images.php 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521
  1. <!DOCTYPE html>
  2. <!--suppress JSAnnotator -->
  3. <html lang="zh-CN">
  4. <head>
  5. <link rel="stylesheet" href="{__FRAME_PATH}font-awesome/css/font-awesome.min.css">
  6. <link href="{__PLUG_PATH}layui/css/layui.css" rel="stylesheet">
  7. <link rel="stylesheet" href="{__ADMIN_PATH}css/images.css?v=100">
  8. <script src="{__PLUG_PATH}jquery-1.10.2.min.js"></script>
  9. <script src="{__PLUG_PATH}layui/layui.all.js"></script>
  10. <script src="{__PLUG_PATH}vue/dist/vue.min.js"></script>
  11. </head>
  12. <style>
  13. </style>
  14. <body>
  15. <div v-cloak id="app" class="layui-fluid">
  16. <div class="layui-fluid-side">
  17. <div class="layui-tree" lay-filter="LAY-tree-8">
  18. <div data-id="2" class="layui-tree-set layui-tree-setHide">
  19. <div class="layui-tree-entry">
  20. <div class="layui-tree-main" @click="OpenTree({ child: [], id: 0 })">
  21. <span style="visibility: hidden;" class="layui-tree-iconClick">
  22. <i class="layui-tree-iconArrow"></i>
  23. </span>
  24. <span :class="{ on: pid == 0 }" class="layui-tree-txt">全部图片</span>
  25. </div>
  26. </div>
  27. </div>
  28. <div v-for="(item, index) in categoryList" :key="item.id" :class="{ 'layui-tree-spread': item.isOpen }" data-id="2" class="layui-tree-set layui-tree-setHide">
  29. <div class="layui-tree-entry">
  30. <div class="layui-tree-main" @click="OpenTree(item, index)">
  31. <span :style="{ visibility: item.child.length ? 'visible' : 'hidden' }" class="layui-tree-iconClick">
  32. <i class="layui-tree-iconArrow"></i>
  33. </span>
  34. <span :class="{ on: pid == item.id }" class="layui-tree-txt">{{ item.name }}</span>
  35. </div>
  36. <div class="layui-layer layui-layer-tips">
  37. <div id="" class="layui-layer-content">
  38. <div class="layui-btn-group layui-tree-btnGroup">
  39. <i class="layui-icon layui-icon-add-1" data-type="add" @click.stop="addCategory(item)"></i>
  40. <i class="layui-icon layui-icon-edit" data-type="update" @click.stop="updateCategory(item)"></i>
  41. <i v-if="!item.child.length" class="layui-icon layui-icon-delete" data-type="del" @click.stop="delCategory(item)"></i>
  42. </div>
  43. <i class="layui-layer-TipsG layui-layer-TipsT"></i>
  44. </div>
  45. <span class="layui-layer-setwin"></span>
  46. </div>
  47. </div>
  48. <div :style="{ display: item.isOpen ? 'block' : 'none' }" class="layui-tree-pack layui-tree-lineExtend layui-tree-showLine">
  49. <div v-for="cell in item.child" :key="cell.id" data-id="2000" class="layui-tree-set">
  50. <div class="layui-tree-entry">
  51. <div class="layui-tree-main" @click="OpenTree(cell, index)">
  52. <span class="layui-tree-iconClick" style="visibility: hidden;">
  53. <i class="layui-tree-iconArrow" style="display: none;"></i>
  54. </span>
  55. <span :class="{ on: pid == cell.id }" class="layui-tree-txt">{{ cell.name }}</span>
  56. </div>
  57. <div class="layui-layer layui-layer-tips">
  58. <div id="" class="layui-layer-content">
  59. <div class="layui-btn-group layui-tree-btnGroup">
  60. <i class="layui-icon layui-icon-edit" data-type="update" @click.stop="updateCategory(cell)"></i>
  61. <i class="layui-icon layui-icon-delete" data-type="del" @click.stop="delCategory(cell)"></i>
  62. </div>
  63. <i class="layui-layer-TipsG layui-layer-TipsT"></i>
  64. </div>
  65. <span class="layui-layer-setwin"></span>
  66. </div>
  67. </div>
  68. </div>
  69. </div>
  70. </div>
  71. </div>
  72. </div>
  73. <div class="layui-fluid-main">
  74. <div class="main-header">
  75. <div class="layui-btn-group">
  76. <button :class="{ 'layui-btn-normal': selectedImage.length, 'layui-btn-disabled': !selectedImage.length }" type="button" class="layui-btn layui-btn-sm" @click="useImage">使用选中的图片</button>
  77. <button type="button" class="layui-btn layui-btn-normal layui-btn-sm" @click="addCategory">添加分类</button>
  78. <button ref="upload" type="button" class="layui-btn layui-btn-normal layui-btn-sm">上传图片</button>
  79. <button :class="{ 'layui-btn-normal': selectedImage.length, 'layui-btn-disabled': !selectedImage.length }" type="button" class="layui-btn layui-btn-sm" @click="moveCategory">图片移至</button>
  80. <button :class="{ 'layui-btn-danger': selectedImage.length, 'layui-btn-disabled': !selectedImage.length }" type="button" class="layui-btn layui-btn-sm" @click="deleteImage">删除图片</button>
  81. <button type="button" class="layui-btn layui-btn-normal layui-btn-sm" @click="searchImage"><i class="layui-icon layui-icon-search"></i></button>
  82. </div>
  83. </div>
  84. <div class="main-content">
  85. <div v-for="item in imageList" :key="item.att_id" class="image-item">
  86. <div :class="{ on: item.selected }" class="image-wrap" @click="selectImage(item)">
  87. <img :src="item.att_dir">
  88. <span v-show="item.number" class="layui-badge layui-bg-cyan">{{ item.number }}</span>
  89. <div class="layui-btn-group">
  90. <button type="button" class="layui-btn layui-btn-primary layui-btn-xs" @click.stop="lookImage(item.att_dir)">
  91. <i class="fa fa-eye"></i>
  92. </button>
  93. <button type="button" class="layui-btn layui-btn-primary layui-btn-xs" @click.stop="handleEdit(item)">
  94. <i class="fa fa-edit"></i>
  95. </button>
  96. </div>
  97. </div>
  98. <input :ref="item.att_id" v-model.trim="item.title" type="text" :disabled="item.disabled" @blur="handleBlur(item)">
  99. </div>
  100. <img v-if="!imageList.length && !loading" class="empty" src="{__ADMIN_PATH}images/empty.jpg">
  101. </div>
  102. <div ref="main-footer" class="main-footer"></div>
  103. </div>
  104. </div>
  105. </body>
  106. </html>
  107. <script src="{__ADMIN_PATH}js/layuiList.js"></script>
  108. <script>
  109. var form = layui.form;
  110. var pid = {$pid}, small = {$Request.param.small ? : 0} , parentinputname = '{$fodder}', maxLength = {$maxLength};//当前图片分类ID
  111. new Vue({
  112. el: "#app",
  113. data: {
  114. categoryList: [],
  115. searchTitle: '',
  116. pid: pid,
  117. imageList: [],
  118. page: 1,
  119. limit: 20,
  120. loading: false,
  121. small: small,
  122. selectedImage: [],
  123. selectedImageIDS: [],
  124. uploadInst: null,
  125. searchContent: '',
  126. files: {}
  127. },
  128. watch: {
  129. page: function () {
  130. this.getImageList();
  131. },
  132. },
  133. methods: {
  134. searchImage: function () {
  135. var vm = this;
  136. layer.prompt({
  137. title: '输入图片名称,并确认',
  138. formType: 0
  139. }, function (text, index) {
  140. if (!text.trim().length) {
  141. return;
  142. }
  143. this.pid = 0;
  144. this.searchContent = text.trim();
  145. this.page = 1;
  146. layer.close(index);
  147. var load = layer.load(1);
  148. layList.baseGet(this.U({
  149. a: 'get_image_list',
  150. q: {pid: this.pid, page: this.page, limit: this.limit, title: this.searchContent}
  151. }), function (res) {
  152. layer.close(load);
  153. var list = res.data.list;
  154. list.forEach(function (item) {
  155. item.disabled = true;
  156. item.hover = false;
  157. if (!item.title) {
  158. item.title = item.name.slice(0, item.name.lastIndexOf('.'));
  159. }
  160. });
  161. this.$set(this, 'imageList', list);
  162. if (this.page == 1) {
  163. layList.laypage.render({
  164. elem: this.$refs['main-footer']
  165. , count: res.data.count
  166. , limit: this.limit
  167. , theme: '#191C6E',
  168. groups: 3,
  169. jump: function (obj) {
  170. vm.page = obj.curr;
  171. }
  172. });
  173. }
  174. }.bind(this), function (res) {
  175. layer.close(load);
  176. layList.msg(res.msg);
  177. });
  178. }.bind(this));
  179. },
  180. handleEdit: function (item) {
  181. item.disabled = false;
  182. this.oldTitle = item.title;
  183. this.$nextTick(function () {
  184. this.$refs[item.att_id][0].focus();
  185. });
  186. },
  187. handleBlur: function (item) {
  188. item.disabled = true;
  189. if (this.oldTitle !== item.title) {
  190. this.updateTitle(item);
  191. }
  192. },
  193. lookImage: function (image) {
  194. layui.layer.photos({
  195. photos: {
  196. data: [
  197. {
  198. src: image
  199. }
  200. ]
  201. },
  202. anim: 5
  203. });
  204. },
  205. // 修改名称
  206. updateTitle: function (item) {
  207. layList.basePost(this.U({
  208. a: 'updateImageTitle'
  209. }), {
  210. att_id: item.att_id,
  211. title: item.title
  212. }, function (res) {
  213. layList.msg(res.msg);
  214. }, function (res) {
  215. layList.msg(res.msg);
  216. });
  217. },
  218. //删除图片
  219. deleteImage: function () {
  220. var that = this;
  221. if (!this.selectedImage.length) return;
  222. layList.layer.confirm('是否要删除选中图片?', {
  223. btn: ['是的我要删除', '我想想吧'] //按钮
  224. }, function () {
  225. layList.basePost(that.U({a: 'delete'}), {imageid: that.selectedImageIDS}, function (res) {
  226. layList.msg(res.msg);
  227. that.getImageList();
  228. window.location.reload()
  229. }, function (res) {
  230. layList.msg(res.msg);
  231. })
  232. })
  233. },
  234. //移动图片分类
  235. moveCategory: function () {
  236. var that = this;
  237. window.formSuccessBack = function () {
  238. that.getImageList();
  239. that.selectedImage = [];
  240. that.selectedImageIDS = [];
  241. };
  242. this.getOpenWindow('移动图片', this.U({a: 'moveimg'}) + '?imgaes=' + this.selectedImageIDS, {
  243. end: function () {
  244. window.formSuccessBack = null;
  245. }
  246. });
  247. },
  248. //使用选中图片
  249. useImage: function () {
  250. if (!this.selectedImage.length) return;
  251. //判断表单限制图片个数
  252. if (typeof parent.$f != 'undefined') {
  253. //已有图片个数
  254. var nowpics = parent.$f.getValue(parentinputname).length,
  255. props = parent.$f.model()[parentinputname].props || {},
  256. maxlength = props.maxLength || 0;
  257. //已选图片个数
  258. var selectlength = this.selectedImage.length;
  259. //还可以选择多少张
  260. var surplus = maxlength - nowpics;
  261. if (nowpics + selectlength > maxlength) {
  262. return layList.msg('最多只能选择 ' + surplus + ' 张');
  263. }
  264. }
  265. //编辑器中
  266. if (parentinputname.includes('editor')) {
  267. var list = this.selectedImage.map(function (image) {
  268. return {
  269. _src: image,
  270. src: image
  271. };
  272. });
  273. parent.insertEditor(list, parentinputname);
  274. var index = parent.layer.getFrameIndex(window.name);
  275. parent.layer.close(index)
  276. } else {
  277. //form表单中
  278. if (parent.$f) {
  279. var value = parent.$f.getValue(parentinputname);//父级input 值
  280. var list = value || [];
  281. for (var i = 0; i < this.selectedImage.length; i++) {
  282. if (value.indexOf(this.selectedImage[i]) == -1) list.push(this.selectedImage[i]);
  283. }
  284. parent.$f.changeField(parentinputname, list);
  285. parent.$f.closeModal(parentinputname);
  286. } else {
  287. //独立图片选择页面
  288. if(maxLength > 0 ){
  289. if(this.selectedImage.length > maxLength){
  290. return layList.msg('最多能选择' + maxLength + '张');
  291. }
  292. parent.changeIMG(parentinputname, this.selectedImage,1);
  293. }else{
  294. if(this.selectedImage.length > 1){
  295. return layList.msg('只能选择一张图片');
  296. }
  297. parent.changeIMG(parentinputname, this.selectedImage[0]);
  298. }
  299. var index = parent.layer.getFrameIndex(window.name);
  300. parent.layer.close(index);
  301. }
  302. }
  303. },
  304. //图片选中和取消
  305. selectImage: function (item, index) {
  306. if (!item.selected) {
  307. if (maxLength) {
  308. if (this.selectedImage.length === maxLength) {
  309. return layList.msg('最多选择' + maxLength + '张图片');
  310. }
  311. } else {
  312. if (parentinputname.indexOf('editor') === -1) {
  313. if (this.selectedImage.length) {
  314. return layList.msg('只能选择1张图片');
  315. }
  316. }
  317. }
  318. }
  319. this.$set(item, 'selected', item.selected == undefined ? true : !item.selected);
  320. var val = small == 1 ? item['satt_dir'] : item['att_dir'];
  321. if (item.selected === true) {
  322. this.selectedImage[this.selectedImage.length] = val;
  323. this.selectedImageIDS[this.selectedImage.length] = item['att_id'];
  324. item.number = this.selectedImage.length;
  325. } else {
  326. this.selectedImage.splice(this.selectedImage.indexOf(val), 1);
  327. this.selectedImageIDS.splice(this.selectedImage.indexOf(item['att_id']), 1);
  328. for (var i = 0; i < this.imageList.length; i++) {
  329. if (this.imageList[i].number > item.number) {
  330. this.imageList[i].number = (this.imageList[i].number - 1) >= 0 ? this.imageList[i].number - 1 : 0;
  331. }
  332. }
  333. item.number = 0;
  334. }
  335. this.$set(this, 'selectedImage', this.selectedImage);
  336. this.$set(this, 'selectedImageIDS', this.selectedImageIDS);
  337. },
  338. //获取图片列表
  339. getImageList: function () {
  340. var that = this;
  341. if (that.loading) return;
  342. that.loading = true;
  343. var index = layList.layer.load(1, {shade: [0.1, '#fff']});
  344. layList.baseGet(this.U({
  345. a: 'get_image_list',
  346. q: {pid: this.pid, page: this.page, limit: this.limit, title: that.searchContent }
  347. }), function (res) {
  348. that.loading = false;
  349. var list = res.data.list;
  350. for (var i = list.length; i--;) {
  351. list[i].disabled = true;
  352. list[i].hover = false;
  353. if (!list[i].title) {
  354. list[i].title = list[i].name.slice(0, list[i].name.lastIndexOf('.'));
  355. }
  356. }
  357. that.$set(that, 'imageList', res.data.list);
  358. layList.layer.close(index);
  359. if (that.page == 1) {
  360. layList.laypage.render({
  361. elem: that.$refs['main-footer']
  362. , count: res.data.count
  363. , limit: that.limit
  364. , theme: '#191C6E',
  365. jump: function (obj) {
  366. that.page = obj.curr;
  367. }
  368. });
  369. }
  370. }, function () {
  371. that.loading = false;
  372. layList.layer.close(index);
  373. });
  374. },
  375. //查询分类
  376. search: function () {
  377. this.getCategoryList();
  378. },
  379. //打开和关闭树形
  380. OpenTree: function (item, index) {
  381. this.searchContent = '';
  382. this.pid = item.id;
  383. if (item.child.length) {
  384. item.isOpen == undefined ? false : item.isOpen;
  385. this.$set(this.categoryList[index], 'isOpen', !item.isOpen);
  386. } else {
  387. this.page = 1;
  388. this.$set(this, 'selectedImage', []);
  389. this.$set(this, 'selectedImageIDS', []);
  390. this.getImageList();
  391. }
  392. this.uploadInst.reload({
  393. url: this.U({a: 'upload'}) + '?pid=' + this.pid
  394. });
  395. },
  396. //组装URL
  397. U: function (opt) {
  398. opt = typeof opt == 'object' ? opt : {};
  399. return layList.U({m: 'admin', c: "widget.images", a: opt.a || '', q: opt.q || {}, p: opt.q || {}});
  400. },
  401. //获取分类
  402. getCategoryList: function () {
  403. var that = this;
  404. layList.baseGet(that.U({a: 'get_image_cate', q: {name: this.searchTitle}}), function (res) {
  405. that.$set(that, 'categoryList', res.data);
  406. });
  407. },
  408. //鼠标移入显示图标
  409. changeActive: function (item, indexK, index) {
  410. if (index)
  411. this.$set(this.categoryList[indexK]['child'], 'isShow', true);
  412. else
  413. this.$set(this.categoryList[indexK], 'isShow', true);
  414. },
  415. //鼠标移出隐藏
  416. removeActive: function (item, indexK, index) {
  417. if (index)
  418. this.$set(this.categoryList[indexK]['child'], 'isShow', false);
  419. else
  420. this.$set(this.categoryList[indexK], 'isShow', false);
  421. },
  422. //添加分类
  423. addCategory: function (item, pid) {
  424. item = item == undefined ? {} : item;
  425. var id = item.id == undefined ? 0 : item.id,
  426. pid = pid == undefined ? 0 : pid;
  427. return this.getOpenWindow(item.name ? item.name + '编辑' : '新增分类', this.U({
  428. a: 'addcate',
  429. q: {id: pid == 0 ? id : pid}
  430. }));
  431. },
  432. //修改分类
  433. updateCategory: function (item, pid) {
  434. item = item == undefined ? {} : item;
  435. pid = pid == undefined ? 0 : pid;
  436. return this.getOpenWindow(item.name + '编辑', this.U({a: 'editcate', q: {id: item.id}}));
  437. },
  438. //删除分类
  439. delCategory: function (item, pid) {
  440. var that = this;
  441. if (item.child.length) return layList.msg('请先删除子分类再尝试删除此分类!');
  442. layList.layer.confirm('是否要删除[' + item.name + ']分类?', {
  443. btn: ['是的我要删除', '我想想吧'] //按钮
  444. }, function () {
  445. layList.baseGet(that.U({a: 'deletecate', q: {id: item.id}}), function (res) {
  446. layList.msg(res.msg, function () {
  447. that.getCategoryList();
  448. });
  449. }, function (err) {
  450. layList.msg(err.msg);
  451. });
  452. });
  453. },
  454. //打开一个窗口
  455. getOpenWindow: function (title, url, opt) {
  456. opt = opt || {};
  457. return layList.layer.open({
  458. type: 2,
  459. title: title,
  460. shade: [0],
  461. area: [opt.w || 340 + "px", opt.h || 265 + 'px'],
  462. anim: 2,
  463. content: [url, 'no'],
  464. end: opt.end || null
  465. });
  466. },
  467. //回调
  468. SuccessCateg: function () {
  469. this.getCategoryList();
  470. },
  471. renderUpload: function () {
  472. var vm = this;
  473. this.uploadInst = layui.upload.render({
  474. elem: this.$refs.upload,
  475. url: this.U({ a: 'upload' }),
  476. acceptMime: 'image/*',
  477. size: 2097152,
  478. before: function (obj) {
  479. layui.layer.load(1);
  480. vm.files = obj.pushFile();
  481. var file;
  482. for (var key in vm.files) {
  483. if (Object.hasOwnProperty.call(vm.files, key)) {
  484. file = vm.files[key];
  485. }
  486. }
  487. this.data = {
  488. pid: vm.pid,
  489. title: file.name.slice(0, file.name.lastIndexOf('.'))
  490. };
  491. },
  492. done: function (res, index, upload) {
  493. for (var key in vm.files) {
  494. if (Object.hasOwnProperty.call(vm.files, key)) {
  495. delete vm.files[key];
  496. }
  497. }
  498. layui.layer.closeAll('loading');
  499. layui.layer.msg(res.msg);
  500. vm.page = 1;
  501. vm.getImageList();
  502. },
  503. error: function (index, upload) {
  504. layui.layer.closeAll('loading');
  505. }
  506. });
  507. }
  508. },
  509. mounted: function () {
  510. this.getCategoryList();
  511. this.getImageList();
  512. window.SuccessCateg = this.SuccessCateg;
  513. this.renderUpload();
  514. }
  515. });
  516. </script>