mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-18 10:41:08 +01:00
Begin lifting column layout logic out of ColumnPositionQuery
Summary: Ref T10010. This is a precursor to D15171, which I'll eventually rebuild on top of these changes. Currently, ColumnPositionQuery does a lot of "column layout" stuff that's very similar to the Milestone/Subproject stuff that needs to happen in D15171. The current approach there ended up splitting this layout stuff across two unrelated classes (ColumnPositionQuery + BoardViewController), neither of which is a particularly great place to do it -- the Query is too low-level, and the Controller is too high-level. Instead, introduce a new "LayoutEngine" which does all this layout stuff. Swap two of the four places that we query this stuff over to the new engine: - "Project (Column)" on tasks. - Transaction generation when moving cards. These sites aren't swapped by this diff, but will be by the next one: - Actually applying transactions. - Main layout for boards (this could swap easily now, but applying transactions currently relies on position writes having taken place, so it can't swap until the other one swaps). Once everything is swapped over, I should be able to add the D15171 logic to LayoutEngine instead of BoardViewController and end up with a cleaner approach overall. One particularly benefit is that //looking// at a board won't do a bunch of position writes anymore, which wasn't a big deal, but which I was a bit uneasy with. Test Plan: - Viewed tasks that are on boards, saw column annotations in project list. - Moved cards between columns on a board. Reviewers: chad Reviewed By: chad Maniphest Tasks: T10010 Differential Revision: https://secure.phabricator.com/D15174
This commit is contained in:
parent
0a735694ae
commit
23b835b647
5 changed files with 218 additions and 29 deletions
|
@ -1814,6 +1814,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorBcryptPasswordHasher' => 'infrastructure/util/password/PhabricatorBcryptPasswordHasher.php',
|
'PhabricatorBcryptPasswordHasher' => 'infrastructure/util/password/PhabricatorBcryptPasswordHasher.php',
|
||||||
'PhabricatorBinariesSetupCheck' => 'applications/config/check/PhabricatorBinariesSetupCheck.php',
|
'PhabricatorBinariesSetupCheck' => 'applications/config/check/PhabricatorBinariesSetupCheck.php',
|
||||||
'PhabricatorBitbucketAuthProvider' => 'applications/auth/provider/PhabricatorBitbucketAuthProvider.php',
|
'PhabricatorBitbucketAuthProvider' => 'applications/auth/provider/PhabricatorBitbucketAuthProvider.php',
|
||||||
|
'PhabricatorBoardLayoutEngine' => 'applications/project/engine/PhabricatorBoardLayoutEngine.php',
|
||||||
'PhabricatorBot' => 'infrastructure/daemon/bot/PhabricatorBot.php',
|
'PhabricatorBot' => 'infrastructure/daemon/bot/PhabricatorBot.php',
|
||||||
'PhabricatorBotChannel' => 'infrastructure/daemon/bot/target/PhabricatorBotChannel.php',
|
'PhabricatorBotChannel' => 'infrastructure/daemon/bot/target/PhabricatorBotChannel.php',
|
||||||
'PhabricatorBotDebugLogHandler' => 'infrastructure/daemon/bot/handler/PhabricatorBotDebugLogHandler.php',
|
'PhabricatorBotDebugLogHandler' => 'infrastructure/daemon/bot/handler/PhabricatorBotDebugLogHandler.php',
|
||||||
|
@ -6038,6 +6039,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorBcryptPasswordHasher' => 'PhabricatorPasswordHasher',
|
'PhabricatorBcryptPasswordHasher' => 'PhabricatorPasswordHasher',
|
||||||
'PhabricatorBinariesSetupCheck' => 'PhabricatorSetupCheck',
|
'PhabricatorBinariesSetupCheck' => 'PhabricatorSetupCheck',
|
||||||
'PhabricatorBitbucketAuthProvider' => 'PhabricatorOAuth1AuthProvider',
|
'PhabricatorBitbucketAuthProvider' => 'PhabricatorOAuth1AuthProvider',
|
||||||
|
'PhabricatorBoardLayoutEngine' => 'Phobject',
|
||||||
'PhabricatorBot' => 'PhabricatorDaemon',
|
'PhabricatorBot' => 'PhabricatorDaemon',
|
||||||
'PhabricatorBotChannel' => 'PhabricatorBotTarget',
|
'PhabricatorBotChannel' => 'PhabricatorBotTarget',
|
||||||
'PhabricatorBotDebugLogHandler' => 'PhabricatorBotHandler',
|
'PhabricatorBotDebugLogHandler' => 'PhabricatorBotHandler',
|
||||||
|
|
|
@ -26,6 +26,8 @@ final class PhabricatorProjectMoveController
|
||||||
return new Aphront404Response();
|
return new Aphront404Response();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$board_phid = $project->getPHID();
|
||||||
|
|
||||||
$object = id(new ManiphestTaskQuery())
|
$object = id(new ManiphestTaskQuery())
|
||||||
->setViewer($viewer)
|
->setViewer($viewer)
|
||||||
->withPHIDs(array($object_phid))
|
->withPHIDs(array($object_phid))
|
||||||
|
@ -54,11 +56,14 @@ final class PhabricatorProjectMoveController
|
||||||
return new Aphront404Response();
|
return new Aphront404Response();
|
||||||
}
|
}
|
||||||
|
|
||||||
$positions = id(new PhabricatorProjectColumnPositionQuery())
|
$engine = id(new PhabricatorBoardLayoutEngine())
|
||||||
->setViewer($viewer)
|
->setViewer($viewer)
|
||||||
->withColumns($columns)
|
->setBoardPHIDs(array($board_phid))
|
||||||
->withObjectPHIDs(array($object_phid))
|
->setObjectPHIDs(array($object_phid))
|
||||||
->execute();
|
->executeLayout();
|
||||||
|
|
||||||
|
$columns = $engine->getObjectColumns($board_phid, $object_phid);
|
||||||
|
$old_column_phids = mpull($columns, 'getPHID');
|
||||||
|
|
||||||
$xactions = array();
|
$xactions = array();
|
||||||
|
|
||||||
|
@ -80,7 +85,7 @@ final class PhabricatorProjectMoveController
|
||||||
) + $order_params)
|
) + $order_params)
|
||||||
->setOldValue(
|
->setOldValue(
|
||||||
array(
|
array(
|
||||||
'columnPHIDs' => mpull($positions, 'getColumnPHID'),
|
'columnPHIDs' => $old_column_phids,
|
||||||
'projectPHID' => $column->getProjectPHID(),
|
'projectPHID' => $column->getProjectPHID(),
|
||||||
));
|
));
|
||||||
|
|
||||||
|
|
187
src/applications/project/engine/PhabricatorBoardLayoutEngine.php
Normal file
187
src/applications/project/engine/PhabricatorBoardLayoutEngine.php
Normal file
|
@ -0,0 +1,187 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorBoardLayoutEngine extends Phobject {
|
||||||
|
|
||||||
|
private $viewer;
|
||||||
|
private $boardPHIDs;
|
||||||
|
private $objectPHIDs;
|
||||||
|
private $boards;
|
||||||
|
private $columnMap;
|
||||||
|
private $objectColumnMap = array();
|
||||||
|
private $boardLayout = array();
|
||||||
|
|
||||||
|
public function setViewer(PhabricatorUser $viewer) {
|
||||||
|
$this->viewer = $viewer;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getViewer() {
|
||||||
|
return $this->viewer;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setBoardPHIDs(array $board_phids) {
|
||||||
|
$this->boardPHIDs = $board_phids;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getBoardPHIDs() {
|
||||||
|
return $this->boardPHIDs;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setObjectPHIDs(array $object_phids) {
|
||||||
|
$this->objectPHIDs = $object_phids;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getObjectPHIDs() {
|
||||||
|
return $this->objectPHIDs;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function executeLayout() {
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
|
||||||
|
$boards = $this->loadBoards();
|
||||||
|
if (!$boards) {
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
$columns = $this->loadColumns($boards);
|
||||||
|
$positions = $this->loadPositions($boards);
|
||||||
|
|
||||||
|
foreach ($boards as $board_phid => $board) {
|
||||||
|
$board_columns = idx($columns, $board_phid);
|
||||||
|
|
||||||
|
// Don't layout boards with no columns. These boards need to be formally
|
||||||
|
// created first.
|
||||||
|
if (!$columns) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$board_positions = idx($positions, $board_phid, array());
|
||||||
|
|
||||||
|
$this->layoutBoard($board, $board_columns, $board_positions);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getObjectColumns($board_phid, $object_phid) {
|
||||||
|
$board_map = idx($this->objectColumnMap, $board_phid, array());
|
||||||
|
|
||||||
|
$column_phids = idx($board_map, $object_phid);
|
||||||
|
if (!$column_phids) {
|
||||||
|
return array();
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_select_keys($this->columnMap, $column_phids);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function loadBoards() {
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
$board_phids = $this->getBoardPHIDs();
|
||||||
|
|
||||||
|
$boards = id(new PhabricatorObjectQuery())
|
||||||
|
->setViewer($viewer)
|
||||||
|
->withPHIDs($board_phids)
|
||||||
|
->execute();
|
||||||
|
$boards = mpull($boards, null, 'getPHID');
|
||||||
|
|
||||||
|
foreach ($boards as $key => $board) {
|
||||||
|
if (!$board->getHasWorkboard()) {
|
||||||
|
unset($boards[$key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $boards;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function loadColumns(array $boards) {
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
|
||||||
|
$columns = id(new PhabricatorProjectColumnQuery())
|
||||||
|
->setViewer($viewer)
|
||||||
|
->withProjectPHIDs(array_keys($boards))
|
||||||
|
->execute();
|
||||||
|
$columns = msort($columns, 'getSequence');
|
||||||
|
$columns = mpull($columns, null, 'getPHID');
|
||||||
|
|
||||||
|
$this->columnMap = $columns;
|
||||||
|
$columns = mgroup($columns, 'getProjectPHID');
|
||||||
|
|
||||||
|
return $columns;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function loadPositions(array $boards) {
|
||||||
|
$viewer = $this->getViewer();
|
||||||
|
|
||||||
|
$positions = id(new PhabricatorProjectColumnPositionQuery())
|
||||||
|
->setViewer($viewer)
|
||||||
|
->withBoardPHIDs(array_keys($boards))
|
||||||
|
->withObjectPHIDs($this->getObjectPHIDs())
|
||||||
|
->execute();
|
||||||
|
$positions = msort($positions, 'getOrderingKey');
|
||||||
|
$positions = mgroup($positions, 'getBoardPHID');
|
||||||
|
|
||||||
|
return $positions;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function layoutBoard(
|
||||||
|
$board,
|
||||||
|
array $columns,
|
||||||
|
array $positions) {
|
||||||
|
|
||||||
|
$board_phid = $board->getPHID();
|
||||||
|
$position_groups = mgroup($positions, 'getObjectPHID');
|
||||||
|
|
||||||
|
foreach ($columns as $column) {
|
||||||
|
if ($column->isDefaultColumn()) {
|
||||||
|
$default_phid = $column->getPHID();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$layout = array();
|
||||||
|
|
||||||
|
$object_phids = $this->getObjectPHIDs();
|
||||||
|
foreach ($object_phids as $object_phid) {
|
||||||
|
$positions = idx($position_groups, $object_phid, array());
|
||||||
|
|
||||||
|
// Remove any positions in columns which no longer exist.
|
||||||
|
foreach ($positions as $key => $position) {
|
||||||
|
$column_phid = $position->getColumnPHID();
|
||||||
|
if (empty($columns[$column_phid])) {
|
||||||
|
unset($positions[$key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the object has no position, put it on the default column.
|
||||||
|
if (!$positions) {
|
||||||
|
$new_position = id(new PhabricatorProjectColumnPosition())
|
||||||
|
->setBoardPHID($board_phid)
|
||||||
|
->setColumnPHID($default_phid)
|
||||||
|
->setObjectPHID($object_phid)
|
||||||
|
->setSequence(0);
|
||||||
|
$positions = array(
|
||||||
|
$new_position,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($positions as $position) {
|
||||||
|
$column_phid = $position->getColumnPHID();
|
||||||
|
$layout[$column_phid][$object_phid] = $position;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($layout as $column_phid => $map) {
|
||||||
|
$map = msort($map, 'getOrderingKey');
|
||||||
|
$layout[$column_phid] = $map;
|
||||||
|
|
||||||
|
foreach ($map as $object_phid => $position) {
|
||||||
|
$this->objectColumnMap[$board_phid][$object_phid][] = $column_phid;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->boardLayout[$board_phid] = $layout;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -49,37 +49,24 @@ final class PhabricatorProjectUIEventListener
|
||||||
|
|
||||||
$annotations = array();
|
$annotations = array();
|
||||||
if ($handles && $can_appear_on_boards) {
|
if ($handles && $can_appear_on_boards) {
|
||||||
|
$engine = id(new PhabricatorBoardLayoutEngine())
|
||||||
|
->setViewer($user)
|
||||||
|
->setBoardPHIDs($project_phids)
|
||||||
|
->setObjectPHIDs(array($object->getPHID()))
|
||||||
|
->executeLayout();
|
||||||
|
|
||||||
// TDOO: Generalize this UI and move it out of Maniphest.
|
// TDOO: Generalize this UI and move it out of Maniphest.
|
||||||
|
|
||||||
require_celerity_resource('maniphest-task-summary-css');
|
require_celerity_resource('maniphest-task-summary-css');
|
||||||
|
|
||||||
$positions_query = id(new PhabricatorProjectColumnPositionQuery())
|
|
||||||
->setViewer($user)
|
|
||||||
->withBoardPHIDs($project_phids)
|
|
||||||
->withObjectPHIDs(array($object->getPHID()))
|
|
||||||
->needColumns(true);
|
|
||||||
|
|
||||||
// This is important because positions will be created "on demand"
|
|
||||||
// based on the set of columns. If we don't specify it, positions
|
|
||||||
// won't be created.
|
|
||||||
$columns = id(new PhabricatorProjectColumnQuery())
|
|
||||||
->setViewer($user)
|
|
||||||
->withProjectPHIDs($project_phids)
|
|
||||||
->execute();
|
|
||||||
if ($columns) {
|
|
||||||
$positions_query->withColumns($columns);
|
|
||||||
}
|
|
||||||
$positions = $positions_query->execute();
|
|
||||||
$positions = mpull($positions, null, 'getBoardPHID');
|
|
||||||
|
|
||||||
foreach ($project_phids as $project_phid) {
|
foreach ($project_phids as $project_phid) {
|
||||||
$handle = $handles[$project_phid];
|
$handle = $handles[$project_phid];
|
||||||
|
|
||||||
$position = idx($positions, $project_phid);
|
$columns = $engine->getObjectColumns(
|
||||||
if ($position) {
|
$project_phid,
|
||||||
$column = $position->getColumn();
|
$object->getPHID());
|
||||||
|
|
||||||
|
$annotation = array();
|
||||||
|
foreach ($columns as $column) {
|
||||||
$column_name = pht('(%s)', $column->getDisplayName());
|
$column_name = pht('(%s)', $column->getDisplayName());
|
||||||
$column_link = phutil_tag(
|
$column_link = phutil_tag(
|
||||||
'a',
|
'a',
|
||||||
|
@ -89,9 +76,13 @@ final class PhabricatorProjectUIEventListener
|
||||||
),
|
),
|
||||||
$column_name);
|
$column_name);
|
||||||
|
|
||||||
|
$annotation[] = $column_link;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($annotation) {
|
||||||
$annotations[$project_phid] = array(
|
$annotations[$project_phid] = array(
|
||||||
' ',
|
' ',
|
||||||
$column_link,
|
phutil_implode_html(', ', $annotation),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -41,6 +41,10 @@ final class PhabricatorProjectColumnPosition extends PhabricatorProjectDAO
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getOrderingKey() {
|
public function getOrderingKey() {
|
||||||
|
if (!$this->getID()) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
// Low sequence numbers go above high sequence numbers.
|
// Low sequence numbers go above high sequence numbers.
|
||||||
// High position IDs go above low position IDs.
|
// High position IDs go above low position IDs.
|
||||||
// Broadly, this makes newly added stuff float to the top.
|
// Broadly, this makes newly added stuff float to the top.
|
||||||
|
|
Loading…
Reference in a new issue