index.vue 45 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374
  1. <template>
  2. <div class="app-container">
  3. <el-form :model="queryParams" ref="queryForm" :inline="true" v-show="showSearch" label-width="68px">
  4. <el-form-item label="订单编号" prop="orderCodes">
  5. <div class="tag-input-container">
  6. <!-- 标签显示区域 -->
  7. <div class="tags-wrapper" @click="focusInput">
  8. <!-- 已添加的订单号标签 -->
  9. <el-tag
  10. v-for="(code, index) in queryParams.orderCodes"
  11. :key="index"
  12. closable
  13. size="small"
  14. @close="removeOrderCode(index)"
  15. class="order-tag"
  16. :class="{ 'tag-error': false }"
  17. >
  18. {{ code }}
  19. </el-tag>
  20. <!-- 输入框 -->
  21. <el-input
  22. ref="tagInput"
  23. v-model="currentInput"
  24. v-show="inputVisible || queryParams.orderCodes.length === 0"
  25. :placeholder="queryParams.orderCodes.length === 0 ? '请输入订单号,按回车或逗号分隔' : '继续输入...'"
  26. size="small"
  27. class="tag-input"
  28. @keydown.native="handleKeyDown"
  29. @keyup.native="handleKeyUp"
  30. @blur="handleInputConfirm"
  31. @focus="inputVisible = true"
  32. clearable
  33. />
  34. <!-- 添加按钮(当没有输入时显示) -->
  35. <el-button
  36. v-if="!inputVisible && queryParams.orderCodes.length > 0"
  37. class="button-new-tag"
  38. size="small"
  39. @click="showInput"
  40. icon="el-icon-plus"
  41. type="text"
  42. >
  43. 添加订单号
  44. </el-button>
  45. </div>
  46. <!-- 输入提示 -->
  47. <div class="input-tips">
  48. <span class="tip-text">
  49. 支持:回车、逗号、空格分隔 |
  50. 已添加 {{ queryParams.orderCodes.length }} 个订单号
  51. <span v-if="maxOrderCodes > 0"> (最多{{ maxOrderCodes }}个)</span>
  52. </span>
  53. </div>
  54. </div>
  55. </el-form-item>
  56. <el-form-item label="用户名称" prop="userName">
  57. <el-input
  58. v-model="queryParams.userName"
  59. placeholder="请输入用户名称"
  60. clearable
  61. size="small"
  62. @keyup.enter.native="handleQuery"
  63. />
  64. </el-form-item>
  65. <el-form-item label="用户电话" prop="userPhone">
  66. <el-input
  67. v-model="queryParams.userPhone"
  68. placeholder="请输入用户电话"
  69. clearable
  70. size="small"
  71. @keyup.enter.native="handleQuery"
  72. />
  73. </el-form-item>
  74. <el-form-item label="状态" prop="status">
  75. <el-select v-model="queryParams.status" placeholder="状态" clearable size="small">
  76. <el-option
  77. v-for="dict in statusOptions"
  78. :key="dict.dictValue"
  79. :label="dict.dictLabel"
  80. :value="dict.dictValue"
  81. />
  82. </el-select>
  83. </el-form-item>
  84. <el-form-item label="快递单号" prop="deliverySn">
  85. <el-input
  86. v-model="queryParams.deliverySn"
  87. placeholder="请输入快递单号"
  88. clearable
  89. size="small"
  90. @keyup.enter.native="handleQuery"
  91. />
  92. </el-form-item>
  93. <el-form-item label="提交时间" prop="createTime">
  94. <el-date-picker v-model="createTime" size="small" style="width: 220px" value-format="yyyy-MM-dd" type="daterange" range-separator="-" start-placeholder="开始日期" end-placeholder="结束日期" @change="change"></el-date-picker>
  95. </el-form-item>
  96. <el-form-item label="销售公司" prop="companyId">
  97. <el-select v-model="queryParams.companyId" placeholder="销售公司" size="small" @change="getAllUserlist(queryParams.companyId)" clearable>
  98. <el-option
  99. v-for="dict in qwCompanyList"
  100. :key="dict.companyId"
  101. :label="dict.companyName"
  102. :value="dict.companyId"
  103. />
  104. </el-select>
  105. </el-form-item>
  106. <el-form-item label="销售账号" prop="userId">
  107. <el-select v-model="queryParams.companyUserId" placeholder="销售账号" size="small" clearable>
  108. <el-option
  109. v-for="dict in companyUserNameList"
  110. :key="dict.userId"
  111. :label="dict.userName"
  112. :value="dict.userId"
  113. />
  114. </el-select>
  115. </el-form-item>
  116. <el-form-item label="企微微信" prop="companyUserName">
  117. <el-select v-model="queryParams.qwUserId" placeholder="企微微信" size="small" clearable>
  118. <el-option
  119. v-for="dict in qwUserList"
  120. :key="dict.id"
  121. :label="dict.qwUserName"
  122. :value="dict.id"
  123. />
  124. </el-select>
  125. </el-form-item>
  126. <!-- 这里就是之前添加的ERP账号下拉框 -->
  127. <el-form-item label="ERP" prop="loginAccount">
  128. <el-select v-model="queryParams.loginAccount" placeholder="ERP账号" size="small" clearable>
  129. <el-option
  130. v-for="account in erpAccountList"
  131. :key="account"
  132. :label="account"
  133. :value="account"
  134. />
  135. </el-select>
  136. </el-form-item>
  137. <el-form-item>
  138. <el-button type="primary" icon="el-icon-search" size="mini" @click="handleQuery">搜索</el-button>
  139. <el-button icon="el-icon-refresh" size="mini" @click="resetQuery">重置</el-button>
  140. </el-form-item>
  141. </el-form>
  142. <el-row :gutter="10" class="mb8">
  143. <el-col :span="1.5">
  144. <el-button
  145. type="warning"
  146. plain
  147. icon="el-icon-download"
  148. size="mini"
  149. :loading="exportLoading"
  150. @click="handleExport"
  151. v-hasPermi="['his:integralOrder:export']"
  152. >导出</el-button>
  153. </el-col>
  154. <el-col :span="1.5">
  155. <el-button
  156. type="info"
  157. plain
  158. icon="el-icon-upload2"
  159. size="mini"
  160. @click="handleImportStatus"
  161. >导入订单状态</el-button>
  162. </el-col>
  163. <el-col :span="1.5">
  164. <el-tooltip content="默认erp推送手机号" placement="top">
  165. <el-button
  166. type="warning"
  167. plain
  168. icon="el-icon-phone"
  169. size="mini"
  170. @click="handleErpPhone"
  171. >推送手机号码</el-button>
  172. </el-tooltip>
  173. </el-col>
  174. <el-col :span="1.5">
  175. <el-button
  176. type="info"
  177. plain
  178. icon="el-icon-upload2"
  179. size="mini"
  180. @click="handleImport"
  181. v-hasPermi="['his:integralOrder:exportDeliver']"
  182. v-show="actName === '6'||actName === '1'"
  183. >导入发货</el-button>
  184. </el-col>
  185. <el-col :span="1.5">
  186. <el-button
  187. type="success"
  188. plain
  189. icon="el-icon-s-operation"
  190. size="mini"
  191. @click="handleDataSort"
  192. v-show="actName === '6'"
  193. >数据分拣</el-button>
  194. </el-col>
  195. <el-col :span="1.5">
  196. <el-button
  197. type="primary"
  198. plain
  199. icon="el-icon-plus"
  200. size="mini"
  201. @click="handleCreateErp"
  202. v-show="actName === '6'"
  203. >创建ERP</el-button>
  204. </el-col>
  205. <right-toolbar :showSearch.sync="showSearch" @queryTable="getList"></right-toolbar>
  206. </el-row>
  207. <el-tabs type="card" v-model="actName" @tab-click="handleClickX">
  208. <el-tab-pane label="全部订单" name="10"></el-tab-pane>
  209. <el-tab-pane v-for="(item,index) in statusOptions" :label="item.dictLabel" :name="item.dictValue"></el-tab-pane>
  210. </el-tabs>
  211. <el-table v-loading="loading" border :data="integralOrderList" @selection-change="handleSelectionChange">
  212. <el-table-column type="selection" width="55" align="center" />
  213. <el-table-column label="ERP账号" align="center" prop="loginAccount" />
  214. <el-table-column label="ERP电话" align="center" prop="erpPhone" />
  215. <el-table-column label="订单编号" align="center" prop="orderCode" />
  216. <el-table-column label="商品名称" align="center" width="200">
  217. <template slot-scope="scope">
  218. <div style="display: flex; align-items: center; justify-content: center;">
  219. <span>{{ scope.row.goodsName }}</span>
  220. <el-image
  221. v-if="scope.row.goodsImage"
  222. :src="scope.row.goodsImage"
  223. style="width: 40px; height: 40px; margin-left: 8px;"
  224. fit="cover"
  225. :preview-src-list="[scope.row.goodsImage]"
  226. />
  227. </div>
  228. </template>
  229. </el-table-column>
  230. <el-table-column label="用户名称" align="center" prop="userName" />
  231. <el-table-column label="用户电话" align="center" prop="userPhone" />
  232. <el-table-column label="用户地址" align="center" prop="userAddress" show-overflow-tooltip />
  233. <el-table-column label="支付积分" align="center" prop="integral" />
  234. <el-table-column label="支付金额" align="center" prop="payMoney" />
  235. <el-table-column label="状态" align="center" prop="status">
  236. <template slot-scope="scope">
  237. <dict-tag :options="statusOptions" :value="scope.row.status"/>
  238. </template>
  239. </el-table-column>
  240. <el-table-column label="快递公司编号" align="center" prop="deliveryCode" />
  241. <el-table-column label="快递名称" align="center" prop="deliveryName" />
  242. <el-table-column label="快递单号" align="center" prop="deliverySn" />
  243. <el-table-column label="发货时间" align="center" prop="deliveryTime" width="180"/>
  244. <el-table-column label="提交时间" align="center" prop="createTime" width="180"/>
  245. <el-table-column label="销售公司ID" align="center" prop="companyId" width="180"/>
  246. <el-table-column label="销售ID" align="center" prop="companyUserId" width="180"/>
  247. <el-table-column label="企业微信ID" align="center" prop="qwUserId" width="180"/>
  248. <el-table-column label="备注" align="center" prop="remark" show-overflow-tooltip/>
  249. <el-table-column label="操作" align="center" class-name="small-padding fixed-width" fixed="right">
  250. <template slot-scope="scope">
  251. <el-button
  252. size="mini"
  253. type="text"
  254. @click="handledetails(scope.row)"
  255. >详情
  256. </el-button>
  257. <el-button
  258. size="mini"
  259. type="text"
  260. @click="cancelOrder(scope.row.orderCode)"
  261. v-if="scope.row.status === '1'"
  262. >取消订单
  263. </el-button>
  264. </template>
  265. </el-table-column>
  266. </el-table>
  267. <pagination
  268. v-show="total>0"
  269. :total="total"
  270. :page.sync="queryParams.pageNum"
  271. :limit.sync="queryParams.pageSize"
  272. @pagination="getList"
  273. />
  274. <!-- 添加或修改积分商品订单对话框 -->
  275. <el-dialog :title="title" :visible.sync="open" width="500px" append-to-body>
  276. <el-form ref="form" :model="form" :rules="rules" label-width="80px">
  277. <el-form-item label="订单编号" prop="orderCode">
  278. <el-input v-model="form.orderCode" placeholder="请输入订单编号" />
  279. </el-form-item>
  280. <el-form-item label="用户id" prop="userId">
  281. <el-input v-model="form.userId" placeholder="请输入用户id" />
  282. </el-form-item>
  283. <el-form-item label="用户名称" prop="userName">
  284. <el-input v-model="form.userName" placeholder="请输入用户名称" />
  285. </el-form-item>
  286. <el-form-item label="用户电话" prop="userPhone">
  287. <el-input v-model="form.userPhone" placeholder="请输入用户电话" />
  288. </el-form-item>
  289. <el-form-item label="用户地址" prop="userAddress">
  290. <el-input v-model="form.userAddress" placeholder="请输入用户地址" />
  291. </el-form-item>
  292. <el-form-item label="商品信息" prop="itemJson">
  293. <el-input v-model="form.itemJson" type="textarea" placeholder="请输入内容" />
  294. </el-form-item>
  295. <el-form-item label="支付积分" prop="integral">
  296. <el-input v-model="form.integral" placeholder="请输入支付积分" />
  297. </el-form-item>
  298. <el-form-item label="1:待发货;2:待收货;3:已完成" prop="status">
  299. <el-select v-model="form.status" placeholder="请选择1:待发货;2:待收货;3:已完成">
  300. <el-option
  301. v-for="dict in statusOptions"
  302. :key="dict.dictValue"
  303. :label="dict.dictLabel"
  304. :value="dict.dictValue"
  305. ></el-option>
  306. </el-select>
  307. </el-form-item>
  308. <el-form-item label="快递公司编号" prop="deliveryCode">
  309. <el-input v-model="form.deliveryCode" placeholder="请输入快递公司编号" />
  310. </el-form-item>
  311. <el-form-item label="快递名称" prop="deliveryName">
  312. <el-input v-model="form.deliveryName" placeholder="请输入快递名称" />
  313. </el-form-item>
  314. <el-form-item label="快递单号" prop="deliverySn">
  315. <el-input v-model="form.deliverySn" placeholder="请输入快递单号" />
  316. </el-form-item>
  317. <el-form-item label="发货时间" prop="deliveryTime">
  318. <el-date-picker clearable size="small"
  319. v-model="form.deliveryTime"
  320. type="date"
  321. value-format="yyyy-MM-dd"
  322. placeholder="选择发货时间">
  323. </el-date-picker>
  324. </el-form-item>
  325. <el-form-item label="备注" prop="remark">
  326. <el-input v-model="form.remark" placeholder="请输入备注" />
  327. </el-form-item>
  328. </el-form>
  329. <div slot="footer" class="dialog-footer">
  330. <el-button type="primary" @click="submitForm">确 定</el-button>
  331. <el-button @click="cancel">取 消</el-button>
  332. </div>
  333. </el-dialog>
  334. <el-drawer
  335. :with-header="false"
  336. size="75%"
  337. :title="show.title" :visible.sync="show.open">
  338. <integralOrderDetails ref="Details" />
  339. </el-drawer>
  340. <el-dialog :title="upload.title" :visible.sync="upload.open" width="400px" append-to-body>
  341. <el-upload
  342. ref="upload"
  343. :limit="1"
  344. accept=".xlsx, .xls"
  345. :headers="upload.headers"
  346. :action="upload.url + '?updateSupport=' + upload.updateSupport"
  347. :disabled="upload.isUploading"
  348. :on-progress="handleFileUploadProgress"
  349. :on-success="handleFileSuccess"
  350. :auto-upload="false"
  351. drag
  352. >
  353. <i class="el-icon-upload"></i>
  354. <div class="el-upload__text">将文件拖到此处,或<em>点击上传</em></div>
  355. <div class="el-upload__tip text-center" slot="tip">
  356. <div class="el-upload__tip" slot="tip">
  357. </div>
  358. <span>仅允许导入xls、xlsx格式文件。</span>
  359. <el-link type="primary" :underline="false" style="font-size:12px;vertical-align: baseline;" @click="importTemplate">下载模板</el-link>
  360. </div>
  361. </el-upload>
  362. <div slot="footer" class="dialog-footer">
  363. <el-button type="primary" @click="submitFileForm">确 定</el-button>
  364. <el-button @click="upload.open = false">取 消</el-button>
  365. </div>
  366. </el-dialog>
  367. <!-- 数据分拣弹窗 -->
  368. <el-dialog :title="erpAccountDialog.title" :visible.sync="erpAccountDialog.open" width="600px" append-to-body>
  369. <div v-loading="erpAccountDialog.loading">
  370. <el-form :model="erpAccountForm" label-width="100px">
  371. <el-form-item label="ERP账户" required>
  372. <el-select
  373. v-model="erpAccountForm.selectedAccount"
  374. placeholder="请选择ERP账户"
  375. style="width: 100%"
  376. filterable
  377. >
  378. <el-option
  379. v-for="account in erpAccountList"
  380. :key="account"
  381. :label="account"
  382. :value="account"
  383. />
  384. </el-select>
  385. </el-form-item>
  386. <el-form-item label="推送手机号">
  387. <el-select
  388. v-model="erpAccountForm.selectedPhones"
  389. multiple
  390. placeholder="请选择推送手机号,不填默认订单用户电话"
  391. style="width: 100%"
  392. filterable
  393. >
  394. <el-option
  395. v-for="phone in phoneList"
  396. :key="phone.phone"
  397. :label="phone.phone"
  398. :value="phone.phone"
  399. />
  400. </el-select>
  401. </el-form-item>
  402. </el-form>
  403. <!-- 订单统计信息 -->
  404. <div class="order-summary" v-if="orderSummary">
  405. <el-divider content-position="left">订单统计</el-divider>
  406. <el-row :gutter="20">
  407. <el-col :span="8">
  408. <div class="summary-item">
  409. <span class="label">选中订单数:</span>
  410. <span class="value">{{ orderSummary.selectedCount }}</span>
  411. </div>
  412. </el-col>
  413. <el-col :span="8">
  414. <div class="summary-item">
  415. <span class="label">总金额:</span>
  416. <span class="value">¥{{ orderSummary.totalAmount }}</span>
  417. </div>
  418. </el-col>
  419. <el-col :span="8">
  420. <div class="summary-item">
  421. <span class="label">查询条件订单:</span>
  422. <span class="value">{{ orderSummary.queryCount }}</span>
  423. </div>
  424. </el-col>
  425. </el-row>
  426. </div>
  427. </div>
  428. <div slot="footer" class="dialog-footer">
  429. <el-button @click="cancelErpAccountDialog">取 消</el-button>
  430. <el-button
  431. type="primary"
  432. @click="confirmCreateErpOrder"
  433. :disabled="!erpAccountForm.selectedAccount"
  434. :loading="erpAccountDialog.submitting"
  435. >确认</el-button>
  436. </div>
  437. </el-dialog>
  438. <!-- 推送手机号码管理弹窗 -->
  439. <el-dialog :title="erpPhone.title" :visible.sync="erpPhone.open" width="600px" append-to-body>
  440. <div style="margin-bottom: 20px;">
  441. <el-button type="primary" size="small" @click="handleAddPhone">新增手机号</el-button>
  442. </div>
  443. <el-table :data="phoneList" border style="width: 100%">
  444. <el-table-column prop="phone" label="手机号" align="center">
  445. <template slot-scope="scope">
  446. <el-input
  447. v-if="scope.row.editing"
  448. v-model="scope.row.phone"
  449. placeholder="请输入手机号"
  450. @blur="validatePhone(scope.row)"
  451. @keyup.enter.native="handleSavePhone(scope.$index)"
  452. />
  453. <span v-else>{{ scope.row.phone }}</span>
  454. </template>
  455. </el-table-column>
  456. <el-table-column label="操作" align="center" width="300">
  457. <template slot-scope="scope">
  458. <el-button
  459. v-if="scope.row.editing"
  460. type="success"
  461. size="mini"
  462. @click="handleSavePhone(scope.$index)"
  463. >保存</el-button>
  464. <el-button
  465. v-if="scope.row.editing"
  466. type="info"
  467. size="mini"
  468. @click="handleCancelEdit(scope.$index)"
  469. >取消</el-button>
  470. <el-button
  471. v-if="!scope.row.editing"
  472. type="primary"
  473. size="mini"
  474. @click="handleEditPhone(scope.$index)"
  475. >修改</el-button>
  476. <el-button
  477. type="danger"
  478. size="mini"
  479. @click="handleDeletePhone(scope.$index)"
  480. >删除</el-button>
  481. </template>
  482. </el-table-column>
  483. </el-table>
  484. <div slot="footer" class="dialog-footer">
  485. <el-button type="primary" @click="handleSavePhoneList">确 定</el-button>
  486. <el-button @click="handleCancelPhoneDialog">取 消</el-button>
  487. </div>
  488. </el-dialog>
  489. <el-dialog :title="uploadStatus.title" :visible.sync="uploadStatus.open" width="400px" append-to-body>
  490. <el-upload
  491. ref="uploadStatus"
  492. :limit="1"
  493. accept=".xlsx, .xls"
  494. :headers="uploadStatus.headers"
  495. :action="uploadStatus.url + '?updateSupport=' + upload.updateSupport"
  496. :disabled="uploadStatus.isUploading"
  497. :on-progress="handleFileUploadProgressOrder"
  498. :on-success="handleFileSuccessOrder"
  499. :auto-upload="false"
  500. drag
  501. >
  502. <i class="el-icon-upload"></i>
  503. <div class="el-upload__text">将文件拖到此处,或<em>点击上传</em></div>
  504. <div class="el-upload__tip text-center" slot="tip">
  505. <div class="el-upload__tip" slot="tip">
  506. </div>
  507. <span>仅允许导入xls、xlsx格式文件。</span>
  508. <el-link type="primary" :underline="false" style="font-size:12px;vertical-align: baseline;" @click="importUpdateOrderTemplate">下载模板</el-link>
  509. </div>
  510. </el-upload>
  511. <div slot="footer" class="dialog-footer">
  512. <el-button type="primary" @click="submitOrderStatusFileForm">确 定</el-button>
  513. <el-button @click="uploadStatus.open = false">取 消</el-button>
  514. </div>
  515. </el-dialog>
  516. </div>
  517. </template>
  518. <script>
  519. import {importTemplate,batchSetErpOrder,batchCreateErpOrder,listIntegralOrder,getIntegralTemplate, getIntegralOrder, delIntegralOrder, addIntegralOrder, updateIntegralOrder, exportIntegralOrder,cancelOrder } from "@/api/his/integralOrder";
  520. import integralOrderDetails from '../../components/his/integralOrderDetails.vue';
  521. import { getToken } from "@/utils/auth";
  522. import {getCompanyList} from "@/api/company/company";
  523. import {getAllUserlist} from "@/api/company/companyUser";
  524. import {getQwUserInfo} from "@/api/qw/qwUser";
  525. import {getErpAccount } from "@/api/his/storeOrder";
  526. import {queryErpPhone, saveErpPhone} from "@/api/his/storeOrder";
  527. export default {
  528. name: "IntegralOrder",
  529. components: { integralOrderDetails },
  530. data() {
  531. return {
  532. show:{
  533. open:false,
  534. },
  535. upload: {
  536. // 是否显示弹出层
  537. open: false,
  538. // 弹出层标题
  539. title: "",
  540. // 是否禁用上传
  541. isUploading: false,
  542. // 是否更新已经存在的用户数据
  543. updateSupport: 0,
  544. // 设置上传的请求头部
  545. headers: { Authorization: "Bearer " + getToken() },
  546. // 上传的地址
  547. url: process.env.VUE_APP_BASE_API + "/his/integralOrder/importData"
  548. },
  549. // 遮罩层
  550. loading: true,
  551. actName:"10",
  552. // 导出遮罩层
  553. exportLoading: false,
  554. // 选中数组
  555. ids: [],
  556. // 非单个禁用
  557. single: true,
  558. // 非多个禁用
  559. multiple: true,
  560. // 显示搜索条件
  561. showSearch: true,
  562. // 总条数
  563. total: 0,
  564. // 积分商品订单表格数据
  565. integralOrderList: [],
  566. // 弹出层标题
  567. title: "",
  568. // 是否显示弹出层
  569. open: false,
  570. // 1:待发货;2:待收货;3:已完成字典
  571. statusOptions: [],
  572. // 查询参数
  573. queryParams: {
  574. pageNum: 1,
  575. pageSize: 10,
  576. orderCode: null,
  577. orderCodes: [], // 添加订单号数组
  578. userName: null,
  579. userPhone: null,
  580. integral: null,
  581. status: null,
  582. deliverySn: null,
  583. createTime: null,
  584. sTime:null,
  585. eTime:null,
  586. companyUserId:null,
  587. qwUserId:null,
  588. companyId:null,
  589. loginAccount: null // 添加ERP账号筛选字段
  590. },
  591. // 最大订单号数量限制
  592. maxOrderCodes: {
  593. type: Number,
  594. default: 50
  595. },
  596. // 输入框是否可见
  597. inputVisible: false,
  598. // 当前输入值
  599. currentInput: '',
  600. createTime:null,
  601. qwCompanyList:[],
  602. companyUserNameList:[],
  603. qwUserList:[],
  604. // 表单参数
  605. form: {},
  606. // 表单校验
  607. rules: {
  608. userId: [
  609. { required: true, message: "用户id不能为空", trigger: "blur" }
  610. ],
  611. status: [
  612. { required: true, message: "1:待发货;2:待收货;3:已完成不能为空", trigger: "change" }
  613. ],
  614. },
  615. dataSortOpen: false, // 数据分拣弹窗显示控制
  616. erpAccountDialog: {
  617. open: false,
  618. loading: false,
  619. submitting: false,
  620. title: "数据分拣"
  621. },
  622. erpAccountForm: {
  623. selectedAccount: null,
  624. selectedPhones: [] // 添加推送手机号字段
  625. },
  626. erpAccountList: [],
  627. orderSummary: {
  628. selectedCount: 0,
  629. totalAmount: 0,
  630. queryCount: 0
  631. },
  632. erpPhone: {
  633. open: false,
  634. title: "设置推送手机号"
  635. },
  636. phoneList: [], // 手机号列表
  637. originalPhoneList: [], // 原始手机号列表,用于取消时恢复
  638. uploadStatus: {
  639. // 是否显示弹出层
  640. open: false,
  641. // 弹出层标题
  642. title: "",
  643. // 是否禁用上传
  644. isUploading: false,
  645. // 是否更新已经存在的用户数据
  646. updateSupport: 0,
  647. // 设置上传的请求头部
  648. headers: { Authorization: "Bearer " + getToken() },
  649. // 上传的地址
  650. url: process.env.VUE_APP_BASE_API + "/his/integralOrder/importOrderStatusData"
  651. },
  652. };
  653. },
  654. created() {
  655. this.getList();
  656. this.getDicts("sys_integral_order_status").then(response => {
  657. this.statusOptions = response.data;
  658. });
  659. //获取企业
  660. getCompanyList().then(response => {
  661. this.qwCompanyList = response.data;
  662. });
  663. this.loadErpAccountData();
  664. },
  665. methods: {
  666. handleImportStatus() {
  667. this.uploadStatus.title = "导入";
  668. this.uploadStatus.open = true;
  669. },
  670. handleFileUploadProgressOrder(event, file, fileList) {
  671. this.uploadStatus.isUploading = true;
  672. },
  673. // 文件上传成功处理
  674. handleFileSuccessOrder(response, file, fileList) {
  675. this.uploadStatus.open = false;
  676. this.uploadStatus.isUploading = false;
  677. this.$refs.uploadStatus.clearFiles();
  678. this.$alert(response.msg, "导入结果", { dangerouslyUseHTMLString: true });
  679. this.getList();
  680. },
  681. importUpdateOrderTemplate(){
  682. getIntegralTemplate().then(response => {
  683. this.download(response.msg);
  684. });
  685. },
  686. //取消订单
  687. cancelOrder(orderCode){
  688. this.$confirm('确定取消此订单?', '提示', {
  689. confirmButtonText: '确定',
  690. cancelButtonText: '取消',
  691. type: 'warning'
  692. }).then(() => {
  693. console.log("orderCode",orderCode)
  694. cancelOrder(orderCode).then(()=>{
  695. this.$message({
  696. type: 'success',
  697. message: '取消成功!'
  698. });
  699. this.getList();
  700. })
  701. }).catch(() => {
  702. });
  703. },
  704. change(){
  705. if(this.createTime!=null){
  706. this.queryParams.sTime=this.createTime[0];
  707. this.queryParams.eTime=this.createTime[1];
  708. }else{
  709. this.queryParams.sTime=null;
  710. this.queryParams.eTime=null;
  711. }
  712. },
  713. handleImport() {
  714. this.upload.title = "导入";
  715. this.upload.open = true;
  716. },
  717. importTemplate() {
  718. importTemplate().then(response => {
  719. this.download(response.msg);
  720. });
  721. },
  722. // 文件上传中处理
  723. handleFileUploadProgress(event, file, fileList) {
  724. this.upload.isUploading = true;
  725. },
  726. // 文件上传成功处理
  727. handleFileSuccess(response, file, fileList) {
  728. this.upload.open = false;
  729. this.upload.isUploading = false;
  730. this.$refs.upload.clearFiles();
  731. this.$alert(response.msg, "导入结果", { dangerouslyUseHTMLString: true });
  732. this.getList();
  733. },
  734. // 提交上传文件
  735. submitFileForm() {
  736. this.$refs.upload.submit();
  737. },
  738. handledetails(row){
  739. this.show.open=true;
  740. setTimeout(() => {
  741. this.$refs.Details.getDetails(row.orderId);
  742. }, 1);
  743. },
  744. /** 查询积分商品订单列表 */
  745. getList() {
  746. this.loading = true;
  747. // 直接传递订单编号数组给后端
  748. listIntegralOrder(this.queryParams).then(response => {
  749. // 解析itemJson字段,提取goodsName和图片
  750. const processedData = response.rows.map(item => {
  751. let goodsName = '';
  752. let goodsImage = '';
  753. try {
  754. if (item.itemJson) {
  755. const itemData = JSON.parse(item.itemJson);
  756. // 如果itemJson是数组格式,取第一个商品的名称和图片
  757. if (Array.isArray(itemData) && itemData.length > 0) {
  758. goodsName = itemData[0].goodsName || '';
  759. // 提取图片,优先使用imgUrl,如果没有则使用images的第一张
  760. if (itemData[0].imgUrl) {
  761. goodsImage = itemData[0].imgUrl;
  762. } else if (itemData[0].images && itemData[0].images.split(',').length > 0) {
  763. goodsImage = itemData[0].images.split(',')[0];
  764. }
  765. }
  766. // 如果itemJson是对象格式,直接取goodsName和图片
  767. else if (itemData.goodsName) {
  768. goodsName = itemData.goodsName;
  769. // 提取图片,优先使用imgUrl,如果没有则使用images的第一张
  770. if (itemData.imgUrl) {
  771. goodsImage = itemData.imgUrl;
  772. } else if (itemData.images && itemData.images.split(',').length > 0) {
  773. goodsImage = itemData.images.split(',')[0];
  774. }
  775. }
  776. }
  777. } catch (error) {
  778. console.error('解析itemJson失败:', error);
  779. goodsName = '';
  780. goodsImage = '';
  781. }
  782. return {
  783. ...item,
  784. goodsName: goodsName,
  785. goodsImage: goodsImage
  786. };
  787. });
  788. this.integralOrderList = processedData;
  789. this.total = response.total;
  790. this.loading = false;
  791. }).catch(error => {
  792. console.error('查询订单列表失败:', error);
  793. this.loading = false;
  794. this.$message.error('查询订单列表失败');
  795. });
  796. },
  797. // 取消按钮
  798. cancel() {
  799. this.open = false;
  800. this.reset();
  801. },
  802. // 表单重置
  803. reset() {
  804. this.form = {
  805. orderId: null,
  806. orderCode: null,
  807. userId: null,
  808. userName: null,
  809. userPhone: null,
  810. userAddress: null,
  811. itemJson: null,
  812. integral: null,
  813. status: null,
  814. deliveryCode: null,
  815. deliveryName: null,
  816. deliverySn: null,
  817. deliveryTime: null,
  818. createTime: null,
  819. remark: null,
  820. sTime:null,
  821. eTime:null,
  822. };
  823. this.resetForm("form");
  824. },
  825. /** 搜索按钮操作 */
  826. handleQuery() {
  827. this.queryParams.pageNum = 1;
  828. this.getList();
  829. },
  830. /** 重置按钮操作 */
  831. resetQuery() {
  832. this.resetForm("queryForm");
  833. this.createTime=null;
  834. this.queryParams.sTime=null;
  835. this.queryParams.eTime=null;
  836. this.queryParams.qwUserId=null;
  837. this.queryParams.companyId=null;
  838. this.queryParams.companyUserId=null;
  839. // 清除订单号标签
  840. this.queryParams.orderCodes = [];
  841. this.currentInput = '';
  842. this.inputVisible = false;
  843. this.handleQuery();
  844. },
  845. // 多选框选中数据
  846. handleSelectionChange(selection) {
  847. this.ids = selection.map(item => item.orderId)
  848. this.single = selection.length!==1
  849. this.multiple = !selection.length
  850. },
  851. /** 新增按钮操作 */
  852. handleAdd() {
  853. this.reset();
  854. this.open = true;
  855. this.title = "添加积分商品订单";
  856. },
  857. /** 修改按钮操作 */
  858. handleUpdate(row) {
  859. this.reset();
  860. const orderId = row.orderId || this.ids
  861. getIntegralOrder(orderId).then(response => {
  862. this.form = response.data;
  863. this.open = true;
  864. this.title = "修改积分商品订单";
  865. });
  866. },
  867. /** 提交按钮 */
  868. submitForm() {
  869. this.$refs["form"].validate(valid => {
  870. if (valid) {
  871. if (this.form.orderId != null) {
  872. updateIntegralOrder(this.form).then(response => {
  873. this.msgSuccess("修改成功");
  874. this.open = false;
  875. this.getList();
  876. });
  877. } else {
  878. addIntegralOrder(this.form).then(response => {
  879. this.msgSuccess("新增成功");
  880. this.open = false;
  881. this.getList();
  882. });
  883. }
  884. }
  885. });
  886. },
  887. handleClickX(tab, event) {
  888. if(tab.name=="10"){
  889. this.queryParams.status=null;
  890. }else{
  891. this.queryParams.status=tab.name;
  892. }
  893. this.handleQuery();
  894. },
  895. /** 删除按钮操作 */
  896. handleDelete(row) {
  897. const orderIds = row.orderId || this.ids;
  898. this.$confirm('是否确认删除积分商品订单编号为"' + orderIds + '"的数据项?', "警告", {
  899. confirmButtonText: "确定",
  900. cancelButtonText: "取消",
  901. type: "warning"
  902. }).then(function() {
  903. return delIntegralOrder(orderIds);
  904. }).then(() => {
  905. this.getList();
  906. this.msgSuccess("删除成功");
  907. }).catch(() => {});
  908. },
  909. /** 导出按钮操作 */
  910. handleExport() {
  911. const queryParams = this.queryParams;
  912. this.$confirm('是否确认导出所有积分商品订单数据项?', "警告", {
  913. confirmButtonText: "确定",
  914. cancelButtonText: "取消",
  915. type: "warning"
  916. }).then(() => {
  917. this.exportLoading = true;
  918. return exportIntegralOrder(queryParams);
  919. }).then(response => {
  920. this.download(response.msg);
  921. this.exportLoading = false;
  922. }).catch(() => {});
  923. },
  924. //选择企业后触发
  925. getAllUserlist(companyId){
  926. if(companyId){
  927. getAllUserlist({companyId}).then(response => {
  928. this.companyUserNameList=response.data;
  929. });
  930. //企业微信
  931. getQwUserInfo({companyId}).then(response => {
  932. this.qwUserList=response.data;
  933. })
  934. }
  935. },
  936. // 数据分拣按钮点击事件
  937. handleDataSort() {
  938. this.erpAccountDialog.open = true;
  939. this.erpAccountDialog.title = "数据分拣";
  940. this.getErpPhoneList(); // 加载手机号列表
  941. this.updateOrderSummary();
  942. },
  943. // 加载ERP账户数据
  944. async loadErpAccountData() {
  945. try {
  946. const response = await getErpAccount();
  947. this.erpAccountList = response.data;
  948. } catch (error) {
  949. console.error('加载ERP账户失败:', error);
  950. }
  951. },
  952. // 更新订单统计
  953. updateOrderSummary() {
  954. // 如果没有选择任何数据,则使用查询出来的所有数据
  955. if (this.ids.length === 0) {
  956. this.orderSummary.selectedCount = this.total;
  957. this.orderSummary.queryCount = this.total;
  958. } else {
  959. // 如果选择了数据,则使用选中的数据
  960. this.orderSummary.selectedCount = this.ids.length;
  961. this.orderSummary.queryCount = this.total;
  962. }
  963. // 计算总金额(这里需要根据实际数据结构调整)
  964. this.calculateTotalAmount();
  965. },
  966. // 计算总金额
  967. calculateTotalAmount() {
  968. let totalAmount = 0;
  969. if (this.ids.length === 0) {
  970. // 使用所有查询数据计算金额
  971. this.integralOrderList.forEach(order => {
  972. totalAmount += parseFloat(order.payMoney || 0);
  973. });
  974. } else {
  975. // 使用选中的数据计算金额
  976. const selectedOrders = this.integralOrderList.filter(order =>
  977. this.ids.includes(order.orderId)
  978. );
  979. selectedOrders.forEach(order => {
  980. totalAmount += parseFloat(order.payMoney || 0);
  981. });
  982. }
  983. this.orderSummary.totalAmount = totalAmount.toFixed(2);
  984. },
  985. // 取消ERP账户对话框
  986. cancelErpAccountDialog() {
  987. this.erpAccountDialog.open = false;
  988. this.erpAccountForm.selectedAccount = null;
  989. },
  990. // 确认创建ERP订单
  991. async confirmCreateErpOrder() {
  992. // 收集选中的orderId
  993. let selectedOrderIds = [];
  994. if (this.ids.length === 0) {
  995. // 如果没有选择任何数据,使用查询出来的所有数据的orderId
  996. selectedOrderIds = this.integralOrderList.map(order => order.orderId);
  997. } else {
  998. // 如果选择了数据,使用选中的orderId
  999. selectedOrderIds = this.ids;
  1000. }
  1001. // 收集ERP账户和手机号
  1002. const selectedErpAccount = this.erpAccountForm.selectedAccount;
  1003. const selectedPhones = this.erpAccountForm.selectedPhones;
  1004. // 准备请求参数
  1005. const params = {
  1006. orderIds: selectedOrderIds,
  1007. loginAccount: selectedErpAccount,
  1008. erpPhones: selectedPhones // 添加手机号参数
  1009. };
  1010. try {
  1011. this.erpAccountDialog.submitting = true;
  1012. // 根据弹窗标题判断是数据分拣还是创建ERP
  1013. if (this.erpAccountDialog.title === "数据分拣") {
  1014. // 调用数据分拣接口
  1015. const response = await batchSetErpOrder(params);
  1016. this.$message.success('数据分拣成功');
  1017. } else if (this.erpAccountDialog.title === "创建ERP") {
  1018. // 调用创建ERP接口
  1019. const response = await batchCreateErpOrder(params);
  1020. // console.log("参数:",params)
  1021. this.$message.success('创建ERP成功');
  1022. }
  1023. // 关闭弹窗
  1024. this.cancelErpAccountDialog();
  1025. // 刷新列表
  1026. this.getList();
  1027. } catch (error) {
  1028. console.error('操作失败:', error);
  1029. this.$message.error('操作失败');
  1030. } finally {
  1031. this.erpAccountDialog.submitting = false;
  1032. }
  1033. },
  1034. handleCreateErp() {
  1035. this.erpAccountDialog.open = true;
  1036. this.erpAccountDialog.title = "创建ERP";
  1037. this.getErpPhoneList(); // 加载手机号列表
  1038. this.updateOrderSummary();
  1039. },
  1040. // 获取ERP手机号列表
  1041. getErpPhoneList() {
  1042. queryErpPhone().then(response => {
  1043. if (response.data && response.data != null && response.data.length > 0) {
  1044. const phones = response.data.filter(phone => phone.trim());
  1045. this.phoneList = phones.map(phone => ({
  1046. phone: phone.trim(),
  1047. editing: false,
  1048. originalPhone: phone.trim()
  1049. }));
  1050. } else {
  1051. this.phoneList = [];
  1052. }
  1053. });
  1054. },
  1055. // 取消ERP账户对话框
  1056. cancelErpAccountDialog() {
  1057. this.erpAccountDialog.open = false;
  1058. this.erpAccountForm.selectedAccount = null;
  1059. this.erpAccountForm.selectedPhones = []; // 清空手机号选择
  1060. },
  1061. // 推送手机号码管理
  1062. handleErpPhone() {
  1063. this.getErpPhoneList();
  1064. this.erpPhone.open = true;
  1065. },
  1066. // 新增手机号
  1067. handleAddPhone() {
  1068. this.phoneList.push({
  1069. phone: '',
  1070. editing: true,
  1071. originalPhone: '',
  1072. isNew: true
  1073. });
  1074. },
  1075. // 编辑手机号
  1076. handleEditPhone(index) {
  1077. this.$set(this.phoneList[index], 'editing', true);
  1078. this.$set(this.phoneList[index], 'originalPhone', this.phoneList[index].phone);
  1079. },
  1080. // 保存手机号
  1081. handleSavePhone(index) {
  1082. const phone = this.phoneList[index].phone.trim();
  1083. if (!phone) {
  1084. this.$message.error('手机号不能为空');
  1085. return;
  1086. }
  1087. if (!this.validatePhoneFormat(phone)) {
  1088. this.$message.error('请输入正确的手机号格式');
  1089. return;
  1090. }
  1091. // 检查是否重复
  1092. const duplicateIndex = this.phoneList.findIndex((item, idx) =>
  1093. idx !== index && item.phone === phone
  1094. );
  1095. if (duplicateIndex !== -1) {
  1096. this.$message.error('手机号已存在');
  1097. return;
  1098. }
  1099. this.$set(this.phoneList[index], 'editing', false);
  1100. this.$set(this.phoneList[index], 'isNew', false);
  1101. },
  1102. // 取消编辑
  1103. handleCancelEdit(index) {
  1104. if (this.phoneList[index].isNew) {
  1105. // 如果是新增的,直接删除
  1106. this.phoneList.splice(index, 1);
  1107. } else {
  1108. // 如果是编辑的,恢复原值
  1109. this.$set(this.phoneList[index], 'phone', this.phoneList[index].originalPhone);
  1110. this.$set(this.phoneList[index], 'editing', false);
  1111. }
  1112. },
  1113. // 删除手机号
  1114. handleDeletePhone(index) {
  1115. this.$confirm('确认删除该手机号?', '提示', {
  1116. confirmButtonText: '确定',
  1117. cancelButtonText: '取消',
  1118. type: 'warning'
  1119. }).then(() => {
  1120. this.phoneList.splice(index, 1);
  1121. this.$message.success('删除成功');
  1122. });
  1123. },
  1124. // 验证手机号格式
  1125. validatePhoneFormat(phone) {
  1126. const phoneRegex = /^1[3-9]\d{9}$/;
  1127. return phoneRegex.test(phone);
  1128. },
  1129. // 验证手机号
  1130. validatePhone(row) {
  1131. if (row.phone && !this.validatePhoneFormat(row.phone)) {
  1132. this.$message.error('请输入正确的手机号格式');
  1133. }
  1134. },
  1135. // 保存手机号列表
  1136. handleSavePhoneList() {
  1137. // 检查是否有正在编辑的项
  1138. const editingItem = this.phoneList.find(item => item.editing);
  1139. if (editingItem) {
  1140. this.$message.error('请先保存正在编辑的手机号');
  1141. return;
  1142. }
  1143. // 检查是否有空的手机号
  1144. const emptyPhone = this.phoneList.find(item => !item.phone.trim());
  1145. if (emptyPhone) {
  1146. this.$message.error('存在空的手机号,请删除或填写完整');
  1147. return;
  1148. }
  1149. // 构造手机号列表
  1150. const phoneList = this.phoneList.map(item => item.phone);
  1151. // 调用保存接口
  1152. saveErpPhone(phoneList).then(response => {
  1153. if (response.code === 200) {
  1154. this.$message.success('保存成功');
  1155. this.erpPhone.open = false;
  1156. } else {
  1157. this.$message.error(response.msg || '保存失败');
  1158. }
  1159. }).catch(() => {
  1160. this.$message.error('保存失败');
  1161. });
  1162. },
  1163. // 取消手机号对话框
  1164. handleCancelPhoneDialog() {
  1165. this.erpPhone.open = false;
  1166. this.getErpPhoneList(); // 重新加载原始数据
  1167. },
  1168. submitOrderStatusFileForm(){
  1169. this.$refs.uploadStatus.submit();
  1170. },
  1171. // 处理键盘按下事件
  1172. handleKeyDown(event) {
  1173. const { key, target } = event
  1174. // 处理退格键删除标签
  1175. if (key === 'Backspace' && !target.value && this.queryParams.orderCodes.length > 0) {
  1176. event.preventDefault()
  1177. this.removeOrderCode(this.queryParams.orderCodes.length - 1)
  1178. }
  1179. // 处理分隔符
  1180. if ([',', ',', ' ', 'Enter'].includes(key)) {
  1181. event.preventDefault()
  1182. this.handleInputConfirm()
  1183. }
  1184. },
  1185. // 处理键盘抬起事件(实时分割输入)
  1186. handleKeyUp(event) {
  1187. const value = event.target.value
  1188. // 检查是否包含分隔符
  1189. if (/[,,\s]/.test(value)) {
  1190. this.handleInputConfirm()
  1191. }
  1192. },
  1193. // 确认输入
  1194. handleInputConfirm() {
  1195. const inputValue = this.currentInput.trim()
  1196. if (inputValue) {
  1197. // 分割多个订单号
  1198. const codes = inputValue.split(/[,,\s]+/).filter(code => code.trim())
  1199. codes.forEach(code => {
  1200. this.addOrderCode(code.trim())
  1201. })
  1202. }
  1203. this.currentInput = ''
  1204. },
  1205. // 添加订单号
  1206. addOrderCode(code) {
  1207. if (!code) return
  1208. // 检查数量限制
  1209. if (this.maxOrderCodes > 0 && this.queryParams.orderCodes.length >= this.maxOrderCodes) {
  1210. this.$message.warning(`最多只能添加 ${this.maxOrderCodes} 个订单号`)
  1211. return
  1212. }
  1213. // 检查重复
  1214. if (this.queryParams.orderCodes.includes(code)) {
  1215. this.$message.warning(`订单号 "${code}" 已存在`)
  1216. return
  1217. }
  1218. // 添加到列表
  1219. this.queryParams.orderCodes.push(code)
  1220. },
  1221. // 删除订单号
  1222. removeOrderCode(index) {
  1223. this.queryParams.orderCodes.splice(index, 1)
  1224. },
  1225. // 显示输入框
  1226. showInput() {
  1227. this.inputVisible = true
  1228. this.$nextTick(() => {
  1229. this.$refs.tagInput.focus()
  1230. })
  1231. },
  1232. // 聚焦输入框
  1233. focusInput() {
  1234. if (!this.inputVisible) {
  1235. this.showInput()
  1236. }
  1237. },
  1238. }
  1239. };
  1240. </script>
  1241. <style scoped>
  1242. .tag-input-container {
  1243. min-width: 445px;
  1244. }
  1245. .tags-wrapper {
  1246. min-height: 32px;
  1247. padding: 4px 8px;
  1248. border: 1px solid #dcdfe6;
  1249. border-radius: 4px;
  1250. cursor: text;
  1251. display: flex;
  1252. flex-wrap: wrap;
  1253. align-items: center;
  1254. gap: 4px;
  1255. transition: border-color 0.2s;
  1256. }
  1257. .tags-wrapper:hover {
  1258. border-color: #c0c4cc;
  1259. }
  1260. .tags-wrapper:focus-within {
  1261. border-color: #409eff;
  1262. box-shadow: 0 0 0 2px rgba(64, 158, 255, 0.2);
  1263. }
  1264. .order-tag {
  1265. margin: 2px;
  1266. flex-shrink: 0;
  1267. }
  1268. .tag-error {
  1269. background-color: #fef0f0;
  1270. border-color: #fbc4c4;
  1271. color: #f56c6c;
  1272. }
  1273. .tag-input {
  1274. border: none;
  1275. outline: none;
  1276. flex: 1;
  1277. min-width: 120px;
  1278. }
  1279. .tag-input >>> .el-input__inner {
  1280. border: none;
  1281. padding: 0;
  1282. height: 24px;
  1283. line-height: 24px;
  1284. }
  1285. .button-new-tag {
  1286. height: 24px;
  1287. line-height: 22px;
  1288. padding: 0 8px;
  1289. margin: 2px;
  1290. }
  1291. .input-tips {
  1292. margin-top: 4px;
  1293. font-size: 12px;
  1294. color: #909399;
  1295. }
  1296. .tip-text {
  1297. display: flex;
  1298. align-items: center;
  1299. gap: 8px;
  1300. }
  1301. </style>