Edit.vue 13 KB

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