index.blade.php 21 KB

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