index.vue 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. <template>
  2. <div v-if="!show">
  3. <div class="upload">
  4. <Icon type="md-reverse-camera" />
  5. <form id="psd">
  6. <input type="file" accept="image/*" @change="selectImg($event)" />
  7. </form>
  8. </div>
  9. <div class="content">
  10. <div class="list">
  11. <img
  12. v-for="(item, id) in newImg"
  13. :key="id + '6s'"
  14. :src="item.url"
  15. alt=""
  16. />
  17. <p v-for="(item, id) in table" :key="id">
  18. {{ item.name }}:{{ item.num }}
  19. </p>
  20. </div>
  21. <div class="btn">
  22. <Button type="primary" @click="save" size="large">开始</Button>
  23. </div>
  24. </div>
  25. </div>
  26. <div class="loading" v-else>
  27. <div class="spinner"></div>
  28. </div>
  29. </template>
  30. <script>
  31. import { Icon, Button, Notice, Message } from "view-design";
  32. import "view-design/dist/styles/iview.css";
  33. import axios from "@/utils/axios";
  34. import ImageCompressor from "js-image-compressor";
  35. export default {
  36. components: {
  37. Icon,
  38. Button,
  39. },
  40. data() {
  41. return {
  42. show: false,
  43. newImg: [],
  44. table: [],
  45. URL: process.env.SERVERURL,
  46. };
  47. },
  48. methods: {
  49. // 选择多张图片
  50. selectImg(event) {
  51. const file = event.target.files;
  52. const size = 2097152; //2M
  53. const newArr = [];
  54. if (file.length > 1) {
  55. Notice.warning({
  56. desc: "不能超过1张",
  57. });
  58. return;
  59. }
  60. for (let j = 0; j < file.length; j++) {
  61. const el = file[j];
  62. if (el.size >= size) {
  63. Notice.warning({
  64. desc: el.name + "大小超过2M限制请重新上传",
  65. });
  66. break;
  67. } else if (el.size >= size * 0.5) {
  68. Notice.warning({
  69. desc: el.name + "大小超过限制将被压缩",
  70. });
  71. // 压缩
  72. new ImageCompressor({
  73. file: el,
  74. quality: 0.3,
  75. mimeType: "image/jpeg",
  76. success: (result) => {
  77. newArr.push({
  78. url: this.getObjectURL(result),
  79. data: result,
  80. });
  81. },
  82. error: function (msg) {
  83. Notice.warning({
  84. desc: "请重新上传",
  85. });
  86. },
  87. });
  88. continue;
  89. } else {
  90. newArr.push({
  91. url: this.getObjectURL(el),
  92. data: el,
  93. });
  94. }
  95. }
  96. this.newImg = newArr;
  97. document.getElementById("psd").reset();
  98. this.table = [];
  99. },
  100. // 图片临时路径
  101. getObjectURL(file) {
  102. var url = null;
  103. if (window.createObjcectURL != undefined) {
  104. url = window.createOjcectURL(file);
  105. } else if (window.URL != undefined) {
  106. url = window.URL.createObjectURL(file);
  107. } else if (window.webkitURL != undefined) {
  108. url = window.webkitURL.createObjectURL(file);
  109. }
  110. return url;
  111. },
  112. // 识别
  113. async save() {
  114. if (!this.newImg.length) return;
  115. const start = Date.now();
  116. try {
  117. this.show = true;
  118. const formData = new FormData();
  119. formData.append("pic", this.newImg[0].data);
  120. const { data,success } = await axios.post("/uploadOneInvoicePic", formData);
  121. if(!success) throw new Error("上传错误");
  122. await this.reGetData(data.result_url,start);
  123. } catch (error) {
  124. this.show = false;
  125. Notice.warning({
  126. desc: error,
  127. });
  128. }
  129. },
  130. // 处理数据
  131. setData(data) {
  132. if (!data && !data.length) return [];
  133. let content = [];
  134. for (const key in data) {
  135. content.push({
  136. name: this.names(key) || "无",
  137. num: data[key].length ? data[key][0].word : "",
  138. });
  139. }
  140. return content;
  141. },
  142. // 名词
  143. names(key) {
  144. let name = "";
  145. switch (key) {
  146. case "AmountInFiguers":
  147. name = "合计";
  148. break;
  149. case "AmountInWords":
  150. name = "合计大写";
  151. break;
  152. case "CheckCode":
  153. name = "校验码";
  154. break;
  155. case "Checker":
  156. name = "复核";
  157. break;
  158. case "InvoiceCode":
  159. name = "发票代码";
  160. break;
  161. case "InvoiceCodeConfirm":
  162. name = "发票代码确认";
  163. break;
  164. case "InvoiceDate":
  165. name = "发票日期";
  166. break;
  167. case "InvoiceNum":
  168. name = "发票编号";
  169. break;
  170. case "InvoiceNumConfirm":
  171. name = "发票编号确认";
  172. break;
  173. case "NoteDrawer":
  174. name = "开票人";
  175. break;
  176. case "Payee":
  177. name = "收款人";
  178. break;
  179. case "PurchaserAddress":
  180. name = "买方地址";
  181. break;
  182. case "PurchaserBank":
  183. name = "买方银行信息";
  184. break;
  185. case "PurchaserName":
  186. name = "买方名称";
  187. break;
  188. case "PurchaserRegisterNum":
  189. name = "买方识别号";
  190. break;
  191. case "SellerAddress":
  192. name = "卖方地址";
  193. break;
  194. case "SellerBank":
  195. name = "卖方银行信息";
  196. break;
  197. case "SellerName":
  198. name = "卖方名称";
  199. break;
  200. case "SellerRegisterNum":
  201. name = "卖方识别号";
  202. break;
  203. default:
  204. break;
  205. }
  206. return name;
  207. },
  208. // 递归获取数据
  209. async reGetData(url,start) {
  210. const res = await axios.get(url);
  211. if (res.success) {
  212. this.table = this.setData(res.data.invoice_data.words_result[0].result);
  213. const end = Date.now();
  214. Message.info({
  215. content: "识别总用时:" + Math.abs((start - end) / 1000) + "s",
  216. duration: 5,
  217. });
  218. this.show = false;
  219. } else {
  220. this.reGetData(url,start);
  221. }
  222. },
  223. },
  224. };
  225. </script>
  226. <style lang="less" scoped>
  227. .upload {
  228. width: 0.4rem;
  229. height: 0.4rem;
  230. overflow: hidden;
  231. border: dashed 1px #dcdee2;
  232. color: #dcdee2;
  233. position: relative;
  234. margin: 0 auto;
  235. margin-top: 0.1rem;
  236. font-size: 0.2rem;
  237. text-align: center;
  238. line-height: 0.35rem;
  239. input {
  240. width: 0.4rem;
  241. height: 0.4rem;
  242. position: absolute;
  243. left: 0;
  244. top: 0;
  245. opacity: 0;
  246. }
  247. }
  248. .content {
  249. .list {
  250. padding: 0.1rem 0;
  251. max-height: 6rem;
  252. color: #dcdee2;
  253. text-align: center;
  254. font-size: 0.16rem;
  255. overflow-x: hidden;
  256. overflow-y: auto;
  257. img {
  258. max-width: 100%;
  259. max-height: 3rem;
  260. font-size: 0;
  261. }
  262. p {
  263. text-align: left;
  264. max-width: 4rem;
  265. margin: 0 auto;
  266. }
  267. }
  268. .btn {
  269. text-align: center;
  270. padding-top: 0.1rem;
  271. }
  272. }
  273. .loading {
  274. position: fixed;
  275. width: 100vw;
  276. height: 100vh;
  277. z-index: 96;
  278. left: 0;
  279. top: 0;
  280. background: rgba(0, 0, 0, 0.5);
  281. display: flex;
  282. justify-content: center;
  283. align-items: center;
  284. .spinner {
  285. width: 0.4rem;
  286. height: 0.4rem;
  287. background-color: #dcdee2;
  288. margin: 0 auto;
  289. -webkit-animation: sk-rotateplane 1.2s infinite ease-in-out;
  290. animation: sk-rotateplane 1.2s infinite ease-in-out;
  291. }
  292. @-webkit-keyframes sk-rotateplane {
  293. 0% {
  294. -webkit-transform: perspective(1.2rem);
  295. }
  296. 50% {
  297. -webkit-transform: perspective(1.2rem) rotateY(180deg);
  298. }
  299. 100% {
  300. -webkit-transform: perspective(1.2rem) rotateY(180deg) rotateX(180deg);
  301. }
  302. }
  303. @keyframes sk-rotateplane {
  304. 0% {
  305. transform: perspective(1.2rem) rotateX(0deg) rotateY(0deg);
  306. -webkit-transform: perspective(1.2rem) rotateX(0deg) rotateY(0deg);
  307. }
  308. 50% {
  309. transform: perspective(1.2rem) rotateX(-180.1deg) rotateY(0deg);
  310. -webkit-transform: perspective(1.2rem) rotateX(-180.1deg) rotateY(0deg);
  311. }
  312. 100% {
  313. transform: perspective(1.2rem) rotateX(-180deg) rotateY(-179.9deg);
  314. -webkit-transform: perspective(1.2rem) rotateX(-180deg) rotateY(-179.9deg);
  315. }
  316. }
  317. }
  318. </style>