index.blade.php 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447
  1. @extends('layouts.app')
  2. @section('title')快递查询-包裹管理@endsection
  3. @section('content')
  4. <span id="nav2">
  5. @component('package.menu')@endcomponent
  6. </span>
  7. <div id="list" class="d-none">
  8. <div class="container-fluid">
  9. <div id="form_div"></div>
  10. <div class="ml-3 form-inline" id="btn">
  11. @can('包裹管理-快递-异常类型-编辑')
  12. <span class="dropdown">
  13. <button type="button"
  14. class="btn btn-outline-dark btn-sm form-control-sm dropdown-toggle tooltipTarget"
  15. data-toggle="dropdown" title="导出所有页将会以搜索条件得到的筛选结果,将其全部记录(每一页)导出">导出Excel
  16. </button>
  17. <div class="dropdown-menu">
  18. <a class="dropdown-item" @click="orderPackageExport(false)" href="javascript:">导出勾选内容</a>
  19. <a class="dropdown-item" @click="orderPackageExport(true)" href="javascript:">导出所有页</a>
  20. </div>
  21. </span>
  22. <span class="ml-1">
  23. <select class="form-control-sm" v-model="batchExceptionType">
  24. <option v-for="(value,index) in exception_types" :value="value"
  25. :name="value" :key="index">@{{ value }}
  26. </option>
  27. </select>
  28. <button @click="batchExceptionTypeUpdate()" type="button"
  29. class="btn btn-sm ml-2 btn-outline-primary">批量异常状态修改
  30. </button>
  31. </span>
  32. <span class="ml-1">
  33. <select class="form-control-sm ml-2" v-model="batchStatus">
  34. <option v-for="(value,index) in statuses" :value="value"
  35. :name="value" :key="index">@{{ value }}
  36. </option>
  37. </select>
  38. <button @click="batchStatusUpdate()" type="button"
  39. class="btn btn-sm ml-2 btn-outline-danger">批量状态修改
  40. </button>
  41. </span>
  42. <span class="ml-1">
  43. <button type="button" class="btn btn-outline-dark btn-sm form-control-sm tooltipTarget"
  44. @click="copyLogisticNumber" style="background: #dad7e8;">复制快递单号</button>
  45. </span>
  46. @endcan
  47. </div>
  48. <table class="table table-striped table-bordered table-hover card-body td-min-width-80" id="table">
  49. <tr v-for="(package,i) in packages" @click="selectTr===i+1?selectTr=0:selectTr=i+1"
  50. :class="selectTr===i+1?'focusing' : ''">
  51. <td>
  52. <input class="checkItem" type="checkbox" :value="package.logistic_number">
  53. </td>
  54. <td><span>@{{ i+1 }} <span class="badge badge-danger" v-if="package.order.issue">问题件</span></span>
  55. </td>
  56. <td>
  57. <select class="form-control-sm" v-model="package.exception_type"
  58. @change="package.showEditButton = true">
  59. <option :disabled="!exception_editable" v-for="(value,index) in exception_types"
  60. :value="value" :name="value" :key="index">@{{ value }}
  61. </option>
  62. </select>
  63. <button class="btn btn-info btn-sm text-white mr-1" @click="updateExceptionType(package)"
  64. v-if="package.showEditButton">
  65. 更新状态
  66. </button>
  67. </td>
  68. <td>@{{ package.logistic_number }}</td>
  69. <td>@{{ package.status }}</td>
  70. <td>@{{ package.order != null ? package.order.logistic.name : '#' }}</td>
  71. <td>@{{ package.order != null ? package.order.owner.name : '#' }}</td>
  72. <td>@{{ package.order != null ? package.order.province : '#' }}</td>
  73. <td>@{{ package.sent_at }}</td>
  74. <td>@{{ package.received_at }}</td>
  75. <td>@{{ package.weighed_at }}</td>
  76. <td>
  77. <div v-if="package.transfer_status && package.transfer_status.length>0"
  78. class="text-overflow-warp-200 up" :id="'route-'+i">
  79. <p v-for="route in package.transfer_status">
  80. @{{ route.accept_address+" "+ route.remark+" "+route.accept_time}}
  81. </p>
  82. </div>
  83. <div class="text-overflow-warp-200 " v-if=" package.transfer_status && !showList[i] && package.transfer_status.length > 0">
  84. @{{ package.transfer_status[0].accept_address+" "+ package.transfer_status[0].remark+" "+package.transfer_status[0].accept_time}}
  85. </div>
  86. <div @click="showRoute(i)" v-if="package.transfer_status && package.transfer_status.length > 1">
  87. <label class="text-center mt-0 p-0 cursor-pointer pull-left">
  88. <span class="fa"
  89. :class="package.isShowTransferStatus ? 'fa-angle-double-down' : 'fa-angle-double-right'"></span>
  90. &nbsp;<span v-if="package.transfer_status && showList[i]">收起</span><span
  91. v-else>展开</span>&nbsp;@{{ package.transfer_status.length }} 条
  92. </label>
  93. </div>
  94. </td>
  95. <td @mouseover="remarkHover = package.id"
  96. @mouseleave="remarkHover=null;remark=null;isShowRemarkInput = false">
  97. @can('包裹管理-快递-客服备注')
  98. <button @click="isShowRemarkInput = true" v-if="remarkHover===package.id">新增</button>
  99. <input @keydown.enter="submitRemark(package)"
  100. v-if="isShowRemarkInput && remarkHover===package.id" type="text" v-model="remark">
  101. <div v-if="package.remark && package.remark.length>0" :id="'remark-'+i">
  102. <div v-if="showRemarkList[i]" class="text-overflow-warp-200 up">
  103. <p v-for="remark_item in package.remark">
  104. @{{ remark_item }}
  105. </p>
  106. </div>
  107. <div class="text-overflow-warp-200" v-else>
  108. @{{ package.remark[0] }}
  109. </div>
  110. </div>
  111. <div @click="showRemarkItem(i)" v-if="package.remark && package.remark.length > 1">
  112. <label class="text-center mt-0 p-0 cursor-pointer pull-left">
  113. <span class="fa"
  114. :class="package.isShowRemark ? 'fa-angle-double-down' : 'fa-angle-double-right'"></span>
  115. &nbsp;<span v-if="package.remark && showRemarkList[i]">收起</span><span
  116. v-else>展开</span>&nbsp;@{{ package.remark.length }} 条
  117. </label>
  118. </div>
  119. @endcan
  120. </td>
  121. <td class="text-overflow-warp-200"><span v-if="package.order && package.order.issue">@{{ package.order.issue.result_explain }}</span>
  122. </td>
  123. <td class="text-overflow-warp-200"><span
  124. v-if="package.order && package.order.issue && package.order.issue.issue_type">@{{ package.order.issue.issue_type.name }}</span>
  125. </td>
  126. <td class="text-overflow-warp-200"><span
  127. v-if="package.order && package.order.issue && package.order.issue.logs.length >0"><span
  128. v-for="log in package.order.issue.logs">@{{ log.content }}<br></span></span></td>
  129. <td class="text-overflow-warp-200"><span
  130. v-if="package.order && package.order.issue && package.order.issue.logs.length >0"><span
  131. v-for="log in package.order.issue.logs">@{{ log.user.name }}<br></span></span></td>
  132. <td class="text-overflow-warp-200"><span
  133. v-if="package.order && package.order.issue && package.order.issue.logs.length >0"><span
  134. v-for="log in package.order.issue.logs">@{{ log.created_at }}<br></span></span></td>
  135. </tr>
  136. </table>
  137. <div class="text-info h5 btn btn">{{$orderPackages->count()}}/{{$orderPackages->total()}}</div>
  138. {{$orderPackages->appends($paginateParams)->links()}}
  139. </div>
  140. <textarea id="clipboardDiv" style="opacity:0"></textarea>
  141. </div>
  142. @endsection
  143. @section('lastScript')
  144. <script type="text/javascript" src="{{mix('js/queryForm/export.js')}}"></script>
  145. <script type="text/javascript" src="{{mix('js/queryForm/queryForm.js')}}"></script>
  146. <script type="text/javascript" src="{{mix('js/queryForm/header.js')}}"></script>{{--新版2--}}
  147. <script>
  148. let vue = new Vue({
  149. el: "#list",
  150. data: {
  151. packages: [
  152. @foreach($orderPackages as $package)
  153. {!! $package !!},
  154. @endforeach
  155. ],
  156. logistics: [
  157. @foreach($logistics as $logistic)
  158. {
  159. name: '{{$logistic->id}}', value: '{{$logistic->name}}'
  160. },
  161. @endforeach
  162. ],
  163. owners: [
  164. @foreach($owners as $owner)
  165. {
  166. name: '{{$owner->id}}', value: '{{$owner->name}}'
  167. },
  168. @endforeach
  169. ],
  170. showList: {},
  171. showRemarkList: {},
  172. selectTr: 0,
  173. exception_types: [
  174. '疑似库内丢件',
  175. '揽件异常',
  176. '中转异常',
  177. '疑似丢件',
  178. '派件异常',
  179. '其他',
  180. '无',
  181. ],
  182. statuses: [
  183. '已揽收',
  184. '无',
  185. ],
  186. exception_editable: @can('包裹管理-快递-异常类型-编辑') true @else false @endcan,
  187. selectedExceptionType: '修改异常类型',
  188. batchExceptionType: null,
  189. batchStatus: null,
  190. remarkHover: null,
  191. remark: null,
  192. isShowRemarkInput: false
  193. },
  194. created() {
  195. $.each(this.packages, function (index, item) {
  196. item.isShowTransferStatus = false;
  197. item.isShowRemark = false;
  198. if (item.transfer_status != null && item.transfer_status.length > 1) {
  199. item.transfer_status.sort(function (item1, item2) {
  200. let date1 = new Date(item1.accept_time);
  201. let date2 = new Date(item2.accept_time);
  202. if (date1 - date2 > 0) return -1;
  203. if (date1 - date2 < 0) return 1;
  204. return 0;
  205. });
  206. }
  207. });
  208. },
  209. mounted() {
  210. $('#list').removeClass('d-none');
  211. let _this = this;
  212. $(".up").slideUp();
  213. let data = [
  214. [
  215. /*"","","","","","","","","","",""*/
  216. {name: 'logistic_number', type: 'input', tip: '可支持多快递单号,糊模查找需要在右边打上%符号', placeholder: '快递单号'},
  217. {
  218. name: 'status',
  219. type: 'select',
  220. tip: ['输入关键词快速定位下拉列表,回车确定', '选择要显示的状态'],
  221. placeholder: '状态',
  222. data: [{name: '无', value: '无'}, {name: '已称重', value: '已称重'}, {
  223. name: '已揽收',
  224. value: '已揽收'
  225. }, {name: '在途', value: '在途'}, {name: '在途异常', value: '在途异常'}, {
  226. name: '派送中',
  227. value: '派送中'
  228. }, {
  229. name: '已收件',
  230. value: '已收件'
  231. }, {name: '派送异常', value: '派送异常'}, {name: '返回中', value: '返回中'}, {
  232. name: '返回异常',
  233. value: '返回异常'
  234. }, {name: '返回派件', value: '返回派件'}, {name: '其他异常', value: '其他异常'},]
  235. },
  236. {name: 'sent_at_start', type: 'dateTime', tip: '选择显示发出时间的起始时间'},
  237. {
  238. name: 'is_weighed',
  239. type: 'select',
  240. tip: ['输入关键词快速定位下拉列表,回车确定', '选择要显示的状态'],
  241. placeholder: '是否称重',
  242. data: [{name: false, value: '无'}, {name: true, value: '已称重'}]
  243. },
  244. {name: 'received_at_start', type: 'dateTime', tip: '选择显示收货时间的起始时间'},
  245. {
  246. name: 'has_transfer_status',
  247. type: 'select',
  248. tip: ['输入关键词快速定位下拉列表,回车确定', '选择要显示的状态'],
  249. placeholder: '是否有物流信息',
  250. data: [{name: '是', value: '是'}, {name: '否', value: '否'}]
  251. },
  252. {name: 'default_date', type: 'checkbox', tip: '默认15天', data: [{name: 'ture', value: '默认15天'}]},
  253. ], [
  254. {
  255. name: 'logistic_id',
  256. type: 'select_multiple_select',
  257. tip: ['输入关键词快速定位下拉列表,回车确定', '选择要显示的快递'],
  258. placeholder: ['快递', '定位或多选快递'],
  259. data: _this.logistics
  260. },
  261. {
  262. name: 'owner_id',
  263. type: 'select_multiple_select',
  264. tip: ['输入关键词快速定位下拉列表,回车确定', '选择要显示的货主'],
  265. placeholder: ['货主', '定位或多选货主'],
  266. data: _this.owners
  267. },
  268. {name: 'sent_at_end', type: 'dateTime', tip: '选择显示发出时间的截止时间'},
  269. {
  270. name: 'is_exception',
  271. type: 'select',
  272. tip: ['输入关键词快速定位下拉列表,回车确定', '选择要显示的状态'],
  273. placeholder: '是否有异常',
  274. data: [{name: '是', value: '是'}, {name: '否', value: '否'}]
  275. },
  276. {name: 'received_at_end', type: 'dateTime', tip: '选择显示收货时间的截止时间'},
  277. {
  278. name: 'exception_type',
  279. type: 'select',
  280. tip: ['输入关键词快速定位下拉列表,回车确定', '选择要显示的状态'],
  281. placeholder: '异常类型',
  282. data: [
  283. {name: '疑似库内丢件', value: '疑似库内丢件'},
  284. {name: '揽件异常', value: '揽件异常'},
  285. {name: '中转异常', value: '中转异常'},
  286. {name: '疑似丢件', value: '疑似丢件'},
  287. {name: '派件异常', value: '派件异常'},
  288. {name: '其他', value: '其他'},
  289. {name: '无', value: '无'}
  290. ]
  291. },
  292. ]
  293. ];
  294. _this.form = new query({
  295. el: '#form_div',
  296. condition: data,
  297. appendDom : "btn",
  298. });
  299. _this.form.init();
  300. let column = [
  301. {name: 'index', value: '序号', neglect: true},
  302. {name: 'exception_type', value: '异常类型'},
  303. {name: 'logistic_number', value: '单号'},
  304. {name: 'status', value: '状态'},
  305. {name: 'logistic_name', value: '快递公司'},
  306. {name: 'owner_name', value: '货主'},
  307. {name: 'province', value: '省份'},
  308. {name: 'sent_at', value: '发出日期'},
  309. {name: 'received_at', value: '收货日期'},
  310. {name: 'weighed_at', value: '称重日期'},
  311. {name: 'transfer_status', value: '快递路由'},
  312. {name: 'remark', value: '客服备注'},
  313. {name: 'result_explain', value: '情况说明'},
  314. {name: 'issue_type', value: '问题类别'},
  315. {name: 'content', value: '说明'},
  316. {name: 'operation_name', value: '操作者'},
  317. {name: 'operation_date', value: '时间'},
  318. ];
  319. new Header({
  320. el: "table",
  321. name: "package",
  322. column: column,
  323. data: this.packages,
  324. restorationColumn: 'addtime',
  325. fixedTop: ($('#form_div').height()) + ($('#btn').height()) + 1,
  326. }).init();
  327. },
  328. methods: {
  329. showRoute(id) {
  330. if (this.showList[id]) {
  331. this.$set(this.showList, id, false);
  332. $("#route-" + id).slideUp();
  333. } else {
  334. this.$set(this.showList, id, true);
  335. $("#route-" + id).slideDown();
  336. }
  337. this.$forceUpdate();
  338. },
  339. showRemarkItem(id) {
  340. if (this.showRemarkList[id]) {
  341. this.$set(this.showRemarkList, id, false);
  342. $("#remark-" + id).slideUp();
  343. } else {
  344. this.$set(this.showRemarkList, id, true);
  345. $("#remark-" + id).slideDown();
  346. }
  347. this.$forceUpdate();
  348. },
  349. updateExceptionType(orderPackages) {
  350. let url = '{{ url("package/logistic") }}' + '/' + orderPackages.id;
  351. let data = {id: orderPackages['id'], exception_type: orderPackages.exception_type}
  352. axios.patch(url, data).then((res) => {
  353. window.tempTip.showSuccess('异常状态修改成功');
  354. });
  355. },
  356. batchExceptionTypeUpdate() {
  357. let _this = this;
  358. if (checkData.length === 0) {
  359. tempTip.show('没有勾选记录');
  360. return
  361. }
  362. axios.put('{{url('package/logistic/batchUpdate')}}', {
  363. exception_type: this.batchExceptionType,
  364. logistic_numbers: checkData
  365. }).then(() => {
  366. tempTip.setDuration(1000);
  367. tempTip.showSuccess('批量更新异常状态成功');
  368. })
  369. },
  370. batchStatusUpdate() {
  371. let _this = this;
  372. if (checkData.length === 0) {
  373. tempTip.show('没有勾选记录');
  374. return
  375. }
  376. axios.put('{{url('package/logistic/batchUpdate')}}', {
  377. status: this.batchStatus,
  378. logistic_numbers: checkData
  379. }).then(() => {
  380. tempTip.setDuration(1000);
  381. tempTip.showSuccess('批量状态成功');
  382. })
  383. },
  384. submitRemark(orderPackage) {
  385. if (this.remark === null) {
  386. return
  387. }
  388. let url = '{{url('apiLocal/package/logistic/')}}';
  389. let _this = this;
  390. axios.put(url, {remark: _this.remark, orderPackageId: orderPackage.id}).then(function (response) {
  391. if (response.data.success) {
  392. orderPackage.remark = response.data.data
  393. tempTip.setDuration(2000);
  394. tempTip.showSuccess('成功!');
  395. } else {
  396. tempTip.setDuration(5000);
  397. tempTip.show(response.data.fail_info);
  398. }
  399. }).catch(function (err) {
  400. tempTip.setDuration(3000);
  401. tempTip.show("网络错误:" + err)
  402. });
  403. },
  404. copyLogisticNumber() {
  405. if (checkData.length === 0) {
  406. tempTip.show('没有勾选');
  407. return;
  408. }
  409. this.copyText(checkData.join('\n'));
  410. },
  411. copyText(text) {
  412. try {
  413. $('#clipboardDiv').text(text).select().focus();
  414. document.execCommand("Copy");
  415. tempTip.setIndex(1052)
  416. tempTip.setDuration(2000)
  417. tempTip.showSuccess('复制成功')
  418. } catch (e) {
  419. tempTip.setIndex(1052)
  420. tempTip.setDuration(2000)
  421. tempTip.showSuccess('复制失败:' + e)
  422. }
  423. },
  424. orderPackageExport(sign) {
  425. let url = '{{url('package/logistic/export')}}';
  426. let token = '{{ csrf_token() }}';
  427. if (sign) {
  428. excelExport(true, checkData, url, this.total, token);
  429. } else {
  430. excelExport(false, checkData, url, null, token);
  431. }
  432. },
  433. },
  434. filters: {
  435. toObjected: function (value) {
  436. return JSON.parse(value);
  437. }
  438. },
  439. });
  440. </script>
  441. @endsection