diff --git a/src/__phutil_library_map__.php b/src/__phutil_library_map__.php index c3897cce57..801f4516ef 100644 --- a/src/__phutil_library_map__.php +++ b/src/__phutil_library_map__.php @@ -1165,7 +1165,9 @@ phutil_register_library_map(array( 'PHUIFormLayoutView' => 'view/form/PHUIFormLayoutView.php', 'PHUIFormMultiSubmitControl' => 'view/form/control/PHUIFormMultiSubmitControl.php', 'PHUIFormPageView' => 'view/form/PHUIFormPageView.php', + 'PHUIHandleListView' => 'applications/phid/view/PHUIHandleListView.php', 'PHUIHandleTagListView' => 'applications/phid/view/PHUIHandleTagListView.php', + 'PHUIHandleView' => 'applications/phid/view/PHUIHandleView.php', 'PHUIHeaderView' => 'view/phui/PHUIHeaderView.php', 'PHUIIconExample' => 'applications/uiexample/examples/PHUIIconExample.php', 'PHUIIconView' => 'view/phui/PHUIIconView.php', @@ -4423,7 +4425,9 @@ phutil_register_library_map(array( 'PHUIFormLayoutView' => 'AphrontView', 'PHUIFormMultiSubmitControl' => 'AphrontFormControl', 'PHUIFormPageView' => 'AphrontView', + 'PHUIHandleListView' => 'AphrontTagView', 'PHUIHandleTagListView' => 'AphrontTagView', + 'PHUIHandleView' => 'AphrontView', 'PHUIHeaderView' => 'AphrontView', 'PHUIIconExample' => 'PhabricatorUIExample', 'PHUIIconView' => 'AphrontTagView', diff --git a/src/applications/maniphest/controller/ManiphestTaskDetailController.php b/src/applications/maniphest/controller/ManiphestTaskDetailController.php index 537ed9c780..147e9dfc6e 100644 --- a/src/applications/maniphest/controller/ManiphestTaskDetailController.php +++ b/src/applications/maniphest/controller/ManiphestTaskDetailController.php @@ -85,10 +85,6 @@ final class ManiphestTaskDetailController extends ManiphestController { $phids = array_keys($phids); $handles = $user->loadHandles($phids); - // TODO: This is double-loading because we have a separate call to - // renderHandlesForPHIDs(). Clean this up in the next pass. - $this->loadHandles($phids); - $info_view = null; if ($parent_task) { $info_view = new PHUIInfoView(); @@ -100,8 +96,8 @@ final class ManiphestTaskDetailController extends ManiphestController { ->setText(pht('Create Another Subtask'))); $info_view->appendChild(hsprintf( - 'Created a subtask of %s', - $handles[$parent_task->getPHID()]->renderLink())); + 'Created a subtask of %s.', + $handles->renderHandle($parent_task->getPHID()))); } else if ($workflow == 'create') { $info_view = new PHUIInfoView(); $info_view->setSeverity(PHUIInfoView::SEVERITY_NOTICE); @@ -457,7 +453,7 @@ final class ManiphestTaskDetailController extends ManiphestController { $view->addProperty( pht('Assigned To'), $task->getOwnerPHID() - ? $handles[$task->getOwnerPHID()]->renderLink() + ? $handles->renderHandle($task->getOwnerPHID()) : phutil_tag('em', array(), pht('None'))); $view->addProperty( @@ -466,7 +462,7 @@ final class ManiphestTaskDetailController extends ManiphestController { $view->addProperty( pht('Author'), - $handles[$task->getAuthorPHID()]->renderLink()); + $handles->renderHandle($task->getAuthorPHID())); $source = $task->getOriginalEmailSource(); if ($source) { @@ -504,7 +500,7 @@ final class ManiphestTaskDetailController extends ManiphestController { ->execute(); foreach ($commit_phids as $phid) { - $revisions_commits[$phid] = $handles[$phid]->renderLink(); + $revisions_commits[$phid] = $handles->renderHandle($phid); $revision_phid = key($drev_edges[$phid][$commit_drev]); $revision_handle = $handles->getHandleIfExists($revision_phid); if ($revision_handle) { @@ -520,9 +516,10 @@ final class ManiphestTaskDetailController extends ManiphestController { foreach ($edge_types as $edge_type => $edge_name) { if ($edges[$edge_type]) { + $edge_handles = $viewer->loadHandles(array_keys($edges[$edge_type])); $view->addProperty( $edge_name, - $this->renderHandlesForPHIDs(array_keys($edges[$edge_type]))); + $edge_handles->renderList()); } } diff --git a/src/applications/phid/handle/pool/PhabricatorHandleList.php b/src/applications/phid/handle/pool/PhabricatorHandleList.php index 0a487cdd2b..d21e461871 100644 --- a/src/applications/phid/handle/pool/PhabricatorHandleList.php +++ b/src/applications/phid/handle/pool/PhabricatorHandleList.php @@ -26,6 +26,7 @@ final class PhabricatorHandleList private $phids; private $handles; private $cursor; + private $map; public function setHandlePool(PhabricatorHandlePool $pool) { $this->handlePool = $pool; @@ -71,6 +72,33 @@ final class PhabricatorHandleList } +/* -( Rendering )---------------------------------------------------------- */ + + + /** + * Return a @{class:PHUIHandleListView} which can render the handles in + * this list. + */ + public function renderList() { + return id(new PHUIHandleListView()) + ->setHandleList($this); + } + + + /** + * Return a @{class:PHUIHandleView} which can render a specific handle. + */ + public function renderHandle($phid) { + if (!isset($this[$phid])) { + throw new Exception( + pht('Trying to render a handle which does not exist!')); + } + + return id(new PHUIHandleView()) + ->setHandleList($this) + ->setHandlePHID($phid); + } + /* -( Iterator )----------------------------------------------------------- */ @@ -99,10 +127,15 @@ final class PhabricatorHandleList public function offsetExists($offset) { - if ($this->handles === null) { - $this->loadHandles(); + // NOTE: We're intentionally not loading handles here so that isset() + // checks do not trigger fetches. This gives us better bulk loading + // behavior, particularly when invoked through methods like renderHandle(). + + if ($this->map === null) { + $this->map = array_fill_keys($this->phids, true); } - return isset($this->handles[$offset]); + + return isset($this->map[$offset]); } public function offsetGet($offset) { diff --git a/src/applications/phid/view/PHUIHandleListView.php b/src/applications/phid/view/PHUIHandleListView.php new file mode 100644 index 0000000000..4ea52ed58a --- /dev/null +++ b/src/applications/phid/view/PHUIHandleListView.php @@ -0,0 +1,51 @@ +handleList = $list; + return $this; + } + + public function setInline($inline) { + $this->inline = $inline; + return $this; + } + + public function getInline() { + return $this->inline; + } + + protected function getTagName() { + // TODO: It would be nice to render this with a proper