我使用包含一些 1:1 和 1:n 关系的 extbase 扩展构建器“启动”了一个扩展。它会自动将字段类型设置为“内联”并在后端显示一个漂亮的 IRRE UI。
但默认情况下,无法选择现有记录,只能创建新记录。
我找到了关于如何使用“foreign_selector”实现这一目标的各种解释,但它们都非常粗略。该功能本身应该可以正常工作,请参阅 https://forge.typo3.org/issues/43239
有人可以引导我完成这个或指向 TER 中的一个工作示例吗?一旦我开始使用它,我就可以根据示例创建一个分步教程。
PS 该字段的 TCA 配置由 extension_builder
生成:
'myfield' => array(
'exclude' => 1,
'label' => 'LLL:EXT:myextension/Resources/Private/Language/locallang_db.xlf:tx_myextension_domain_model_myitem.myfield',
'config' => array(
'type' => 'inline',
'foreign_table' => 'tx_myextension_domain_model_myfield',
'foreign_field' => 'myitem',
'maxitems' => 9999,
'appearance' => array(
'collapseAll' => 0,
'levelLinksPosition' => 'top',
'showSynchronizationLink' => 1,
'showPossibleLocalizationRecords' => 1,
'showAllLocalizationLink' => 1
),
),
),
最佳答案
主要问题是 1:n 类型的 IRRE 关系是这样工作的:子记录保存其父记录的 uid。所以你的表 tx_myext_domain_model_city 保存了你的(虚构的)tx_myext_domain_model_address 的 UID。
因此,使用默认配置,您将无法多次选择一个城市,因为它只能有一个父级。
因此,您需要为此字段使用关系表。此表需要包含地址 (uid_address) 和城市 (uid_city) 的 uid 字段:
CREATE TABLE tx_irreforeignselectordemo_address_city_mm (
uid int(11) NOT NULL auto_increment,
pid int(11) DEFAULT '0' NOT NULL,
uid_address int(11) unsigned DEFAULT '0' NOT NULL,
uid_city int(11) unsigned DEFAULT '0' NOT NULL,
sorting int(11) unsigned DEFAULT '0' NOT NULL,
PRIMARY KEY (uid),
KEY parent (pid)
);
并且它需要对这些字段进行 TCA 配置(而表本身可以隐藏):
return array(
'ctrl' => array(
'title' => 'Relation table',
'hideTable' => TRUE,
'sortby' => 'sorting',
),
'columns' => array(
'uid_address' => Array(
'label' => 'Address',
'config' => Array(
'type' => 'select',
'foreign_table' => 'tx_irreforeignselectordemo_domain_model_address',
'size' => 1,
'minitems' => 0,
'maxitems' => 1,
),
),
'uid_city' => Array(
'label' => 'City',
'config' => Array(
'type' => 'select',
'foreign_table' => 'tx_irreforeignselectordemo_domain_model_city',
'foreign_table_where' => ' AND sys_language_uid IN (0,-1)',
'size' => 1,
'minitems' => 0,
'maxitems' => 1,
),
),
),
'types' => array(
'0' => array('showitem' => 'uid_address,uid_city')
),
'palettes' => array()
);
然后,您可以配置地址的 TCA 以使其成为 IRRE 字段:
'type' => 'inline',
'foreign_table' => 'tx_yourext_address_city_mm',
'foreign_field' => 'uid_address',
'foreign_label' => 'uid_city',
'foreign_selector' => 'uid_city',
'foreign_unique' => 'uid_city',
'foreign_sortby' => 'sorting',
请注意,
foreign_unique
告诉 TYPO3 一个城市只能选择一次。并且您需要从另一端(来自您所在城市的 TCA)定义关系:
'addresses' => array(
'exclude' => 1,
'label' => 'Addresses',
'config' => array(
'type' => 'inline',
'foreign_table' => 'tx_irreforeignselectordemo_address_city_mm',
'foreign_field' => 'uid_city',
'foreign_label' => 'uid_address',
),
),
配置完成后,您将能够在后端使用它。
由于这是一个非标准的MM关系,默认情况下Extbase将无法处理它。但是我们可以将其与 TYPO3 6 中引入的 sys_file_reference 表进行比较。 因此,我们为 CityRelation 构建了一个具有“address”和“city”属性的 Extbase 模型,并将此模型映射到我们的 mm 表:
config.tx_extbase.persistence.classes {
Visol\Irreforeignselectordemo\Domain\Model\CityRelation {
mapping {
tableName = tx_irreforeignselectordemo_address_city_mm
columns {
uid_address.mapOnProperty = address
uid_city.mapOnProperty = city
}
}
}
}
现在在我们的地址模型中,我们将城市(或多个城市 - 允许多个选择)定义为 CityRelation 类型的 ObjectStorage:
/**
* Cities
*
* @var \TYPO3\CMS\Extbase\Persistence\ObjectStorage<\Visol\Irreforeignselectordemo\Domain\Model\CityRelation>
*/
protected $cities = NULL;
我们现在有一个属性“城市”,其中包含对所有选定城市的引用。您可以遍历它们并使用它们:
<f:for each="{address.cities}" as="cityRelation">
<li>{cityRelation.city.name}</li>
</f:for>
由于我找不到针对此的多合一演示并且对该主题感兴趣,因此我创建了一个演示扩展来执行我刚刚描述的操作 - 基于 Core 和处理该主题的两个扩展:https://github.com/lorenzulrich/irreforeignselectordemo
无论如何,解决方案是 m:n 方法(因为 1:n 由于上述原因不起作用)所以我决定使用“城市”而不是“城市”。虽然这对于选择城市可能没有意义(如您的帖子所建议的那样),但对于其他机会可能有意义。随意将“城市”替换为“城市”,并将内联配置中的 maxItems 设置为 1 - 那么您就有了一种 1:n。
关于TYPO3 extbase 和 IRRE : add existing records with 'foreign_selector' ,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/26733567/