1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-23 13:08:18 +01:00

Allow applications to have multiple "help" menu items

Summary:
Ref T7199. Convert the single help menu item into a dropdown and allow applications to list multiple items there.

When an application has mail command objects, link them in the menu.

Test Plan:
{F355925}

{F355926}

Reviewers: btrahan

Reviewed By: btrahan

Subscribers: epriestley

Maniphest Tasks: T7199

Differential Revision: https://secure.phabricator.com/D12244
This commit is contained in:
epriestley 2015-04-01 08:13:12 -07:00
parent 6f95b325c6
commit c169199e64
22 changed files with 260 additions and 55 deletions

View file

@ -1894,6 +1894,7 @@ phutil_register_library_map(array(
'PhabricatorHashTestCase' => 'infrastructure/util/__tests__/PhabricatorHashTestCase.php', 'PhabricatorHashTestCase' => 'infrastructure/util/__tests__/PhabricatorHashTestCase.php',
'PhabricatorHelpApplication' => 'applications/help/application/PhabricatorHelpApplication.php', 'PhabricatorHelpApplication' => 'applications/help/application/PhabricatorHelpApplication.php',
'PhabricatorHelpController' => 'applications/help/controller/PhabricatorHelpController.php', 'PhabricatorHelpController' => 'applications/help/controller/PhabricatorHelpController.php',
'PhabricatorHelpDocumentationController' => 'applications/help/controller/PhabricatorHelpDocumentationController.php',
'PhabricatorHelpEditorProtocolController' => 'applications/help/controller/PhabricatorHelpEditorProtocolController.php', 'PhabricatorHelpEditorProtocolController' => 'applications/help/controller/PhabricatorHelpEditorProtocolController.php',
'PhabricatorHelpKeyboardShortcutController' => 'applications/help/controller/PhabricatorHelpKeyboardShortcutController.php', 'PhabricatorHelpKeyboardShortcutController' => 'applications/help/controller/PhabricatorHelpKeyboardShortcutController.php',
'PhabricatorHeraldApplication' => 'applications/herald/application/PhabricatorHeraldApplication.php', 'PhabricatorHeraldApplication' => 'applications/herald/application/PhabricatorHeraldApplication.php',
@ -5232,6 +5233,7 @@ phutil_register_library_map(array(
'PhabricatorHashTestCase' => 'PhabricatorTestCase', 'PhabricatorHashTestCase' => 'PhabricatorTestCase',
'PhabricatorHelpApplication' => 'PhabricatorApplication', 'PhabricatorHelpApplication' => 'PhabricatorApplication',
'PhabricatorHelpController' => 'PhabricatorController', 'PhabricatorHelpController' => 'PhabricatorController',
'PhabricatorHelpDocumentationController' => 'PhabricatorHelpController',
'PhabricatorHelpEditorProtocolController' => 'PhabricatorHelpController', 'PhabricatorHelpEditorProtocolController' => 'PhabricatorHelpController',
'PhabricatorHelpKeyboardShortcutController' => 'PhabricatorHelpController', 'PhabricatorHelpKeyboardShortcutController' => 'PhabricatorHelpController',
'PhabricatorHeraldApplication' => 'PhabricatorApplication', 'PhabricatorHeraldApplication' => 'PhabricatorApplication',

View file

@ -26,8 +26,13 @@ final class PhabricatorAlmanacApplication extends PhabricatorApplication {
return self::GROUP_UTILITIES; return self::GROUP_UTILITIES;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Almanac User Guide'); return array(
array(
'name' => pht('Alamanac User Guide'),
'href' => PhabricatorEnv::getDoclink('Almanac User Guide'),
),
);
} }
public function isPrototype() { public function isPrototype() {

View file

@ -22,8 +22,13 @@ final class PhabricatorAuditApplication extends PhabricatorApplication {
return true; return true;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Audit User Guide'); return array(
array(
'name' => pht('Audit User Guide'),
'href' => PhabricatorEnv::getDoclink('Audit User Guide'),
),
);
} }
public function getRoutes() { public function getRoutes() {

View file

@ -26,16 +26,16 @@ final class PhabricatorAuthApplication extends PhabricatorApplication {
return pht('Login/Registration'); return pht('Login/Registration');
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
// NOTE: Although reasonable help exists for this in "Configuring Accounts // NOTE: Although reasonable help exists for this in "Configuring Accounts
// and Registration", specifying a help URI here means we get the menu // and Registration", specifying help items here means we get the menu
// item in all the login/link interfaces, which is confusing and not // item in all the login/link interfaces, which is confusing and not
// helpful. // helpful.
// TODO: Special case this, or split the auth and auth administration // TODO: Special case this, or split the auth and auth administration
// applications? // applications?
return null; return array();
} }
public function buildMainMenuItems( public function buildMainMenuItems(

View file

@ -169,8 +169,48 @@ abstract class PhabricatorApplication implements PhabricatorPolicyInterface {
return null; return null;
} }
public function getHelpURI() { public function getHelpMenuItems(PhabricatorUser $viewer) {
return null; $items = array();
$articles = $this->getHelpDocumentationArticles($viewer);
if ($articles) {
$items[] = id(new PHUIListItemView())
->setType(PHUIListItemView::TYPE_LABEL)
->setName(pht('%s Documentation', $this->getName()));
foreach ($articles as $article) {
$item = id(new PHUIListItemView())
->setName($article['name'])
->setIcon('fa-book')
->setHref($article['href']);
$items[] = $item;
}
}
$command_specs = $this->getMailCommandObjects();
if ($command_specs) {
$items[] = id(new PHUIListItemView())
->setType(PHUIListItemView::TYPE_LABEL)
->setName(pht('Email Help'));
foreach ($command_specs as $key => $spec) {
$object = $spec['object'];
$class = get_class($this);
$href = '/applications/mailcommands/'.$class.'/'.$key.'/';
$item = id(new PHUIListItemView())
->setName($spec['name'])
->setIcon('fa-envelope-o')
->setHref($href);
$items[] = $item;
}
}
return $items;
}
public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return array();
} }
public function getOverview() { public function getOverview() {

View file

@ -14,8 +14,13 @@ final class PhabricatorConduitApplication extends PhabricatorApplication {
return false; return false;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Conduit Technical Documentation'); return array(
array(
'name' => pht('Conduit Technical Documentation'),
'href' => PhabricatorEnv::getDoclink('Conduit Technical Documentation'),
),
);
} }
public function getName() { public function getName() {

View file

@ -22,8 +22,13 @@ final class PhabricatorDifferentialApplication extends PhabricatorApplication {
return true; return true;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Differential User Guide'); return array(
array(
'name' => pht('Differential User Guide'),
'href' => PhabricatorEnv::getDoclink('Differential User Guide'),
),
);
} }
public function getFactObjectsForAnalysis() { public function getFactObjectsForAnalysis() {
@ -189,6 +194,7 @@ EOTEXT
public function getMailCommandObjects() { public function getMailCommandObjects() {
return array( return array(
'revision' => array( 'revision' => array(
'name' => pht('Email Commands: Revisions'),
'object' => new DifferentialRevision(), 'object' => new DifferentialRevision(),
), ),
); );

View file

@ -22,8 +22,13 @@ final class PhabricatorDiffusionApplication extends PhabricatorApplication {
return true; return true;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Diffusion User Guide'); return array(
array(
'name' => pht('Diffusion User Guide'),
'href' => PhabricatorEnv::getDoclink('Diffusion User Guide'),
),
);
} }
public function getFactObjectsForAnalysis() { public function getFactObjectsForAnalysis() {

View file

@ -34,8 +34,13 @@ final class PhabricatorDrydockApplication extends PhabricatorApplication {
return true; return true;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Drydock User Guide'); return array(
array(
'name' => pht('Drydock User Guide'),
'href' => PhabricatorEnv::getDoclink('Drydock User Guide'),
),
);
} }
public function getRoutes() { public function getRoutes() {

View file

@ -19,6 +19,8 @@ final class PhabricatorHelpApplication extends PhabricatorApplication {
'/help/' => array( '/help/' => array(
'keyboardshortcut/' => 'PhabricatorHelpKeyboardShortcutController', 'keyboardshortcut/' => 'PhabricatorHelpKeyboardShortcutController',
'editorprotocol/' => 'PhabricatorHelpEditorProtocolController', 'editorprotocol/' => 'PhabricatorHelpEditorProtocolController',
'documentation/(?P<application>\w+)/'
=> 'PhabricatorHelpDocumentationController',
), ),
); );
} }
@ -27,27 +29,75 @@ final class PhabricatorHelpApplication extends PhabricatorApplication {
PhabricatorUser $user, PhabricatorUser $user,
PhabricatorController $controller = null) { PhabricatorController $controller = null) {
$items = array();
$application = null; $application = null;
if ($controller) { if ($controller) {
$application = $controller->getCurrentApplication(); $application = $controller->getCurrentApplication();
} }
if ($application && $application->getHelpURI()) { $items = array();
$help_name = pht('%s Help', $application->getName()); if ($application) {
$help_items = $application->getHelpMenuItems($user);
if ($help_items) {
$help_id = celerity_generate_unique_node_id();
$item = id(new PHUIListItemView()) Javelin::initBehavior(
->setName($help_name) 'aphlict-dropdown',
->addClass('core-menu-item') array(
->setIcon('fa-info-circle') 'bubbleID' => $help_id,
->setAural($help_name) 'dropdownID' => 'phabricator-help-menu',
->setOrder(200) 'local' => true,
->setHref($application->getHelpURI()); 'desktop' => true,
$items[] = $item; 'right' => true,
));
$help_name = pht('%s Help', $application->getName());
$item = id(new PHUIListItemView())
->setName($help_name)
->setIcon('fa-life-ring')
->setHref('/help/documentation/'.get_class($application).'/')
->addClass('core-menu-item')
->setID($help_id)
->setAural($help_name)
->setOrder(200);
$items[] = $item;
}
} }
return $items; return $items;
} }
public function buildMainMenuExtraNodes(
PhabricatorUser $viewer,
PhabricatorController $controller = null) {
if (!$controller) {
return null;
}
$application = $controller->getCurrentApplication();
if (!$application) {
return null;
}
$help_items = $application->getHelpMenuItems($viewer);
if (!$help_items) {
return null;
}
$view = new PHUIListView();
foreach ($help_items as $item) {
$view->addMenuItem($item);
}
return phutil_tag(
'div',
array(
'id' => 'phabricator-help-menu',
'class' => 'phabricator-main-menu-dropdown phui-list-sidenav',
'style' => 'display: none',
),
$view);
}
} }

View file

@ -0,0 +1,52 @@
<?php
final class PhabricatorHelpDocumentationController
extends PhabricatorHelpController {
public function shouldAllowPublic() {
return true;
}
public function handleRequest(AphrontRequest $request) {
$viewer = $this->getViewer();
$application_class = $request->getURIData('application');
$application = id(new PhabricatorApplicationQuery())
->setViewer($viewer)
->withClasses(array($application_class))
->executeOne();
if (!$application) {
return new Aphront404Response();
}
$items = $application->getHelpMenuItems($viewer);
$title = pht('%s Help', $application->getName());
$list = id(new PHUIObjectItemListView())
->setUser($viewer);
foreach ($items as $item) {
if ($item->getType() == PHUIListItemView::TYPE_LABEL) {
continue;
}
$list->addItem(
id(new PHUIObjectItemView())
->setHeader($item->getName())
->setWorkflow($item->getWorkflow())
->setHref($item->getHref()));
}
$crumbs = $this->buildApplicationCrumbs();
$crumbs->addTextCrumb($title);
return $this->buildApplicationPage(
array(
$crumbs,
$list,
),
array(
'title' => $title,
));
}
}

View file

@ -22,8 +22,13 @@ final class PhabricatorHeraldApplication extends PhabricatorApplication {
return "\xE2\x98\xBF"; return "\xE2\x98\xBF";
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Herald User Guide'); return array(
array(
'name' => pht('Herald User Guide'),
'href' => PhabricatorEnv::getDoclink('Herald User Guide'),
),
);
} }
public function getFlavorText() { public function getFlavorText() {

View file

@ -32,5 +32,4 @@ final class PhabricatorHomeQuickCreateController
)); ));
} }
} }

View file

@ -32,8 +32,13 @@ final class PhabricatorLegalpadApplication extends PhabricatorApplication {
); );
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Legalpad User Guide'); return array(
array(
'name' => pht('Legalpad User Guide'),
'href' => PhabricatorEnv::getDoclink('Legalpad User Guide'),
),
);
} }
public function getOverview() { public function getOverview() {

View file

@ -146,6 +146,7 @@ final class PhabricatorManiphestApplication extends PhabricatorApplication {
public function getMailCommandObjects() { public function getMailCommandObjects() {
return array( return array(
'task' => array( 'task' => array(
'name' => pht('Email Commands: Tasks'),
'object' => new ManiphestTask(), 'object' => new ManiphestTask(),
), ),
); );

View file

@ -138,14 +138,6 @@ final class PhabricatorApplicationDetailViewController
->setUser($user) ->setUser($user)
->setObjectURI($this->getRequest()->getRequestURI()); ->setObjectURI($this->getRequest()->getRequestURI());
if ($selected->getHelpURI()) {
$view->addAction(
id(new PhabricatorActionView())
->setName(pht('Help / Documentation'))
->setIcon('fa-life-ring')
->setHref($selected->getHelpURI()));
}
$can_edit = PhabricatorPolicyFilter::hasCapability( $can_edit = PhabricatorPolicyFilter::hasCapability(
$user, $user,
$selected, $selected,

View file

@ -78,9 +78,11 @@ final class PhabricatorApplicationEmailCommandsController
$content = implode("\n\n", $content); $content = implode("\n\n", $content);
$title = $spec['name'];
$crumbs = $this->buildApplicationCrumbs(); $crumbs = $this->buildApplicationCrumbs();
$this->addApplicationCrumb($crumbs, $selected); $this->addApplicationCrumb($crumbs, $selected);
$crumbs->addTextCrumb(pht('Mail Commands')); $crumbs->addTextCrumb($title);
$content_box = id(new PHUIBoxView()) $content_box = id(new PHUIBoxView())
->addMargin(PHUI::MARGIN_LARGE) ->addMargin(PHUI::MARGIN_LARGE)
@ -91,7 +93,7 @@ final class PhabricatorApplicationEmailCommandsController
$viewer)); $viewer));
$box = id(new PHUIObjectBoxView()) $box = id(new PHUIObjectBoxView())
->setHeaderText(pht('Mail Commands')) ->setHeaderText($title)
->appendChild($content_box); ->appendChild($content_box);
return $this->buildApplicationPage( return $this->buildApplicationPage(
@ -100,7 +102,7 @@ final class PhabricatorApplicationEmailCommandsController
$box, $box,
), ),
array( array(
'title' => 'asdf', 'title' => $title,
)); ));
} }

View file

@ -34,8 +34,14 @@ final class PhabricatorOAuthServerApplication extends PhabricatorApplication {
return true; return true;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Using the Phabricator OAuth Server'); return array(
array(
'name' => pht('Using the Phabricator OAuth Server'),
'href' => PhabricatorEnv::getDoclink(
'Using the Phabricator OAuth Server'),
),
);
} }
public function getRoutes() { public function getRoutes() {

View file

@ -22,8 +22,13 @@ final class PhabricatorOwnersApplication extends PhabricatorApplication {
return "\xE2\x98\x81"; return "\xE2\x98\x81";
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Owners Tool User Guide'); return array(
array(
'name' => pht('Owners User Guide'),
'href' => PhabricatorEnv::getDoclink('Owners Tool User Guide'),
),
);
} }
public function getFlavorText() { public function getFlavorText() {

View file

@ -22,8 +22,13 @@ final class PhabricatorPhameApplication extends PhabricatorApplication {
return "\xe2\x9c\xa9"; return "\xe2\x9c\xa9";
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Phame User Guide'); return array(
array(
'name' => pht('Phame User Guide'),
'href' => PhabricatorEnv::getDoclink('Phame User Guide'),
),
);
} }
public function isPrototype() { public function isPrototype() {

View file

@ -22,8 +22,13 @@ final class PhabricatorPhrictionApplication extends PhabricatorApplication {
return true; return true;
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Phriction User Guide'); return array(
array(
'name' => pht('Phriction User Guide'),
'href' => PhabricatorEnv::getDoclink('Phriction User Guide'),
),
);
} }
public function getTitleGlyph() { public function getTitleGlyph() {

View file

@ -22,8 +22,13 @@ final class PhabricatorSlowvoteApplication extends PhabricatorApplication {
return "\xE2\x9C\x94"; return "\xE2\x9C\x94";
} }
public function getHelpURI() { public function getHelpDocumentationArticles(PhabricatorUser $viewer) {
return PhabricatorEnv::getDoclink('Slowvote User Guide'); return array(
array(
'name' => pht('Slowvote User Guide'),
'href' => PhabricatorEnv::getDoclink('Slowvote User Guide'),
),
);
} }
public function getFlavorText() { public function getFlavorText() {