我在TYPO3 V6.2中有一个工作正常的Extbase扩展,用于存储产品。
现在,我想了解有关使用信号/插槽(Hooks的Extbase变体)的信息。我不知道为什么这个例子不起作用。当我在TYPO3后端的“列表”模块中更新产品时,它可以正确保存,但不会出现任何消息。
文件 typo3conf/ext/myext/ext_localconf.php
$signalSlotDispatcher = \TYPO3\CMS\Core\Utility\GeneralUtility::makeInstance( 'TYPO3\\CMS\\Extbase\\SignalSlot\\Dispatcher' );
$signalSlotDispatcher->connect(
'TYPO3\\CMS\\Extbase\\Persistence\\Generic\\Backend',
'afterUpdateObject',
'MyVendor\\MyExt\\Service\\Signalservice',
'myAfterUpdate',
FALSE
);
文件 typo3conf/ext/myext/Service/Signalservice.php
namespace MyVendor\MyExt\Service;
class Signalservice implements \TYPO3\CMS\Core\SingletonInterface {
/**
* @param \TYPO3\CMS\Extbase\DomainObject\DomainObjectInterface $object
*/
public function myAfterUpdate(\TYPO3\CMS\Extbase\DomainObject\DomainObjectInterface $object){
if ($object instanceof \MyVendor\MyExt\Domain\Model\Products) {
// check if we come to this point
\TYPO3\CMS\Extbase\Utility\DebuggerUtility::var_dump('Successfully hooked - I am a slot.');
die();
}
}
}
2015年6月15日更新
帕特里克·洛巴赫(Patrick Lobacher)暗示说,在这种情况下我们不能使用die()。相反,我们应该编写一个日志文件。但这对我也不起作用。未写入文件:
文件 typo3conf/ext/myext/ext_localconf.php
/**
* @var \TYPO3\CMS\Extbase\SignalSlot\Dispatcher $signalSlotDispatcher
* */
$signalSlotDispatcher = \TYPO3\CMS\Core\Utility\GeneralUtility::makeInstance('TYPO3\\CMS\\Extbase\\Object\\ObjectManager')->get('TYPO3\\CMS\\Extbase\\SignalSlot\\Dispatcher');
$signalSlotDispatcher->connect(
'TYPO3\\CMS\\Extbase\\Persistence\\Generic\\Backend',
'afterUpdateObject',
function ($payload) {
$logfile = "fileadmin/test/logfile.txt";
$handle = fopen($logfile, "a+");
fwrite ($handle, 'Hi. I was written by ext_localconf.php. ' . time());
fclose ($handle);
});
更新2015年6月29日
在https://forge.typo3.org/issues/61979上,Francois写道:“对象管理器只能在Extbase上下文中使用,而不能在ext_localconf.php中使用”。
但是,给出的答案甚至对我不起作用。但也许它可以帮助其他人。
最佳答案
当前没有官方文档,但是在此问题中,您可以找到非官方文档:https://forge.typo3.org/issues/59089
问题在于您正在列表模块中使用Extbase的Signal Slots。在6.2中,列表模块不是使用Extbase实现的。因此,没有可以使用的插槽。
相反,您需要使用Hooks遵循记录在案的旧方法:https://docs.typo3.org/typo3cms/CoreApiReference/ApiOverview/Hooks/Concept/Index.html
在您的情况下,以下代码应作为入口点:ext_localconf.php
:
$GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_tcemain.php']['processDatamapClass'][$_EXTKEY]
= 'Vendor\ExtName\Hook\DataMapHook';
在那里,您可以配置该类以用作TYPO3 6.2之前处理旧类名的
t3lib_tcemain
的钩子(Hook),而不仅仅是处理 ListView 的数据。在您的类内部,您可以实现已经由您完成的代码:
Classes/Hook/DataMapHook.php
:<?php
namespace Vendor\ExtName\Hook;
/**
* Hook to process updated records.
*
* @author Daniel Siepmann <d.siepmann@web-vision.de>
*/
class DataMapHook
{
/**
* Hook to add latitude and longitude to locations.
*
* @param string $action The action to perform, e.g. 'update'.
* @param string $table The table affected by action, e.g. 'fe_users'.
* @param int $uid The uid of the record affected by action.
* @param array $modifiedFields The modified fields of the record.
*
* @return void
*/
public function processDatamap_postProcessFieldArray(
$action, $table, $uid, array &$modifiedFields
) {
if(!$this->executeHook($table, $action)) {
return;
}
// check if we come to this point
\TYPO3\CMS\Extbase\Utility\DebuggerUtility::var_dump('Successfully hooked - I am a slot.');
die();
}
/**
* Check whether to execute hook or not.
*
* @param string $table
* @param string $action
* @param array $modifiedFields
*
* @return bool
*/
protected function executeHook($table, $action)
{
// Do not process if foreign table, unintended action,
// or fields were changed explicitly.
if ($table !== 'tx_extname_domain_model_modelname' || $action !== 'update') {
return false;
}
return false;
}
}
是的,您可以在这种情况下使用
die
进行调试等。就像TYPO3一样,它将遍历已配置的钩子(Hook)并调用方法。所以在这里没什么好看的。您将获得一些由实现定义的参数,并可以使用它们。
在上面的示例中,有一个检查仅在表和操作匹配时才执行 Hook 。由于有许多原因调用此代码,因此请确保将其列入白名单,以便仅在您了解的环境中执行。出于安全和性能方面的考虑。
关于signals-slots - TYPO3 Extbase-如何使用核心信号/插槽,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/30544422/