upload.vue 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. <template>
  2. <div>
  3. <el-dialog
  4. v-model="dialogVisible"
  5. width="750px"
  6. title="Upload Statement"
  7. :show-close="false"
  8. :close-on-click-modal="false"
  9. :close-on-press-escape="false"
  10. :before-close="handleClose"
  11. >
  12. <div
  13. v-loading="loading"
  14. class="flex start"
  15. >
  16. <el-form
  17. ref="mainForm"
  18. :rules="formRule"
  19. class="flex-auto"
  20. :model="form"
  21. label-width="150px"
  22. >
  23. <el-form-item
  24. label="Statement Name"
  25. prop="statement_name"
  26. >
  27. <el-input
  28. v-model="form.statement_name"
  29. style="width: 190px"
  30. type="textarea"
  31. rows="3"
  32. />
  33. </el-form-item>
  34. <el-form-item label="Currency">
  35. <el-select v-model="form.currency">
  36. <el-option
  37. v-for="option in currencyList as IOptionItem[]"
  38. :key="option.value"
  39. :label="option.label"
  40. :value="option.value"
  41. ></el-option>
  42. </el-select>
  43. </el-form-item>
  44. <el-form-item label="Payment Type">
  45. <el-select v-model="form.paymentType">
  46. <el-option
  47. v-for="option in paymentOption"
  48. :key="option.value"
  49. :label="option.label"
  50. :value="option.value"
  51. ></el-option>
  52. </el-select>
  53. </el-form-item>
  54. <el-form-item label="Upload Mode">
  55. <el-select v-model="form.mode">
  56. <el-option
  57. v-for="option in uploadOption"
  58. :key="option.value"
  59. :label="option.label"
  60. :value="option.value"
  61. ></el-option>
  62. </el-select>
  63. </el-form-item>
  64. </el-form>
  65. <div
  66. class="flex-auto drag-area"
  67. @dragenter="stop"
  68. @dragover="stop"
  69. @dragleave="stop"
  70. @drop="processExcel"
  71. >
  72. <label for="fileInput">
  73. <div
  74. class="flex column stretch"
  75. style="text-align: center; padding: 44px 20px; cursor: pointer"
  76. >
  77. <div>
  78. <el-icon
  79. size="60px"
  80. color="#999"
  81. >
  82. <upload-filled />
  83. </el-icon>
  84. </div>
  85. <br />
  86. <div class="el-upload__text">拖动文件到这或者点击选择</div>
  87. <br />
  88. <div class="el-upload">
  89. 单个Excel数据最好控制在100行内, 处理起来会慢
  90. </div>
  91. <br>
  92. <div
  93. v-if="tableData.length"
  94. style="color: green"
  95. >
  96. 读取文件成功!
  97. </div>
  98. </div>
  99. </label>
  100. </div>
  101. </div>
  102. <br />
  103. <div class="flex end">
  104. <el-button
  105. :loading="loading"
  106. @click="handleClose"
  107. >
  108. 关闭
  109. </el-button>
  110. <el-button
  111. type="primary"
  112. :loading="loading"
  113. @click="next(mainForm)"
  114. >
  115. 确认
  116. </el-button>
  117. </div>
  118. </el-dialog>
  119. <!-- multiple -->
  120. <input
  121. id="fileInput"
  122. type="file"
  123. style="display: none"
  124. accept="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet,application/vnd.ms-excel"
  125. @change="processExcel"
  126. />
  127. </div>
  128. </template>
  129. <script lang="ts">
  130. import { defineComponent } from 'vue'
  131. export default defineComponent({
  132. name: 'DialogUploadExcel',
  133. })
  134. </script>
  135. <script lang="ts" setup>
  136. import { watchEffect, ref } from 'vue'
  137. import {
  138. ElDialog,
  139. ElMessage,
  140. ElForm,
  141. ElFormItem,
  142. ElSelect,
  143. ElOption,
  144. ElInput,
  145. ElButton,
  146. ElIcon,
  147. } from 'element-plus'
  148. import { UploadFilled } from '@element-plus/icons-vue'
  149. import * as XLSX from 'xlsx'
  150. import type { FormInstance, FormRules } from 'element-plus'
  151. import { IPoItem, IOptionItem } from '../inteface'
  152. import request from '@/utils/axios'
  153. const props = defineProps({
  154. visible: {
  155. type: Boolean,
  156. default: false,
  157. },
  158. currencyList: {
  159. type: Array,
  160. default: () => {
  161. return []
  162. },
  163. },
  164. })
  165. const emit = defineEmits(['update:visible', 'update-table-data'])
  166. const dialogVisible = ref(false)
  167. watchEffect(() => {
  168. dialogVisible.value = props.visible
  169. })
  170. const tableData = ref([] as IPoItem[])
  171. const handleClose = function (done: any) {
  172. emit('update:visible', false)
  173. tableData.value = []
  174. form.value = {
  175. statement_name: '',
  176. currency: 'CNY',
  177. mode: 'Replace',
  178. paymentType: '货款',
  179. }
  180. const target = document.getElementById('fileInput') as HTMLInputElement
  181. if (target) {
  182. target.value = ''
  183. }
  184. if (typeof done === 'function') {
  185. done()
  186. }
  187. }
  188. const mainForm = ref<FormInstance>()
  189. const formRule = ref<FormRules>({
  190. statement_name: {
  191. required: true,
  192. message: '必填项',
  193. trigger: 'blur',
  194. },
  195. })
  196. const form = ref({
  197. statement_name: '',
  198. currency: 'CNY',
  199. mode: 'Replace',
  200. paymentType: '货款',
  201. })
  202. const uploadOption = [
  203. {
  204. label: '追加(Append)',
  205. value: 'Append',
  206. },
  207. {
  208. label: '替换(Replace)',
  209. value: 'Replace',
  210. },
  211. ]
  212. const paymentOption = [
  213. {
  214. label: '货款',
  215. value: '货款',
  216. },
  217. {
  218. label: '快递款',
  219. value: '快递款',
  220. },
  221. ]
  222. const stop = (e: any) => {
  223. e.preventDefault()
  224. e.stopPropagation()
  225. }
  226. const processExcel = (event: any) => {
  227. const files = event.target.files || event.dataTransfer.files
  228. // console.log('files:', files)
  229. let str = ''
  230. let arr: IPoItem[] = []
  231. tableData.value = []
  232. try {
  233. for (let i = 0; i < files.length; i++) {
  234. if (
  235. ![
  236. 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
  237. 'application/vnd.ms-excel',
  238. ].includes(files[i].type)
  239. ) {
  240. ElMessage.error('读取数据出错, 请确认选择了正确的Excel文件')
  241. return
  242. }
  243. str =
  244. str +
  245. `${str.length ? ', ' : ''}` +
  246. (files[i].name.replace(/\.xlsx?/, '') || 'unNameFile')
  247. const fileReader = new FileReader()
  248. fileReader.onload = (e: any) => {
  249. const data = XLSX.read(e.target.result, { type: 'binary' })
  250. // 重命名列名
  251. data.Sheets[data.SheetNames[0]].A1.w = 'po_number'
  252. data.Sheets[data.SheetNames[0]].B1.w = 'sku'
  253. data.Sheets[data.SheetNames[0]].C1.w = 'description'
  254. data.Sheets[data.SheetNames[0]].D1.w = 'unit_price'
  255. data.Sheets[data.SheetNames[0]].E1.w = 'quantity'
  256. data.Sheets[data.SheetNames[0]].F1.w = 'sample_fee'
  257. data.Sheets[data.SheetNames[0]].G1.w = 'setup_service_fee'
  258. data.Sheets[data.SheetNames[0]].H1.w = 'total'
  259. const jsonData = XLSX.utils.sheet_to_json(
  260. data.Sheets[data.SheetNames[0]],
  261. ) as IPoItem[]
  262. jsonData.forEach((i) => {
  263. tableData.value.push(i)
  264. })
  265. // tableData.value = tableData.value.concat(jsonData)
  266. }
  267. fileReader.readAsBinaryString(files[i])
  268. }
  269. form.value.statement_name = str
  270. } catch (error) {
  271. console.log('处理文件出错:', error)
  272. }
  273. event.preventDefault()
  274. event.stopPropagation()
  275. }
  276. const loading = ref(false)
  277. const next = (formEl: FormInstance | undefined) => {
  278. if (!formEl) return
  279. formEl.validate((valid, fields) => {
  280. if (valid) {
  281. loading.value = true
  282. request
  283. .post('/payment_request/createStatementData', [
  284. {
  285. Name: form.value.statement_name,
  286. },
  287. ])
  288. .then((response) => {
  289. if (response.data.code !== 1) return
  290. const res = response.data.result
  291. let result = {
  292. mode: form.value.mode,
  293. data: tableData.value.map((i) => {
  294. return {
  295. ...i,
  296. po_number: i.po_number.toUpperCase(),
  297. payment_type: form.value.paymentType,
  298. statement_name: form.value.statement_name,
  299. currency: form.value.currency,
  300. statement_id: res.data[0].details.id,
  301. }
  302. }),
  303. }
  304. emit('update-table-data', result)
  305. handleClose(false)
  306. })
  307. .finally(() => {
  308. loading.value = false
  309. })
  310. } else {
  311. console.log('check form has not pass!', fields)
  312. ElMessage.error('请检查表单必填项')
  313. }
  314. })
  315. }
  316. </script>
  317. <style lang="scss" scoped>
  318. .drag-area {
  319. height: 240px;
  320. border: 1px solid #ddd;
  321. border-radius: 4px;
  322. // padding: 24px;
  323. }
  324. </style>