Asset.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784
  1. <?php
  2. namespace Dcat\Admin\Layout;
  3. use Dcat\Admin\Admin;
  4. use Dcat\Admin\Color;
  5. use Illuminate\Support\Str;
  6. class Asset
  7. {
  8. /**
  9. * 别名.
  10. *
  11. * @var array
  12. */
  13. protected $alias = [
  14. // Dcat Admin静态资源路径别名
  15. '@admin' => 'vendor/dcat-admin',
  16. // Dcat Acmin扩展静态资源路径别名
  17. '@extension' => 'vendor/dcat-admin-extensions',
  18. '@adminlte' => [
  19. 'js' => [
  20. '@admin/adminlte/adminlte.js',
  21. ],
  22. 'css' => [
  23. '@admin/adminlte/adminlte.css',
  24. ],
  25. ],
  26. '@nunito' => [
  27. //'css' => 'https://fonts.googleapis.com/css?family=Nunito:200,200i,300,300i,400,400i,600,600i,800,800i,900,900i',
  28. 'css' => '@admin/dcat/css/nunito.css',
  29. ],
  30. '@dcat' => [
  31. 'js' => '@admin/dcat/js/dcat-app.js',
  32. 'css' => '@admin/dcat/css/dcat-app.css',
  33. ],
  34. '@vendors' => [
  35. 'js' => '@admin/dcat/plugins/vendors.min.js',
  36. 'css' => '@admin/dcat/plugins/vendors.min.css',
  37. ],
  38. '@jquery.initialize' => [
  39. 'js' => '@admin/dcat/plugins/jquery.initialize/jquery.initialize.min.js',
  40. ],
  41. '@datatables' => [
  42. 'css' => '@admin/dcat/plugins/tables/datatable/datatables.min.css',
  43. ],
  44. '@grid-extension' => [
  45. 'js' => '@admin/dcat/extra/grid-extend.js',
  46. ],
  47. '@resource-selector' => [
  48. 'js' => '@admin/dcat/extra/resource-selector.js',
  49. ],
  50. '@select-table' => [
  51. 'js' => '@admin/dcat/extra/select-table.js',
  52. ],
  53. '@layer' => [
  54. 'js' => '@admin/dcat/plugins/layer/layer.js',
  55. ],
  56. '@tinymce' => [
  57. 'js' => '@admin/dcat/plugins/tinymce/tinymce.min.js',
  58. ],
  59. '@pjax' => [
  60. 'js' => '@admin/dcat/plugins/jquery-pjax/jquery.pjax.min.js',
  61. ],
  62. '@toastr' => [
  63. 'js' => '@admin/dcat/plugins/extensions/toastr.min.js',
  64. 'css' => '@admin/dcat/plugins/extensions/toastr.css',
  65. ],
  66. '@jquery.nestable' => [
  67. 'js' => '@admin/dcat/plugins/nestable/jquery.nestable.min.js',
  68. 'css' => '@admin/dcat/plugins/nestable/nestable.css',
  69. ],
  70. '@validator' => [
  71. 'js' => '@admin/dcat/plugins/bootstrap-validator/validator.min.js',
  72. ],
  73. '@select2' => [
  74. 'js' => [
  75. '@admin/dcat/plugins/select/select2.full.min.js',
  76. '@admin/dcat/plugins/select/i18n/{lang}.js',
  77. ],
  78. 'css' => '@admin/dcat/plugins/select/select2.min.css',
  79. ],
  80. '@bootstrap-datetimepicker' => [
  81. 'js' => '@admin/dcat/plugins/bootstrap-datetimepicker/bootstrap-datetimepicker.min.js',
  82. 'css' => '@admin/dcat/plugins/bootstrap-datetimepicker/bootstrap-datetimepicker.min.css',
  83. ],
  84. '@moment' => [
  85. 'js' => [
  86. '@admin/dcat/plugins/moment/moment-with-locales.min.js',
  87. ],
  88. ],
  89. '@moment-timezone' => [
  90. 'js' => [
  91. '@admin/dcat/plugins/moment-timezone/moment-timezone-with-data.min.js',
  92. ],
  93. ],
  94. '@jstree' => [
  95. 'js' => '@admin/dcat/plugins/jstree-theme/jstree.min.js',
  96. 'css' => '@admin/dcat/plugins/jstree-theme/themes/proton/style.min.css',
  97. ],
  98. '@switchery' => [
  99. 'js' => '@admin/dcat/plugins/switchery/switchery.min.js',
  100. 'css' => '@admin/dcat/plugins/switchery/switchery.min.css',
  101. ],
  102. '@webuploader' => [
  103. 'js' => [
  104. '@admin/dcat/plugins/webuploader/webuploader.min.js',
  105. '@admin/dcat/extra/upload.js',
  106. ],
  107. 'css' => '@admin/dcat/extra/upload.css',
  108. ],
  109. '@chartjs' => [
  110. 'js' => '@admin/dcat/plugins/chart.js/chart.bundle.min.js',
  111. ],
  112. '@jquery.sparkline' => [
  113. 'js' => '@admin/dcat/plugins/jquery.sparkline/jquery.sparkline.min.js',
  114. ],
  115. '@jquery.bootstrap-duallistbox' => [
  116. 'js' => '@admin/dcat/plugins/bootstrap-duallistbox/dist/jquery.bootstrap-duallistbox.min.js',
  117. 'css' => '@admin/dcat/plugins/bootstrap-duallistbox/dist/bootstrap-duallistbox.min.css',
  118. ],
  119. '@number-input' => [
  120. 'js' => '@admin/dcat/plugins/number-input/bootstrap-number-input.js',
  121. ],
  122. '@ionslider' => [
  123. 'js' => [
  124. '@admin/dcat/plugins/ionslider/ion.rangeSlider.min.js',
  125. ],
  126. 'css' => [
  127. '@admin/dcat/plugins/ionslider/ion.rangeSlider.css',
  128. '@admin/dcat/plugins/ionslider/ion.rangeSlider.skinNice.css',
  129. ],
  130. ],
  131. '@editor-md' => [
  132. 'js' => [
  133. '@admin/dcat/plugins/editor-md/lib/raphael.min.js',
  134. '@admin/dcat/plugins/editor-md/lib/marked.min.js',
  135. '@admin/dcat/plugins/editor-md/lib/prettify.min.js',
  136. '@admin/dcat/plugins/editor-md/lib/underscore.min.js',
  137. '@admin/dcat/plugins/editor-md/lib/sequence-diagram.min.js',
  138. '@admin/dcat/plugins/editor-md/lib/flowchart.min.js',
  139. '@admin/dcat/plugins/editor-md/lib/jquery.flowchart.min.js',
  140. '@admin/dcat/plugins/editor-md/editormd.min.js',
  141. ],
  142. 'css' => [
  143. '@admin/dcat/plugins/editor-md/css/editormd.preview.min.css',
  144. '@admin/dcat/extra/markdown.css',
  145. ],
  146. ],
  147. '@editor-md-form' => [
  148. 'js' => [
  149. '@admin/dcat/plugins/editor-md/lib/raphael.min.js',
  150. '@admin/dcat/plugins/editor-md/editormd.min.js',
  151. ],
  152. 'css' => [
  153. '@admin/dcat/plugins/editor-md/css/editormd.min.css',
  154. ],
  155. ],
  156. '@jquery.inputmask' => [
  157. 'js' => '@admin/dcat/plugins/input-mask/jquery.inputmask.bundle.min.js',
  158. ],
  159. '@apex-charts' => [
  160. 'js' => '@admin/dcat/plugins/charts/apexcharts.min.js',
  161. ],
  162. '@fontawesome-iconpicker' => [
  163. 'js' => '@admin/dcat/plugins/fontawesome-iconpicker/dist/js/fontawesome-iconpicker.js',
  164. 'css' => '@admin/dcat/plugins/fontawesome-iconpicker/dist/css/fontawesome-iconpicker.min.css',
  165. ],
  166. '@color' => [
  167. 'js' => '@admin/dcat/plugins/bootstrap-colorpicker/js/bootstrap-colorpicker.min.js',
  168. 'css' => '@admin/dcat/plugins/bootstrap-colorpicker/css/bootstrap-colorpicker.min.css',
  169. ],
  170. '@qrcode' => [
  171. 'js' => '@admin/dcat/plugins/jquery-qrcode/dist/jquery-qrcode.min.js',
  172. ],
  173. '@sortable' => [
  174. 'js' => '@admin/dcat/plugins/sortable/Sortable.min.js',
  175. ],
  176. '@autocomplete' => [
  177. 'js' => '@admin/dcat/plugins/autocomplete/jquery.autocomplete.min.js',
  178. ],
  179. ];
  180. /**
  181. * js代码.
  182. *
  183. * @var array
  184. */
  185. public $script = [];
  186. /**
  187. * @var array
  188. */
  189. public $directScript = [];
  190. /**
  191. * css代码.
  192. *
  193. * @var array
  194. */
  195. public $style = [];
  196. /**
  197. * css脚本路径.
  198. *
  199. * @var array
  200. */
  201. public $css = [];
  202. /**
  203. * js脚本路径.
  204. *
  205. * @var array
  206. */
  207. public $js = [];
  208. /**
  209. * 在head标签内加载的js脚本.
  210. *
  211. * @var array
  212. */
  213. public $headerJs = [
  214. 'vendors' => '@vendors',
  215. 'dcat' => '@dcat',
  216. ];
  217. /**
  218. * 基础css.
  219. *
  220. * @var array
  221. */
  222. public $baseCss = [
  223. 'adminlte' => '@adminlte',
  224. 'vendors' => '@vendors',
  225. 'toastr' => '@toastr',
  226. 'datatables' => '@datatables',
  227. 'dcat' => '@dcat',
  228. ];
  229. /**
  230. * 基础js.
  231. *
  232. * @var array
  233. */
  234. public $baseJs = [
  235. 'adminlte' => '@adminlte',
  236. 'toastr' => '@toastr',
  237. 'pjax' => '@pjax',
  238. 'validator' => '@validator',
  239. 'layer' => '@layer',
  240. 'init' => '@jquery.initialize',
  241. ];
  242. /**
  243. * @var array
  244. */
  245. public $fonts = [
  246. '@nunito',
  247. ];
  248. /**
  249. * 初始化主题样式.
  250. */
  251. protected function setUpTheme()
  252. {
  253. $color = Admin::color()->getName();
  254. if ($color === Color::DEFAULT_COLOR) {
  255. return;
  256. }
  257. $alias = [
  258. '@adminlte',
  259. '@dcat',
  260. ];
  261. foreach ($alias as $n) {
  262. $before = (array) $this->alias[$n]['css'];
  263. $this->alias[$n]['css'] = [];
  264. foreach ($before as $css) {
  265. $this->alias[$n]['css'][] = str_replace('.css', "-{$color}.css", $css);
  266. }
  267. }
  268. }
  269. /**
  270. * 设置或获取别名.
  271. *
  272. * @param string|array $name
  273. * @param string|array $value
  274. * @return void|array
  275. */
  276. public function alias($name, $value = null)
  277. {
  278. if (is_array($name)) {
  279. foreach ($name as $key => $value) {
  280. $this->alias($key, $value);
  281. }
  282. return;
  283. }
  284. if ($value === null) {
  285. return $this->getAlias($name);
  286. }
  287. if (mb_strpos($name, '@') !== 0) {
  288. $name = '@'.$name;
  289. }
  290. $this->alias[$name] = $value;
  291. }
  292. /**
  293. * 获取别名.
  294. *
  295. * @param string $name
  296. * @param array $params
  297. * @return array|string
  298. */
  299. public function getAlias($name, array $params = [])
  300. {
  301. if (mb_strpos($name, '@') !== 0) {
  302. $name = '@'.$name;
  303. }
  304. [$name, $query] = $this->parseParams($name);
  305. $assets = $this->alias[$name] ?? [];
  306. // 路径别名
  307. if (is_string($assets)) {
  308. return $assets;
  309. }
  310. $params += $query;
  311. return [
  312. 'js' => $this->normalizeAliasPaths($assets['js'] ?? [], $params) ?: null,
  313. 'css' => $this->normalizeAliasPaths($assets['css'] ?? [], $params) ?: null,
  314. ];
  315. }
  316. /**
  317. * @param array $files
  318. * @param array $params
  319. * @return array
  320. */
  321. protected function normalizeAliasPaths($files, array $params)
  322. {
  323. $files = (array) $files;
  324. foreach ($files as &$file) {
  325. foreach ($params as $k => $v) {
  326. if ($v !== '' && $v !== null) {
  327. $file = str_replace("{{$k}}", $v, $file);
  328. }
  329. }
  330. }
  331. return array_filter($files, function ($file) {
  332. return ! mb_strpos($file, '{');
  333. });
  334. }
  335. /**
  336. * 解析参数.
  337. *
  338. * @param string $name
  339. * @return array
  340. */
  341. protected function parseParams($name)
  342. {
  343. $name = explode('?', $name);
  344. if (empty($name[1])) {
  345. return [$name[0], []];
  346. }
  347. parse_str($name[1], $params);
  348. return [$name[0], $params];
  349. }
  350. /**
  351. * 根据别名设置需要载入的js和css脚本.
  352. *
  353. * @param string|array $alias
  354. * @param array $params
  355. * @return void
  356. */
  357. public function require($alias, array $params = [])
  358. {
  359. if (is_array($alias)) {
  360. foreach ($alias as $v) {
  361. $this->require($v, $params);
  362. }
  363. return;
  364. }
  365. $assets = $this->getAlias($alias, $params);
  366. $this->js($assets['js']);
  367. $this->css($assets['css']);
  368. }
  369. /**
  370. * 设置需要载入的css脚本.
  371. *
  372. * @param string|array $css
  373. */
  374. public function css($css)
  375. {
  376. if (! $css) {
  377. return;
  378. }
  379. $this->css = array_merge(
  380. $this->css,
  381. (array) $css
  382. );
  383. }
  384. /**
  385. * 设置需要载入的基础css脚本.
  386. *
  387. * @param array $css
  388. */
  389. public function baseCss(array $css, bool $merge = false)
  390. {
  391. if ($merge) {
  392. $this->baseCss = array_merge($this->baseCss, $css);
  393. } else {
  394. $this->baseCss = $css;
  395. }
  396. }
  397. /**
  398. * 设置需要载入的js脚本.
  399. *
  400. * @param string|array $js
  401. */
  402. public function js($js)
  403. {
  404. if (! $js) {
  405. return;
  406. }
  407. $this->js = array_merge(
  408. $this->js,
  409. (array) $js
  410. );
  411. }
  412. /**
  413. * 根据别名获取资源路径.
  414. *
  415. * @param string $path
  416. * @param string $type
  417. * @return string|array|null
  418. */
  419. public function get($path, string $type = 'js')
  420. {
  421. if (empty($this->alias[$path])) {
  422. return $this->url($path);
  423. }
  424. $paths = isset($this->alias[$path][$type]) ? (array) $this->alias[$path][$type] : null;
  425. if (! $paths) {
  426. return $paths;
  427. }
  428. foreach ($paths as &$value) {
  429. $value = $this->url($value);
  430. }
  431. return $paths;
  432. }
  433. /**
  434. * 获取静态资源完整URL.
  435. *
  436. * @param string $path
  437. * @return string
  438. */
  439. public function url($path)
  440. {
  441. if (! $path) {
  442. return $path;
  443. }
  444. $path = $this->getRealPath($path);
  445. if (mb_strpos($path, '//') === false) {
  446. $path = config('admin.assets_server').'/'.trim($path, '/');
  447. }
  448. return (config('admin.https') || config('admin.secure')) ? secure_asset($path) : asset($path);
  449. }
  450. /**
  451. * 获取真实路径.
  452. *
  453. * @param string|null $path
  454. * @return string|null
  455. */
  456. public function getRealPath(?string $path)
  457. {
  458. if (! $this->containsAlias($path)) {
  459. return $path;
  460. }
  461. return implode(
  462. '/',
  463. array_map(
  464. function ($v) {
  465. if (! $this->isPathAlias($v)) {
  466. return $v;
  467. }
  468. return $this->getRealPath($this->alias($v));
  469. },
  470. explode('/', $path)
  471. )
  472. );
  473. }
  474. /**
  475. * 判断是否是路径别名.
  476. *
  477. * @param mixed $value
  478. * @return bool
  479. */
  480. public function isPathAlias($value)
  481. {
  482. return $this->hasAlias($value) && is_string($this->alias[$value]);
  483. }
  484. /**
  485. * 判断别名是否存在.
  486. *
  487. * @param $value
  488. * @return bool
  489. */
  490. public function hasAlias($value)
  491. {
  492. return isset($this->alias[$value]);
  493. }
  494. /**
  495. * 判断是否含有别名.
  496. *
  497. * @param string $value
  498. * @return bool
  499. */
  500. protected function containsAlias($value)
  501. {
  502. return $value && mb_strpos($value, '@') === 0;
  503. }
  504. /**
  505. * 设置在head标签内加载的js.
  506. *
  507. * @param string|array $js
  508. */
  509. public function headerJs($js, bool $merge = true)
  510. {
  511. if ($merge) {
  512. $this->headerJs = $js ? array_merge($this->headerJs, (array) $js) : $this->headerJs;
  513. } else {
  514. $this->headerJs = (array) $js;
  515. }
  516. }
  517. /**
  518. * 设置基础js脚本.
  519. *
  520. * @param array $js
  521. * @param bool $merge
  522. */
  523. public function baseJs(array $js, bool $merge = true)
  524. {
  525. if ($merge) {
  526. $this->baseJs = array_merge($this->baseJs, $js);
  527. } else {
  528. $this->baseJs = $js;
  529. }
  530. }
  531. /**
  532. * 设置js代码.
  533. *
  534. * @param string|array $script
  535. * @param bool $direct
  536. */
  537. public function script($script, bool $direct = false)
  538. {
  539. if (! $script) {
  540. return;
  541. }
  542. if ($direct) {
  543. $this->directScript = array_merge($this->directScript, (array) $script);
  544. } else {
  545. $this->script = array_merge($this->script, (array) $script);
  546. }
  547. }
  548. /**
  549. * 设置css代码.
  550. *
  551. * @param string $style
  552. */
  553. public function style($style)
  554. {
  555. if (! $style) {
  556. return;
  557. }
  558. $this->style = array_merge($this->style, (array) $style);
  559. }
  560. /**
  561. * 字体css脚本路径.
  562. */
  563. protected function addFontCss()
  564. {
  565. $this->fonts && ($this->baseCss = array_merge(
  566. $this->baseCss,
  567. (array) $this->fonts
  568. ));
  569. }
  570. protected function isPjax()
  571. {
  572. return request()->pjax();
  573. }
  574. /**
  575. * 合并基础css脚本.
  576. */
  577. protected function mergeBaseCss()
  578. {
  579. if ($this->isPjax()) {
  580. return;
  581. }
  582. $this->addFontCss();
  583. $this->css = array_merge($this->baseCss, $this->css);
  584. }
  585. /**
  586. * @return string
  587. */
  588. public function cssToHtml()
  589. {
  590. $this->setUpTheme();
  591. $this->mergeBaseCss();
  592. $html = '';
  593. foreach (array_unique($this->css) as &$v) {
  594. if (! $paths = $this->get($v, 'css')) {
  595. continue;
  596. }
  597. foreach ((array) $paths as $path) {
  598. $html .= "<link rel=\"stylesheet\" href=\"{$this->withVersionQuery($path)}\">";
  599. }
  600. }
  601. return $html;
  602. }
  603. /**
  604. * @param string $url
  605. * @return string
  606. */
  607. public function withVersionQuery($url)
  608. {
  609. if (! Str::contains($url, '?')) {
  610. $url .= '?';
  611. }
  612. $ver = 'v'.Admin::VERSION;
  613. return Str::endsWith($url, '?') ? $url.$ver : $url.'&'.$ver;
  614. }
  615. /**
  616. * 合并基础js脚本.
  617. */
  618. protected function mergeBaseJs()
  619. {
  620. if ($this->isPjax()) {
  621. return;
  622. }
  623. $this->js = array_merge($this->baseJs, $this->js);
  624. }
  625. /**
  626. * @return string
  627. */
  628. public function jsToHtml()
  629. {
  630. $this->mergeBaseJs();
  631. $html = '';
  632. foreach (array_unique($this->js) as &$v) {
  633. if (! $paths = $this->get($v, 'js')) {
  634. continue;
  635. }
  636. foreach ((array) $paths as $path) {
  637. $html .= "<script src=\"{$this->withVersionQuery($path)}\"></script>";
  638. }
  639. }
  640. return $html;
  641. }
  642. /**
  643. * @return string
  644. */
  645. public function headerJsToHtml()
  646. {
  647. $html = '';
  648. foreach (array_unique($this->headerJs) as &$v) {
  649. if (! $paths = $this->get($v, 'js')) {
  650. continue;
  651. }
  652. foreach ((array) $paths as $path) {
  653. $html .= "<script src=\"{$this->withVersionQuery($path)}\"></script>";
  654. }
  655. }
  656. return $html;
  657. }
  658. /**
  659. * @return string
  660. */
  661. public function scriptToHtml()
  662. {
  663. $script = implode(";\n", array_unique($this->script));
  664. $directScript = implode(";\n", array_unique($this->directScript));
  665. return <<<HTML
  666. <script data-exec-on-popstate>
  667. (function () {
  668. try {
  669. {$directScript}
  670. } catch (e) {
  671. console.error(e)
  672. }
  673. })();
  674. Dcat.ready(function () {
  675. try {
  676. {$script}
  677. } catch (e) {
  678. console.error(e)
  679. }
  680. })
  681. </script>
  682. HTML;
  683. }
  684. /**
  685. * @return string
  686. */
  687. public function styleToHtml()
  688. {
  689. $style = implode('', array_unique($this->style));
  690. return "<style>$style</style>";
  691. }
  692. }