SelectDialog 单选对话框

SelectDialog 单选对话框 同样是基于Field的扩展类,支持 string,int,float,array 类型的Model属性字段注解,其中如果是对 array 类型的注解,则保存相应的值和文本,否则只保存相应的值。
该字段插件主要解决下拉选项过多,使用对话框形式打开页面进行选择。

除了Field 的属性支持以外,SelectDialog 单选对话框自己的属性设置选项如下:

public string $url = '';

打开的对话框URL地址,支持虚拟URL地址。

public string $textSql = '';

使用值获取对应文本的SQL语句,仅适用于非 array 类型的Model属性字段修饰。

例如:

textSql:'select name from @pf_category where id=?'

 

public string|array|null $textFunc = null;

指定值获取对应文本的函数指定,仅适用于非 array 类型的Model属性字段修饰。

函数定义为:

function($value):string

如:

    #[SelectDialog(
        label: '选择文章',
        url: '~/select_dialog',
        textFunc: [self::class, 'getSelectName']
    )]
    public ?int $select1 = null;

    public static function getSelectName(int|string $value): string
    {
        $row = DB::getRow('select title from @pf_article where id=?', $value);
        if ($row == null) {
            return '';
        }
        return $row['title'];
    }


为了演示效果:我们先制作一个简单的对话框页面:

控制器代码如下:
/app/home/controller/SelectDialog.php

namespace app\home\controller;


use beacon\core\Controller;
use beacon\core\DBSelector;
use beacon\core\Method;

class SelectDialog extends Controller
{
    //用于提供穿梭框的数据的
    #[Method(act: 'index', method: Method::POST | Method::GET)]
    public function index(string $keyword = '')
    {
        #非ajax请求直接返回页面。
        if (!$this->isAjax()) {
            $this->display('select_dialog.tpl');
            return;
        }

        $selector = new DBSelector('@pf_article');
        $selector->search("title like concat('%',?,'%')", $keyword);
        $data = $selector->pageData();
        #定义一个字符串模板,用于修饰 $data['list'] 数据。
        $template = <<<'TPL'
        {hook fn='_choice' rs}<a href="javascript:;" class="yee-btn check-item" data-value="{$rs.id}" data-text="{$rs.title}">选择</a>{/hook}
        {hook fn='columnName' rs}{$rs.columnId|column}{/hook}
        {hook fn='categoryName' rs}{$rs.categoryId|category}{/hook}
        TPL;
        # 以上模板用到 修饰器 column 和 category 是自定义的模板修饰器,可参考 sdopx.wj008.net 模板引擎 修饰器的制作。
        # column 是columnId 兑换 栏目名称的修饰器
        # category 是categoryId 兑换 栏目名称的修饰器

        $data['list'] = $this->hookData($data['list'], 'string:' . $template, 'rs', true);
        $this->success('ok', $data);
    }
}


对话框列表页面使用的模板代码如下
/app/home/view/select_dialog.tpl

<!doctype html>
<html lang="zh-cn">
<head>
    <meta charset="utf-8">
    <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/third/vue.min.js"></script>
    <script src="/yeeui/yee.js"></script>
    {literal}
        <style>
            html, body {
                background: #fff;
            }
        </style>
    {/literal}
</head>
<body>
<div class="yee-wrap yee-dialog">

    <div class="yee-list-header">
        <div class="yee-caption">选择列表</div>
        <div class="yee-toolbar">
            <span> 共 <span id="records-count">0</span> 条记录</span>
            <a href="javascript:window.location.reload()" class="refresh-btn"><i class="icofont-refresh"></i>刷新</a>
        </div>
    </div>

    <div name="main-list">

        <div class="yee-list-search">
            <!--YeeUI ajax搜索表单控件-->
            <form id="searchForm" yee-module="search-form" data-bind="#list">
                <div class="yee-cell">
                    <label class="yee-label">文章标题:</label>
                    <span><input name="keyword" class="form-inp text" type="text"/></span>
                </div>
                <div class="yee-cell">
                    <input class="form-btn blue" value="查询" type="submit"/>
                    <input class="form-btn normal" value="重置" type="reset"/>
                    <input type="hidden" name="sort">
                </div>
            </form>
        </div>

        <div class="scrollbar" style="height:calc(100vh - 180px);overflow-y: auto">
            <!--YeeUI数据表格控件-->
            <table id="list" width=100% class="yee-datatable" yee-module="datatable" data-auto-load="true">
                <thead>
                <tr>
                    <th width="80">选择</th>
                    <th align="center" width="60">id</th>
                    <th align="left">文章标题</th>
                    <th align="center" width="80">栏目类别</th>
                    <th align="center" width="80">文章分类</th>
                    <th align="center" width="80">点击数</th>
                    <th align="center" width="180">创建时间</th>
                </tr>
                </thead>
                <tbody yee-template="vue">
                <tr v-for="rs in list">
                    <td align="center" v-html="rs._choice"></td>
                    <td align="center" v-html="rs.id"></td>
                    <td align="left" v-html="rs.title"></td>
                    <td align="center" v-html="rs.columnName"></td>
                    <td align="center" v-html="rs.categoryName"></td>
                    <td align="center" v-html="rs.hits"></td>
                    <td align="center" v-html="rs.addTime"></td>
                </tr>
                <tr v-if="list.length==0">
                    <td colspan="100"> 没有任何数据信息....</td>
                </tr>
                </tbody>
            </table>
        </div>
        <!--YeeUI分页控件-->
        <div yee-module="pagebar" data-bind="#list" class="yee-pagebar">
            <div yee-template="vue">
                <div class="pagebar" v-html="barCode"></div>
                <div class="pagebar-info">共有信息:<span v-text="recordsCount"></span> 页次:<span v-text="page"></span>/<span v-text="pageCount"></span> 每页<span v-text="pageSize"></span></div>
            </div>
        </div>
    </div>

</div>

{literal}
    <script>
        //当对话框初始化完成后,该函数会返回 dialog 的对话框句柄。
        Yee.readyDialog(function (dialog) {
            let table = $("#list");
            //更新
            table.on("render", function (ev, source) {
                $("#records-count").text(source.pageInfo.recordsCount || "0");
            });
            table.on('click', 'a.check-item', function () {
                let data = $(this).data();
                //使用对话框句柄返回数据 data
                dialog.success(data);
                //关闭对话框
                dialog.close();
            });
        });
    </script>
{/literal}

</html>


接着 我们用示例来演示如何使用 单选对话框

UserModel:

namespace app\home\model;

use beacon\core\DB;
use beacon\core\Form;
use beacon\widget\SelectDialog;
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 = '';


    #[SelectDialog(
        label: '选择文章1',
        url: '~/select_dialog',
        textFunc: [self::class, 'getSelectName']
    )]
    public int $select1 = 0;


    #[SelectDialog(
        label: '选择文章2',
        url: '~/select_dialog',
    )]
    public array $select2 = [];


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


    public static function getSelectName(int|string $value): string
    {
        $row = DB::getRow('select title from @pf_article where id=?', $value);
        if ($row == null) {
            return '';
        }
        return $row['title'];
    }

}


模板页面:

<!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>


控制器页面:

namespace app\home\controller;


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": "测试团队",
    "select1": 5,
    "select2": {
        "text": "今天聊聊 Elasticsearch 在PHP项目中的使用。",
        "value": 7
    },
    "intro": "就你最帅。"
}

{
    "___class_name": "app\\home\\model\\UserModel",
    "groupName": "测试团队",
    "select1": "5",
    "select2": {
        "text": "今天聊聊 Elasticsearch 在PHP项目中的使用。",
        "value": 7
    },
    "intro": "就你最帅。"
}


这里 $select1 因为是整形,所以只赋予其 值,而 $select2 是数组,所以把完整的数据结构填入其中。
$select1 需要在回显数据的时候 设置 textSql 或者 textFunc 属性用于显示对应的文本信息。
$select2 不需要设置 textSql textFunc 因为其结果中已经存在有对应的文本信息。



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

琼公网安备 46900702000037号