SiteAlbumController.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. <?php
  2. namespace App\Admin\Controllers;
  3. use App\Admin\Repositories\SiteAlbum;
  4. use App\Admin\Repositories\SiteAlbumFolder;
  5. use App\Libraries\CommonHelper;
  6. use App\Models\SiteAlbumFolder as SiteAlbumFolderModel;
  7. use Dcat\Admin\Form;
  8. use Dcat\Admin\Grid;
  9. use Dcat\Admin\Show;
  10. use Dcat\Admin\Http\Controllers\AdminController;
  11. use Dcat\Admin\Layout\Content;
  12. use Dcat\Admin\Admin;
  13. use App\Admin\Repositories\NullRepository;
  14. use Dcat\Admin\Traits\HasUploadedFile;
  15. use function Symfony\Component\Translation\t;
  16. class SiteAlbumController extends AdminController
  17. {
  18. use HasUploadedFile;
  19. public function title()
  20. {
  21. return admin_trans( 'admin.album');
  22. }
  23. /**
  24. * page index
  25. */
  26. public function index(Content $content)
  27. {
  28. //记录folder_id
  29. $folderId = isset($_GET['folder_id']) ? intval($_GET['folder_id']) : 0;
  30. //保存临时变量
  31. setTempValue('folderId', $folderId);
  32. $html = $content
  33. ->header(admin_trans( 'admin.album'))
  34. ->body($this->indexForm());
  35. $html = $html->render();
  36. //删除第一个formID对应的JS代码
  37. preg_match('/<form[^>]*id="([^"]*)"[^>]*>/', $html, $matches);
  38. if (isset($matches[1])) {
  39. $formId = $matches[1]; // 获取 id 的值
  40. // echo "找到的 form id: " . $formId . "\n";
  41. // 2. 根据 id 值,删除对应的 JavaScript 代码
  42. $pattern = '/\$\(\'#' . preg_quote($formId, '/') . '\'\)\.form\(\{.*?\}\);/s';
  43. $html = preg_replace($pattern, '', $html);
  44. }
  45. //把第一个form标签替换成div标签
  46. $html = preg_replace('/<form([^>]*)>(.*?)<\/form>/s', '<div$1>$2</div>', $html, 1);
  47. return $html;
  48. }
  49. protected function indexForm()
  50. {
  51. return Form::make(new NullRepository(), function (Form $form) {
  52. $form->action('/site-album');
  53. $folderModel = new SiteAlbumFolderModel();
  54. $folderTree = $folderModel->allNodes();
  55. $form->block(2, function (Form\BlockForm $form) use ($folderTree) {
  56. $type = [
  57. 'default' => [
  58. 'icon' => true,
  59. ],
  60. ];
  61. $plugins = ['types'];
  62. $form->tree()
  63. ->setTitleColumn('title')
  64. ->nodes($folderTree)
  65. ->type($type)
  66. ->plugins($plugins)
  67. ->width(12,0);
  68. });
  69. $form->block(10, function (Form\BlockForm $form) {
  70. $form->html($this->grid())->width(12);
  71. });
  72. //以下JS代码用于点击文件夹时,自动跳转到相应页面
  73. Admin::script(
  74. <<<JS
  75. setTimeout(() => {
  76. // 获取所有具有 class="jstree-anchor" 的 <a> 元素
  77. const anchors = document.querySelectorAll('a.jstree-anchor');
  78. anchors.forEach(anchor => {
  79. // 提取 id 中的数字部分
  80. const id = anchor.id.split('_')[0];
  81. // 动态生成跳转链接
  82. const href = `/prime-control/site-album?folder_id=`+id;
  83. // 绑定点击事件
  84. anchor.addEventListener('click', function(event) {
  85. event.preventDefault(); // 阻止默认的链接跳转行为
  86. window.location.href = href; // 跳转到目标页面
  87. });
  88. });
  89. }, 200);
  90. const firstCheckbox = document.querySelector('.vs-checkbox-primary');
  91. // 如果找到元素,则隐藏它
  92. if (firstCheckbox) {
  93. firstCheckbox.style.display = 'none';
  94. }
  95. //清空_previous_
  96. const input = document.querySelector('input[name="_previous_"]');
  97. if (input) {
  98. // 清空其值
  99. input.value = '';
  100. }
  101. JS
  102. );
  103. });
  104. }
  105. protected function grid()
  106. {
  107. return Grid::make(new SiteAlbum(), function (Grid $grid) {
  108. //默认分页条数
  109. $grid->paginate(config('admin.per_page'));
  110. $grid->column('id')->sortable();
  111. $grid->column('cover')->display(function ($images) {
  112. $images = json_decode($images);
  113. // 限制最多显示2个缩略图
  114. $dataImages = array_slice($images, 0, 1);
  115. return CommonHelper::displayImage($dataImages,80);
  116. });
  117. $grid->column('model');
  118. $grid->column('folder_id',admin_trans_label('folder'))
  119. ->display(function ($folder_id) {
  120. $folderModel = new SiteAlbumFolderModel();
  121. $folderName = $folderModel->find($folder_id)->title;
  122. return $folderName;
  123. });
  124. $grid->column('enabled')->using(admin_trans_array(config('dictionary.enabled'))) ->label([
  125. 0 => 'danger',
  126. 1 => 'success',
  127. ]);
  128. $grid->column('missing_content')->display(function ($missing_content) {
  129. $missing_content = [];
  130. if ($this->cover == '[]') {$missing_content[] = '主图';}
  131. if ($this->en_detail == '[]') {$missing_content[] = '英文详情';}
  132. if ($this->cn_detail == '[]') {$missing_content[] = '中文详情';}
  133. if ($this->video == '[]') {$missing_content[] = '视频';}
  134. if ($this->poster == '[]') {$missing_content[] = '海报';}
  135. if ($this->cert == '[]') {$missing_content[] = '证书';}
  136. if ($this->pdf == '[]') {$missing_content[] = 'PDF';}
  137. return implode(' / ', $missing_content);
  138. });
  139. // 筛选
  140. $grid->filter(function (Grid\Filter $filter) {
  141. $selectOptions = SiteAlbumFolderModel::selectOptions();
  142. unset($selectOptions[0]);
  143. $filter->panel();
  144. $filter->expand();
  145. $filter->like('model')->width(2);
  146. $filter->equal('folder_id',admin_trans_label('folder'))->select($selectOptions)->width(3);
  147. });
  148. $grid->disableViewButton();
  149. $grid->disablePerPages();
  150. $grid->disableRefreshButton();
  151. $grid->model()->orderBy('order', 'asc')->orderBy('id', 'desc');
  152. //弹窗大小
  153. $grid->setDialogFormDimensions('830px','670px');
  154. });
  155. }
  156. protected function form()
  157. {
  158. $thisObj = $this;
  159. return Form::make(new SiteAlbum(), function (Form $form) use ($thisObj) {
  160. $form->width(9, 1);
  161. $form->disableViewButton();
  162. $form->disableViewCheck();
  163. $form->saving(function (Form $form) use ($thisObj) {
  164. //处理video
  165. $videos = $form->input('video');
  166. if ($videos) {
  167. foreach ($videos as $key => $value) {
  168. if (empty($value['cover']) && $value['_remove_'] != 1) {
  169. //自动生成封面
  170. $result = $thisObj->autoGenerateCover($value['video_src']);
  171. if ($result['status']) {
  172. $videos[$key]['cover'] = $result['path'];
  173. } else {
  174. return $form->response()->error($result['msg']);
  175. }
  176. }
  177. }
  178. } else {
  179. $videos = [];
  180. }
  181. $form->input('video', $videos);
  182. //处理pdf
  183. $pdfs = $form->input('pdf');
  184. $pdfs = empty($pdfs) ? [] : $pdfs;
  185. $form->input('pdf', $pdfs);
  186. });
  187. $form->tab(admin_trans_label('basic_info'), function (Form $form) {
  188. $selectOptions = SiteAlbumFolder::selectOptions();
  189. unset($selectOptions[0]);
  190. $folderId = getTempValue('folderId');
  191. if ($folderId == 0) {
  192. $folderId = array_key_first($selectOptions);
  193. }
  194. $form->select('folder_id')->options($selectOptions)->default($folderId)->required();
  195. $form->text('model')->required();
  196. $form->table('parameters',admin_trans_label('attribute_name'), function (Form\NestedForm $table) {
  197. $table->text('key')->required();
  198. $table->text('value')->required();
  199. })->setView('admin.form_custom.hasmanytable')
  200. ->saving(function ($input) {
  201. return json_encode($input);
  202. });
  203. $form->switch('enabled')->default(1);
  204. })->tab(admin_trans_label('cover'), function (Form $form) {
  205. $form->multipleImage('cover')
  206. ->retainable()//禁止删OSS图
  207. ->sortable() // 可拖动排序
  208. ->removable() // 可移除图片
  209. ->autoUpload() // 自动上传
  210. ->uniqueName()
  211. ->limit(config('admin.upload.oss_image.limit'))
  212. ->accept(config('admin.upload.oss_image.accept'))
  213. ->maxSize(config('admin.upload.oss_image.max_size'))
  214. ->dir(config("admin.upload.directory.image").'/uploads/'.date("Ymd"))
  215. ->saving(function ($images) use ($form) {
  216. return json_encode($images);
  217. });
  218. })->tab(admin_trans_label('en_detail'), function (Form $form) {
  219. $form->multipleImage('en_detail')
  220. ->retainable()//禁止删OSS图
  221. ->sortable() // 可拖动排序
  222. ->removable() // 可移除图片
  223. ->autoUpload() // 自动上传
  224. ->uniqueName()
  225. ->limit(config('admin.upload.oss_image.limit'))
  226. ->accept(config('admin.upload.oss_image.accept'))
  227. ->maxSize(config('admin.upload.oss_image.max_size'))
  228. ->dir(config("admin.upload.directory.image").'/uploads/'.date("Ymd"))
  229. ->saving(function ($images) use ($form) {
  230. return json_encode($images);
  231. });
  232. })->tab(admin_trans_label('cn_detail'), function (Form $form) {
  233. $form->multipleImage('cn_detail')
  234. ->retainable()//禁止删OSS图
  235. ->sortable() // 可拖动排序
  236. ->removable() // 可移除图片
  237. ->autoUpload() // 自动上传
  238. ->uniqueName()
  239. ->limit(config('admin.upload.oss_image.limit'))
  240. ->accept(config('admin.upload.oss_image.accept'))
  241. ->maxSize(config('admin.upload.oss_image.max_size'))
  242. ->dir(config("admin.upload.directory.image").'/uploads/'.date("Ymd"))
  243. ->saving(function ($images) use ($form) {
  244. return json_encode($images);
  245. });
  246. })->tab(admin_trans_label('video'), function (Form $form) {
  247. $count = 0;
  248. $form->hasMany('video', function (Form\NestedForm $form) use (&$count) {
  249. $videos = $form->model()->video;
  250. $imgArray = "";
  251. if ($videos) {
  252. $videos = json_decode($videos,true);
  253. foreach ($videos as $key => $value) {
  254. if ($value['cover'] && $key == $count-1) {
  255. $imgArray = [$value['cover']];
  256. }
  257. }
  258. }
  259. $imgHtml = CommonHelper::displayImage($imgArray);
  260. $form->html($imgHtml,admin_trans_label('image_preview'));
  261. $count++;
  262. $form->text('cover',admin_trans_label('video_cover'))->placeholder('自动生成')->readOnly();
  263. $form->tradFile('video_src')
  264. ->retainable()//禁止删OSS图
  265. ->removable() // 可移除图片
  266. ->autoUpload() // 自动上传
  267. ->uniqueName()//
  268. ->downloadable()
  269. ->accept(config('admin.upload.oss_video.accept'))
  270. ->maxSize(config('admin.upload.oss_video.max_size'))
  271. ->dir(config("admin.upload.directory.video").'/uploads/'.date("Ymd"))
  272. ->chunkSize(1024)
  273. ->required();
  274. })->useTable()
  275. ->customFormat(function ($data) {return json_decode($data,true);})
  276. ->setView('admin.form_custom.hasmanytable')
  277. ->saving(function ($input) {
  278. $data = [];
  279. foreach ($input as $value) {
  280. if ($value['_remove_'] != 1){
  281. $data[] = ['cover'=>$value['cover'],'video_src'=>$value['video_src']];
  282. }
  283. }
  284. return json_encode($data);
  285. });
  286. })->tab(admin_trans_label('poster'), function (Form $form) {
  287. $form->multipleImage('poster')
  288. ->retainable()//禁止删OSS图
  289. ->sortable() // 可拖动排序
  290. ->removable() // 可移除图片
  291. ->autoUpload() // 自动上传
  292. ->uniqueName()
  293. ->limit(config('admin.upload.oss_image.limit'))
  294. ->accept(config('admin.upload.oss_image.accept'))
  295. ->maxSize(config('admin.upload.oss_image.max_size'))
  296. ->dir(config("admin.upload.directory.image").'/uploads/'.date("Ymd"))
  297. ->saving(function ($images) use ($form) {
  298. return json_encode($images);
  299. });
  300. })->tab(admin_trans_label('cert'), function (Form $form) {
  301. $form->multipleImage('cert')
  302. ->retainable()//禁止删OSS图
  303. ->sortable() // 可拖动排序
  304. ->removable() // 可移除图片
  305. ->autoUpload() // 自动上传
  306. ->uniqueName()
  307. ->limit(config('admin.upload.oss_image.limit'))
  308. ->accept(config('admin.upload.oss_image.accept'))
  309. ->maxSize(config('admin.upload.oss_image.max_size'))
  310. ->dir(config("admin.upload.directory.image").'/uploads/'.date("Ymd"))
  311. ->saving(function ($images) use ($form) {
  312. return json_encode($images);
  313. });
  314. })->tab(admin_trans_label('pdf'), function (Form $form) {
  315. $form->hasMany('pdf', function ($form) {
  316. $form->text('pdf_title')->required();
  317. $form->tradFile('pdf_src')
  318. ->retainable()//禁止删OSS图
  319. ->removable() // 可移除图片
  320. ->autoUpload() // 自动上传
  321. ->uniqueName()
  322. ->downloadable()
  323. ->accept(config('admin.upload.oss_pdf.accept'))
  324. ->maxSize(config('admin.upload.oss_pdf.max_size'))
  325. ->dir(config("admin.upload.directory.pdf").'/uploads/'.date("Ymd"))
  326. ->chunkSize(1024)
  327. ->required();
  328. })->useTable()
  329. ->customFormat(function ($data) {
  330. return json_decode($data,true);
  331. })
  332. ->setView('admin.form_custom.hasmanytable')
  333. ->saving(function ($input) {
  334. $data = [];
  335. foreach ($input as $value) {
  336. if ($value['_remove_'] != 1){
  337. $data[] = ['pdf_title'=>$value['pdf_title'],'pdf_src'=>$value['pdf_src']];
  338. }
  339. }
  340. return json_encode($data);
  341. });
  342. });
  343. $folderTabs = SiteAlbumFolder::getAllFolderTabs();
  344. $folderTabs = json_encode($folderTabs);
  345. //以下JS作用:1.点击列表时,把folder_id参数传递给表单 2.切换文件夹时,显示隐藏相应的tab
  346. Admin::script(
  347. <<<JS
  348. const featherIcon = document.querySelector('i.feather.icon-list');
  349. if (featherIcon) {
  350. // 找到 <i> 的上级 <a> 元素
  351. const parentLink = featherIcon.closest('a');
  352. if (parentLink) {
  353. // 绑定 onclick 事件
  354. parentLink.onclick = function(event) {
  355. // 阻止默认行为(如跳转)
  356. event.preventDefault();
  357. // 获取 folder_id 的值
  358. let folderIdValue = $('select[name="folder_id"]').val();
  359. // 在 href 后追加 ?folder_id=xxx
  360. if (parentLink.href) {
  361. const newHref = parentLink.href + '?folder_id=' + folderIdValue;
  362. window.location.href = newHref; // 跳转到新的 URL
  363. }
  364. };
  365. }
  366. }
  367. // 监听 <select name="folder_id"> 的变化事件
  368. $('select[name="folder_id"]').change(function() {
  369. showHideTabs($(this).val());
  370. });
  371. folderIdValue = $('select[name="folder_id"]').val();
  372. showHideTabs(folderIdValue);
  373. function showHideTabs(fid) {
  374. // 获取当前选中的值
  375. let folderTabs = {$folderTabs};
  376. const folderIdValue = fid;
  377. // 获取当前 folderIdValue 对应的标签索引
  378. const tabIndexes = folderTabs[folderIdValue];
  379. //console.log(tabIndexes);
  380. if (tabIndexes) {
  381. // 处理上方的 <li>,限定在 .nav-tabs 内
  382. $('.nav-tabs .nav-item').each(function(index) {
  383. if (index > 0) { // 跳过第一个固定元素
  384. $(this).hide();
  385. }
  386. });
  387. tabIndexes.forEach(function(index) {
  388. $('.nav-tabs .nav-item:eq(' + (Number(index) + 1) + ')').show(); // 使用字符串拼接
  389. });
  390. // 处理下方的 tab-pane,限定在 .tab-content 内
  391. // $('.tab-content .tab-pane').each(function(index) {
  392. // if (index > 0) { // 跳过第一个固定元素
  393. // $(this).hide();
  394. // }
  395. // });
  396. //
  397. // tabIndexes.forEach(function(index) {
  398. // $('.tab-content .tab-pane:eq(' + (Number(index) + 1) + ')').show(); // 使用字符串拼接
  399. // });
  400. }
  401. }
  402. JS
  403. );
  404. });
  405. }
  406. /*
  407. * 自动生成视频封面
  408. */
  409. private function autoGenerateCover($videoSrc)
  410. {
  411. $cover = $videoSrc.'?x-oss-process=video/snapshot,t_2000,f_jpg,h_500,m_fast';
  412. //TODO 上传到OSS
  413. $cover = CommonHelper::ossUrl($cover);
  414. $path = $this->upload($cover,'.jpg');
  415. return $path;
  416. }
  417. private function upload($file,$imgType='.jpg')
  418. {
  419. $disk = $this->disk('oss');
  420. $newName = uniqueCode("video_cover_").$imgType;
  421. $dir = config("admin.upload.directory.image").'/uploads/'.date("Ymd").'/'.$newName;
  422. $contents = file_get_contents($file);
  423. if (!$contents) {
  424. return ['status'=>false,'msg'=>'图片上传失败,请检查PHP配置'];
  425. }
  426. $disk->put($dir, $contents);
  427. return ['status'=>true,'path'=>$dir];
  428. }
  429. }