Container 复杂容器

 

Container 复杂容器(也可以称为可动态添加子表单的容器) 可以支持 Field 的部分属性,是基于Field的扩展类,仅支持array 类型的Model属性字段注解。

Single 简单容器不同的是,Container 复杂容器可以添加多行。

Container 复杂容器的其他属性选项如下:

 

public string $itemClass = '';

子表单的Model类名,仅支持书写类名不支持实例。

public string $subType = 'add';

创建子表单是什么类型,一般不用设置会保持和主表单一致,除非有特殊的要求。

public string|array|\Closure|null $modifyFunc = null;

设置一个函数 用于调整子表单的数据,一般在开发中不经常使用到,除非你需要根据数据对子表单进行调整。
函数格式如下

//$subForm 为以参数传递的子表单,可用于针对子表单进行调整子表单的字段等信息
function(Form $subForm){}

 

public string $template = '';

为子表单指定模板文件,一般情况下 Model 在Form注解中已经设置了模板,但是这里可在此指定模板覆盖Model中使用Form 注解种指定的模板。

 

public bool $removeBtn = true;

是否显示移除按钮 (需要在模板中对应的控制)

public bool $insertBtn = false;

是否显示插入按钮 (需要在模板中对应的控制)

 

public bool $sortBtn = false;

是否显示顺序调整按钮 (需要在模板中对应的控制)


为了演示效果
我们先制作一个 子Model 和一个 子模板

/app/home/model/ContainerModel.php

namespace app\home\model;

use beacon\core\Form;
use beacon\widget\Integer;
use beacon\widget\Select;
use beacon\widget\Text;

#[Form(title: '子表单', template: 'container.form.tpl')]
class ContainerModel
{
    #[Text(
        label: '姓名',
        validRule: ['r' => '请填写姓名'],
        #将错误内容输出到#container-error的html元素上。
        validDisplay: '#container-error'
    )]
    public string $name = '';

    #[Select(
        label: '性别',
        header: '请选择',
        options: [['value' => '男', 'text' => '男'], ['value' => '女', 'text' => '女']],
        validRule: ['r' => '请选择性别'],
        validDisplay: '#container-error'
    )]
    public string $sex = '';

    #[Integer(
        label: '年龄',
        after: '岁',
        validRule: ['r' => '请填写年龄'],
        validDisplay: '#container-error'
    )]
    public ?int $age = null;


}


子表单模板:
Container 复杂容器 的模板定义与其他模板定义不同,这里需要分为两部分设计。
所以这里的子表单需要定义两个 hook 方法。

#wrap 方法用于输出整个区域的模板数据内容,即整个容器字段所在的区域内容,外层模板。
# 其中 field 是主表单的字段,body 是所有子表单生成的html 代码 需要用raw 修饰符原义输出。
{hook fn='wrap' field  body}{/hook}

#item 方法用于每个子表单的内容部分 field 是主表单的字段,form 是子表单,index 是索引占位符(保持原样即可)。
{hook fn='item' field form index='@@index@@'}{/hook}


/app/home/view/container.form.tpl
代码如下:

{*控件外层包裹 field 是主控件的字段,body 子表单生成的内容*}
{hook fn='wrap' field  body}
    <div class="yee-row" id="row_{$field->boxId}">
        <label class="row-label" style="line-height: 40px">{if $field->star}<em></em>{/if}{$field->label}:</label>
        <div class="row-cell">
            <div style="display: block;">{$body|raw}</div>
            <div style="display: block;">
                <a href="javascript:;" name="add" class="yee-btn"><i class="icofont-plus-circle"></i>新增行</a>
                <!--用于输出错误信息-->
                <span id="container-error"></span>
            </div>
        </div>
    </div>
{/hook}

{*这里是子表单的渲染节点,field 是主控件的字段 form 是子表单*}
{hook fn='item' field form index='@@index@@'}
    <div class="yee-row">
        <div class="row-cell">
            <span>姓名: {input field=$form->getField('name')}</span>
            <span>性别: {input field=$form->getField('sex')}</span>
            <span>年龄: {input field=$form->getField('age')}</span>
            {* yee-module 控件会检索 name=add  name=remove  name=upsort|dnsort name=insert 的a标签,并赋予相应的操作,所以我们指定 name 就可以了  *}
            {if $field->insertBtn}<a href="javascript:;" name="insert" class="yee-btn"><i class="icofont-puzzle"></i>插入</a>{/if}
            {if $field->sortBtn}
                <a href="javascript:;" name="upsort" class="yee-btn"><i class="icofont-long-arrow-up"></i>上移</a>
                <a href="javascript:;" name="dnsort" class="yee-btn"><i class="icofont-long-arrow-down"></i>下移</a>
            {/if}
            {if $field->removeBtn}<a href="javascript:;" name="remove" class="yee-btn"><i class="icofont-minus-circle"></i>移除</a>{/if}
        </div>
    </div>
{/hook}



接着我们的主表单使用如下
UserModel

namespace app\home\model;

use beacon\core\Form;
use beacon\widget\Container;
use beacon\widget\Text;
use beacon\widget\Textarea;

#注意 这里需要使用表单注解
#[Form(title: '用户表单', table: '@pf_user', template: 'user.form.tpl')]
class UserModel
{

    #[Text(
        label: '团队名称',
        validRule: ['r' => '请填写团队名称'],
    )]
    public string $groupName = '';


    #[Container(
        label: '人员名单',
        #使用主表单的模型类类名。
        itemClass: ContainerModel::class,
        insertBtn:true
    )]
    public array $container = [];

    #[Textarea(
        label: '团队介绍',
        validRule: ['r' => '请填写个人介绍'],
    )]
    public string $intro = '';


}


主表单的 模板如下:

<!doctype html>
<html lang="zh-cn">
<head>
    <meta charset="utf-8">
    <title>{$form->title}</title>
    <link type="text/css" rel="stylesheet" href="/yeeui/css/yeeui.css"/>
    <link type="text/css" rel="stylesheet" href="/icofont/icofont.css"/>
    <script src="/yeeui/third/jquery-3.3.1.min.js"></script>
    <script src="/yeeui/yee.js"></script>
</head>
<body>
<div style="margin: 20px">
    <form method="post" yee-module="ajax validate">
        <div class="yee-panel">
            <div class="panel-caption">
                {if $form->getType()=='add'}添加{elseif $form->getType()=='edit'}编辑{/if}{$form->title}
            </div>
            <div class="panel-content">
                {foreach from=$form->getViewFields() item=field}
                    {field_row field=$field}
                {/foreach}
            </div>
            <div class="yee-submit">
                <label class="submit-label"></label>
                <div class="submit-cell">
                    {*输出隐藏域*}
                    {$form->fetchHideBox()|raw}
                    <input type="submit" class="form-btn red" value="提交">
                </div>
            </div>
        </div>
    </form>
</div>
</html>


控制器代码如下:

use app\home\model\UserModel;
use beacon\core\Controller;
use beacon\core\Form;
use beacon\core\Logger;
use beacon\core\Method;

class User extends Controller
{
    #[Method(act: 'index', method: Method::GET)]
    public function index()
    {
        $form = Form::create(UserModel::class, 'add');
        $this->displayForm($form);
    }

    #[Method(act: 'index', method: Method::POST)]
    public function add()
    {
        $user = new UserModel();
        $form = Form::create($user, 'add');
        $input = $this->completeForm($form);
        Logger::log($input); #input 可以用用于插入或者更新数据库。
        Logger::log($user);
        //这里需要有返回值 否则 ajax 提交会没有任何内容返回
        $this->success('ok');
    }

}

 

最后呈现效果如下:



提交后debug.php 输出的数据如下:

{
    "groupName": "测试团队",
    "container": [
        {
            "name": "张三",
            "sex": "男",
            "age": 30
        },
        {
            "name": "王五",
            "sex": "女",
            "age": 20
        }
    ],
    "intro": "介绍信息"
}

{
    "___class_name": "app\\home\\model\\UserModel",
    "groupName": "测试团队",
    "container": [
        {
            "name": "张三",
            "sex": "男",
            "age": 30
        },
        {
            "name": "王五",
            "sex": "女",
            "age": 20
        }
    ],
    "intro": "介绍信息"
}

 

Copyright © 2021 海南的叶子 All Rights Reserved 琼ICP备2021000725号

琼公网安备 46900702000037号