mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-18 10:41:08 +01:00
Modernize pinned homepage applications settings
Summary: Ref T4103. A few bits here: - We have an ancient "tiles" preference which was just a fallback from 2-3 years ago. Throw that away. - Modenize the other pinned stuff. We should likely revisit this after the next homepage update but I just left the actual defaults alone for now. - Lightly prepare for global default editing. - Add a "reset to defaults" option. Test Plan: - Pinned, unpinned, reordered and reset application homepage order. Reviewers: chad Reviewed By: chad Maniphest Tasks: T4103 Differential Revision: https://secure.phabricator.com/D16028
This commit is contained in:
parent
44e88f186c
commit
64d6593e9c
7 changed files with 131 additions and 73 deletions
|
@ -3008,6 +3008,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorPhurlURLTransactionComment' => 'applications/phurl/storage/PhabricatorPhurlURLTransactionComment.php',
|
'PhabricatorPhurlURLTransactionComment' => 'applications/phurl/storage/PhabricatorPhurlURLTransactionComment.php',
|
||||||
'PhabricatorPhurlURLTransactionQuery' => 'applications/phurl/query/PhabricatorPhurlURLTransactionQuery.php',
|
'PhabricatorPhurlURLTransactionQuery' => 'applications/phurl/query/PhabricatorPhurlURLTransactionQuery.php',
|
||||||
'PhabricatorPhurlURLViewController' => 'applications/phurl/controller/PhabricatorPhurlURLViewController.php',
|
'PhabricatorPhurlURLViewController' => 'applications/phurl/controller/PhabricatorPhurlURLViewController.php',
|
||||||
|
'PhabricatorPinnedApplicationsSetting' => 'applications/settings/setting/PhabricatorPinnedApplicationsSetting.php',
|
||||||
'PhabricatorPirateEnglishTranslation' => 'infrastructure/internationalization/translation/PhabricatorPirateEnglishTranslation.php',
|
'PhabricatorPirateEnglishTranslation' => 'infrastructure/internationalization/translation/PhabricatorPirateEnglishTranslation.php',
|
||||||
'PhabricatorPlatformSite' => 'aphront/site/PhabricatorPlatformSite.php',
|
'PhabricatorPlatformSite' => 'aphront/site/PhabricatorPlatformSite.php',
|
||||||
'PhabricatorPointsEditField' => 'applications/transactions/editfield/PhabricatorPointsEditField.php',
|
'PhabricatorPointsEditField' => 'applications/transactions/editfield/PhabricatorPointsEditField.php',
|
||||||
|
@ -7680,6 +7681,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorPhurlURLTransactionComment' => 'PhabricatorApplicationTransactionComment',
|
'PhabricatorPhurlURLTransactionComment' => 'PhabricatorApplicationTransactionComment',
|
||||||
'PhabricatorPhurlURLTransactionQuery' => 'PhabricatorApplicationTransactionQuery',
|
'PhabricatorPhurlURLTransactionQuery' => 'PhabricatorApplicationTransactionQuery',
|
||||||
'PhabricatorPhurlURLViewController' => 'PhabricatorPhurlController',
|
'PhabricatorPhurlURLViewController' => 'PhabricatorPhurlController',
|
||||||
|
'PhabricatorPinnedApplicationsSetting' => 'PhabricatorInternalSetting',
|
||||||
'PhabricatorPirateEnglishTranslation' => 'PhutilTranslation',
|
'PhabricatorPirateEnglishTranslation' => 'PhutilTranslation',
|
||||||
'PhabricatorPlatformSite' => 'PhabricatorSite',
|
'PhabricatorPlatformSite' => 'PhabricatorSite',
|
||||||
'PhabricatorPointsEditField' => 'PhabricatorEditField',
|
'PhabricatorPointsEditField' => 'PhabricatorEditField',
|
||||||
|
|
|
@ -15,9 +15,8 @@ abstract class PhabricatorHomeController extends PhabricatorController {
|
||||||
->withLaunchable(true)
|
->withLaunchable(true)
|
||||||
->execute();
|
->execute();
|
||||||
|
|
||||||
$pinned = $user->loadPreferences()->getPinnedApplications(
|
$pinned = $user->getUserSetting(
|
||||||
$applications,
|
PhabricatorPinnedApplicationsSetting::SETTINGKEY);
|
||||||
$user);
|
|
||||||
|
|
||||||
// Force "Applications" to appear at the bottom.
|
// Force "Applications" to appear at the bottom.
|
||||||
$meta_app = 'PhabricatorApplicationsApplication';
|
$meta_app = 'PhabricatorApplicationsApplication';
|
||||||
|
|
|
@ -26,12 +26,7 @@ abstract class PhabricatorEditEngineSettingsPanel
|
||||||
->setIsSelfEdit($is_self)
|
->setIsSelfEdit($is_self)
|
||||||
->setProfileURI($profile_uri);
|
->setProfileURI($profile_uri);
|
||||||
|
|
||||||
$preferences = $user->loadPreferences();
|
$preferences = $this->loadTargetPreferences();
|
||||||
|
|
||||||
PhabricatorPolicyFilter::requireCapability(
|
|
||||||
$viewer,
|
|
||||||
$preferences,
|
|
||||||
PhabricatorPolicyCapability::CAN_EDIT);
|
|
||||||
|
|
||||||
$engine->setTargetObject($preferences);
|
$engine->setTargetObject($preferences);
|
||||||
|
|
||||||
|
|
|
@ -16,18 +16,19 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
}
|
}
|
||||||
|
|
||||||
public function processRequest(AphrontRequest $request) {
|
public function processRequest(AphrontRequest $request) {
|
||||||
$user = $request->getUser();
|
$viewer = $this->getViewer();
|
||||||
$preferences = $user->loadPreferences();
|
$preferences = $this->loadTargetPreferences();
|
||||||
|
|
||||||
|
$pinned_key = PhabricatorPinnedApplicationsSetting::SETTINGKEY;
|
||||||
|
$pinned = $preferences->getSettingValue($pinned_key);
|
||||||
|
|
||||||
$apps = id(new PhabricatorApplicationQuery())
|
$apps = id(new PhabricatorApplicationQuery())
|
||||||
->setViewer($user)
|
->setViewer($viewer)
|
||||||
->withInstalled(true)
|
->withInstalled(true)
|
||||||
->withUnlisted(false)
|
->withUnlisted(false)
|
||||||
->withLaunchable(true)
|
->withLaunchable(true)
|
||||||
->execute();
|
->execute();
|
||||||
|
|
||||||
$pinned = $preferences->getPinnedApplications($apps, $user);
|
|
||||||
|
|
||||||
$app_list = array();
|
$app_list = array();
|
||||||
foreach ($pinned as $app) {
|
foreach ($pinned as $app) {
|
||||||
if (isset($apps[$app])) {
|
if (isset($apps[$app])) {
|
||||||
|
@ -35,6 +36,23 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($request->getBool('reset')) {
|
||||||
|
if ($request->isFormPost()) {
|
||||||
|
$this->writePinnedApplications($preferences, null);
|
||||||
|
return id(new AphrontRedirectResponse())
|
||||||
|
->setURI($this->getPanelURI());
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->newDialog()
|
||||||
|
->setTitle(pht('Reset Applications'))
|
||||||
|
->addHiddenInput('reset', 'true')
|
||||||
|
->appendParagraph(
|
||||||
|
pht('Reset pinned applications to their defaults?'))
|
||||||
|
->addSubmitButton(pht('Reset Applications'))
|
||||||
|
->addCancelButton($this->getPanelURI());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if ($request->getBool('add')) {
|
if ($request->getBool('add')) {
|
||||||
$options = array();
|
$options = array();
|
||||||
foreach ($apps as $app) {
|
foreach ($apps as $app) {
|
||||||
|
@ -48,10 +66,8 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
$pin = $request->getStr('pin');
|
$pin = $request->getStr('pin');
|
||||||
if (isset($options[$pin]) && !in_array($pin, $pinned)) {
|
if (isset($options[$pin]) && !in_array($pin, $pinned)) {
|
||||||
$pinned[] = $pin;
|
$pinned[] = $pin;
|
||||||
$preferences->setPreference(
|
|
||||||
PhabricatorUserPreferences::PREFERENCE_APP_PINNED,
|
$this->writePinnedApplications($preferences, $pinned);
|
||||||
$pinned);
|
|
||||||
$preferences->save();
|
|
||||||
|
|
||||||
return id(new AphrontRedirectResponse())
|
return id(new AphrontRedirectResponse())
|
||||||
->setURI($this->getPanelURI());
|
->setURI($this->getPanelURI());
|
||||||
|
@ -65,21 +81,18 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
->setDisabledOptions(array_keys($app_list));
|
->setDisabledOptions(array_keys($app_list));
|
||||||
|
|
||||||
$form = id(new AphrontFormView())
|
$form = id(new AphrontFormView())
|
||||||
->setUser($user)
|
->setViewer($viewer)
|
||||||
->addHiddenInput('add', 'true')
|
->addHiddenInput('add', 'true')
|
||||||
->appendRemarkupInstructions(
|
->appendRemarkupInstructions(
|
||||||
pht('Choose an application to pin to your home page.'))
|
pht('Choose an application to pin to your home page.'))
|
||||||
->appendChild($options_control);
|
->appendChild($options_control);
|
||||||
|
|
||||||
$dialog = id(new AphrontDialogView())
|
return $this->newDialog()
|
||||||
->setUser($user)
|
|
||||||
->setWidth(AphrontDialogView::WIDTH_FORM)
|
->setWidth(AphrontDialogView::WIDTH_FORM)
|
||||||
->setTitle(pht('Pin Application'))
|
->setTitle(pht('Pin Application'))
|
||||||
->appendChild($form->buildLayoutView())
|
->appendChild($form->buildLayoutView())
|
||||||
->addSubmitButton(pht('Pin Application'))
|
->addSubmitButton(pht('Pin Application'))
|
||||||
->addCancelButton($this->getPanelURI());
|
->addCancelButton($this->getPanelURI());
|
||||||
|
|
||||||
return id(new AphrontDialogResponse())->setDialog($dialog);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$unpin = $request->getStr('unpin');
|
$unpin = $request->getStr('unpin');
|
||||||
|
@ -88,35 +101,28 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
if ($app) {
|
if ($app) {
|
||||||
if ($request->isFormPost()) {
|
if ($request->isFormPost()) {
|
||||||
$pinned = array_diff($pinned, array($unpin));
|
$pinned = array_diff($pinned, array($unpin));
|
||||||
$preferences->setPreference(
|
|
||||||
PhabricatorUserPreferences::PREFERENCE_APP_PINNED,
|
$this->writePinnedApplications($preferences, $pinned);
|
||||||
$pinned);
|
|
||||||
$preferences->save();
|
|
||||||
|
|
||||||
return id(new AphrontRedirectResponse())
|
return id(new AphrontRedirectResponse())
|
||||||
->setURI($this->getPanelURI());
|
->setURI($this->getPanelURI());
|
||||||
}
|
}
|
||||||
|
|
||||||
$dialog = id(new AphrontDialogView())
|
return $this->newDialog()
|
||||||
->setUser($user)
|
|
||||||
->setTitle(pht('Unpin Application'))
|
->setTitle(pht('Unpin Application'))
|
||||||
|
->addHiddenInput('unpin', $unpin)
|
||||||
->appendParagraph(
|
->appendParagraph(
|
||||||
pht(
|
pht(
|
||||||
'Unpin the %s application from your home page?',
|
'Unpin the %s application from your home page?',
|
||||||
phutil_tag('strong', array(), $app->getName())))
|
phutil_tag('strong', array(), $app->getName())))
|
||||||
->addSubmitButton(pht('Unpin Application'))
|
->addSubmitButton(pht('Unpin Application'))
|
||||||
->addCanceLButton($this->getPanelURI());
|
->addCancelButton($this->getPanelURI());
|
||||||
|
|
||||||
return id(new AphrontDialogResponse())->setDialog($dialog);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$order = $request->getStrList('order');
|
$order = $request->getStrList('order');
|
||||||
if ($order && $request->validateCSRF()) {
|
if ($order && $request->validateCSRF()) {
|
||||||
$preferences->setPreference(
|
$this->writePinnedApplications($preferences, $order);
|
||||||
PhabricatorUserPreferences::PREFERENCE_APP_PINNED,
|
|
||||||
$order);
|
|
||||||
$preferences->save();
|
|
||||||
|
|
||||||
return id(new AphrontRedirectResponse())
|
return id(new AphrontRedirectResponse())
|
||||||
->setURI($this->getPanelURI());
|
->setURI($this->getPanelURI());
|
||||||
|
@ -125,7 +131,7 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
$list_id = celerity_generate_unique_node_id();
|
$list_id = celerity_generate_unique_node_id();
|
||||||
|
|
||||||
$list = id(new PHUIObjectItemListView())
|
$list = id(new PHUIObjectItemListView())
|
||||||
->setUser($user)
|
->setViewer($viewer)
|
||||||
->setID($list_id);
|
->setID($list_id);
|
||||||
|
|
||||||
Javelin::initBehavior(
|
Javelin::initBehavior(
|
||||||
|
@ -182,7 +188,14 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
->setText(pht('Pin Application'))
|
->setText(pht('Pin Application'))
|
||||||
->setHref($this->getPanelURI().'?add=true')
|
->setHref($this->getPanelURI().'?add=true')
|
||||||
->setWorkflow(true)
|
->setWorkflow(true)
|
||||||
->setIcon('fa-thumb-tack'));
|
->setIcon('fa-thumb-tack'))
|
||||||
|
->addActionLink(
|
||||||
|
id(new PHUIButtonView())
|
||||||
|
->setTag('a')
|
||||||
|
->setText(pht('Reset to Defaults'))
|
||||||
|
->setHref($this->getPanelURI().'?reset=true')
|
||||||
|
->setWorkflow(true)
|
||||||
|
->setIcon('fa-recycle'));
|
||||||
|
|
||||||
$box = id(new PHUIObjectBoxView())
|
$box = id(new PHUIObjectBoxView())
|
||||||
->setHeader($header)
|
->setHeader($header)
|
||||||
|
@ -191,4 +204,23 @@ final class PhabricatorHomePreferencesSettingsPanel
|
||||||
return $box;
|
return $box;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function writePinnedApplications(
|
||||||
|
PhabricatorUserPreferences $preferences,
|
||||||
|
$pinned) {
|
||||||
|
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
$request = $this->getController()->getRequest();
|
||||||
|
$pinned_key = PhabricatorPinnedApplicationsSetting::SETTINGKEY;
|
||||||
|
|
||||||
|
$editor = id(new PhabricatorUserPreferencesEditor())
|
||||||
|
->setActor($viewer)
|
||||||
|
->setContentSourceFromRequest($request)
|
||||||
|
->setContinueOnNoEffect(true)
|
||||||
|
->setContinueOnMissingFields(true);
|
||||||
|
|
||||||
|
$xactions = array();
|
||||||
|
$xactions[] = $preferences->newTransaction($pinned_key, $pinned);
|
||||||
|
$editor->applyTransactions($preferences, $xactions);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -217,4 +217,22 @@ abstract class PhabricatorSettingsPanel extends Phobject {
|
||||||
->addString($this->getPanelName());
|
->addString($this->getPanelName());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function loadTargetPreferences() {
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
$user = $this->getUser();
|
||||||
|
|
||||||
|
$preferences = PhabricatorUserPreferences::loadUserPreferences($user);
|
||||||
|
|
||||||
|
PhabricatorPolicyFilter::requireCapability(
|
||||||
|
$viewer,
|
||||||
|
$preferences,
|
||||||
|
PhabricatorPolicyCapability::CAN_EDIT);
|
||||||
|
|
||||||
|
return $preferences;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function newDialog() {
|
||||||
|
return $this->getController()->newDialog();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,33 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorPinnedApplicationsSetting
|
||||||
|
extends PhabricatorInternalSetting {
|
||||||
|
|
||||||
|
const SETTINGKEY = 'app-pinned';
|
||||||
|
|
||||||
|
public function getSettingName() {
|
||||||
|
return pht('Pinned Applications');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSettingDefaultValue() {
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
|
||||||
|
$applications = id(new PhabricatorApplicationQuery())
|
||||||
|
->setViewer($viewer)
|
||||||
|
->withInstalled(true)
|
||||||
|
->withUnlisted(false)
|
||||||
|
->withLaunchable(true)
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
$pinned = array();
|
||||||
|
foreach ($applications as $application) {
|
||||||
|
if ($application->isPinnedByDefault($viewer)) {
|
||||||
|
$pinned[] = get_class($application);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $pinned;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -73,49 +73,28 @@ final class PhabricatorUserPreferences
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$setting = id(clone $setting)
|
||||||
|
->setViewer($this->getUser());
|
||||||
|
|
||||||
return $setting->getSettingDefaultValue();
|
return $setting->getSettingDefaultValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getSettingValue($key) {
|
||||||
|
if (array_key_exists($key, $this->preferences)) {
|
||||||
|
return $this->preferences[$key];
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: If this setting set inherits from another preference set,
|
||||||
|
// we would look it up here.
|
||||||
|
|
||||||
|
return $this->getDefaultValue($key);
|
||||||
|
}
|
||||||
|
|
||||||
private static function getSettingObject($key) {
|
private static function getSettingObject($key) {
|
||||||
$settings = PhabricatorSetting::getAllSettings();
|
$settings = PhabricatorSetting::getAllSettings();
|
||||||
return idx($settings, $key);
|
return idx($settings, $key);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getPinnedApplications(array $apps, PhabricatorUser $viewer) {
|
|
||||||
$pref_pinned = self::PREFERENCE_APP_PINNED;
|
|
||||||
$pinned = $this->getPreference($pref_pinned);
|
|
||||||
|
|
||||||
if ($pinned) {
|
|
||||||
return $pinned;
|
|
||||||
}
|
|
||||||
|
|
||||||
$pref_tiles = self::PREFERENCE_APP_TILES;
|
|
||||||
$tiles = $this->getPreference($pref_tiles, array());
|
|
||||||
$full_tile = 'full';
|
|
||||||
|
|
||||||
$large = array();
|
|
||||||
foreach ($apps as $app) {
|
|
||||||
$show = $app->isPinnedByDefault($viewer);
|
|
||||||
|
|
||||||
// TODO: This is legacy stuff, clean it up eventually. This approximately
|
|
||||||
// retains the old "tiles" preference.
|
|
||||||
if (isset($tiles[get_class($app)])) {
|
|
||||||
$show = ($tiles[get_class($app)] == $full_tile);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($show) {
|
|
||||||
$large[] = get_class($app);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $large;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function filterMonospacedCSSRule($monospaced) {
|
|
||||||
// Prevent the user from doing dangerous things.
|
|
||||||
return preg_replace('([^a-z0-9 ,"./]+)i', '', $monospaced);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function attachUser(PhabricatorUser $user = null) {
|
public function attachUser(PhabricatorUser $user = null) {
|
||||||
$this->user = $user;
|
$this->user = $user;
|
||||||
return $this;
|
return $this;
|
||||||
|
|
Loading…
Reference in a new issue