表单字段的管理

扩展自定义组件

{tip} 扩展自定义Form组件后IDE默认是不会自动补全的,这时候可以通过php artisan admin:ide-helper生成IDE提示文件。

集成富文本编辑器wangEditor

wangEditor是一个优秀的国产的轻量级富文本编辑器,如果dcat-admin自带的基于ckeditor的编辑器组件使用上有问题,可以通过下面的步骤可以集成它,并覆盖掉ckeditor

先下载前端库文件wangEditor,解压到目录public/vendor/wangEditor-3.0.9

然后新建组件类app/Admin/Extensions/WangEditor.php

  1. <?php
  2. namespace App\Admin\Extensions;
  3. use Dcat\Admin\Form\Field;
  4. class WangEditor extends Field
  5. {
  6. protected $view = 'admin.wang-editor';
  7. protected static $css = [
  8. '/vendor/wangEditor-3.0.9/release/wangEditor.min.css',
  9. ];
  10. protected static $js = [
  11. '/vendor/wangEditor-3.0.9/release/wangEditor.min.js',
  12. ];
  13. public function render()
  14. {
  15. $name = $this->formatName($this->column);
  16. $this->script = <<<EOT
  17. // 注意这里的ID一定要通过 replaceNestedFormIndex 函数转换,否则将无法兼容 hasMany 表单
  18. var E = window.wangEditor
  19. var editor = new E(replaceNestedFormIndex('#{$this->id}'));
  20. editor.customConfig.zIndex = 0
  21. editor.customConfig.uploadImgShowBase64 = true
  22. editor.customConfig.onchange = function (html) {
  23. $('input[name=$name]').val(html);
  24. }
  25. editor.create()
  26. EOT;
  27. return parent::render();
  28. }
  29. }

新建视图文件resources/views/admin/wang-editor.blade.php

  1. <div class="{{$viewClass['form-group']}} {!! !$errors->has($label) ? '' : 'has-error' !!}">
  2. <label for="{{$id}}" class="{{$viewClass['label']}} control-label">{{$label}}</label>
  3. <div class="{{$viewClass['field']}}">
  4. @include('admin::form.error')
  5. <div id="{{$id}}" style="width: 100%; height: 100%;">
  6. <p>{!! old($column, $value) !!}</p>
  7. </div>
  8. <input type="hidden" name="{{$name}}" value="{{ old($column, $value) }}" />
  9. @include('admin::form.help-block')
  10. </div>
  11. </div>

然后注册进dcat-admin,在app/Admin/bootstrap.php中添加以下代码:

  1. <?php
  2. use App\Admin\Extensions\WangEditor;
  3. use Dcat\Admin\Form;
  4. Form::extend('editor', WangEditor::class);

调用:

  1. $form->editor('body');

集成富文本编辑器ckeditor

先下载ckeditor 并解压到/public目录,比如放在/public/packages/目录下。

然后新建扩展文件app/Admin/Extensions/Form/CKEditor.php:

  1. <?php
  2. namespace App\Admin\Extensions\Form;
  3. use Dcat\Admin\Form\Field;
  4. class CKEditor extends Field
  5. {
  6. public static $js = [
  7. '/packages/ckeditor/ckeditor.js',
  8. '/packages/ckeditor/adapters/jquery.js',
  9. ];
  10. protected $view = 'admin.ckeditor';
  11. public function render()
  12. {
  13. $this->script = "$('{$this->getElementClassSelector()}').ckeditor();";
  14. return parent::render();
  15. }
  16. }

新建view resources/views/admin/ckeditor.blade.php:

  1. <div class="{{$viewClass['form-group']}} {!! !$errors->has($label) ? '' : 'has-error' !!}">
  2. <label for="{{$id}}" class="{{$viewClass['label']}} control-label">{{$label}}</label>
  3. <div class="{{$viewClass['field']}}">
  4. @include('admin::form.error')
  5. <textarea class="form-control {{ $class }}" name="{{$name}}" placeholder="{{ $placeholder }}" {!! $attributes !!} >{{ old($column, $value) }}</textarea>
  6. @include('admin::form.help-block')
  7. </div>
  8. </div>

然后在app/Admin/bootstrap.php中引入扩展:

  1. use App\Admin\Extensions\Form\CKEditor;
  2. use Dcat\Admin\Form;
  3. Form::extend('ckeditor', CKEditor::class);

然后就能在form中使用了:

  1. $form->ckeditor('content');

集成PHP editor

通过下面的步骤来扩展一个基于codemirror的PHP代码编辑器,效果参考PHP mode

先将codemirror库下载并解压到前端资源目录下,比如放在public/packages/codemirror-5.20.2目录下。

新建组件类app/Admin/Extensions/PHPEditor.php:

  1. <?php
  2. namespace App\Admin\Extensions;
  3. use Dcat\Admin\Form\Field;
  4. class PHPEditor extends Field
  5. {
  6. protected $view = 'admin.php-editor';
  7. protected static $css = [
  8. '/packages/codemirror-5.20.2/lib/codemirror.css',
  9. ];
  10. protected static $js = [
  11. '/packages/codemirror-5.20.2/lib/codemirror.js',
  12. '/packages/codemirror-5.20.2/addon/edit/matchbrackets.js',
  13. '/packages/codemirror-5.20.2/mode/htmlmixed/htmlmixed.js',
  14. '/packages/codemirror-5.20.2/mode/xml/xml.js',
  15. '/packages/codemirror-5.20.2/mode/javascript/javascript.js',
  16. '/packages/codemirror-5.20.2/mode/css/css.js',
  17. '/packages/codemirror-5.20.2/mode/clike/clike.js',
  18. '/packages/codemirror-5.20.2/mode/php/php.js',
  19. ];
  20. public function render()
  21. {
  22. $this->script = <<<EOT
  23. // 注意这里的ID一定要通过 replaceNestedFormIndex 函数转换,否则将无法兼容 hasMany 表单
  24. CodeMirror.fromTextArea(document.getElementById(replaceNestedFormIndex("{$this->id}")), {
  25. lineNumbers: true,
  26. mode: "text/x-php",
  27. extraKeys: {
  28. "Tab": function(cm){
  29. cm.replaceSelection(" " , "end");
  30. }
  31. }
  32. });
  33. EOT;
  34. return parent::render();
  35. }
  36. }

{tip} 类中的静态资源也同样可以从外部引入,参考Editor.php

创建视图resources/views/admin/php-editor.blade.php:

  1. <div class="{{$viewClass['form-group']}} {!! !$errors->has($label) ? '' : 'has-error' !!}">
  2. <label for="{{$id}}" class="{{$viewClass['label']}} control-label">{{$label}}</label>
  3. <div class="{{$viewClass['field']}}">
  4. @include('admin::form.error')
  5. <textarea class="form-control" id="{{$id}}" name="{{$name}}" placeholder="{{ trans('admin::lang.input') }} {{$label}}" {!! $attributes !!} >{{ old($column, $value) }}</textarea>
  6. @include('admin::form.help-block')
  7. </div>
  8. </div>

最后找到文件app/Admin/bootstrap.php,如果文件不存在,请更新dcat-admin,然后新建该文件,添加下面代码:

  1. <?php
  2. use App\Admin\Extensions\PHPEditor;
  3. use Dcat\Admin\Form;
  4. Form::extend('php', PHPEditor::class);

这样就能在model-form中使用PHP编辑器了:

  1. $form->php('code');

通过这种方式,可以添加任意你想要添加的form组件。

常用方法

表单组件是Dcat Admin中最为复杂的组件,下面列一些在扩展表单组件中可能需要用到的方法

处理用户输入的表单值 (prepareInputValue)

通过prepareInputValue方法可以处理用户输入的表单值,默认不做任何处理。此方法在Form表单的saving事件触发之后,表单字段的saving方法之前执行

{tip} 这个功能类似Laravel model中的修改器

  1. class PHPEditor extends Field
  2. {
  3. ...
  4. // 把用户输入的表单值转化为 string 格式保存到数据库
  5. protected function prepareInputValue($value)
  6. {
  7. return (string) $value;
  8. }
  9. }

处理待显示的字段值 (formatFieldData)

通过formatFieldData方法可以处理处理待显示的字段值。此方法在表单字段的customFormat方法之前执行

{tip} 这个功能类似Laravel model中的访问器

  1. class PHPEditor extends Field
  2. {
  3. ...
  4. // 把字段值转化为数组格式
  5. // $data是当前表单的编辑数据,数据格式是 array
  6. protected function formatFieldData($data)
  7. {
  8. return (array) parent::formatFieldData($data);
  9. }
  10. }

获取元素CSS选择器 (getElementClassSelector)

表单组件类实例化后会根据字段名称生成一个css class,然后传递到模板中,我们通常可以通过这个class获取到当前表单的元素对象

模板

  1. <div class="{{$viewClass['form-group']}} {!! !$errors->has($errorKey) ? '' : 'has-error' !!}">
  2. <div for="{{ $id }}" class="{{ $viewClass['label'] }} control-label">
  3. <span>{!! $label !!}</span>
  4. </div>
  5. <div class="{{$viewClass['field']}}">
  6. @include('admin::form.error')
  7. <input type="hidden" name="{{$name}}"/>
  8. <!-- $class 即是根据字段名称生成的class -->
  9. <select class="form-control {{$class}}" style="width: 100%;" name="{{$name}}" {!! $attributes !!} >
  10. <option value=""></option>
  11. @foreach($options as $select => $option)
  12. <option value="{{$select}}" {{ Dcat\Admin\Support\Helper::equal($select, old($column, $value)) ?'selected':'' }}>{{$option}}</option>
  13. @endforeach
  14. </select>
  15. @include('admin::form.help-block')
  16. </div>
  17. </div>

使用

  1. class Select extends Field
  2. {
  3. ...
  4. /**
  5. * {@inheritdoc}
  6. */
  7. public function render()
  8. {
  9. ...
  10. if (empty($this->script)) {
  11. // 通过 getElementClassSelector 方法可以获取到表单中 select 标签的css选择器
  12. $this->script = "$(\"{$this->getElementClassSelector()}\").select2($configs);";
  13. }
  14. ...
  15. return parent::render();
  16. }
  17. }

ID属性

表单组件类实例化后会生成一个唯一的随机字符串并保存到id属性中,然后传递到模板中,我们通常可以通过这个id获取到当前表单组件的关键元素

模板

  1. <div class="{{$viewClass['form-group']}} {!! !$errors->has($label) ? '' : 'has-error' !!}">
  2. <label for="{{$id}}" class="{{$viewClass['label']}} control-label">{{$label}}</label>
  3. <div class="{{$viewClass['field']}}">
  4. @include('admin::form.error')
  5. <!-- $id -->
  6. <textarea class="form-control" id="{{$id}}" name="{{$name}}" placeholder="{{ trans('admin::lang.input') }} {{$label}}" {!! $attributes !!} >{{ old($column, $value) }}</textarea>
  7. @include('admin::form.help-block')
  8. </div>
  9. </div>

使用$id属性

  1. class PHPEditor extends Field
  2. {
  3. ...
  4. public function render()
  5. {
  6. // 通过 $this->id 定位元素
  7. $this->script = <<<JS
  8. // 注意这里的ID一定要通过 replaceNestedFormIndex 函数转换,否则将无法兼容 hasMany 表单
  9. CodeMirror.fromTextArea(document.getElementById(replaceNestedFormIndex("{$this->id}")), {
  10. lineNumbers: true,
  11. mode: "text/x-php",
  12. extraKeys: {
  13. "Tab": function(cm){
  14. cm.replaceSelection(" " , "end");
  15. }
  16. }
  17. });
  18. JS;
  19. return parent::render();
  20. }
  21. }

JS代码

为了让扩展的表单组件能够兼容HasMany以及Table表单,我们必须把JS代码保存在$script属性中

  1. class Select extends Field
  2. {
  3. ...
  4. /**
  5. * {@inheritdoc}
  6. */
  7. public function render()
  8. {
  9. ...
  10. $this->script = "$(\"{$this->getElementClassSelector()}\").select2($configs);";
  11. ...
  12. return parent::render();
  13. }
  14. }

并且如果你是通过id属性去定位元素,则必须使用replaceNestedFormIndex函数对原始id进行转换,否则将无法兼容HasMany以及Table表单

  1. class PHPEditor extends Field
  2. {
  3. ...
  4. public function render()
  5. {
  6. // 通过 $this->id 定位元素
  7. $this->script = <<<JS
  8. // 注意这里的ID一定要通过 replaceNestedFormIndex 函数转换,否则将无法兼容 hasMany 表单
  9. CodeMirror.fromTextArea(document.getElementById(replaceNestedFormIndex("{$this->id}")), {
  10. lineNumbers: true,
  11. mode: "text/x-php",
  12. extraKeys: {
  13. "Tab": function(cm){
  14. cm.replaceSelection(" " , "end");
  15. }
  16. }
  17. });
  18. JS;
  19. return parent::render();
  20. }
  21. }