2009-05-06 08:36:16 +02:00
|
|
|
<?php
|
|
|
|
|
2016-06-15 06:03:16 +02:00
|
|
|
use SilverStripe\ORM\ValidationException;
|
|
|
|
use SilverStripe\ORM\Versioning\Versioned;
|
|
|
|
use SilverStripe\ORM\DataObject;
|
|
|
|
|
2009-05-06 08:36:16 +02:00
|
|
|
class HierarchyTest extends SapphireTest {
|
2011-03-30 08:49:11 +02:00
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
protected static $fixture_file = 'HierarchyTest.yml';
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
protected $requiredExtensions = array(
|
2016-06-15 06:03:16 +02:00
|
|
|
'HierarchyTest_Object' => array('SilverStripe\\ORM\\Hierarchy\\Hierarchy', 'SilverStripe\\ORM\\Versioning\\Versioned'),
|
|
|
|
'HierarchyHideTest_Object' => array('SilverStripe\\ORM\\Hierarchy\\Hierarchy', 'SilverStripe\\ORM\\Versioning\\Versioned'),
|
2011-03-23 04:32:24 +01:00
|
|
|
);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
protected $extraDataObjects = array(
|
2016-05-15 05:19:39 +02:00
|
|
|
'HierarchyTest_Object',
|
|
|
|
'HierarchyHideTest_Object'
|
2011-03-23 04:32:24 +01:00
|
|
|
);
|
2009-05-06 08:36:16 +02:00
|
|
|
|
2012-04-13 06:19:06 +02:00
|
|
|
/**
|
|
|
|
* Test the Hierarchy prevents infinite loops.
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testPreventLoop() {
|
2012-04-13 06:19:06 +02:00
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
|
|
|
|
$obj2->ParentID = $obj2aa->ID;
|
|
|
|
try {
|
|
|
|
$obj2->write();
|
|
|
|
}
|
|
|
|
catch (ValidationException $e) {
|
|
|
|
$this->assertContains('Infinite loop found within the "HierarchyTest_Object" hierarchy', $e->getMessage());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->fail('Failed to prevent infinite loop in hierarchy.');
|
|
|
|
}
|
|
|
|
|
2009-05-06 08:36:16 +02:00
|
|
|
/**
|
|
|
|
* Test Hierarchy::AllHistoricalChildren().
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testAllHistoricalChildren() {
|
2011-03-23 04:32:24 +01:00
|
|
|
// Delete some objs
|
|
|
|
$this->objFromFixture('HierarchyTest_Object', 'obj2b')->delete();
|
|
|
|
$this->objFromFixture('HierarchyTest_Object', 'obj3a')->delete();
|
|
|
|
$this->objFromFixture('HierarchyTest_Object', 'obj3')->delete();
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
// Check that obj1-3 appear at the top level of the AllHistoricalChildren tree
|
2014-08-15 08:53:05 +02:00
|
|
|
$this->assertEquals(array("Obj 1", "Obj 2", "Obj 3"),
|
2011-03-23 04:32:24 +01:00
|
|
|
singleton('HierarchyTest_Object')->AllHistoricalChildren()->column('Title'));
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2010-04-12 04:33:46 +02:00
|
|
|
// Check numHistoricalChildren
|
2011-03-23 04:32:24 +01:00
|
|
|
$this->assertEquals(3, singleton('HierarchyTest_Object')->numHistoricalChildren());
|
2010-04-12 04:33:46 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
// Check that both obj 2 children are returned
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
2014-08-15 08:53:05 +02:00
|
|
|
$this->assertEquals(array("Obj 2a", "Obj 2b"),
|
2011-03-23 04:32:24 +01:00
|
|
|
$obj2->AllHistoricalChildren()->column('Title'));
|
2010-04-12 04:33:46 +02:00
|
|
|
|
|
|
|
// Check numHistoricalChildren
|
2011-03-23 04:32:24 +01:00
|
|
|
$this->assertEquals(2, $obj2->numHistoricalChildren());
|
2010-04-12 04:33:46 +02:00
|
|
|
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
// Obj 3 has been deleted; let's bring it back from the grave
|
|
|
|
$obj3 = Versioned::get_including_deleted("HierarchyTest_Object", "\"Title\" = 'Obj 3'")->First();
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2016-04-20 01:21:14 +02:00
|
|
|
// Check that all obj 3 children are returned
|
|
|
|
$this->assertEquals(array("Obj 3a", "Obj 3b", "Obj 3c", "Obj 3d"),
|
2011-03-23 04:32:24 +01:00
|
|
|
$obj3->AllHistoricalChildren()->column('Title'));
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2010-04-12 04:33:46 +02:00
|
|
|
// Check numHistoricalChildren
|
2016-04-20 01:21:14 +02:00
|
|
|
$this->assertEquals(4, $obj3->numHistoricalChildren());
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2009-05-06 08:36:16 +02:00
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2009-05-06 08:36:16 +02:00
|
|
|
/**
|
2011-03-23 04:32:24 +01:00
|
|
|
* Test that you can call Hierarchy::markExpanded/Unexpanded/Open() on a obj, and that
|
2009-05-06 08:36:16 +02:00
|
|
|
* calling Hierarchy::isMarked() on a different instance of that object will return true.
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testItemMarkingIsntRestrictedToSpecificInstance() {
|
2011-03-23 04:32:24 +01:00
|
|
|
// Mark a few objs
|
|
|
|
$this->objFromFixture('HierarchyTest_Object', 'obj2')->markExpanded();
|
|
|
|
$this->objFromFixture('HierarchyTest_Object', 'obj2a')->markExpanded();
|
|
|
|
$this->objFromFixture('HierarchyTest_Object', 'obj2b')->markExpanded();
|
|
|
|
$this->objFromFixture('HierarchyTest_Object', 'obj3')->markUnexpanded();
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
// Query some objs in a different context and check their m
|
|
|
|
$objs = DataObject::get("HierarchyTest_Object", '', '"ID" ASC');
|
2009-05-06 08:36:16 +02:00
|
|
|
$marked = $expanded = array();
|
2011-03-23 04:32:24 +01:00
|
|
|
foreach($objs as $obj) {
|
|
|
|
if($obj->isMarked()) $marked[] = $obj->Title;
|
|
|
|
if($obj->isExpanded()) $expanded[] = $obj->Title;
|
2009-05-06 08:36:16 +02:00
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
$this->assertEquals(array('Obj 2', 'Obj 3', 'Obj 2a', 'Obj 2b'), $marked);
|
|
|
|
$this->assertEquals(array('Obj 2', 'Obj 2a', 'Obj 2b'), $expanded);
|
2009-10-23 00:18:02 +02:00
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testNumChildren() {
|
2011-03-23 04:32:24 +01:00
|
|
|
$this->assertEquals($this->objFromFixture('HierarchyTest_Object', 'obj1')->numChildren(), 0);
|
|
|
|
$this->assertEquals($this->objFromFixture('HierarchyTest_Object', 'obj2')->numChildren(), 2);
|
2016-04-20 01:21:14 +02:00
|
|
|
$this->assertEquals($this->objFromFixture('HierarchyTest_Object', 'obj3')->numChildren(), 4);
|
2011-03-23 04:32:24 +01:00
|
|
|
$this->assertEquals($this->objFromFixture('HierarchyTest_Object', 'obj2a')->numChildren(), 2);
|
|
|
|
$this->assertEquals($this->objFromFixture('HierarchyTest_Object', 'obj2b')->numChildren(), 0);
|
|
|
|
$this->assertEquals($this->objFromFixture('HierarchyTest_Object', 'obj3a')->numChildren(), 2);
|
2016-04-20 01:21:14 +02:00
|
|
|
$this->assertEquals($this->objFromFixture('HierarchyTest_Object', 'obj3d')->numChildren(), 0);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$this->assertEquals($obj1->numChildren(), 0);
|
|
|
|
$obj1Child1 = new HierarchyTest_Object();
|
|
|
|
$obj1Child1->ParentID = $obj1->ID;
|
|
|
|
$obj1Child1->write();
|
|
|
|
$this->assertEquals($obj1->numChildren(false), 1,
|
2009-10-26 21:56:54 +01:00
|
|
|
'numChildren() caching can be disabled through method parameter'
|
|
|
|
);
|
2011-03-23 04:32:24 +01:00
|
|
|
$obj1Child2 = new HierarchyTest_Object();
|
|
|
|
$obj1Child2->ParentID = $obj1->ID;
|
|
|
|
$obj1Child2->write();
|
|
|
|
$obj1->flushCache();
|
|
|
|
$this->assertEquals($obj1->numChildren(), 2,
|
2009-10-26 21:56:54 +01:00
|
|
|
'numChildren() caching can be disabled by flushCache()'
|
|
|
|
);
|
2009-10-23 00:18:02 +02:00
|
|
|
}
|
2010-04-13 05:20:33 +02:00
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testLoadDescendantIDListIntoArray() {
|
2011-03-23 04:32:24 +01:00
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2b = $this->objFromFixture('HierarchyTest_Object', 'obj2b');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
$obj2ab = $this->objFromFixture('HierarchyTest_Object', 'obj2ab');
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
$obj2IdList = $obj2->getDescendantIDList();
|
|
|
|
$obj2aIdList = $obj2a->getDescendantIDList();
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
$this->assertContains($obj2a->ID, $obj2IdList);
|
|
|
|
$this->assertContains($obj2b->ID, $obj2IdList);
|
|
|
|
$this->assertContains($obj2aa->ID, $obj2IdList);
|
|
|
|
$this->assertContains($obj2ab->ID, $obj2IdList);
|
|
|
|
$this->assertEquals(4, count($obj2IdList));
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2011-03-23 04:32:24 +01:00
|
|
|
$this->assertContains($obj2aa->ID, $obj2aIdList);
|
|
|
|
$this->assertContains($obj2ab->ID, $obj2aIdList);
|
|
|
|
$this->assertEquals(2, count($obj2aIdList));
|
2010-04-13 05:20:33 +02:00
|
|
|
}
|
|
|
|
|
2012-02-08 06:38:37 +01:00
|
|
|
/**
|
|
|
|
* The "only deleted from stage" argument to liveChildren() should exclude
|
|
|
|
* any page that has been moved to another location on the stage site
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testLiveChildrenOnlyDeletedFromStage() {
|
2012-02-08 06:38:37 +01:00
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2b = $this->objFromFixture('HierarchyTest_Object', 'obj2b');
|
|
|
|
|
|
|
|
// Get a published set of objects for our fixture
|
2016-04-01 05:27:59 +02:00
|
|
|
$obj1->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
$obj2->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
$obj2a->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
$obj2b->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2012-02-08 06:38:37 +01:00
|
|
|
// Then delete 2a from stage and move 2b to a sub-node of 1.
|
|
|
|
$obj2a->delete();
|
|
|
|
$obj2b->ParentID = $obj1->ID;
|
|
|
|
$obj2b->write();
|
|
|
|
|
|
|
|
// Get live children, excluding pages that have been moved on the stage site
|
|
|
|
$children = $obj2->liveChildren(true, true)->column("Title");
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2012-02-08 06:38:37 +01:00
|
|
|
// 2a has been deleted from stage and should be shown
|
|
|
|
$this->assertContains("Obj 2a", $children);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2012-02-08 06:38:37 +01:00
|
|
|
// 2b has merely been moved to a different parent and so shouldn't be shown
|
|
|
|
$this->assertNotContains("Obj 2b", $children);
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testBreadcrumbs() {
|
2012-03-02 17:10:47 +01:00
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
|
|
|
|
$this->assertEquals('Obj 1', $obj1->getBreadcrumbs());
|
|
|
|
$this->assertEquals('Obj 2 » Obj 2a', $obj2a->getBreadcrumbs());
|
|
|
|
$this->assertEquals('Obj 2 » Obj 2a » Obj 2aa', $obj2aa->getBreadcrumbs());
|
|
|
|
}
|
|
|
|
|
2016-04-20 01:21:14 +02:00
|
|
|
/**
|
|
|
|
* @covers Hierarchy::markChildren()
|
|
|
|
*/
|
|
|
|
public function testMarkChildrenDoesntUnmarkPreviouslyMarked() {
|
|
|
|
$obj3 = $this->objFromFixture('HierarchyTest_Object', 'obj3');
|
|
|
|
$obj3aa = $this->objFromFixture('HierarchyTest_Object', 'obj3aa');
|
|
|
|
$obj3ba = $this->objFromFixture('HierarchyTest_Object', 'obj3ba');
|
|
|
|
$obj3ca = $this->objFromFixture('HierarchyTest_Object', 'obj3ca');
|
|
|
|
|
|
|
|
$obj3->markPartialTree();
|
|
|
|
$obj3->markToExpose($obj3aa);
|
|
|
|
$obj3->markToExpose($obj3ba);
|
|
|
|
$obj3->markToExpose($obj3ca);
|
|
|
|
|
|
|
|
$expected = <<<EOT
|
|
|
|
<ul>
|
|
|
|
<li>Obj 3a
|
|
|
|
<ul>
|
|
|
|
<li>Obj 3aa
|
|
|
|
</li>
|
|
|
|
<li>Obj 3ab
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li>Obj 3b
|
|
|
|
<ul>
|
|
|
|
<li>Obj 3ba
|
|
|
|
</li>
|
|
|
|
<li>Obj 3bb
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li>Obj 3c
|
|
|
|
<ul>
|
|
|
|
<li>Obj 3c
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li>Obj 3d
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
|
|
|
|
EOT;
|
|
|
|
|
|
|
|
$this->assertSame($expected, $obj3->getChildrenAsUL());
|
|
|
|
}
|
|
|
|
|
2013-03-12 23:01:43 +01:00
|
|
|
public function testGetChildrenAsUL() {
|
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
|
|
|
|
$nodeCountThreshold = 30;
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
|
|
|
$root->markPartialTree($nodeCountThreshold);
|
|
|
|
$html = $root->getChildrenAsUL(
|
2014-08-15 08:53:05 +02:00
|
|
|
"",
|
|
|
|
'"<li id=\"" . $child->ID . "\">" . $child->Title',
|
|
|
|
null,
|
|
|
|
false,
|
|
|
|
"AllChildrenIncludingDeleted",
|
|
|
|
"numChildren",
|
2013-03-12 23:01:43 +01:00
|
|
|
true, // rootCall
|
|
|
|
$nodeCountThreshold
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains root elements'
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2, $obj2a),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains child elements (in correct nesting)'
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2, $obj2a, $obj2aa),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains grandchild elements (in correct nesting)'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetChildrenAsULMinNodeCount() {
|
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
|
|
|
|
// Set low enough that it should be fulfilled by root only elements
|
|
|
|
$nodeCountThreshold = 3;
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
|
|
|
$root->markPartialTree($nodeCountThreshold);
|
|
|
|
$html = $root->getChildrenAsUL(
|
2014-08-15 08:53:05 +02:00
|
|
|
"",
|
|
|
|
'"<li id=\"" . $child->ID . "\">" . $child->Title',
|
|
|
|
null,
|
|
|
|
false,
|
|
|
|
"AllChildrenIncludingDeleted",
|
|
|
|
"numChildren",
|
|
|
|
true,
|
2013-03-12 23:01:43 +01:00
|
|
|
$nodeCountThreshold
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj1),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains root elements'
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains root elements'
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeNotContains($html, array($obj2, $obj2a),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Does not contains child elements because they exceed minNodeCount'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetChildrenAsULMinNodeCountWithMarkToExpose() {
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
|
|
|
|
// Set low enough that it should be fulfilled by root only elements
|
|
|
|
$nodeCountThreshold = 3;
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
|
|
|
$root->markPartialTree($nodeCountThreshold);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2013-03-12 23:01:43 +01:00
|
|
|
// Mark certain node which should be included regardless of minNodeCount restrictions
|
|
|
|
$root->markToExpose($obj2aa);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2013-03-12 23:01:43 +01:00
|
|
|
$html = $root->getChildrenAsUL(
|
2014-08-15 08:53:05 +02:00
|
|
|
"",
|
|
|
|
'"<li id=\"" . $child->ID . "\">" . $child->Title',
|
|
|
|
null,
|
|
|
|
false,
|
|
|
|
"AllChildrenIncludingDeleted",
|
|
|
|
"numChildren",
|
|
|
|
true,
|
2013-03-12 23:01:43 +01:00
|
|
|
$nodeCountThreshold
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains root elements'
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2, $obj2a, $obj2aa),
|
|
|
|
'Does contain marked children nodes regardless of configured threshold'
|
2013-03-12 23:01:43 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetChildrenAsULMinNodeCountWithFilters() {
|
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
|
|
|
|
// Set low enough that it should fit all search matches without lazy loading
|
|
|
|
$nodeCountThreshold = 3;
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2013-03-12 23:01:43 +01:00
|
|
|
// Includes nodes by filter regardless of minNodeCount restrictions
|
|
|
|
$root->setMarkingFilterFunction(function($record) use($obj2, $obj2a, $obj2aa) {
|
|
|
|
// Results need to include parent hierarchy, even if we just want to
|
|
|
|
// match the innermost node.
|
|
|
|
return in_array($record->ID, array($obj2->ID, $obj2a->ID, $obj2aa->ID));
|
|
|
|
});
|
|
|
|
$root->markPartialTree($nodeCountThreshold);
|
|
|
|
|
|
|
|
$html = $root->getChildrenAsUL(
|
2014-08-15 08:53:05 +02:00
|
|
|
"",
|
|
|
|
'"<li id=\"" . $child->ID . "\">" . $child->Title',
|
|
|
|
null,
|
2013-03-12 23:01:43 +01:00
|
|
|
true, // limit to marked
|
2014-08-15 08:53:05 +02:00
|
|
|
"AllChildrenIncludingDeleted",
|
|
|
|
"numChildren",
|
|
|
|
true,
|
2013-03-12 23:01:43 +01:00
|
|
|
$nodeCountThreshold
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeNotContains($html, array($obj1),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Does not contain root elements which dont match the filter'
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2, $obj2a, $obj2aa),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains non-root elements which match the filter'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetChildrenAsULHardLimitsNodes() {
|
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
|
|
|
|
// Set low enough that it should fit all search matches without lazy loading
|
|
|
|
$nodeCountThreshold = 3;
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2013-03-12 23:01:43 +01:00
|
|
|
// Includes nodes by filter regardless of minNodeCount restrictions
|
|
|
|
$root->setMarkingFilterFunction(function($record) use($obj2, $obj2a, $obj2aa) {
|
|
|
|
// Results need to include parent hierarchy, even if we just want to
|
|
|
|
// match the innermost node.
|
|
|
|
return in_array($record->ID, array($obj2->ID, $obj2a->ID, $obj2aa->ID));
|
|
|
|
});
|
|
|
|
$root->markPartialTree($nodeCountThreshold);
|
|
|
|
|
|
|
|
$html = $root->getChildrenAsUL(
|
2014-08-15 08:53:05 +02:00
|
|
|
"",
|
|
|
|
'"<li id=\"" . $child->ID . "\">" . $child->Title',
|
|
|
|
null,
|
2013-03-12 23:01:43 +01:00
|
|
|
true, // limit to marked
|
2014-08-15 08:53:05 +02:00
|
|
|
"AllChildrenIncludingDeleted",
|
|
|
|
"numChildren",
|
|
|
|
true,
|
2013-03-12 23:01:43 +01:00
|
|
|
$nodeCountThreshold
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeNotContains($html, array($obj1),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Does not contain root elements which dont match the filter'
|
|
|
|
);
|
2013-03-19 22:26:48 +01:00
|
|
|
$this->assertTreeContains($html, array($obj2, $obj2a, $obj2aa),
|
2013-03-12 23:01:43 +01:00
|
|
|
'Contains non-root elements which match the filter'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2013-03-19 22:26:48 +01:00
|
|
|
public function testGetChildrenAsULNodeThresholdLeaf() {
|
|
|
|
$obj1 = $this->objFromFixture('HierarchyTest_Object', 'obj1');
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj3 = $this->objFromFixture('HierarchyTest_Object', 'obj3');
|
|
|
|
$obj3a = $this->objFromFixture('HierarchyTest_Object', 'obj3a');
|
|
|
|
|
|
|
|
$nodeCountThreshold = 99;
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
|
|
|
$root->markPartialTree($nodeCountThreshold);
|
|
|
|
$nodeCountCallback = function($parent, $numChildren) {
|
|
|
|
// Set low enough that it the fixture structure should exceed it
|
|
|
|
if($parent->ID && $numChildren > 2) {
|
|
|
|
return '<span class="exceeded">Exceeded!</span>';
|
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
};
|
2013-03-19 22:26:48 +01:00
|
|
|
|
|
|
|
$html = $root->getChildrenAsUL(
|
2014-08-15 08:53:05 +02:00
|
|
|
"",
|
|
|
|
'"<li id=\"" . $child->ID . "\">" . $child->Title',
|
|
|
|
null,
|
2013-03-19 22:26:48 +01:00
|
|
|
true, // limit to marked
|
2014-08-15 08:53:05 +02:00
|
|
|
"AllChildrenIncludingDeleted",
|
|
|
|
"numChildren",
|
|
|
|
true,
|
2013-03-19 22:26:48 +01:00
|
|
|
$nodeCountThreshold,
|
|
|
|
$nodeCountCallback
|
|
|
|
);
|
|
|
|
$this->assertTreeContains($html, array($obj1),
|
|
|
|
'Does contain root elements regardless of count'
|
|
|
|
);
|
|
|
|
$this->assertTreeContains($html, array($obj3),
|
|
|
|
'Does contain root elements regardless of count'
|
|
|
|
);
|
|
|
|
$this->assertTreeContains($html, array($obj2, $obj2a),
|
|
|
|
'Contains children which do not exceed threshold'
|
|
|
|
);
|
|
|
|
$this->assertTreeNotContains($html, array($obj3, $obj3a),
|
|
|
|
'Does not contain children which exceed threshold'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2014-07-20 01:49:17 +02:00
|
|
|
/**
|
|
|
|
* This test checks that deleted ('archived') child pages don't set a css class on the parent
|
|
|
|
* node that makes it look like it has children
|
|
|
|
*/
|
|
|
|
public function testGetChildrenAsULNodeDeletedOnLive() {
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
$obj2ab = $this->objFromFixture('HierarchyTest_Object', 'obj2b');
|
|
|
|
|
|
|
|
// delete all children under obj2
|
|
|
|
$obj2a->delete();
|
|
|
|
$obj2aa->delete();
|
|
|
|
$obj2ab->delete();
|
|
|
|
// Don't pre-load all children
|
|
|
|
$nodeCountThreshold = 1;
|
|
|
|
|
|
|
|
$childrenMethod = 'AllChildren';
|
|
|
|
$numChildrenMethod = 'numChildren';
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
|
|
|
$root->markPartialTree($nodeCountThreshold, null, $childrenMethod, $numChildrenMethod);
|
|
|
|
|
|
|
|
// As in LeftAndMain::getSiteTreeFor() but simpler and more to the point for testing purposes
|
|
|
|
$titleFn = function(&$child, $numChildrenMethod="") {
|
|
|
|
return '<li class="' . $child->markingClasses($numChildrenMethod).
|
|
|
|
'" id="' . $child->ID . '">"' . $child->Title;
|
|
|
|
};
|
|
|
|
|
|
|
|
$html = $root->getChildrenAsUL(
|
|
|
|
"",
|
|
|
|
$titleFn,
|
|
|
|
null,
|
|
|
|
true, // limit to marked
|
|
|
|
$childrenMethod,
|
|
|
|
$numChildrenMethod,
|
|
|
|
true,
|
|
|
|
$nodeCountThreshold
|
|
|
|
);
|
|
|
|
|
|
|
|
// Get the class attribute from the $obj2 node in the sitetree, class 'jstree-leaf' means it's a leaf node
|
|
|
|
$nodeClass = $this->getNodeClassFromTree($html, $obj2);
|
|
|
|
$this->assertEquals('jstree-leaf closed', $nodeClass, 'object2 should not have children in the sitetree');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This test checks that deleted ('archived') child pages _do_ set a css class on the parent
|
|
|
|
* node that makes it look like it has children when getting all children including deleted
|
|
|
|
*/
|
|
|
|
public function testGetChildrenAsULNodeDeletedOnStage() {
|
|
|
|
$obj2 = $this->objFromFixture('HierarchyTest_Object', 'obj2');
|
|
|
|
$obj2a = $this->objFromFixture('HierarchyTest_Object', 'obj2a');
|
|
|
|
$obj2aa = $this->objFromFixture('HierarchyTest_Object', 'obj2aa');
|
|
|
|
$obj2ab = $this->objFromFixture('HierarchyTest_Object', 'obj2b');
|
|
|
|
|
|
|
|
// delete all children under obj2
|
|
|
|
$obj2a->delete();
|
|
|
|
$obj2aa->delete();
|
|
|
|
$obj2ab->delete();
|
|
|
|
// Don't pre-load all children
|
|
|
|
$nodeCountThreshold = 1;
|
|
|
|
|
|
|
|
$childrenMethod = 'AllChildrenIncludingDeleted';
|
|
|
|
$numChildrenMethod = 'numHistoricalChildren';
|
|
|
|
|
|
|
|
$root = new HierarchyTest_Object();
|
|
|
|
$root->markPartialTree($nodeCountThreshold, null, $childrenMethod, $numChildrenMethod);
|
|
|
|
|
|
|
|
// As in LeftAndMain::getSiteTreeFor() but simpler and more to the point for testing purposes
|
|
|
|
$titleFn = function(&$child, $numChildrenMethod="") {
|
|
|
|
return '<li class="' . $child->markingClasses($numChildrenMethod).
|
|
|
|
'" id="' . $child->ID . '">"' . $child->Title;
|
|
|
|
};
|
|
|
|
|
|
|
|
$html = $root->getChildrenAsUL(
|
|
|
|
"",
|
|
|
|
$titleFn,
|
|
|
|
null,
|
|
|
|
true, // limit to marked
|
|
|
|
$childrenMethod,
|
|
|
|
$numChildrenMethod,
|
|
|
|
true,
|
|
|
|
$nodeCountThreshold
|
|
|
|
);
|
|
|
|
|
|
|
|
// Get the class attribute from the $obj2 node in the sitetree
|
|
|
|
$nodeClass = $this->getNodeClassFromTree($html, $obj2);
|
|
|
|
// Object2 can now be expanded
|
|
|
|
$this->assertEquals('unexpanded jstree-closed closed', $nodeClass, 'obj2 should have children in the sitetree');
|
|
|
|
}
|
|
|
|
|
2016-05-15 05:19:39 +02:00
|
|
|
public function testNoHideFromHeirarchy() {
|
|
|
|
$obj4 = $this->objFromFixture('HierarchyHideTest_Object', 'obj4');
|
|
|
|
$obj4->publish("Stage", "Live");
|
|
|
|
|
|
|
|
foreach($obj4->stageChildren() as $child) {
|
|
|
|
$child->publish("Stage", "Live");
|
|
|
|
}
|
|
|
|
$this->assertEquals($obj4->stageChildren()->Count(), 2);
|
|
|
|
$this->assertEquals($obj4->liveChildren()->Count(), 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testHideFromHeirarchy() {
|
|
|
|
HierarchyHideTest_Object::config()->hide_from_hierarchy = array('HierarchyHideTest_SubObject');
|
|
|
|
$obj4 = $this->objFromFixture('HierarchyHideTest_Object', 'obj4');
|
|
|
|
$obj4->publish("Stage", "Live");
|
|
|
|
|
|
|
|
// load without using stage children otherwise it'll bbe filtered before it's publish
|
|
|
|
// we need to publish all of them, and expect liveChildren to return some.
|
|
|
|
$children = HierarchyHideTest_Object::get()
|
|
|
|
->filter('ParentID', (int)$obj4->ID)
|
|
|
|
->exclude('ID', (int)$obj4->ID);
|
|
|
|
|
|
|
|
foreach($children as $child) {
|
|
|
|
$child->publish("Stage", "Live");
|
|
|
|
}
|
|
|
|
$this->assertEquals($obj4->stageChildren()->Count(), 1);
|
|
|
|
$this->assertEquals($obj4->liveChildren()->Count(), 1);
|
|
|
|
}
|
|
|
|
|
2013-03-19 22:26:48 +01:00
|
|
|
/**
|
|
|
|
* @param String $html [description]
|
|
|
|
* @param array $nodes Breadcrumb path as array
|
|
|
|
* @param String $message
|
|
|
|
*/
|
|
|
|
protected function assertTreeContains($html, $nodes, $message = null) {
|
|
|
|
$parser = new CSSContentParser($html);
|
|
|
|
$xpath = '/';
|
|
|
|
foreach($nodes as $node) $xpath .= '/ul/li[@id="' . $node->ID . '"]';
|
|
|
|
$match = $parser->getByXpath($xpath);
|
|
|
|
self::assertThat((bool)$match, self::isTrue(), $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param String $html [description]
|
|
|
|
* @param array $nodes Breadcrumb path as array
|
|
|
|
* @param String $message
|
|
|
|
*/
|
|
|
|
protected function assertTreeNotContains($html, $nodes, $message = null) {
|
|
|
|
$parser = new CSSContentParser($html);
|
|
|
|
$xpath = '/';
|
|
|
|
foreach($nodes as $node) $xpath .= '/ul/li[@id="' . $node->ID . '"]';
|
|
|
|
$match = $parser->getByXpath($xpath);
|
2014-08-15 08:53:05 +02:00
|
|
|
self::assertThat((bool)$match, self::isFalse(), $message);
|
2013-03-19 22:26:48 +01:00
|
|
|
}
|
|
|
|
|
2014-07-20 01:49:17 +02:00
|
|
|
/**
|
|
|
|
* Get the HTML class attribute from a node in the sitetree
|
|
|
|
*
|
|
|
|
* @param $html
|
|
|
|
* @param $node
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function getNodeClassFromTree($html, $node) {
|
|
|
|
$parser = new CSSContentParser($html);
|
|
|
|
$xpath = '//ul/li[@id="' . $node->ID . '"]';
|
|
|
|
$object = $parser->getByXpath($xpath);
|
|
|
|
|
|
|
|
foreach($object[0]->attributes() as $key => $attr) {
|
|
|
|
if($key == 'class') {
|
|
|
|
return (string)$attr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return '';
|
|
|
|
}
|
2011-03-23 04:32:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
class HierarchyTest_Object extends DataObject implements TestOnly {
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $db = array(
|
2011-03-23 04:32:24 +01:00
|
|
|
'Title' => 'Varchar'
|
|
|
|
);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $extensions = array(
|
2016-06-15 06:03:16 +02:00
|
|
|
'SilverStripe\\ORM\\Hierarchy\\Hierarchy',
|
|
|
|
'SilverStripe\\ORM\\Versioning\\Versioned',
|
2011-03-23 04:32:24 +01:00
|
|
|
);
|
2014-07-20 01:49:17 +02:00
|
|
|
|
2016-04-20 01:21:14 +02:00
|
|
|
private static $default_sort = 'Title ASC';
|
|
|
|
|
2014-07-20 01:49:17 +02:00
|
|
|
public function cmstreeclasses() {
|
|
|
|
return $this->markingClasses();
|
|
|
|
}
|
|
|
|
}
|
2016-05-15 05:19:39 +02:00
|
|
|
|
|
|
|
class HierarchyHideTest_Object extends DataObject implements TestOnly {
|
|
|
|
private static $db = array(
|
|
|
|
'Title' => 'Varchar'
|
|
|
|
);
|
|
|
|
|
|
|
|
private static $extensions = array(
|
2016-06-15 06:03:16 +02:00
|
|
|
'SilverStripe\\ORM\\Hierarchy\\Hierarchy',
|
|
|
|
"SilverStripe\\ORM\\Versioning\\Versioned('Stage', 'Live')",
|
2016-05-15 05:19:39 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
public function cmstreeclasses() {
|
|
|
|
return $this->markingClasses();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class HierarchyHideTest_SubObject extends HierarchyHideTest_Object {
|
|
|
|
|
|
|
|
}
|