gpt4 book ai didi

php - 在带有枚举的 DataObject 的 ModelAdmin 过滤器中使用脚手架列表框进行多项选择

转载 作者:可可西里 更新时间:2023-10-31 23:17:34 29 4
gpt4 key购买 nike

目前,对于有枚举的搜索字段,自动脚手架会生成一个下拉列表,只允许进行一个选择。我有兴趣使用现有过滤器来更改它以允许多项选择。

给定以下数据对象...

class MyDataObject extends DataObject {
static $db = array(
'Name' => "Varchar(255)",
'MyEnum' => "Enum('Option1,Option2,Option3','Option1')"
);
}

...以及以下 ModelAdmin...

class MyModelAdmin extends ModelAdmin {
static $mangaged_models = array(
'MyDataObject',
);
static $url_segment = 'mymodeladmin';
static $menu_title = 'MyModelAdmin';
static $menu_priority = 9;
}

...我正在寻找一个模块或某种简单的过滤器来将 Enum 搭建成一个多选列表框

多选列表框定义为...

  • 允许多项选择
  • 输入一些字符后会提供建议

我要求一个通用 解决方案 - 我可以为每个模型管理员构建一个搜索上下文,但这非常令人沮丧。使用现有过滤器(ExactMatchMultiFilter 看起来很完美但似乎实际上工作)或者如果模块中有一个过滤器或者有人可以建议如何为此修改现有过滤器会很棒。

class MyDataObject extends DataObject {
static $db = array(
'Name' => "Varchar(255)",
'MyEnum' => "Enum('Option1,Option2,Option3','Option1')"
);
public static $searchable_fields = array (
'MyEnum' => array('filter' => 'ExactMatchMultiFilter')
);
}

非常感谢任何帮助。

最佳答案

从您的问题来看,您似乎打算将传递给可搜索字段的 filter 更改脚手架。我做了一些挖掘,但似乎并非如此。但是,如果您使用了 field相反,您可以实现您想要的。

您确实特别提到了 ListboxField虽然它确实支持多个,但字段 which is how it would be instantiated 上的默认构造函数并未启用它.

你想要的东西可以通过开箱即用的方式来完成 CheckboxSetField . (我承认,在 ModelAdmin 中使用时,UI 有点一般)

生成的代码可能如下所示:

class MyDataObject extends DataObject {
static $db = array(
'Name' => "Varchar(255)",
'MyEnum' => "Enum('Option1,Option2,Option3','Option1')"
);
public static $searchable_fields = array (
'MyEnum' => array('field' => 'CheckboxSetField')
);
}

不幸的是,这并不是那么容易,您会注意到,只要这样做,它就会出现“没有可用的选项”,而不是复选框列表。这是因为当我们提供我之前提到的 field 选项时,SilverStripe 的行为有所不同。

这种解决方法不是很好,但可以说仍然是通用的。我制作了 ModelAdmin 的扩展类,它在搜索表单中查找 CheckboxSetField 并为其设置 Enum 值。

class MyModelAdminExtension extends Extension {
public function updateSearchForm($form) {

$modelClass = $form->getController()->modelClass;

foreach ($form->Fields() as $field) {
if ($field->class == 'CheckboxSetField') {
//We need to remove the "q[]" around the field name set by ModelAdmin
$fieldName = substr($field->getName(), 2, -1);
$dbObj = singleton($modelClass)->dbObject($fieldName);
if ($dbObj->class == 'Enum') {
$enumValues = $dbObj->enumValues();
$field->setSource($enumValues);
}
}
}
}
}

这是一个相对安全的 ModelAdmin 扩展,因为它专门查找映射到 CheckboxSetField 的 Enum 的组合,这只有在您手动指定时才会发生。

说到这里,我们实际上可以回顾一下 ListboxField,克服被禁用的多个选项并用值填充它(因为它会遇到上面提到的相同问题)。这个解决方案将不那么通用,因为我们将强制从枚举映射的所有 ListboxField 为倍数,但如果我们想要一个更好的解决方案,这就是我们可以获得它的方法。

class MyModelAdminExtension extends Extension {
public function updateSearchForm($form) {

$modelClass = $form->getController()->modelClass;

foreach ($form->Fields() as $field) {
if ($field->class == 'ListboxField') {
//We need to remove the "q[]" around the field name set by ModelAdmin
$fieldName = substr($field->getName(), 2, -1);
$dbObj = singleton($modelClass)->dbObject($fieldName);
if ($dbObj->class == 'Enum') {
$field->setMultiple(true);

$enumValues = $dbObj->enumValues();
$field->setSource($enumValues);
}
}
}
}
}

对于我们的模型...

class MyDataObject extends DataObject {

private static $db = array(
'Name' => "Varchar(255)",
'MyEnum' => "Enum('Option1,Option2,Option3','Option1')"
);

public static $searchable_fields = array (
'MyEnum' => array('field' => 'ListboxField')
);
}

您现在拥有了您想要的东西 - 具有枚举值的多选 ListBoxField

您现在可能会问,为什么我要覆盖 CheckboxSetField?好吧,我认为研究所有可能的解决方案很重要。我通过尝试 CheckboxSetField 找到了我提供的解决方案,这真的只是最后一分钟的事情,我意识到通过一些小的修改,我可以让它为 ListboxField 工作.


如您所提,上述代码处理 HasOne 关系中的 Enum 时存在问题。这是由于 ModelAdmin 扩展在表单的模型类中采用字段名称并将其专门视为数据库字段(通过 dbObject )。相反,我们可以从字段名称上的双下划线检测到关系,将其替换为点语法并将其视为关系(通过 relObject )。

我们更新后的 updateSearchForm 函数如下所示:

public function updateSearchForm($form) {

$modelClass = $form->getController()->modelClass;

foreach ($form->Fields() as $field) {
if ($field->class == 'ListboxField') {
//We need to remove the "q[]" around the field name set by Model Admin
$fieldName = substr($field->getName(), 2, -1);
$dbObj = null;

//Check if the field name represents a value across a relationship
if (strpos($fieldName, '__') !== false) {
//To use "relObject", we need dot-syntax
$fieldName = str_replace('__', '.', $fieldName);
$dbObj = singleton($modelClass)->relObject($fieldName);
}
else {
$dbObj = singleton($modelClass)->dbObject($fieldName);
}

if ($dbObj != null && $dbObj->class == 'Enum') {
$field->setMultiple(true);

$enumValues = $dbObj->enumValues();
$field->setSource($enumValues);
}
}
}
}

关于php - 在带有枚举的 DataObject 的 ModelAdmin 过滤器中使用脚手架列表框进行多项选择,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35910394/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com