4

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

给定以下数据对象...

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')
    );
}

任何帮助深表感谢。

4

1 回答 1

3

从您的问题来看,您似乎打算将filter传递给可搜索字段的内容更改为脚手架。我已经进行了一些挖掘,但似乎并非如此。但是,如果您改用该field选项,您可能会实现您想要的。

您确实特别提到ListboxField了,虽然它确实支持多个,但它并没有被字段上的默认构造函数启用,这就是它的实例化方式

您想要的可以通过 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在搜索表单中查找并为其设置枚举值。

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扩展,因为它专门寻找映射到 a 的 Enum 的组合,CheckboxSetField这只能在您手动指定它时发生。

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

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带有 Enum 值的多选。

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


正如您所提出的,上述代码在HasOne关系中处理枚举存在问题。这是由于ModelAdmin扩展采用字段名称并将其视为dbObject表单模型类上的数据库字段(via)。相反,我们可以从字段名称上的双下划线检测关系,将其替换为点语法并将其视为关系(通过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);
            }
        }
    }
}
于 2016-04-17T07:42:17.437 回答