Asset.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799
  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. '@markdown' => [
  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. ];
  177. /**
  178. * js代码.
  179. *
  180. * @var array
  181. */
  182. public $script = [];
  183. /**
  184. * @var array
  185. */
  186. public $directScript = [];
  187. /**
  188. * css代码.
  189. *
  190. * @var array
  191. */
  192. public $style = [];
  193. /**
  194. * css脚本路径.
  195. *
  196. * @var array
  197. */
  198. public $css = [];
  199. /**
  200. * js脚本路径.
  201. *
  202. * @var array
  203. */
  204. public $js = [];
  205. /**
  206. * 在head标签内加载的js脚本.
  207. *
  208. * @var array
  209. */
  210. public $headerJs = [
  211. 'vendors' => '@vendors',
  212. 'dcat' => '@dcat',
  213. ];
  214. /**
  215. * 基础css.
  216. *
  217. * @var array
  218. */
  219. public $baseCss = [
  220. 'adminlte' => '@adminlte',
  221. 'vendors' => '@vendors',
  222. 'toastr' => '@toastr',
  223. 'datatables' => '@datatables',
  224. 'dcat' => '@dcat',
  225. ];
  226. /**
  227. * 基础js.
  228. *
  229. * @var array
  230. */
  231. public $baseJs = [
  232. 'adminlte' => '@adminlte',
  233. 'toastr' => '@toastr',
  234. 'pjax' => '@pjax',
  235. 'validator' => '@validator',
  236. 'layer' => '@layer',
  237. 'init' => '@jquery.initialize',
  238. ];
  239. /**
  240. * @var array
  241. */
  242. public $fonts = [
  243. '@nunito',
  244. ];
  245. /**
  246. * @var bool
  247. */
  248. protected $isPjax = false;
  249. /**
  250. * Assets constructor.
  251. */
  252. public function __construct()
  253. {
  254. $this->isPjax = request()->pjax();
  255. $this->initTheme();
  256. }
  257. /**
  258. * 初始化主题样式.
  259. */
  260. protected function initTheme()
  261. {
  262. $color = Admin::color()->getName();
  263. if ($color === Color::DEFAULT_COLOR) {
  264. return;
  265. }
  266. $alias = [
  267. '@adminlte',
  268. '@dcat',
  269. ];
  270. foreach ($alias as $n) {
  271. $before = (array) $this->alias[$n]['css'];
  272. $this->alias[$n]['css'] = [];
  273. foreach ($before as $css) {
  274. $this->alias[$n]['css'][] = str_replace('.css', "-{$color}.css", $css);
  275. }
  276. }
  277. }
  278. /**
  279. * 设置或获取别名.
  280. *
  281. * @param string|array $name
  282. * @param string|array $value
  283. *
  284. * @return void|array
  285. */
  286. public function alias($name, $value = null)
  287. {
  288. if (is_array($name)) {
  289. foreach ($name as $key => $value) {
  290. $this->alias($key, $value);
  291. }
  292. return;
  293. }
  294. if ($value === null) {
  295. return $this->getAlias($name);
  296. }
  297. if (mb_strpos($name, '@') !== 0) {
  298. $name = '@'.$name;
  299. }
  300. $this->alias[$name] = $value;
  301. }
  302. /**
  303. * 获取别名.
  304. *
  305. * @param string $name
  306. * @param array $params
  307. *
  308. * @return array|string
  309. */
  310. public function getAlias($name, array $params = [])
  311. {
  312. if (mb_strpos($name, '@') !== 0) {
  313. $name = '@'.$name;
  314. }
  315. [$name, $query] = $this->parseParams($name);
  316. $assets = $this->alias[$name] ?? [];
  317. // 路径别名
  318. if (is_string($assets)) {
  319. return $assets;
  320. }
  321. $params += $query;
  322. return [
  323. 'js' => $this->normalizeAliasPaths($assets['js'] ?? [], $params) ?: null,
  324. 'css' => $this->normalizeAliasPaths($assets['css'] ?? [], $params) ?: null,
  325. ];
  326. }
  327. /**
  328. * @param array $files
  329. * @param array $params
  330. *
  331. * @return array
  332. */
  333. protected function normalizeAliasPaths($files, array $params)
  334. {
  335. $files = (array) $files;
  336. foreach ($files as &$file) {
  337. foreach ($params as $k => $v) {
  338. $file = str_replace("{{$k}}", $v, $file);
  339. }
  340. }
  341. return array_filter($files, function ($file) {
  342. return ! mb_strpos($file, '{');
  343. });
  344. }
  345. /**
  346. * 解析参数.
  347. *
  348. * @param string $name
  349. *
  350. * @return array
  351. */
  352. protected function parseParams($name)
  353. {
  354. $name = explode('?', $name);
  355. if (empty($name[1])) {
  356. return [$name[0], []];
  357. }
  358. parse_str($name[1], $params);
  359. return [$name[0], $params];
  360. }
  361. /**
  362. * 根据别名设置需要载入的js和css脚本.
  363. *
  364. * @param string|array $alias
  365. * @param array $params
  366. *
  367. * @return void
  368. */
  369. public function require($alias, array $params = [])
  370. {
  371. if (is_array($alias)) {
  372. foreach ($alias as $v) {
  373. $this->require($v);
  374. }
  375. return;
  376. }
  377. $assets = $this->getAlias($alias, $params);
  378. $this->js($assets['js']);
  379. $this->css($assets['css']);
  380. }
  381. /**
  382. * 设置需要载入的css脚本.
  383. *
  384. * @param string|array $css
  385. */
  386. public function css($css)
  387. {
  388. if (! $css) {
  389. return;
  390. }
  391. $this->css = array_merge(
  392. $this->css,
  393. (array) $css
  394. );
  395. }
  396. /**
  397. * 设置需要载入的基础css脚本.
  398. *
  399. * @param array $css
  400. */
  401. public function baseCss(array $css, bool $merge = false)
  402. {
  403. if ($merge) {
  404. $this->baseCss = array_merge($this->baseCss, $css);
  405. } else {
  406. $this->baseCss = $css;
  407. }
  408. }
  409. /**
  410. * 设置需要载入的js脚本.
  411. *
  412. * @param string|array $js
  413. */
  414. public function js($js)
  415. {
  416. if (! $js) {
  417. return;
  418. }
  419. $this->js = array_merge(
  420. $this->js,
  421. (array) $js
  422. );
  423. }
  424. /**
  425. * 根据别名获取资源路径.
  426. *
  427. * @param string $path
  428. * @param string $type
  429. *
  430. * @return string|array|null
  431. */
  432. public function get($path, string $type = 'js')
  433. {
  434. if (empty($this->alias[$path])) {
  435. return $this->url($path);
  436. }
  437. $paths = isset($this->alias[$path][$type]) ? (array) $this->alias[$path][$type] : null;
  438. if (! $paths) {
  439. return $paths;
  440. }
  441. foreach ($paths as &$value) {
  442. $value = $this->url($value);
  443. }
  444. return $paths;
  445. }
  446. /**
  447. * 获取静态资源完整URL.
  448. *
  449. * @param string $path
  450. *
  451. * @return string
  452. */
  453. public function url($path)
  454. {
  455. if (! $path) {
  456. return $path;
  457. }
  458. $path = $this->getRealPath($path);
  459. if (mb_strpos($path, '//') === false) {
  460. $path = config('admin.assets_server').'/'.trim($path, '/');
  461. }
  462. return (config('admin.https') || config('admin.secure')) ? secure_asset($path) : asset($path);
  463. }
  464. /**
  465. * 获取真实路径.
  466. *
  467. * @param string|null $path
  468. *
  469. * @return string|null
  470. */
  471. public function getRealPath(?string $path)
  472. {
  473. if (! $this->containsAlias($path)) {
  474. return $path;
  475. }
  476. return implode(
  477. '/',
  478. array_map(
  479. function ($v) {
  480. if (! $this->isPathAlias($v)) {
  481. return $v;
  482. }
  483. return $this->getRealPath($this->alias($v));
  484. },
  485. explode('/', $path)
  486. )
  487. );
  488. }
  489. /**
  490. * 判断是否是路径别名.
  491. *
  492. * @param mixed $value
  493. *
  494. * @return bool
  495. */
  496. public function isPathAlias($value)
  497. {
  498. return $this->hasAlias($value) && is_string($this->alias[$value]);
  499. }
  500. /**
  501. * 判断别名是否存在.
  502. *
  503. * @param $value
  504. *
  505. * @return bool
  506. */
  507. public function hasAlias($value)
  508. {
  509. return isset($this->alias[$value]);
  510. }
  511. /**
  512. * 判断是否含有别名.
  513. *
  514. * @param string $value
  515. *
  516. * @return bool
  517. */
  518. protected function containsAlias($value)
  519. {
  520. return $value && mb_strpos($value, '@') === 0;
  521. }
  522. /**
  523. * 设置在head标签内加载的js.
  524. *
  525. * @param string|array $js
  526. */
  527. public function headerJs($js)
  528. {
  529. if (! $js) {
  530. return;
  531. }
  532. $this->headerJs = array_merge($this->headerJs, (array) $js);
  533. }
  534. /**
  535. * 设置基础js脚本.
  536. *
  537. * @param array $js
  538. * @param bool $merge
  539. */
  540. public function baseJs(array $js, bool $merge = true)
  541. {
  542. if ($merge) {
  543. $this->baseJs = array_merge($this->baseJs, $js);
  544. } else {
  545. $this->baseJs = $js;
  546. }
  547. }
  548. /**
  549. * 设置js代码.
  550. *
  551. * @param string|array $script
  552. * @param bool $direct
  553. */
  554. public function script($script, bool $direct = false)
  555. {
  556. if (! $script) {
  557. return;
  558. }
  559. if ($direct) {
  560. $this->directScript = array_merge($this->directScript, (array) $script);
  561. } else {
  562. $this->script = array_merge($this->script, (array) $script);
  563. }
  564. }
  565. /**
  566. * 设置css代码.
  567. *
  568. * @param string $style
  569. */
  570. public function style($style)
  571. {
  572. if (! $style) {
  573. return;
  574. }
  575. $this->style = array_merge($this->style, (array) $style);
  576. }
  577. /**
  578. * 字体css脚本路径.
  579. */
  580. protected function addFontCss()
  581. {
  582. $this->fonts && ($this->baseCss = array_merge(
  583. $this->baseCss,
  584. (array) $this->fonts
  585. ));
  586. }
  587. /**
  588. * 合并基础css脚本.
  589. */
  590. protected function mergeBaseCss()
  591. {
  592. if ($this->isPjax) {
  593. return;
  594. }
  595. $this->addFontCss();
  596. $this->css = array_merge($this->baseCss, $this->css);
  597. }
  598. /**
  599. * @return string
  600. */
  601. public function cssToHtml()
  602. {
  603. $this->mergeBaseCss();
  604. $html = '';
  605. foreach (array_unique($this->css) as &$v) {
  606. if (! $paths = $this->get($v, 'css')) {
  607. continue;
  608. }
  609. foreach ((array) $paths as $path) {
  610. $html .= "<link rel=\"stylesheet\" href=\"{$this->withVersionQuery($path)}\">";
  611. }
  612. }
  613. return $html;
  614. }
  615. /**
  616. * @param string $url
  617. *
  618. * @return string
  619. */
  620. public function withVersionQuery($url)
  621. {
  622. if (! Str::contains($url, '?')) {
  623. $url .= '?';
  624. }
  625. $ver = 'v'.Admin::VERSION;
  626. return Str::endsWith($url, '?') ? $url.$ver : $url.'&'.$ver;
  627. }
  628. /**
  629. * 合并基础js脚本.
  630. */
  631. protected function mergeBaseJs()
  632. {
  633. if ($this->isPjax) {
  634. return;
  635. }
  636. $this->js = array_merge($this->baseJs, $this->js);
  637. }
  638. /**
  639. * @return string
  640. */
  641. public function jsToHtml()
  642. {
  643. $this->mergeBaseJs();
  644. $html = '';
  645. foreach (array_unique($this->js) as &$v) {
  646. if (! $paths = $this->get($v, 'js')) {
  647. continue;
  648. }
  649. foreach ((array) $paths as $path) {
  650. $html .= "<script src=\"{$this->withVersionQuery($path)}\"></script>";
  651. }
  652. }
  653. return $html;
  654. }
  655. /**
  656. * @return string
  657. */
  658. public function headerJsToHtml()
  659. {
  660. $html = '';
  661. foreach (array_unique($this->headerJs) as &$v) {
  662. if (! $paths = $this->get($v, 'js')) {
  663. continue;
  664. }
  665. foreach ((array) $paths as $path) {
  666. $html .= "<script src=\"{$this->withVersionQuery($path)}\"></script>";
  667. }
  668. }
  669. return $html;
  670. }
  671. /**
  672. * @return string
  673. */
  674. public function scriptToHtml()
  675. {
  676. $script = implode(';', array_unique($this->script));
  677. $directScript = implode(';', array_unique($this->directScript));
  678. return <<<HTML
  679. <script data-exec-on-popstate>
  680. (function () {
  681. try {
  682. {$directScript}
  683. } catch (e) {
  684. console.error(e)
  685. }
  686. })();
  687. Dcat.ready(function () {
  688. try {
  689. {$script}
  690. } catch (e) {
  691. console.error(e)
  692. }
  693. })
  694. </script>
  695. HTML;
  696. }
  697. /**
  698. * @return string
  699. */
  700. public function styleToHtml()
  701. {
  702. $style = implode('', array_unique($this->style));
  703. return "<style>$style</style>";
  704. }
  705. }