mirror of
https://we.phorge.it/source/phorge.git
synced 2025-04-06 09:28:25 +02:00
Use a function typeahead for "projects" in Maniphest
Summary: Ref T4100. Collapse the five inputs into one. Test Plan: - Searched for a bunch of stuff. - Used "Group By: Project", which is a bit of a special case and possibly tricky. - Created an old query with all the fields, then updated; verified it was preserved/transformed correctly. {F379971} Reviewers: btrahan Reviewed By: btrahan Subscribers: epriestley Maniphest Tasks: T4100 Differential Revision: https://secure.phabricator.com/D12525
This commit is contained in:
parent
3e9ebb8e20
commit
7ab025eef5
4 changed files with 68 additions and 110 deletions
|
@ -403,23 +403,13 @@ final class ManiphestTaskQuery extends PhabricatorCursorPagedPolicyAwareQuery {
|
||||||
$this->subpriorityMax);
|
$this->subpriorityMax);
|
||||||
}
|
}
|
||||||
|
|
||||||
$where[] = $this->buildPagingClause($conn);
|
$where[] = $this->buildWhereClauseParts($conn);
|
||||||
|
|
||||||
$where = $this->formatWhereClause($where);
|
$where = $this->formatWhereClause($where);
|
||||||
|
|
||||||
$having = '';
|
|
||||||
$count = '';
|
$count = '';
|
||||||
|
|
||||||
if (count($this->projectPHIDs) > 1) {
|
if (count($this->projectPHIDs) > 1) {
|
||||||
// We want to treat the query as an intersection query, not a union
|
|
||||||
// query. We sum the project count and require it be the same as the
|
|
||||||
// number of projects we're searching for.
|
|
||||||
|
|
||||||
$count = ', COUNT(project.dst) projectCount';
|
$count = ', COUNT(project.dst) projectCount';
|
||||||
$having = qsprintf(
|
|
||||||
$conn,
|
|
||||||
'HAVING projectCount = %d',
|
|
||||||
count($this->projectPHIDs));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$group_column = '';
|
$group_column = '';
|
||||||
|
@ -433,14 +423,15 @@ final class ManiphestTaskQuery extends PhabricatorCursorPagedPolicyAwareQuery {
|
||||||
|
|
||||||
$rows = queryfx_all(
|
$rows = queryfx_all(
|
||||||
$conn,
|
$conn,
|
||||||
'SELECT task.* %Q %Q FROM %T task %Q %Q %Q %Q %Q %Q',
|
'%Q %Q %Q FROM %T task %Q %Q %Q %Q %Q %Q',
|
||||||
|
$this->buildSelectClause($conn),
|
||||||
$count,
|
$count,
|
||||||
$group_column,
|
$group_column,
|
||||||
$task_dao->getTableName(),
|
$task_dao->getTableName(),
|
||||||
$this->buildJoinsClause($conn),
|
$this->buildJoinClause($conn),
|
||||||
$where,
|
$where,
|
||||||
$this->buildGroupClause($conn),
|
$this->buildGroupClause($conn),
|
||||||
$having,
|
$this->buildHavingClause($conn),
|
||||||
$this->buildOrderClause($conn),
|
$this->buildOrderClause($conn),
|
||||||
$this->buildLimitClause($conn));
|
$this->buildLimitClause($conn));
|
||||||
|
|
||||||
|
@ -761,7 +752,7 @@ final class ManiphestTaskQuery extends PhabricatorCursorPagedPolicyAwareQuery {
|
||||||
'xproject.dst IS NULL');
|
'xproject.dst IS NULL');
|
||||||
}
|
}
|
||||||
|
|
||||||
private function buildJoinsClause(AphrontDatabaseConnection $conn_r) {
|
protected function buildJoinClauseParts(AphrontDatabaseConnection $conn_r) {
|
||||||
$edge_table = PhabricatorEdgeConfig::TABLE_NAME_EDGE;
|
$edge_table = PhabricatorEdgeConfig::TABLE_NAME_EDGE;
|
||||||
|
|
||||||
$joins = array();
|
$joins = array();
|
||||||
|
@ -856,9 +847,9 @@ final class ManiphestTaskQuery extends PhabricatorCursorPagedPolicyAwareQuery {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
$joins[] = $this->buildApplicationSearchJoinClause($conn_r);
|
$joins[] = parent::buildJoinClauseParts($conn_r);
|
||||||
|
|
||||||
return implode(' ', $joins);
|
return $joins;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function buildGroupClause(AphrontDatabaseConnection $conn_r) {
|
protected function buildGroupClause(AphrontDatabaseConnection $conn_r) {
|
||||||
|
@ -866,7 +857,7 @@ final class ManiphestTaskQuery extends PhabricatorCursorPagedPolicyAwareQuery {
|
||||||
(count($this->anyProjectPHIDs) > 1) ||
|
(count($this->anyProjectPHIDs) > 1) ||
|
||||||
$this->shouldJoinBlockingTasks() ||
|
$this->shouldJoinBlockingTasks() ||
|
||||||
$this->shouldJoinBlockedTasks() ||
|
$this->shouldJoinBlockedTasks() ||
|
||||||
($this->getApplicationSearchMayJoinMultipleRows());
|
($this->shouldGroupQueryResultRows());
|
||||||
|
|
||||||
$joined_project_name = ($this->groupBy == self::GROUP_PROJECT);
|
$joined_project_name = ($this->groupBy == self::GROUP_PROJECT);
|
||||||
|
|
||||||
|
@ -920,6 +911,18 @@ final class ManiphestTaskQuery extends PhabricatorCursorPagedPolicyAwareQuery {
|
||||||
return array_mergev($phids);
|
return array_mergev($phids);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Remove this when moving fully to edge logic.
|
||||||
|
protected function buildHavingClauseParts(AphrontDatabaseConnection $conn) {
|
||||||
|
$having = parent::buildHavingClauseParts($conn);
|
||||||
|
if (count($this->projectPHIDs) > 1) {
|
||||||
|
$having[] = qsprintf(
|
||||||
|
$conn,
|
||||||
|
'projectCount = %d',
|
||||||
|
count($this->projectPHIDs));
|
||||||
|
}
|
||||||
|
return $having;
|
||||||
|
}
|
||||||
|
|
||||||
protected function getResultCursor($result) {
|
protected function getResultCursor($result) {
|
||||||
$id = $result->getID();
|
$id = $result->getID();
|
||||||
|
|
||||||
|
|
|
@ -89,24 +89,8 @@ final class ManiphestTaskSearchEngine
|
||||||
$saved->setParameter('fulltext', $request->getStr('fulltext'));
|
$saved->setParameter('fulltext', $request->getStr('fulltext'));
|
||||||
|
|
||||||
$saved->setParameter(
|
$saved->setParameter(
|
||||||
'allProjectPHIDs',
|
'projects',
|
||||||
$this->readPHIDsFromRequest($request, 'allProjects'));
|
$this->readProjectsFromRequest($request, 'projects'));
|
||||||
|
|
||||||
$saved->setParameter(
|
|
||||||
'withNoProject',
|
|
||||||
$request->getBool('withNoProject'));
|
|
||||||
|
|
||||||
$saved->setParameter(
|
|
||||||
'anyProjectPHIDs',
|
|
||||||
$this->readPHIDsFromRequest($request, 'anyProjects'));
|
|
||||||
|
|
||||||
$saved->setParameter(
|
|
||||||
'excludeProjectPHIDs',
|
|
||||||
$this->readPHIDsFromRequest($request, 'excludeProjects'));
|
|
||||||
|
|
||||||
$saved->setParameter(
|
|
||||||
'userProjectPHIDs',
|
|
||||||
$this->readUsersFromRequest($request, 'userProjects'));
|
|
||||||
|
|
||||||
$saved->setParameter('createdStart', $request->getStr('createdStart'));
|
$saved->setParameter('createdStart', $request->getStr('createdStart'));
|
||||||
$saved->setParameter('createdEnd', $request->getStr('createdEnd'));
|
$saved->setParameter('createdEnd', $request->getStr('createdEnd'));
|
||||||
|
@ -183,30 +167,9 @@ final class ManiphestTaskSearchEngine
|
||||||
$query->withFullTextSearch($fulltext);
|
$query->withFullTextSearch($fulltext);
|
||||||
}
|
}
|
||||||
|
|
||||||
$with_no_project = $saved->getParameter('withNoProject');
|
$projects = $this->readProjectTokens($saved);
|
||||||
if ($with_no_project) {
|
$adjusted = id(clone $saved)->setParameter('projects', $projects);
|
||||||
$query->withAllProjects(array(ManiphestTaskOwner::PROJECT_NO_PROJECT));
|
$this->setQueryProjects($query, $adjusted);
|
||||||
} else {
|
|
||||||
$project_phids = $saved->getParameter('allProjectPHIDs');
|
|
||||||
if ($project_phids) {
|
|
||||||
$query->withAllProjects($project_phids);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$any_project_phids = $saved->getParameter('anyProjectPHIDs');
|
|
||||||
if ($any_project_phids) {
|
|
||||||
$query->withAnyProjects($any_project_phids);
|
|
||||||
}
|
|
||||||
|
|
||||||
$exclude_project_phids = $saved->getParameter('excludeProjectPHIDs');
|
|
||||||
if ($exclude_project_phids) {
|
|
||||||
$query->withoutProjects($exclude_project_phids);
|
|
||||||
}
|
|
||||||
|
|
||||||
$user_project_phids = $saved->getParameter('userProjectPHIDs');
|
|
||||||
if ($user_project_phids) {
|
|
||||||
$query->withAnyUserProjects($user_project_phids);
|
|
||||||
}
|
|
||||||
|
|
||||||
$start = $this->parseDateTime($saved->getParameter('createdStart'));
|
$start = $this->parseDateTime($saved->getParameter('createdStart'));
|
||||||
$end = $this->parseDateTime($saved->getParameter('createdEnd'));
|
$end = $this->parseDateTime($saved->getParameter('createdEnd'));
|
||||||
|
@ -242,21 +205,9 @@ final class ManiphestTaskSearchEngine
|
||||||
$assigned_phids = $this->readAssignedPHIDs($saved);
|
$assigned_phids = $this->readAssignedPHIDs($saved);
|
||||||
|
|
||||||
$author_phids = $saved->getParameter('authorPHIDs', array());
|
$author_phids = $saved->getParameter('authorPHIDs', array());
|
||||||
$all_project_phids = $saved->getParameter(
|
$projects = $this->readProjectTokens($saved);
|
||||||
'allProjectPHIDs',
|
|
||||||
array());
|
|
||||||
$any_project_phids = $saved->getParameter(
|
|
||||||
'anyProjectPHIDs',
|
|
||||||
array());
|
|
||||||
$exclude_project_phids = $saved->getParameter(
|
|
||||||
'excludeProjectPHIDs',
|
|
||||||
array());
|
|
||||||
$user_project_phids = $saved->getParameter(
|
|
||||||
'userProjectPHIDs',
|
|
||||||
array());
|
|
||||||
$subscriber_phids = $saved->getParameter('subscriberPHIDs', array());
|
|
||||||
|
|
||||||
$with_no_projects = $saved->getParameter('withNoProject');
|
$subscriber_phids = $saved->getParameter('subscriberPHIDs', array());
|
||||||
|
|
||||||
$statuses = $saved->getParameter('statuses', array());
|
$statuses = $saved->getParameter('statuses', array());
|
||||||
$statuses = array_fuse($statuses);
|
$statuses = array_fuse($statuses);
|
||||||
|
@ -317,41 +268,10 @@ final class ManiphestTaskSearchEngine
|
||||||
->setValue($assigned_phids))
|
->setValue($assigned_phids))
|
||||||
->appendControl(
|
->appendControl(
|
||||||
id(new AphrontFormTokenizerControl())
|
id(new AphrontFormTokenizerControl())
|
||||||
->setDatasource(new PhabricatorProjectDatasource())
|
->setDatasource(new PhabricatorProjectLogicalDatasource())
|
||||||
->setName('allProjects')
|
->setName('projects')
|
||||||
->setLabel(pht('In All Projects'))
|
->setLabel(pht('Projects'))
|
||||||
->setValue($all_project_phids));
|
->setValue($projects))
|
||||||
|
|
||||||
if (!$this->getIsBoardView()) {
|
|
||||||
$form
|
|
||||||
->appendChild(
|
|
||||||
id(new AphrontFormCheckboxControl())
|
|
||||||
->addCheckbox(
|
|
||||||
'withNoProject',
|
|
||||||
1,
|
|
||||||
pht('Show only tasks with no projects.'),
|
|
||||||
$with_no_projects));
|
|
||||||
}
|
|
||||||
|
|
||||||
$form
|
|
||||||
->appendControl(
|
|
||||||
id(new AphrontFormTokenizerControl())
|
|
||||||
->setDatasource(new PhabricatorProjectDatasource())
|
|
||||||
->setName('anyProjects')
|
|
||||||
->setLabel(pht('In Any Project'))
|
|
||||||
->setValue($any_project_phids))
|
|
||||||
->appendControl(
|
|
||||||
id(new AphrontFormTokenizerControl())
|
|
||||||
->setDatasource(new PhabricatorProjectDatasource())
|
|
||||||
->setName('excludeProjects')
|
|
||||||
->setLabel(pht('Not In Projects'))
|
|
||||||
->setValue($exclude_project_phids))
|
|
||||||
->appendControl(
|
|
||||||
id(new AphrontFormTokenizerControl())
|
|
||||||
->setDatasource(new PhabricatorPeopleDatasource())
|
|
||||||
->setName('userProjects')
|
|
||||||
->setLabel(pht('In Users\' Projects'))
|
|
||||||
->setValue($user_project_phids))
|
|
||||||
->appendControl(
|
->appendControl(
|
||||||
id(new AphrontFormTokenizerControl())
|
id(new AphrontFormTokenizerControl())
|
||||||
->setDatasource(new PhabricatorPeopleDatasource())
|
->setDatasource(new PhabricatorPeopleDatasource())
|
||||||
|
@ -566,4 +486,35 @@ final class ManiphestTaskSearchEngine
|
||||||
return $assigned_phids;
|
return $assigned_phids;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function readProjectTokens(PhabricatorSavedQuery $saved) {
|
||||||
|
$projects = $saved->getParameter('projects', array());
|
||||||
|
|
||||||
|
$all = $saved->getParameter('allProjectPHIDs', array());
|
||||||
|
foreach ($all as $phid) {
|
||||||
|
$projects[] = $phid;
|
||||||
|
}
|
||||||
|
|
||||||
|
$any = $saved->getParameter('anyProjectPHIDs', array());
|
||||||
|
foreach ($any as $phid) {
|
||||||
|
$projects[] = 'any('.$phid.')';
|
||||||
|
}
|
||||||
|
|
||||||
|
$not = $saved->getParameter('excludeProjectPHIDs', array());
|
||||||
|
foreach ($not as $phid) {
|
||||||
|
$projects[] = 'not('.$phid.')';
|
||||||
|
}
|
||||||
|
|
||||||
|
$users = $saved->getParameter('userProjectPHIDs', array());
|
||||||
|
foreach ($users as $phid) {
|
||||||
|
$projects[] = 'projects('.$phid.')';
|
||||||
|
}
|
||||||
|
|
||||||
|
$no = $saved->getParameter('withNoProject');
|
||||||
|
if ($no) {
|
||||||
|
$projects[] = 'null()';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $projects;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -85,7 +85,8 @@ final class PhabricatorProjectLogicalOrNotDatasource
|
||||||
foreach ($results as $result) {
|
foreach ($results as $result) {
|
||||||
$result
|
$result
|
||||||
->setTokenType(PhabricatorTypeaheadTokenView::TYPE_FUNCTION)
|
->setTokenType(PhabricatorTypeaheadTokenView::TYPE_FUNCTION)
|
||||||
->setIcon('fa-asterisk');
|
->setIcon('fa-asterisk')
|
||||||
|
->setColor(null);
|
||||||
|
|
||||||
if ($return_any) {
|
if ($return_any) {
|
||||||
$return[] = id(clone $result)
|
$return[] = id(clone $result)
|
||||||
|
@ -134,6 +135,7 @@ final class PhabricatorProjectLogicalOrNotDatasource
|
||||||
|
|
||||||
$tokens = $this->renderTokens($phids);
|
$tokens = $this->renderTokens($phids);
|
||||||
foreach ($tokens as $token) {
|
foreach ($tokens as $token) {
|
||||||
|
$token->setColor(null);
|
||||||
if ($token->isInvalid()) {
|
if ($token->isInvalid()) {
|
||||||
if ($function == 'any') {
|
if ($function == 'any') {
|
||||||
$token->setValue(pht('In Any: Invalid Project'));
|
$token->setValue(pht('In Any: Invalid Project'));
|
||||||
|
|
|
@ -42,6 +42,7 @@ final class PhabricatorProjectLogicalUserDatasource
|
||||||
protected function didLoadResults(array $results) {
|
protected function didLoadResults(array $results) {
|
||||||
foreach ($results as $result) {
|
foreach ($results as $result) {
|
||||||
$result
|
$result
|
||||||
|
->setColor(null)
|
||||||
->setTokenType(PhabricatorTypeaheadTokenView::TYPE_FUNCTION)
|
->setTokenType(PhabricatorTypeaheadTokenView::TYPE_FUNCTION)
|
||||||
->setIcon('fa-asterisk')
|
->setIcon('fa-asterisk')
|
||||||
->setPHID('projects('.$result->getPHID().')')
|
->setPHID('projects('.$result->getPHID().')')
|
||||||
|
@ -85,6 +86,7 @@ final class PhabricatorProjectLogicalUserDatasource
|
||||||
|
|
||||||
$tokens = $this->renderTokens($phids);
|
$tokens = $this->renderTokens($phids);
|
||||||
foreach ($tokens as $token) {
|
foreach ($tokens as $token) {
|
||||||
|
$token->setColor(null);
|
||||||
if ($token->isInvalid()) {
|
if ($token->isInvalid()) {
|
||||||
$token
|
$token
|
||||||
->setValue(pht("User's Projects: Invalid User"));
|
->setValue(pht("User's Projects: Invalid User"));
|
||||||
|
|
Loading…
Add table
Reference in a new issue