salesToProduction.vue 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600
  1. <template>
  2. <div class="ele-body">
  3. <el-card shadow="never">
  4. <div class="body-title">
  5. <div class="title-left">销售订单转生产计划</div>
  6. <div class="title-right">
  7. <el-button @click="cancel">取消</el-button>
  8. <el-button type="primary" @click="toSubmit" :loading="loading">
  9. 提交计划
  10. </el-button>
  11. </div>
  12. </div>
  13. <el-form ref="form" :model="form" :rules="rules" label-width="90px" class="formbox">
  14. <el-row :gutter="24">
  15. <el-col v-bind="styleResponsive ? { lg: 6, md: 12 } : { span: 6 }">
  16. <el-form-item label="计划类型:">
  17. <DictSelection dictName="订单计划类型" clearable v-model="form.planType" disabled>
  18. </DictSelection>
  19. </el-form-item>
  20. </el-col>
  21. <el-col v-bind="styleResponsive ? { lg: 6, md: 12 } : { span: 6 }">
  22. <el-form-item label="工艺路线:" prop="produceRoutingName">
  23. <el-input @click.native="openVersion" placeholder="请选择工艺路线" readonly
  24. v-model="form.produceRoutingName"></el-input>
  25. </el-form-item>
  26. </el-col>
  27. <el-col v-bind="styleResponsive ? { lg: 3, md: 6 } : { span: 3 }">
  28. <el-form-item label="使用改型:" prop="modification">
  29. <el-checkbox v-model="form.modification" :true-label="1" :false-label="0"></el-checkbox>
  30. </el-form-item>
  31. </el-col>
  32. <el-col v-bind="styleResponsive ? { lg: 6, md: 12 } : { span: 6 }">
  33. <el-form-item label="余量系数:" prop="marginCoefficient">
  34. <DictSelection @itemChange="itemChange" style="width: 200px" dictName="余量系数"
  35. v-model="form.marginCoefficient"></DictSelection>
  36. </el-form-item>
  37. </el-col>
  38. </el-row>
  39. <el-row :gutter="24" class="row-intro">
  40. <el-col v-bind="styleResponsive ? { lg: 3, md: 12 } : { span: 3 }">
  41. <el-form-item label="销售单数量:">{{ form.codeNum }}</el-form-item>
  42. </el-col>
  43. <el-col v-bind="styleResponsive ? { lg: 3, md: 12 } : { span: 3 }">
  44. <el-form-item label="订单总数量:">{{
  45. form.contractNum
  46. }}</el-form-item>
  47. </el-col>
  48. <el-col v-bind="styleResponsive ? { lg: 4, md: 12 } : { span: 4 }">
  49. <el-form-item label="订单总重量:">{{
  50. form.sumOrderWeight
  51. }}</el-form-item>
  52. </el-col>
  53. <el-col v-bind="styleResponsive ? { lg: 3, md: 12 } : { span: 3 }">
  54. <el-form-item label="物料编码:">{{
  55. form.productCode
  56. }}</el-form-item>
  57. </el-col>
  58. <el-col v-bind="styleResponsive ? { lg: 3, md: 12 } : { span: 3 }">
  59. <el-form-item label="牌号:">{{ form.brandNo }}</el-form-item>
  60. </el-col>
  61. <el-col v-bind="styleResponsive ? { lg: 5, md: 12 } : { span: 5 }">
  62. <el-form-item label="型号:">{{ form.model }}</el-form-item>
  63. </el-col>
  64. <el-col v-bind="styleResponsive ? { lg: 3, md: 12 } : { span: 3 }">
  65. <el-form-item label="库存数量:">{{
  66. form.stockCountBase
  67. }}</el-form-item>
  68. </el-col>
  69. </el-row>
  70. <el-table :data="form.salesOrders" border>
  71. <el-table-column label="序号" align="center" width="60">
  72. <template slot-scope="scope">
  73. <span>{{ scope.$index + 1 }}</span>
  74. </template>
  75. </el-table-column>
  76. <el-table-column label="销售订单号" align="center" prop="code" width="120">
  77. </el-table-column>
  78. <el-table-column label="行号" align="center" prop="lineNumber">
  79. </el-table-column>
  80. <el-table-column label="合同数量" align="center" prop="contractNum">
  81. </el-table-column>
  82. <el-table-column label="合同重量" align="center" prop="productSumWeight">
  83. </el-table-column>
  84. <el-table-column label="欠交数量" align="center" prop="lackNum">
  85. </el-table-column>
  86. <el-table-column label="计划生产数量" align="center" prop="planProductNum" width="120">
  87. <template slot-scope="scope">
  88. <el-form-item label-width="0px" :prop="'salesOrders.' + scope.$index + '.planProductNum'" :rules="{
  89. required: true,
  90. message: '请输入计划生产数量',
  91. trigger: 'blur'
  92. }" class="table-item">
  93. <el-input v-model.number="scope.row.planProductNum" size="small"
  94. oninput="value=value.replace(/[^\d]/g,'')" style="width: 100%"
  95. @blur="scope.row.requiredFormingNum = toInt(scope.row.planProductNum)" placeholder="输入数量"></el-input>
  96. </el-form-item>
  97. </template>
  98. </el-table-column>
  99. <el-table-column label="要求成型数量" align="center" prop="requiredFormingNum" width="120">
  100. <template slot-scope="scope">
  101. <el-form-item label-width="0px" :prop="'salesOrders.' + scope.$index + '.requiredFormingNum'" :rules="{
  102. required: true,
  103. message: '请输入要求成型数量',
  104. trigger: 'blur'
  105. }" class="table-item">
  106. <el-input v-model.number="scope.row.requiredFormingNum" size="small" disabled
  107. oninput="value=value.replace(/[^\d]/g,'')" style="width: 100%" placeholder="输入数量"></el-input>
  108. </el-form-item>
  109. </template>
  110. </el-table-column>
  111. <el-table-column label="按单按库" align="center" prop="orderLibraryType">
  112. <template slot-scope="{ row }">
  113. {{ getDictValue('按单按库', row.orderLibraryType) }}
  114. </template>
  115. </el-table-column>
  116. <el-table-column label="订单类型" align="center" prop="orderType">
  117. <template slot-scope="{ row }">
  118. {{ getDictValue('订单类型', row.orderType) }}
  119. </template>
  120. </el-table-column>
  121. <el-table-column label="交付日期" align="center" prop="deliveryTime" width="160">
  122. </el-table-column>
  123. <el-table-column label="要求成型日期" align="center" prop="reqMoldTime" width="180">
  124. <template slot-scope="scope">
  125. <el-form-item label-width="0px" :prop="'salesOrders.' + scope.$index + '.reqMoldTime'" :rules="{
  126. required: true,
  127. message: '请选择要求成型日期',
  128. trigger: 'blur'
  129. }" class="table-item">
  130. <el-date-picker style="width: 100%" v-model="scope.row.reqMoldTime" :pickerOptions="{
  131. disabledDate: (time) =>
  132. time.getTime() <
  133. new Date(new Date().setHours(0, 0, 0, 0)).getTime()
  134. }" type="date" placeholder="选择日期" value-format="yyyy-MM-dd">
  135. </el-date-picker>
  136. <!-- <el-date-picker
  137. style="width: 100%;"
  138. v-model="scope.row.reqMoldTime"
  139. type="datetime"
  140. placeholder="选择日期"
  141. value-format="yyyy-MM-dd HH:mm:ss">
  142. </el-date-picker> -->
  143. </el-form-item>
  144. </template>
  145. </el-table-column>
  146. <el-table-column label="客户名称" align="center" prop="customerName">
  147. </el-table-column>
  148. <el-table-column label="业务员" align="center" prop="salesman">
  149. </el-table-column>
  150. <el-table-column label="交付要求" align="center" prop="deliveryRequirements">
  151. <template slot-scope="{ row }">
  152. {{ getDictValue('交付要求', row.deliveryRequirements) }}
  153. </template>
  154. </el-table-column>
  155. <el-table-column prop="priority" label="优先级" width="100" align="center">
  156. <template slot-scope="{ row }">{{ row.priority }}
  157. <div class="sort-wrap">
  158. <i class="el-icon-caret-top" @click="sortTop(row)"></i>
  159. <i class="el-icon-caret-bottom" @click="sortBottom(row)"></i>
  160. </div>
  161. </template>
  162. </el-table-column>
  163. <el-table-column label="操作" align="center" width="70" fixed="right">
  164. <template slot-scope="scope">
  165. <el-button type="text" @click="handleDeleteItem(scope.$index)">删除</el-button>
  166. </template>
  167. </el-table-column>
  168. </el-table>
  169. <div class="add-product" @click="addEquipment">
  170. <i class="el-icon-circle-plus-outline"></i>
  171. </div>
  172. <el-row :gutter="24">
  173. <el-col v-bind="styleResponsive ? { lg: 24, md: 24 } : { span: 24 }">
  174. <el-form-item label="计划备注:">
  175. <el-input type="textarea" :rows="4" v-model="form.notes" resize="none"></el-input>
  176. </el-form-item>
  177. </el-col>
  178. </el-row>
  179. </el-form>
  180. <AdditionalOrder ref="additionalRefs" :productCode="form.productCode" :selectList="form.salesOrders"
  181. @choose="confirmChoose"></AdditionalOrder>
  182. <ProductionVersion ref="versionRefs" @changeProduct="changeProduct"></ProductionVersion>
  183. <PlanSubmit ref="submitRefs" :type="$route.query.type" :info="form" @publish="publishData"></PlanSubmit>
  184. </el-card>
  185. </div>
  186. </template>
  187. <script>
  188. import AdditionalOrder from './components/AdditionalOrder.vue';
  189. import PlanSubmit from './components/plan-submit.vue';
  190. import ProductionVersion from '@/components/CreatePlan/ProductionVersion2.vue';
  191. import {
  192. productionToPlan,
  193. saveSaleToPlan,
  194. updateSaleToPlan,
  195. releaseSave,
  196. getInventory,
  197. getUpdateInfoById,
  198. getProductVersion
  199. } from '@/api/saleOrder';
  200. import dictMixins from '@/mixins/dictMixins';
  201. import { deepClone } from '@/utils/index';
  202. import { getRouteTabKey, removePageTab } from '@/utils/page-tab-util';
  203. import { getCode } from '@/api/codeManagement';
  204. import dayjs from 'dayjs';
  205. export default {
  206. mixins: [dictMixins],
  207. components: {
  208. AdditionalOrder,
  209. ProductionVersion,
  210. PlanSubmit
  211. },
  212. data() {
  213. return {
  214. type: this.$route.query.type,
  215. form: {
  216. planType: 1,
  217. produceRoutingId: '',
  218. stockCountBase: '',
  219. salesOrders: [],
  220. produceRoutingName: '',
  221. marginCoefficient: '1',
  222. },
  223. // 表单验证规则
  224. rules: {
  225. produceRoutingName: [
  226. { required: true, message: '请选择生产版本', trigger: 'change' }
  227. ]
  228. },
  229. // selection: [],
  230. loading: false
  231. };
  232. },
  233. computed: {
  234. // 是否开启响应式布局
  235. styleResponsive() {
  236. return this.$store.state.theme.styleResponsive;
  237. }
  238. },
  239. created() {
  240. this.requestDict('按单按库');
  241. this.requestDict('订单类型');
  242. this.requestDict('交付要求');
  243. if (this.type == 'edit') {
  244. this.getPlanInfo(this.$route.query.id);
  245. } else {
  246. this.getSaleInfo();
  247. }
  248. },
  249. methods: {
  250. async getPlanInfo(id) {
  251. const data = await getUpdateInfoById(id);
  252. this.form = data;
  253. },
  254. async _getInventory() {
  255. const res = await getInventory(
  256. this.form.productCode,
  257. this.form.planType
  258. );
  259. this.form.stockCountBase = res;
  260. },
  261. getSaleInfo() {
  262. let params = JSON.parse(this.$route.query.selection)
  263. productionToPlan(params).then((res) => {
  264. this.form = deepClone(res);
  265. this.form.salesOrders.map((item, index) => {
  266. item.priority = index + 1;
  267. item.planProductNum = item.lackNum;
  268. item.requiredFormingNum = item.lackNum;
  269. item.reqMoldTime = dayjs(
  270. new Date(item.deliveryTime).getTime() - 3600 * 1000 * 24 * 10
  271. ).format('YYYY-MM-DD');
  272. });
  273. if (this.form.salesOrders.every((itm) => itm.orderType == 2)) {
  274. this.form.planType = 2;
  275. } else if (this.form.salesOrders.every((itm) => itm.orderType == 1)) {
  276. this.form.planType = 1;
  277. } else {
  278. this.form.planType = 3;
  279. }
  280. this._getInventory();
  281. });
  282. this.$forceUpdate()
  283. },
  284. itemChange() {
  285. this.form.salesOrders.map((item, index) => {
  286. item.requiredFormingNum = (item.lackNum * (this.form.marginCoefficient || 1)).toFixed(0);
  287. })
  288. },
  289. toInt(planProductNum) {
  290. return (planProductNum * (this.form.marginCoefficient || 1)).toFixed(0)
  291. },
  292. cancel() {
  293. const key = getRouteTabKey();
  294. this.$router.go(-1);
  295. removePageTab({ key });
  296. },
  297. toSubmit() {
  298. this.$refs.form.validate((valid) => {
  299. if (valid) {
  300. this.mapList();
  301. this.$refs.submitRefs.open();
  302. }
  303. });
  304. },
  305. // 对比日期,计算要求成型重量
  306. mapList() {
  307. var requiredFormingWeight = 0;
  308. var requiredFormingNum = 0;
  309. var productNum = 0;
  310. this.form.salesOrders.map((item, index) => {
  311. requiredFormingWeight =
  312. requiredFormingWeight +
  313. Number(item.productUnitWeight) * Number(item.requiredFormingNum);
  314. requiredFormingNum = requiredFormingNum + item.requiredFormingNum;
  315. productNum += Number(item.planProductNum);
  316. });
  317. this.form.productNum = productNum;
  318. this.form.productUnitWeight =
  319. this.form.salesOrders[0]?.productUnitWeight;
  320. this.form.requiredFormingWeight = requiredFormingWeight.toFixed(2);
  321. this.form.requiredFormingNum = requiredFormingNum;
  322. const collection = deepClone(this.form.salesOrders);
  323. const sortedCollection = collection.sort(
  324. (a, b) => new Date(b.reqMoldTime) - new Date(a.reqMoldTime)
  325. );
  326. let latestData = {};
  327. for (let i = 0; i < sortedCollection.length; i++) {
  328. const data = sortedCollection[i];
  329. if (
  330. !latestData.reqMoldTime ||
  331. new Date(data.reqMoldTime) >= new Date(latestData.reqMoldTime)
  332. ) {
  333. latestData = data;
  334. }
  335. }
  336. this.form.reqMoldTime = latestData.reqMoldTime;
  337. },
  338. sortTop(row) {
  339. if (row.priority <= 1) {
  340. return;
  341. }
  342. let rowN = this.form.salesOrders.find(
  343. (n) => n.priority == row.priority - 1
  344. );
  345. rowN.priority += 1;
  346. row.priority -= 1;
  347. this.form.salesOrders.sort((a, b) => {
  348. return a.priority - b.priority;
  349. });
  350. },
  351. sortBottom(row) {
  352. if (row.priority >= this.form.salesOrders.length) {
  353. return;
  354. }
  355. let rowN = this.form.salesOrders.find(
  356. (n) => n.priority == row.priority + 1
  357. );
  358. rowN.priority -= 1;
  359. row.priority += 1;
  360. this.form.salesOrders.sort((a, b) => {
  361. return a.priority - b.priority;
  362. });
  363. },
  364. againSort() {
  365. this.form.salesOrders.forEach((n, index) => {
  366. n.priority = index + 1;
  367. });
  368. },
  369. // 删除产品
  370. handleDeleteItem(index) {
  371. this.form.salesOrders.splice(index, 1);
  372. },
  373. addEquipment() {
  374. this.$refs.additionalRefs.open(this.form.planType);
  375. },
  376. openVersion() {
  377. this.$refs.versionRefs.open();
  378. },
  379. changeProduct(data) {
  380. this.$set(this.form, 'produceRoutingName', data.name);
  381. this.$set(this.form, 'produceRoutingId', data.id);
  382. this.$set(this.form, 'produceVersionName', data.produceVersionName)
  383. },
  384. confirmChoose(list) {
  385. // 取出在弹窗中选中并且不在表格中的数据
  386. const result = list.filter(
  387. (i) => this.form.salesOrders.findIndex((p) => p.id === i.id) === -1
  388. );
  389. // 取出在表格中并且不在弹窗中选中的数据 即取消选中的数据
  390. const del = this.form.salesOrders.filter(
  391. (i) => list.findIndex((p) => p.id === i.id) === -1
  392. );
  393. for (let i = this.form.salesOrders.length - 1; i >= 0; i--) {
  394. for (let j in del) {
  395. if (this.form.salesOrders[i].id === del[j].id) {
  396. this.form.salesOrders.splice(i, 1);
  397. }
  398. }
  399. }
  400. let priority =
  401. this.form.salesOrders[this.form.salesOrders.length - 1]?.priority ||
  402. 0;
  403. this.form.salesOrders = this.form.salesOrders.concat(
  404. result.map((item, index) => {
  405. item.priority = ++priority;
  406. item.planProductNum = item.lackNum;
  407. item.requiredFormingNum = item.lackNum;
  408. item.reqMoldTime = dayjs(
  409. new Date(item.deliveryTime).getTime() - 3600 * 1000 * 24 * 10
  410. ).format('YYYY-MM-DD');
  411. return item;
  412. })
  413. );
  414. this.changeData();
  415. },
  416. changeData() {
  417. var planProductNum = 0;
  418. var productWeight = 0;
  419. this.form.salesOrders.map((item, index) => {
  420. item.priority = index + 1;
  421. planProductNum = planProductNum + item.contractNum;
  422. productWeight = productWeight + Number(item.productSumWeight);
  423. });
  424. this.$set(this.form, 'codeNum', this.form.salesOrders.length);
  425. this.$set(this.form, 'contractNum', planProductNum);
  426. this.$set(this.form, 'sumOrderWeight', productWeight.toFixed(2));
  427. },
  428. async publishData(type) {
  429. const key = getRouteTabKey();
  430. let params = deepClone(this.form);
  431. params.categoryId = params.salesOrders[0]?.categoryId;
  432. if (this.$route.query.type != 'edit') {
  433. delete params.id;
  434. }
  435. if (type === 2) {
  436. this.$confirm('发布工单后不可撤回,确定发布吗?', '发布确认').then(
  437. async () => {
  438. const loading = this.$loading({
  439. lock: true,
  440. fullscreen: true,
  441. text: '工单发布中...'
  442. });
  443. try {
  444. const code = await getCode('product_order_code');
  445. const data = {
  446. productionPlan: params,
  447. workOrder: {
  448. productionPlanCode: params.code,
  449. code: code,
  450. formingNum: params.contractNum,
  451. formingWeight: params.sumOrderWeight,
  452. produceRoutingId: params.produceRoutingId,
  453. status: 4,
  454. model: params.model,
  455. brandNo: params.brandNo,
  456. categoryId: params.categoryId,
  457. productCode: params.productCode,
  458. productName: params.productName
  459. }
  460. };
  461. if (this.$route.query.type == 'edit') {
  462. data.workOrder.productionPlanId = params.id;
  463. }
  464. await releaseSave(data)
  465. .then((res) => {
  466. if (res === 1) {
  467. this.$message.success('工单已发布!');
  468. this.$router.push({
  469. path: '/productionPlan'
  470. });
  471. } else {
  472. this.$confirm(
  473. '生产计划创建成功,但工单发布失败。请前往【生产计划】列表【重新发布】工单',
  474. '提示',
  475. {
  476. confirmButtonText: '返回',
  477. cancelButtonText: '立即前往',
  478. type: 'warning'
  479. }
  480. )
  481. .then(() => {
  482. this.$router.push({
  483. path: '/productionPlan'
  484. });
  485. })
  486. .catch(() => {
  487. this.$router.go(-1);
  488. });
  489. }
  490. removePageTab({ key });
  491. })
  492. .catch(() => {
  493. this.$message.error('发布失败,请重新发布!');
  494. });
  495. } catch (error) { }
  496. loading.close();
  497. }
  498. );
  499. } else {
  500. let request =
  501. this.$route.query.type == 'edit'
  502. ? updateSaleToPlan
  503. : saveSaleToPlan;
  504. request(params)
  505. .then(async (res) => {
  506. // 提交
  507. this.$router.push({
  508. path: '/productionPlan'
  509. });
  510. removePageTab({ key });
  511. })
  512. .catch(() => {
  513. this.$message.error('提交失败,请重新提交!');
  514. });
  515. }
  516. }
  517. }
  518. };
  519. </script>
  520. <style lang="scss" scoped>
  521. .ele-body {
  522. background: #fff;
  523. }
  524. .body-title {
  525. width: 100%;
  526. display: flex;
  527. align-items: center;
  528. justify-content: space-between;
  529. }
  530. .title-left {
  531. font-size: 20px;
  532. color: #333;
  533. }
  534. .formbox {
  535. margin: 20px auto;
  536. }
  537. .row-intro {
  538. border-bottom: 1px dashed #ccc;
  539. margin-bottom: 20px;
  540. }
  541. .sort-wrap {
  542. i {
  543. font-size: 30px;
  544. cursor: pointer;
  545. }
  546. .el-icon-caret-top {
  547. color: red;
  548. }
  549. .el-icon-caret-bottom {
  550. color: #157a2c;
  551. }
  552. }
  553. .add-product {
  554. width: 100%;
  555. display: flex;
  556. align-items: center;
  557. justify-content: flex-end;
  558. font-size: 30px;
  559. color: #1890ff;
  560. margin: 10px 0;
  561. cursor: pointer;
  562. }
  563. .table-item {
  564. margin-bottom: 0;
  565. }
  566. </style>