setAccessible(true); $config = new GridFieldConfig_RelationEditor(); $config->addComponent($orderable); list($parentClass, $parentInstanceID) = explode('.', $fixtureID ?? ''); $parent = $this->objFromFixture($parentClass, $parentInstanceID); $grid = new GridField( $relationName, 'Testing Many Many', $parent->$relationName()->sort($sortName), $config ); $originalOrder = $parent->$relationName()->sort($sortName)->column('ID'); $desiredOrder = []; // Make order non-contiguous, and 1-based foreach (array_reverse($originalOrder ?? []) as $index => $id) { $desiredOrder[$index * 2 + 1] = $id; } $this->assertNotEquals($originalOrder, $desiredOrder); $reflection->invoke($orderable, $grid, $desiredOrder); $newOrder = $parent->$relationName()->sort($sortName)->map($sortName, 'ID')->toArray(); $this->assertEquals($desiredOrder, $newOrder); } public function testManyManyThroughListSortOrdersAreUsedForInitialRender() { /** @var ThroughDefiner $record */ $record = $this->objFromFixture(ThroughDefiner::class, 'DefinerOne'); $orderable = new GridFieldOrderableRows('Sort'); $config = new GridFieldConfig_RelationEditor(); $config->addComponent($orderable); $grid = new GridField( 'Belongings', 'Testing Many Many', $record->Belongings()->sort('Sort'), $config ); // Get the first record, which would be the first one to have column contents generated /** @var ThroughIntermediary $expected */ $intermediary = $this->objFromFixture(ThroughIntermediary::class, 'One'); $result = $orderable->getColumnContent($grid, $record, 'irrelevant'); $this->assertStringContainsString( 'Belongings[GridFieldEditableColumns][' . $record->ID . '][Sort]', $result, 'The field name is indexed under the record\'s ID' ); $this->assertStringContainsString( 'value="' . $intermediary->Sort . '"', $result, 'The value comes from the MMTL intermediary Sort value' ); } public function testPolymorphicManyManyListSortOrdersAreUsedForInitialRender() { $this->markTestSkipped('TODO: Introduce this test in the next minor release (3.3)'); $record = $this->objFromFixture(PolymorphM2MParent::class, 'ParentOne'); $orderable = new GridFieldOrderableRows('Sort'); $config = new GridFieldConfig_RelationEditor(); $config->addComponent($orderable); $grid = new GridField( 'Children', 'Testing Polymorphic Many Many', $record->Children()->sort('Sort'), $config ); // Get the first record, which would be the first one to have column contents generated $intermediary = $this->objFromFixture(PolymorphM2MMapper::class, 'MapP1ToC1'); $result = $orderable->getColumnContent($grid, $record, 'irrelevant'); $this->assertStringContainsString( 'Children[GridFieldEditableColumns][' . $record->ID . '][Sort]', $result, 'The field name is indexed under the record\'s ID' ); $this->assertStringContainsString( 'value="' . $intermediary->Sort . '"', $result, 'The value comes from the MMTL intermediary Sort value' ); } public function testSortableChildClass() { $orderable = new GridFieldOrderableRows('Sort'); $reflection = new ReflectionMethod($orderable, 'executeReorder'); $reflection->setAccessible(true); $parent = $this->objFromFixture(StubOrdered::class, 'nestedtest'); $config = new GridFieldConfig_RelationEditor(); $config->addComponent($orderable); $grid = new GridField( 'Children', 'Children', $parent->Children(), $config ); $originalOrder = $parent->Children()->column('ID'); $desiredOrder = array_reverse($originalOrder ?? []); $this->assertNotEquals($originalOrder, $desiredOrder); $reflection->invoke($orderable, $grid, $desiredOrder); $newOrder = $parent->Children()->column('ID'); $this->assertEquals($desiredOrder, $newOrder); } public function testGetSortTable() { $orderable = new GridFieldOrderableRows(); $parent = new StubParent(); $parent->write(); $this->assertEquals( 'StubOrdered', $orderable->getSortTable($parent->MyHasMany()) ); $this->assertEquals( 'StubOrdered', $orderable->getSortTable($parent->MyHasManySubclass()) ); $this->assertEquals( 'StubOrdered', $orderable->getSortTable($parent->MyManyMany()) ); $this->assertEquals( 'StubParent_MyManyMany', $orderable->setSortField('ManyManySort')->getSortTable($parent->MyManyMany()) ); $this->assertEquals( 'StubOrderedVersioned', $orderable->setSortField('Sort')->getSortTable($parent->MyHasManySubclassOrderedVersioned()) ); } public function testReorderItemsSubclassVersioned() { $orderable = new GridFieldOrderableRows('Sort'); $reflection = new ReflectionMethod($orderable, 'executeReorder'); $reflection->setAccessible(true); $parent = $this->objFromFixture(StubParent::class, 'parent-subclass-ordered-versioned'); // make sure all items are published foreach ($parent->MyHasManySubclassOrderedVersioned() as $item) { $item->publishRecursive(); } // there should be no difference between stages at this point $differenceFound = false; foreach ($parent->MyHasManySubclassOrderedVersioned() as $item) { /** @var StubSubclassOrderedVersioned|Versioned $item */ if ($item->stagesDiffer()) { $this->fail('Unexpected difference found on stages'); } } // reorder items $config = new GridFieldConfig_RelationEditor(); $config->addComponent($orderable); $grid = new GridField( 'TestField', 'TestField', $parent->MyHasManySubclassOrderedVersioned()->sort('Sort', 'ASC'), $config ); $originalOrder = $parent->MyHasManySubclassOrderedVersioned() ->sort('Sort', 'ASC') ->column('ID'); $desiredOrder = []; // Make order non-contiguous, and 1-based foreach (array_reverse($originalOrder ?? []) as $index => $id) { $desiredOrder[$index * 2 + 1] = $id; } $this->assertNotEquals($originalOrder, $desiredOrder); $reflection->invoke($orderable, $grid, $desiredOrder); $newOrder = $parent->MyHasManySubclassOrderedVersioned() ->sort('Sort', 'ASC') ->map('Sort', 'ID') ->toArray(); $this->assertEquals($desiredOrder, $newOrder); // reorder should have been handled as versioned - there should be a difference between stages now $differenceFound = false; foreach ($parent->MyHasManySubclassOrderedVersioned() as $item) { if ($item->stagesDiffer()) { $differenceFound = true; break; } } $this->assertTrue($differenceFound); } #[DataProvider('provideGetManipulatedData')] public function testGetManipulatedData(string $dataClass, string $listClass, array $data, array $expected) { $list = $listClass == DataList::class ? new DataList($dataClass) : new ArrayList(); foreach ($data as $values) { $item = new $dataClass(); $item->update($values); $item->write(); $list->add($item); } $orderable = new GridFieldOrderableRows('Title'); $config = new GridFieldConfig_RelationEditor(); $config->addComponent($orderable); $grid = new GridField('MyName', 'MyTitle', $list, $config); $sortedList = $orderable->getManipulatedData($grid, $list); $col = $dataClass === TitleObject::class ? 'Title' : 'Iden'; $this->assertSame($expected, $sortedList->column($col)); } public static function provideGetManipulatedData(): array { return [ [ TitleObject::class, ArrayList::class, [ ['Title' => 'C'], ['Title' => 'A'], ['Title' => 'B'] ], ['A', 'B', 'C'] ], [ TitleObject::class, DataList::class, [ ['Title' => 'C'], ['Title' => 'A'], ['Title' => 'B'], ], ['A', 'B', 'C'] ], [ TitleSortedObject::class, ArrayList::class, [ ['Title' => '1', 'Iden' => 'C'], ['Title' => '2', 'Iden' => 'A'], ['Title' => '3', 'Iden' => 'B'], ], ['C', 'A', 'B'] ], [ TitleSortedObject::class, DataList::class, [ ['Title' => '1', 'Iden' => 'C'], ['Title' => '2', 'Iden' => 'A'], ['Title' => '3', 'Iden' => 'B'], ], ['C', 'A', 'B'] ], [ TitleSortedObject::class, DataList::class, [ ['Title' => 'Z', 'Iden' => 'C', 'DefaultSort' => 3], ['Title' => 'Z', 'Iden' => 'A', 'DefaultSort' => 2], ['Title' => 'Z', 'Iden' => 'B', 'DefaultSort' => 1], ], ['B', 'A', 'C'] ], [ TitleArraySortedObject::class, DataList::class, [ ['Title' => 'X', 'Iden' => 'C', 'OtherSort' => 3], ['Title' => 'Z', 'Iden' => 'A', 'OtherSort' => 2], ['Title' => 'Z', 'Iden' => 'B', 'OtherSort' => 1], ], ['C', 'B', 'A'] ], ]; } }