diff --git a/typo3/sysext/backend/Classes/Authentication/Event/SwitchUserEvent.php b/typo3/sysext/backend/Classes/Authentication/Event/SwitchUserEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..366c9075b06d07dd395aea4bb553f32f64a30ec1 --- /dev/null +++ b/typo3/sysext/backend/Classes/Authentication/Event/SwitchUserEvent.php @@ -0,0 +1,59 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Backend\Authentication\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +/** + * This event is triggered when a "SU" (switch user) action has been triggered + */ +final class SwitchUserEvent +{ + /** + * @var string + */ + private $sessionId; + + /** + * @var array + */ + private $targetUser; + + /** + * @var array + */ + private $currentUser; + + public function __construct(string $sessionId, array $targetUser, array $currentUser) + { + $this->sessionId = $sessionId; + $this->targetUser = $targetUser; + $this->currentUser = $currentUser; + } + + public function getSessionId(): string + { + return $this->sessionId; + } + + public function getTargetUser(): array + { + return $this->targetUser; + } + + public function getCurrentUser(): array + { + return $this->currentUser; + } +} diff --git a/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php index a0df516f0eb3ce77e76672845c9cd00a434e39b8..088766c86935501f1aaaa18055d646446cffc3b4 100644 --- a/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php +++ b/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php @@ -17,6 +17,13 @@ namespace TYPO3\CMS\Backend\Compatibility; use TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent; use TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem; +use TYPO3\CMS\Backend\Controller\EditDocumentController; +use TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent; +use TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider; +use TYPO3\CMS\Backend\Utility\BackendUtility; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; /** @@ -51,4 +58,43 @@ class SlotReplacement [$event->getToolbarItem()] ); } + + public function onLoginProviderGetPageRenderer(ModifyPageLayoutOnLoginProviderSelectionEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + UsernamePasswordLoginProvider::class, + 'getPageRenderer', + [$event->getPageRenderer()] + ); + } + + public function onPreInitEditDocumentController(BeforeFormEnginePageInitializedEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + EditDocumentController::class, + 'preInitAfter', + [$event->getController(), 'request' => $event->getRequest()] + ); + } + + public function onInitEditDocumentController(AfterFormEnginePageInitializedEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + EditDocumentController::class, + 'initAfter', + [$event->getController(), 'request' => $event->getRequest()] + ); + } + + public function emitGetPagesTSconfigPreIncludeSignalBackendUtility(ModifyLoadedPageTsConfigEvent $event): void + { + $rootLine = $event->getRootLine(); + $page = end($rootLine); + $signalArguments = $this->signalSlotDispatcher->dispatch( + BackendUtility::class, + 'getPagesTSconfigPreInclude', + [$event->getTsConfig(), (int)$page['uid'], $rootLine, false] + ); + $event->setTsConfig($signalArguments[0]); + } } diff --git a/typo3/sysext/backend/Classes/Controller/EditDocumentController.php b/typo3/sysext/backend/Classes/Controller/EditDocumentController.php index a4ec19da5ae1c52e6a2189e63cd34d8fb22e3a43..bea6ac250387dac83d52eb9ac863a581dfb22258 100644 --- a/typo3/sysext/backend/Classes/Controller/EditDocumentController.php +++ b/typo3/sysext/backend/Classes/Controller/EditDocumentController.php @@ -15,8 +15,11 @@ namespace TYPO3\CMS\Backend\Controller; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; +use TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent; use TYPO3\CMS\Backend\Form\Exception\AccessDeniedException; use TYPO3\CMS\Backend\Form\Exception\DatabaseRecordException; use TYPO3\CMS\Backend\Form\FormDataCompiler; @@ -51,7 +54,6 @@ use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Utility\HttpUtility; use TYPO3\CMS\Core\Utility\MathUtility; use TYPO3\CMS\Core\Utility\PathUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; /** * Main backend controller almost always used if some database record is edited in the backend. @@ -347,11 +349,6 @@ class EditDocumentController */ protected $dontStoreDocumentRef = 0; - /** - * @var \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected $signalSlotDispatcher; - /** * Stores information needed to preview the currently saved record * @@ -381,10 +378,13 @@ class EditDocumentController protected $isPageInFreeTranslationMode = false; /** - * Constructor + * @var EventDispatcherInterface */ - public function __construct() + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher; $this->moduleTemplate = GeneralUtility::makeInstance(ModuleTemplate::class); $this->moduleTemplate->setUiBlock(true); // @todo Used by TYPO3\CMS\Backend\Configuration\TypoScript\ConditionMatching @@ -489,8 +489,8 @@ class EditDocumentController $this->getBackendUser()->setTemporaryWorkspace($this->workspace); } - $this->emitFunctionAfterSignal('preInit', $request); - + $event = new BeforeFormEnginePageInitializedEvent($this, $request); + $this->eventDispatcher->dispatch($event); return null; } @@ -779,7 +779,8 @@ class EditDocumentController // Set context sensitive menu $this->moduleTemplate->getPageRenderer()->loadRequireJsModule('TYPO3/CMS/Backend/ContextMenu'); - $this->emitFunctionAfterSignal('init', $request); + $event = new AfterFormEnginePageInitializedEvent($this, $request); + $this->eventDispatcher->dispatch($event); } /** @@ -2452,30 +2453,6 @@ class EditDocumentController return new RedirectResponse($retUrl, 303); } - /** - * Emits a signal after a function was executed - * - * @param string $signalName - * @param ServerRequestInterface $request - */ - protected function emitFunctionAfterSignal($signalName, ServerRequestInterface $request): void - { - $this->getSignalSlotDispatcher()->dispatch(__CLASS__, $signalName . 'After', [$this, 'request' => $request]); - } - - /** - * Get the SignalSlot dispatcher - * - * @return \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected function getSignalSlotDispatcher() - { - if (!isset($this->signalSlotDispatcher)) { - $this->signalSlotDispatcher = GeneralUtility::makeInstance(Dispatcher::class); - } - return $this->signalSlotDispatcher; - } - /** * @return \TYPO3\CMS\Core\Authentication\BackendUserAuthentication */ diff --git a/typo3/sysext/backend/Classes/Controller/Event/AfterFormEnginePageInitializedEvent.php b/typo3/sysext/backend/Classes/Controller/Event/AfterFormEnginePageInitializedEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..af5f41f8689617c5b76663d68be1fff1dfb2fd78 --- /dev/null +++ b/typo3/sysext/backend/Classes/Controller/Event/AfterFormEnginePageInitializedEvent.php @@ -0,0 +1,51 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Backend\Controller\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use Psr\Http\Message\ServerRequestInterface; +use TYPO3\CMS\Backend\Controller\EditDocumentController; + +/** + * Event to listen to after the form engine has been initialized (= all data has been persisted) + */ +final class AfterFormEnginePageInitializedEvent +{ + /** + * @var EditDocumentController + */ + private $controller; + + /** + * @var ServerRequestInterface + */ + private $request; + + public function __construct(EditDocumentController $controller, ServerRequestInterface $request) + { + $this->controller = $controller; + $this->request = $request; + } + + public function getController(): EditDocumentController + { + return $this->controller; + } + + public function getRequest(): ServerRequestInterface + { + return $this->request; + } +} diff --git a/typo3/sysext/backend/Classes/Controller/Event/BeforeFormEnginePageInitializedEvent.php b/typo3/sysext/backend/Classes/Controller/Event/BeforeFormEnginePageInitializedEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..585fac7e505fa5fbb8e99ab49d49daa8814766d6 --- /dev/null +++ b/typo3/sysext/backend/Classes/Controller/Event/BeforeFormEnginePageInitializedEvent.php @@ -0,0 +1,51 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Backend\Controller\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use Psr\Http\Message\ServerRequestInterface; +use TYPO3\CMS\Backend\Controller\EditDocumentController; + +/** + * Event to listen to before the form engine has been initialized (= before all data will be persisted) + */ +final class BeforeFormEnginePageInitializedEvent +{ + /** + * @var EditDocumentController + */ + private $controller; + + /** + * @var ServerRequestInterface + */ + private $request; + + public function __construct(EditDocumentController $controller, ServerRequestInterface $request) + { + $this->controller = $controller; + $this->request = $request; + } + + public function getController(): EditDocumentController + { + return $this->controller; + } + + public function getRequest(): ServerRequestInterface + { + return $this->request; + } +} diff --git a/typo3/sysext/backend/Classes/Controller/LoginController.php b/typo3/sysext/backend/Classes/Controller/LoginController.php index 73f3d8919cf92c44199e4382e05053e761de785f..39566ec6b2b0502691097b4c339121ce6c223180 100644 --- a/typo3/sysext/backend/Classes/Controller/LoginController.php +++ b/typo3/sysext/backend/Classes/Controller/LoginController.php @@ -15,11 +15,13 @@ namespace TYPO3\CMS\Backend\Controller; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; use Psr\Log\LoggerAwareInterface; use Psr\Log\LoggerAwareTrait; use TYPO3\CMS\Backend\Exception; +use TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent; use TYPO3\CMS\Backend\LoginProvider\LoginProviderInterface; use TYPO3\CMS\Backend\Routing\UriBuilder; use TYPO3\CMS\Backend\Template\DocumentTemplate; @@ -99,13 +101,17 @@ class LoginController implements LoggerAwareInterface * @var DocumentTemplate */ protected $documentTemplate; + + protected $eventDispatcher; + /** - * @var \TYPO3\CMS\Core\Information\Typo3Copyright + * @var Typo3Copyright */ private $copyright; - public function __construct(Typo3Copyright $copyright) + public function __construct(Typo3Copyright $copyright, EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher; $this->copyright = $copyright; } @@ -302,6 +308,15 @@ class LoginController implements LoggerAwareInterface /** @var LoginProviderInterface $loginProvider */ $loginProvider = GeneralUtility::makeInstance($this->loginProviders[$this->loginProviderIdentifier]['provider']); + + $this->eventDispatcher->dispatch( + new ModifyPageLayoutOnLoginProviderSelectionEvent( + $this, + $this->view, + $pageRenderer + ) + ); + $loginProvider->render($this->view, $pageRenderer, $this); $content = $this->documentTemplate->startPage('TYPO3 CMS Login: ' . $GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']); diff --git a/typo3/sysext/backend/Classes/LoginProvider/Event/ModifyPageLayoutOnLoginProviderSelectionEvent.php b/typo3/sysext/backend/Classes/LoginProvider/Event/ModifyPageLayoutOnLoginProviderSelectionEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..3987f2568478ee096b0bffce8aa930f03e1926f1 --- /dev/null +++ b/typo3/sysext/backend/Classes/LoginProvider/Event/ModifyPageLayoutOnLoginProviderSelectionEvent.php @@ -0,0 +1,62 @@ +<?php +namespace TYPO3\CMS\Backend\LoginProvider\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Backend\Controller\LoginController; +use TYPO3\CMS\Core\Page\PageRenderer; +use TYPO3\CMS\Fluid\View\StandaloneView; + +/** + * Allows to modify variables for the view depending on a special login provider set in the controller. + */ +final class ModifyPageLayoutOnLoginProviderSelectionEvent +{ + /** + * @var LoginController + */ + private $controller; + + /** + * @var StandaloneView + */ + private $view; + + /** + * @var PageRenderer + */ + private $pageRenderer; + + public function __construct(LoginController $controller, StandaloneView $view, PageRenderer $pageRenderer) + { + $this->controller = $controller; + $this->view = $view; + $this->pageRenderer = $pageRenderer; + } + + public function getController(): LoginController + { + return $this->controller; + } + + public function getView(): StandaloneView + { + return $this->view; + } + + public function getPageRenderer(): PageRenderer + { + return $this->pageRenderer; + } +} diff --git a/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php b/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php index e649f2da2ba9af8b91d3e5a73d1a92c0077849dd..c965b392df00f3100f831574729e53e20a3b5f6b 100644 --- a/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php +++ b/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php @@ -17,8 +17,6 @@ namespace TYPO3\CMS\Backend\LoginProvider; use TYPO3\CMS\Backend\Controller\LoginController; use TYPO3\CMS\Core\Page\PageRenderer; use TYPO3\CMS\Core\Utility\GeneralUtility; -use TYPO3\CMS\Extbase\Object\ObjectManager; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; use TYPO3\CMS\Fluid\View\StandaloneView; /** @@ -27,8 +25,6 @@ use TYPO3\CMS\Fluid\View\StandaloneView; */ class UsernamePasswordLoginProvider implements LoginProviderInterface { - const SIGNAL_getPageRenderer = 'getPageRenderer'; - /** * @param StandaloneView $view * @param PageRenderer $pageRenderer @@ -37,8 +33,6 @@ class UsernamePasswordLoginProvider implements LoginProviderInterface */ public function render(StandaloneView $view, PageRenderer $pageRenderer, LoginController $loginController) { - GeneralUtility::makeInstance(ObjectManager::class)->get(Dispatcher::class)->dispatch(__CLASS__, self::SIGNAL_getPageRenderer, [$pageRenderer]); - $pageRenderer->loadRequireJsModule('TYPO3/CMS/Backend/UserPassLogin'); $view->setTemplatePathAndFilename(GeneralUtility::getFileAbsFileName('EXT:backend/Resources/Private/Templates/UserPassLoginForm.html')); diff --git a/typo3/sysext/backend/Classes/Utility/BackendUtility.php b/typo3/sysext/backend/Classes/Utility/BackendUtility.php index faccb2c1f97197b6a69ee0f293e62f317b862008..62b8b467706d0f46acfeae3dccc3108b80341d78 100644 --- a/typo3/sysext/backend/Classes/Utility/BackendUtility.php +++ b/typo3/sysext/backend/Classes/Utility/BackendUtility.php @@ -14,12 +14,14 @@ namespace TYPO3\CMS\Backend\Utility; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Log\LoggerInterface; use TYPO3\CMS\Backend\Configuration\TypoScript\ConditionMatching\ConditionMatcher; use TYPO3\CMS\Backend\Routing\UriBuilder; use TYPO3\CMS\Core\Authentication\BackendUserAuthentication; use TYPO3\CMS\Core\Cache\CacheManager; use TYPO3\CMS\Core\Cache\Frontend\FrontendInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader; use TYPO3\CMS\Core\Configuration\Parser\PageTsConfigParser; use TYPO3\CMS\Core\Core\Environment; @@ -762,10 +764,9 @@ class BackendUtility $tsDataArray['uid_' . $v['uid']] = $v['TSconfig']; } - $tsDataArray = static::emitGetPagesTSconfigPreIncludeSignal($tsDataArray, $id, $rootLine); - $tsDataArray = TypoScriptParser::checkIncludeLines_array($tsDataArray); - - return $tsDataArray; + $eventDispatcher = GeneralUtility::getContainer()->get(EventDispatcherInterface::class); + $event = $eventDispatcher->dispatch(new ModifyLoadedPageTsConfigEvent($tsDataArray, $rootLine)); + return TypoScriptParser::checkIncludeLines_array($event->getTsConfig()); } /******************************************* @@ -3942,37 +3943,6 @@ class BackendUtility return !empty($GLOBALS['TCA'][$table]['ctrl']['security']['ignoreRootLevelRestriction']); } - /** - * Get the SignalSlot dispatcher - * - * @return \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected static function getSignalSlotDispatcher() - { - return GeneralUtility::makeInstance(\TYPO3\CMS\Extbase\SignalSlot\Dispatcher::class); - } - - /** - * Emits signal to modify the page TSconfig before include - * - * @param array $TSdataArray Current TSconfig data array - Can be modified by slots! - * @param int $id Page ID we are handling - * @param array $rootLine Rootline array of page - * @return array Modified Data array - */ - protected static function emitGetPagesTSconfigPreIncludeSignal( - array $TSdataArray, - $id, - array $rootLine - ) { - $signalArguments = static::getSignalSlotDispatcher()->dispatch( - __CLASS__, - 'getPagesTSconfigPreInclude', - [$TSdataArray, $id, $rootLine, false] - ); - return $signalArguments[0]; - } - /** * @param string $table * @return Connection diff --git a/typo3/sysext/backend/Configuration/Services.yaml b/typo3/sysext/backend/Configuration/Services.yaml index b9a64911f0fa7bfc0d04040ab64740c7d1cc2a3e..90f55c8a6256c6e02bf315db1cb73af8ffde1d86 100644 --- a/typo3/sysext/backend/Configuration/Services.yaml +++ b/typo3/sysext/backend/Configuration/Services.yaml @@ -19,6 +19,9 @@ services: TYPO3\CMS\Backend\History\RecordHistoryRollback: public: true + TYPO3\CMS\Backend\Controller\EditDocumentController: + tags: ['backend.controller'] + TYPO3\CMS\Backend\Controller\LoginController: tags: ['backend.controller'] @@ -33,6 +36,22 @@ services: identifier: 'legacy-slot' method: 'onSystemInformationToolbarEvent' event: TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onLoginProviderGetPageRenderer' + event: TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'emitGetPagesTSconfigPreIncludeSignalBackendUtility' + event: TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onPreInitEditDocumentController' + event: TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onInitEditDocumentController' + event: TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent # Category security checks for backend users TYPO3\CMS\Backend\Security\CategoryPermissionsAspect: diff --git a/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php b/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php index 8630752d5967335d6f3beb804d4f7ea4879ec95b..6c3bf1d7c7bcd45ff7934af5693de2a29ada3de8 100644 --- a/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php +++ b/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php @@ -16,6 +16,7 @@ namespace TYPO3\CMS\Backend\Tests\Unit\Controller; */ use Prophecy\Argument; +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Backend\Controller\EditDocumentController; use TYPO3\CMS\Backend\Template\ModuleTemplate; use TYPO3\CMS\Core\Localization\LanguageService; @@ -52,10 +53,11 @@ class EditDocumentControllerTest extends UnitTestCase $GLOBALS['LANG'] = $this->prophesize(LanguageService::class)->reveal(); GeneralUtility::addInstance(ModuleTemplate::class, $moduleTemplate->reveal()); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); $mock = \Closure::bind(static function (EditDocumentController $editDocumentController) use (&$result, $typoScript) { return $editDocumentController->parseAdditionalGetParameters($result, $typoScript); }, null, EditDocumentController::class); - $mock(new EditDocumentController()); + $mock(new EditDocumentController($eventDispatcher->reveal())); self::assertSame($expectedParameters, $result); } diff --git a/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php b/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php index c11d1e2d5407047e595011ba0478ea354b0e1944..0e78b11f4d311f0baff365f7deb3061a739291f1 100644 --- a/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php +++ b/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php @@ -16,6 +16,7 @@ namespace TYPO3\CMS\Backend\Tests\Unit\Utility; use Prophecy\Argument; use Prophecy\Prophecy\ObjectProphecy; +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Backend\Configuration\TypoScript\ConditionMatching\ConditionMatcher; use TYPO3\CMS\Backend\Tests\Unit\Utility\Fixtures\LabelFromItemListMergedReturnsCorrectFieldsFixture; use TYPO3\CMS\Backend\Tests\Unit\Utility\Fixtures\ProcessedValueForGroupWithMultipleAllowedTablesFixture; @@ -24,6 +25,8 @@ use TYPO3\CMS\Backend\Tests\Unit\Utility\Fixtures\ProcessedValueForSelectWithMMR use TYPO3\CMS\Backend\Utility\BackendUtility; use TYPO3\CMS\Core\Cache\CacheManager; use TYPO3\CMS\Core\Cache\Frontend\FrontendInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; +use TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader; use TYPO3\CMS\Core\Configuration\Parser\PageTsConfigParser; use TYPO3\CMS\Core\Database\Connection; use TYPO3\CMS\Core\Database\ConnectionPool; @@ -283,7 +286,7 @@ class BackendUtilityTest extends UnitTestCase $relationHandlerInstance = $relationHandlerProphet->reveal(); $relationHandlerInstance->tableArray['sys_category'] = [1, 2]; - list($queryBuilderProphet, $connectionPoolProphet) = $this->mockDatabaseConnection('sys_category'); + [$queryBuilderProphet, $connectionPoolProphet] = $this->mockDatabaseConnection('sys_category'); $statementProphet = $this->prophesize(\Doctrine\DBAL\Driver\Statement::class); $statementProphet->fetch()->shouldBeCalled()->willReturn( [ @@ -1035,6 +1038,10 @@ class BackendUtilityTest extends UnitTestCase { $expected = ['called.' => ['config']]; $pageId = 13; + $eventDispatcherProphecy = $this->prophesize(EventDispatcherInterface::class); + $eventDispatcherProphecy->dispatch(Argument::any())->willReturn(new ModifyLoadedPageTsConfigEvent([], [])); + $loader = new PageTsConfigLoader($eventDispatcherProphecy->reveal()); + GeneralUtility::addInstance(PageTsConfigLoader::class, $loader); $parserProphecy = $this->prophesize(PageTsConfigParser::class); $parserProphecy->parse(Argument::cetera())->willReturn($expected); GeneralUtility::addInstance(PageTsConfigParser::class, $parserProphecy->reveal()); diff --git a/typo3/sysext/beuser/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/beuser/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 0000000000000000000000000000000000000000..54e864ea98655392bfbb70e249cae879d12fdd73 --- /dev/null +++ b/typo3/sysext/beuser/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,49 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Beuser\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent; +use TYPO3\CMS\Beuser\Controller\BackendUserController; +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; + +/** + * This class provides a replacement for all existing signals in EXT:beuser of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function onSwitchUser(SwitchUserEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + BackendUserController::class, + 'switchUser', + [$event->getTargetUser()] + ); + } +} diff --git a/typo3/sysext/beuser/Classes/Controller/BackendUserController.php b/typo3/sysext/beuser/Classes/Controller/BackendUserController.php index fc7a32e537efd081de989910c29df54f605ae638..61a4dab44c4dc50d373944d9db1ade1e60407e90 100644 --- a/typo3/sysext/beuser/Classes/Controller/BackendUserController.php +++ b/typo3/sysext/beuser/Classes/Controller/BackendUserController.php @@ -14,10 +14,13 @@ namespace TYPO3\CMS\Beuser\Controller; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; +use TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent; use TYPO3\CMS\Core\Authentication\BackendUserAuthentication; use TYPO3\CMS\Core\Session\Backend\SessionBackendInterface; use TYPO3\CMS\Core\Session\SessionManager; use TYPO3\CMS\Core\Utility\GeneralUtility; +use TYPO3\CMS\Core\Utility\HttpUtility; use TYPO3\CMS\Extbase\Mvc\Controller\ActionController; use TYPO3\CMS\Extbase\Mvc\View\ViewInterface; use TYPO3\CMS\Extbase\Utility\LocalizationUtility; @@ -58,6 +61,11 @@ class BackendUserController extends ActionController */ protected $backendUserSessionRepository; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + /** * @param \TYPO3\CMS\Beuser\Service\ModuleDataStorageService $moduleDataStorageService */ @@ -90,6 +98,11 @@ class BackendUserController extends ActionController $this->backendUserSessionRepository = $backendUserSessionRepository; } + public function injectEventDispatcher(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * Load and persist module data * @@ -279,10 +292,15 @@ class BackendUserController extends ActionController ] ); - $this->emitSwitchUserSignal($targetUser); + $event = new SwitchUserEvent( + $this->getBackendUserAuthentication()->getSessionId(), + $targetUser, + $this->getBackendUserAuthentication()->user + ); + $this->eventDispatcher->dispatch($event); $redirectUrl = 'index.php' . ($GLOBALS['TYPO3_CONF_VARS']['BE']['interfaces'] ? '' : '?commandLI=1'); - \TYPO3\CMS\Core\Utility\HttpUtility::redirect($redirectUrl); + HttpUtility::redirect($redirectUrl); } } @@ -313,16 +331,6 @@ class BackendUserController extends ActionController return $latestUserUids; } - /** - * Emit a signal when using the "switch to user" functionality - * - * @param array $targetUser - */ - protected function emitSwitchUserSignal(array $targetUser) - { - $this->signalSlotDispatcher->dispatch(__CLASS__, 'switchUser', [$targetUser]); - } - /** * @return BackendUserAuthentication */ diff --git a/typo3/sysext/beuser/Configuration/Services.yaml b/typo3/sysext/beuser/Configuration/Services.yaml index 2415fd433f1be46ab9d679a99d8fd4cd7ed2a0b8..7e181563c246eecda7f2a4703d3d4f3e403e47b7 100644 --- a/typo3/sysext/beuser/Configuration/Services.yaml +++ b/typo3/sysext/beuser/Configuration/Services.yaml @@ -6,3 +6,11 @@ services: TYPO3\CMS\Beuser\: resource: '../Classes/*' + + # Listener for old Signal Slots + TYPO3\CMS\Beuser\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'onSwitchUser' + event: TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent diff --git a/typo3/sysext/core/Classes/Configuration/Event/ModifyLoadedPageTsConfigEvent.php b/typo3/sysext/core/Classes/Configuration/Event/ModifyLoadedPageTsConfigEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..5989e205e58ecd06e067db3abc3e779a84097db3 --- /dev/null +++ b/typo3/sysext/core/Classes/Configuration/Event/ModifyLoadedPageTsConfigEvent.php @@ -0,0 +1,59 @@ +<?php +declare(strict_types = 1); + +namespace TYPO3\CMS\Core\Configuration\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +/** + * Extensions can modify pageTSConfig entries that can be overridden or added, based on the root line + */ +final class ModifyLoadedPageTsConfigEvent +{ + /** + * @var array + */ + private $tsConfig; + + /** + * @var array + */ + private $rootLine; + + public function __construct(array $tsConfig, array $rootLine) + { + $this->tsConfig = $tsConfig; + $this->rootLine = $rootLine; + } + + public function getTsConfig(): array + { + return $this->tsConfig; + } + + public function addTsConfig(string $tsConfig): void + { + $this->tsConfig[] = $tsConfig; + } + + public function setTsConfig(array $tsConfig): void + { + $this->tsConfig = $tsConfig; + } + + public function getRootLine(): array + { + return $this->rootLine; + } +} diff --git a/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php b/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php index ee5c631ad0951020c0a26690f4cb4de29d4d561d..6c88536044ccb4b871c0cf6eac5f6fb9929aee51 100644 --- a/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php +++ b/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php @@ -15,6 +15,8 @@ namespace TYPO3\CMS\Core\Configuration\Loader; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\TypoScript\Parser\TypoScriptParser; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Core\Utility\GeneralUtility; @@ -33,6 +35,16 @@ use TYPO3\CMS\Core\Utility\PathUtility; */ class PageTsConfigLoader { + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * Main method to get all PageTSconfig from the rootline including the defaultTSconfig settings. * @param array $rootLine @@ -87,7 +99,10 @@ class PageTsConfigLoader } $tsData['page_' . $page['uid']] = $page['TSconfig'] ?? ''; } + + $event = $this->eventDispatcher->dispatch(new ModifyLoadedPageTsConfigEvent($tsData, $rootLine)); + // Apply includes - return TypoScriptParser::checkIncludeLines_array($tsData); + return TypoScriptParser::checkIncludeLines_array($event->getTsConfig()); } } diff --git a/typo3/sysext/core/Configuration/Services.yaml b/typo3/sysext/core/Configuration/Services.yaml index dc518deb9ad1839c544baa54858950f7d7907853..fc6ed9326f698ed5b6bbcfedd7686511e839b28a 100644 --- a/typo3/sysext/core/Configuration/Services.yaml +++ b/typo3/sysext/core/Configuration/Services.yaml @@ -26,6 +26,9 @@ services: TYPO3\CMS\Core\Http\MiddlewareDispatcher: autoconfigure: false + TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader: + public: true + TYPO3\CMS\Core\Database\Schema\SqlReader: public: true diff --git a/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst b/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst index a44f9dcbe90eb93261081d9583d75ee9daa65db8..7b0c050bb1036b328f07cc4f8bfd85365312ac01 100644 --- a/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst +++ b/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst @@ -10,21 +10,37 @@ Description =========== The following Signal Slots have been replaced by new PSR-14 events -which are a 1:1 equivalent: +which can be used as 1:1 equivalents: -- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider::getPageRenderer` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::preInitAfter` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::initAfter` +- :php:`TYPO3\CMS\Backend\Utility\BackendUtility::getPagesTSconfigPreInclude` +- :php:`TYPO3\CMS\Beuser\Controller\BackendUserController::switchUser` - :php:`TYPO3\CMS\Core\Database\SoftReferenceIndex::setTypoLinkPartsElement` - :php:`TYPO3\CMS\Core\Database\ReferenceIndex::shouldExcludeTableFromReferenceIndex` +- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` - :php:`TYPO3\CMS\Core\Utility\ExtensionManagementUtility::tcaIsBeingBuilt` +- :php:`TYPO3\CMS\Impexp\Utility\ImportExportUtility::afterImportExportInitialisation` - :php:`TYPO3\CMS\Install\Service\SqlExpectedSchemaService::tablesDefinitionIsBeingBuilt` -- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Lang\Service\TranslationService::postProcessMirrorUrl` +- :php:`TYPO3\CMS\Linkvalidator\LinkAnalyzer::beforeAnalyzeRecord` +- :php:`TYPO3\CMS\Seo\Canonical\CanonicalGenerator::beforeGeneratingCanonical` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss` -In addition, the following public constant, marking a signal name, is deprecated: +In addition, the following public constants, marking a signal name, are deprecated: - :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::SIGNAL_PostProcessTreeData` - +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss` Impact ====== @@ -42,12 +58,25 @@ Migration Use the new PSR-14 alternatives: +- :php:`TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent` +- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent` - :php:`TYPO3\CMS\Core\Imaging\Event\ModifyIconForResourcePropertiesEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\IsTableExcludedFromReferenceIndexEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\AppendLinkHandlerElementsEvent` - :php:`TYPO3\CMS\Core\Configuration\Event\AfterTcaCompilationEvent` - :php:`TYPO3\CMS\Core\Database\Event\AlterTableDefinitionStatementsEvent` - :php:`TYPO3\CMS\Core\Tree\Event\ModifyTreeDataEvent` -- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent` +- :php:`TYPO3\CMS\Impexp\Event\BeforeImportEvent` +- :php:`TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent` +- :php:`TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent` +- :php:`TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent` +- :php:`TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent` .. index:: PHP-API, FullyScanned, ext:core diff --git a/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst b/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst index 078fd11018a2e15896f078586e4d887bad845514..a193a51ec01c83dc6a9018b4bb4c7c079d7088d6 100644 --- a/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst +++ b/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst @@ -13,24 +13,50 @@ PSR-14 EventDispatching allows for TYPO3 Extensions or PHP packages to extend TY The following new PSR-14 events have been introduced: +- :php:`TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent` +- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent` - :php:`TYPO3\CMS\Core\Imaging\Event\ModifyIconForResourcePropertiesEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\IsTableExcludedFromReferenceIndexEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\AppendLinkHandlerElementsEvent` - :php:`TYPO3\CMS\Core\Configuration\Event\AfterTcaCompilationEvent` - :php:`TYPO3\CMS\Core\Database\Event\AlterTableDefinitionStatementsEvent` - :php:`TYPO3\CMS\Core\Tree\Event\ModifyTreeDataEvent` -- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent` +- :php:`TYPO3\CMS\Impexp\Event\BeforeImportEvent` +- :php:`TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent` +- :php:`TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent` +- :php:`TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent` +- :php:`TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent` They replace the existing Extbase-based Signal Slots -- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider::getPageRenderer` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::preInitAfter` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::initAfter` +- :php:`TYPO3\CMS\Backend\Utility\BackendUtility::getPagesTSconfigPreInclude` +- :php:`TYPO3\CMS\Beuser\Controller\BackendUserController::switchUser` - :php:`TYPO3\CMS\Core\Database\SoftReferenceIndex::setTypoLinkPartsElement` - :php:`TYPO3\CMS\Core\Database\ReferenceIndex::shouldExcludeTableFromReferenceIndex` +- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` - :php:`TYPO3\CMS\Core\Utility\ExtensionManagementUtility::tcaIsBeingBuilt` +- :php:`TYPO3\CMS\Impexp\Utility\ImportExportUtility::afterImportExportInitialisation` - :php:`TYPO3\CMS\Install\Service\SqlExpectedSchemaService::tablesDefinitionIsBeingBuilt` -- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Lang\Service\TranslationService::postProcessMirrorUrl` +- :php:`TYPO3\CMS\Linkvalidator\LinkAnalyzer::beforeAnalyzeRecord` +- :php:`TYPO3\CMS\Seo\Canonical\CanonicalGenerator::beforeGeneratingCanonical` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss` Impact diff --git a/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php b/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php index 75e03915564a3a21eccae167fb3f79ff0d3afc7e..9083a654ddbecbdc052e8cd5fb8e466652ab823e 100644 --- a/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php +++ b/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php @@ -15,6 +15,9 @@ namespace TYPO3\CMS\Core\Tests\Unit\Configuration\Loader; * The TYPO3 project - inspiring people to share! */ +use Prophecy\Argument; +use Psr\EventDispatcher\EventDispatcherInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader; use TYPO3\TestingFramework\Core\Unit\UnitTestCase; @@ -29,7 +32,10 @@ class PageTsConfigLoaderTest extends UnitTestCase 'default' => $GLOBALS['TYPO3_CONF_VARS']['BE']['defaultPageTSconfig'] ]; $expectedString = implode('"\n[GLOBAL]\n"', $expected); - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); + $event = new ModifyLoadedPageTsConfigEvent($expected, []); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); $result = $subject->collect([]); self::assertSame($expected, $result); @@ -48,7 +54,10 @@ class PageTsConfigLoaderTest extends UnitTestCase 'page_27' => '', ]; $rootLine = [['uid' => 0, 'pid' => 0], ['uid' => 13, 'TSconfig' => 'waiting for = love'], ['uid' => 27, 'TSconfig' => '']]; - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $event = new ModifyLoadedPageTsConfigEvent($expected, $rootLine); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); $result = $subject->collect($rootLine); self::assertSame($expected, $result); } @@ -66,7 +75,10 @@ class PageTsConfigLoaderTest extends UnitTestCase 'page_27' => '', ]; $rootLine = [['uid' => 13, 'TSconfig' => 'waiting for = love', 'tsconfig_includes' => 'EXT:core/Tests/Unit/Configuration/Loader/Fixtures/included.typoscript'], ['uid' => 27, 'TSconfig' => '']]; - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $event = new ModifyLoadedPageTsConfigEvent($expected, $rootLine); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); $result = $subject->collect($rootLine); self::assertSame($expected, $result); } @@ -83,7 +95,10 @@ class PageTsConfigLoaderTest extends UnitTestCase ]; $expectedString = implode("\n[GLOBAL]\n", $expected); $rootLine = [['uid' => 13, 'TSconfig' => 'waiting for = love', 'tsconfig_includes' => 'EXT:core/Tests/Unit/Configuration/Loader/Fixtures/me_does_not_exist.typoscript'], ['uid' => 27, 'TSconfig' => '']]; - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $event = new ModifyLoadedPageTsConfigEvent($expected, $rootLine); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); $result = $subject->collect($rootLine); self::assertSame($expected, $result); diff --git a/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php b/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php index 1d239c58f40e46c6de1469f13d8de553241e397f..287347e97d407eca6e0b4265be253b445e8e626e 100644 --- a/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php +++ b/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php @@ -17,9 +17,18 @@ namespace TYPO3\CMS\Extbase\SignalSlot; */ use Psr\Log\LoggerInterface; +use TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent; use TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent; use TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem; +use TYPO3\CMS\Backend\Controller\EditDocumentController; +use TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent; +use TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider; +use TYPO3\CMS\Backend\Utility\BackendUtility; +use TYPO3\CMS\Beuser\Controller\BackendUserController; use TYPO3\CMS\Core\Configuration\Event\AfterTcaCompilationEvent; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\Database\Event\AlterTableDefinitionStatementsEvent; use TYPO3\CMS\Core\Database\ReferenceIndex; use TYPO3\CMS\Core\Database\SoftReferenceIndex; @@ -78,6 +87,18 @@ use TYPO3\CMS\Core\Tree\Event\ModifyTreeDataEvent; use TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Extbase\Object\ObjectManagerInterface; +use TYPO3\CMS\Impexp\Event\BeforeImportEvent; +use TYPO3\CMS\Impexp\Utility\ImportExportUtility; +use TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent; +use TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent; +use TYPO3\CMS\Linkvalidator\LinkAnalyzer; +use TYPO3\CMS\Seo\Canonical\CanonicalGenerator; +use TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent; +use TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent; +use TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent; +use TYPO3\CMS\Workspaces\Service\GridDataService; /** * A dispatcher which dispatches signals by calling its registered slot methods @@ -174,9 +195,40 @@ class Dispatcher implements \TYPO3\CMS\Core\SingletonInterface DatabaseTreeDataProvider::class => [ 'PostProcessTreeData' => ModifyTreeDataEvent::class, ], + BackendUserController::class => [ + 'switchUser' => SwitchUserEvent::class + ], + BackendUtility::class => [ + 'getPagesTSconfigPreInclude' => ModifyLoadedPageTsConfigEvent::class + ], + EditDocumentController::class => [ + 'preInitAfter' => BeforeFormEnginePageInitializedEvent::class, + 'initAfter' => AfterFormEnginePageInitializedEvent::class, + ], SystemInformationToolbarItem::class => [ 'getSystemInformation' => SystemInformationToolbarCollectorEvent::class, 'loadMessages' => SystemInformationToolbarCollectorEvent::class + ], + UsernamePasswordLoginProvider::class => [ + 'getPageRenderer' => ModifyPageLayoutOnLoginProviderSelectionEvent::class + ], + 'TYPO3\\CMS\\Lang\\Service\\TranslationService' => [ + 'postProcessMirrorUrl' => ModifyLanguagePackRemoteBaseUrlEvent::class + ], + LinkAnalyzer::class => [ + 'beforeAnalyzeRecord' => BeforeRecordIsAnalyzedEvent::class + ], + ImportExportUtility::class => [ + 'afterImportExportInitialisation' => BeforeImportEvent::class + ], + CanonicalGenerator::class => [ + 'beforeGeneratingCanonical' => ModifyUrlForCanonicalTagEvent::class + ], + GridDataService::class => [ + GridDataService::SIGNAL_GenerateDataArray_BeforeCaching => AfterCompiledCacheableDataForWorkspaceEvent::class, + GridDataService::SIGNAL_GenerateDataArray_PostProcesss => AfterDataGeneratedForWorkspaceEvent::class, + GridDataService::SIGNAL_GetDataArray_PostProcesss => GetVersionedDataEvent::class, + GridDataService::SIGNAL_SortDataArray_PostProcesss => SortVersionedDataEvent::class, ] ]; diff --git a/typo3/sysext/impexp/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/impexp/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 0000000000000000000000000000000000000000..04d9c80afe9ea1f9f4b233b296ef6f612dfcdfe6 --- /dev/null +++ b/typo3/sysext/impexp/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,49 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Impexp\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Impexp\Event\BeforeImportEvent; +use TYPO3\CMS\Impexp\Utility\ImportExportUtility; + +/** + * This class provides a replacement for all existing signals in EXT:impexp of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function emitAfterImportExportInitialisationSignal(BeforeImportEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + ImportExportUtility::class, + 'afterImportExportInitialisation', + [$event->getImport()] + ); + } +} diff --git a/typo3/sysext/impexp/Classes/Event/BeforeImportEvent.php b/typo3/sysext/impexp/Classes/Event/BeforeImportEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..608fd995fdf15a699080fb9eaf031ec94096008a --- /dev/null +++ b/typo3/sysext/impexp/Classes/Event/BeforeImportEvent.php @@ -0,0 +1,39 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Impexp\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Impexp\Import; + +/** + * This event is triggered when an import file is about to be imported + */ +final class BeforeImportEvent +{ + /** + * @var Import + */ + private $import; + + public function __construct(Import $import) + { + $this->import = $import; + } + + public function getImport(): Import + { + return $this->import; + } +} diff --git a/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php b/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php index 485e41b00628aa667010e853358110c3761850b6..864a2c6f3f9b2afd6dc789a94b1ec0428e0f62d3 100644 --- a/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php +++ b/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php @@ -14,9 +14,10 @@ namespace TYPO3\CMS\Impexp\Utility; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Core\Log\LogManager; use TYPO3\CMS\Core\Utility\GeneralUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; +use TYPO3\CMS\Impexp\Event\BeforeImportEvent; use TYPO3\CMS\Impexp\Import; /** @@ -31,6 +32,16 @@ class ImportExportUtility */ protected $import; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * @return Import|null */ @@ -59,7 +70,7 @@ class ImportExportUtility $this->import = GeneralUtility::makeInstance(Import::class); $this->import->init(); - $this->emitAfterImportExportInitialisationSignal($this->import); + $this->eventDispatcher->dispatch(new BeforeImportEvent($this->import)); $importResponse = 0; if ($file && @is_file($file)) { @@ -84,26 +95,4 @@ class ImportExportUtility } return $importResponse; } - - /** - * Get the SignalSlot dispatcher - * - * @return Dispatcher - */ - protected function getSignalSlotDispatcher() - { - return GeneralUtility::makeInstance(Dispatcher::class); - } - - /** - * Emits a signal after initialization - * - * @param Import $import - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotException - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotReturnException - */ - protected function emitAfterImportExportInitialisationSignal(Import $import) - { - $this->getSignalSlotDispatcher()->dispatch(__CLASS__, 'afterImportExportInitialisation', [$import]); - } } diff --git a/typo3/sysext/impexp/Configuration/Services.yaml b/typo3/sysext/impexp/Configuration/Services.yaml index 825fd786666ca6ba5921e7856d6b914941792b6d..4099126b0450823b4e91fee7841d40c4c8af4bb1 100644 --- a/typo3/sysext/impexp/Configuration/Services.yaml +++ b/typo3/sysext/impexp/Configuration/Services.yaml @@ -6,3 +6,14 @@ services: TYPO3\CMS\Impexp\: resource: '../Classes/*' + + TYPO3\CMS\Impexp\Utility\ImportExportUtility: + public: true + + # Listener for old Signal Slots + TYPO3\CMS\Impexp\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'emitAfterImportExportInitialisationSignal' + event: TYPO3\CMS\Impexp\Event\BeforeImportEvent diff --git a/typo3/sysext/install/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/install/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 0000000000000000000000000000000000000000..03c21602830a68fde94ddc41e1bd5b7a5b522baf --- /dev/null +++ b/typo3/sysext/install/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,54 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Install\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Core\Http\Uri; +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent; + +/** + * This class provides a replacement for all existing signals in EXT:install of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function onLanguagePackProcessMirrorUrl(ModifyLanguagePackRemoteBaseUrlEvent $event): void + { + $languagePackBaseUrl = (string)$event->getBaseUrl(); + $this->signalSlotDispatcher->dispatch( + 'TYPO3\\CMS\\Lang\\Service\\TranslationService', + 'postProcessMirrorUrl', + [ + 'extensionKey' => $event->getPackageKey(), + 'mirrorUrl' => &$languagePackBaseUrl, + ] + ); + $event->setBaseUrl(new Uri($languagePackBaseUrl)); + } +} diff --git a/typo3/sysext/install/Classes/Service/Event/ModifyLanguagePackRemoteBaseUrlEvent.php b/typo3/sysext/install/Classes/Service/Event/ModifyLanguagePackRemoteBaseUrlEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..64e3dd8794034d23c5bdde1f52f28972a8a2c421 --- /dev/null +++ b/typo3/sysext/install/Classes/Service/Event/ModifyLanguagePackRemoteBaseUrlEvent.php @@ -0,0 +1,55 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Install\Service\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use Psr\Http\Message\UriInterface; + +/** + * Event to modify the main URL of a language + */ +final class ModifyLanguagePackRemoteBaseUrlEvent +{ + /** + * @var UriInterface + */ + private $baseUrl; + + /** + * @var string + */ + private $packageKey; + + public function __construct(UriInterface $baseUrl, string $packageKey) + { + $this->baseUrl = $baseUrl; + $this->packageKey = $packageKey; + } + + public function getBaseUrl(): UriInterface + { + return $this->baseUrl; + } + + public function setBaseUrl(UriInterface $baseUrl): void + { + $this->baseUrl = $baseUrl; + } + + public function getPackageKey(): string + { + return $this->packageKey; + } +} diff --git a/typo3/sysext/install/Classes/Service/LanguagePackService.php b/typo3/sysext/install/Classes/Service/LanguagePackService.php index 480411ab0a575ede522d8bce04a3d5b53f3d2716..89c73fca93bd2a0c0392f1e50fa70614b7c61f53 100644 --- a/typo3/sysext/install/Classes/Service/LanguagePackService.php +++ b/typo3/sysext/install/Classes/Service/LanguagePackService.php @@ -15,16 +15,17 @@ namespace TYPO3\CMS\Install\Service; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Symfony\Component\Finder\Finder; use TYPO3\CMS\Core\Configuration\Features; use TYPO3\CMS\Core\Core\Environment; +use TYPO3\CMS\Core\Http\Uri; use TYPO3\CMS\Core\Localization\Locales; use TYPO3\CMS\Core\Package\PackageManager; use TYPO3\CMS\Core\Registry; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Utility\PathUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; /** * Service class handling language pack details @@ -44,11 +45,17 @@ class LanguagePackService */ protected $registry; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + private const DEFAULT_LANGUAGE_PACK_URL = 'https://typo3.org/fileadmin/ter/'; private const BETA_LANGUAGE_PACK_URL = 'https://beta-translation.typo3.org/fileadmin/ter/'; - public function __construct() + public function __construct(EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher ?? GeneralUtility::getContainer()->get(EventDispatcherInterface::class); $this->locales = GeneralUtility::makeInstance(Locales::class); $this->registry = GeneralUtility::makeInstance(Registry::class); } @@ -232,17 +239,9 @@ class LanguagePackService $languagePackBaseUrl = self::BETA_LANGUAGE_PACK_URL; } - // Allow to modify the base url on the fly by calling a signal - $signalSlotDispatcher = GeneralUtility::makeInstance(Dispatcher::class); - $signalSlotDispatcher->dispatch( - 'TYPO3\\CMS\\Lang\\Service\\TranslationService', - 'postProcessMirrorUrl', - [ - 'extensionKey' => $key, - 'mirrorUrl' => &$languagePackBaseUrl, - ] - ); - + // Allow to modify the base url on the fly + $event = $this->eventDispatcher->dispatch(new Event\ModifyLanguagePackRemoteBaseUrlEvent(new Uri($languagePackBaseUrl), $key)); + $languagePackBaseUrl = $event->getBaseUrl(); $path = ExtensionManagementUtility::extPath($key); $majorVersion = explode('.', TYPO3_branch)[0]; if (strpos($path, '/sysext/') !== false) { diff --git a/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php b/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php index c3b01d1381c925b61fe1fba3b0a6dd939ecc95ae..887b0689306f2e00bf777c98c1dde4756df12a16 100644 --- a/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php +++ b/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php @@ -357,5 +357,29 @@ return [ 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], ] ]; diff --git a/typo3/sysext/install/Configuration/Services.yaml b/typo3/sysext/install/Configuration/Services.yaml index 15f76349eb9f8ad387c25395a2ed65fe957e3d36..001682a48bb9f3b59964ff3c7858585d56932bec 100644 --- a/typo3/sysext/install/Configuration/Services.yaml +++ b/typo3/sysext/install/Configuration/Services.yaml @@ -8,3 +8,12 @@ services: identifier: 'install/show-latest-errors' method: 'appendMessage' event: TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent + TYPO3\CMS\Install\Compatibility\SlotReplacement: + autowire: true + autoconfigure: true + public: false + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'onLanguagePackProcessMirrorUrl' + event: TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent diff --git a/typo3/sysext/linkvalidator/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/linkvalidator/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 0000000000000000000000000000000000000000..9bf46d91cdcd8505cf526bd77c130045ea7a4e89 --- /dev/null +++ b/typo3/sysext/linkvalidator/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,57 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Linkvalidator\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent; +use TYPO3\CMS\Linkvalidator\LinkAnalyzer; + +/** + * This class provides a replacement for all existing signals in EXT:linkvalidator of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function beforeLinkAnalyzerRecordIsAnalyzed(BeforeRecordIsAnalyzedEvent $event): void + { + [$results, $record] = $this->signalSlotDispatcher->dispatch( + LinkAnalyzer::class, + 'beforeAnalyzeRecord', + [ + $event->getResults(), + $event->getRecord(), + $event->getTableName(), + $event->getFields(), + $event->getLinkAnalyzer() + ] + ); + $event->setRecord($record); + $event->setResults($results); + } +} diff --git a/typo3/sysext/linkvalidator/Classes/Event/BeforeRecordIsAnalyzedEvent.php b/typo3/sysext/linkvalidator/Classes/Event/BeforeRecordIsAnalyzedEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..52ce7443cbbe5a5dd333cbf524ef8d49f4c5df3f --- /dev/null +++ b/typo3/sysext/linkvalidator/Classes/Event/BeforeRecordIsAnalyzedEvent.php @@ -0,0 +1,94 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Linkvalidator\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Linkvalidator\LinkAnalyzer; + +/** + * Event that is fired to modify results (= add results) or modify the record before the linkanalyzer analyzes + * the record. + */ +final class BeforeRecordIsAnalyzedEvent +{ + /** + * @var string + */ + private $tableName; + + /** + * @var array + */ + private $record; + + /** + * @var array + */ + private $fields; + + /** + * @var array + */ + private $results; + + /** + * @var LinkAnalyzer + */ + private $linkAnalyzer; + + public function __construct(string $tableName, array $record, array $fields, LinkAnalyzer $linkAnalyzer, array $results) + { + $this->tableName = $tableName; + $this->record = $record; + $this->fields = $fields; + $this->linkAnalyzer = $linkAnalyzer; + $this->results = $results; + } + + public function getTableName(): string + { + return $this->tableName; + } + + public function getRecord(): array + { + return $this->record; + } + + public function setRecord(array $record): void + { + $this->record = $record; + } + + public function getFields(): array + { + return $this->fields; + } + + public function getResults(): array + { + return $this->results; + } + + public function setResults(array $results): void + { + $this->results = $results; + } + + public function getLinkAnalyzer(): LinkAnalyzer + { + return $this->linkAnalyzer; + } +} diff --git a/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php b/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php index c7a8006b721f0b03f3f5e8a67eaa5c636916a751..21da1eb79863acb9c7d0c47dba1636694078de08 100644 --- a/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php +++ b/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php @@ -14,6 +14,7 @@ namespace TYPO3\CMS\Linkvalidator; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Backend\Utility\BackendUtility; use TYPO3\CMS\Core\Database\Connection; use TYPO3\CMS\Core\Database\ConnectionPool; @@ -94,10 +95,13 @@ class LinkAnalyzer protected $tsConfig = []; /** - * Fill hookObjectsArr with different link types and possible XClasses. + * @var EventDispatcherInterface */ - public function __construct() + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher; $this->getLanguageService()->includeLLFile('EXT:linkvalidator/Resources/Private/Language/Module/locallang.xlf'); } @@ -268,7 +272,10 @@ class LinkAnalyzer */ public function analyzeRecord(array &$results, $table, array $fields, array $record) { - list($results, $record) = $this->emitBeforeAnalyzeRecordSignal($results, $record, $table, $fields); + $event = new Event\BeforeRecordIsAnalyzedEvent($table, $record, $fields, $this, $results); + $this->eventDispatcher->dispatch($event); + $results = $event->getResults(); + $record = $event->getRecord(); // Put together content of all relevant fields $haystack = ''; @@ -572,40 +579,6 @@ class LinkAnalyzer return false; } - /** - * Emits a signal before the record is analyzed - * - * @param array $results Array of broken links - * @param array $record Record to analyze - * @param string $table Table name of the record - * @param array $fields Array of fields to analyze - * @return array - */ - protected function emitBeforeAnalyzeRecordSignal($results, $record, $table, $fields) - { - return $this->getSignalSlotDispatcher()->dispatch( - self::class, - 'beforeAnalyzeRecord', - [$results, $record, $table, $fields, $this] - ); - } - - /** - * @return \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected function getSignalSlotDispatcher() - { - return $this->getObjectManager()->get(\TYPO3\CMS\Extbase\SignalSlot\Dispatcher::class); - } - - /** - * @return \TYPO3\CMS\Extbase\Object\ObjectManager - */ - protected function getObjectManager() - { - return GeneralUtility::makeInstance(\TYPO3\CMS\Extbase\Object\ObjectManager::class); - } - /** * @return LanguageService */ diff --git a/typo3/sysext/linkvalidator/Configuration/Services.yaml b/typo3/sysext/linkvalidator/Configuration/Services.yaml index 90fb8b53ff1473de906d47a351698350af391a22..ac4cecda0067cd7b987eef46d8225035e4fbb415 100644 --- a/typo3/sysext/linkvalidator/Configuration/Services.yaml +++ b/typo3/sysext/linkvalidator/Configuration/Services.yaml @@ -24,3 +24,10 @@ services: identifier: 'rte-check-link-to-file' event: TYPO3\CMS\Core\Html\Event\BrokenLinkAnalysisEvent method: 'checkFileLink' + + TYPO3\CMS\Linkvalidator\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'beforeLinkAnalyzerRecordIsAnalyzed' + event: TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent diff --git a/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php b/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php index e3cd3cafff80f895c897304d5ad24d71378fd532..032842db047be51d37428ee66883a640355c0c22 100644 --- a/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php +++ b/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php @@ -16,6 +16,7 @@ namespace TYPO3\CMS\Linkvalidator\Tests\Functional; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Core\Core\Bootstrap; use TYPO3\CMS\Linkvalidator\LinkAnalyzer; use TYPO3\TestingFramework\Core\Functional\FunctionalTestCase; @@ -34,7 +35,6 @@ class LinkAnalyzerTest extends FunctionalTestCase protected function setUp(): void { parent::setUp(); - Bootstrap::initializeLanguageObject(); } @@ -110,7 +110,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); @@ -172,7 +172,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); @@ -234,7 +234,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); @@ -295,7 +295,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); diff --git a/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php b/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php index 6b5b4ca7f1cbffe8dd9d67382b121ed26889c973..c87a1b7d8ff7aa8e2a9ceafe9a9b2d05642f2774 100644 --- a/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php +++ b/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php @@ -16,11 +16,12 @@ namespace TYPO3\CMS\Seo\Canonical; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Core\Domain\Repository\PageRepository; use TYPO3\CMS\Core\Utility\GeneralUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; use TYPO3\CMS\Frontend\Controller\TypoScriptFrontendController; use TYPO3\CMS\Frontend\Utility\CanonicalizationUtility; +use TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent; /** * Class to add the canonical tag to the page @@ -40,38 +41,21 @@ class CanonicalGenerator protected $pageRepository; /** - * @var Dispatcher + * @var EventDispatcherInterface */ - protected $signalSlotDispatcher; + protected $eventDispatcher; - /** - * CanonicalGenerator constructor - * - * @param TypoScriptFrontendController $typoScriptFrontendController - * @param Dispatcher $signalSlotDispatcher - */ - public function __construct(TypoScriptFrontendController $typoScriptFrontendController = null, Dispatcher $signalSlotDispatcher = null) + public function __construct(TypoScriptFrontendController $typoScriptFrontendController = null, EventDispatcherInterface $eventDispatcher = null) { - if ($typoScriptFrontendController === null) { - $typoScriptFrontendController = $this->getTypoScriptFrontendController(); - } - if ($signalSlotDispatcher === null) { - $signalSlotDispatcher = GeneralUtility::makeInstance(Dispatcher::class); - } - $this->typoScriptFrontendController = $typoScriptFrontendController; - $this->signalSlotDispatcher = $signalSlotDispatcher; + $this->eventDispatcher = $eventDispatcher ?? GeneralUtility::getContainer()->get(EventDispatcherInterface::class); + $this->typoScriptFrontendController = $typoScriptFrontendController ?? $this->getTypoScriptFrontendController(); $this->pageRepository = GeneralUtility::makeInstance(PageRepository::class); } - /** - * @return string - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotException - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotReturnException - */ public function generate(): string { - $href = ''; - $this->signalSlotDispatcher->dispatch(self::class, 'beforeGeneratingCanonical', [&$href]); + $event = $this->eventDispatcher->dispatch(new ModifyUrlForCanonicalTagEvent('')); + $href = $event->getUrl(); if (empty($href) && (int)$this->typoScriptFrontendController->page['no_index'] === 1) { return ''; diff --git a/typo3/sysext/seo/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/seo/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 0000000000000000000000000000000000000000..756b45fbdbe958bba82755735719cd9c04303784 --- /dev/null +++ b/typo3/sysext/seo/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,51 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Seo\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Seo\Canonical\CanonicalGenerator; +use TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent; + +/** + * This class provides a replacement for all existing signals in EXT:seo of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function beforeGeneratingCanonical(ModifyUrlForCanonicalTagEvent $event): void + { + $href = $event->getUrl(); + $this->signalSlotDispatcher->dispatch( + CanonicalGenerator::class, + 'beforeGeneratingCanonical', + [&$href] + ); + $event->setUrl($href); + } +} diff --git a/typo3/sysext/seo/Classes/Event/ModifyUrlForCanonicalTagEvent.php b/typo3/sysext/seo/Classes/Event/ModifyUrlForCanonicalTagEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..4c634e4c0c03779eaca20b79f9f23ede21eb1b0f --- /dev/null +++ b/typo3/sysext/seo/Classes/Event/ModifyUrlForCanonicalTagEvent.php @@ -0,0 +1,43 @@ +<?php +declare(strict_types = 1); + +namespace TYPO3\CMS\Seo\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +/** + * PSR-14 to alter (or empty) a canonical URL for the href="" attribute of a canonical URL. + */ +final class ModifyUrlForCanonicalTagEvent +{ + /** + * @var string + */ + private $url; + + public function __construct(string $url) + { + $this->url = $url; + } + + public function getUrl(): string + { + return $this->url; + } + + public function setUrl(string $url): void + { + $this->url = $url; + } +} diff --git a/typo3/sysext/seo/Configuration/Services.yaml b/typo3/sysext/seo/Configuration/Services.yaml index 484cb85e965146f65737a38f5f78d8ee0f38bb7e..79ba4065337e11296dfeb5d227005a88e625bb38 100644 --- a/typo3/sysext/seo/Configuration/Services.yaml +++ b/typo3/sysext/seo/Configuration/Services.yaml @@ -6,3 +6,10 @@ services: TYPO3\CMS\Seo\: resource: '../Classes/*' + + TYPO3\CMS\Seo\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'beforeGeneratingCanonical' + event: TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent diff --git a/typo3/sysext/workspaces/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/workspaces/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 0000000000000000000000000000000000000000..88be5cd7c7a2af0e9b970a7627d2f662b633ad2e --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,87 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent; +use TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent; +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * This class provides a replacement for all existing signals in EXT:workspaces of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function onGenerateDataArrayBeforeCaching(AfterCompiledCacheableDataForWorkspaceEvent $event): void + { + [$obj, $dataArray, $versions] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_GenerateDataArray_BeforeCaching, + [$event->getGridService(), $event->getData(), $event->getVersions()] + ); + $event->setData($dataArray); + $event->setVersions($versions); + } + + public function onGenerateDataArrayPostProcessing(AfterDataGeneratedForWorkspaceEvent $event): void + { + [$obj, $dataArray] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_GenerateDataArray_PostProcesss, + [$event->getGridService(), $event->getData(), $event->getVersions()] + ); + $event->setData($dataArray); + } + + public function onGetDataPostProcessing(GetVersionedDataEvent $event): void + { + [$obj, $dataArray, $start, $limit, $dataArrayPart] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_GetDataArray_PostProcesss, + [$event->getGridService(), $event->getData(), $event->getStart(), $event->getLimit(), $event->getDataArrayPart()] + ); + $event->setData($dataArray); + $event->setDataArrayPart($dataArrayPart); + } + + public function onSortDataPostProcessing(SortVersionedDataEvent $event): void + { + [$obj, $dataArray, $sortingColumn, $sortingDirection] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_SortDataArray_PostProcesss, + [$event->getGridService(), $event->getData(), $event->getSortColumn(), $event->getSortDirection()] + ); + $event->setData($dataArray); + $event->setSortColumn($sortingColumn); + $event->setSortDirection($sortingDirection); + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/AfterCompiledCacheableDataForWorkspaceEvent.php b/typo3/sysext/workspaces/Classes/Event/AfterCompiledCacheableDataForWorkspaceEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..db81d8605b9130593b4ff59cce47e12b9c4e10ef --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/AfterCompiledCacheableDataForWorkspaceEvent.php @@ -0,0 +1,71 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module to find all chacheable data of versions of a workspace. + */ +final class AfterCompiledCacheableDataForWorkspaceEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var array + */ + private $versions; + + public function __construct(GridDataService $gridService, array $data, array $versions) + { + $this->gridService = $gridService; + $this->data = $data; + $this->versions = $versions; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getVersions(): array + { + return $this->versions; + } + + public function setVersions(array $versions): void + { + $this->versions = $versions; + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/AfterDataGeneratedForWorkspaceEvent.php b/typo3/sysext/workspaces/Classes/Event/AfterDataGeneratedForWorkspaceEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..70ba704e6ca978db50e02fa7a6c92a4af649677f --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/AfterDataGeneratedForWorkspaceEvent.php @@ -0,0 +1,71 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module to find all data of versions of a workspace. + */ +final class AfterDataGeneratedForWorkspaceEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var array + */ + private $versions; + + public function __construct(GridDataService $gridService, array $data, array $versions) + { + $this->gridService = $gridService; + $this->data = $data; + $this->versions = $versions; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getVersions(): array + { + return $this->versions; + } + + public function setVersions(array $versions): void + { + $this->versions = $versions; + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/GetVersionedDataEvent.php b/typo3/sysext/workspaces/Classes/Event/GetVersionedDataEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..c9b5fc933793189af9c0d553dd37e9e78ce8a87a --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/GetVersionedDataEvent.php @@ -0,0 +1,95 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module to find all data of versions of a workspace. + * In comparison to AfterDataGeneratedForWorkspaceEvent, this one contains the + * cleaned / prepared data with an optional limit applied depending on the view. + */ +final class GetVersionedDataEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var array + */ + private $dataArrayPart; + + /** + * @var int + */ + private $start; + + /** + * @var int + */ + private $limit; + + public function __construct(GridDataService $gridService, array $data, int $start, int $limit, array $dataArrayPart) + { + $this->gridService = $gridService; + $this->data = $data; + $this->start = $start; + $this->limit = $limit; + $this->dataArrayPart = $dataArrayPart; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getDataArrayPart(): array + { + return $this->dataArrayPart; + } + + public function setDataArrayPart(array $dataArrayPart): void + { + $this->dataArrayPart = $dataArrayPart; + } + + public function getStart(): int + { + return $this->start; + } + + public function getLimit(): int + { + return $this->limit; + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/SortVersionedDataEvent.php b/typo3/sysext/workspaces/Classes/Event/SortVersionedDataEvent.php new file mode 100644 index 0000000000000000000000000000000000000000..b23dfb6c7713269e4dceb5336689fcf07363bca9 --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/SortVersionedDataEvent.php @@ -0,0 +1,87 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module after sorting all data for versions of a workspace. + */ +final class SortVersionedDataEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var string + */ + private $sortColumn; + + /** + * @var string + */ + private $sortDirection; + + public function __construct(GridDataService $gridService, array $data, string $sortColumn, string $sortDirection) + { + $this->gridService = $gridService; + $this->data = $data; + $this->sortColumn = $sortColumn; + $this->sortDirection = $sortDirection; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getSortColumn(): string + { + return $this->sortColumn; + } + + public function setSortColumn(string $sortColumn): void + { + $this->sortColumn = $sortColumn; + } + + public function getSortDirection(): string + { + return $this->sortDirection; + } + + public function setSortDirection(string $sortDirection): void + { + $this->sortDirection = $sortDirection; + } +} diff --git a/typo3/sysext/workspaces/Classes/Service/GridDataService.php b/typo3/sysext/workspaces/Classes/Service/GridDataService.php index cf22174f578622eac8db423b4f678764adcdabe1..ba7469136d375584a26e4a0f5d758ac41d4d55bc 100644 --- a/typo3/sysext/workspaces/Classes/Service/GridDataService.php +++ b/typo3/sysext/workspaces/Classes/Service/GridDataService.php @@ -14,6 +14,7 @@ namespace TYPO3\CMS\Workspaces\Service; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Log\LoggerAwareInterface; use Psr\Log\LoggerAwareTrait; use TYPO3\CMS\Backend\Configuration\TranslationConfigurationProvider; @@ -24,8 +25,11 @@ use TYPO3\CMS\Core\Imaging\IconFactory; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Versioning\VersionState; use TYPO3\CMS\Extbase\Object\ObjectManager; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; use TYPO3\CMS\Workspaces\Domain\Model\CombinedRecord; +use TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent; +use TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent; use TYPO3\CMS\Workspaces\Preview\PreviewUriBuilder; /** @@ -35,9 +39,21 @@ class GridDataService implements LoggerAwareInterface { use LoggerAwareTrait; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_GenerateDataArray_BeforeCaching = 'generateDataArray.beforeCaching'; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_GenerateDataArray_PostProcesss = 'generateDataArray.postProcess'; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_GetDataArray_PostProcesss = 'getDataArray.postProcess'; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_SortDataArray_PostProcesss = 'sortDataArray.postProcess'; const GridColumn_Collection = 'Workspaces_Collection'; @@ -84,6 +100,16 @@ class GridDataService implements LoggerAwareInterface */ protected $integrityService; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * Generates grid list array from given versions. * @@ -214,9 +240,12 @@ class GridDataService implements LoggerAwareInterface } } } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, array $versions) - list($this->dataArray, $versions) = $this->emitSignal(self::SIGNAL_GenerateDataArray_BeforeCaching, $this->dataArray, $versions); + + // Trigger a PSR-14 event + $event = new AfterCompiledCacheableDataForWorkspaceEvent($this, $this->dataArray, $versions); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); + $versions = $event->getVersions(); // Enrich elements after everything has been processed: foreach ($this->dataArray as &$element) { $identifier = $element['table'] . ':' . $element['t3ver_oid']; @@ -227,9 +256,11 @@ class GridDataService implements LoggerAwareInterface } $this->setDataArrayIntoCache($versions, $filterTxt); } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, array $versions) - list($this->dataArray) = $this->emitSignal(self::SIGNAL_GenerateDataArray_PostProcesss, $this->dataArray, $versions); + + // Trigger a PSR-14 event + $event = new AfterDataGeneratedForWorkspaceEvent($this, $this->dataArray, $versions); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); $this->sortDataArray(); $this->resolveDataArrayDependencies(); } @@ -277,9 +308,11 @@ class GridDataService implements LoggerAwareInterface } } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, $start, $limit, array $dataArrayPart) - list($this->dataArray, $start, $limit, $dataArrayPart) = $this->emitSignal(self::SIGNAL_GetDataArray_PostProcesss, $this->dataArray, $start, $limit, $dataArrayPart); + // Trigger a PSR-14 event + $event = new GetVersionedDataEvent($this, $this->dataArray, $start, $limit, $dataArrayPart); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); + return $event->getDataArrayPart(); return $dataArrayPart; } @@ -375,9 +408,12 @@ class GridDataService implements LoggerAwareInterface } else { $this->logger->critical('Try to sort "' . $this->sort . '" in "\\TYPO3\\CMS\\Workspaces\\Service\\GridDataService::sortDataArray" but $this->dataArray is empty! This might be the bug #26422 which could not be reproduced yet.'); } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, $sortColumn, $sortDirection) - list($this->dataArray, $this->sort, $this->sortDir) = $this->emitSignal(self::SIGNAL_SortDataArray_PostProcesss, $this->dataArray, $this->sort, $this->sortDir); + // Trigger an event for extensibility + $event = new SortVersionedDataEvent($this, $this->dataArray, $this->sort, $this->sortDir); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); + $this->sort = $event->getSortColumn(); + $this->sortDir = $event->getSortDirection(); } /** @@ -623,22 +659,6 @@ class GridDataService implements LoggerAwareInterface return $this->integrityService; } - /** - * Emits a signal to be handled by any registered slots. - * - * @param string $signalName Name of the signal - * @param array|mixed[] $arguments - * @return array - */ - protected function emitSignal($signalName, ...$arguments) - { - // Arguments are always ($this, [method argument], [method argument], ...) - $signalArguments = $arguments; - array_unshift($signalArguments, $this); - $slotReturn = $this->getSignalSlotDispatcher()->dispatch(GridDataService::class, $signalName, $signalArguments); - return array_slice($slotReturn, 1); - } - /** * @return Dependency\CollectionService */ @@ -655,14 +675,6 @@ class GridDataService implements LoggerAwareInterface return $this->getObjectManager()->get(AdditionalColumnService::class); } - /** - * @return Dispatcher - */ - protected function getSignalSlotDispatcher() - { - return $this->getObjectManager()->get(Dispatcher::class); - } - /** * @return ObjectManager */ diff --git a/typo3/sysext/workspaces/Configuration/Services.yaml b/typo3/sysext/workspaces/Configuration/Services.yaml index a2a64f288aa873c0f0e00f1a78709de981656266..9c62f633de7cbe600cf03e5cf8d68873a5ffc1c8 100644 --- a/typo3/sysext/workspaces/Configuration/Services.yaml +++ b/typo3/sysext/workspaces/Configuration/Services.yaml @@ -11,3 +11,25 @@ services: class: TYPO3\CMS\Core\Cache\Frontend\FrontendInterface factory: ['@TYPO3\CMS\Core\Cache\CacheManager', 'getCache'] arguments: ['workspaces_cache'] + + TYPO3\CMS\Workspaces\Service\GridDataService: + public: true + + TYPO3\CMS\Workspaces\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'onGenerateDataArrayBeforeCaching' + event: TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onGenerateDataArrayPostProcessing' + event: TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onGetDataPostProcessing' + event: TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onSortDataPostProcessing' + event: TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent diff --git a/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php b/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php index bb1d88cfa0aad9eb4b25126bc8b95d2cb64ad850..3dfb34e9a38bf5f56f5248805c7ba32348280dae 100644 --- a/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php +++ b/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php @@ -112,7 +112,7 @@ class RemoteServerTest extends UnitTestCase $liveFileReferences = $this->getFileReferenceProphecies($fileFileReferenceList); $versionFileReferences = $this->getFileReferenceProphecies($versionFileReferenceList); - $subject = $this->getAccessibleMock(RemoteServer::class, ['__none']); + $subject = $this->getAccessibleMock(RemoteServer::class, ['__none'], [], '', false); $result = $subject->_call( 'prepareFileReferenceDifferences', $liveFileReferences,