Add filter by task link
This commit is contained in:
parent
e582d4047b
commit
f837e70a2d
|
|
@ -39,6 +39,7 @@ class Lexer
|
|||
"/^(swimlane:)/" => 'T_SWIMLANE',
|
||||
"/^(ref:)/" => 'T_REFERENCE',
|
||||
"/^(reference:)/" => 'T_REFERENCE',
|
||||
"/^(link:)/" => 'T_LINK',
|
||||
"/^(\s+)/" => 'T_WHITESPACE',
|
||||
'/^([<=>]{0,2}[0-9]{4}-[0-9]{2}-[0-9]{2})/' => 'T_DATE',
|
||||
'/^(yesterday|tomorrow|today)/' => 'T_DATE',
|
||||
|
|
@ -118,6 +119,7 @@ class Lexer
|
|||
case 'T_COLUMN':
|
||||
case 'T_PROJECT':
|
||||
case 'T_SWIMLANE':
|
||||
case 'T_LINK':
|
||||
$next = next($tokens);
|
||||
|
||||
if ($next !== false && $next['token'] === 'T_STRING') {
|
||||
|
|
|
|||
|
|
@ -30,6 +30,7 @@ class TaskFilter extends Base
|
|||
'T_COLUMN' => 'filterByColumnName',
|
||||
'T_REFERENCE' => 'filterByReference',
|
||||
'T_SWIMLANE' => 'filterBySwimlaneName',
|
||||
'T_LINK' => 'filterByLinkName',
|
||||
);
|
||||
|
||||
/**
|
||||
|
|
@ -107,6 +108,22 @@ class TaskFilter extends Base
|
|||
->neq(Subtask::TABLE.'.status', Subtask::STATUS_DONE);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a new link query
|
||||
*
|
||||
* @access public
|
||||
* @return \PicoDb\Table
|
||||
*/
|
||||
public function createLinkQuery()
|
||||
{
|
||||
return $this->db->table(TaskLink::TABLE)
|
||||
->columns(
|
||||
TaskLink::TABLE.'.task_id',
|
||||
Link::TABLE.'.label'
|
||||
)
|
||||
->join(Link::TABLE, 'id', 'link_id', TaskLink::TABLE);
|
||||
}
|
||||
|
||||
/**
|
||||
* Clone the filter
|
||||
*
|
||||
|
|
@ -506,6 +523,30 @@ class TaskFilter extends Base
|
|||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter by link
|
||||
*
|
||||
* @access public
|
||||
* @param array $values List of links
|
||||
* @return TaskFilter
|
||||
*/
|
||||
public function filterByLinkName(array $values)
|
||||
{
|
||||
$this->query->beginOr();
|
||||
|
||||
$link_query = $this->createLinkQuery()->in(Link::TABLE.'.label', $values);
|
||||
$matching_task_ids = $link_query->findAllByColumn('task_id');
|
||||
if (empty($matching_task_ids)) {
|
||||
$this->query->eq(Task::TABLE.'.id', 0);
|
||||
} else {
|
||||
$this->query->in(Task::TABLE.'.id', $matching_task_ids);
|
||||
}
|
||||
|
||||
$this->query->closeOr();
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter by due date
|
||||
*
|
||||
|
|
|
|||
|
|
@ -136,3 +136,11 @@ Attribute: **swimlane**
|
|||
- Find tasks in the default swimlane: `swimlane:default`
|
||||
- Find tasks into several swimlanes: `swimlane:"Version 1.2" swimlane:"Version 1.3"`
|
||||
|
||||
Search by task link
|
||||
------------------
|
||||
|
||||
Attribute: **link**
|
||||
|
||||
- Find tasks by link name: `link:"is a milestone of"`
|
||||
- Find tasks into several links: `link:"is a milestone of" link:"relates to"`
|
||||
|
||||
|
|
|
|||
|
|
@ -116,6 +116,31 @@ class LexerTest extends Base
|
|||
);
|
||||
}
|
||||
|
||||
public function testLinkQuery()
|
||||
{
|
||||
$lexer = new Lexer;
|
||||
|
||||
$this->assertEquals(
|
||||
array(array('match' => 'link:', 'token' => 'T_LINK'), array('match' => 'is a milestone of', 'token' => 'T_STRING')),
|
||||
$lexer->tokenize('link:"is a milestone of"')
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
array('T_LINK' => array('is a milestone of')),
|
||||
$lexer->map($lexer->tokenize('link:"is a milestone of"'))
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
array('T_LINK' => array('is a milestone of', 'fixes')),
|
||||
$lexer->map($lexer->tokenize('link:"is a milestone of" link:fixes'))
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
array(),
|
||||
$lexer->map($lexer->tokenize('link: '))
|
||||
);
|
||||
}
|
||||
|
||||
public function testColumnQuery()
|
||||
{
|
||||
$lexer = new Lexer;
|
||||
|
|
|
|||
|
|
@ -6,6 +6,7 @@ use Kanboard\Model\Project;
|
|||
use Kanboard\Model\User;
|
||||
use Kanboard\Model\TaskFilter;
|
||||
use Kanboard\Model\TaskCreation;
|
||||
use Kanboard\Model\TaskLink;
|
||||
use Kanboard\Core\DateParser;
|
||||
use Kanboard\Model\Category;
|
||||
use Kanboard\Model\Subtask;
|
||||
|
|
@ -552,6 +553,65 @@ class TaskFilterTest extends Base
|
|||
$this->assertEquals('task3', $tasks[0]['title']);
|
||||
}
|
||||
|
||||
public function testSearchWithLink()
|
||||
{
|
||||
$p = new Project($this->container);
|
||||
$u = new User($this->container);
|
||||
$tc = new TaskCreation($this->container);
|
||||
$tl = new TaskLink($this->container);
|
||||
$tf = new TaskFilter($this->container);
|
||||
|
||||
$this->assertEquals(1, $p->create(array('name' => 'test')));
|
||||
$this->assertEquals(2, $u->create(array('username' => 'bob', 'name' => 'Bob Ryan')));
|
||||
$this->assertNotFalse($tc->create(array('project_id' => 1, 'title' => 'my task title is awesome', 'color_id' => 'light_green')));
|
||||
$this->assertNotFalse($tc->create(array('project_id' => 1, 'title' => 'my task title is amazing', 'color_id' => 'blue')));
|
||||
$this->assertNotFalse($tc->create(array('project_id' => 1, 'title' => 'Bob at work')));
|
||||
$this->assertNotFalse($tc->create(array('project_id' => 1, 'title' => 'I have a bad feeling about that')));
|
||||
$this->assertEquals(1, $tl->create(1, 2, 9)); // #1 is a milestone of #2
|
||||
$this->assertEquals(3, $tl->create(2, 1, 2)); // #2 blocks #1
|
||||
$this->assertEquals(5, $tl->create(3, 2, 2)); // #3 blocks #2
|
||||
|
||||
$tf->search('link:"is a milestone of"');
|
||||
$tasks = $tf->findAll();
|
||||
$this->assertNotEmpty($tasks);
|
||||
$this->assertCount(1, $tasks);
|
||||
$this->assertEquals('my task title is awesome', $tasks[0]['title']);
|
||||
|
||||
$tf->search('link:"is a milestone of" amazing');
|
||||
$tasks = $tf->findAll();
|
||||
$this->assertEmpty($tasks);
|
||||
|
||||
$tf->search('link:"unknown"');
|
||||
$tasks = $tf->findAll();
|
||||
$this->assertEmpty($tasks);
|
||||
|
||||
$tf->search('link:unknown');
|
||||
$tasks = $tf->findAll();
|
||||
$this->assertEmpty($tasks);
|
||||
|
||||
$tf->search('link:blocks amazing');
|
||||
$tasks = $tf->findAll();
|
||||
$this->assertNotEmpty($tasks);
|
||||
$this->assertCount(1, $tasks);
|
||||
$this->assertEquals('my task title is amazing', $tasks[0]['title']);
|
||||
|
||||
$tf->search('link:"is a milestone of" link:blocks');
|
||||
$tasks = $tf->findAll();
|
||||
$this->assertNotEmpty($tasks);
|
||||
$this->assertCount(3, $tasks);
|
||||
$this->assertEquals('my task title is awesome', $tasks[0]['title']);
|
||||
$this->assertEquals('my task title is amazing', $tasks[1]['title']);
|
||||
$this->assertEquals('Bob at work', $tasks[2]['title']);
|
||||
|
||||
$tf->search('link:"is a milestone of" link:blocks link:unknown');
|
||||
$tasks = $tf->findAll();
|
||||
$this->assertNotEmpty($tasks);
|
||||
$this->assertCount(3, $tasks);
|
||||
$this->assertEquals('my task title is awesome', $tasks[0]['title']);
|
||||
$this->assertEquals('my task title is amazing', $tasks[1]['title']);
|
||||
$this->assertEquals('Bob at work', $tasks[2]['title']);
|
||||
}
|
||||
|
||||
public function testCopy()
|
||||
{
|
||||
$tf = new TaskFilter($this->container);
|
||||
|
|
|
|||
Loading…
Reference in New Issue