index.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588
  1. <template>
  2. <el-dialog
  3. v-loading="loading"
  4. :title="title"
  5. :center="true"
  6. align="left"
  7. top="8vh"
  8. width="1040px"
  9. :close-on-click-modal="false"
  10. :visible.sync="showModelThis"
  11. element-loading-text="拼命加载中"
  12. element-loading-spinner="el-icon-loading"
  13. element-loading-background="rgba(0, 0, 0, 0.8)"
  14. append-to-body
  15. @close="showModelThis = false"
  16. >
  17. <el-card style="margin-top: -25px">
  18. <upload-excel :on-success="handleSuccess" :before-upload="beforeUpload" />
  19. <el-form ref="ruleForm" :model="ruleForm" :size="'mini'">
  20. <el-table
  21. :data="ruleForm.order_addr"
  22. border
  23. :size="'mini'"
  24. style="width: 100%"
  25. max-height="400px"
  26. row-key="key"
  27. >
  28. <el-table-column
  29. prop="receipt_quantity"
  30. label="收货总数"
  31. width="100"
  32. show-overflow-tooltip
  33. />
  34. <el-table-column
  35. prop="contactor"
  36. label="收件联系人"
  37. show-overflow-tooltip
  38. width="85"
  39. />
  40. <el-table-column
  41. prop="mobile"
  42. label="收货联系电话"
  43. show-overflow-tooltip
  44. width="100"
  45. />
  46. <el-table-column
  47. show-overflow-tooltip
  48. prop="in_addr"
  49. label="收货省市区(文件导入)"
  50. width="150"
  51. />
  52. <el-table-column
  53. show-overflow-tooltip
  54. prop="addr_code_name"
  55. label="收货省市区(系统解析)"
  56. width="150"
  57. />
  58. <el-table-column prop="addr" label="详细地址" show-overflow-tooltip />
  59. <el-table-column fixed="right" width="80" label="操作">
  60. <template slot-scope="scope">
  61. <el-tooltip effect="dark" content="编辑" placement="top">
  62. <i class="el-icon-edit tb-icon" @click="openHouseModal(scope.$index)" />
  63. </el-tooltip>
  64. <el-tooltip effect="dark" content="删除" placement="top">
  65. <i
  66. class="el-icon-delete tb-icon"
  67. @click="deleteRow(scope.$index, ruleForm.order_addr)"
  68. />
  69. </el-tooltip>
  70. </template>
  71. </el-table-column>
  72. </el-table>
  73. </el-form>
  74. <div class="tr" style="padding: 10px 0 0 0">
  75. <el-button type="primary" size="small" @click="submitForm">保 存 </el-button>
  76. </div>
  77. </el-card>
  78. </el-dialog>
  79. </template>
  80. <script>
  81. import resToken from '@/mixins/resToken'
  82. import { province_list, city_list, county_list } from '@/assets/js/area-data'
  83. import {
  84. isnumber,
  85. isMobile,
  86. isChinese,
  87. isEmoticon,
  88. isSpecialSymbol,
  89. hasSpace,
  90. isAddr
  91. } from '@/utils/validate'
  92. export default {
  93. name: 'InAddrModel',
  94. mixins: [resToken],
  95. props: ['showModel', 'id', 'sitem'],
  96. data() {
  97. const validatemobile = (rule, value, callback) => {
  98. if (value === '') {
  99. callback(new Error('联系电话不能为空!'))
  100. } else {
  101. if (!isMobile(value)) {
  102. callback(new Error('联系电话格式不正确!'))
  103. } else {
  104. callback()
  105. }
  106. }
  107. }
  108. const validateWeight = (rule, value, callback) => {
  109. if (value === '') {
  110. callback(new Error('收货总数不能为空!'))
  111. } else {
  112. if (!isnumber(value)) {
  113. callback(new Error('收货总数仅支持整数!'))
  114. } else {
  115. callback()
  116. }
  117. }
  118. }
  119. const validatecontactor = (rule, value, callback) => {
  120. if (value === '') {
  121. callback(new Error('联系人不能为空!'))
  122. } else {
  123. if (value.length < 2 || value.length > 10) {
  124. callback(new Error('联系人规则为2~10位汉字!'))
  125. } else {
  126. const isOk = value.split("").every(s => isChinese(s) || /^[A-Za-z]+$/.test(s));
  127. if (!isOk) {
  128. callback(new Error('联系人规则为2~10位汉字!'))
  129. } else if (isEmoticon(value)) {
  130. callback(new Error('联系人规则为2~10位汉字!'))
  131. } else {
  132. callback()
  133. }
  134. }
  135. }
  136. }
  137. const validateAddr = (rule, value, callback) => {
  138. if (value === '') {
  139. callback(new Error('详细地址不能为空!'))
  140. } else {
  141. if (hasSpace(value)) {
  142. callback(new Error('系统不允许输入转义字符,请检查后重新输入!'))
  143. } else if (isSpecialSymbol(value)) {
  144. callback(new Error('不能使用英文特殊字符!'))
  145. } else if (isAddr(value)) {
  146. callback()
  147. } else {
  148. callback(new Error('详细地址填写不规范!'))
  149. }
  150. }
  151. }
  152. return {
  153. showAddrAddEditModal: false,
  154. AddrAddEditModalIndex: -1,
  155. AddrAddEditModalSitem: {},
  156. tableData: [],
  157. tableHeader: [],
  158. title: '',
  159. showModelThis: false,
  160. loading: false,
  161. newTime: 0,
  162. pickerOptions: {
  163. disabledDate: (time) => {
  164. return time.getTime() < new Date().valueOf()
  165. }
  166. },
  167. rulesThis: this.rules,
  168. ruleForm: {
  169. order_addr: [] // 收货地址
  170. },
  171. rules: {
  172. receipt_quantity: [
  173. {
  174. required: true,
  175. validator: validateWeight,
  176. trigger: 'blur'
  177. }
  178. ],
  179. contactor: [
  180. {
  181. required: true,
  182. trigger: 'blur',
  183. validator: validatecontactor
  184. }
  185. ],
  186. mobile: [
  187. {
  188. required: true,
  189. validator: validatemobile,
  190. trigger: 'blur'
  191. }
  192. ],
  193. addr_code: [
  194. {
  195. type: 'array',
  196. required: false,
  197. trigger: 'change'
  198. }
  199. ],
  200. addr: [
  201. {
  202. required: true,
  203. validator: validateAddr,
  204. trigger: 'blur'
  205. }
  206. ]
  207. }
  208. }
  209. },
  210. watch: {
  211. showModel: function(val) {
  212. this.showModelThis = val
  213. if (val) {
  214. this.initForm()
  215. }
  216. },
  217. showModelThis(val) {
  218. if (!val) {
  219. this.$emit('cancel')
  220. this.$emit('update:showModel', val)
  221. }
  222. }
  223. },
  224. mounted() {
  225. this.get_code()
  226. },
  227. methods: {
  228. async resetForm() {
  229. // 重置
  230. await this.$nextTick(() => {
  231. if (this.$refs.ruleForm) {
  232. this.$refs.ruleForm.resetFields()
  233. this.$refs.ruleForm.clearValidate()
  234. this.ruleForm = {
  235. order_addr: []
  236. }
  237. }
  238. })
  239. },
  240. // 省市区删除行操作
  241. deleteRow(index, rows) {
  242. rows.splice(index, 1)
  243. },
  244. showAddrAddEditModalRefresh(e) {
  245. const { index, item } = e
  246. if (index === -1) {
  247. this.ruleForm.order_addr.push(JSON.parse(JSON.stringify(item)))
  248. } else {
  249. const {
  250. receipt_quantity,
  251. contactor,
  252. mobile,
  253. addr_code,
  254. addr_code_name,
  255. addr,
  256. id
  257. } = JSON.parse(JSON.stringify(item))
  258. this.ruleForm.order_addr[index].receipt_quantity = receipt_quantity
  259. this.ruleForm.order_addr[index].contactor = contactor
  260. this.ruleForm.order_addr[index].mobile = mobile
  261. this.ruleForm.order_addr[index].addr_code = addr_code
  262. this.ruleForm.order_addr[index].addr_code_name = addr_code_name
  263. this.ruleForm.order_addr[index].addr = addr
  264. this.ruleForm.order_addr[index].id = id
  265. }
  266. this.$refs.ruleForm.validateField('order_addr')
  267. },
  268. beforeUpload(file) {
  269. const isLt1M = file.size / 1024 < 500
  270. if (isLt1M) {
  271. return true
  272. }
  273. this.$message({
  274. message: '请不要上传大于500KB的文件.',
  275. type: 'warning'
  276. })
  277. return false
  278. },
  279. handleSuccess({ results, header }) {
  280. if (results.length === 0) {
  281. this.$message.error('表格无有效数据!')
  282. return
  283. }
  284. if (results.length > 500) {
  285. this.$message.error('地址数据不能超过500条!')
  286. return
  287. }
  288. const head = [
  289. '收货总数',
  290. '收货联系人',
  291. '收货联系电话',
  292. '收货省名称',
  293. '收货市名称',
  294. '收货区名称',
  295. '详细地址'
  296. ]
  297. if (head.length !== header.length) {
  298. this.$message.error('表头与导入模板不匹配!')
  299. return
  300. }
  301. let hederOk = true
  302. head.forEach((v1, i1) => {
  303. if (v1 !== header[i1].replace(/\s*/g, '')) {
  304. hederOk = false
  305. }
  306. })
  307. if (!hederOk) {
  308. this.$message.error('表头与导入模板不匹配!')
  309. return
  310. }
  311. this.tableHeader = header
  312. this.tableData = []
  313. const list = results
  314. const tableOk = true
  315. this.ruleForm.order_addr = []
  316. list.forEach((v1) => {
  317. const b = Object.values(v1)
  318. const item = this.get_code(b[3], b[4], b[5])
  319. const model = {
  320. receipt_quantity: b[0] + '',
  321. contactor: b[1] + '',
  322. mobile: b[2] + '',
  323. in_addr: b[3] + '/' + b[4] + '/' + b[5],
  324. addr_code_name: item.name + '',
  325. addr_code: item.code,
  326. addr: b[6] + '',
  327. edit: false
  328. }
  329. this.ruleForm.order_addr.push(model)
  330. })
  331. if (!tableOk) {
  332. this.$message.error('最晚收货日期不正确,请将表格格式转为文本上传!')
  333. }
  334. },
  335. openHouseModal(index) {
  336. this.AddrAddEditModalIndex = index
  337. if (index === -1) {
  338. this.AddrAddEditModalSitem = {}
  339. } else {
  340. this.AddrAddEditModalSitem = JSON.parse(
  341. JSON.stringify(this.ruleForm.order_addr[index])
  342. )
  343. }
  344. this.showAddrAddEditModal = true
  345. // let findex = this.ruleForm.order_addr.findIndex((v) => v.edit === true);
  346. // if (findex !== -1) {
  347. // this.$message.warning("当前已有地址在编辑,请保存后再试!");
  348. // return;
  349. // } else {
  350. // this.ruleForm.order_addr[index].edit = true;
  351. // }
  352. },
  353. unique(arr) {
  354. const hash = []
  355. for (let i = 0; i < arr.length; i++) {
  356. const index = hash.findIndex((v1) => v1.inv_number === arr[i].inv_number)
  357. if (index === -1) {
  358. hash.push(arr[i])
  359. }
  360. }
  361. return hash
  362. },
  363. setTime(time) {
  364. time = time.replace(/\//g, '-')
  365. time = time.replace(/\./g, '-')
  366. const key = new Date(time).valueOf() + ''
  367. if (key.length !== 13) {
  368. time = ''
  369. }
  370. return time
  371. },
  372. async initForm() {
  373. this.rulesThis = this.rules
  374. await this.resetForm()
  375. this.newTime = 0
  376. this.loading = true
  377. this.tableData = []
  378. this.tableHeader = []
  379. this.step = 1
  380. this.title = '批量导入收货地址信息'
  381. this.changea()
  382. this.loading = false
  383. },
  384. refreshAll() {
  385. this.showModelThis = false
  386. this.$emit('refresh')
  387. },
  388. changea() {
  389. this.newTime = new Date().valueOf()
  390. },
  391. async submitForm() {
  392. const { order_addr } = this.ruleForm
  393. if (order_addr.length < 1) {
  394. this.$message.error('导入数据不能为空!')
  395. this.loading = false
  396. return
  397. }
  398. let isEdit = false
  399. order_addr.forEach((v) => {
  400. v.err = false
  401. if (v.edit) {
  402. v.err = true
  403. isEdit = true
  404. }
  405. })
  406. if (isEdit) {
  407. this.$message.error('当前收货地址已在编辑,请保存后再试!')
  408. this.loading = false
  409. return
  410. }
  411. let cItem = null
  412. order_addr.some((x, i) => {
  413. cItem = this.checkItem(x)
  414. if (!cItem.isok) {
  415. cItem.index = i
  416. return true
  417. }
  418. })
  419. if (cItem && !cItem.isok) {
  420. this.$message.error(`第${cItem.index + 1}行,${cItem.message}`)
  421. this.loading = false
  422. return
  423. }
  424. const list = JSON.parse(JSON.stringify(order_addr))
  425. this.$emit('refresh', { list: list })
  426. },
  427. checkItem(sitem) {
  428. const model = {
  429. isok: true,
  430. message: ''
  431. }
  432. const { arrive_time, receipt_quantity, contactor, mobile, addr_code, addr } = sitem
  433. if (receipt_quantity === '' && model.isok) {
  434. model.isok = false
  435. model.message = '收货总数不能为空!'
  436. }
  437. if (!isnumber(receipt_quantity) && model.isok) {
  438. model.isok = false
  439. model.message = '收货总数仅支持整数!'
  440. }
  441. if (arrive_time === '' && model.isok) {
  442. model.isok = false
  443. model.message = '最晚收货日期不能为空!'
  444. }
  445. if (contactor === '' && model.isok) {
  446. model.isok = false
  447. model.message = '收货联系人不能为空!'
  448. }
  449. if (mobile === '' && model.isok) {
  450. model.isok = false
  451. model.message = '收货联系电话不能为空!'
  452. }
  453. if (!isMobile(mobile) && model.isok) {
  454. model.isok = false
  455. model.message = '收货联系电话格式不正确!'
  456. }
  457. // if (addr_code.length !== 3 && model.isok) {
  458. // model.isok = false;
  459. // model.message = "收货省市区不能为空!";
  460. // }
  461. if (addr === '' && model.isok) {
  462. model.isok = false
  463. model.message = '详细地址不能为空!'
  464. }
  465. return model
  466. },
  467. get_code(name1, name2, name3) {
  468. let name = ''
  469. let code = []
  470. if (name1 && name2 && name3) {
  471. for (const x in province_list) {
  472. if (name1 === province_list[x]) {
  473. code.push(x)
  474. name += province_list[x]
  475. break
  476. }
  477. }
  478. if (code.length === 1) {
  479. for (const y in city_list) {
  480. if (name2 === city_list[y]) {
  481. code.push(y)
  482. name += '/' + city_list[y]
  483. break
  484. }
  485. }
  486. }
  487. if (code.length === 2) {
  488. for (const z in county_list) {
  489. if (name3 === county_list[z]) {
  490. code.push(z)
  491. name += '/' + county_list[z]
  492. break
  493. }
  494. }
  495. }
  496. if (code.length === 3) {
  497. let str1 = ''
  498. let str2 = ''
  499. let isok = false
  500. str1 = code[0].slice(0, 2)
  501. str2 = code[1].slice(2, 4)
  502. if (
  503. code[1].indexOf(str1) === 0 &&
  504. code[2].indexOf(str1) === 0 &&
  505. code[2].indexOf(str2) == 2
  506. ) {
  507. isok = true
  508. }
  509. if (!isok) {
  510. name = ''
  511. code = []
  512. }
  513. } else {
  514. name = ''
  515. code = []
  516. }
  517. }
  518. return { name: name, code: code }
  519. }
  520. }
  521. }
  522. </script>
  523. <style lang="scss" scoped>
  524. .account {
  525. .gongshi {
  526. span {
  527. vertical-align: top;
  528. display: inline-block;
  529. color: #000;
  530. }
  531. .icon-span {
  532. padding: 0 5px;
  533. height: 40px;
  534. line-height: 40px;
  535. font-size: 20px;
  536. color: #606266;
  537. display: inline-block;
  538. // vertical-align: top;
  539. // display: inline-block;
  540. }
  541. .label {
  542. height: 40px;
  543. line-height: 40px;
  544. }
  545. .tuan {
  546. &.chu {
  547. width: 60px;
  548. height: 40px;
  549. display: inline-block;
  550. span {
  551. width: 60px;
  552. display: inline-block;
  553. line-height: 20px;
  554. text-align: center;
  555. font-size: 12px;
  556. height: 20px;
  557. &:last-child {
  558. border-top: 1px solid #606266;
  559. }
  560. }
  561. }
  562. &.cheng {
  563. .name {
  564. height: 40px;
  565. line-height: 40px;
  566. }
  567. .icon-span {
  568. line-height: 40px;
  569. font-size: 16px;
  570. padding: 0 1px;
  571. }
  572. }
  573. }
  574. }
  575. }
  576. </style>