主要问题是 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 构建了一个具有“地址”和“城市”属性的 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>
由于我找不到一个多合一的演示并且对该主题感兴趣,因此我创建了一个演示扩展来完成我刚才描述的操作 - 基于核心和两个处理该主题的扩展:https:/ /github.com/lorenzulrich/irreforeignselectordemo
无论如何,解决方案是一种 m:n 方法(因为 1:n 由于上述原因不起作用)所以我决定使用“城市”而不是“城市”。虽然这对于选择城市可能没有意义(如您的帖子所建议的那样),但对于其他机会可能有意义。随意将“城市”替换为“城市”,并将内联配置中的 maxItems 设置为 1 - 然后您就有了一种 1:n。