518 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			518 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | ||
| /**
 | ||
|  * 商业使用授权协议
 | ||
|  * 
 | ||
|  * Copyright (c) 2025 [云泽网]. 保留所有权利.
 | ||
|  * 
 | ||
|  * 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
 | ||
|  * 未经授权商业使用本软件属于侵权行为,将承担法律责任。
 | ||
|  * 
 | ||
|  * 授权购买请联系: 357099073@qq.com
 | ||
|  * 官方网站: https://www.yunzer.cn
 | ||
|  * 
 | ||
|  * 评估用户须知:
 | ||
|  * 1. 禁止移除版权声明
 | ||
|  * 2. 禁止用于生产环境
 | ||
|  * 3. 禁止转售或分发
 | ||
|  */
 | ||
| 
 | ||
| /**
 | ||
|  *	后台管理系统-首页
 | ||
|  */
 | ||
| namespace app\admin\controller;
 | ||
| use app\admin\controller\Base;
 | ||
| use app\admin\model\DailyStats;
 | ||
| use app\admin\model\Log\LogsOperation;
 | ||
| use app\index\model\Attachments;
 | ||
| use think\facade\Db;
 | ||
| use think\facade\View;
 | ||
| use think\facade\Env;
 | ||
| use think\facade\Config;
 | ||
| use app\admin\controller\Log;
 | ||
| use \think\facade\Filesystem;
 | ||
| 
 | ||
| use app\admin\model\AdminUserGroup;
 | ||
| use app\admin\model\AdminSysMenu;
 | ||
| 
 | ||
| class IndexController extends Base{
 | ||
| 	# 首页
 | ||
| 	public function index(){
 | ||
| 		$menus = [];
 | ||
| 		$menu = [];
 | ||
| 		$where = ['group_id'=>$this->aUser['group_id']];
 | ||
| 		$role = AdminUserGroup::where($where)->find();
 | ||
| 		if($role){
 | ||
| 			$role['rights'] = (isset($role['rights']) && $role['rights']) ? json_decode($role['rights'],true) : [];
 | ||
| 		}
 | ||
| 		if($role['rights']){
 | ||
| 			$where = [
 | ||
| 				['smid','in',implode(',',$role['rights']) ],
 | ||
| 				['status','=',1]
 | ||
| 			];
 | ||
| 			// 获取所有菜单
 | ||
| 			$menus = AdminSysMenu::order('type,sort desc')->where($where)->select()->toArray();
 | ||
| 			
 | ||
| 			// 构建树形结构菜单
 | ||
| 			$menuTree = [];
 | ||
| 			$menuMap = [];
 | ||
| 			
 | ||
| 			// 先将所有菜单项映射到一个关联数组中
 | ||
| 			foreach($menus as $item){
 | ||
| 				$item['children'] = [];
 | ||
| 				$menuMap[$item['smid']] = $item;
 | ||
| 			}
 | ||
| 			
 | ||
| 			// 构建树形结构
 | ||
| 			foreach($menus as $item){
 | ||
| 				if($item['parent_id'] == 0){
 | ||
| 					// 顶级菜单
 | ||
| 					$menuTree[$item['smid']] = &$menuMap[$item['smid']];
 | ||
| 				}else{
 | ||
| 					// 子菜单,添加到父菜单的children数组中
 | ||
| 					if(isset($menuMap[$item['parent_id']])){
 | ||
| 						$menuMap[$item['parent_id']]['children'][] = &$menuMap[$item['smid']];
 | ||
| 					}
 | ||
| 				}
 | ||
| 			}
 | ||
| 			
 | ||
| 			$menu = $menuTree;
 | ||
| 		}
 | ||
| 		
 | ||
| 		View::assign([
 | ||
| 			'role' => $role,
 | ||
| 			'menu' => $menu
 | ||
| 		]);
 | ||
| 		return View::fetch();
 | ||
| 	}
 | ||
| 	# 欢迎页面
 | ||
| 	public function welcome(){
 | ||
| 		try {
 | ||
| 			// 获取最近7天的日期
 | ||
| 			$dates = [];
 | ||
| 			for ($i = 6; $i >= 0; $i--) {
 | ||
| 				$dates[] = date('Y-m-d', strtotime("-$i day"));
 | ||
| 			}
 | ||
| 
 | ||
| 			// 初始化数据数组
 | ||
| 			$visitData = [];
 | ||
| 			$userData = [];
 | ||
| 			$resourceData = [];
 | ||
| 			$articleData = [];
 | ||
| 
 | ||
| 			// 直接查询每天的数据
 | ||
| 			foreach ($dates as $date) {
 | ||
| 				$dayStats = Db::name('daily_stats')
 | ||
| 					->where('date', $date)
 | ||
| 					->find();
 | ||
| 
 | ||
| 				// 访问数据
 | ||
| 				$visitData[] = [
 | ||
| 					'date' => $date,
 | ||
| 					'visits' => $dayStats ? intval($dayStats['daily_visits']) : 0,
 | ||
| 					'uv' => $dayStats ? intval($dayStats['unique_visitors']) : 0
 | ||
| 				];
 | ||
| 
 | ||
| 				// 用户数据
 | ||
| 				$userData[] = [
 | ||
| 					'date' => $date,
 | ||
| 					'total' => $dayStats ? intval($dayStats['total_users']) : 0,
 | ||
| 					'new' => $dayStats ? intval($dayStats['new_users']) : 0
 | ||
| 				];
 | ||
| 
 | ||
| 				// 资源数据
 | ||
| 				$resourceData[] = [
 | ||
| 					'date' => $date,
 | ||
| 					'total' => $dayStats ? intval($dayStats['total_resources']) : 0,
 | ||
| 					'new' => $dayStats ? intval($dayStats['daily_resources']) : 0,
 | ||
| 					'downloads' => $dayStats ? intval($dayStats['resource_downloads']) : 0
 | ||
| 				];
 | ||
| 
 | ||
| 				// 文章数据
 | ||
| 				$articleData[] = [
 | ||
| 					'date' => $date,
 | ||
| 					'total' => $dayStats ? intval($dayStats['total_articles']) : 0,
 | ||
| 					'new' => $dayStats ? intval($dayStats['daily_articles']) : 0,
 | ||
| 					'views' => $dayStats ? intval($dayStats['article_views']) : 0
 | ||
| 				];
 | ||
| 			}
 | ||
| 
 | ||
| 			// 获取今日统计数据
 | ||
| 			$today = date('Y-m-d');
 | ||
| 			$todayStats = Db::name('daily_stats')
 | ||
| 				->where('date', $today)
 | ||
| 				->find();
 | ||
| 
 | ||
| 			// 获取最近的操作日志
 | ||
| 			$recentActivities = Db::name('logs_operation')
 | ||
| 				->field('operation_time, module, operation')
 | ||
| 				->order('operation_time DESC')
 | ||
| 				->limit(5)
 | ||
| 				->select()
 | ||
| 				->each(function($item) {
 | ||
| 					$item['content'] = date('Y年m月d日 H:i:s', strtotime($item['operation_time'])) . ' 在 ' . 
 | ||
| 						($item['module'] ?: '未知模块') . ' ' . 
 | ||
| 						($item['operation'] ?: '未知操作');
 | ||
| 					$item['icon'] = $this->getActivityIcon($item['module'] ?: '其他');
 | ||
| 					return $item;
 | ||
| 				});
 | ||
| 
 | ||
| 			// 处理图表数据
 | ||
| 			$chartData = [
 | ||
| 				'visitTrend' => [
 | ||
| 					'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $visitData),
 | ||
| 					'visits' => array_column($visitData, 'visits'),
 | ||
| 					'uvs' => array_column($visitData, 'uv')
 | ||
| 				],
 | ||
| 				'userGrowth' => [
 | ||
| 					'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $userData),
 | ||
| 					'newUsers' => array_column($userData, 'new'),
 | ||
| 					'totalUsers' => array_column($userData, 'total')
 | ||
| 				],
 | ||
| 				'resourceStats' => [
 | ||
| 					'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $resourceData),
 | ||
| 					'newResources' => array_column($resourceData, 'new'),
 | ||
| 					'totalResources' => array_column($resourceData, 'total'),
 | ||
| 					'downloads' => array_column($resourceData, 'downloads')
 | ||
| 				],
 | ||
| 				'articleStats' => [
 | ||
| 					'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $articleData),
 | ||
| 					'newArticles' => array_column($articleData, 'new'),
 | ||
| 					'totalArticles' => array_column($articleData, 'total'),
 | ||
| 					'views' => array_column($articleData, 'views')
 | ||
| 				]
 | ||
| 			];
 | ||
| 
 | ||
| 			// 传递给视图
 | ||
| 			View::assign([
 | ||
| 				'todayStats' => $todayStats ?: [
 | ||
| 					'total_users' => 0,
 | ||
| 					'new_users' => 0,
 | ||
| 					'total_visits' => 0,
 | ||
| 					'daily_visits' => 0,
 | ||
| 					'unique_visitors' => 0,
 | ||
| 					'total_articles' => 0,
 | ||
| 					'daily_articles' => 0,
 | ||
| 					'article_views' => 0,
 | ||
| 					'total_resources' => 0,
 | ||
| 					'daily_resources' => 0,
 | ||
| 					'resource_downloads' => 0
 | ||
| 				],
 | ||
| 				'recentActivities' => $recentActivities,
 | ||
| 				'chartData' => $chartData
 | ||
| 			]);
 | ||
| 
 | ||
| 			return View::fetch();
 | ||
| 		} catch (\Exception $e) {
 | ||
| 			// 记录错误日志
 | ||
| 			\think\facade\Log::error('获取统计数据失败:' . $e->getMessage());
 | ||
| 			// 返回空数据
 | ||
| 			View::assign([
 | ||
| 				'todayStats' => [
 | ||
| 					'total_users' => 0,
 | ||
| 					'new_users' => 0,
 | ||
| 					'total_visits' => 0,
 | ||
| 					'daily_visits' => 0,
 | ||
| 					'unique_visitors' => 0,
 | ||
| 					'total_articles' => 0,
 | ||
| 					'daily_articles' => 0,
 | ||
| 					'article_views' => 0,
 | ||
| 					'total_resources' => 0,
 | ||
| 					'daily_resources' => 0,
 | ||
| 					'resource_downloads' => 0
 | ||
| 				],
 | ||
| 				'recentActivities' => [],
 | ||
| 				'chartData' => [
 | ||
| 					'visitTrend' => ['dates' => [], 'visits' => [], 'uvs' => []],
 | ||
| 					'userGrowth' => ['dates' => [], 'newUsers' => [], 'totalUsers' => []],
 | ||
| 					'resourceStats' => ['dates' => [], 'newResources' => [], 'totalResources' => [], 'downloads' => []],
 | ||
| 					'articleStats' => ['dates' => [], 'newArticles' => [], 'totalArticles' => [], 'views' => []]
 | ||
| 				]
 | ||
| 			]);
 | ||
| 			return View::fetch();
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * 根据操作类型获取对应的图标
 | ||
| 	 */
 | ||
| 	private function getActivityIcon($type)
 | ||
| 	{
 | ||
| 		$icons = [
 | ||
| 			'用户管理' => '👥',
 | ||
| 			'文章管理' => '📝',
 | ||
| 			'资源管理' => '📦',
 | ||
| 			'系统设置' => '⚙️',
 | ||
| 			'登录' => '🔑',
 | ||
| 			'退出' => '🚪',
 | ||
| 			'其他' => '📌'
 | ||
| 		];
 | ||
| 		
 | ||
| 		return $icons[$type] ?? '📌';
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * 格式化访问趋势数据
 | ||
| 	 */
 | ||
| 	private function formatVisitTrendData($data)
 | ||
| 	{
 | ||
| 		$dates = [];
 | ||
| 		$visits = [];
 | ||
| 		$uvs = [];
 | ||
| 
 | ||
| 		foreach ($data as $item) {
 | ||
| 			$dates[] = date('m-d', strtotime($item['date']));
 | ||
| 			$visits[] = $item['daily_visits'];
 | ||
| 			$uvs[] = $item['unique_visitors'];
 | ||
| 		}
 | ||
| 
 | ||
| 		return [
 | ||
| 			'dates' => $dates,
 | ||
| 			'visits' => $visits,
 | ||
| 			'uvs' => $uvs
 | ||
| 		];
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * 格式化用户增长数据
 | ||
| 	 */
 | ||
| 	private function formatUserGrowthData($data)
 | ||
| 	{
 | ||
| 		$dates = [];
 | ||
| 		$newUsers = [];
 | ||
| 		$totalUsers = [];
 | ||
| 
 | ||
| 		foreach ($data as $item) {
 | ||
| 			$dates[] = date('m-d', strtotime($item['date']));
 | ||
| 			$newUsers[] = $item['new_users'];
 | ||
| 			$totalUsers[] = $item['total_users'];
 | ||
| 		}
 | ||
| 
 | ||
| 		return [
 | ||
| 			'dates' => $dates,
 | ||
| 			'newUsers' => $newUsers,
 | ||
| 			'totalUsers' => $totalUsers
 | ||
| 		];
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * 格式化资源统计数据
 | ||
| 	 */
 | ||
| 	private function formatResourceStatsData($data)
 | ||
| 	{
 | ||
| 		$dates = [];
 | ||
| 		$resources = [];
 | ||
| 		$downloads = [];
 | ||
| 
 | ||
| 		foreach ($data as $item) {
 | ||
| 			$dates[] = date('m-d', strtotime($item['date']));
 | ||
| 			$resources[] = $item['daily_resources'];
 | ||
| 			$downloads[] = $item['resource_downloads'];
 | ||
| 		}
 | ||
| 
 | ||
| 		return [
 | ||
| 			'dates' => $dates,
 | ||
| 			'resources' => $resources,
 | ||
| 			'downloads' => $downloads
 | ||
| 		];
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * 格式化文章统计数据
 | ||
| 	 */
 | ||
| 	private function formatArticleStatsData($data)
 | ||
| 	{
 | ||
| 		$dates = [];
 | ||
| 		$articles = [];
 | ||
| 		$views = [];
 | ||
| 
 | ||
| 		foreach ($data as $item) {
 | ||
| 			$dates[] = date('m-d', strtotime($item['date']));
 | ||
| 			$articles[] = $item['daily_articles'];
 | ||
| 			$views[] = $item['article_views'];
 | ||
| 		}
 | ||
| 
 | ||
| 		return [
 | ||
| 			'dates' => $dates,
 | ||
| 			'articles' => $articles,
 | ||
| 			'views' => $views
 | ||
| 		];
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * 保存附件信息到数据库
 | ||
| 	 * @param string $name 文件名
 | ||
| 	 * @param int $type 附件类型
 | ||
| 	 * @param int $size 文件大小
 | ||
| 	 * @param string $src 文件路径
 | ||
| 	 * @return int 附件ID
 | ||
| 	 */
 | ||
| 	private function saveAttachment($name, $type, $size, $src) {
 | ||
| 		$data = [
 | ||
| 			'name' => $name,
 | ||
| 			'type' => $type,
 | ||
| 			'size' => $size,
 | ||
| 			'src' => $src,
 | ||
| 			'create_time' => time(),
 | ||
| 			'update_time' => time()
 | ||
| 		];
 | ||
| 		return Attachments::insertGetId($data);
 | ||
| 	}
 | ||
| 
 | ||
| 	# 图片上传
 | ||
| 	public function upload_img(){
 | ||
| 		// 获取上传的文件
 | ||
| 		$file = request()->file();
 | ||
| 		$files = request()->file('file');
 | ||
| 		
 | ||
| 		// 检查是否有文件上传
 | ||
| 		if(empty($file)){
 | ||
| 			return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
 | ||
| 		}
 | ||
| 		
 | ||
| 		try {
 | ||
| 			// 验证上传的文件
 | ||
| 			validate([
 | ||
| 				'image'=>'filesize:51200|fileExt:jpg,png,gif,jpeg'
 | ||
| 			])->check($file);
 | ||
| 			
 | ||
| 			// 存储文件到public磁盘的uploads目录
 | ||
| 			$info = Filesystem::disk('public')->putFile('uploads', $files);
 | ||
| 			
 | ||
| 			// 处理文件路径,统一使用正斜杠
 | ||
| 			$info = str_replace("\\", "/", $info);
 | ||
| 			$img = '/storage/'.$info;
 | ||
| 			
 | ||
| 			// 保存附件信息
 | ||
| 			$fileName = $files->getOriginalName();
 | ||
| 			$fileSize = $files->getSize();
 | ||
| 			$attachmentId = $this->saveAttachment($fileName, 1, $fileSize, $img); // 1: 图片
 | ||
| 			
 | ||
| 			// 返回成功信息
 | ||
| 			return json([
 | ||
| 				'code' => 0, 
 | ||
| 				'data' => $img, 
 | ||
| 				'url' => $this->config['admin_domain'].$img,
 | ||
| 				'attachment_id' => $attachmentId
 | ||
| 			])->send();
 | ||
| 			
 | ||
| 		} catch (\think\exception\ValidateException $e) {
 | ||
| 			// 捕获验证异常并返回错误信息
 | ||
| 			return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
 | ||
| 		} catch (\Exception $e) {
 | ||
| 			// 捕获其他异常
 | ||
| 			return json(['code'=>1, 'msg'=>'上传失败:'.$e->getMessage()])->send();
 | ||
| 		}
 | ||
| 	}
 | ||
| 	# 清除缓存
 | ||
| 	public function clear(){
 | ||
| 		$a = delete_dir_file(Env::get('runtime_path').'cache/');
 | ||
| 		$b = delete_dir_file(Env::get('runtime_path').'temp/');
 | ||
| 		if ($a || $b) {
 | ||
| 			$this->returnCode(0, '清除缓存成功');
 | ||
| 		} else {
 | ||
| 			$this->returnCode(1, '清除缓存失败');
 | ||
| 		}
 | ||
| 	}
 | ||
| 	# 文件上传
 | ||
| 	public function upload_file(){
 | ||
| 		$file = request()->file();
 | ||
| 		$files = request()->file('file');
 | ||
| 		if(empty($file)){
 | ||
| 			return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
 | ||
| 		}
 | ||
| 		try {
 | ||
| 			// 只验证文件扩展名,不验证大小
 | ||
| 			validate([
 | ||
| 				'file'=>'fileExt:doc,docx,xls,xlsx,ppt,pptx,pdf,txt,zip,rar,7z'
 | ||
| 			])->check($file);
 | ||
| 			
 | ||
| 			$info = Filesystem::disk('public')->putFile('uploads/files', $files);
 | ||
| 			
 | ||
| 			// 处理文件路径
 | ||
| 			$info = str_replace("\\", "/", $info);
 | ||
| 			$filePath = '/storage/'.$info;
 | ||
| 			
 | ||
| 			// 保存附件信息
 | ||
| 			$fileName = $files->getOriginalName();
 | ||
| 			$fileSize = $files->getSize();
 | ||
| 			$attachmentId = $this->saveAttachment($fileName, 2, $fileSize, $filePath); // 2: 文件
 | ||
| 			
 | ||
| 			return json([
 | ||
| 				'code' => 0,
 | ||
| 				'data' => [
 | ||
| 					'src' => $filePath,
 | ||
| 					'attachment_id' => $attachmentId
 | ||
| 				]
 | ||
| 			])->send();
 | ||
| 		} catch (\think\exception\ValidateException $e) {
 | ||
| 			return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
 | ||
| 		} catch (\Exception $e) {
 | ||
| 			return json(['code'=>1, 'msg'=>'上传失败:'.$e->getMessage()])->send();
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	# 视频上传
 | ||
| 	public function upload_video(){
 | ||
| 		$file = request()->file();
 | ||
| 		$files = request()->file('file');
 | ||
| 		if(empty($file)){
 | ||
| 			return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
 | ||
| 		}
 | ||
| 		try {
 | ||
| 			validate(['video'=>'filesize:102400|fileExt:mp4,avi,mov,wmv,flv'])->check($file);
 | ||
| 			$info = Filesystem::disk('public')->putFile('uploads/videos', $files);
 | ||
| 			
 | ||
| 			// 处理文件路径
 | ||
| 			$info = str_replace("\\", "/", $info);
 | ||
| 			$videoPath = '/storage/'.$info;
 | ||
| 			
 | ||
| 			// 保存附件信息
 | ||
| 			$fileName = $files->getOriginalName();
 | ||
| 			$fileSize = $files->getSize();
 | ||
| 			$attachmentId = $this->saveAttachment($fileName, 3, $fileSize, $videoPath); // 3: 视频
 | ||
| 			
 | ||
| 			return json([
 | ||
| 				'code' => 0,
 | ||
| 				'data' => [
 | ||
| 					'src' => $videoPath,
 | ||
| 					'attachment_id' => $attachmentId
 | ||
| 				]
 | ||
| 			])->send();
 | ||
| 		} catch (\think\exception\ValidateException $e) {
 | ||
| 			return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	# 音频上传
 | ||
| 	public function upload_audio(){
 | ||
| 		$file = request()->file();
 | ||
| 		$files = request()->file('file');
 | ||
| 		if(empty($file)){
 | ||
| 			return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
 | ||
| 		}
 | ||
| 		try {
 | ||
| 			validate(['audio'=>'filesize:51200|fileExt:mp3,wav,ogg,m4a'])->check($file);
 | ||
| 			$info = Filesystem::disk('public')->putFile('uploads/audios', $files);
 | ||
| 			
 | ||
| 			// 处理文件路径
 | ||
| 			$info = str_replace("\\", "/", $info);
 | ||
| 			$audioPath = '/storage/'.$info;
 | ||
| 			
 | ||
| 			// 保存附件信息
 | ||
| 			$fileName = $files->getOriginalName();
 | ||
| 			$fileSize = $files->getSize();
 | ||
| 			$attachmentId = $this->saveAttachment($fileName, 4, $fileSize, $audioPath); // 4: 音频
 | ||
| 			
 | ||
| 			return json([
 | ||
| 				'code' => 0,
 | ||
| 				'data' => [
 | ||
| 					'src' => $audioPath,
 | ||
| 					'attachment_id' => $attachmentId
 | ||
| 				]
 | ||
| 			])->send();
 | ||
| 		} catch (\think\exception\ValidateException $e) {
 | ||
| 			return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| } |