mirror of
https://we.phorge.it/source/phorge.git
synced 2025-04-05 00:48:22 +02:00
Summary: Depends on D20635. Ref T4900. Fixes T13316. Currently, "Move Tasks to Column..." first prompts you to select a project, then prompts you for a column. The first step is prefilled with the current project, so the common case (moving to another column on the same board) requires you to confirm that you aren't doing an off-project move by clicking "Continue", then you can select a column. This isn't a huge inconvenience and the workflow isn't terribly common, but it's surprising enough that it has come up a few times as a stumbling block. Particularly, we're suggesting to users that they're about to pick a column, then we're asking them to pick a project. The prompt also says "Project: XYZ", not "Project: Keep in current project" or something like that. Smooth this out by splitting the action into two better-cued flows: - "Move Tasks to Project..." is the current flow: pick a project, then pick a column. - The project selection no longer defaults to the current project, since we now expect you to usually use this flow to move tasks to a different project. - "Move Tasks to Column..." prompts you to select a column on the same board. - This just skips step 1 of the workflow. - This now defaults to the current column, which isn't a useful selection, but is more clear. In both cases, the action cue ("Move tasks to X...") now matches what the dialog actually asks you for ("Pick an X"). Test Plan: - Moved tasks across projects and columns within the same project. - Hit all (I think?) the error cases and got sensible error and recovery behavior. Reviewers: amckinley Reviewed By: amckinley Maniphest Tasks: T13316, T4900 Differential Revision: https://secure.phabricator.com/D20636
264 lines
8.2 KiB
PHP
264 lines
8.2 KiB
PHP
<?php
|
|
|
|
final class PhabricatorProjectColumnBulkMoveController
|
|
extends PhabricatorProjectBoardController {
|
|
|
|
public function handleRequest(AphrontRequest $request) {
|
|
$viewer = $request->getViewer();
|
|
|
|
$response = $this->loadProject();
|
|
if ($response) {
|
|
return $response;
|
|
}
|
|
|
|
// See T13316. If we're operating in "column" mode, we're going to skip
|
|
// the prompt for a project and just have the user select a target column.
|
|
// In "project" mode, we prompt them for a project first.
|
|
$is_column_mode = ($request->getURIData('mode') === 'column');
|
|
|
|
$src_project = $this->getProject();
|
|
$state = $this->getViewState();
|
|
$board_uri = $state->newWorkboardURI();
|
|
|
|
$layout_engine = $state->getLayoutEngine();
|
|
|
|
$board_phid = $src_project->getPHID();
|
|
$columns = $layout_engine->getColumns($board_phid);
|
|
$columns = mpull($columns, null, 'getID');
|
|
|
|
$column_id = $request->getURIData('columnID');
|
|
$src_column = idx($columns, $column_id);
|
|
if (!$src_column) {
|
|
return new Aphront404Response();
|
|
}
|
|
|
|
$move_task_phids = $layout_engine->getColumnObjectPHIDs(
|
|
$board_phid,
|
|
$src_column->getPHID());
|
|
|
|
$tasks = $state->getObjects();
|
|
|
|
$move_tasks = array_select_keys($tasks, $move_task_phids);
|
|
|
|
$move_tasks = id(new PhabricatorPolicyFilter())
|
|
->setViewer($viewer)
|
|
->requireCapabilities(array(PhabricatorPolicyCapability::CAN_EDIT))
|
|
->apply($move_tasks);
|
|
|
|
if (!$move_tasks) {
|
|
return $this->newDialog()
|
|
->setTitle(pht('No Movable Tasks'))
|
|
->appendParagraph(
|
|
pht(
|
|
'The selected column contains no visible tasks which you '.
|
|
'have permission to move.'))
|
|
->addCancelButton($board_uri);
|
|
}
|
|
|
|
$dst_project_phid = null;
|
|
$dst_project = null;
|
|
$has_project = false;
|
|
if ($is_column_mode) {
|
|
$has_project = true;
|
|
$dst_project_phid = $src_project->getPHID();
|
|
} else {
|
|
if ($request->isFormOrHiSecPost()) {
|
|
$has_project = $request->getStr('hasProject');
|
|
if ($has_project) {
|
|
// We may read this from a tokenizer input as an array, or from a
|
|
// hidden input as a string.
|
|
$dst_project_phid = head($request->getArr('dstProjectPHID'));
|
|
if (!$dst_project_phid) {
|
|
$dst_project_phid = $request->getStr('dstProjectPHID');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$errors = array();
|
|
$hidden = array();
|
|
|
|
if ($has_project) {
|
|
if (!$dst_project_phid) {
|
|
$errors[] = pht('Choose a project to move tasks to.');
|
|
} else {
|
|
$dst_project = id(new PhabricatorProjectQuery())
|
|
->setViewer($viewer)
|
|
->withPHIDs(array($dst_project_phid))
|
|
->executeOne();
|
|
if (!$dst_project) {
|
|
$errors[] = pht('Choose a valid project to move tasks to.');
|
|
}
|
|
|
|
if (!$dst_project->getHasWorkboard()) {
|
|
$errors[] = pht('You must choose a project with a workboard.');
|
|
$dst_project = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($dst_project) {
|
|
$same_project = ($src_project->getID() === $dst_project->getID());
|
|
|
|
$layout_engine = id(new PhabricatorBoardLayoutEngine())
|
|
->setViewer($viewer)
|
|
->setBoardPHIDs(array($dst_project->getPHID()))
|
|
->setFetchAllBoards(true)
|
|
->executeLayout();
|
|
|
|
$dst_columns = $layout_engine->getColumns($dst_project->getPHID());
|
|
$dst_columns = mpull($columns, null, 'getPHID');
|
|
|
|
$has_column = false;
|
|
$dst_column = null;
|
|
|
|
// If we're performing a move on the same board, default the
|
|
// control value to the current column.
|
|
if ($same_project) {
|
|
$dst_column_phid = $src_column->getPHID();
|
|
} else {
|
|
$dst_column_phid = null;
|
|
}
|
|
|
|
if ($request->isFormOrHiSecPost()) {
|
|
$has_column = $request->getStr('hasColumn');
|
|
if ($has_column) {
|
|
$dst_column_phid = $request->getStr('dstColumnPHID');
|
|
}
|
|
}
|
|
|
|
if ($has_column) {
|
|
$dst_column = idx($dst_columns, $dst_column_phid);
|
|
if (!$dst_column) {
|
|
$errors[] = pht('Choose a column to move tasks to.');
|
|
} else {
|
|
if ($dst_column->isHidden()) {
|
|
$errors[] = pht('You can not move tasks to a hidden column.');
|
|
$dst_column = null;
|
|
} else if ($dst_column->getPHID() === $src_column->getPHID()) {
|
|
$errors[] = pht('You can not move tasks from a column to itself.');
|
|
$dst_column = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($dst_column) {
|
|
foreach ($move_tasks as $move_task) {
|
|
$xactions = array();
|
|
|
|
// If we're switching projects, get out of the old project first
|
|
// and move to the new project.
|
|
if (!$same_project) {
|
|
$xactions[] = id(new ManiphestTransaction())
|
|
->setTransactionType(PhabricatorTransactions::TYPE_EDGE)
|
|
->setMetadataValue(
|
|
'edge:type',
|
|
PhabricatorProjectObjectHasProjectEdgeType::EDGECONST)
|
|
->setNewValue(
|
|
array(
|
|
'-' => array(
|
|
$src_project->getPHID() => $src_project->getPHID(),
|
|
),
|
|
'+' => array(
|
|
$dst_project->getPHID() => $dst_project->getPHID(),
|
|
),
|
|
));
|
|
}
|
|
|
|
$xactions[] = id(new ManiphestTransaction())
|
|
->setTransactionType(PhabricatorTransactions::TYPE_COLUMNS)
|
|
->setNewValue(
|
|
array(
|
|
array(
|
|
'columnPHID' => $dst_column->getPHID(),
|
|
),
|
|
));
|
|
|
|
$editor = id(new ManiphestTransactionEditor())
|
|
->setActor($viewer)
|
|
->setContinueOnMissingFields(true)
|
|
->setContinueOnNoEffect(true)
|
|
->setContentSourceFromRequest($request)
|
|
->setCancelURI($board_uri);
|
|
|
|
$editor->applyTransactions($move_task, $xactions);
|
|
}
|
|
|
|
// If we did a move on the same workboard, redirect and preserve the
|
|
// state parameters. If we moved to a different workboard, go there
|
|
// with clean default state.
|
|
if ($same_project) {
|
|
$done_uri = $board_uri;
|
|
} else {
|
|
$done_uri = $dst_project->getWorkboardURI();
|
|
}
|
|
|
|
return id(new AphrontRedirectResponse())->setURI($done_uri);
|
|
}
|
|
|
|
$title = pht('Move Tasks to Column');
|
|
|
|
$form = id(new AphrontFormView())
|
|
->setViewer($viewer);
|
|
|
|
// If we're moving between projects, add a reminder about which project
|
|
// you selected in the previous step.
|
|
if (!$is_column_mode) {
|
|
$form->appendControl(
|
|
id(new AphrontFormStaticControl())
|
|
->setLabel(pht('Project'))
|
|
->setValue($dst_project->getDisplayName()));
|
|
}
|
|
|
|
$form->appendControl(
|
|
id(new AphrontFormSelectControl())
|
|
->setName('dstColumnPHID')
|
|
->setLabel(pht('Move to Column'))
|
|
->setValue($dst_column_phid)
|
|
->setOptions(mpull($dst_columns, 'getDisplayName', 'getPHID')));
|
|
|
|
$submit = pht('Move Tasks');
|
|
|
|
$hidden['dstProjectPHID'] = $dst_project->getPHID();
|
|
$hidden['hasColumn'] = true;
|
|
$hidden['hasProject'] = true;
|
|
} else {
|
|
$title = pht('Move Tasks to Project');
|
|
|
|
if ($dst_project_phid) {
|
|
$dst_project_phid_value = array($dst_project_phid);
|
|
} else {
|
|
$dst_project_phid_value = array();
|
|
}
|
|
|
|
$form = id(new AphrontFormView())
|
|
->setViewer($viewer)
|
|
->appendControl(
|
|
id(new AphrontFormTokenizerControl())
|
|
->setName('dstProjectPHID')
|
|
->setLimit(1)
|
|
->setLabel(pht('Move to Project'))
|
|
->setValue($dst_project_phid_value)
|
|
->setDatasource(new PhabricatorProjectDatasource()));
|
|
|
|
$submit = pht('Continue');
|
|
|
|
$hidden['hasProject'] = true;
|
|
}
|
|
|
|
$dialog = $this->newWorkboardDialog()
|
|
->setWidth(AphrontDialogView::WIDTH_FORM)
|
|
->setTitle($title)
|
|
->setErrors($errors)
|
|
->appendForm($form)
|
|
->addSubmitButton($submit)
|
|
->addCancelButton($board_uri);
|
|
|
|
foreach ($hidden as $key => $value) {
|
|
$dialog->addHiddenInput($key, $value);
|
|
}
|
|
|
|
return $dialog;
|
|
}
|
|
|
|
}
|