TYPO3 extbase& IRRE:使用' foreign_selector'添加现有记录

时间:2014-11-04 11:05:13

标签: typo3 extbase typo3-6.2.x

我" kickstarted" extbase扩展构建器的扩展,包含一些1:1和1:n关系。它会自动将字段类型设置为' inline'并在后端显示了一个很好的IRRE UI。

但默认情况下,无法选择现有记录,只需创建新记录即可。

enter image description here

我找到了关于如何使用&foreign_selector'来实现这一点的各种解释,但所有这些都非常粗略。功能本身应该有效,请参阅https://forge.typo3.org/issues/43239

有人可以通过这个或指向TER中的工作示例吗?一旦我开始工作,我就可以从这个例子中创建一个分步教程。

PS由extension_builder生成的字段的TCA配置:

'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 个答案:

答案 0 :(得分:7)

主要问题是类型1的IRRE关系:n的工作方式如下:子记录保存其父级的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表进行比较。因此,我们使用属性" address"为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;

我们现在有一个房产&#34;城市&#34;包含对所有选定城市的引用。您可以遍历它们并使用它们:

<f:for each="{address.cities}" as="cityRelation">
  <li>{cityRelation.city.name}</li>
</f:for>

由于我无法找到一个这样的一体化演示,并对该主题感兴趣,我创建了一个演示扩展,完成我刚才描述的 - 基于Core和两个扩展处理话题:https://github.com/lorenzulrich/irreforeignselectordemo

无论如何,解决方案是一种m:n方法(因为1:n因上述原因而无法工作)所以我决定使用&#34; cities&#34;而不是&#34; city&#34;。虽然这可能对选择城市没有意义(如你的帖子所建议的),但对其他机会可能有意义。随意更换&#34;城市&#34; by&#34; city&#34;并将内联配置中的maxItems设置为1 - 然后你就有了1:n。