1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-10 14:51:06 +01:00

Merge profile "About" into main profile and simplify some custom field stuff

Summary:
Ref T1703. Drive "user since" with a custom field and make the other fields render into a property list.

Users can make their profiles a little more personal/obnoxious now.

Also delete a bunch of code.

Test Plan: {F49415}

Reviewers: chad, btrahan

Reviewed By: chad

CC: aran

Maniphest Tasks: T1703

Differential Revision: https://secure.phabricator.com/D6401
This commit is contained in:
epriestley 2013-07-10 05:09:59 -07:00
parent c20b7540c9
commit b92fe7dbda
13 changed files with 197 additions and 106 deletions

View file

@ -1616,6 +1616,7 @@ phutil_register_library_map(array(
'PhabricatorUserRealNameField' => 'applications/people/customfield/PhabricatorUserRealNameField.php', 'PhabricatorUserRealNameField' => 'applications/people/customfield/PhabricatorUserRealNameField.php',
'PhabricatorUserSSHKey' => 'applications/settings/storage/PhabricatorUserSSHKey.php', 'PhabricatorUserSSHKey' => 'applications/settings/storage/PhabricatorUserSSHKey.php',
'PhabricatorUserSearchIndexer' => 'applications/people/search/PhabricatorUserSearchIndexer.php', 'PhabricatorUserSearchIndexer' => 'applications/people/search/PhabricatorUserSearchIndexer.php',
'PhabricatorUserSinceField' => 'applications/people/customfield/PhabricatorUserSinceField.php',
'PhabricatorUserStatus' => 'applications/people/storage/PhabricatorUserStatus.php', 'PhabricatorUserStatus' => 'applications/people/storage/PhabricatorUserStatus.php',
'PhabricatorUserStatusInvalidEpochException' => 'applications/people/exception/PhabricatorUserStatusInvalidEpochException.php', 'PhabricatorUserStatusInvalidEpochException' => 'applications/people/exception/PhabricatorUserStatusInvalidEpochException.php',
'PhabricatorUserStatusOverlapException' => 'applications/people/exception/PhabricatorUserStatusOverlapException.php', 'PhabricatorUserStatusOverlapException' => 'applications/people/exception/PhabricatorUserStatusOverlapException.php',
@ -3568,6 +3569,7 @@ phutil_register_library_map(array(
'PhabricatorUserRealNameField' => 'PhabricatorUserCustomField', 'PhabricatorUserRealNameField' => 'PhabricatorUserCustomField',
'PhabricatorUserSSHKey' => 'PhabricatorUserDAO', 'PhabricatorUserSSHKey' => 'PhabricatorUserDAO',
'PhabricatorUserSearchIndexer' => 'PhabricatorSearchDocumentIndexer', 'PhabricatorUserSearchIndexer' => 'PhabricatorSearchDocumentIndexer',
'PhabricatorUserSinceField' => 'PhabricatorUserCustomField',
'PhabricatorUserStatus' => 'PhabricatorUserDAO', 'PhabricatorUserStatus' => 'PhabricatorUserDAO',
'PhabricatorUserStatusInvalidEpochException' => 'Exception', 'PhabricatorUserStatusInvalidEpochException' => 'Exception',
'PhabricatorUserStatusOverlapException' => 'Exception', 'PhabricatorUserStatusOverlapException' => 'Exception',

View file

@ -16,6 +16,7 @@ final class PhabricatorUserConfigOptions
$default = array( $default = array(
id(new PhabricatorUserRealNameField())->getFieldKey() => true, id(new PhabricatorUserRealNameField())->getFieldKey() => true,
id(new PhabricatorUserTitleField())->getFieldKey() => true, id(new PhabricatorUserTitleField())->getFieldKey() => true,
id(new PhabricatorUserSinceField())->getFieldKey() => true,
id(new PhabricatorUserBlurbField())->getFieldKey() => true, id(new PhabricatorUserBlurbField())->getFieldKey() => true,
); );

View file

@ -5,7 +5,6 @@ final class PhabricatorPeopleProfileController
private $username; private $username;
private $page; private $page;
private $profileUser;
public function shouldRequireAdmin() { public function shouldRequireAdmin() {
// Default for people app is true // Default for people app is true
@ -18,10 +17,6 @@ final class PhabricatorPeopleProfileController
$this->page = idx($data, 'page'); $this->page = idx($data, 'page');
} }
public function getProfileUser() {
return $this->profileUser;
}
private function getMainFilters($username) { private function getMainFilters($username) {
return array( return array(
array( array(
@ -29,11 +24,6 @@ final class PhabricatorPeopleProfileController
'name' => pht('Feed'), 'name' => pht('Feed'),
'href' => '/p/'.$username.'/feed/' 'href' => '/p/'.$username.'/feed/'
), ),
array(
'key' => 'about',
'name' => pht('About'),
'href' => '/p/'.$username.'/about/'
)
); );
} }
@ -48,8 +38,6 @@ final class PhabricatorPeopleProfileController
return new Aphront404Response(); return new Aphront404Response();
} }
$this->profileUser = $user;
require_celerity_resource('phabricator-profile-css'); require_celerity_resource('phabricator-profile-css');
$profile = $user->loadUserProfile(); $profile = $user->loadUserProfile();
@ -76,16 +64,7 @@ final class PhabricatorPeopleProfileController
$this->page = $nav->selectFilter($this->page, 'feed'); $this->page = $nav->selectFilter($this->page, 'feed');
switch ($this->page) {
case 'feed':
$content = $this->renderUserFeed($user); $content = $this->renderUserFeed($user);
break;
case 'about':
$content = $this->renderBasicInformation($user, $profile);
break;
default:
throw new Exception("Unknown page '{$this->page}'!");
}
$picture = $user->loadProfileImageURI(); $picture = $user->loadProfileImageURI();
@ -133,8 +112,11 @@ final class PhabricatorPeopleProfileController
->setHref($this->getApplicationURI('edit/'.$user->getID().'/'))); ->setHref($this->getApplicationURI('edit/'.$user->getID().'/')));
} }
$properties = $this->buildPropertyView($user);
$nav->appendChild($header); $nav->appendChild($header);
$nav->appendChild($actions); $nav->appendChild($actions);
$nav->appendChild($properties);
$nav->appendChild($content); $nav->appendChild($content);
return $this->buildApplicationPage( return $this->buildApplicationPage(
@ -146,55 +128,24 @@ final class PhabricatorPeopleProfileController
)); ));
} }
private function renderBasicInformation($user, $profile) { private function buildPropertyView(PhabricatorUser $user) {
$blurb = nonempty(
$profile->getBlurb(),
'//'.pht('Nothing is known about this rare specimen.').'//');
$viewer = $this->getRequest()->getUser(); $viewer = $this->getRequest()->getUser();
$engine = PhabricatorMarkupEngine::newProfileMarkupEngine(); $view = id(new PhabricatorPropertyListView())
$engine->setConfig('viewer', $viewer); ->setUser($viewer)
$blurb = $engine->markupText($blurb); ->setObject($user);
$content = hsprintf( $fields = PhabricatorCustomField::getObjectFields(
'<div class="phabricator-profile-info-group profile-wrap-responsive"> $user,
<h1 class="phabricator-profile-info-header">%s</h1> PhabricatorCustomField::ROLE_VIEW);
<div class="phabricator-profile-info-pane">
<table class="phabricator-profile-info-table">
<tr>
<th>%s</th>
<td>%s</td>
</tr>
<tr>
<th>%s</th>
<td>%s</td>
</tr>
</table>
</div>
</div>'.
'<div class="phabricator-profile-info-group profile-wrap-responsive">
<h1 class="phabricator-profile-info-header">%s</h1>
<div class="phabricator-profile-info-pane">
<table class="phabricator-profile-info-table">
<tr>
<th>%s</th>
<td>%s</td>
</tr>
</table>
</div>
</div>',
pht('Basic Information'),
pht('PHID'),
$user->getPHID(),
pht('User Since'),
phabricator_datetime($user->getDateCreated(), $viewer),
pht('Flavor Text'),
pht('Blurb'),
$blurb);
return $content; foreach ($fields as $field) {
$field->setViewer($viewer);
}
$view->applyCustomFields($fields);
return $view;
} }
private function renderUserFeed(PhabricatorUser $user) { private function renderUserFeed(PhabricatorUser $user) {

View file

@ -34,7 +34,7 @@ final class PhabricatorPeopleProfileEditController
$fields = PhabricatorCustomField::getObjectFields( $fields = PhabricatorCustomField::getObjectFields(
$user, $user,
PhabricatorUserCustomFieldInterface::ROLE_EDIT); PhabricatorCustomField::ROLE_EDIT);
if ($request->isFormPost()) { if ($request->isFormPost()) {
$xactions = array(); $xactions = array();

View file

@ -17,7 +17,15 @@ final class PhabricatorUserBlurbField
return pht('Short blurb about the user.'); return pht('Short blurb about the user.');
} }
public function canDisableField() { public function shouldAppearInApplicationTransactions() {
return true;
}
public function shouldAppearInEditView() {
return true;
}
public function shouldAppearInPropertyView() {
return true; return true;
} }
@ -49,4 +57,23 @@ final class PhabricatorUserBlurbField
->setLabel($this->getFieldName()); ->setLabel($this->getFieldName());
} }
public function renderPropertyViewLabel() {
return null;
}
public function renderPropertyViewValue() {
$blurb = $this->getObject()->loadUserProfile()->getBlurb();
if (!strlen($blurb)) {
return null;
}
return PhabricatorMarkupEngine::renderOneObject(
id(new PhabricatorMarkupOneOff())->setContent($blurb),
'default',
$this->getViewer());
}
public function getStyleForPropertyView() {
return 'block';
}
} }

View file

@ -5,28 +5,4 @@ abstract class PhabricatorUserCustomField
implements PhabricatorUserCustomFieldInterface { implements PhabricatorUserCustomFieldInterface {
public function shouldEnableForRole($role) {
switch ($role) {
case PhabricatorUserCustomFieldInterface::ROLE_EDIT:
return $this->shouldAppearOnProfileEdit();
}
return parent::shouldEnableForRole($role);
}
public function shouldAppearOnProfileEdit() {
return true;
}
/* -( PhabricatorCustomField )--------------------------------------------- */
public function canDisableField() {
return false;
}
public function shouldAppearInApplicationTransactions() {
return true;
}
} }

View file

@ -2,8 +2,5 @@
interface PhabricatorUserCustomFieldInterface { interface PhabricatorUserCustomFieldInterface {
const ROLE_EDIT = 'user.edit';
public function shouldAppearOnProfileEdit();
} }

View file

@ -17,6 +17,18 @@ final class PhabricatorUserRealNameField
return pht('Stores the real name of the user, like "Abraham Lincoln".'); return pht('Stores the real name of the user, like "Abraham Lincoln".');
} }
public function canDisableField() {
return false;
}
public function shouldAppearInApplicationTransactions() {
return true;
}
public function shouldAppearInEditView() {
return true;
}
protected function didSetObject(PhabricatorCustomFieldInterface $object) { protected function didSetObject(PhabricatorCustomFieldInterface $object) {
$this->value = $object->getRealName(); $this->value = $object->getRealName();
} }

View file

@ -0,0 +1,30 @@
<?php
final class PhabricatorUserSinceField
extends PhabricatorUserCustomField {
private $value;
public function getFieldKey() {
return 'user:since';
}
public function getFieldName() {
return pht('User Since');
}
public function getFieldDescription() {
return pht('Shows user join date.');
}
public function shouldAppearInPropertyView() {
return true;
}
public function renderPropertyViewValue() {
return phabricator_datetime(
$this->getObject()->getDateCreated(),
$this->getViewer());
}
}

View file

@ -17,6 +17,18 @@ final class PhabricatorUserTitleField
return pht('User title, like "CEO" or "Assistant to the Manager".'); return pht('User title, like "CEO" or "Assistant to the Manager".');
} }
public function canDisableField() {
return false;
}
public function shouldAppearInApplicationTransactions() {
return true;
}
public function shouldAppearInEditView() {
return true;
}
protected function didSetObject(PhabricatorCustomFieldInterface $object) { protected function didSetObject(PhabricatorCustomFieldInterface $object) {
$this->value = $object->loadUserProfile()->getTitle(); $this->value = $object->loadUserProfile()->getTitle();
} }

View file

@ -7,6 +7,9 @@
* @task storage Field Storage * @task storage Field Storage
* @task appsearch Integration with ApplicationSearch * @task appsearch Integration with ApplicationSearch
* @task appxaction Integration with ApplicationTransactions * @task appxaction Integration with ApplicationTransactions
* @task edit Integration with edit views
* @task view Integration with property views
* @task list Integration with list views
*/ */
abstract class PhabricatorCustomField { abstract class PhabricatorCustomField {
@ -17,6 +20,9 @@ abstract class PhabricatorCustomField {
const ROLE_APPLICATIONSEARCH = 'ApplicationSearch'; const ROLE_APPLICATIONSEARCH = 'ApplicationSearch';
const ROLE_STORAGE = 'storage'; const ROLE_STORAGE = 'storage';
const ROLE_DEFAULT = 'default'; const ROLE_DEFAULT = 'default';
const ROLE_EDIT = 'edit';
const ROLE_VIEW = 'view';
const ROLE_LIST = 'list';
/* -( Building Applications with Custom Fields )--------------------------- */ /* -( Building Applications with Custom Fields )--------------------------- */
@ -219,6 +225,12 @@ abstract class PhabricatorCustomField {
return $this->shouldAppearInApplicationSearch(); return $this->shouldAppearInApplicationSearch();
case self::ROLE_STORAGE: case self::ROLE_STORAGE:
return ($this->getStorageKey() !== null); return ($this->getStorageKey() !== null);
case self::ROLE_EDIT:
return $this->shouldAppearInEditView();
case self::ROLE_VIEW:
return $this->shouldAppearInPropertyView();
case self::ROLE_LIST:
return $this->shouldAppearInListView();
case self::ROLE_DEFAULT: case self::ROLE_DEFAULT:
return true; return true;
default: default:
@ -605,7 +617,7 @@ abstract class PhabricatorCustomField {
/** /**
* @task edit * @task edit
*/ */
public function shouldAppearOnEditView() { public function shouldAppearInEditView() {
return false; return false;
} }
@ -626,4 +638,59 @@ abstract class PhabricatorCustomField {
} }
/* -( Property View )------------------------------------------------------ */
/**
* @task view
*/
public function shouldAppearInPropertyView() {
return false;
}
/**
* @task view
*/
public function renderPropertyViewLabel() {
return $this->getFieldName();
}
/**
* @task view
*/
public function renderPropertyViewValue() {
throw new PhabricatorCustomFieldImplementationIncompleteException($this);
}
/**
* @task view
*/
public function getStyleForPropertyView() {
return 'property';
}
/* -( List View )---------------------------------------------------------- */
/**
* @task list
*/
public function shouldAppearInListView() {
return false;
}
/**
* @task list
*/
public function renderOnListItem(PhabricatorObjectItemView $view) {
throw new PhabricatorCustomFieldImplementationIncompleteException($this);
}
} }

View file

@ -346,15 +346,6 @@ final class PhabricatorMarkupEngine {
} }
/**
* @task engine
*/
public static function newProfileMarkupEngine() {
return self::newMarkupEngine(array(
));
}
/** /**
* @task engine * @task engine
*/ */

View file

@ -81,7 +81,32 @@ final class PhabricatorPropertyListView extends AphrontView {
$this->invokedWillRenderEvent = true; $this->invokedWillRenderEvent = true;
} }
public function applyCustomFields(array $fields) {
assert_instances_of($fields, 'PhabricatorCustomField');
foreach ($fields as $field) {
$label = $field->renderPropertyViewLabel();
$value = $field->renderPropertyViewValue();
if ($value !== null) {
switch ($field->getStyleForPropertyView()) {
case 'property':
$this->addProperty($label, $value);
break;
case 'block':
$this->invokeWillRenderEvent();
if ($label !== null) {
$this->addSectionHeader($label);
}
$this->addTextContent($value);
break;
default:
throw new Exception(
"Unknown field property view style; valid styles are ".
"'block' and 'property'.");
}
}
}
}
public function render() { public function render() {
$this->invokeWillRenderEvent(); $this->invokeWillRenderEvent();