dcat-admin

dcat-admin


表单字段的管理

<h1>表单字段的管理</h1> <h2>扩展自定义组件</h2> <p>在开始扩展表单组件之前需要先了解<a href="js.md#init">动态监听元素生成 (init)</a>功能,表单组件的<code>JS</code>代码建议都使用<code>Dcat.init</code>方法监听并初始化,否则可能无法兼容<code>Form::hasMany</code>以及<code>Form::array</code>等动态生成字段的表单类型。</p> <h3>集成富文本编辑器wangEditor</h3> <p><a href="http://www.wangeditor.com/">wangEditor</a>是一个优秀的国产的轻量级富文本编辑器,如果<code>dcat-admin</code>自带的基于<code>ckeditor</code>的编辑器组件使用上有问题,可以通过下面的步骤可以集成它,并覆盖掉默认的<code>editor</code>:</p> <p>&gt; 为了方便演示,示例中直接使用CDN链接。实际开发中需要先下载前端库文件<a href="https://github.com/wangfupeng1988/wangEditor/releases">wangEditor</a>到服务器,解压到目录<code>public/vendor/wangEditor-4.7.1</code>。</p> <p>新建组件类<code>app/Admin/Extensions/Form/WangEditor.php</code>。</p> <pre><code class="language-php">&amp;lt;?php namespace App\Admin\Extensions\Form; use Dcat\Admin\Form\Field; class WangEditor extends Field { protected $view = 'admin.wang-editor'; }</code></pre> <p>新建视图文件<code>resources/views/admin/wang-editor.blade.php</code>:</p> <pre><code class="language-html">&amp;lt;div class=&amp;quot;{{$viewClass['form-group']}}&amp;quot;&amp;gt; &amp;lt;label class=&amp;quot;{{$viewClass['label']}} control-label&amp;quot;&amp;gt;{{$label}}&amp;lt;/label&amp;gt; &amp;lt;div class=&amp;quot;{{$viewClass['field']}}&amp;quot;&amp;gt; @include('admin::form.error') &amp;lt;div {!! $attributes !!} style=&amp;quot;width: 100%; height: 100%;&amp;quot;&amp;gt; &amp;lt;p&amp;gt;{!! $value !!}&amp;lt;/p&amp;gt; &amp;lt;/div&amp;gt; &amp;lt;input type=&amp;quot;hidden&amp;quot; name=&amp;quot;{{$name}}&amp;quot; value=&amp;quot;{{ old($column, $value) }}&amp;quot; /&amp;gt; @include('admin::form.help-block') &amp;lt;/div&amp;gt; &amp;lt;/div&amp;gt; &amp;lt;!-- script标签加上 &amp;quot;init&amp;quot; 属性后会自动使用 Dcat.init() 方法动态监听元素生成 --&amp;gt; &amp;lt;script require=&amp;quot;@wang-editor&amp;quot; init=&amp;quot;{!! $selector !!}&amp;quot;&amp;gt; var E = window.wangEditor // id 变量是 Dcat.init() 自动生成的,是一个唯一的随机字符串 var editor = new E('#' + id); editor.config.zIndex = 0 editor.config.uploadImgShowBase64 = true editor.config.onchange = function (html) { $this.parents('.form-field').find('input[type=&amp;quot;hidden&amp;quot;]').val(html); } editor.create() &amp;lt;/script&amp;gt;</code></pre> <p>然后注册进<code>dcat-admin</code>,在<code>app/Admin/bootstrap.php</code>中添加以下代码:</p> <pre><code class="language-php">&amp;lt;?php use App\Admin\Extensions\Form\WangEditor; use Dcat\Admin\Form; // 注册前端组件别名 Admin::asset()-&amp;gt;alias('@wang-editor', [ // 为了方便演示效果,这里直接加载CDN链接,实际开发中可以下载到服务器加载 'js' =&amp;gt; ['https://cdn.jsdelivr.net/npm/wangeditor@4.7.1/dist/wangEditor.min.js'], ]); Form::extend('editor', WangEditor::class);</code></pre> <p>调用:</p> <pre><code class="language-php">$form-&amp;gt;editor('body');</code></pre> <h3>集成富文本编辑器ckeditor</h3> <p>先下载<a href="http://ckeditor.com/download">ckeditor</a> 并解压到/public目录,比如放在<code>/public/packages/</code>目录下。</p> <p>然后新建扩展文件<code>app/Admin/Extensions/Form/CKEditor.php</code>:</p> <pre><code class="language-php">&amp;lt;?php namespace App\Admin\Extensions\Form; use Dcat\Admin\Form\Field; class CKEditor extends Field { protected $view = 'admin.ckeditor'; }</code></pre> <p>新建view <code>resources/views/admin/ckeditor.blade.php</code>:</p> <pre><code class="language-html">&amp;lt;div class=&amp;quot;{{$viewClass['form-group']}}&amp;quot;&amp;gt; &amp;lt;label class=&amp;quot;{{$viewClass['label']}} control-label&amp;quot;&amp;gt;{{$label}}&amp;lt;/label&amp;gt; &amp;lt;div class=&amp;quot;{{$viewClass['field']}}&amp;quot;&amp;gt; @include('admin::form.error') &amp;lt;textarea name=&amp;quot;{{ $name}}&amp;quot; placeholder=&amp;quot;{{ $placeholder }}&amp;quot; {!! $attributes !!} &amp;gt;{!! $value !!}&amp;lt;/textarea&amp;gt; @include('admin::form.help-block') &amp;lt;/div&amp;gt; &amp;lt;/div&amp;gt; &amp;lt;script require=&amp;quot;@ckeditor&amp;quot; init=&amp;quot;{!! $selector !!}&amp;quot;&amp;gt; $this.ckeditor(); &amp;lt;/script&amp;gt;</code></pre> <p>然后在<code>app/Admin/bootstrap.php</code>中引入扩展:</p> <pre><code class="language-php">use App\Admin\Extensions\Form\CKEditor; use Dcat\Admin\Form; // 注册前端组件别名 Admin::asset()-&amp;gt;alias('@ckeditor', [ 'js' =&amp;gt; [ '/packages/ckeditor/ckeditor.js', '/packages/ckeditor/adapters/jquery.js', ], ]); Form::extend('ckeditor', CKEditor::class);</code></pre> <p>然后就能在form中使用了:</p> <pre><code class="language-php">$form-&amp;gt;ckeditor('content');</code></pre> <h3>集成PHP editor</h3> <p>通过下面的步骤来扩展一个基于<a href="http://codemirror.net/index.html">codemirror</a>的PHP代码编辑器,效果参考<a href="http://codemirror.net/mode/php/">PHP mode</a>。</p> <p>先将<a href="http://codemirror.net/codemirror.zip">codemirror</a>库下载并解压到前端资源目录下,比如放在<code>public/packages/codemirror-5.20.2</code>目录下。</p> <p>新建组件类<code>app/Admin/Extensions/Form/PHPEditor.php</code>:</p> <pre><code class="language-php">&amp;lt;?php namespace App\Admin\Extensions\Form; use Dcat\Admin\Form\Field; class PHPEditor extends Field { protected $view = 'admin.php-editor'; } </code></pre> <p>&gt; {tip} 类中的静态资源也同样可以从外部引入,参考<a href="https://github.com/jqhph/dcat-admin/blob/master/src/Form/Field/Editor.php">Editor.php</a></p> <p>创建视图<code>resources/views/admin/php-editor.blade.php</code>:</p> <pre><code class="language-html">&amp;lt;div class=&amp;quot;{{$viewClass['form-group']}}&amp;quot;&amp;gt; &amp;lt;label class=&amp;quot;{{$viewClass['label']}} control-label&amp;quot;&amp;gt;{{$label}}&amp;lt;/label&amp;gt; &amp;lt;div class=&amp;quot;{{$viewClass['field']}}&amp;quot;&amp;gt; @include('admin::form.error') &amp;lt;textarea class=&amp;quot;{{ $class }}&amp;quot; {!! $attributes !!} &amp;gt;{!! $value !!}&amp;lt;/textarea&amp;gt; &amp;lt;input type=&amp;quot;hidden&amp;quot; name=&amp;quot;{{$name}}&amp;quot; value=&amp;quot;{{ old($column, $value) }}&amp;quot; /&amp;gt; @include('admin::form.help-block') &amp;lt;/div&amp;gt; &amp;lt;/div&amp;gt; &amp;lt;script require=&amp;quot;@phpeditor&amp;quot; init=&amp;quot;{!! $selector !!}&amp;quot;&amp;gt; var Editor = CodeMirror.fromTextArea(document.getElementById(id), { lineNumbers: true, mode: &amp;quot;text/x-php&amp;quot;, extraKeys: { &amp;quot;Tab&amp;quot;: function(cm){ cm.replaceSelection(&amp;quot; &amp;quot; , &amp;quot;end&amp;quot;); } } }); Editor.on(&amp;quot;change&amp;quot;, function (Editor, changes) { let val = Editor.getValue(); //console.log(val); $this.parents('.form-field').find('input[type=&amp;quot;hidden&amp;quot;]').val(val); }); &amp;lt;/script&amp;gt;</code></pre> <p>最后找到文件<code>app/Admin/bootstrap.php</code>,如果文件不存在,请更新<code>dcat-admin</code>,然后新建该文件,添加下面代码:</p> <pre><code>&amp;lt;?php use App\Admin\Extensions\Form\PHPEditor; use Dcat\Admin\Form; Admin::asset()-&amp;gt;alias('@phpeditor', [ 'js' =&amp;gt; [ '/packages/codemirror-5.20.2/lib/codemirror.js', '/packages/codemirror-5.20.2/addon/edit/matchbrackets.js', '/packages/codemirror-5.20.2/mode/htmlmixed/htmlmixed.js', '/packages/codemirror-5.20.2/mode/xml/xml.js', '/packages/codemirror-5.20.2/mode/javascript/javascript.js', '/packages/codemirror-5.20.2/mode/css/css.js', '/packages/codemirror-5.20.2/mode/clike/clike.js', '/packages/codemirror-5.20.2/mode/php/php.js', ], 'css' =&amp;gt; '/packages/codemirror-5.20.2/lib/codemirror.css', ]); Form::extend('php', PHPEditor::class);</code></pre> <p>这样就能在<a href="model-form.md">model-form</a>中使用PHP编辑器了:</p> <pre><code>$form-&amp;gt;php('code');</code></pre> <p>通过这种方式,可以添加任意你想要添加的<code>form</code>组件。</p> <h2>常用方法</h2> <p>表单组件是<code>Dcat Admin</code>中最为复杂的组件,下面列一些在扩展表单组件中可能需要用到的方法</p> <h3>处理用户输入的表单值 (prepareInputValue)</h3> <p>通过<code>prepareInputValue</code>方法可以处理用户输入的表单值,默认不做任何处理。此方法在<code>Form</code>表单的<code>saving</code>事件触发之后,表单字段的<code>saving</code>方法之前执行</p> <p>&gt; {tip} 这个功能类似<code>Laravel model</code>中的<strong>修改器</strong>。</p> <pre><code class="language-php">class PHPEditor extends Field { ... // 把用户输入的表单值转化为 string 格式保存到数据库 protected function prepareInputValue($value) { return (string) $value; } }</code></pre> <h3>处理待显示的字段值 (formatFieldData)</h3> <p>通过<code>formatFieldData</code>方法可以处理处理待显示的字段值。此方法在表单字段的<code>customFormat</code>方法之前执行</p> <p>&gt; {tip} 这个功能类似<code>Laravel model</code>中的<strong>访问器</strong>。</p> <pre><code class="language-php">class PHPEditor extends Field { ... // 把字段值转化为数组格式 // $data是当前表单的编辑数据,数据格式是 array protected function formatFieldData($data) { // 获取到当前字段值 $value = parent::formatFieldData($data); // 处理字段值 ... return $value; } }</code></pre> <h3>获取元素CSS选择器 (getElementClassSelector)</h3> <p>表单组件类实例化后会根据字段名称生成一个<code>css class</code>,然后传递到模板中,我们通常可以通过这个<code>class</code>获取到当前表单的元素对象</p> <p>模板</p> <pre><code class="language-html">&amp;lt;div class=&amp;quot;{{$viewClass['form-group']}}&amp;quot;&amp;gt; &amp;lt;div class=&amp;quot;{{ $viewClass['label'] }} control-label&amp;quot;&amp;gt; &amp;lt;span&amp;gt;{!! $label !!}&amp;lt;/span&amp;gt; &amp;lt;/div&amp;gt; &amp;lt;div class=&amp;quot;{{$viewClass['field']}}&amp;quot;&amp;gt; @include('admin::form.error') &amp;lt;input type=&amp;quot;hidden&amp;quot; name=&amp;quot;{{$name}}&amp;quot;/&amp;gt; &amp;lt;select style=&amp;quot;width: 100%;&amp;quot; name=&amp;quot;{{$name}}&amp;quot; {!! $attributes !!} &amp;gt; &amp;lt;option value=&amp;quot;&amp;quot;&amp;gt;&amp;lt;/option&amp;gt; @foreach($options as $select =&amp;gt; $option) &amp;lt;option value=&amp;quot;{{$select}}&amp;quot; {{ Dcat\Admin\Support\Helper::equal($select, $value) ?'selected':'' }}&amp;gt;{{$option}}&amp;lt;/option&amp;gt; @endforeach &amp;lt;/select&amp;gt; @include('admin::form.help-block') &amp;lt;/div&amp;gt; &amp;lt;/div&amp;gt; &amp;lt;script require=&amp;quot;...&amp;quot; init=&amp;quot;{!! $selector !!}&amp;quot;&amp;gt; // 这里的 $selector 即是当前字段的 css选择器 $this.select2($configs); &amp;lt;/script&amp;gt;</code></pre> <h3>JS代码</h3> <p>为了让扩展的表单组件能够兼容<code>HasMany</code>、<code>array</code>以及<code>Table</code>表单,我们必须使用<a href="js.md#init">动态监听元素生成 (init)</a>功能</p> <pre><code class="language-html">&amp;lt;div class=&amp;quot;{{$viewClass['form-group']}}&amp;quot;&amp;gt; ... &amp;lt;/div&amp;gt; &amp;lt;script require=&amp;quot;...&amp;quot; init=&amp;quot;{!! $selector !!}&amp;quot;&amp;gt; $this.select2($configs); &amp;lt;/script&amp;gt;</code></pre>

页面列表

ITEM_HTML