1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-11 23:31:03 +01:00

Allow the "Create Subtask" workflow to prompt for a subtype selection, and prepare for customizable options

Summary:
Ref T13222. Ref T12588. See PHI683. Currently, "Create Subtask" always uses the first edit form that the user has access to for the same task subtype. (For example, if you "Create Subtask" from a "Bug", you get the first edit form for "Bugs".)

I didn't want to go too crazy with the initial subtype implementation, but it seems like we're generally on firm ground and it's working fairly well: user requests are for more flexibility in using the system as implemented, not changes to the system or confusion/difficulty with any of the tradeoffs. Thus, I'm generally comfortable continuing to build it out in the same direction. To improve flexibility, I want to make the options from "Create Subtask" more flexible/configurable.

I plan to let you specify that a given subtype (say, "Quest") prompts you with creation options for a set of other subtypes (say, "Objective"), or prompts you with a particular set of forms.

If we end up with a single option, we just go into the current flow (directly to the edit form). If we end up with more than one option, we prompt the user to choose between them.

This change is a first step toward this:

  - When building "Create Subtask", query for multiple forms.
  - The default behavior is now "prompt user to choose among create forms of the same subtype". Previously, it was "use the first edit form of the same subtype". This is a behavioral change.
  - The next change will make the selected forms configurable.
  - (I also plan to make the dialog itself less rough.)

Test Plan: {F6051067}

Reviewers: amckinley

Reviewed By: amckinley

Maniphest Tasks: T13222, T12588

Differential Revision: https://secure.phabricator.com/D19853
This commit is contained in:
epriestley 2018-12-06 08:25:25 -08:00
parent da4341cf8b
commit d1bcdaeda4
6 changed files with 147 additions and 15 deletions

View file

@ -1760,6 +1760,7 @@ phutil_register_library_map(array(
'ManiphestTaskStatusTestCase' => 'applications/maniphest/constants/__tests__/ManiphestTaskStatusTestCase.php', 'ManiphestTaskStatusTestCase' => 'applications/maniphest/constants/__tests__/ManiphestTaskStatusTestCase.php',
'ManiphestTaskStatusTransaction' => 'applications/maniphest/xaction/ManiphestTaskStatusTransaction.php', 'ManiphestTaskStatusTransaction' => 'applications/maniphest/xaction/ManiphestTaskStatusTransaction.php',
'ManiphestTaskSubpriorityTransaction' => 'applications/maniphest/xaction/ManiphestTaskSubpriorityTransaction.php', 'ManiphestTaskSubpriorityTransaction' => 'applications/maniphest/xaction/ManiphestTaskSubpriorityTransaction.php',
'ManiphestTaskSubtaskController' => 'applications/maniphest/controller/ManiphestTaskSubtaskController.php',
'ManiphestTaskSubtypeDatasource' => 'applications/maniphest/typeahead/ManiphestTaskSubtypeDatasource.php', 'ManiphestTaskSubtypeDatasource' => 'applications/maniphest/typeahead/ManiphestTaskSubtypeDatasource.php',
'ManiphestTaskTestCase' => 'applications/maniphest/__tests__/ManiphestTaskTestCase.php', 'ManiphestTaskTestCase' => 'applications/maniphest/__tests__/ManiphestTaskTestCase.php',
'ManiphestTaskTitleHeraldField' => 'applications/maniphest/herald/ManiphestTaskTitleHeraldField.php', 'ManiphestTaskTitleHeraldField' => 'applications/maniphest/herald/ManiphestTaskTitleHeraldField.php',
@ -7347,6 +7348,7 @@ phutil_register_library_map(array(
'ManiphestTaskStatusTestCase' => 'PhabricatorTestCase', 'ManiphestTaskStatusTestCase' => 'PhabricatorTestCase',
'ManiphestTaskStatusTransaction' => 'ManiphestTaskTransactionType', 'ManiphestTaskStatusTransaction' => 'ManiphestTaskTransactionType',
'ManiphestTaskSubpriorityTransaction' => 'ManiphestTaskTransactionType', 'ManiphestTaskSubpriorityTransaction' => 'ManiphestTaskTransactionType',
'ManiphestTaskSubtaskController' => 'ManiphestController',
'ManiphestTaskSubtypeDatasource' => 'PhabricatorTypeaheadDatasource', 'ManiphestTaskSubtypeDatasource' => 'PhabricatorTypeaheadDatasource',
'ManiphestTaskTestCase' => 'PhabricatorTestCase', 'ManiphestTaskTestCase' => 'PhabricatorTestCase',
'ManiphestTaskTitleHeraldField' => 'ManiphestTaskHeraldField', 'ManiphestTaskTitleHeraldField' => 'ManiphestTaskHeraldField',

View file

@ -52,6 +52,7 @@ final class PhabricatorManiphestApplication extends PhabricatorApplication {
'task/' => array( 'task/' => array(
$this->getEditRoutePattern('edit/') $this->getEditRoutePattern('edit/')
=> 'ManiphestTaskEditController', => 'ManiphestTaskEditController',
'subtask/(?P<id>[1-9]\d*)/' => 'ManiphestTaskSubtaskController',
), ),
'subpriority/' => 'ManiphestSubpriorityController', 'subpriority/' => 'ManiphestSubpriorityController',
), ),

View file

@ -281,29 +281,39 @@ final class ManiphestTaskDetailController extends ManiphestController {
->setDisabled(!$can_edit) ->setDisabled(!$can_edit)
->setWorkflow($workflow_edit)); ->setWorkflow($workflow_edit));
$edit_config = $edit_engine->loadDefaultEditConfiguration($task); $subtype_map = $task->newEditEngineSubtypeMap();
$can_create = (bool)$edit_config; $subtask_options = $subtype_map->getCreateFormsForSubtype(
$edit_engine,
$task);
if ($can_create) { // If no forms are available, we want to show the user an error.
$form_key = $edit_config->getIdentifier(); // If one form is available, we take them user directly to the form.
$edit_uri = id(new PhutilURI("/task/edit/form/{$form_key}/")) // If two or more forms are available, we give the user a choice.
// The "subtask" controller handles the first case (no forms) and the
// third case (more than one form). In the case of one form, we link
// directly to the form.
$subtask_uri = "/task/subtask/{$id}/";
$subtask_workflow = true;
if (count($subtask_options) == 1) {
$subtask_form = head($subtask_options);
$form_key = $subtask_form->getIdentifier();
$subtask_uri = id(new PhutilURI("/task/edit/form/{$form_key}/"))
->setQueryParam('parent', $id) ->setQueryParam('parent', $id)
->setQueryParam('template', $id) ->setQueryParam('template', $id)
->setQueryParam('status', ManiphestTaskStatus::getDefaultStatus()); ->setQueryParam('status', ManiphestTaskStatus::getDefaultStatus());
$edit_uri = $this->getApplicationURI($edit_uri); $subtask_workflow = false;
} else {
// TODO: This will usually give us a somewhat-reasonable error page, but
// could be a bit cleaner.
$edit_uri = "/task/edit/{$id}/";
$edit_uri = $this->getApplicationURI($edit_uri);
} }
$subtask_uri = $this->getApplicationURI($subtask_uri);
$subtask_item = id(new PhabricatorActionView()) $subtask_item = id(new PhabricatorActionView())
->setName(pht('Create Subtask')) ->setName(pht('Create Subtask'))
->setHref($edit_uri) ->setHref($subtask_uri)
->setIcon('fa-level-down') ->setIcon('fa-level-down')
->setDisabled(!$can_create) ->setDisabled(!$subtask_options)
->setWorkflow(!$can_create); ->setWorkflow($subtask_workflow);
$relationship_list = PhabricatorObjectRelationshipList::newForObject( $relationship_list = PhabricatorObjectRelationshipList::newForObject(
$viewer, $viewer,

View file

@ -0,0 +1,76 @@
<?php
final class ManiphestTaskSubtaskController
extends ManiphestController {
public function handleRequest(AphrontRequest $request) {
$viewer = $this->getViewer();
$id = $request->getURIData('id');
$task = id(new ManiphestTaskQuery())
->setViewer($viewer)
->withIDs(array($id))
->executeOne();
if (!$task) {
return new Aphront404Response();
}
$cancel_uri = $task->getURI();
$edit_engine = id(new ManiphestEditEngine())
->setViewer($viewer)
->setTargetObject($task);
$subtype_map = $task->newEditEngineSubtypeMap();
$subtype_options = $subtype_map->getCreateFormsForSubtype(
$edit_engine,
$task);
if (!$subtype_options) {
return $this->newDialog()
->setTitle(pht('No Forms'))
->appendParagraph(
pht(
'You do not have access to any forms which can be used to '.
'create a subtask.'))
->addCancelButton($cancel_uri, pht('Close'));
}
if ($request->isFormPost()) {
$form_key = $request->getStr('formKey');
if (isset($subtype_options[$form_key])) {
$subtask_uri = id(new PhutilURI("/task/edit/form/{$form_key}/"))
->setQueryParam('parent', $id)
->setQueryParam('template', $id)
->setQueryParam('status', ManiphestTaskStatus::getDefaultStatus());
$subtask_uri = $this->getApplicationURI($subtask_uri);
return id(new AphrontRedirectResponse())
->setURI($subtask_uri);
}
}
$control = id(new AphrontFormRadioButtonControl())
->setName('formKey')
->setLabel(pht('Subtype'));
foreach ($subtype_options as $key => $subtype_form) {
$control->addButton(
$key,
$subtype_form->getDisplayName(),
null);
}
$form = id(new AphrontFormView())
->setViewer($viewer)
->appendControl($control);
return $this->newDialog()
->setTitle(pht('Choose Subtype'))
->appendForm($form)
->addSubmitButton(pht('Continue'))
->addCancelButton($cancel_uri);
}
}

View file

@ -358,7 +358,7 @@ abstract class PhabricatorEditEngine
return $this->editEngineConfiguration; return $this->editEngineConfiguration;
} }
private function newConfigurationQuery() { public function newConfigurationQuery() {
return id(new PhabricatorEditEngineConfigurationQuery()) return id(new PhabricatorEditEngineConfigurationQuery())
->setViewer($this->getViewer()) ->setViewer($this->getViewer())
->withEngineKeys(array($this->getEngineKey())); ->withEngineKeys(array($this->getEngineKey()));

View file

@ -39,4 +39,47 @@ final class PhabricatorEditEngineSubtypeMap
return $this->subtypes[$subtype_key]; return $this->subtypes[$subtype_key];
} }
public function getCreateFormsForSubtype(
PhabricatorEditEngine $edit_engine,
PhabricatorEditEngineSubtypeInterface $object) {
$subtype_key = $object->getEditEngineSubtype();
$subtype = $this->getSubtype($subtype_key);
// TODO: Allow subtype configuration to specify that children should be
// created from particular forms or subtypes.
$select_ids = array();
$select_subtypes = array();
$query = $edit_engine->newConfigurationQuery()
->withIsDisabled(false);
if ($select_ids) {
$query->withIDs($select_ids);
} else {
// If we're selecting by subtype rather than selecting specific forms,
// only select create forms.
$query->withIsDefault(true);
if ($select_subtypes) {
$query->withSubtypes($select_subtypes);
} else {
$query->withSubtypes(array($subtype_key));
}
}
$forms = $query->execute();
$forms = mpull($forms, null, 'getIdentifier');
// If we're selecting by ID, respect the order specified in the
// constraint. Otherwise, use the create form sort order.
if ($select_ids) {
$forms = array_select_keys($forms, $select_ids) + $forms;
} else {
$forms = msort($forms, 'getCreateSortKey');
}
return $forms;
}
} }