From 4a10147c5be220e7ea814ddf22d7a46638104ebd Mon Sep 17 00:00:00 2001 From: Chris Penny Date: Fri, 4 Mar 2022 09:19:15 +1300 Subject: [PATCH 1/2] ENH: Update reorderItems() to use ORM where possible --- src/GridFieldOrderableRows.php | 45 ++++++---------------------------- 1 file changed, 7 insertions(+), 38 deletions(-) diff --git a/src/GridFieldOrderableRows.php b/src/GridFieldOrderableRows.php index c03b9bb..f44631f 100755 --- a/src/GridFieldOrderableRows.php +++ b/src/GridFieldOrderableRows.php @@ -587,9 +587,10 @@ protected function executeReorder(GridField $grid, $sortedIDs) */ protected function reorderItems($list, array $values, array $sortedIDs) { + $this->extend('onBeforeReorderItems', $list, $values, $sortedIDs); + // setup $sortField = $this->getSortField(); - $class = $list->dataClass(); // The problem is that $sortedIDs is a list of the _related_ item IDs, which causes trouble // with ManyManyThrough, where we need the ID of the _join_ item in order to set the value. $itemToSortReference = ($list instanceof ManyManyThroughList) ? 'getJoin' : 'Me'; @@ -598,53 +599,20 @@ protected function reorderItems($list, array $values, array $sortedIDs) // sanity check. $this->validateSortField($list); - $isVersioned = false; - // check if sort column is present on the model provided by dataClass() and if it's versioned - // cases: - // Model has sort column and is versioned - handle as versioned - // Model has sort column and is NOT versioned - handle as NOT versioned - // Model doesn't have sort column because sort column is on ManyManyList - handle as NOT versioned - // Model doesn't have sort column because sort column is on ManyManyThroughList - inspect through object - if ($list instanceof ManyManyThroughList) { - // We'll be updating the join class, not the relation class. - $class = $this->getManyManyInspector($list)->getJoinClass(); - $isVersioned = $class::create()->hasExtension(Versioned::class); - } elseif (!$this->isManyMany($list)) { - $isVersioned = $class::create()->hasExtension(Versioned::class); - } - - // Loop through each item, and update the sort values which do not - // match to order the objects. - if (!$isVersioned) { + // ManyManyList models are not represented by the ORM, and so they need to be updated through an SQL Query + if ($list instanceof ManyManyList) { $sortTable = $this->getSortTable($list); - $now = DBDatetime::now()->Rfc2822(); - $additionalSQL = ''; - $baseTable = DataObject::getSchema()->baseDataTable($class); - - $isBaseTable = ($baseTable == $sortTable); - if (!$list instanceof ManyManyList && $isBaseTable) { - $additionalSQL = ", \"LastEdited\" = '$now'"; - } + // Loop through each item, and update the sort values which do not match to order the objects. foreach ($sortedIDs as $newSortValue => $targetRecordID) { if ($currentSortList[$targetRecordID]->$sortField != $newSortValue) { DB::query(sprintf( - 'UPDATE "%s" SET "%s" = %d%s WHERE %s', + 'UPDATE "%s" SET "%s" = %d WHERE %s', $sortTable, $sortField, $newSortValue, - $additionalSQL, $this->getSortTableClauseForIds($list, $targetRecordID) )); - - if (!$isBaseTable && !$list instanceof ManyManyList) { - DB::query(sprintf( - 'UPDATE "%s" SET "LastEdited" = \'%s\' WHERE %s', - $baseTable, - $now, - $this->getSortTableClauseForIds($list, $targetRecordID) - )); - } } } } else { @@ -656,6 +624,7 @@ protected function reorderItems($list, array $values, array $sortedIDs) // either the list data class (has_many, (belongs_)many_many) // or the intermediary join class (many_many through) $record = $currentSortList[$targetRecordID]; + if ($record->$sortField != $newSortValue) { $record->$sortField = $newSortValue; $record->write(); From 5041e6c348dc59d4219a8f2905a82954b2cdef8d Mon Sep 17 00:00:00 2001 From: Chris Penny Date: Thu, 4 Aug 2022 07:19:37 +1200 Subject: [PATCH 2/2] Increase test coverage of orderable rows --- src/GridFieldOrderableRows.php | 3 +- tests/GridFieldOrderableRowsTest.php | 14 ++++++-- tests/GridFieldOrderableRowsTest.yml | 38 +++++++++++++++++++++ tests/OrderableRowsThroughTest.yml | 31 +++++++++++++++++ tests/Stub/StubParent.php | 2 ++ tests/Stub/ThroughBelongsVersioned.php | 25 ++++++++++++++ tests/Stub/ThroughDefinerVersioned.php | 33 ++++++++++++++++++ tests/Stub/ThroughIntermediaryVersioned.php | 32 +++++++++++++++++ 8 files changed, 175 insertions(+), 3 deletions(-) create mode 100644 tests/Stub/ThroughBelongsVersioned.php create mode 100644 tests/Stub/ThroughDefinerVersioned.php create mode 100644 tests/Stub/ThroughIntermediaryVersioned.php diff --git a/src/GridFieldOrderableRows.php b/src/GridFieldOrderableRows.php index f44631f..6b3d8bb 100755 --- a/src/GridFieldOrderableRows.php +++ b/src/GridFieldOrderableRows.php @@ -599,7 +599,8 @@ protected function reorderItems($list, array $values, array $sortedIDs) // sanity check. $this->validateSortField($list); - // ManyManyList models are not represented by the ORM, and so they need to be updated through an SQL Query + // ManyManyList extra fields aren't easily updated via the ORM, and so they need to be updated through an SQL + // Query if ($list instanceof ManyManyList) { $sortTable = $this->getSortTable($list); diff --git a/tests/GridFieldOrderableRowsTest.php b/tests/GridFieldOrderableRowsTest.php index 664ef11..1b23da9 100644 --- a/tests/GridFieldOrderableRowsTest.php +++ b/tests/GridFieldOrderableRowsTest.php @@ -8,7 +8,6 @@ use SilverStripe\Forms\GridField\GridFieldConfig_RelationEditor; use SilverStripe\ORM\DataList; use Symbiote\GridFieldExtensions\GridFieldOrderableRows; -use Symbiote\GridFieldExtensions\Tests\Stub\PolymorphM2MChild; use Symbiote\GridFieldExtensions\Tests\Stub\PolymorphM2MMapper; use Symbiote\GridFieldExtensions\Tests\Stub\PolymorphM2MParent; use Symbiote\GridFieldExtensions\Tests\Stub\StubOrderableChild; @@ -18,11 +17,14 @@ use Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass; use Symbiote\GridFieldExtensions\Tests\Stub\StubSubclassOrderedVersioned; use Symbiote\GridFieldExtensions\Tests\Stub\StubUnorderable; +use Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongs; +use Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongsVersioned; use Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefiner; +use Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefinerVersioned; use Symbiote\GridFieldExtensions\Tests\Stub\ThroughIntermediary; -use Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongs; use Symbiote\GridFieldExtensions\Tests\Stub\TitleObject; use Symbiote\GridFieldExtensions\Tests\Stub\TitleSortedObject; +use Symbiote\GridFieldExtensions\Tests\Stub\ThroughIntermediaryVersioned; /** * Tests for the {@link GridFieldOrderableRows} component. @@ -51,13 +53,21 @@ class GridFieldOrderableRowsTest extends SapphireTest ThroughBelongs::class, TitleObject::class, TitleSortedObject::class, + ThroughDefinerVersioned::class, + ThroughIntermediaryVersioned::class, + ThroughBelongsVersioned::class, ]; public function reorderItemsProvider() { return [ + [StubParent::class . '.parent', 'MyHasMany', 'Sort'], + [StubParent::class . '.parent', 'MyHasManySubclass', 'Sort'], + [StubParent::class . '.parent-subclass-ordered-versioned', 'MyHasManySubclassOrderedVersioned', 'Sort'], [StubParent::class . '.parent', 'MyManyMany', 'ManyManySort'], + [StubParent::class . '.parent', 'MyManyManyVersioned', 'ManyManySort'], [ThroughDefiner::class . '.DefinerOne', 'Belongings', 'Sort'], + [ThroughDefinerVersioned::class . '.DefinerOne', 'Belongings', 'Sort'], // [PolymorphM2MParent::class . '.ParentOne', 'Children', 'Sort'] ]; } diff --git a/tests/GridFieldOrderableRowsTest.yml b/tests/GridFieldOrderableRowsTest.yml index 8efc5ec..1fbe431 100644 --- a/tests/GridFieldOrderableRowsTest.yml +++ b/tests/GridFieldOrderableRowsTest.yml @@ -7,6 +7,7 @@ Symbiote\GridFieldExtensions\Tests\Stub\StubOrderableChild: Sort: 3 item4: Sort: 4 + Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered: item1: Sort: 1 @@ -27,6 +28,20 @@ Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered: - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrderableChild.item3 - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrderableChild.item4 +Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass: + item1: + Sort: 1 + item2: + Sort: 2 + item3: + Sort: 3 + item4: + Sort: 4 + item5: + Sort: 5 + item6: + Sort: 6 + Symbiote\GridFieldExtensions\Tests\Stub\StubSubclassOrderedVersioned: item1: ExtraField: 1 @@ -56,6 +71,29 @@ Symbiote\GridFieldExtensions\Tests\Stub\StubParent: ManyManySort: 108 - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered.item6: ManyManySort: 108 + MyManyManyVersioned: + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclassOrderedVersioned.item1: + ManyManySort: 1 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclassOrderedVersioned.item2: + ManyManySort: 1 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclassOrderedVersioned.item3: + ManyManySort: 108 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclassOrderedVersioned.item4: + ManyManySort: 108 + MyHasMany: + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered.item1 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered.item2 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered.item3 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered.item4 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered.item5 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubOrdered.item6 + MyHasManySubclass: + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass.item1 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass.item2 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass.item3 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass.item4 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass.item5 + - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclass.item6 parent-subclass-ordered-versioned: MyHasManySubclassOrderedVersioned: - =>Symbiote\GridFieldExtensions\Tests\Stub\StubSubclassOrderedVersioned.item1 diff --git a/tests/OrderableRowsThroughTest.yml b/tests/OrderableRowsThroughTest.yml index a0e11ea..7052d4d 100644 --- a/tests/OrderableRowsThroughTest.yml +++ b/tests/OrderableRowsThroughTest.yml @@ -28,3 +28,34 @@ Symbiote\GridFieldExtensions\Tests\Stub\ThroughIntermediary: Defining: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefiner.DefinerTwo Belonging: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongs.BelongsThree Sort: 2 + +Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefinerVersioned: + DefinerOne: + DefinerTwo: + +Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongsVersioned: + BelongsOne: + BelongsTwo: + BelongsThree: + +Symbiote\GridFieldExtensions\Tests\Stub\ThroughIntermediaryVersioned: + One: + Defining: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefinerVersioned.DefinerOne + Belonging: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongsVersioned.BelongsOne + Sort: 3 + Two: + Defining: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefinerVersioned.DefinerOne + Belonging: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongsVersioned.BelongsTwo + Sort: 2 + Three: + Defining: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefinerVersioned.DefinerOne + Belonging: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongsVersioned.BelongsThree + Sort: 1 + Four: + Defining: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefinerVersioned.DefinerTwo + Belonging: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongsVersioned.BelongsTwo + Sort: 1 + Five: + Defining: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughDefinerVersioned.DefinerTwo + Belonging: =>Symbiote\GridFieldExtensions\Tests\Stub\ThroughBelongsVersioned.BelongsThree + Sort: 2 diff --git a/tests/Stub/StubParent.php b/tests/Stub/StubParent.php index 8904318..e964eb0 100644 --- a/tests/Stub/StubParent.php +++ b/tests/Stub/StubParent.php @@ -15,10 +15,12 @@ class StubParent extends DataObject implements TestOnly private static $many_many = [ 'MyManyMany' => StubOrdered::class, + 'MyManyManyVersioned' => StubSubclassOrderedVersioned::class, ]; private static $many_many_extraFields = [ 'MyManyMany' => ['ManyManySort' => 'Int'], + 'MyManyManyVersioned' => ['ManyManySort' => 'Int'], ]; private static $table_name = 'StubParent'; diff --git a/tests/Stub/ThroughBelongsVersioned.php b/tests/Stub/ThroughBelongsVersioned.php new file mode 100644 index 0000000..d6f695b --- /dev/null +++ b/tests/Stub/ThroughBelongsVersioned.php @@ -0,0 +1,25 @@ + ThroughDefinerVersioned::class, + ]; + + private static array $extensions = [ + Versioned::class, + ]; +} diff --git a/tests/Stub/ThroughDefinerVersioned.php b/tests/Stub/ThroughDefinerVersioned.php new file mode 100644 index 0000000..9cb6af2 --- /dev/null +++ b/tests/Stub/ThroughDefinerVersioned.php @@ -0,0 +1,33 @@ + [ + 'through' => ThroughIntermediaryVersioned::class, + 'from' => 'Defining', + 'to' => 'Belonging', + ] + ]; + + private static array $owns = [ + 'Belongings' + ]; + + private static array $extensions = [ + Versioned::class, + ]; +} diff --git a/tests/Stub/ThroughIntermediaryVersioned.php b/tests/Stub/ThroughIntermediaryVersioned.php new file mode 100644 index 0000000..5ad3f14 --- /dev/null +++ b/tests/Stub/ThroughIntermediaryVersioned.php @@ -0,0 +1,32 @@ + 'Int', + ]; + + private static array $has_one = [ + 'Defining' => ThroughDefinerVersioned::class, + 'Belonging' => ThroughBelongsVersioned::class, + ]; + + private static array $extensions = [ + Versioned::class, + ]; +}