From 713b256a90bfb6f82812f17b2421c530ec0f76d8 Mon Sep 17 00:00:00 2001 From: Sabina Talipova Date: Thu, 11 Jan 2024 15:12:17 +1300 Subject: [PATCH] ENH Add sorting by link type --- lang/en.yml | 9 ++++ src/Form/Traits/AllowedLinkClassesTrait.php | 7 ++- src/Models/EmailLink.php | 14 ++++++ src/Models/ExternalLink.php | 14 ++++++ src/Models/FileLink.php | 14 ++++++ src/Models/Link.php | 16 ++++++ src/Models/PhoneLink.php | 14 ++++++ src/Models/SiteTreeLink.php | 14 ++++++ .../Traits/AllowedLinkClassesTraitTest.php | 49 +++++++++++++++++++ 9 files changed, 149 insertions(+), 2 deletions(-) diff --git a/lang/en.yml b/lang/en.yml index abbba2b1..3e99bb1f 100644 --- a/lang/en.yml +++ b/lang/en.yml @@ -37,8 +37,17 @@ en: MISSING_DEFAULT_TITLE: 'No link provided' SilverStripe\LinkField\Models\SiteTreeLink: MISSING_DEFAULT_TITLE: 'Page missing' + LINKLABEL: 'Page on this site' SilverStripe\LinkField\Models\FileLink: MISSING_DEFAULT_TITLE: 'File missing' + LINKLABEL: 'Link to a file' + SilverStripe\LinkField\Models\EmailLink: + LINKLABEL: 'Link to email address' + SilverStripe\LinkField\Models\ExternalLink: + LINKLABEL: 'Link to external URL' + SilverStripe\LinkField\Models\PhoneLink: + LINKLABEL: 'Phone number' + SilverStripe\LinkField\Form\Traits\AllowedLinkClassesTrait: INVALID_TYPECLASS: '"{class}": {typeclass} is not a valid Link Type' INVALID_TYPECLASS_EMPTY: '"{class}": Allowed types cannot be empty' diff --git a/src/Form/Traits/AllowedLinkClassesTrait.php b/src/Form/Traits/AllowedLinkClassesTrait.php index d21fef27..3007eced 100644 --- a/src/Form/Traits/AllowedLinkClassesTrait.php +++ b/src/Form/Traits/AllowedLinkClassesTrait.php @@ -93,10 +93,14 @@ public function getTypesProps(): string } $typesList[$key] = [ 'key' => $key, - 'title' => $type->i18n_singular_name(), + 'title' => $type->getMenuTitle(), 'handlerName' => $type->LinkTypeHandlerName(), + 'priority' => $class::config()->get('menu_priority'), ]; } + uasort($typesList, function ($a, $b) { + return $a['priority'] - $b['priority']; + }); return json_encode($typesList); } @@ -121,7 +125,6 @@ private function genarateAllowedTypes(): array $result[$type] = $class; } } - return $result; } } diff --git a/src/Models/EmailLink.php b/src/Models/EmailLink.php index 2410af4a..8abf5ff0 100644 --- a/src/Models/EmailLink.php +++ b/src/Models/EmailLink.php @@ -18,6 +18,11 @@ class EmailLink extends Link 'Email' => 'Varchar(255)', ]; + /** + * Set the priority of this link type in the CMS menu + */ + private static int $menu_priority = 30; + public function getCMSFields(): FieldList { $this->beforeUpdateCMSFields(function (FieldList $fields) { @@ -38,4 +43,13 @@ public function getURL(): string { return $this->Email ? sprintf('mailto:%s', $this->Email) : ''; } + + /** + * The title that will be displayed in the dropdown + * for selecting the link type to create. + */ + public function getMenuTitle(): string + { + return _t(__CLASS__ . '.LINKLABEL', 'Link to email address'); + } } diff --git a/src/Models/ExternalLink.php b/src/Models/ExternalLink.php index 27c167db..1b08336f 100644 --- a/src/Models/ExternalLink.php +++ b/src/Models/ExternalLink.php @@ -17,6 +17,11 @@ class ExternalLink extends Link 'ExternalUrl' => 'Varchar', ]; + /** + * Set the priority of this link type in the CMS menu + */ + private static int $menu_priority = 20; + public function getCMSFields(): FieldList { $this->beforeUpdateCMSFields(function (FieldList $fields) { @@ -35,4 +40,13 @@ public function getURL(): string { return $this->ExternalUrl ?: ''; } + + /** + * The title that will be displayed in the dropdown + * for selecting the link type to create. + */ + public function getMenuTitle(): string + { + return _t(__CLASS__ . '.LINKLABEL', 'Link to external URL'); + } } diff --git a/src/Models/FileLink.php b/src/Models/FileLink.php index 0d118b8e..101cfa82 100644 --- a/src/Models/FileLink.php +++ b/src/Models/FileLink.php @@ -19,6 +19,11 @@ class FileLink extends Link 'File' => File::class, ]; + /** + * Set the priority of this link type in the CMS menu + */ + private static int $menu_priority = 10; + public function getCMSFields(): FieldList { $this->beforeUpdateCMSFields(function (FieldList $fields) { @@ -55,4 +60,13 @@ public function getDefaultTitle(): string return (string) $this->getDescription(); } + + /** + * The title that will be displayed in the dropdown + * for selecting the link type to create. + */ + public function getMenuTitle(): string + { + return _t(__CLASS__ . '.LINKLABEL', 'Link to a file'); + } } diff --git a/src/Models/Link.php b/src/Models/Link.php index 2beea0b2..8990b3c6 100644 --- a/src/Models/Link.php +++ b/src/Models/Link.php @@ -57,6 +57,11 @@ class Link extends DataObject */ private ?string $linkType = null; + /** + * Set the priority of this link type in the CMS menu + */ + private static int $menu_priority = 100; + public function getDescription(): string { return ''; @@ -454,4 +459,15 @@ public function getShortCode(): string { return strtolower(rtrim(ClassInfo::shortName($this), 'Link')) ?? ''; } + + /** + * The title that will be displayed in the dropdown + * for selecting the link type to create. + * Subclasses should override this. + * It will use the singular_name by default. + */ + public function getMenuTitle(): string + { + return $this->i18n_singular_name(); + } } diff --git a/src/Models/PhoneLink.php b/src/Models/PhoneLink.php index 63fdf45d..eee764f9 100644 --- a/src/Models/PhoneLink.php +++ b/src/Models/PhoneLink.php @@ -15,6 +15,11 @@ class PhoneLink extends Link 'Phone' => 'Varchar(255)', ]; + /** + * Set the priority of this link type in the CMS menu + */ + private static int $menu_priority = 40; + public function getCMSFields(): FieldList { $this->beforeUpdateCMSFields(function (FieldList $fields) { @@ -33,4 +38,13 @@ public function getURL(): string { return $this->Phone ? sprintf('tel:%s', $this->Phone) : ''; } + + /** + * The title that will be displayed in the dropdown + * for selecting the link type to create. + */ + public function getMenuTitle(): string + { + return _t(__CLASS__ . '.LINKLABEL', 'Phone number'); + } } diff --git a/src/Models/SiteTreeLink.php b/src/Models/SiteTreeLink.php index 95dce8b1..e8a0900d 100644 --- a/src/Models/SiteTreeLink.php +++ b/src/Models/SiteTreeLink.php @@ -30,6 +30,11 @@ class SiteTreeLink extends Link 'Page' => SiteTree::class, ]; + /** + * Set the priority of this link type in the CMS menu + */ + private static int $menu_priority = 0; + public function getDescription(): string { $page = $this->Page(); @@ -131,4 +136,13 @@ public function getDefaultTitle(): string } return $page->Title; } + + /** + * The title that will be displayed in the dropdown + * for selecting the link type to create. + */ + public function getMenuTitle(): string + { + return _t(__CLASS__ . '.LINKLABEL', 'Page on this site'); + } } diff --git a/tests/php/Traits/AllowedLinkClassesTraitTest.php b/tests/php/Traits/AllowedLinkClassesTraitTest.php index abffd775..bb7f2fcc 100644 --- a/tests/php/Traits/AllowedLinkClassesTraitTest.php +++ b/tests/php/Traits/AllowedLinkClassesTraitTest.php @@ -5,6 +5,7 @@ use ArrayIterator; use InvalidArgumentException; use ReflectionMethod; +use SilverStripe\Core\Injector\Injector; use SilverStripe\Dev\SapphireTest; use SilverStripe\LinkField\Form\Traits\AllowedLinkClassesTrait; use SilverStripe\LinkField\Form\LinkField; @@ -105,6 +106,54 @@ public function provideTypesExceptionDataProvider() : array ]; } + public function sortedTypesDataProvider() : array + { + return [ + 'sort all allowed Link classes' => [ + 'enabled' => [ + SiteTreeLink::class, + ExternalLink::class, + FileLink::class, + EmailLink::class, + PhoneLink::class, + TestPhoneLink::class, + ], + 'expected' => [ + 'sitetree', + 'testphone', + 'file', + 'external', + 'email', + 'phone', + ], + ], + 'sort only particular allowed Link class' => [ + 'enabled' => [ + SiteTreeLink::class, + TestPhoneLink::class, + EmailLink::class, + ], + 'expected' => [ + 'sitetree', + 'testphone', + 'email', + ], + ], + ]; + } + + /** + * @dataProvider sortedTypesDataProvider + */ + public function testGetTypesProps(array $enabled, array $expected): void + { + Injector::inst()->get(TestPhoneLink::class)->config()->set('menu_priority', 5); + $linkField = LinkField::create('LinkField'); + $linkField->setAllowedTypes($enabled); + $json = json_decode($linkField->getTypesProps(), true); + $this->assertEquals(array_keys($json), $expected); + } + public function testGetTypesPropsCanCreate(): void { $linkField = LinkField::create('LinkField');