From 31b6f69ff7c21e04cd537e6ae46b0c5895bf44bb Mon Sep 17 00:00:00 2001 From: epriestley Date: Thu, 2 Jan 2014 11:59:35 -0800 Subject: [PATCH] Allow CelerityResourceResponse to hold resources from multiple maps Summary: Ref T4222. Currently, CelerityResourceResponse holds response resources in flat maps. Instead, specify which map resources appear in. Also, provide `requireResource()` and `initBehavior()` APIs on the Controller and View base classes. These provide a cleaner abstraction over `require_celerity_resource()` and `Javelin::initBehavior()`, but are otherwise the same. Move a few callsites over. Test Plan: - Reloaded pages. - Browsed around Differential. Reviewers: btrahan, hach-que Reviewed By: btrahan CC: aran Maniphest Tasks: T4222 Differential Revision: https://secure.phabricator.com/D7876 --- src/aphront/AphrontController.php | 21 +++- .../base/controller/PhabricatorController.php | 5 + .../DifferentialRevisionViewController.php | 4 +- .../view/DifferentialAddCommentView.php | 2 +- .../view/DifferentialChangesetDetailView.php | 4 +- .../view/DifferentialChangesetListView.php | 10 +- .../DifferentialDiffTableOfContentsView.php | 4 +- .../view/DifferentialLocalCommitsView.php | 2 +- .../view/DifferentialResultsTableView.php | 4 +- .../DifferentialRevisionCommentListView.php | 4 +- .../view/DifferentialRevisionCommentView.php | 4 +- .../view/DifferentialRevisionDetailView.php | 2 +- .../view/DifferentialRevisionListView.php | 4 +- .../DifferentialRevisionUpdateHistoryView.php | 4 +- .../celerity/CelerityResourceMap.php | 4 + .../CelerityStaticResourceResponse.php | 100 +++++++++++++----- src/infrastructure/celerity/api.php | 4 +- src/view/AphrontView.php | 17 +++ 18 files changed, 146 insertions(+), 53 deletions(-) diff --git a/src/aphront/AphrontController.php b/src/aphront/AphrontController.php index 6d84c46195..a591271270 100644 --- a/src/aphront/AphrontController.php +++ b/src/aphront/AphrontController.php @@ -9,7 +9,6 @@ abstract class AphrontController extends Phobject { private $currentApplication; private $delegatingController; - public function setDelegatingController( AphrontController $delegating_controller) { $this->delegatingController = $delegating_controller; @@ -64,4 +63,24 @@ abstract class AphrontController extends Phobject { return $this->currentApplication; } + public function getDefaultResourceSource() { + throw new Exception( + pht( + 'A Controller must implement getDefaultResourceSource() before you '. + 'can invoke requireResource() or initBehavior().')); + } + + public function requireResource($symbol) { + $response = CelerityAPI::getStaticResourceResponse(); + $response->requireResource($symbol, $this->getDefaultResourceSource()); + return $this; + } + + public function initBehavior($name, $config = array()) { + Javelin::initBehavior( + $name, + $config, + $this->getDefaultResourceSource()); + } + } diff --git a/src/applications/base/controller/PhabricatorController.php b/src/applications/base/controller/PhabricatorController.php index 86009fd9fc..4d7d6e3243 100644 --- a/src/applications/base/controller/PhabricatorController.php +++ b/src/applications/base/controller/PhabricatorController.php @@ -408,4 +408,9 @@ abstract class PhabricatorController extends AphrontController { return array($can_act, $message); } + public function getDefaultResourceSource() { + return 'phabricator'; + } + + } diff --git a/src/applications/differential/controller/DifferentialRevisionViewController.php b/src/applications/differential/controller/DifferentialRevisionViewController.php index 14d37ffb29..8e15dd7de3 100644 --- a/src/applications/differential/controller/DifferentialRevisionViewController.php +++ b/src/applications/differential/controller/DifferentialRevisionViewController.php @@ -497,8 +497,8 @@ final class DifferentialRevisionViewController extends DifferentialController { ); } - require_celerity_resource('phabricator-object-selector-css'); - require_celerity_resource('javelin-behavior-phabricator-object-selector'); + $this->requireResource('phabricator-object-selector-css'); + $this->requireResource('javelin-behavior-phabricator-object-selector'); $links[] = array( 'icon' => 'link', diff --git a/src/applications/differential/view/DifferentialAddCommentView.php b/src/applications/differential/view/DifferentialAddCommentView.php index 822a85d6ac..a95ee9254f 100644 --- a/src/applications/differential/view/DifferentialAddCommentView.php +++ b/src/applications/differential/view/DifferentialAddCommentView.php @@ -48,7 +48,7 @@ final class DifferentialAddCommentView extends AphrontView { public function render() { - require_celerity_resource('differential-revision-add-comment-css'); + $this->requireResource('differential-revision-add-comment-css'); $is_serious = PhabricatorEnv::getEnvConfig('phabricator.serious-business'); diff --git a/src/applications/differential/view/DifferentialChangesetDetailView.php b/src/applications/differential/view/DifferentialChangesetDetailView.php index 029bd0a2a0..0e9a1711f3 100644 --- a/src/applications/differential/view/DifferentialChangesetDetailView.php +++ b/src/applications/differential/view/DifferentialChangesetDetailView.php @@ -94,8 +94,8 @@ final class DifferentialChangesetDetailView extends AphrontView { } public function render() { - require_celerity_resource('differential-changeset-view-css'); - require_celerity_resource('syntax-highlighting-css'); + $this->requireResource('differential-changeset-view-css'); + $this->requireResource('syntax-highlighting-css'); Javelin::initBehavior('phabricator-oncopy', array()); diff --git a/src/applications/differential/view/DifferentialChangesetListView.php b/src/applications/differential/view/DifferentialChangesetListView.php index 17ed42eac9..069fd2bc4b 100644 --- a/src/applications/differential/view/DifferentialChangesetListView.php +++ b/src/applications/differential/view/DifferentialChangesetListView.php @@ -103,7 +103,7 @@ final class DifferentialChangesetListView extends AphrontView { } public function render() { - require_celerity_resource('differential-changeset-view-css'); + $this->requireResource('differential-changeset-view-css'); $changesets = $this->changesets; @@ -183,20 +183,20 @@ final class DifferentialChangesetListView extends AphrontView { $output[] = $detail->render(); } - require_celerity_resource('aphront-tooltip-css'); + $this->requireResource('aphront-tooltip-css'); - Javelin::initBehavior('differential-populate', array( + $this->initBehavior('differential-populate', array( 'registry' => $mapping, 'whitespace' => $this->whitespace, 'uri' => $this->renderURI, )); - Javelin::initBehavior('differential-show-more', array( + $this->initBehavior('differential-show-more', array( 'uri' => $this->renderURI, 'whitespace' => $this->whitespace, )); - Javelin::initBehavior('differential-comment-jump', array()); + $this->initBehavior('differential-comment-jump', array()); if ($this->inlineURI) { $undo_templates = $this->renderUndoTemplates(); diff --git a/src/applications/differential/view/DifferentialDiffTableOfContentsView.php b/src/applications/differential/view/DifferentialDiffTableOfContentsView.php index 84fae42243..74391ef5f9 100644 --- a/src/applications/differential/view/DifferentialDiffTableOfContentsView.php +++ b/src/applications/differential/view/DifferentialDiffTableOfContentsView.php @@ -54,8 +54,8 @@ final class DifferentialDiffTableOfContentsView extends AphrontView { public function render() { - require_celerity_resource('differential-core-view-css'); - require_celerity_resource('differential-table-of-contents-css'); + $this->requireResource('differential-core-view-css'); + $this->requireResource('differential-table-of-contents-css'); $rows = array(); diff --git a/src/applications/differential/view/DifferentialLocalCommitsView.php b/src/applications/differential/view/DifferentialLocalCommitsView.php index 4321807139..d4fbe45431 100644 --- a/src/applications/differential/view/DifferentialLocalCommitsView.php +++ b/src/applications/differential/view/DifferentialLocalCommitsView.php @@ -20,7 +20,7 @@ final class DifferentialLocalCommitsView extends AphrontView { return null; } - require_celerity_resource('differential-local-commits-view-css'); + $this->requireResource('differential-local-commits-view-css'); $has_tree = false; $has_local = false; diff --git a/src/applications/differential/view/DifferentialResultsTableView.php b/src/applications/differential/view/DifferentialResultsTableView.php index f2be210728..226c68484c 100644 --- a/src/applications/differential/view/DifferentialResultsTableView.php +++ b/src/applications/differential/view/DifferentialResultsTableView.php @@ -97,10 +97,10 @@ final class DifferentialResultsTableView extends AphrontView { ), phutil_tag('th', array('colspan' => 2), $hide_more)); - Javelin::initBehavior('differential-show-field-details'); + $this->initBehavior('differential-show-field-details'); } - require_celerity_resource('differential-results-table-css'); + $this->requireResource('differential-results-table-css'); return javelin_tag( 'table', diff --git a/src/applications/differential/view/DifferentialRevisionCommentListView.php b/src/applications/differential/view/DifferentialRevisionCommentListView.php index 86d0c76bfa..b03cee03db 100644 --- a/src/applications/differential/view/DifferentialRevisionCommentListView.php +++ b/src/applications/differential/view/DifferentialRevisionCommentListView.php @@ -53,7 +53,7 @@ final class DifferentialRevisionCommentListView extends AphrontView { public function render() { - require_celerity_resource('differential-revision-comment-list-css'); + $this->requireResource('differential-revision-comment-list-css'); $engine = new PhabricatorMarkupEngine(); $engine->setViewer($this->user); @@ -154,7 +154,7 @@ final class DifferentialRevisionCommentListView extends AphrontView { $visible = array_reverse($visible); if ($hidden) { - Javelin::initBehavior( + $this->initBehavior( 'differential-show-all-comments', array( 'markup' => implode("\n", $hidden), diff --git a/src/applications/differential/view/DifferentialRevisionCommentView.php b/src/applications/differential/view/DifferentialRevisionCommentView.php index 820954dfff..293d93a4a6 100644 --- a/src/applications/differential/view/DifferentialRevisionCommentView.php +++ b/src/applications/differential/view/DifferentialRevisionCommentView.php @@ -67,8 +67,8 @@ final class DifferentialRevisionCommentView extends AphrontView { throw new Exception("Call setUser() before rendering!"); } - require_celerity_resource('phabricator-remarkup-css'); - require_celerity_resource('differential-revision-comment-css'); + $this->requireResource('phabricator-remarkup-css'); + $this->requireResource('differential-revision-comment-css'); $comment = $this->comment; diff --git a/src/applications/differential/view/DifferentialRevisionDetailView.php b/src/applications/differential/view/DifferentialRevisionDetailView.php index fad95b5ada..7e0f3c5de6 100644 --- a/src/applications/differential/view/DifferentialRevisionDetailView.php +++ b/src/applications/differential/view/DifferentialRevisionDetailView.php @@ -45,7 +45,7 @@ final class DifferentialRevisionDetailView extends AphrontView { public function render() { - require_celerity_resource('differential-core-view-css'); + $this->requireResource('differential-core-view-css'); $revision = $this->revision; $user = $this->getUser(); diff --git a/src/applications/differential/view/DifferentialRevisionListView.php b/src/applications/differential/view/DifferentialRevisionListView.php index 86bab4dbc6..0addfa5b62 100644 --- a/src/applications/differential/view/DifferentialRevisionListView.php +++ b/src/applications/differential/view/DifferentialRevisionListView.php @@ -102,8 +102,8 @@ final class DifferentialRevisionListView extends AphrontView { -$stale); } - Javelin::initBehavior('phabricator-tooltips', array()); - require_celerity_resource('aphront-tooltip-css'); + $this->initBehavior('phabricator-tooltips', array()); + $this->requireResource('aphront-tooltip-css'); $flagged = mpull($this->flags, null, 'getObjectPHID'); diff --git a/src/applications/differential/view/DifferentialRevisionUpdateHistoryView.php b/src/applications/differential/view/DifferentialRevisionUpdateHistoryView.php index e4a78860df..5029a3bb44 100644 --- a/src/applications/differential/view/DifferentialRevisionUpdateHistoryView.php +++ b/src/applications/differential/view/DifferentialRevisionUpdateHistoryView.php @@ -30,8 +30,8 @@ final class DifferentialRevisionUpdateHistoryView extends AphrontView { public function render() { - require_celerity_resource('differential-core-view-css'); - require_celerity_resource('differential-revision-history-css'); + $this->requireResource('differential-core-view-css'); + $this->requireResource('differential-revision-history-css'); $data = array( array( diff --git a/src/infrastructure/celerity/CelerityResourceMap.php b/src/infrastructure/celerity/CelerityResourceMap.php index ebaa4fb9a6..c9beffc2d3 100644 --- a/src/infrastructure/celerity/CelerityResourceMap.php +++ b/src/infrastructure/celerity/CelerityResourceMap.php @@ -236,4 +236,8 @@ final class CelerityResourceMap { return isset($this->packageMap[$name]); } + public function getResourceTypeForName($name) { + return $this->resources->getResourceType($name); + } + } diff --git a/src/infrastructure/celerity/CelerityStaticResourceResponse.php b/src/infrastructure/celerity/CelerityStaticResourceResponse.php index b52274f323..21ad591e87 100644 --- a/src/infrastructure/celerity/CelerityStaticResourceResponse.php +++ b/src/infrastructure/celerity/CelerityStaticResourceResponse.php @@ -39,8 +39,12 @@ final class CelerityStaticResourceResponse { * a behavior will execute only once even if it is initialized multiple times. * If $config is nonempty, the behavior will be invoked once for each config. */ - public function initBehavior($behavior, array $config = array()) { - $this->requireResource('javelin-behavior-'.$behavior); + public function initBehavior( + $behavior, + array $config = array(), + $source_name = 'phabricator') { + + $this->requireResource('javelin-behavior-'.$behavior, $source_name); if (empty($this->behaviors[$behavior])) { $this->behaviors[$behavior] = array(); @@ -53,19 +57,39 @@ final class CelerityStaticResourceResponse { return $this; } - public function requireResource($symbol) { - $this->symbols[$symbol] = true; + public function requireResource($symbol, $source_name) { + if (isset($this->symbols[$source_name][$symbol])) { + return $this; + } + + // Verify that the resource exists. + $map = CelerityResourceMap::getNamedInstance($source_name); + $name = $map->getResourceNameForSymbol($symbol); + if ($name === null) { + throw new Exception( + pht( + 'No resource with symbol "%s" exists in source "%s"!', + $symbol, + $source_name)); + } + + $this->symbols[$source_name][$symbol] = true; $this->needsResolve = true; + return $this; } private function resolveResources() { if ($this->needsResolve) { - $map = CelerityResourceMap::getNamedInstance('phabricator'); + $this->packaged = array(); + foreach ($this->symbols as $source_name => $symbols_map) { + $symbols = array_keys($symbols_map); - $symbols = array_keys($this->symbols); - $this->packaged = $map->getPackagedNamesForSymbols($symbols); + $map = CelerityResourceMap::getNamedInstance($source_name); + $packaged = $map->getPackagedNamesForSymbols($symbols); + $this->packaged[$source_name] = $packaged; + } $this->needsResolve = false; } return $this; @@ -74,24 +98,34 @@ final class CelerityStaticResourceResponse { public function renderSingleResource($symbol, $source_name) { $map = CelerityResourceMap::getNamedInstance($source_name); $packaged = $map->getPackagedNamesForSymbols(array($symbol)); - return $this->renderPackagedResources($packaged); + return $this->renderPackagedResources($map, $packaged); } public function renderResourcesOfType($type) { $this->resolveResources(); - $resources = array(); - foreach ($this->packaged as $name) { - $resource_type = CelerityResourceTransformer::getResourceType($name); - if ($resource_type == $type) { - $resources[] = $name; + $result = array(); + foreach ($this->packaged as $source_name => $resource_names) { + $map = CelerityResourceMap::getNamedInstance($source_name); + + $resources_of_type = array(); + foreach ($resource_names as $resource_name) { + $resource_type = $map->getResourceTypeForName($resource_name); + if ($resource_type == $type) { + $resources_of_type[] = $resource_name; + } } + + $result[] = $this->renderPackagedResources($map, $resources_of_type); } - return $this->renderPackagedResources($resources); + return phutil_implode_html('', $result); } - private function renderPackagedResources(array $resources) { + private function renderPackagedResources( + CelerityResourceMap $map, + array $resources) { + $output = array(); foreach ($resources as $name) { if (isset($this->hasRendered[$name])) { @@ -99,15 +133,19 @@ final class CelerityStaticResourceResponse { } $this->hasRendered[$name] = true; - $output[] = $this->renderResource($name); - $output[] = "\n"; + $output[] = $this->renderResource($map, $name); } - return phutil_implode_html('', $output); + + return $output; } - private function renderResource($name) { - $uri = $this->getURI($name); - $type = CelerityResourceTransformer::getResourceType($name); + private function renderResource( + CelerityResourceMap $map, + $name) { + + $uri = $this->getURI($map, $name); + $type = $map->getResourceTypeForName($name); + switch ($type) { case 'css': return phutil_tag( @@ -126,7 +164,12 @@ final class CelerityStaticResourceResponse { ), ''); } - throw new Exception("Unable to render resource."); + + throw new Exception( + pht( + 'Unable to render resource "%s", which has unknown type "%s".', + $name, + $type)); } public function renderHTMLFooter() { @@ -225,8 +268,11 @@ final class CelerityStaticResourceResponse { $this->resolveResources(); $resources = array(); - foreach ($this->packaged as $resource) { - $resources[] = $this->getURI($resource); + foreach ($this->packaged as $source_name => $resource_names) { + $map = CelerityResourceMap::getNamedInstance($source_name); + foreach ($resource_names as $resource_name) { + $resources[] = $this->getURI($map, $resource_name); + } } if ($resources) { $response['javelin_resources'] = $resources; @@ -235,8 +281,10 @@ final class CelerityStaticResourceResponse { return $response; } - private function getURI($name) { - $map = CelerityResourceMap::getNamedInstance('phabricator'); + private function getURI( + CelerityResourceMap $map, + $name) { + $uri = $map->getURIForName($name); // In developer mode, we dump file modification times into the URI. When a diff --git a/src/infrastructure/celerity/api.php b/src/infrastructure/celerity/api.php index 8b04b49fd3..c10b7a1d40 100644 --- a/src/infrastructure/celerity/api.php +++ b/src/infrastructure/celerity/api.php @@ -15,9 +15,9 @@ * * @group celerity */ -function require_celerity_resource($symbol) { +function require_celerity_resource($symbol, $source_name = 'phabricator') { $response = CelerityAPI::getStaticResourceResponse(); - $response->requireResource($symbol); + $response->requireResource($symbol, $source_name); } diff --git a/src/view/AphrontView.php b/src/view/AphrontView.php index 33bddb701e..788ea50ab3 100644 --- a/src/view/AphrontView.php +++ b/src/view/AphrontView.php @@ -128,6 +128,23 @@ abstract class AphrontView extends Phobject return $children; } + public function getDefaultResourceSource() { + return 'phabricator'; + } + + public function requireResource($symbol) { + $response = CelerityAPI::getStaticResourceResponse(); + $response->requireResource($symbol, $this->getDefaultResourceSource()); + return $this; + } + + public function initBehavior($name, $config = array()) { + Javelin::initBehavior( + $name, + $config, + $this->getDefaultResourceSource()); + } + /* -( Rendering )---------------------------------------------------------- */