号码管理
This commit is contained in:
@@ -0,0 +1,77 @@
|
||||
-- 分流管理 - 号码管理表与菜单权限
|
||||
SET NAMES utf8mb4;
|
||||
SET FOREIGN_KEY_CHECKS = 0;
|
||||
|
||||
DROP TABLE IF EXISTS `fa_split_number`;
|
||||
CREATE TABLE `fa_split_number` (
|
||||
`id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '号码ID',
|
||||
`admin_id` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '管理员ID',
|
||||
`split_link_id` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '关联分流链接ID',
|
||||
`ticket_name` varchar(100) NOT NULL DEFAULT '' COMMENT '工单名称',
|
||||
`number` varchar(50) NOT NULL DEFAULT '' COMMENT '号码',
|
||||
`number_type` varchar(32) NOT NULL DEFAULT '' COMMENT '号码类型',
|
||||
`number_type_custom` varchar(50) NOT NULL DEFAULT '' COMMENT '自定义号码类型',
|
||||
`visit_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '访问次数',
|
||||
`inbound_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '进线人数',
|
||||
`manual_manage` tinyint(1) unsigned NOT NULL DEFAULT '0' COMMENT '手动管理:0=否,1=是',
|
||||
`status` enum('normal','hidden') NOT NULL DEFAULT 'normal' COMMENT '状态:normal=正常,hidden=停用',
|
||||
`createtime` bigint(16) DEFAULT NULL COMMENT '创建时间',
|
||||
`updatetime` bigint(16) DEFAULT NULL COMMENT '更新时间',
|
||||
PRIMARY KEY (`id`),
|
||||
UNIQUE KEY `uk_link_number` (`split_link_id`,`number`),
|
||||
KEY `admin_id` (`admin_id`),
|
||||
KEY `split_link_id` (`split_link_id`),
|
||||
KEY `status` (`status`),
|
||||
KEY `number` (`number`)
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='分流号码表';
|
||||
|
||||
SET FOREIGN_KEY_CHECKS = 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', r.id, 'split.number', '号码管理', 'fa fa-phone', '', '', 1, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` r
|
||||
WHERE r.name = 'split' AND r.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.number' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.number/index', '查看', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.number' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.number/index' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.number/add', '添加', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.number' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.number/add' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.number/edit', '编辑', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.number' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.number/edit' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.number/del', '删除', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.number' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.number/del' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.number/batchupdate', '批量更新号码', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.number' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.number/batchupdate' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.number/multi', '批量更新', 'fa fa-circle-o', '', '列表状态开关', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.number' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.number/multi' LIMIT 1)
|
||||
LIMIT 1;
|
||||
@@ -0,0 +1,109 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\validate\split;
|
||||
|
||||
use app\admin\model\split\Link;
|
||||
use app\admin\model\split\Number as NumberModel;
|
||||
use think\Validate;
|
||||
|
||||
/**
|
||||
* 分流号码验证器(部署源文件,由 deploy_split_number.php 复制到 application)
|
||||
*/
|
||||
class Number extends Validate
|
||||
{
|
||||
protected $rule = [
|
||||
'number_type' => 'require|checkNumberType',
|
||||
'number_type_custom' => 'max:50|checkNumberTypeCustom',
|
||||
'split_link_id' => 'require|integer|gt:0|checkSplitLink',
|
||||
'ticket_name' => 'require|max:100',
|
||||
'numbers' => 'require|checkNumbers',
|
||||
'number' => 'require|max:50',
|
||||
'status' => 'require|checkStatus',
|
||||
];
|
||||
|
||||
protected $message = [
|
||||
'number_type.require' => '请选择号码类型',
|
||||
'split_link_id.require' => '请选择分流链接',
|
||||
'ticket_name.require' => '请填写工单名称',
|
||||
'ticket_name.max' => '工单名称不能超过100个字符',
|
||||
'numbers.require' => '请填写号码',
|
||||
'number.require' => '请填写号码',
|
||||
'number.max' => '号码不能超过50个字符',
|
||||
'status.require' => '请选择状态',
|
||||
];
|
||||
|
||||
protected $scene = [
|
||||
'add' => ['number_type', 'number_type_custom', 'split_link_id', 'ticket_name', 'numbers', 'status'],
|
||||
'edit' => ['number_type', 'number_type_custom', 'split_link_id', 'ticket_name', 'number', 'status'],
|
||||
];
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkNumberType($value)
|
||||
{
|
||||
$key = (string) $value;
|
||||
$list = (new NumberModel())->getNumberTypeList();
|
||||
return isset($list[$key]) ? true : '号码类型无效';
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @param mixed $rule
|
||||
* @param array $data
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkNumberTypeCustom($value, $rule, array $data = [])
|
||||
{
|
||||
if (($data['number_type'] ?? '') !== 'custom') {
|
||||
return true;
|
||||
}
|
||||
if (trim((string) $value) === '') {
|
||||
return '请填写自定义号码类型';
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @param mixed $rule
|
||||
* @param array $data
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkSplitLink($value, $rule, array $data = [])
|
||||
{
|
||||
$linkId = (int) $value;
|
||||
$link = Link::get($linkId);
|
||||
if (!$link) {
|
||||
return '所选分流链接不存在';
|
||||
}
|
||||
if (($link['status'] ?? '') !== 'normal') {
|
||||
return '所选分流链接已停用';
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkNumbers($value)
|
||||
{
|
||||
$list = NumberModel::parseNumbersText((string) $value);
|
||||
return $list !== [] ? true : '请至少填写一个有效号码(每行一个)';
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkStatus($value)
|
||||
{
|
||||
$key = (string) $value;
|
||||
$list = (new NumberModel())->getStatusList();
|
||||
return isset($list[$key]) ? true : '状态无效';
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,89 @@
|
||||
-- 分流管理 - 工单管理表与菜单权限
|
||||
SET NAMES utf8mb4;
|
||||
SET FOREIGN_KEY_CHECKS = 0;
|
||||
|
||||
DROP TABLE IF EXISTS `fa_split_ticket`;
|
||||
CREATE TABLE `fa_split_ticket` (
|
||||
`id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT 'ID',
|
||||
`admin_id` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '管理员ID',
|
||||
`ticket_type` varchar(32) NOT NULL DEFAULT '' COMMENT '工单类型',
|
||||
`ticket_name` varchar(100) NOT NULL DEFAULT '' COMMENT '工单名称',
|
||||
`ticket_url` varchar(1000) NOT NULL DEFAULT '' COMMENT '工单链接',
|
||||
`ticket_total` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '工单总量',
|
||||
`split_link_id` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '关联分流链接ID',
|
||||
`number_type` varchar(32) NOT NULL DEFAULT '' COMMENT '号码类型',
|
||||
`number_type_custom` varchar(50) NOT NULL DEFAULT '' COMMENT '自定义号码类型',
|
||||
`start_time` bigint(16) DEFAULT NULL COMMENT '开始时间',
|
||||
`end_time` bigint(16) DEFAULT NULL COMMENT '到期时间',
|
||||
`order_limit` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '单号上限',
|
||||
`assign_ratio` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '下号比率',
|
||||
`account` varchar(50) NOT NULL DEFAULT '' COMMENT '工单账号',
|
||||
`password` varchar(50) NOT NULL DEFAULT '' COMMENT '工单密码',
|
||||
`status` enum('normal','hidden') NOT NULL DEFAULT 'normal' COMMENT '状态:normal=正常,hidden=停用',
|
||||
`complete_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '完成数量(同步)',
|
||||
`inbound_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '进线人数(同步)',
|
||||
`speed_per_hour` decimal(10,2) unsigned NOT NULL DEFAULT '0.00' COMMENT '当前速度:每小时进线(同步)',
|
||||
`number_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '号码数量含离线封号(同步)',
|
||||
`number_offline_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '离线号码数(同步)',
|
||||
`number_banned_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '封号号码数(同步)',
|
||||
`online_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '在线人数(同步)',
|
||||
`sync_status` enum('success','error','pending') NOT NULL DEFAULT 'pending' COMMENT '同步状态',
|
||||
`sync_time` bigint(16) DEFAULT NULL COMMENT '最近同步时间',
|
||||
`sync_message` varchar(255) NOT NULL DEFAULT '' COMMENT '同步失败摘要',
|
||||
`createtime` bigint(16) DEFAULT NULL COMMENT '创建时间',
|
||||
`updatetime` bigint(16) DEFAULT NULL COMMENT '更新时间',
|
||||
PRIMARY KEY (`id`),
|
||||
KEY `admin_id` (`admin_id`),
|
||||
KEY `split_link_id` (`split_link_id`),
|
||||
KEY `ticket_type` (`ticket_type`),
|
||||
KEY `start_time` (`start_time`),
|
||||
KEY `end_time` (`end_time`),
|
||||
KEY `status` (`status`),
|
||||
KEY `sync_status` (`sync_status`),
|
||||
KEY `sync_time` (`sync_time`)
|
||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='分流工单表';
|
||||
|
||||
SET FOREIGN_KEY_CHECKS = 1;
|
||||
|
||||
-- 菜单:工单管理(父级 split)
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', r.id, 'split.ticket', '工单管理', 'fa fa-ticket', '', '', 1, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` r
|
||||
WHERE r.name = 'split' AND r.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.ticket' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.ticket/index', '查看', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.ticket' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.ticket/index' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.ticket/add', '添加', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.ticket' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.ticket/add' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.ticket/edit', '编辑', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.ticket' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.ticket/edit' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.ticket/del', '删除', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.ticket' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.ticket/del' LIMIT 1)
|
||||
LIMIT 1;
|
||||
|
||||
INSERT INTO `fa_auth_rule` (`type`, `pid`, `name`, `title`, `icon`, `condition`, `remark`, `ismenu`, `createtime`, `updatetime`, `weigh`, `status`)
|
||||
SELECT 'file', m.id, 'split.ticket/multi', '批量更新', 'fa fa-circle-o', '', '', 0, UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), 0, 'normal'
|
||||
FROM `fa_auth_rule` m
|
||||
WHERE m.name = 'split.ticket' AND m.ismenu = 1
|
||||
AND NOT EXISTS (SELECT 1 FROM `fa_auth_rule` WHERE `name` = 'split.ticket/multi' LIMIT 1)
|
||||
LIMIT 1;
|
||||
@@ -0,0 +1,6 @@
|
||||
-- 工单表补充 status 字段(列表开关依赖此列)
|
||||
SET NAMES utf8mb4;
|
||||
|
||||
ALTER TABLE `fa_split_ticket`
|
||||
ADD COLUMN `status` enum('normal','hidden') NOT NULL DEFAULT 'normal' COMMENT '状态:normal=正常,hidden=停用' AFTER `password`,
|
||||
ADD KEY `status` (`status`);
|
||||
@@ -0,0 +1,28 @@
|
||||
-- 工单表:统计字段 + ticket_total 整数化(供列表展示与后续同步接口写入)
|
||||
SET NAMES utf8mb4;
|
||||
|
||||
-- 1. 同步统计字段(若已存在则跳过需手工处理)
|
||||
ALTER TABLE `fa_split_ticket`
|
||||
ADD COLUMN `complete_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '完成数量(同步)' AFTER `status`,
|
||||
ADD COLUMN `inbound_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '进线人数(同步)' AFTER `complete_count`,
|
||||
ADD COLUMN `speed_per_hour` decimal(10,2) unsigned NOT NULL DEFAULT '0.00' COMMENT '当前速度:每小时进线(同步)' AFTER `inbound_count`,
|
||||
ADD COLUMN `number_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '号码数量含离线封号(同步)' AFTER `speed_per_hour`,
|
||||
ADD COLUMN `number_offline_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '离线号码数(同步)' AFTER `number_count`,
|
||||
ADD COLUMN `number_banned_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '封号号码数(同步)' AFTER `number_offline_count`,
|
||||
ADD COLUMN `online_count` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '在线人数(同步)' AFTER `number_banned_count`,
|
||||
ADD COLUMN `sync_status` enum('success','error','pending') NOT NULL DEFAULT 'pending' COMMENT '同步状态' AFTER `online_count`,
|
||||
ADD COLUMN `sync_time` bigint(16) DEFAULT NULL COMMENT '最近同步时间' AFTER `sync_status`,
|
||||
ADD COLUMN `sync_message` varchar(255) NOT NULL DEFAULT '' COMMENT '同步失败摘要' AFTER `sync_time`;
|
||||
|
||||
ALTER TABLE `fa_split_ticket`
|
||||
ADD KEY `sync_status` (`sync_status`),
|
||||
ADD KEY `sync_time` (`sync_time`);
|
||||
|
||||
-- 2. ticket_total:varchar -> int unsigned(非数字旧值变为 0)
|
||||
UPDATE `fa_split_ticket` SET `ticket_total` = '0' WHERE `ticket_total` REGEXP '[^0-9]' OR `ticket_total` = '';
|
||||
ALTER TABLE `fa_split_ticket`
|
||||
MODIFY COLUMN `ticket_total` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '工单总量';
|
||||
|
||||
-- 3. 号码表组合索引:加速按工单汇总 visit_count
|
||||
ALTER TABLE `fa_split_number`
|
||||
ADD KEY `idx_ticket_agg` (`admin_id`, `split_link_id`, `ticket_name`(50));
|
||||
@@ -0,0 +1,154 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\validate\split;
|
||||
|
||||
use app\admin\model\split\Link;
|
||||
use app\admin\model\split\Ticket as TicketModel;
|
||||
use think\Validate;
|
||||
|
||||
/**
|
||||
* 分流工单验证器(部署源文件,由 deploy_split_ticket.php 复制到 application)
|
||||
*/
|
||||
class Ticket extends Validate
|
||||
{
|
||||
protected $rule = [
|
||||
'ticket_type' => 'require|checkTicketType',
|
||||
'ticket_name' => 'require|max:100',
|
||||
'ticket_url' => 'require|max:1000',
|
||||
'ticket_total' => 'require|integer|egt:0',
|
||||
'split_link_id' => 'require|integer|gt:0|checkSplitLink',
|
||||
'number_type' => 'require|checkNumberType',
|
||||
'number_type_custom' => 'max:50|checkNumberTypeCustom',
|
||||
'start_time' => 'require|checkStartTime',
|
||||
'end_time' => 'require|checkEndTime',
|
||||
'order_limit' => 'require|integer|egt:0',
|
||||
'assign_ratio' => 'require|integer|egt:0',
|
||||
'account' => 'max:50',
|
||||
'password' => 'max:50',
|
||||
];
|
||||
|
||||
protected $message = [
|
||||
'ticket_type.require' => '请选择工单类型',
|
||||
'ticket_name.require' => '请填写工单名称',
|
||||
'ticket_name.max' => '工单名称不能超过100个字符',
|
||||
'ticket_url.require' => '请填写工单链接',
|
||||
'ticket_url.max' => '工单链接不能超过1000个字符',
|
||||
'ticket_total.require' => '请填写工单总量',
|
||||
'ticket_total.integer' => '工单总量必须为整数',
|
||||
'ticket_total.egt' => '工单总量不能小于0',
|
||||
'split_link_id.require' => '请选择分流链接',
|
||||
'number_type.require' => '请选择号码类型',
|
||||
'start_time.require' => '请选择开始时间',
|
||||
'end_time.require' => '请选择到期时间',
|
||||
'order_limit.require' => '请填写单号上限',
|
||||
'order_limit.egt' => '单号上限不能小于0',
|
||||
'assign_ratio.require' => '请填写下号比率',
|
||||
'assign_ratio.egt' => '下号比率不能小于0',
|
||||
];
|
||||
|
||||
protected $scene = [
|
||||
'add' => [
|
||||
'ticket_type', 'ticket_name', 'ticket_url', 'ticket_total', 'split_link_id',
|
||||
'number_type', 'number_type_custom', 'start_time', 'end_time',
|
||||
'order_limit', 'assign_ratio', 'account', 'password',
|
||||
],
|
||||
'edit' => [
|
||||
'ticket_type', 'ticket_name', 'ticket_url', 'ticket_total', 'split_link_id',
|
||||
'number_type', 'number_type_custom', 'start_time', 'end_time',
|
||||
'order_limit', 'assign_ratio', 'account', 'password',
|
||||
],
|
||||
];
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkTicketType($value)
|
||||
{
|
||||
$key = (string) $value;
|
||||
$list = (new TicketModel())->getTicketTypeList();
|
||||
return isset($list[$key]) ? true : '工单类型无效';
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkNumberType($value)
|
||||
{
|
||||
$key = (string) $value;
|
||||
$list = (new TicketModel())->getNumberTypeList();
|
||||
return isset($list[$key]) ? true : '号码类型无效';
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @param mixed $rule
|
||||
* @param array $data
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkNumberTypeCustom($value, $rule, array $data = [])
|
||||
{
|
||||
if (($data['number_type'] ?? '') !== 'custom') {
|
||||
return true;
|
||||
}
|
||||
if (trim((string) $value) === '') {
|
||||
return '请填写自定义号码类型';
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @param mixed $rule
|
||||
* @param array $data
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkSplitLink($value, $rule, array $data = [])
|
||||
{
|
||||
$linkId = (int) $value;
|
||||
$link = Link::get($linkId);
|
||||
if (!$link) {
|
||||
return '所选分流链接不存在';
|
||||
}
|
||||
if (($link['status'] ?? '') !== 'normal') {
|
||||
return '所选分流链接已停用';
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkStartTime($value)
|
||||
{
|
||||
$ts = is_numeric($value) ? (int) $value : strtotime((string) $value);
|
||||
return ($ts !== false && $ts > 0) ? true : '开始时间格式无效';
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
* @param mixed $rule
|
||||
* @param array $data
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function checkEndTime($value, $rule, array $data = [])
|
||||
{
|
||||
$endTs = is_numeric($value) ? (int) $value : strtotime((string) $value);
|
||||
if ($endTs === false || $endTs <= 0) {
|
||||
return '到期时间格式无效';
|
||||
}
|
||||
$startRaw = $data['start_time'] ?? '';
|
||||
$startTs = is_numeric($startRaw) ? (int) $startRaw : strtotime((string) $startRaw);
|
||||
if ($startTs === false || $startTs <= 0) {
|
||||
return true;
|
||||
}
|
||||
if ($endTs <= $startTs) {
|
||||
return '到期时间必须晚于开始时间';
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,383 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller\split;
|
||||
|
||||
use app\admin\model\split\Link as LinkModel;
|
||||
use app\admin\model\split\Number as NumberModel;
|
||||
use app\common\controller\Backend;
|
||||
use think\Db;
|
||||
use think\Exception;
|
||||
use think\exception\DbException;
|
||||
use think\exception\PDOException;
|
||||
use think\exception\ValidateException;
|
||||
use think\response\Json;
|
||||
|
||||
/**
|
||||
* 号码管理
|
||||
*
|
||||
* @icon fa fa-phone
|
||||
* @remark 分流号码管理,关联分流链接
|
||||
*/
|
||||
class Number extends Backend
|
||||
{
|
||||
/** @var NumberModel */
|
||||
protected $model = null;
|
||||
|
||||
protected $searchFields = 'ticket_name,number';
|
||||
|
||||
protected $dataLimit = 'personal';
|
||||
|
||||
protected $modelValidate = true;
|
||||
|
||||
protected $modelSceneValidate = true;
|
||||
|
||||
/** @var string[] */
|
||||
protected $noNeedRight = ['script'];
|
||||
|
||||
/** @var string */
|
||||
private const PATCH_VIEW_DIR = 'patches/application/admin/view/split/number/';
|
||||
|
||||
public function _initialize()
|
||||
{
|
||||
parent::_initialize();
|
||||
|
||||
$lang = $this->request->langset();
|
||||
$lang = preg_match('/^([a-zA-Z\-_]{2,10})$/i', $lang) ? $lang : 'zh-cn';
|
||||
$langFile = ROOT_PATH . 'patches/application/admin/lang/' . $lang . '/split/number.php';
|
||||
if (is_file($langFile)) {
|
||||
\think\Lang::load($langFile);
|
||||
}
|
||||
|
||||
$this->model = new NumberModel();
|
||||
$this->view->assign('numberTypeList', $this->model->getNumberTypeList());
|
||||
$this->view->assign('statusList', $this->model->getStatusList());
|
||||
$this->view->assign('manualManageList', $this->model->getManualManageList());
|
||||
$this->view->assign('splitLinkList', $this->buildSplitLinkList());
|
||||
$this->assignconfig('numberTypeList', $this->model->getNumberTypeList());
|
||||
$this->assignconfig('statusList', $this->model->getStatusList());
|
||||
$this->assignconfig('manualManageList', $this->model->getManualManageList());
|
||||
|
||||
$this->setupPatchFrontend();
|
||||
}
|
||||
|
||||
private function setupPatchFrontend(): void
|
||||
{
|
||||
$patchJs = ROOT_PATH . 'patches/public/assets/js/backend/split/number.js';
|
||||
$publicJs = ROOT_PATH . 'public/assets/js/backend/split/number.js';
|
||||
$usePatchJs = is_file($patchJs) && (
|
||||
!is_file($publicJs) || filemtime($patchJs) >= filemtime($publicJs)
|
||||
);
|
||||
if (!$usePatchJs) {
|
||||
return;
|
||||
}
|
||||
|
||||
$cfg = is_array($this->view->config ?? null) ? $this->view->config : [];
|
||||
$version = (string) \think\Config::get('site.version');
|
||||
$scriptUrl = (string) url('split.number/script', ['v' => $version], false, true);
|
||||
if (strpos($scriptUrl, '?') === false) {
|
||||
$scriptUrl .= '?v=' . $version;
|
||||
}
|
||||
if (strpos($scriptUrl, '://') === false) {
|
||||
$scriptUrl = $this->request->domain() . $scriptUrl;
|
||||
}
|
||||
$cfg['jsname'] = $scriptUrl;
|
||||
$this->view->assign('config', $cfg);
|
||||
$this->view->config = $cfg;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<int, array<string, string>>
|
||||
*/
|
||||
private function buildSplitLinkList(): array
|
||||
{
|
||||
$query = (new LinkModel())->where('status', 'normal')->order('id', 'desc');
|
||||
if ($this->dataLimit) {
|
||||
$adminIds = $this->getDataLimitAdminIds();
|
||||
if (is_array($adminIds)) {
|
||||
$query->where('admin_id', 'in', $adminIds);
|
||||
}
|
||||
}
|
||||
$list = [];
|
||||
foreach ($query->select() as $row) {
|
||||
$code = (string) $row['link_code'];
|
||||
$desc = (string) $row['description'];
|
||||
$label = $desc !== '' ? $code . ' - ' . $desc : $code;
|
||||
$list[] = [
|
||||
'id' => (string) $row['id'],
|
||||
'label' => $label,
|
||||
];
|
||||
}
|
||||
return $list;
|
||||
}
|
||||
|
||||
private function fetchPatch(string $template): string
|
||||
{
|
||||
$patchFile = ROOT_PATH . self::PATCH_VIEW_DIR . $template . '.html';
|
||||
$appFile = APP_PATH . 'admin/view/split/number/' . $template . '.html';
|
||||
if (is_file($patchFile)) {
|
||||
$file = $patchFile;
|
||||
} elseif (is_file($appFile)) {
|
||||
$file = $appFile;
|
||||
} else {
|
||||
$this->error('模板文件不存在');
|
||||
}
|
||||
return (string) $this->view->fetch($file);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string|Json
|
||||
* @throws DbException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
$this->request->filter(['strip_tags', 'trim']);
|
||||
if (false === $this->request->isAjax()) {
|
||||
return $this->fetchPatch('index');
|
||||
}
|
||||
if ($this->request->request('keyField')) {
|
||||
return $this->selectpage();
|
||||
}
|
||||
[$where, $sort, $order, $offset, $limit] = $this->buildparams();
|
||||
$list = $this->model
|
||||
->with(['splitLink'])
|
||||
->where($where)
|
||||
->order($sort, $order)
|
||||
->paginate($limit);
|
||||
$result = ['total' => $list->total(), 'rows' => $list->items()];
|
||||
return json($result);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string
|
||||
* @throws DbException
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
if (false === $this->request->isPost()) {
|
||||
return $this->fetchPatch('add');
|
||||
}
|
||||
$params = $this->request->post('row/a', []);
|
||||
if ($params === []) {
|
||||
$this->error(__('Parameter %s can not be empty', ''));
|
||||
}
|
||||
$params = $this->preExcludeFields($params);
|
||||
if (($params['number_type'] ?? '') !== 'custom') {
|
||||
$params['number_type_custom'] = '';
|
||||
}
|
||||
|
||||
$numbersText = (string) ($params['numbers'] ?? '');
|
||||
unset($params['numbers']);
|
||||
$numberList = NumberModel::parseNumbersText($numbersText);
|
||||
if ($numberList === []) {
|
||||
$this->error(__('Please fill at least one number'));
|
||||
}
|
||||
|
||||
$validateData = array_merge($params, ['numbers' => $numbersText]);
|
||||
$adminId = $this->dataLimit && $this->dataLimitFieldAutoFill ? (int) $this->auth->id : 0;
|
||||
|
||||
$inserted = 0;
|
||||
$skipped = 0;
|
||||
Db::startTrans();
|
||||
try {
|
||||
if ($this->modelValidate) {
|
||||
$name = str_replace('\\model\\', '\\validate\\', get_class($this->model));
|
||||
$validate = $this->modelSceneValidate ? $name . '.add' : $name;
|
||||
$this->model->validateFailException()->validate($validate, $validateData);
|
||||
}
|
||||
$splitLinkId = (int) ($params['split_link_id'] ?? 0);
|
||||
$baseRow = [
|
||||
'split_link_id' => $splitLinkId,
|
||||
'ticket_name' => (string) ($params['ticket_name'] ?? ''),
|
||||
'number_type' => (string) ($params['number_type'] ?? ''),
|
||||
'number_type_custom' => (string) ($params['number_type_custom'] ?? ''),
|
||||
'status' => (string) ($params['status'] ?? 'normal'),
|
||||
'visit_count' => 0,
|
||||
'inbound_count' => 0,
|
||||
'manual_manage' => 0,
|
||||
];
|
||||
if ($adminId > 0) {
|
||||
$baseRow['admin_id'] = $adminId;
|
||||
}
|
||||
foreach ($numberList as $num) {
|
||||
$exists = $this->model
|
||||
->where('split_link_id', $splitLinkId)
|
||||
->where('number', $num)
|
||||
->find();
|
||||
if ($exists) {
|
||||
$skipped++;
|
||||
continue;
|
||||
}
|
||||
$row = $baseRow;
|
||||
$row['number'] = $num;
|
||||
$item = new NumberModel();
|
||||
$item->allowField(true)->isUpdate(false)->save($row);
|
||||
$inserted++;
|
||||
}
|
||||
if ($inserted === 0) {
|
||||
throw new Exception(__('All numbers already exist for this link'));
|
||||
}
|
||||
Db::commit();
|
||||
} catch (ValidateException $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
} catch (PDOException|Exception $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
}
|
||||
|
||||
$msg = __('Inserted %d number(s)', $inserted);
|
||||
if ($skipped > 0) {
|
||||
$msg .= ',' . __('Skipped %d duplicate(s)', $skipped);
|
||||
}
|
||||
$this->success($msg);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string|null $ids
|
||||
* @return string
|
||||
* @throws DbException
|
||||
*/
|
||||
public function edit($ids = null)
|
||||
{
|
||||
$row = $this->model->get($ids);
|
||||
if (!$row) {
|
||||
$this->error(__('No Results were found'));
|
||||
}
|
||||
$adminIds = $this->getDataLimitAdminIds();
|
||||
if (is_array($adminIds) && !in_array((int) $row[$this->dataLimitField], $adminIds, true)) {
|
||||
$this->error(__('You have no permission'));
|
||||
}
|
||||
if (false === $this->request->isPost()) {
|
||||
$this->view->assign('row', $row);
|
||||
return $this->fetchPatch('edit');
|
||||
}
|
||||
$params = $this->request->post('row/a', []);
|
||||
if ($params === []) {
|
||||
$this->error(__('Parameter %s can not be empty', ''));
|
||||
}
|
||||
$params = $this->preExcludeFields($params);
|
||||
if (($params['number_type'] ?? '') !== 'custom') {
|
||||
$params['number_type_custom'] = '';
|
||||
}
|
||||
|
||||
$newNumber = trim((string) ($params['number'] ?? ''));
|
||||
$splitLinkId = (int) ($params['split_link_id'] ?? $row['split_link_id']);
|
||||
$exists = $this->model
|
||||
->where('split_link_id', $splitLinkId)
|
||||
->where('number', $newNumber)
|
||||
->where('id', '<>', (int) $row['id'])
|
||||
->find();
|
||||
if ($exists) {
|
||||
$this->error(__('Number already exists for this link'));
|
||||
}
|
||||
|
||||
$result = false;
|
||||
Db::startTrans();
|
||||
try {
|
||||
if ($this->modelValidate) {
|
||||
$name = str_replace('\\model\\', '\\validate\\', get_class($this->model));
|
||||
$validate = $this->modelSceneValidate ? $name . '.edit' : $name;
|
||||
$row->validateFailException()->validate($validate, $params);
|
||||
}
|
||||
$result = $row->allowField(true)->save($params);
|
||||
Db::commit();
|
||||
} catch (ValidateException $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
} catch (PDOException|Exception $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
}
|
||||
if ($result === false) {
|
||||
$this->error(__('No rows were updated'));
|
||||
}
|
||||
$this->success();
|
||||
}
|
||||
|
||||
/**
|
||||
* 批量更新号码状态与手动管理
|
||||
*/
|
||||
public function batchupdate(): void
|
||||
{
|
||||
if (false === $this->request->isPost()) {
|
||||
$this->error(__('Invalid parameters'));
|
||||
}
|
||||
$ids = $this->request->post('ids', '');
|
||||
if ($ids === '' || $ids === null) {
|
||||
$this->error(__('Parameter %s can not be empty', 'ids'));
|
||||
}
|
||||
if (!is_array($ids)) {
|
||||
$ids = explode(',', (string) $ids);
|
||||
}
|
||||
$ids = array_filter(array_map('intval', $ids));
|
||||
if ($ids === []) {
|
||||
$this->error(__('Parameter %s can not be empty', 'ids'));
|
||||
}
|
||||
|
||||
$status = (string) $this->request->post('status', '');
|
||||
$manualManage = $this->request->post('manual_manage', '');
|
||||
$statusList = $this->model->getStatusList();
|
||||
if (!isset($statusList[$status])) {
|
||||
$this->error(__('Invalid status'));
|
||||
}
|
||||
if (!in_array((string) $manualManage, ['0', '1'], true)) {
|
||||
$this->error(__('Invalid manual manage'));
|
||||
}
|
||||
|
||||
$query = $this->model->where('id', 'in', $ids);
|
||||
if ($this->dataLimit) {
|
||||
$adminIds = $this->getDataLimitAdminIds();
|
||||
if (is_array($adminIds)) {
|
||||
$query->where('admin_id', 'in', $adminIds);
|
||||
}
|
||||
}
|
||||
$count = $query->update([
|
||||
'status' => $status,
|
||||
'manual_manage' => (int) $manualManage,
|
||||
]);
|
||||
if ($count === false || $count === 0) {
|
||||
$this->error(__('No rows were updated'));
|
||||
}
|
||||
$this->success(__('Batch update success'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 排除不可由表单提交的字段
|
||||
*
|
||||
* @param array<string, mixed> $params
|
||||
* @return array<string, mixed>
|
||||
*/
|
||||
protected function preExcludeFields($params): array
|
||||
{
|
||||
$params = parent::preExcludeFields($params);
|
||||
unset(
|
||||
$params['visit_count'],
|
||||
$params['inbound_count'],
|
||||
$params['manual_manage'],
|
||||
$params['id']
|
||||
);
|
||||
return $params;
|
||||
}
|
||||
|
||||
public function script(): void
|
||||
{
|
||||
$jsFile = ROOT_PATH . 'patches/public/assets/js/backend/split/number.js';
|
||||
if (!is_file($jsFile)) {
|
||||
$jsFile = ROOT_PATH . 'public/assets/js/backend/split/number.js';
|
||||
}
|
||||
if (!is_file($jsFile)) {
|
||||
$this->error('脚本文件不存在');
|
||||
}
|
||||
$content = file_get_contents($jsFile);
|
||||
if ($content === false) {
|
||||
$this->error('读取脚本失败');
|
||||
}
|
||||
$response = response($content, 200, [
|
||||
'Content-Type' => 'application/javascript; charset=utf-8',
|
||||
'Cache-Control' => 'public, max-age=3600',
|
||||
]);
|
||||
throw new \think\exception\HttpResponseException($response);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,306 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller\split;
|
||||
|
||||
use app\admin\model\split\Link as LinkModel;
|
||||
use app\admin\model\split\Ticket as TicketModel;
|
||||
use app\common\controller\Backend;
|
||||
use think\Db;
|
||||
use think\Exception;
|
||||
use think\exception\DbException;
|
||||
use think\exception\PDOException;
|
||||
use think\exception\ValidateException;
|
||||
use think\response\Json;
|
||||
|
||||
/**
|
||||
* 工单管理
|
||||
*
|
||||
* @icon fa fa-ticket
|
||||
* @remark 分流工单管理,关联分流链接
|
||||
*/
|
||||
class Ticket extends Backend
|
||||
{
|
||||
/** @var \app\admin\model\split\Ticket */
|
||||
protected $model = null;
|
||||
|
||||
protected $searchFields = 'ticket_name,ticket_url,ticket_total';
|
||||
|
||||
protected $dataLimit = 'personal';
|
||||
|
||||
protected $modelValidate = true;
|
||||
|
||||
protected $modelSceneValidate = true;
|
||||
|
||||
/** @var string[] 无需鉴权的方法 */
|
||||
protected $noNeedRight = ['script'];
|
||||
|
||||
/** @var string patches 视图目录 */
|
||||
private const PATCH_VIEW_DIR = 'patches/application/admin/view/split/ticket/';
|
||||
|
||||
public function _initialize()
|
||||
{
|
||||
parent::_initialize();
|
||||
|
||||
$lang = $this->request->langset();
|
||||
$lang = preg_match('/^([a-zA-Z\-_]{2,10})$/i', $lang) ? $lang : 'zh-cn';
|
||||
$langFile = ROOT_PATH . 'patches/application/admin/lang/' . $lang . '/split/ticket.php';
|
||||
if (is_file($langFile)) {
|
||||
\think\Lang::load($langFile);
|
||||
}
|
||||
$linkLangFile = ROOT_PATH . 'patches/application/admin/lang/' . $lang . '/split/link.php';
|
||||
if (is_file($linkLangFile)) {
|
||||
\think\Lang::load($linkLangFile);
|
||||
}
|
||||
|
||||
$this->model = new \app\admin\model\split\Ticket();
|
||||
$this->view->assign('ticketTypeList', $this->model->getTicketTypeList());
|
||||
$this->view->assign('numberTypeList', $this->model->getNumberTypeList());
|
||||
$this->view->assign('statusList', $this->model->getStatusList());
|
||||
$this->view->assign('splitLinkList', $this->buildSplitLinkList());
|
||||
$this->assignconfig('ticketTypeList', $this->model->getTicketTypeList());
|
||||
$this->assignconfig('numberTypeList', $this->model->getNumberTypeList());
|
||||
$this->assignconfig('statusList', $this->model->getStatusList());
|
||||
|
||||
$this->setupPatchFrontend();
|
||||
}
|
||||
|
||||
/**
|
||||
* 未部署 JS 时指向 script 接口
|
||||
*/
|
||||
private function setupPatchFrontend(): void
|
||||
{
|
||||
$patchJs = ROOT_PATH . 'patches/public/assets/js/backend/split/ticket.js';
|
||||
$publicJs = ROOT_PATH . 'public/assets/js/backend/split/ticket.js';
|
||||
$usePatchJs = is_file($patchJs) && (
|
||||
!is_file($publicJs) || filemtime($patchJs) >= filemtime($publicJs)
|
||||
);
|
||||
if (!$usePatchJs) {
|
||||
return;
|
||||
}
|
||||
|
||||
$cfg = is_array($this->view->config ?? null) ? $this->view->config : [];
|
||||
$version = (string) \think\Config::get('site.version');
|
||||
$scriptUrl = (string) url('split.ticket/script', ['v' => $version], false, true);
|
||||
if (strpos($scriptUrl, '?') === false) {
|
||||
$scriptUrl .= '?v=' . $version;
|
||||
}
|
||||
if (strpos($scriptUrl, '://') === false) {
|
||||
$scriptUrl = $this->request->domain() . $scriptUrl;
|
||||
}
|
||||
$cfg['jsname'] = $scriptUrl;
|
||||
$this->view->assign('config', $cfg);
|
||||
$this->view->config = $cfg;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<int, array<string, string>>
|
||||
*/
|
||||
private function buildSplitLinkList(): array
|
||||
{
|
||||
$query = (new LinkModel())->where('status', 'normal')->order('id', 'desc');
|
||||
if ($this->dataLimit) {
|
||||
$adminIds = $this->getDataLimitAdminIds();
|
||||
if (is_array($adminIds)) {
|
||||
$query->where('admin_id', 'in', $adminIds);
|
||||
}
|
||||
}
|
||||
$list = [];
|
||||
foreach ($query->select() as $row) {
|
||||
$code = (string) $row['link_code'];
|
||||
$desc = (string) $row['description'];
|
||||
$label = $desc !== '' ? $code . ' - ' . $desc : $code;
|
||||
$list[] = [
|
||||
'id' => (string) $row['id'],
|
||||
'label' => $label,
|
||||
];
|
||||
}
|
||||
return $list;
|
||||
}
|
||||
|
||||
private function fetchPatch(string $template): string
|
||||
{
|
||||
$patchFile = ROOT_PATH . self::PATCH_VIEW_DIR . $template . '.html';
|
||||
$appFile = APP_PATH . 'admin/view/split/ticket/' . $template . '.html';
|
||||
if (is_file($patchFile)) {
|
||||
$file = $patchFile;
|
||||
} elseif (is_file($appFile)) {
|
||||
$file = $appFile;
|
||||
} else {
|
||||
$this->error('模板文件不存在');
|
||||
}
|
||||
return (string) $this->view->fetch($file);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string|Json
|
||||
* @throws DbException
|
||||
* @throws \think\Exception
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
$this->request->filter(['strip_tags', 'trim']);
|
||||
if (false === $this->request->isAjax()) {
|
||||
return $this->fetchPatch('index');
|
||||
}
|
||||
if ($this->request->request('keyField')) {
|
||||
return $this->selectpage();
|
||||
}
|
||||
[$where, $sort, $order, $offset, $limit] = $this->buildparams();
|
||||
$ticketTable = $this->model->getTable();
|
||||
$clickSub = TicketModel::buildClickCountSubQuery();
|
||||
// 勿 with(splitLink):eagerlyType=0 为 JOIN 预载入,会重写 field 导致子查询 SQL 语法错误
|
||||
$list = $this->model
|
||||
->field($ticketTable . '.*,' . $clickSub)
|
||||
->where($where)
|
||||
->order($sort, $order)
|
||||
->paginate($limit);
|
||||
$result = ['total' => $list->total(), 'rows' => $list->items()];
|
||||
return json($result);
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步统计字段仅由接口写入,禁止表单提交篡改
|
||||
*
|
||||
* @param array<string, mixed> $params
|
||||
* @return array<string, mixed>
|
||||
*/
|
||||
protected function preExcludeFields($params): array
|
||||
{
|
||||
$params = parent::preExcludeFields($params);
|
||||
unset(
|
||||
$params['complete_count'],
|
||||
$params['inbound_count'],
|
||||
$params['speed_per_hour'],
|
||||
$params['number_count'],
|
||||
$params['number_offline_count'],
|
||||
$params['number_banned_count'],
|
||||
$params['online_count'],
|
||||
$params['sync_status'],
|
||||
$params['sync_time'],
|
||||
$params['sync_message'],
|
||||
$params['click_count']
|
||||
);
|
||||
return $params;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string
|
||||
* @throws DbException
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
if (false === $this->request->isPost()) {
|
||||
return $this->fetchPatch('add');
|
||||
}
|
||||
$params = $this->request->post('row/a', []);
|
||||
if ($params === []) {
|
||||
$this->error(__('Parameter %s can not be empty', ''));
|
||||
}
|
||||
$params = $this->preExcludeFields($params);
|
||||
if (($params['number_type'] ?? '') !== 'custom') {
|
||||
$params['number_type_custom'] = '';
|
||||
}
|
||||
if ($this->dataLimit && $this->dataLimitFieldAutoFill) {
|
||||
$params[$this->dataLimitField] = $this->auth->id;
|
||||
}
|
||||
$result = false;
|
||||
Db::startTrans();
|
||||
try {
|
||||
if ($this->modelValidate) {
|
||||
$name = str_replace('\\model\\', '\\validate\\', get_class($this->model));
|
||||
$validate = $this->modelSceneValidate ? $name . '.add' : $name;
|
||||
$this->model->validateFailException()->validate($validate);
|
||||
}
|
||||
$result = $this->model->allowField(true)->save($params);
|
||||
Db::commit();
|
||||
} catch (ValidateException $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
} catch (PDOException|Exception $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
}
|
||||
if ($result === false) {
|
||||
$this->error(__('No rows were inserted'));
|
||||
}
|
||||
$this->success();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string|null $ids
|
||||
* @return string
|
||||
* @throws DbException
|
||||
*/
|
||||
public function edit($ids = null)
|
||||
{
|
||||
$row = $this->model->get($ids);
|
||||
if (!$row) {
|
||||
$this->error(__('No Results were found'));
|
||||
}
|
||||
$adminIds = $this->getDataLimitAdminIds();
|
||||
if (is_array($adminIds) && !in_array((int) $row[$this->dataLimitField], $adminIds, true)) {
|
||||
$this->error(__('You have no permission'));
|
||||
}
|
||||
if (false === $this->request->isPost()) {
|
||||
$rowData = $row->toArray();
|
||||
$rowData['start_time'] = $rowData['start_time_text'] ?? '';
|
||||
$rowData['end_time'] = $rowData['end_time_text'] ?? '';
|
||||
$this->view->assign('row', $rowData);
|
||||
return $this->fetchPatch('edit');
|
||||
}
|
||||
$params = $this->request->post('row/a', []);
|
||||
if ($params === []) {
|
||||
$this->error(__('Parameter %s can not be empty', ''));
|
||||
}
|
||||
$params = $this->preExcludeFields($params);
|
||||
if (($params['number_type'] ?? '') !== 'custom') {
|
||||
$params['number_type_custom'] = '';
|
||||
}
|
||||
$result = false;
|
||||
Db::startTrans();
|
||||
try {
|
||||
if ($this->modelValidate) {
|
||||
$name = str_replace('\\model\\', '\\validate\\', get_class($this->model));
|
||||
$validate = $this->modelSceneValidate ? $name . '.edit' : $name;
|
||||
$row->validateFailException()->validate($validate);
|
||||
}
|
||||
$result = $row->allowField(true)->save($params);
|
||||
Db::commit();
|
||||
} catch (ValidateException $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
} catch (PDOException|Exception $e) {
|
||||
Db::rollback();
|
||||
$this->error($e->getMessage());
|
||||
}
|
||||
if ($result === false) {
|
||||
$this->error(__('No rows were updated'));
|
||||
}
|
||||
$this->success();
|
||||
}
|
||||
|
||||
/**
|
||||
* 输出后台 JS(patches 未部署到 public 时)
|
||||
*/
|
||||
public function script(): void
|
||||
{
|
||||
$jsFile = ROOT_PATH . 'patches/public/assets/js/backend/split/ticket.js';
|
||||
if (!is_file($jsFile)) {
|
||||
$jsFile = ROOT_PATH . 'public/assets/js/backend/split/ticket.js';
|
||||
}
|
||||
if (!is_file($jsFile)) {
|
||||
$this->error('脚本文件不存在');
|
||||
}
|
||||
$content = file_get_contents($jsFile);
|
||||
if ($content === false) {
|
||||
$this->error('读取脚本失败');
|
||||
}
|
||||
$response = response($content, 200, [
|
||||
'Content-Type' => 'application/javascript; charset=utf-8',
|
||||
'Cache-Control' => 'public, max-age=3600',
|
||||
]);
|
||||
throw new \think\exception\HttpResponseException($response);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,38 @@
|
||||
<?php
|
||||
|
||||
return [
|
||||
'Id' => '号码ID',
|
||||
'Link_url' => '分流链接',
|
||||
'Ticket_name' => '工单名称',
|
||||
'Number' => '号码',
|
||||
'Numbers' => '号码',
|
||||
'Number_type' => '号码类型',
|
||||
'Number_type_custom' => '自定义号码类型',
|
||||
'Split_link_id' => '选择链接',
|
||||
'Visit_count' => '访问次数',
|
||||
'Inbound_count' => '进线人数',
|
||||
'Status' => '状态',
|
||||
'Manual_manage' => '手动管理',
|
||||
'Createtime' => '创建时间',
|
||||
'Section basic' => '基础信息',
|
||||
'Status normal' => '正常',
|
||||
'Status hidden' => '停用',
|
||||
'Number type custom' => '自定义',
|
||||
'Manual manage yes' => '是',
|
||||
'Manual manage no' => '否',
|
||||
'Please select' => '请选择',
|
||||
'Numbers placeholder' => '每行填写一个号码',
|
||||
'Batch update title' => '批量更新号码',
|
||||
'Batch selected count' => '已选号码',
|
||||
'Batch update status' => '更新状态',
|
||||
'Batch update btn' => '更新状态',
|
||||
'Please fill at least one number' => '请至少填写一个有效号码',
|
||||
'All numbers already exist for this link' => '该链接下号码均已存在,未新增任何记录',
|
||||
'Number already exists for this link' => '该链接下已存在相同号码',
|
||||
'Inserted %d number(s)' => '成功添加 %d 条号码',
|
||||
'Skipped %d duplicate(s)' => '跳过 %d 条重复号码',
|
||||
'Invalid status' => '状态无效',
|
||||
'Invalid manual manage' => '手动管理选项无效',
|
||||
'Batch update success' => '批量更新成功',
|
||||
'Unit count' => '个',
|
||||
];
|
||||
@@ -0,0 +1,52 @@
|
||||
<?php
|
||||
|
||||
return [
|
||||
'Ticket_type' => '工单类型',
|
||||
'Ticket_name' => '工单名称',
|
||||
'Ticket_url' => '工单链接',
|
||||
'Ticket_total' => '工单总量',
|
||||
'Split_link_id' => '分流链接',
|
||||
'View split link tip' => '点击查看分流链接与复制完整 URL',
|
||||
'Copy link code tip' => '复制链接码',
|
||||
'Number_type' => '号码类型',
|
||||
'Number_type_custom' => '自定义号码类型',
|
||||
'Start_time' => '开始时间',
|
||||
'End_time' => '到期时间',
|
||||
'Order_limit' => '单号上限',
|
||||
'Assign_ratio' => '下号比率',
|
||||
'Account' => '工单账号',
|
||||
'Password' => '工单密码',
|
||||
'Status' => '状态',
|
||||
'Status normal' => '正常',
|
||||
'Status hidden' => '停用',
|
||||
'Complete_count' => '完成数量',
|
||||
'Ticket_progress' => '工单进度',
|
||||
'Inbound_ratio' => '进线比例',
|
||||
'Speed_per_hour' => '当前速度',
|
||||
'Number_count' => '号码数量',
|
||||
'Number_count_detail' => '离线 %s / 封号 %s',
|
||||
'Sync_status' => '同步状态',
|
||||
'Sync status success' => '同步成功',
|
||||
'Sync status error' => '同步异常',
|
||||
'Sync status pending' => '待同步',
|
||||
'Sync display success' => '同步成功 / 在线人数 %s',
|
||||
'Sync display error' => '同步异常',
|
||||
'Createtime' => '创建时间',
|
||||
'Section basic' => '基础信息',
|
||||
'Section time rule' => '时间与规则',
|
||||
'Section account' => '账号信息',
|
||||
'Number type custom' => '自定义',
|
||||
'Ticket type xinghe' => '星河云控',
|
||||
'Ticket type haiwang' => '海王',
|
||||
'Ticket type taiji' => '太极云控',
|
||||
'Ticket type huojian' => '火箭云控',
|
||||
'Ticket type ss_channel'=> 'SS云控(Channel)',
|
||||
'Ticket type ss_customer'=> 'SS云控(Customer)',
|
||||
'Ticket type yifafa' => '译发发云控',
|
||||
'Ticket type a2c' => 'A2C云控',
|
||||
'Ticket type ceo_scrm' => 'CEO SCRM',
|
||||
'Ticket type whatshub' => 'Whatshub云控',
|
||||
'Ticket type sihai' => '四海云控',
|
||||
'End time must after start' => '到期时间必须晚于开始时间',
|
||||
'Please select' => '请选择',
|
||||
];
|
||||
@@ -0,0 +1,165 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\model\split;
|
||||
|
||||
use app\common\service\SplitPlatformDomainService;
|
||||
use think\Db;
|
||||
use think\Model;
|
||||
|
||||
/**
|
||||
* 分流号码模型
|
||||
*/
|
||||
class Number extends Model
|
||||
{
|
||||
protected $name = 'split_number';
|
||||
|
||||
protected $autoWriteTimestamp = 'integer';
|
||||
|
||||
protected $createTime = 'createtime';
|
||||
protected $updateTime = 'updatetime';
|
||||
protected $deleteTime = false;
|
||||
|
||||
protected $append = [
|
||||
'number_type_text',
|
||||
'link_url_text',
|
||||
'status_text',
|
||||
'manual_manage_text',
|
||||
];
|
||||
|
||||
/**
|
||||
* 号码类型(与工单模块一致)
|
||||
*
|
||||
* @return array<string, string>
|
||||
*/
|
||||
public function getNumberTypeList(): array
|
||||
{
|
||||
return [
|
||||
'whatsapp' => 'WhatsApp',
|
||||
'telegram' => 'Telegram',
|
||||
'line' => 'Line',
|
||||
'custom' => __('Number type custom'),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, string>
|
||||
*/
|
||||
public function getStatusList(): array
|
||||
{
|
||||
return [
|
||||
'normal' => __('Status normal'),
|
||||
'hidden' => __('Status hidden'),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, string>
|
||||
*/
|
||||
public function getManualManageList(): array
|
||||
{
|
||||
return [
|
||||
'0' => __('Manual manage no'),
|
||||
'1' => __('Manual manage yes'),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* 关联分流链接
|
||||
*/
|
||||
public function splitLink()
|
||||
{
|
||||
return $this->belongsTo(Link::class, 'split_link_id', 'id', [], 'LEFT')->setEagerlyType(0);
|
||||
}
|
||||
|
||||
public function setNumberTypeCustomAttr($value): string
|
||||
{
|
||||
return trim((string) $value);
|
||||
}
|
||||
|
||||
public function setManualManageAttr($value): int
|
||||
{
|
||||
return (int) $value === 1 ? 1 : 0;
|
||||
}
|
||||
|
||||
public function getNumberTypeTextAttr($value, $data): string
|
||||
{
|
||||
$type = (string) ($data['number_type'] ?? '');
|
||||
if ($type === 'custom') {
|
||||
$custom = trim((string) ($data['number_type_custom'] ?? ''));
|
||||
return $custom !== '' ? $custom : (string) __('Number type custom');
|
||||
}
|
||||
$list = $this->getNumberTypeList();
|
||||
return $list[$type] ?? $type;
|
||||
}
|
||||
|
||||
/**
|
||||
* 列表展示用:仅显示分流链接码(非完整 URL)
|
||||
*/
|
||||
public function getLinkUrlTextAttr($value, $data): string
|
||||
{
|
||||
if (isset($data['split_link']) && is_array($data['split_link'])) {
|
||||
return (string) ($data['split_link']['link_code'] ?? '');
|
||||
}
|
||||
$linkId = (int) ($data['split_link_id'] ?? 0);
|
||||
if ($linkId <= 0) {
|
||||
return '';
|
||||
}
|
||||
return (string) Link::where('id', $linkId)->value('link_code');
|
||||
}
|
||||
|
||||
public function getStatusTextAttr($value, $data): string
|
||||
{
|
||||
$key = (string) ($data['status'] ?? '');
|
||||
$list = $this->getStatusList();
|
||||
return $list[$key] ?? $key;
|
||||
}
|
||||
|
||||
public function getManualManageTextAttr($value, $data): string
|
||||
{
|
||||
$key = (string) ((int) ($data['manual_manage'] ?? 0));
|
||||
$list = $this->getManualManageList();
|
||||
return $list[$key] ?? $key;
|
||||
}
|
||||
|
||||
/**
|
||||
* 根据链接码生成完整分流 URL
|
||||
*/
|
||||
public static function buildLinkUrl(string $linkCode): string
|
||||
{
|
||||
$linkCode = trim($linkCode);
|
||||
if ($linkCode === '') {
|
||||
return '';
|
||||
}
|
||||
$raw = trim((string) \think\Config::get('site.split_platform_domain'));
|
||||
if ($raw === '') {
|
||||
$raw = trim((string) Db::name('config')->where('name', 'split_platform_domain')->value('value'));
|
||||
}
|
||||
$domains = SplitPlatformDomainService::parseList($raw);
|
||||
$domain = $domains[0] ?? '';
|
||||
if ($domain === '') {
|
||||
return '';
|
||||
}
|
||||
return 'https://' . $domain . '/s/' . $linkCode;
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析 textarea 号码列表
|
||||
*
|
||||
* @return array<int, string>
|
||||
*/
|
||||
public static function parseNumbersText(string $text): array
|
||||
{
|
||||
$lines = preg_split('/\r\n|\r|\n/', $text) ?: [];
|
||||
$numbers = [];
|
||||
foreach ($lines as $line) {
|
||||
$num = trim((string) $line);
|
||||
if ($num === '') {
|
||||
continue;
|
||||
}
|
||||
$numbers[$num] = $num;
|
||||
}
|
||||
return array_values($numbers);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,296 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\model\split;
|
||||
|
||||
use think\Db;
|
||||
use think\Model;
|
||||
|
||||
/**
|
||||
* 分流工单模型
|
||||
*/
|
||||
class Ticket extends Model
|
||||
{
|
||||
protected $name = 'split_ticket';
|
||||
|
||||
protected $autoWriteTimestamp = 'integer';
|
||||
|
||||
protected $createTime = 'createtime';
|
||||
protected $updateTime = 'updatetime';
|
||||
protected $deleteTime = false;
|
||||
|
||||
protected $append = [
|
||||
'ticket_type_text',
|
||||
'number_type_text',
|
||||
'link_code_text',
|
||||
'start_time_text',
|
||||
'end_time_text',
|
||||
'status_text',
|
||||
'click_count',
|
||||
'ticket_progress_text',
|
||||
'inbound_ratio_text',
|
||||
'sync_display_text',
|
||||
];
|
||||
|
||||
/**
|
||||
* 工单类型
|
||||
*
|
||||
* @return array<string, string>
|
||||
*/
|
||||
public function getTicketTypeList(): array
|
||||
{
|
||||
return [
|
||||
'xinghe' => __('Ticket type xinghe'),
|
||||
'haiwang' => __('Ticket type haiwang'),
|
||||
'taiji' => __('Ticket type taiji'),
|
||||
'huojian' => __('Ticket type huojian'),
|
||||
'ss_channel' => __('Ticket type ss_channel'),
|
||||
'ss_customer' => __('Ticket type ss_customer'),
|
||||
'yifafa' => __('Ticket type yifafa'),
|
||||
'a2c' => __('Ticket type a2c'),
|
||||
'ceo_scrm' => __('Ticket type ceo_scrm'),
|
||||
'whatshub' => __('Ticket type whatshub'),
|
||||
'sihai' => __('Ticket type sihai'),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, string>
|
||||
*/
|
||||
public function getStatusList(): array
|
||||
{
|
||||
return [
|
||||
'normal' => __('Status normal'),
|
||||
'hidden' => __('Status hidden'),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, string>
|
||||
*/
|
||||
public function getSyncStatusList(): array
|
||||
{
|
||||
return [
|
||||
'success' => __('Sync status success'),
|
||||
'error' => __('Sync status error'),
|
||||
'pending' => __('Sync status pending'),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* 号码类型
|
||||
*
|
||||
* @return array<string, string>
|
||||
*/
|
||||
public function getNumberTypeList(): array
|
||||
{
|
||||
return [
|
||||
'whatsapp' => 'Whatsapp',
|
||||
'telegram' => 'Telegram',
|
||||
'line' => 'Line',
|
||||
'custom' => __('Number type custom'),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* 关联分流链接
|
||||
*/
|
||||
public function splitLink()
|
||||
{
|
||||
// IN 预载入:避免 eagerlyType=0(JOIN) 与列表 field 子查询冲突导致 SQL 1064
|
||||
return $this->belongsTo(Link::class, 'split_link_id', 'id', [], 'LEFT')->setEagerlyType(1);
|
||||
}
|
||||
|
||||
/**
|
||||
* 列表子查询注入的点击数,无则按关联号码汇总
|
||||
*
|
||||
* @param mixed $value
|
||||
* @param array<string, mixed> $data
|
||||
*/
|
||||
public function getClickCountAttr($value, $data): int
|
||||
{
|
||||
if (isset($data['click_count']) && $data['click_count'] !== '' && $data['click_count'] !== null) {
|
||||
return (int) $data['click_count'];
|
||||
}
|
||||
return self::sumVisitCountForTicket($data);
|
||||
}
|
||||
|
||||
/**
|
||||
* 工单进度:完成数量 / 工单总量
|
||||
*
|
||||
* @param mixed $value
|
||||
* @param array<string, mixed> $data
|
||||
*/
|
||||
public function getTicketProgressTextAttr($value, $data): string
|
||||
{
|
||||
$total = (int) ($data['ticket_total'] ?? 0);
|
||||
$complete = (int) ($data['complete_count'] ?? 0);
|
||||
if ($total <= 0) {
|
||||
return '—';
|
||||
}
|
||||
$percent = round($complete / $total * 100, 2);
|
||||
return $percent . '%';
|
||||
}
|
||||
|
||||
/**
|
||||
* 进线比例:进线人数 / 点击数(点击数=关联号码 visit_count 之和)
|
||||
*
|
||||
* @param mixed $value
|
||||
* @param array<string, mixed> $data
|
||||
*/
|
||||
public function getInboundRatioTextAttr($value, $data): string
|
||||
{
|
||||
$clicks = $this->getClickCountAttr(null, $data);
|
||||
$inbound = (int) ($data['inbound_count'] ?? 0);
|
||||
if ($clicks <= 0) {
|
||||
return '—';
|
||||
}
|
||||
$percent = round($inbound / $clicks * 100, 2);
|
||||
return $percent . '%';
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步状态展示文案
|
||||
*
|
||||
* @param mixed $value
|
||||
* @param array<string, mixed> $data
|
||||
*/
|
||||
public function getSyncDisplayTextAttr($value, $data): string
|
||||
{
|
||||
$status = (string) ($data['sync_status'] ?? 'pending');
|
||||
$online = (int) ($data['online_count'] ?? 0);
|
||||
if ($status === 'success') {
|
||||
return sprintf((string) __('Sync display success'), $online);
|
||||
}
|
||||
return (string) __('Sync display error');
|
||||
}
|
||||
|
||||
public function setStartTimeAttr($value): ?int
|
||||
{
|
||||
return self::parseTimeValue($value);
|
||||
}
|
||||
|
||||
public function setEndTimeAttr($value): ?int
|
||||
{
|
||||
return self::parseTimeValue($value);
|
||||
}
|
||||
|
||||
public function setNumberTypeCustomAttr($value): string
|
||||
{
|
||||
return trim((string) $value);
|
||||
}
|
||||
|
||||
public function setTicketTotalAttr($value): int
|
||||
{
|
||||
return max(0, (int) $value);
|
||||
}
|
||||
|
||||
public function getTicketTypeTextAttr($value, $data): string
|
||||
{
|
||||
$key = (string) ($data['ticket_type'] ?? '');
|
||||
$list = $this->getTicketTypeList();
|
||||
return $list[$key] ?? $key;
|
||||
}
|
||||
|
||||
public function getNumberTypeTextAttr($value, $data): string
|
||||
{
|
||||
$type = (string) ($data['number_type'] ?? '');
|
||||
if ($type === 'custom') {
|
||||
$custom = trim((string) ($data['number_type_custom'] ?? ''));
|
||||
return $custom !== '' ? $custom : (string) __('Number type custom');
|
||||
}
|
||||
$list = $this->getNumberTypeList();
|
||||
return $list[$type] ?? $type;
|
||||
}
|
||||
|
||||
public function getLinkCodeTextAttr($value, $data): string
|
||||
{
|
||||
if ($value !== '' && $value !== null) {
|
||||
return (string) $value;
|
||||
}
|
||||
$linkId = (int) ($data['split_link_id'] ?? 0);
|
||||
if ($linkId <= 0) {
|
||||
return '';
|
||||
}
|
||||
$code = Link::where('id', $linkId)->value('link_code');
|
||||
return (string) $code;
|
||||
}
|
||||
|
||||
public function getStartTimeTextAttr($value, $data): string
|
||||
{
|
||||
return self::formatTimeText($data['start_time'] ?? null);
|
||||
}
|
||||
|
||||
public function getEndTimeTextAttr($value, $data): string
|
||||
{
|
||||
return self::formatTimeText($data['end_time'] ?? null);
|
||||
}
|
||||
|
||||
public function getStatusTextAttr($value, $data): string
|
||||
{
|
||||
$key = (string) ($data['status'] ?? '');
|
||||
$list = $this->getStatusList();
|
||||
return $list[$key] ?? $key;
|
||||
}
|
||||
|
||||
/**
|
||||
* 构建列表用 click_count 子查询 SQL 片段
|
||||
*/
|
||||
public static function buildClickCountSubQuery(string $ticketTableAlias = ''): string
|
||||
{
|
||||
$ticketTable = (new self())->getTable();
|
||||
$numberTable = (new Number())->getTable();
|
||||
$t = $ticketTableAlias !== '' ? $ticketTableAlias : $ticketTable;
|
||||
return "(SELECT COALESCE(SUM(n.visit_count), 0) FROM `{$numberTable}` n "
|
||||
. "WHERE n.ticket_name = `{$t}`.ticket_name "
|
||||
. "AND n.split_link_id = `{$t}`.split_link_id "
|
||||
. "AND n.admin_id = `{$t}`.admin_id) AS click_count";
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array<string, mixed> $data
|
||||
*/
|
||||
private static function sumVisitCountForTicket(array $data): int
|
||||
{
|
||||
$ticketName = (string) ($data['ticket_name'] ?? '');
|
||||
$linkId = (int) ($data['split_link_id'] ?? 0);
|
||||
$adminId = (int) ($data['admin_id'] ?? 0);
|
||||
if ($ticketName === '' || $linkId <= 0) {
|
||||
return 0;
|
||||
}
|
||||
$sum = Db::name('split_number')
|
||||
->where('ticket_name', $ticketName)
|
||||
->where('split_link_id', $linkId)
|
||||
->where('admin_id', $adminId)
|
||||
->sum('visit_count');
|
||||
return (int) $sum;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
*/
|
||||
private static function parseTimeValue($value): ?int
|
||||
{
|
||||
if ($value === '' || $value === null) {
|
||||
return null;
|
||||
}
|
||||
if (is_numeric($value)) {
|
||||
return (int) $value;
|
||||
}
|
||||
$ts = strtotime((string) $value);
|
||||
return $ts === false ? null : $ts;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $value
|
||||
*/
|
||||
private static function formatTimeText($value): string
|
||||
{
|
||||
if ($value === '' || $value === null || !is_numeric($value)) {
|
||||
return '';
|
||||
}
|
||||
$ts = (int) $value;
|
||||
return $ts > 0 ? date('Y-m-d H:i:s', $ts) : '';
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,11 @@
|
||||
<form id="add-form" class="form-horizontal split-number-form" role="form" data-toggle="validator" method="POST" action="">
|
||||
{:token()}
|
||||
{include file="split/number/form_body" /}
|
||||
<div class="form-group layer-footer">
|
||||
<label class="control-label col-xs-12 col-sm-2"></label>
|
||||
<div class="col-xs-12 col-sm-8">
|
||||
<button type="submit" class="btn btn-primary btn-embossed disabled">{:__('OK')}</button>
|
||||
<button type="reset" class="btn btn-default btn-embossed">{:__('Reset')}</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
||||
@@ -0,0 +1,12 @@
|
||||
<form id="edit-form" class="form-horizontal split-number-form" role="form" data-toggle="validator" method="POST" action="">
|
||||
{:token()}
|
||||
<input type="hidden" name="row[id]" value="{$row.id|htmlentities}">
|
||||
{include file="split/number/form_body" /}
|
||||
<div class="form-group layer-footer">
|
||||
<label class="control-label col-xs-12 col-sm-2"></label>
|
||||
<div class="col-xs-12 col-sm-8">
|
||||
<button type="submit" class="btn btn-primary btn-embossed disabled">{:__('OK')}</button>
|
||||
<button type="reset" class="btn btn-default btn-embossed">{:__('Reset')}</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
||||
@@ -0,0 +1,146 @@
|
||||
<style>
|
||||
.split-number-form .panel {
|
||||
border-color: #e8ecf1;
|
||||
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.04);
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
.split-number-form .panel-heading {
|
||||
background: linear-gradient(180deg, #f8fafc 0%, #f1f5f9 100%);
|
||||
border-bottom: 1px solid #e2e8f0;
|
||||
color: #334155;
|
||||
font-weight: 600;
|
||||
font-size: 13px;
|
||||
padding: 10px 16px;
|
||||
}
|
||||
.split-number-form .panel-body {
|
||||
padding: 18px 20px 8px;
|
||||
}
|
||||
.split-number-form .form-group {
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
.split-number-form .control-label {
|
||||
color: #475569;
|
||||
font-weight: 500;
|
||||
font-size: 13px;
|
||||
margin-bottom: 6px;
|
||||
padding-top: 0;
|
||||
text-align: left;
|
||||
width: 100%;
|
||||
display: block;
|
||||
}
|
||||
.split-number-form .control-label .text-danger {
|
||||
margin-left: 2px;
|
||||
font-weight: 700;
|
||||
}
|
||||
.split-number-form .form-control {
|
||||
border-radius: 4px;
|
||||
border-color: #cbd5e1;
|
||||
box-shadow: none;
|
||||
width: 100%;
|
||||
}
|
||||
.split-number-form .st-grid-2 {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr;
|
||||
margin-bottom: 4px;
|
||||
width: 100%;
|
||||
}
|
||||
@media (min-width: 768px) {
|
||||
.split-number-form .st-grid-2 {
|
||||
grid-template-columns: 1fr 1fr;
|
||||
column-gap: 36px;
|
||||
}
|
||||
}
|
||||
.split-number-form .st-grid-1 {
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
.split-number-form .st-grid-cell {
|
||||
min-width: 0;
|
||||
}
|
||||
.split-number-form .radio-inline {
|
||||
margin-right: 18px;
|
||||
}
|
||||
.split-number-form > .layer-footer {
|
||||
margin: 12px 0 0;
|
||||
padding: 16px 20px;
|
||||
border-top: 1px solid #e2e8f0;
|
||||
background: #f8fafc;
|
||||
clear: both;
|
||||
}
|
||||
.split-number-form > .layer-footer > .control-label {
|
||||
display: none !important;
|
||||
}
|
||||
.split-number-form > .layer-footer > div[class*="col-"] {
|
||||
width: 100%;
|
||||
max-width: 100%;
|
||||
text-align: center;
|
||||
padding: 0;
|
||||
}
|
||||
.split-number-form > .layer-footer .btn {
|
||||
min-width: 100px;
|
||||
margin: 0 12px;
|
||||
}
|
||||
</style>
|
||||
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading">{:__('Section basic')}</div>
|
||||
<div class="panel-body">
|
||||
<div class="st-grid-2">
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-number_type" class="control-label">{:__('Number_type')}<span class="text-danger">*</span></label>
|
||||
<select id="c-number_type" name="row[number_type]" class="form-control selectpicker" data-rule="required" data-none-selected-text="请选择" title="请选择">
|
||||
{foreach name="numberTypeList" item="vo" key="key"}
|
||||
<option value="{$key|htmlentities}" {if isset($row) && $row.number_type==$key}selected{elseif !isset($row) && $key=='whatsapp'/}selected{/if}>{$vo|htmlentities}</option>
|
||||
{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-split_link_id" class="control-label">{:__('Split_link_id')}<span class="text-danger">*</span></label>
|
||||
<select id="c-split_link_id" name="row[split_link_id]" class="form-control selectpicker" data-live-search="true" data-rule="required" data-none-selected-text="请选择" title="请选择">
|
||||
<option value="">请选择</option>
|
||||
{foreach name="splitLinkList" item="link"}
|
||||
<option value="{$link.id|htmlentities}" {if isset($row) && $row.split_link_id==$link.id}selected{/if}>{$link.label|htmlentities}</option>
|
||||
{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-1 split-number-type-custom {if !isset($row) || $row.number_type!='custom'}hide{/if}">
|
||||
<div class="form-group">
|
||||
<label for="c-number_type_custom" class="control-label">{:__('Number_type_custom')}<span class="text-danger">*</span></label>
|
||||
<input id="c-number_type_custom" class="form-control" name="row[number_type_custom]" type="text" value="{$row.number_type_custom|default=''|htmlentities}" maxlength="50">
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-1">
|
||||
<div class="form-group">
|
||||
<label for="c-ticket_name" class="control-label">{:__('Ticket_name')}<span class="text-danger">*</span></label>
|
||||
<input id="c-ticket_name" data-rule="required" class="form-control" name="row[ticket_name]" type="text" value="{$row.ticket_name|default=''|htmlentities}" maxlength="100">
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-1">
|
||||
<div class="form-group">
|
||||
{if isset($row)}
|
||||
<label for="c-number" class="control-label">{:__('Number')}<span class="text-danger">*</span></label>
|
||||
<input id="c-number" data-rule="required" class="form-control" name="row[number]" type="text" value="{$row.number|default=''|htmlentities}" maxlength="50">
|
||||
{else}
|
||||
<label for="c-numbers" class="control-label">{:__('Numbers')}<span class="text-danger">*</span></label>
|
||||
<textarea id="c-numbers" data-rule="required" class="form-control" name="row[numbers]" rows="8" placeholder="{:__('Numbers placeholder')}"></textarea>
|
||||
{/if}
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-1">
|
||||
<div class="form-group">
|
||||
<label class="control-label">{:__('Status')}<span class="text-danger">*</span></label>
|
||||
<div>
|
||||
{foreach name="statusList" item="vo" key="key"}
|
||||
<label class="radio-inline">
|
||||
<input type="radio" name="row[status]" value="{$key|htmlentities}" data-rule="required" {if isset($row) && $row.status==$key}checked{elseif !isset($row) && $key=='normal'/}checked{/if}> {$vo|htmlentities}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -0,0 +1,31 @@
|
||||
<div class="panel panel-default panel-intro">
|
||||
<div class="panel-heading">
|
||||
{:build_heading(null,FALSE)}
|
||||
<ul class="nav nav-tabs" data-field="status">
|
||||
<li class="{:$Think.get.status === null ? 'active' : ''}"><a href="#t-all" data-value="" data-toggle="tab">{:__('All')}</a></li>
|
||||
{foreach name="statusList" item="vo" key="key"}
|
||||
<li class="{:$Think.get.status === (string)$key ? 'active' : ''}"><a href="#t-{$key}" data-value="{$key|htmlentities}" data-toggle="tab">{$vo|htmlentities}</a></li>
|
||||
{/foreach}
|
||||
</ul>
|
||||
</div>
|
||||
<div class="panel-body">
|
||||
<div id="myTabContent" class="tab-content">
|
||||
<div class="tab-pane fade active in" id="one">
|
||||
<div class="widget-body no-padding">
|
||||
<div id="toolbar" class="toolbar">
|
||||
<a href="javascript:;" class="btn btn-primary btn-refresh" title="{:__('Refresh')}"><i class="fa fa-refresh"></i></a>
|
||||
<a href="javascript:;" class="btn btn-success btn-add {:$auth->check('split.number/add')?'':'hide'}" title="{:__('Add')}"><i class="fa fa-plus"></i> {:__('Add')}</a>
|
||||
<a href="javascript:;" class="btn btn-success btn-edit btn-disabled disabled {:$auth->check('split.number/edit')?'':'hide'}" title="{:__('Edit')}"><i class="fa fa-pencil"></i> {:__('Edit')}</a>
|
||||
<a href="javascript:;" class="btn btn-danger btn-del btn-disabled disabled {:$auth->check('split.number/del')?'':'hide'}" title="{:__('Delete')}"><i class="fa fa-trash"></i> {:__('Delete')}</a>
|
||||
<a href="javascript:;" class="btn btn-warning btn-batch-update-status btn-disabled disabled {:$auth->check('split.number/batchupdate')?'':'hide'}" title="{:__('Batch update btn')}"><i class="fa fa-edit"></i> {:__('Batch update btn')}</a>
|
||||
</div>
|
||||
<table id="table" class="table table-striped table-bordered table-hover table-nowrap"
|
||||
data-operate-edit="{:$auth->check('split.number/edit')}"
|
||||
data-operate-del="{:$auth->check('split.number/del')}"
|
||||
width="100%">
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -0,0 +1,11 @@
|
||||
<form id="add-form" class="form-horizontal split-ticket-form" role="form" data-toggle="validator" method="POST" action="">
|
||||
{:token()}
|
||||
{include file="split/ticket/form_body" /}
|
||||
<div class="form-group layer-footer">
|
||||
<label class="control-label col-xs-12 col-sm-2"></label>
|
||||
<div class="col-xs-12 col-sm-8">
|
||||
<button type="submit" class="btn btn-primary btn-embossed disabled">{:__('OK')}</button>
|
||||
<button type="reset" class="btn btn-default btn-embossed">{:__('Reset')}</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
||||
@@ -0,0 +1,12 @@
|
||||
<form id="edit-form" class="form-horizontal split-ticket-form" role="form" data-toggle="validator" method="POST" action="">
|
||||
{:token()}
|
||||
<input type="hidden" name="row[id]" value="{$row.id|htmlentities}">
|
||||
{include file="split/ticket/form_body" /}
|
||||
<div class="form-group layer-footer">
|
||||
<label class="control-label col-xs-12 col-sm-2"></label>
|
||||
<div class="col-xs-12 col-sm-8">
|
||||
<button type="submit" class="btn btn-primary btn-embossed disabled">{:__('OK')}</button>
|
||||
<button type="reset" class="btn btn-default btn-embossed">{:__('Reset')}</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
||||
@@ -0,0 +1,239 @@
|
||||
<style>
|
||||
.split-ticket-form .panel {
|
||||
border-color: #e8ecf1;
|
||||
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.04);
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
.split-ticket-form .panel-heading {
|
||||
background: linear-gradient(180deg, #f8fafc 0%, #f1f5f9 100%);
|
||||
border-bottom: 1px solid #e2e8f0;
|
||||
color: #334155;
|
||||
font-weight: 600;
|
||||
font-size: 13px;
|
||||
padding: 10px 16px;
|
||||
}
|
||||
.split-ticket-form .panel-body {
|
||||
padding: 18px 20px 8px;
|
||||
}
|
||||
.split-ticket-form .form-group {
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
.split-ticket-form .control-label {
|
||||
color: #475569;
|
||||
font-weight: 500;
|
||||
font-size: 13px;
|
||||
margin-bottom: 6px;
|
||||
padding-top: 0;
|
||||
text-align: left;
|
||||
width: 100%;
|
||||
display: block;
|
||||
}
|
||||
.split-ticket-form .control-label .text-danger {
|
||||
margin-left: 2px;
|
||||
font-weight: 700;
|
||||
}
|
||||
.split-ticket-form .form-control {
|
||||
border-radius: 4px;
|
||||
border-color: #cbd5e1;
|
||||
box-shadow: none;
|
||||
width: 100%;
|
||||
}
|
||||
.split-ticket-form .form-control:focus {
|
||||
border-color: #3b82f6;
|
||||
box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.15);
|
||||
}
|
||||
/* 单列整行 */
|
||||
.split-ticket-form .st-grid-1 {
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
/* 双列:PC 端 column-gap 控制列间距(不依赖 Bootstrap col padding) */
|
||||
.split-ticket-form .st-grid-2 {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr;
|
||||
column-gap: 0;
|
||||
row-gap: 0;
|
||||
margin-bottom: 4px;
|
||||
width: 100%;
|
||||
}
|
||||
@media (min-width: 768px) {
|
||||
.split-ticket-form .st-grid-2 {
|
||||
grid-template-columns: 1fr 1fr;
|
||||
column-gap: 36px;
|
||||
}
|
||||
}
|
||||
.split-ticket-form .st-grid-cell {
|
||||
min-width: 0;
|
||||
}
|
||||
@media (max-width: 767px) {
|
||||
.split-ticket-form .st-grid-2 {
|
||||
row-gap: 0;
|
||||
}
|
||||
.split-ticket-form .st-grid-2 .st-grid-cell + .st-grid-cell .form-group {
|
||||
margin-top: 0;
|
||||
}
|
||||
}
|
||||
/* 底部按钮区(兼容 layer 弹窗 footer 迁移) */
|
||||
.split-ticket-form > .layer-footer {
|
||||
margin: 12px 0 0;
|
||||
padding: 16px 20px;
|
||||
border-top: 1px solid #e2e8f0;
|
||||
background: #f8fafc;
|
||||
clear: both;
|
||||
}
|
||||
.split-ticket-form > .layer-footer > .control-label {
|
||||
display: none !important;
|
||||
}
|
||||
.split-ticket-form > .layer-footer > div[class*="col-"] {
|
||||
width: 100%;
|
||||
max-width: 100%;
|
||||
flex: none;
|
||||
float: none;
|
||||
text-align: center;
|
||||
padding: 0;
|
||||
}
|
||||
.split-ticket-form > .layer-footer .btn {
|
||||
min-width: 100px;
|
||||
margin: 0 12px;
|
||||
padding: 8px 24px;
|
||||
font-size: 14px;
|
||||
border-radius: 4px;
|
||||
}
|
||||
.split-ticket-form > .layer-footer .btn-primary {
|
||||
background-color: #3b82f6;
|
||||
border-color: #2563eb;
|
||||
}
|
||||
.split-ticket-form > .layer-footer .btn-primary:hover,
|
||||
.split-ticket-form > .layer-footer .btn-primary:focus {
|
||||
background-color: #2563eb;
|
||||
border-color: #1d4ed8;
|
||||
}
|
||||
.split-ticket-form > .layer-footer .btn-default {
|
||||
background-color: #fff;
|
||||
border-color: #cbd5e1;
|
||||
color: #475569;
|
||||
}
|
||||
.split-ticket-form > .layer-footer .btn-default:hover {
|
||||
background-color: #f1f5f9;
|
||||
border-color: #94a3b8;
|
||||
}
|
||||
</style>
|
||||
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading">{:__('Section basic')}</div>
|
||||
<div class="panel-body">
|
||||
<div class="st-grid-2">
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-ticket_type" class="control-label">{:__('Ticket_type')}<span class="text-danger">*</span></label>
|
||||
<select id="c-ticket_type" name="row[ticket_type]" class="form-control selectpicker" data-rule="required" data-none-selected-text="请选择" title="请选择">
|
||||
<option value="">请选择</option>
|
||||
{foreach name="ticketTypeList" item="vo" key="key"}
|
||||
<option value="{$key|htmlentities}" {if isset($row) && $row.ticket_type==$key}selected{/if}>{$vo|htmlentities}</option>
|
||||
{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-ticket_name" class="control-label">{:__('Ticket_name')}<span class="text-danger">*</span></label>
|
||||
<input id="c-ticket_name" data-rule="required" class="form-control" name="row[ticket_name]" type="text" value="{$row.ticket_name|default=''|htmlentities}" maxlength="100">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-1">
|
||||
<div class="form-group">
|
||||
<label for="c-ticket_url" class="control-label">{:__('Ticket_url')}<span class="text-danger">*</span></label>
|
||||
<input id="c-ticket_url" data-rule="required" class="form-control" name="row[ticket_url]" type="text" value="{$row.ticket_url|default=''|htmlentities}" maxlength="1000" placeholder="https://">
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-1">
|
||||
<div class="form-group">
|
||||
<label for="c-ticket_total" class="control-label">{:__('Ticket_total')}<span class="text-danger">*</span></label>
|
||||
<input id="c-ticket_total" data-rule="required" class="form-control" name="row[ticket_total]" type="number" min="0" step="1" value="{$row.ticket_total|default='0'|htmlentities}">
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-2">
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-split_link_id" class="control-label">{:__('Split_link_id')}<span class="text-danger">*</span></label>
|
||||
<select id="c-split_link_id" name="row[split_link_id]" class="form-control selectpicker" data-live-search="true" data-rule="required" data-none-selected-text="请选择" title="请选择">
|
||||
<option value="">请选择</option>
|
||||
{foreach name="splitLinkList" item="link"}
|
||||
<option value="{$link.id|htmlentities}" {if isset($row) && $row.split_link_id==$link.id}selected{/if}>{$link.label|htmlentities}</option>
|
||||
{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-number_type" class="control-label">{:__('Number_type')}<span class="text-danger">*</span></label>
|
||||
<select id="c-number_type" name="row[number_type]" class="form-control selectpicker" data-rule="required">
|
||||
{foreach name="numberTypeList" item="vo" key="key"}
|
||||
<option value="{$key|htmlentities}" {if isset($row) && $row.number_type==$key}selected{elseif !isset($row) && $key=='whatsapp'/}selected{/if}>{$vo|htmlentities}</option>
|
||||
{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-1 split-number-type-custom {if !isset($row) || $row.number_type!='custom'}hide{/if}">
|
||||
<div class="form-group">
|
||||
<label for="c-number_type_custom" class="control-label">{:__('Number_type_custom')}<span class="text-danger split-required-star">*</span></label>
|
||||
<input id="c-number_type_custom" class="form-control" name="row[number_type_custom]" type="text" value="{$row.number_type_custom|default=''|htmlentities}" maxlength="50">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading">{:__('Section time rule')}</div>
|
||||
<div class="panel-body">
|
||||
<div class="st-grid-2">
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-start_time" class="control-label">{:__('Start_time')}<span class="text-danger">*</span></label>
|
||||
<input id="c-start_time" data-rule="required" class="form-control datetimepicker" data-date-format="YYYY-MM-DD HH:mm:ss" data-use-current="true" name="row[start_time]" type="text" value="{$row.start_time|default=''|htmlentities}">
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-end_time" class="control-label">{:__('End_time')}<span class="text-danger">*</span></label>
|
||||
<input id="c-end_time" data-rule="required" class="form-control datetimepicker" data-date-format="YYYY-MM-DD HH:mm:ss" data-use-current="true" name="row[end_time]" type="text" value="{$row.end_time|default=''|htmlentities}">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-2">
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-order_limit" class="control-label">{:__('Order_limit')}<span class="text-danger">*</span></label>
|
||||
<input id="c-order_limit" data-rule="required;integer;range(0~)" class="form-control" name="row[order_limit]" type="number" min="0" step="1" value="{$row.order_limit|default='0'|htmlentities}">
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-assign_ratio" class="control-label">{:__('Assign_ratio')}<span class="text-danger">*</span></label>
|
||||
<input id="c-assign_ratio" data-rule="required;integer;range(0~)" class="form-control" name="row[assign_ratio]" type="number" min="0" step="1" value="{$row.assign_ratio|default='0'|htmlentities}">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="panel panel-default">
|
||||
<div class="panel-heading">{:__('Section account')}</div>
|
||||
<div class="panel-body">
|
||||
<div class="st-grid-2">
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-account" class="control-label">{:__('Account')}</label>
|
||||
<input id="c-account" class="form-control" name="row[account]" type="text" value="{$row.account|default=''|htmlentities}" maxlength="50">
|
||||
</div>
|
||||
</div>
|
||||
<div class="st-grid-cell">
|
||||
<div class="form-group">
|
||||
<label for="c-password" class="control-label">{:__('Password')}</label>
|
||||
<input id="c-password" class="form-control" name="row[password]" type="text" value="{$row.password|default=''|htmlentities}" maxlength="50">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -0,0 +1,24 @@
|
||||
<div class="panel panel-default panel-intro">
|
||||
<div class="panel-heading">
|
||||
{:build_heading(null,FALSE)}
|
||||
</div>
|
||||
<div class="panel-body">
|
||||
<div id="myTabContent" class="tab-content">
|
||||
<div class="tab-pane fade active in" id="one">
|
||||
<div class="widget-body no-padding">
|
||||
<div id="toolbar" class="toolbar">
|
||||
<a href="javascript:;" class="btn btn-primary btn-refresh" title="{:__('Refresh')}"><i class="fa fa-refresh"></i></a>
|
||||
<a href="javascript:;" class="btn btn-success btn-add {:$auth->check('split.ticket/add')?'':'hide'}" title="{:__('Add')}"><i class="fa fa-plus"></i> {:__('Add')}</a>
|
||||
<a href="javascript:;" class="btn btn-success btn-edit btn-disabled disabled {:$auth->check('split.ticket/edit')?'':'hide'}" title="{:__('Edit')}"><i class="fa fa-pencil"></i> {:__('Edit')}</a>
|
||||
<a href="javascript:;" class="btn btn-danger btn-del btn-disabled disabled {:$auth->check('split.ticket/del')?'':'hide'}" title="{:__('Delete')}"><i class="fa fa-trash"></i> {:__('Delete')}</a>
|
||||
</div>
|
||||
<table id="table" class="table table-striped table-bordered table-hover table-nowrap"
|
||||
data-operate-edit="{:$auth->check('split.ticket/edit')}"
|
||||
data-operate-del="{:$auth->check('split.ticket/del')}"
|
||||
width="100%">
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
Reference in New Issue
Block a user