Edit.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549
  1. <template>
  2. <el-dialog
  3. :close-on-click-modal="false" :title="title" :type="type" :visible.sync="isVisible"
  4. center width="48%"
  5. top="50px" class="group-dialog"
  6. >
  7. <div class="add-dialog-component">
  8. <el-alert :style="{ backgroundColor: '#fff' }" title="保存后将生效在移动端的分类里" type="info" :closable="false" />
  9. <el-tree
  10. class="tree-box" :data="treeData" :props="{ children: 'childs' }" node-key="id"
  11. default-expand-all draggable
  12. :expand-on-click-node="false"
  13. >
  14. <div slot-scope="{ node, data }" class="custom-tree-node">
  15. <div class="content">
  16. <template v-if="data.depth < 3">
  17. <el-input
  18. v-model="data.categoryName" class="input" :disabled="isCheck" maxlength="6"
  19. size="mini"
  20. :placeholder="data.depth | placeholderTips"
  21. />
  22. <el-upload
  23. class="upload-uploader" :on-success="handleImageSuccessOne" :multiple="false"
  24. :show-file-list="false" :action="action" :file-list="data.categoryImgArray"
  25. >
  26. <img
  27. v-if="data.categoryImgArray.length && data.categoryImgArray[0].imgPath" width="80" height="80"
  28. :src="data.categoryImgArray.length && data.categoryImgArray[0].imgPath"
  29. >
  30. <i v-else class="el-icon-plus" />
  31. </el-upload>
  32. </template>
  33. <template v-else>
  34. <div class="content" :class="isCheck ? 'disabled' : ''">
  35. <el-input
  36. v-model="data.categoryName" class="input" :disabled="isCheck" size="mini"
  37. maxlength="6"
  38. placeholder="输入三级类别名称(最大6个字符)"
  39. />
  40. <el-upload
  41. :headers="headers" :data="dataObj" :multiple="false" :show-file-list="false"
  42. :file-list="data.categoryImgArray" :on-success="handleImageSuccess" class="upload-uploader"
  43. :action="action"
  44. >
  45. <img
  46. v-if="data.categoryImgArray && data.categoryImgArray[0].url
  47. " width="80" height="80" :src="data.categoryImgArray && data.categoryImgArray[0].url
  48. "
  49. >
  50. <i v-else class="el-icon-plus" />
  51. </el-upload>
  52. </div>
  53. </template>
  54. </div>
  55. <div v-if="!isCheck" class="setting-box">
  56. <el-button type="text" size="mini" @click="() => append(data)">
  57. {{ data.depth | addTips }}
  58. </el-button>
  59. <el-button type="text" size="mini" @click="() => remove(node, data)">
  60. 删除
  61. </el-button>
  62. </div>
  63. </div>
  64. </el-tree>
  65. <div class="add-btn-wrap">
  66. <template v-if="isCheck">
  67. <el-button class="add" type="primary" size="small" @click="close">
  68. 确定
  69. </el-button>
  70. </template>
  71. <template v-else>
  72. <el-button v-if="type === 'add'" class="add" type="primary" size="small" @click="addClassification">
  73. 添加一级类别名称
  74. </el-button>
  75. <el-button type="primary" size="small" @click="onSubmit">
  76. 保存
  77. </el-button>
  78. </template>
  79. </div>
  80. </div>
  81. </el-dialog>
  82. </template>
  83. <script>
  84. import { uploadUrl } from '@/utils/request'
  85. import {
  86. getSelect,
  87. commdityClassAdd,
  88. commdityClassgetById,
  89. commdityClassUpdate
  90. } from '@/api/renovation'
  91. let idx = 1000
  92. let imgId = 1
  93. export default {
  94. filters: {
  95. addTips(depth) {
  96. depth = depth + ''
  97. const tipsMp = {
  98. 1: '添加二级类别名称',
  99. 2: '添加三级类别名称'
  100. }
  101. return tipsMp[depth]
  102. },
  103. placeholderTips(depth) {
  104. depth = depth + ''
  105. const tipsMp = {
  106. 1: '输入一级类别名称(最大6个字符)',
  107. 2: '输入二级类别名称(最大6个字符)'
  108. }
  109. return tipsMp[depth]
  110. }
  111. },
  112. props: {
  113. dialogVisible: {
  114. type: Boolean,
  115. default: false
  116. },
  117. type: {
  118. type: String,
  119. default: 'add'
  120. }
  121. },
  122. data() {
  123. return {
  124. params: {
  125. categoryName: ''
  126. },
  127. file: this.image ? this.image : '',
  128. imgList: [],
  129. customParams: {
  130. current: 1,
  131. map: {},
  132. model: {
  133. config: '',
  134. isCustom: true,
  135. isDelete: 0,
  136. name: ''
  137. },
  138. order: 'descending',
  139. size: 100,
  140. sort: 'id'
  141. },
  142. treeData: [],
  143. headers: {
  144. Authorization: ''
  145. },
  146. action: uploadUrl,
  147. dataObj: {
  148. folderId: 1
  149. },
  150. deleteArr: []
  151. }
  152. },
  153. computed: {
  154. isVisible: {
  155. get() {
  156. return this.dialogVisible
  157. },
  158. set() {
  159. this.close()
  160. this.reset()
  161. }
  162. },
  163. title() {
  164. const stateMap = {
  165. add: '新建类别',
  166. edit: '修改类别',
  167. check: '查看类别'
  168. }
  169. return stateMap[this.type]
  170. },
  171. isCheck() {
  172. return this.type === 'check'
  173. }
  174. },
  175. async created() {
  176. /*
  177. const res = await getSelect({
  178. dictName: '商品类别链接'
  179. })
  180. console.log(res)
  181. this.dictList = res.data
  182. */
  183. },
  184. methods: {
  185. async queryOneCategory(oneClassifyId) {
  186. console.log(oneClassifyId)
  187. if (oneClassifyId === undefined) {
  188. this.treeData = []
  189. return
  190. }
  191. const res = await commdityClassgetById({
  192. oneClassifyId
  193. })
  194. console.log(res)
  195. const resData = res.data
  196. const treeFilter = (item) => {
  197. const {
  198. categoryName,
  199. categoryImg,
  200. categoryPath,
  201. parentName,
  202. categoryImgArray,
  203. depth,
  204. id,
  205. link
  206. } = item
  207. const newMap = {
  208. depth,
  209. categoryName,
  210. categoryPath: categoryPath || '',
  211. parentName,
  212. categoryImgArray,
  213. link,
  214. id
  215. }
  216. console.log(depth)
  217. if (depth === 3) {
  218. newMap.categoryImgArray = [
  219. {
  220. url: categoryImg
  221. }
  222. ]
  223. }
  224. if (item.childs && item.childs.length) {
  225. newMap.childs = item.childs.map(treeFilter)
  226. }
  227. return newMap
  228. }
  229. if (resData) {
  230. resData.childs =
  231. resData && resData.childs && resData.childs.map(treeFilter)
  232. this.treeData = [ resData ]
  233. } else {
  234. this.treeData = []
  235. }
  236. console.log(this.treeData)
  237. },
  238. handleImageSuccess(response, file, fileList) {
  239. console.log(response)
  240. const url = response.data.url
  241. fileList[0].url = url
  242. },
  243. handleImageSuccessOne(response, file, fileList) {
  244. const url = response.data.url
  245. console.log(fileList)
  246. fileList[0].imgPath = url
  247. },
  248. close() {
  249. this.$emit('close')
  250. },
  251. reset() {
  252. this.treeData = []
  253. },
  254. addClassification() {
  255. this.treeData.push({
  256. placeholder: '输入一级类别名称',
  257. addTips: '添加二级类别名称',
  258. categoryName: '',
  259. categoryPath: '',
  260. depth: 1,
  261. idx: idx++,
  262. categoryImgArray: [
  263. {
  264. id: imgId++,
  265. imgPath: ''
  266. }
  267. ]
  268. })
  269. },
  270. append(data) {
  271. console.log(data)
  272. const { categoryName } = data
  273. const depth = data.depth + 1
  274. let newChild
  275. if (!data.childs) {
  276. this.$set(data, 'childs', [])
  277. }
  278. if (depth < 3) {
  279. newChild = {
  280. placeholder: '输入二级类别名称',
  281. addTips: '添加三级类别名称',
  282. depth,
  283. parentName: categoryName,
  284. categoryName: '',
  285. categoryPath: '',
  286. categoryImgArray: [
  287. {
  288. id: imgId++,
  289. imgPath: ''
  290. }
  291. ],
  292. childs: [],
  293. idx: idx++
  294. }
  295. } else {
  296. newChild = {
  297. parentName: categoryName,
  298. categoryImgArray: [
  299. {
  300. id: imgId++,
  301. url: ''
  302. }
  303. ],
  304. depth: 3,
  305. categoryName: '',
  306. idx: idx++
  307. }
  308. }
  309. data.childs.push(newChild)
  310. },
  311. remove(node, data) {
  312. const parent = node.parent
  313. const children = parent.data.childs || parent.data
  314. const index = children.findIndex((d) => d.idx === data.idx)
  315. if (index !== -1) {
  316. children.splice(index, 1)
  317. this.deleteArr.push(data.id || '')
  318. } else {
  319. this.$message.warning('数据错误,请重试')
  320. }
  321. },
  322. onSubmit() {
  323. console.log(this.type)
  324. if (this.type === 'add') {
  325. this.addGroup()
  326. } else {
  327. this.updateGroup()
  328. }
  329. },
  330. async addGroup() {
  331. console.log(this.treeData)
  332. const treeFilter = (item) => {
  333. const {
  334. categoryName,
  335. categoryImgArray,
  336. categoryPath,
  337. parentName,
  338. depth,
  339. link
  340. } = item
  341. const newMap = {
  342. depth,
  343. categoryName,
  344. categoryPath: categoryPath || '',
  345. parentName,
  346. link
  347. }
  348. if (categoryImgArray) {
  349. newMap.categoryImg = categoryImgArray[0].imgPath
  350. }
  351. if (depth === 3) {
  352. newMap.categoryImg = categoryImgArray[0].url
  353. }
  354. if (!newMap.categoryImg) {
  355. this.$message.error('分类\"' + categoryName + '\"请上传分类图片')
  356. throw new Error('未上传分类图片')
  357. }
  358. if (item.childs && item.childs.length) {
  359. newMap.childs = item.childs.map(treeFilter)
  360. }
  361. return newMap
  362. }
  363. const params = this.treeData.map(treeFilter)
  364. console.log(params)
  365. if (params.length === 0) {
  366. this.$message.error('请添加分类')
  367. return
  368. }
  369. const obj = {
  370. classifies: params
  371. }
  372. const res = await commdityClassAdd(obj)
  373. if (res.code === '') {
  374. this.isVisible = false
  375. this.$message({
  376. message: '新增成功',
  377. type: 'success'
  378. })
  379. this.$emit('success')
  380. this.deleteArr = []
  381. }
  382. },
  383. async updateGroup() {
  384. const treeFilter = (item) => {
  385. const {
  386. categoryName,
  387. categoryImgArray,
  388. categoryPath,
  389. parentName,
  390. depth,
  391. id,
  392. link
  393. } = item
  394. const newMap = {
  395. depth,
  396. categoryName,
  397. categoryPath: categoryPath || '',
  398. parentName,
  399. link,
  400. id
  401. }
  402. if (categoryImgArray) {
  403. newMap.categoryImg = categoryImgArray[0].imgPath
  404. }
  405. if (depth === 3) {
  406. newMap.categoryImg = categoryImgArray[0].url
  407. }
  408. if (!newMap.categoryImg) {
  409. this.$message.error('分类\"' + categoryName + '\"请上传分类图片')
  410. throw new Error('未上传分类图片')
  411. }
  412. if (item.childs && item.childs.length) {
  413. newMap.childs = item.childs.map(treeFilter)
  414. }
  415. return newMap
  416. }
  417. console.log(this.treeData)
  418. const params = this.treeData.map(treeFilter)
  419. const obj = {
  420. classifies: params,
  421. deleteIds: this.deleteArr
  422. }
  423. const res = await commdityClassUpdate(obj)
  424. if (res.code === '') {
  425. this.isVisible = false
  426. this.$message({
  427. message: '成功',
  428. type: 'success'
  429. })
  430. this.$emit('success')
  431. this.deleteArr = []
  432. }
  433. },
  434. setParams({ id }) {
  435. console.log(id)
  436. this.queryOneCategory(id)
  437. }
  438. }
  439. }
  440. </script>
  441. <style lang="scss">
  442. .add-dialog-component {
  443. .tree-box {
  444. .el-tree-node__content {
  445. margin-bottom: 15px;
  446. height: auto;
  447. }
  448. }
  449. }
  450. </style>
  451. <style
  452. lang="scss"
  453. scoped
  454. >
  455. @import url("../../../styles/elDialog.scss");
  456. .group-dialog {
  457. .el-dialog {
  458. min-width: 500px;
  459. max-width: 600px;
  460. }
  461. }
  462. .add-dialog-component {
  463. padding: 15px 20px;
  464. max-height: 60vh;
  465. overflow: auto;
  466. .el-tree-node__content {
  467. &:hover {
  468. background-color: #fff;
  469. }
  470. }
  471. .tree-box {
  472. margin: 15px 0;
  473. .custom-tree-node {
  474. display: flex;
  475. width: 100%;
  476. text-align: left;
  477. .content {
  478. flex: 1;
  479. display: flex;
  480. align-items: center;
  481. .input {
  482. width: 60%;
  483. margin-right: 20px;
  484. }
  485. .textarea-input {
  486. width: 100%;
  487. margin-left: 15px;
  488. textarea {
  489. height: 80px;
  490. }
  491. }
  492. }
  493. .level-3-wrap {
  494. display: flex;
  495. width: 300px;
  496. .upload-wrap {
  497. position: relative;
  498. &.disabled::after {
  499. content: "";
  500. position: absolute;
  501. left: 0;
  502. right: 0;
  503. top: 0;
  504. bottom: 0;
  505. z-index: 999;
  506. background-color: #f5f7fa;
  507. opacity: 0.5;
  508. }
  509. .el-upload {
  510. border: 1px dashed #d9d9d9;
  511. }
  512. i {}
  513. img {
  514. width: 80px;
  515. height: 80px;
  516. }
  517. }
  518. }
  519. }
  520. }
  521. .add-btn-wrap {
  522. text-align: center;
  523. }
  524. }
  525. .upload-uploader {
  526. margin-left: 30px;
  527. }
  528. </style>