mirror of
https://github.com/crmeb/CRMEB.git
synced 2026-03-26 07:13:10 +00:00
867 lines
37 KiB
PHP
867 lines
37 KiB
PHP
<?php
|
||
|
||
|
||
// +----------------------------------------------------------------------
|
||
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||
// +----------------------------------------------------------------------
|
||
// | Copyright (c) 2016~2026 https://www.crmeb.com All rights reserved.
|
||
// +----------------------------------------------------------------------
|
||
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||
// +----------------------------------------------------------------------
|
||
// | Author: CRMEB Team <admin@crmeb.com>
|
||
// +----------------------------------------------------------------------
|
||
namespace app\services\diy;
|
||
|
||
use app\dao\diy\ThemeDao;
|
||
use app\services\BaseServices;
|
||
use crmeb\exceptions\AdminException;
|
||
use crmeb\exceptions\ApiException;
|
||
|
||
/**
|
||
* 主题服务类
|
||
*
|
||
* 功能概述:
|
||
* 负责系统主题的管理,包括主题的增删改查、导入导出、应用切换等功能。
|
||
* 提供了对首页、分类页、详情页、个人中心等页面数据的独立管理和组合使用能力。
|
||
*
|
||
* 主要功能:
|
||
* 1. 主题管理 - 主题列表查询、详情获取、创建与编辑
|
||
* 2. 主题应用 - 切换当前使用的主题,或单独应用某个主题的特定页面数据
|
||
* 3. 数据导入 - 支持导入外部主题配置数据
|
||
* 4. 资源管理 - 管理主题相关的图片、标题等资源
|
||
* 5. 版本控制 - 记录主题数据的更新时间和版本信息
|
||
*
|
||
* @package app\services\diy
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2025/12/18
|
||
*/
|
||
class ThemeServices extends BaseServices
|
||
{
|
||
/**
|
||
* 构造函数 - 初始化依赖
|
||
*
|
||
* 注入 ThemeDao 依赖,用于数据库操作。
|
||
*
|
||
* @param ThemeDao $dao 主题数据访问对象
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function __construct(ThemeDao $dao)
|
||
{
|
||
$this->dao = $dao;
|
||
}
|
||
|
||
/**
|
||
* 获取主题列表
|
||
*
|
||
* 功能概述:
|
||
* 根据传入的查询条件,分页获取主题列表数据,并对返回的数据进行格式化处理。
|
||
* 处理内容包括:时间戳转日期字符串、图片路径转完整URL、JSON数据解析等。
|
||
*
|
||
* @param array $where 查询条件数组
|
||
* @return array 包含列表数据 list 和总数 count 的数组
|
||
* @throws \think\db\exception\DataNotFoundException
|
||
* @throws \think\db\exception\DbException
|
||
* @throws \think\db\exception\ModelNotFoundException
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function getThemeList($where)
|
||
{
|
||
[$page, $limit] = $this->getPageValue();
|
||
$field = 'id,title,info,type,home_image,category_image,detail_image,user_image,theme_data,add_time,up_time,is_use,page_type';
|
||
$order = 'id desc';
|
||
if (($where['page_type'] ?? '') === 'all') {
|
||
unset($where['page_type']);
|
||
} else {
|
||
$where['page_type'] = 'theme';
|
||
}
|
||
$list = $this->dao->themeList($where, $field, $page, $limit, $order);
|
||
foreach ($list as &$item) {
|
||
if (isset($item['add_time'])) $item['add_time'] = date('Y-m-d H:i', $item['add_time']);
|
||
if (isset($item['up_time'])) $item['up_time'] = date('Y-m-d H:i', $item['up_time']);
|
||
if (isset($item['home_data_update_time'])) $item['home_data_update_time'] = date('Y-m-d H:i', $item['home_data_update_time']);
|
||
if (isset($item['category_data_update_time'])) $item['category_data_update_time'] = date('Y-m-d H:i', $item['category_data_update_time']);
|
||
if (isset($item['detail_data_update_time'])) $item['detail_data_update_time'] = date('Y-m-d H:i', $item['detail_data_update_time']);
|
||
if (isset($item['user_data_update_time'])) $item['user_data_update_time'] = date('Y-m-d H:i', $item['user_data_update_time']);
|
||
if (isset($item['theme_data_update_time'])) $item['theme_data_update_time'] = date('Y-m-d H:i', $item['theme_data_update_time']);
|
||
if (isset($item['type'])) $item['type'] = $item['type'] == 0 ? '自建主题' : '广场主题';
|
||
if (isset($item['theme_data'])) $item['theme_data'] = json_decode($item['theme_data'], true) ?? [];
|
||
$item['home_image'] = set_file_url($item['home_image']);
|
||
$item['category_image'] = set_file_url($item['category_image']);
|
||
$item['detail_image'] = set_file_url($item['detail_image']);
|
||
$item['user_image'] = set_file_url($item['user_image']);
|
||
}
|
||
$count = $this->dao->themeCount($where);
|
||
return compact('list', 'count');
|
||
}
|
||
|
||
/**
|
||
* 获取主题版本号
|
||
*
|
||
* 功能概述:
|
||
* 根据主题ID获取该主题的当前版本号。
|
||
* 如果ID为0,则获取当前正在使用的主题的版本号。
|
||
*
|
||
* @param int $id 主题ID,0表示当前使用的主题
|
||
* @return mixed 版本号字符串
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function getThemeVersion($id)
|
||
{
|
||
$where = $id == 0 ? ['is_use' => 1] : ['id' => $id];
|
||
return $this->dao->value($where, 'version');
|
||
}
|
||
|
||
/**
|
||
* 获取主题信息
|
||
*
|
||
* 功能概述:
|
||
* 根据主题ID和类型获取主题的详细信息。
|
||
* 支持获取全部信息或指定类型(如首页、分类页、详情页等)的数据。
|
||
* 对返回的数据进行必要的格式化和默认值填充。
|
||
*
|
||
* 返回数据结构:
|
||
* 根据 $type 不同返回不同结构:
|
||
* - 'all'/'base': 返回主题完整记录数组
|
||
* - 'home'/'detail'/'user'/'theme': 返回解析后的配置数组
|
||
* - 'category': 返回包含 status 的数组
|
||
*
|
||
* @param int $id 主题ID,0表示当前使用的主题
|
||
* @param string $type 数据类型:all, home, category, detail, user, theme, base
|
||
* @return array|int[]|mixed|\think\Model|null
|
||
* @throws \think\db\exception\DataNotFoundException
|
||
* @throws \think\db\exception\DbException
|
||
* @throws \think\db\exception\ModelNotFoundException
|
||
* @throws AdminException 数据不存在时抛出
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function getThemeInfo($id, $type = 'all')
|
||
{
|
||
$where = $id == 0 ? ['is_use' => 1] : ['id' => $id];
|
||
$info = $this->dao->get($where);
|
||
if (!$info) throw new AdminException('数据不存在');
|
||
$info = $info->toArray();
|
||
if ($type == 'home') {
|
||
return json_decode($info['home_data'], true) ?? [];
|
||
} elseif ($type == 'category') {
|
||
return ['status' => $info['category_data'] ?? 1];
|
||
} elseif ($type == 'detail') {
|
||
return json_decode($info['detail_data'], true) ?? [];
|
||
} elseif ($type == 'user') {
|
||
return json_decode($info['user_data'], true) ?? [];
|
||
} elseif ($type == 'theme') {
|
||
if ($info['theme_data'] == '' || $info['theme_data'] == null || $info['theme_data'] == 'null') {
|
||
$info['theme_data'] = '{"theme_color":"#E93323","gradient_color":"#FF7931","sub_color":"#FE960F","light_color":"rgba(233, 51, 35, 0.1)"}';
|
||
}
|
||
return json_decode($info['theme_data'], true) ?? [];
|
||
} elseif ($type == 'base') {
|
||
return ['id' => $info['id'], 'type' => $info['type'], 'title' => $info['title'], 'info' => $info['info']];
|
||
} else {
|
||
$info['home_data_update_time'] = date('Y-m-d H:i:s', $info['home_data_update_time']);
|
||
$info['category_data_update_time'] = date('Y-m-d H:i:s', $info['category_data_update_time']);
|
||
$info['detail_data_update_time'] = date('Y-m-d H:i:s', $info['detail_data_update_time']);
|
||
$info['user_data_update_time'] = date('Y-m-d H:i:s', $info['user_data_update_time']);
|
||
$info['theme_data_update_time'] = date('Y-m-d H:i:s', $info['theme_data_update_time']);
|
||
$info['add_time'] = date('Y-m-d H:i:s', $info['add_time']);
|
||
$info['up_time'] = date('Y-m-d H:i:s', $info['up_time']);
|
||
$ids = [$info['home_data_id'], $info['category_data_id'], $info['detail_data_id'], $info['user_data_id'], $info['theme_data_id']];
|
||
$titles = $this->dao->getColumn([['id', 'in', $ids]], 'title', 'id');
|
||
$info['home_data_id_title'] = $titles[$info['home_data_id']] ?? '';
|
||
$info['category_data_id_title'] = $titles[$info['category_data_id']] ?? '';
|
||
$info['detail_data_id_title'] = $titles[$info['detail_data_id']] ?? '';
|
||
$info['user_data_id_title'] = $titles[$info['user_data_id']] ?? '';
|
||
$info['theme_data_id_title'] = $titles[$info['theme_data_id']] ?? '';
|
||
return $info;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 保存主题数据
|
||
*
|
||
* 功能概述:
|
||
* 创建新主题或更新现有主题的数据。
|
||
* 支持从模板主题复制数据创建新主题。
|
||
* 根据不同的页面类型(home, category, detail, user, theme)处理相应的数据保存逻辑。
|
||
* 自动更新版本号和最后修改时间。
|
||
*
|
||
* @param int $id 主题主键,0 表示新增
|
||
* @param array $data 待保存数据,必须包含 type、value,可选 tid、title
|
||
* @return int 新增或更新后的主题 ID
|
||
* @throws AdminException 当指定 tid 但主题不存在时抛出
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function saveTheme($id, $data)
|
||
{
|
||
// 初始化待写入数组
|
||
$saveData = [];
|
||
|
||
// 如果指定了模板主题 ID(tid),则先复制其数据作为基础
|
||
if ($data['tid'] !== 0) {
|
||
// 查询模板主题
|
||
$tInfo = $this->dao->get($data['tid']);
|
||
if (!$tInfo) {
|
||
throw new AdminException('主题不存在');
|
||
}
|
||
// 将模板主题数据转为数组,并剔除主键 id,避免冲突
|
||
$saveData = $tInfo->toArray();
|
||
// 新主题默认未启用
|
||
$saveData['is_use'] = 0;
|
||
unset($saveData['id']);
|
||
}
|
||
|
||
// 如果传入了标题,则覆盖
|
||
if ($data['title'] != '') {
|
||
$saveData['title'] = $data['title'];
|
||
}
|
||
|
||
if ($id == 0) {
|
||
$type = 0;
|
||
$saveData['category_data'] = 1;
|
||
$saveData['category_data_update_time'] = time();
|
||
$saveData['category_image'] = '/statics/images/cate1.png';
|
||
} else {
|
||
$type = $this->dao->value(['id' => $id], 'type');
|
||
}
|
||
|
||
// 将传入的 value 统一转为 JSON 字符串
|
||
$value = json_encode($data['value']);
|
||
|
||
// 根据模块类型分别处理数据、预览图及更新时间
|
||
switch ($data['type']) {
|
||
case 'home':
|
||
// 首页
|
||
$saveData['home_data'] = $value;
|
||
$saveData['home_data_update_time'] = time();
|
||
// 自建主题需要同步写入默认数据
|
||
if ($type == 0) {
|
||
$saveData['home_default_data'] = $value;
|
||
}
|
||
break;
|
||
|
||
case 'category':
|
||
// 分类页
|
||
$saveData['category_data'] = $value;
|
||
$saveData['category_data_update_time'] = time();
|
||
// 根据 value 生成对应预览图路径
|
||
$saveData['category_image'] = '/statics/images/cate' . $value . '.png';
|
||
if ($type == 0) {
|
||
$saveData['category_default_data'] = $value;
|
||
$saveData['category_default_image'] = '/statics/images/cate' . $value . '.png';
|
||
}
|
||
break;
|
||
|
||
case 'detail':
|
||
// 详情页
|
||
$saveData['detail_data'] = $value;
|
||
$saveData['detail_data_update_time'] = time();
|
||
if ($type == 0) {
|
||
$saveData['detail_default_data'] = $value;
|
||
}
|
||
break;
|
||
|
||
case 'user':
|
||
// 用户中心
|
||
$saveData['user_data'] = $value;
|
||
$saveData['user_data_update_time'] = time();
|
||
if ($type == 0) {
|
||
$saveData['user_default_data'] = $value;
|
||
}
|
||
break;
|
||
|
||
case 'theme':
|
||
// 主题自身数据
|
||
$saveData['theme_data'] = $value;
|
||
$saveData['theme_data_update_time'] = time();
|
||
if ($type == 0) {
|
||
$saveData['theme_default_data'] = $value;
|
||
}
|
||
break;
|
||
}
|
||
|
||
// 每次保存都生成新的版本号
|
||
$saveData['version'] = uniqid();
|
||
|
||
// 新增 or 更新
|
||
if ($id) {
|
||
// 更新
|
||
$saveData['up_time'] = time();
|
||
$this->dao->update($id, $saveData);
|
||
} else {
|
||
// 新增
|
||
$saveData['page_type'] = $data['page_type'];
|
||
$saveData['add_time'] = time();
|
||
$saveData['up_time'] = time();
|
||
$id = $this->dao->insertGetId($saveData);
|
||
}
|
||
|
||
// 返回最终主题 ID
|
||
return $id;
|
||
}
|
||
|
||
/**
|
||
* 保存主题标题信息
|
||
*
|
||
* 功能概述:
|
||
* 更新主题的标题和简介信息,或创建新的主题记录(仅包含标题信息)。
|
||
* 更新操作会同步更新版本号和最后修改时间。
|
||
*
|
||
* @param int $id 主题ID,0表示新增
|
||
* @param array $data 包含 title 和 info 的数据数组
|
||
* @return int|mixed|string 主题ID
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function saveThemeTitle($id, $data)
|
||
{
|
||
// 如果指定了模板主题 ID(tid),则先复制其数据作为基础
|
||
if ($data['tid'] !== 0) {
|
||
// 查询模板主题
|
||
$tInfo = $this->dao->get($data['tid']);
|
||
if (!$tInfo) {
|
||
throw new AdminException('主题不存在');
|
||
}
|
||
// 将模板主题数据转为数组,并剔除主键 id,避免冲突
|
||
$saveData = $tInfo->toArray();
|
||
// 新主题默认未启用
|
||
$saveData['is_use'] = 0;
|
||
unset($saveData['id']);
|
||
}
|
||
$saveData['title'] = $data['title'];
|
||
$saveData['info'] = $data['info'];
|
||
$saveData['version'] = uniqid();
|
||
$saveData['page_type'] = $data['page_type'];
|
||
if ($id) {
|
||
$saveData['up_time'] = time();
|
||
$this->dao->update($id, $saveData);
|
||
} else {
|
||
$saveData['add_time'] = time();
|
||
$saveData['up_time'] = time();
|
||
$id = $this->dao->insertGetId($saveData);
|
||
}
|
||
return $id;
|
||
}
|
||
|
||
/**
|
||
* 保存主题图片信息
|
||
*
|
||
* 功能概述:
|
||
* 更新主题各模块(首页、详情页、用户中心)的预览图片。
|
||
* 如果是默认主题(type=0),会同步更新默认图片配置。
|
||
* 自动更新版本号和最后修改时间。
|
||
*
|
||
* @param int $id 主题ID
|
||
* @param array $data 包含 type (home/detail/user) 和 image 的数据数组
|
||
* @return int|mixed|string 主题ID
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2025/12/18
|
||
*/
|
||
public function saveThemeImage($id, $data)
|
||
{
|
||
$type = $id ? $this->dao->value(['id' => $id], 'type') : 0;
|
||
switch ($data['type']) {
|
||
case 'home':
|
||
$saveData['home_image'] = $data['image'];
|
||
if ($type == 0) $saveData['home_default_image'] = $data['image'];
|
||
break;
|
||
case 'detail':
|
||
$saveData['detail_image'] = $data['image'];
|
||
if ($type == 0) $saveData['detail_default_image'] = $data['image'];
|
||
break;
|
||
case 'user':
|
||
$saveData['user_image'] = $data['image'];
|
||
if ($type == 0) $saveData['user_default_image'] = $data['image'];
|
||
break;
|
||
}
|
||
$saveData['version'] = uniqid();
|
||
if ($id) {
|
||
$saveData['up_time'] = time();
|
||
$this->dao->update($id, $saveData);
|
||
} else {
|
||
$saveData['page_type'] = 'theme';
|
||
$saveData['add_time'] = time();
|
||
$saveData['up_time'] = time();
|
||
$id = $this->dao->insertGetId($saveData);
|
||
}
|
||
return $id;
|
||
}
|
||
|
||
/**
|
||
* 导入主题数据
|
||
*
|
||
* 功能概述:
|
||
* 将外部导入的主题配置数据保存到数据库中。
|
||
* 包含主题的所有页面配置(首页、分类、详情、个人中心)及其对应的默认配置。
|
||
*
|
||
* @param array $config 主题配置数据数组
|
||
* @return mixed 新增的主题ID
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function importThemeData($config)
|
||
{
|
||
$data = [];
|
||
$data['version'] = uniqid(); // 版本号
|
||
$data['title'] = $config['title']; // 标题
|
||
$data['info'] = $config['info']; // 简介
|
||
$data['type'] = 1; // 类型
|
||
$data['home_data'] = $data['home_default_data'] = $config['home_data']; // 首页数据
|
||
$data['home_image'] = $data['home_default_image'] = $config['home_image']; // 首页封面
|
||
$data['home_data_id'] = $config['home_data_id']; // 首页数据ID
|
||
$data['home_data_update_time'] = time(); // 首页数据更新时间
|
||
$data['category_data'] = $data['category_default_data'] = $config['category_data']; // 分类页数据
|
||
$data['category_image'] = $data['category_default_image'] = $config['category_image']; // 分类页封面
|
||
$data['category_data_id'] = $config['category_data_id']; // 分类页数据ID
|
||
$data['category_data_update_time'] = time(); // 分类页数据更新时间
|
||
$data['detail_data'] = $data['detail_default_data'] = $config['detail_data']; // 详情页数据
|
||
$data['detail_image'] = $data['detail_default_image'] = $config['detail_image']; // 详情页封面
|
||
$data['detail_data_id'] = $config['detail_data_id']; // 详情页数据ID
|
||
$data['detail_data_update_time'] = time(); // 详情页数据更新时间
|
||
$data['user_data'] = $data['user_default_data'] = $config['user_data']; // 个人中心数据
|
||
$data['user_image'] = $data['user_default_image'] = $config['user_image']; // 个人中心封面
|
||
$data['user_data_id'] = $config['user_data_id']; // 个人中心数据ID
|
||
$data['user_data_update_time'] = time(); // 个人中心数据更新时间
|
||
$data['theme_data'] = $data['theme_default_data'] = json_encode($config['theme_data'], JSON_UNESCAPED_UNICODE); // 主题数据
|
||
$data['theme_data_id'] = $config['theme_data_id']; // 主题数据ID
|
||
$data['theme_data_update_time'] = time(); // 主题数据更新时间
|
||
$data['page_type'] = 'theme'; // 页面类型
|
||
$data['is_use'] = 0; // 是否使用
|
||
$data['is_del'] = 0; // 是否删除
|
||
$data['add_time'] = time(); // 添加时间
|
||
$data['up_time'] = time(); // 更新时间
|
||
$id = $this->dao->insertGetId($data);
|
||
return $id;
|
||
}
|
||
|
||
/**
|
||
* 使用主题
|
||
*
|
||
* 功能概述:
|
||
* 将指定主题设置为当前启用状态。
|
||
* 该操作会先将所有主题设为未启用,然后启用指定ID的主题。
|
||
*
|
||
* @param int $id 要启用的主题ID
|
||
* @return bool 操作成功返回 true
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function useTheme(int $id)
|
||
{
|
||
$this->dao->update(['is_use' => 1], ['is_use' => 0]);
|
||
$this->dao->update($id, ['is_use' => 1]);
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* 使用主题数据
|
||
*
|
||
* 功能概述:
|
||
* 将某个主题的特定模块数据(如首页、详情页等)应用到目标主题数据记录中。
|
||
* 实现主题数据的局部复用或混搭。
|
||
*
|
||
* @param int $id 目标主题数据ID
|
||
* @param int $theme_id 源主题ID
|
||
* @param string $type 数据类型(home/category/detail/user/theme)
|
||
* @return bool 操作成功返回 true
|
||
* @throws AdminException 当源主题数据不存在时抛出
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function useThemeData(int $id, int $theme_id, string $type)
|
||
{
|
||
$data = $this->dao->get(['id' => $theme_id]);
|
||
if (!$data) throw new AdminException('主题数据不存在');
|
||
$this->dao->update(['id' => $id], [$type . '_data_update_time' => time(), $type . '_image' => $data[$type . '_image'], $type . '_data' => $data[$type . '_data'], $type . '_data_id' => $theme_id]);
|
||
return true;
|
||
}
|
||
|
||
|
||
/**
|
||
* 获取当前正在使用的主题信息
|
||
*
|
||
* 功能概述:
|
||
* 查询当前启用的主题(is_use=1),并聚合其关联的各模块(首页、分类、详情等)数据。
|
||
* 如果采用了混搭模式(引用了其他主题的模块),会解析出实际来源主题的标题和图片信息。
|
||
*
|
||
* 返回数据结构:
|
||
* - id, title, info, version: 主题基础信息
|
||
* - confuse: 是否混搭模式 (0/1)
|
||
* - data_info: 各模块详情列表(包含 key, title, image, update_time)
|
||
* - theme_data: 主题全局样式配置
|
||
*
|
||
* @return array 返回包含主题基础信息及各个模块详细配置的数据
|
||
* @throws AdminException 当没有正在使用的主题时抛出异常
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function getUsingTheme()
|
||
{
|
||
// 查询当前正在使用的主题记录(is_use = 1)
|
||
$data = $this->dao->get(['is_use' => 1]);
|
||
if (!$data) throw new AdminException('没有正在使用的主题');
|
||
|
||
// 收集各模块关联的主题ID,并过滤掉空值
|
||
$themeIds = array_filter([
|
||
$data['home_data_id'], // 首页模块关联主题ID
|
||
$data['category_data_id'], // 分类页模块关联主题ID
|
||
$data['detail_data_id'], // 详情页模块关联主题ID
|
||
$data['user_data_id'], // 用户中心模块关联主题ID
|
||
$data['theme_data_id'], // 主题自身数据关联主题ID
|
||
]);
|
||
|
||
|
||
// 组装最终返回的主题信息数组
|
||
$theme = [];
|
||
$theme['id'] = $data['id']; // 主题ID
|
||
$theme['title'] = $data['title']; // 主题名称
|
||
$theme['info'] = $data['info']; // 主题简介
|
||
$theme['version'] = $data['version']; // 主题版本号
|
||
$theme['confuse'] = 0; // 是否混搭使用主题:0否 1是
|
||
// 若存在关联主题ID,则批量查询其标题,供后续拼接使用
|
||
if ($themeIds) {
|
||
$themeData = $this->dao->getColumn([['id', 'in', $themeIds]], 'title', 'id');
|
||
$theme['confuse'] = 1;
|
||
}
|
||
$theme['data_info'] = [
|
||
[
|
||
'key' => 'home',
|
||
'title' => $themeData[$data['home_data_id']] ?? $data['title'], // 首页模块标题(优先取关联主题标题)
|
||
'image' => set_file_url($data['home_image']), // 首页预览图
|
||
'update_time' => date('Y-m-d H:i:s', $data['home_data_update_time']), // 首页数据更新时间
|
||
],
|
||
[
|
||
'key' => 'category',
|
||
'title' => $themeData[$data['category_data_id']] ?? $data['title'], // 分类页模块标题(优先取关联主题标题)
|
||
'image' => set_file_url($data['category_image']), // 分类页预览图
|
||
'update_time' => date('Y-m-d H:i:s', $data['category_data_update_time']), // 分类页数据更新时间
|
||
],
|
||
[
|
||
'key' => 'detail',
|
||
'title' => $themeData[$data['detail_data_id']] ?? $data['title'], // 详情页模块标题(优先取关联主题标题)
|
||
'image' => set_file_url($data['detail_image']), // 详情页预览图
|
||
'update_time' => date('Y-m-d H:i:s', $data['detail_data_update_time']), // 详情页数据更新时间
|
||
],
|
||
[
|
||
'key' => 'user',
|
||
'title' => $themeData[$data['user_data_id']] ?? $data['title'], // 用户中心模块标题(优先取关联主题标题)
|
||
'image' => set_file_url($data['user_image']), // 用户中心预览图
|
||
'update_time' => date('Y-m-d H:i:s', $data['user_data_update_time']), // 用户中心数据更新时间
|
||
],
|
||
];
|
||
$theme['theme_data'] = json_decode($data['theme_data'], true); // 主题自身数据(JSON格式)
|
||
|
||
return $theme;
|
||
}
|
||
|
||
/**
|
||
* @description: 还原主题
|
||
* @param int $id 主题ID
|
||
* @return void
|
||
*/
|
||
public function restoreTheme(int $id)
|
||
{
|
||
$data = $this->dao->get($id);
|
||
if (!$data) throw new AdminException('主题不存在');
|
||
$this->dao->update($id, [
|
||
'home_data' => $data['home_default_data'],
|
||
'home_data_id' => 0,
|
||
'home_image' => $data['home_default_image'],
|
||
'home_data_update_time' => time(),
|
||
'category_data' => $data['category_default_data'],
|
||
'category_data_id' => 0,
|
||
'category_image' => $data['category_default_image'],
|
||
'category_data_update_time' => time(),
|
||
'detail_data' => $data['detail_default_data'],
|
||
'detail_data_id' => 0,
|
||
'detail_image' => $data['detail_default_image'],
|
||
'detail_data_update_time' => time(),
|
||
'user_data' => $data['user_default_data'],
|
||
'user_data_id' => 0,
|
||
'user_image' => $data['user_default_image'],
|
||
'user_data_update_time' => time(),
|
||
'theme_data' => $data['theme_default_data'],
|
||
'theme_data_update_time' => time(),
|
||
'version' => uniqid(), // 更新版本号
|
||
'up_time' => time(), // 更新时间
|
||
]);
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* 删除主题
|
||
*
|
||
* 功能概述:
|
||
* 软删除指定的主题(更新 is_del 字段)。
|
||
*
|
||
* @param int $id 主题ID
|
||
* @return bool 操作成功返回 true
|
||
* @throws AdminException 当主题不存在时抛出
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function deleteTheme(int $id)
|
||
{
|
||
$data = $this->dao->get($id);
|
||
if (!$data) throw new AdminException('主题不存在');
|
||
if ($data['is_use']) throw new AdminException('当前主题正在使用中,不能删除');
|
||
$this->dao->update($id, ['is_del' => 1]);
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* 获取当前启用主题的底部导航配置
|
||
*
|
||
* 功能概述:
|
||
* 解析当前启用主题的首页数据,提取其中的底部导航(pagefoot)组件配置。
|
||
*
|
||
* @return array 返回名为 pagefoot 的组件配置数组,未找到时返回空数组
|
||
* @throws ApiException 当启用主题不存在首页数据时抛出
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function themeNavigation()
|
||
{
|
||
// 查询当前正在使用的主题的首页数据(JSON 字符串)
|
||
$value = $this->dao->value(['is_use' => 1], 'home_data');
|
||
if (!$value) {
|
||
throw new ApiException('数据不存在');
|
||
}
|
||
|
||
// 初始化导航数据为空数组
|
||
$navigation = [];
|
||
|
||
// 若首页数据存在,则进行解析与遍历
|
||
if ($value) {
|
||
// 将 JSON 字符串解码为数组
|
||
$value = json_decode($value, true);
|
||
// 遍历首页组件,查找名称为 pagefoot 的底部导航组件
|
||
foreach ($value['value'] as $item) {
|
||
if (isset($item['name']) && strtolower($item['name']) === 'pagefoot') {
|
||
// 找到后赋值并终止循环
|
||
$navigation = $item;
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 返回导航配置(可能为空数组)
|
||
return $navigation;
|
||
}
|
||
|
||
/**
|
||
* 获取微页面列表
|
||
*
|
||
* 功能概述:
|
||
* 分页查询微页面(page_type='micro')列表数据。
|
||
*
|
||
* 主要功能:
|
||
* 1. 分页查询 - 根据系统分页参数获取数据
|
||
* 2. 数据过滤 - 仅查询未删除且类型为微页面的记录
|
||
* 3. 格式化 - 转换时间戳为可读日期格式
|
||
*
|
||
* @return array 包含列表数据 list 和总数 count 的数组
|
||
* @throws \think\db\exception\DataNotFoundException
|
||
* @throws \think\db\exception\DbException
|
||
* @throws \think\db\exception\ModelNotFoundException
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/02/03
|
||
*/
|
||
public function getMicroPageList()
|
||
{
|
||
[$page, $limit] = $this->getPageValue(); // 获取分页参数
|
||
$field = 'id,title,info,type,add_time,up_time,page_type'; // 查询字段
|
||
$order = 'id desc'; // 排序
|
||
$where = [
|
||
'is_del' => 0, // 未删除
|
||
'page_type' => 'micro', // 微页面类型
|
||
];
|
||
$list = $this->dao->themeList($where, $field, $page, $limit, $order); // 查询列表
|
||
foreach ($list as &$item) {
|
||
// 格式化时间
|
||
if (isset($item['add_time'])) $item['add_time'] = date('Y-m-d H:i', $item['add_time']);
|
||
if (isset($item['up_time'])) $item['up_time'] = date('Y-m-d H:i', $item['up_time']);
|
||
}
|
||
$count = $this->dao->themeCount($where); // 获取总数
|
||
return compact('list', 'count');
|
||
}
|
||
|
||
/**
|
||
* 导出主题数据(核心逻辑)
|
||
* 将主题配置及相关图片打包成 Zip 文件,返回下载地址
|
||
*
|
||
* @param $themeInfo
|
||
* @return string 下载地址
|
||
* @throws hinkdbexceptionDataNotFoundException
|
||
* @throws hinkdbexceptionDbException
|
||
* @throws hinkdbexceptionModelNotFoundException
|
||
* @author wuhaotian
|
||
* @email 442384644@qq.com
|
||
* @date 2026/3/10
|
||
*/
|
||
public function exportThemePackage($info): string
|
||
{
|
||
// 1. 设置导出临时目录
|
||
$dir = public_path() . 'theme/download/' . $info['id'] . '/';
|
||
|
||
// 2. 处理主要图片(首页图、分类图、详情图、个人中心图)
|
||
$images = ['home_image', 'category_image', 'detail_image', 'user_image'];
|
||
$defaultImages = ['home_default_image', 'category_default_image', 'detail_default_image', 'user_default_image'];
|
||
$i = 1;
|
||
foreach ($images as $key => $image) {
|
||
if (isset($info[$image]) && $info[$image]) {
|
||
$originalUrl = $info[$image];
|
||
$isRemote = (bool)preg_match('/^https?:\/\//i', $originalUrl);
|
||
if ($isRemote) {
|
||
$urlPath = parse_url($originalUrl, PHP_URL_PATH) ?: $originalUrl;
|
||
$extension = strtolower(pathinfo($urlPath, PATHINFO_EXTENSION)) ?: 'jpg';
|
||
$newPath = $dir . $i . '_' . $image . '.' . $extension;
|
||
$ch = curl_init();
|
||
curl_setopt_array($ch, [
|
||
CURLOPT_URL => $originalUrl,
|
||
CURLOPT_RETURNTRANSFER => true,
|
||
CURLOPT_FOLLOWLOCATION => true,
|
||
CURLOPT_TIMEOUT => 30,
|
||
CURLOPT_SSL_VERIFYPEER => false,
|
||
CURLOPT_SSL_VERIFYHOST => false,
|
||
CURLOPT_REFERER => rtrim(sys_config('site_url'), '/'),
|
||
CURLOPT_USERAGENT => 'Mozilla/5.0 (compatible; CRMEB/1.0)',
|
||
CURLOPT_HTTPHEADER => ['Accept: image/webp,image/*,*/*'],
|
||
]);
|
||
$content = curl_exec($ch);
|
||
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||
curl_close($ch);
|
||
if ($content && $httpCode === 200 && file_put_contents($newPath, $content) !== false) {
|
||
$info[$image] = 'theme/download/' . $i . '_' . $image . '.' . $extension;
|
||
$info[$defaultImages[$key]] = 'theme/download/' . $i . '_' . $image . '.' . $extension;
|
||
}
|
||
} else {
|
||
$localPath = public_path() . ltrim(preg_replace('/^https?:\/\/[^\/]+/', '', $originalUrl), '/');
|
||
if (!file_exists($localPath)) {
|
||
$i++;
|
||
continue;
|
||
}
|
||
$extension = strtolower(pathinfo($localPath, PATHINFO_EXTENSION)) ?: 'jpg';
|
||
$newPath = $dir . $i . '_' . $image . '.' . $extension;
|
||
if (copy($localPath, $newPath)) {
|
||
$info[$image] = 'theme/download/' . $i . '_' . $image . '.' . $extension;
|
||
$info[$defaultImages[$key]] = 'theme/download/' . $i . '_' . $image . '.' . $extension;
|
||
}
|
||
}
|
||
}
|
||
$i++;
|
||
}
|
||
|
||
// 3. 处理 home_data / detail_data / user_data 中的图片
|
||
$imagesDir = $dir . 'images/';
|
||
$index = 1;
|
||
$map = [];
|
||
$isAttachImage = function ($str) {
|
||
if (!is_string($str) || $str === '') return false;
|
||
if (strpos($str, 'uploads/attach') !== false) return true;
|
||
if (preg_match('/^https?:\/\//i', $str)) return true;
|
||
return false;
|
||
};
|
||
|
||
$process = function (&$value, $key) use (&$index, &$map, $imagesDir, $isAttachImage) {
|
||
if (!is_string($value) || !$isAttachImage($value)) return;
|
||
if (!isset($map[$value])) {
|
||
$path = parse_url($value, PHP_URL_PATH);
|
||
$path = $path ?: $value;
|
||
$basename = basename($path);
|
||
$ext = strtolower(pathinfo($basename, PATHINFO_EXTENSION));
|
||
if (!in_array($ext, ['jpg', 'jpeg', 'png', 'gif', 'webp', 'svg', 'bmp'])) {
|
||
$ext = 'jpg';
|
||
$basename = md5($value) . '.' . $ext;
|
||
}
|
||
$dest = $imagesDir . $basename;
|
||
$rel = 'images/' . $basename;
|
||
$ok = false;
|
||
$isRemote = preg_match('/^https?:\/\//i', $value);
|
||
if ($isRemote) {
|
||
$ch = curl_init();
|
||
curl_setopt_array($ch, [
|
||
CURLOPT_URL => $value,
|
||
CURLOPT_RETURNTRANSFER => true,
|
||
CURLOPT_FOLLOWLOCATION => true,
|
||
CURLOPT_TIMEOUT => 30,
|
||
CURLOPT_SSL_VERIFYPEER => false,
|
||
CURLOPT_SSL_VERIFYHOST => false,
|
||
CURLOPT_REFERER => rtrim(sys_config('site_url'), '/'),
|
||
CURLOPT_USERAGENT => 'Mozilla/5.0 (compatible; CRMEB/1.0)',
|
||
CURLOPT_HTTPHEADER => ['Accept: image/webp,image/*,*/*'],
|
||
]);
|
||
$content = curl_exec($ch);
|
||
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||
curl_close($ch);
|
||
if ($content && $httpCode === 200) {
|
||
$ok = (file_put_contents($dest, $content) !== false);
|
||
}
|
||
} else {
|
||
$src = public_path() . ltrim($path, '/');
|
||
if (file_exists($src)) $ok = @copy($src, $dest);
|
||
}
|
||
if ($ok) {
|
||
$map[$value] = $rel;
|
||
$index++;
|
||
} else {
|
||
return;
|
||
}
|
||
}
|
||
$value = $map[$value] ?? $value;
|
||
};
|
||
|
||
$homeData = json_decode($info['home_data'] ?? '[]', true);
|
||
$detailData = json_decode($info['detail_data'] ?? '[]', true);
|
||
$userData = json_decode($info['user_data'] ?? '[]', true);
|
||
|
||
if (is_array($homeData)) array_walk_recursive($homeData, $process);
|
||
if (is_array($detailData)) array_walk_recursive($detailData, $process);
|
||
if (is_array($userData)) array_walk_recursive($userData, $process);
|
||
|
||
$info['home_data'] = json_encode($homeData, JSON_UNESCAPED_UNICODE);
|
||
$info['detail_data'] = json_encode($detailData, JSON_UNESCAPED_UNICODE);
|
||
$info['user_data'] = json_encode($userData, JSON_UNESCAPED_UNICODE);
|
||
$info['home_default_data'] = json_encode($homeData, JSON_UNESCAPED_UNICODE);
|
||
$info['detail_default_data'] = json_encode($detailData, JSON_UNESCAPED_UNICODE);
|
||
$info['user_default_data'] = json_encode($userData, JSON_UNESCAPED_UNICODE);
|
||
$info['theme_data'] = $info['theme_default_data'] = json_decode($info['theme_data'], true);
|
||
|
||
// 4. 写入 config.json
|
||
file_put_contents($dir . 'config.json', json_encode($info, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE));
|
||
|
||
// 7. 打包成 zip
|
||
$zip = new \ZipArchive();
|
||
$zip->open($dir . $info['title'] . '.zip', \ZipArchive::CREATE | \ZipArchive::OVERWRITE);
|
||
$rootPath = realpath($dir);
|
||
$files = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($dir, \FilesystemIterator::SKIP_DOTS));
|
||
foreach ($files as $file) {
|
||
if ($file->isDir()) continue;
|
||
$filePath = $file->getRealPath();
|
||
if (basename($filePath) === $info['title'] . '.zip') continue;
|
||
$relativePath = ltrim(str_replace($rootPath, '', $filePath), DIRECTORY_SEPARATOR);
|
||
$zip->addFile($filePath, $relativePath);
|
||
}
|
||
$zip->close();
|
||
|
||
return sys_config('site_url') . '/theme/download/' . $info['id'] . '/' . $info['title'] . '.zip';
|
||
}
|
||
}
|