addOpportunityDialog.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. <template>
  2. <div>
  3. <headerTitle title="基本信息"></headerTitle>
  4. <el-form
  5. label-width="120px"
  6. ref="form"
  7. :model="form"
  8. :rules="rules"
  9. style="margin-top: 30px"
  10. >
  11. <el-row>
  12. <el-col :span="8" style="height: 58px">
  13. <el-form-item label="选择客户" prop="contactName">
  14. <el-input
  15. placeholder="请选择"
  16. v-model="form.contactName"
  17. @click.native="handParent"
  18. maxlength="50"
  19. ></el-input>
  20. </el-form-item>
  21. </el-col>
  22. <el-col :span="8">
  23. <el-form-item label=" 商机名称" prop="name">
  24. <el-input
  25. placeholder="请输入"
  26. v-model="form.name"
  27. maxlength="50"
  28. ></el-input>
  29. </el-form-item>
  30. </el-col>
  31. <el-col :span="8" style="height: 58px">
  32. <el-form-item label="选择负责人" prop="responsibleName">
  33. <el-input
  34. placeholder="请选择"
  35. v-model="userInfo.name"
  36. disabled
  37. maxlength="50"
  38. ></el-input>
  39. </el-form-item>
  40. </el-col>
  41. <el-col :span="8" style="height: 58px">
  42. <el-form-item label="商机来源" prop="sourceCode">
  43. <DictSelection
  44. dictName="商机来源"
  45. clearable
  46. v-model="form.sourceCode"
  47. >
  48. </DictSelection>
  49. </el-form-item>
  50. </el-col>
  51. <el-col :span="8" style="height: 58px">
  52. <el-form-item label="赢单率" prop="winRate">
  53. <el-input placeholder="请输入内容" v-model="form.winRate">
  54. <template slot="append">%</template>
  55. </el-input>
  56. </el-form-item>
  57. </el-col>
  58. <el-col :span="8" style="height: 58px">
  59. <el-form-item label="预算" prop="budget">
  60. <el-input placeholder="请输入内容" v-model="form.budget">
  61. <template slot="append">万元</template>
  62. </el-input>
  63. </el-form-item>
  64. </el-col>
  65. <el-col :span="8">
  66. <el-form-item label="预计结单日期">
  67. <el-date-picker
  68. v-model="form.expectedClosingDate"
  69. value-format="yyyy-MM-dd"
  70. placeholder="请选择"
  71. type="date"
  72. style="width: 100%"
  73. class="filter-item"
  74. ></el-date-picker>
  75. </el-form-item>
  76. </el-col>
  77. <!-- <el-col :span="8">
  78. <el-form-item label="商机阶段" prop="stageCode">
  79. <DictSelection
  80. dictName="商机阶段"
  81. clearable
  82. v-model="form.stageCode"
  83. >
  84. </DictSelection>
  85. </el-form-item>
  86. </el-col> -->
  87. <!-- <el-col :span="8">
  88. <el-form-item
  89. label="计价方式"
  90. prop="pricingWay"
  91. >
  92. <el-radio-group v-model="form.pricingWay" @change="changePricingWay" :disabled="!!form.contractId">
  93. <el-radio v-for="item in pricingWayList" :label="item.id">{{ item.name }}</el-radio>
  94. </el-radio-group>
  95. </el-form-item>
  96. </el-col> -->
  97. <el-col :span="8" style="height: 58px">
  98. <el-form-item prop="files" label="附件">
  99. <fileMain v-model="form.files"></fileMain>
  100. <!-- <fileUpload-->
  101. <!-- v-model="form.files"-->
  102. <!-- module="main"-->
  103. <!-- :showLib="false"-->
  104. <!-- :limit="5"-->
  105. <!-- />-->
  106. </el-form-item>
  107. </el-col>
  108. <el-col :span="16">
  109. <el-form-item label="备注" prop="remark">
  110. <el-input
  111. placeholder="请输入"
  112. v-model="form.remark"
  113. type="textarea"
  114. maxlength="300"
  115. ></el-input>
  116. </el-form-item>
  117. </el-col>
  118. </el-row>
  119. </el-form>
  120. <headerTitle title="产品清单"></headerTitle>
  121. <inventoryTable
  122. :isCustomerMark="false"
  123. ref="inventoryTable"
  124. :isDeliveryDeadline="false"
  125. :isGuaranteePeriod="false"
  126. :isDiscount="false"
  127. pageName="businessOpportunity"
  128. :pricing-way="form.pricingWay"
  129. ></inventoryTable>
  130. <!-- <headerTitle title="竞品"></headerTitle>
  131. <businessAddTable
  132. ref="businessAddTable"
  133. @timeAll="getDetailTable"
  134. :delDetailIds="delDetailIds"
  135. ></businessAddTable>
  136. <headerTitle title="关键人信息"></headerTitle>
  137. <personnelAddTable
  138. ref="personnelAddTable"
  139. :contactId="form.contactId"
  140. @timeAll="getPersonnelAddTable"
  141. :delDetailIds="PersonnelDetailIds"
  142. ></personnelAddTable> -->
  143. <head-list ref="headRef" @changeParent="changeHead"></head-list>
  144. <parentList
  145. ref="parentRef"
  146. classType="1"
  147. @changeParent="changeParent"
  148. ></parentList>
  149. </div>
  150. </template>
  151. <script>
  152. import store from '@/store';
  153. import dictMixins from '@/mixins/dictMixins';
  154. import businessAddTable from './businessAddTable.vue';
  155. import personnelAddTable from './personnelAddTable.vue';
  156. import headList from '@/components/headList';
  157. import parentList from '@/views/bpm/handleTask/components/quotation/parentList.vue';
  158. import { getDetail } from '@/api/bpm/components/saleManage/businessOpportunity';
  159. import { copyObj } from '@/utils/util';
  160. import inventoryTable from '@/BIZComponents/inventoryTable.vue';
  161. import fileUpload from '@/components/upload/fileUpload';
  162. import fileMain from '@/components/addDoc/index.vue';
  163. import {contactDetail} from '@/api/bpm/components/saleManage/contact';
  164. export default {
  165. props: {
  166. businessId: {
  167. default: ''
  168. }
  169. },
  170. mixins: [dictMixins],
  171. components: {
  172. fileMain,
  173. fileUpload,
  174. businessAddTable,
  175. personnelAddTable,
  176. headList,
  177. parentList,
  178. inventoryTable
  179. },
  180. data() {
  181. let formDef = {
  182. id: '',
  183. contactName: '',
  184. contactId: '',
  185. budget: null,
  186. expectedClosingDate: '',
  187. sourceCode: '',
  188. sourceName: '',
  189. stageCode: '',
  190. stageName: '',
  191. winRate: null,
  192. name: '',
  193. remark: '',
  194. responsibleName: '',
  195. responsibleId: '',
  196. source: '',
  197. status: 1,
  198. files: []
  199. };
  200. return {
  201. delDetailIds: [],
  202. PersonnelDetailIds: [],
  203. pricingWayList: [
  204. { id: 1, name: '按数量计价' },
  205. { id: 2, name: '按重量计价' }
  206. ],
  207. visible: false,
  208. title: '',
  209. row: {},
  210. userInfo: {},
  211. activeName: 'base',
  212. formDef,
  213. form: copyObj(formDef),
  214. // removeBankList: [],
  215. // removeLinkList: [],
  216. tableBankData: [],
  217. tableLinkData: [],
  218. rules: {
  219. contactName: [
  220. { required: true, message: '请选择客户', trigger: 'change' }
  221. ],
  222. name: [
  223. { required: true, message: '请输入商机名称', trigger: 'blur' }
  224. ],
  225. expectedClosingDate: [
  226. { required: true, message: '请选择预计结单日期', trigger: 'blur' }
  227. ],
  228. sourceCode: [
  229. { required: true, message: '请选择商机来源名称', trigger: 'change' }
  230. ],
  231. stageCode: [
  232. { required: true, message: '请选择商机阶段名称', trigger: 'change' }
  233. ],
  234. winRate: [
  235. {
  236. required: false,
  237. trigger: 'blur',
  238. validator: (_rule, value, callback) => {
  239. console.log(_rule);
  240. const reg = /^\d+(\.\d{1,2})?$/; // 限制为最多两位小数的正数
  241. if (!value) {
  242. callback(); // 验证通过
  243. return;
  244. }
  245. if (reg.test(value) && parseFloat(value) <= 100) {
  246. callback(); // 验证通过
  247. } else {
  248. value = value && value.replace(/[^0-9.]/g, ''); // 只保留数字和小数点
  249. const [integerPart, decimalPart] = value && value.split('.');
  250. if (decimalPart && decimalPart.length > 2) {
  251. callback(new Error('小数位数不能超过两位'));
  252. } else {
  253. callback(new Error('只能输入小于100的数字'));
  254. }
  255. }
  256. }
  257. }
  258. ],
  259. budget: [
  260. {
  261. required: false,
  262. trigger: 'blur',
  263. validator: (_rule, value, callback) => {
  264. const reg = /^\d{1,13}(\.\d{1,2})?$/; // 限制为最多两位小数的13位以内的正数
  265. if (!value) {
  266. callback(); // 验证通过
  267. return;
  268. }
  269. if (reg.test(value)) {
  270. callback(); // 验证通过
  271. } else {
  272. value = value && value.replace(/[^0-9.]/g, ''); // 只保留数字和小数点
  273. const [integerPart, decimalPart] = value && value.split('.');
  274. if (decimalPart && decimalPart.length > 2) {
  275. callback(new Error('小数位数不能超过两位'));
  276. } else if (integerPart.length > 13) {
  277. callback(new Error('整数位数不能超过13位'));
  278. } else {
  279. callback(new Error('只能输入数字'));
  280. }
  281. }
  282. }
  283. }
  284. ]
  285. },
  286. // 提交状态
  287. loading: false,
  288. // 是否是修改
  289. isUpdate: false,
  290. // 组织机构树形结构数据
  291. groupTreeData: [],
  292. // 组织机构平铺数据
  293. groupData: []
  294. };
  295. },
  296. mounted() {
  297. this.$nextTick(() => {
  298. this.userInfo = store.state.user.info;
  299. });
  300. },
  301. created() {
  302. this.getDetailData(this.businessId);
  303. },
  304. methods: {
  305. //获取竞品
  306. getDetailTable(val) {
  307. this.form.competAnalysisList = val;
  308. },
  309. //获取关键人信息
  310. getPersonnelAddTable(val) {
  311. this.form.partyList = val;
  312. },
  313. //选择客户回调
  314. changeParent(obj) {
  315. this.$set(this.form, 'contactId', obj.id);
  316. this.$set(this.form, 'contactName', obj.name);
  317. this.getContactDetail(1);
  318. },
  319. //获取客户
  320. async getContactDetail(type) {
  321. const { base } = await contactDetail(this.form.contactId);
  322. this.customerMark = base.serialNo;
  323. if (type) {
  324. this.$nextTick(() => {
  325. this.$refs.inventoryTable.setCustomerMark(this.customerMark);
  326. });
  327. }
  328. },
  329. //选择负责人回调
  330. changeHead(obj) {
  331. this.$set(this.form, 'responsibleId', obj.id);
  332. this.$set(this.form, 'responsibleName', obj.name);
  333. },
  334. handParent() {
  335. let item = {
  336. id: this.form.contactId
  337. };
  338. this.$refs.parentRef.open(item);
  339. },
  340. handHead() {
  341. let item = {
  342. id: this.form.responsibleId
  343. };
  344. this.$refs.headRef.open(item);
  345. },
  346. //获取详情
  347. async getDetailData(id) {
  348. this.loading = true;
  349. const data = await getDetail(id);
  350. this.loading = false;
  351. if (data) {
  352. this.form = data;
  353. this.getContactDetail()
  354. this.$nextTick(() => {
  355. this.$refs.inventoryTable &&
  356. this.$refs.inventoryTable.putTableValue(data);
  357. });
  358. }
  359. },
  360. salesmanChange(val, info) {
  361. this.otherForm.salesmanName = info.name;
  362. },
  363. settlementModeChange(info) {
  364. this.otherForm.settlementModeName = info.dictValue;
  365. },
  366. ifChiefChange(value, idx) {
  367. if (value === 1) {
  368. this.tableLinkData.forEach((e) => (e.ifChief = 0));
  369. this.tableLinkData[idx].ifChief = 1;
  370. }
  371. },
  372. getValidate() {
  373. return Promise.all([
  374. new Promise((resolve, reject) => {
  375. this.$refs.form.validate((valid) => {
  376. if (!valid) {
  377. reject(false);
  378. } else {
  379. resolve(true);
  380. }
  381. });
  382. }),
  383. new Promise((resolve, reject) => {
  384. this.$refs.inventoryTable.validateForm((valid) => {
  385. if (!valid) {
  386. reject(false);
  387. } else {
  388. resolve(true);
  389. }
  390. });
  391. })
  392. ]);
  393. },
  394. async getTableValue() {
  395. try {
  396. await this.getValidate();
  397. // 表单验证通过,执行保存操作
  398. let _sourceName = this.getDictValue('商机来源', this.form.sourceCode),
  399. _stageName = this.getDictValue('商机阶段', this.form.stageCode);
  400. this.form = Object.assign({}, this.form, {
  401. responsibleId: this.userInfo.userId,
  402. responsibleName: this.userInfo.name,
  403. sourceName: _sourceName,
  404. stageName: _stageName,
  405. files: this.form.files || []
  406. });
  407. if (this.$refs.inventoryTable.getTableValue().length == 0) {
  408. this.$message.warning('产品清单不能为空');
  409. return;
  410. }
  411. this.form.totalPrice= this.$refs.inventoryTable.getPrice()[0]
  412. const commitData = {
  413. opportunity: this.form,
  414. competAnalysisList: this.form.competAnalysisList,
  415. partyList: this.form.partyList,
  416. productList: this.$refs.inventoryTable.getTableValue()
  417. };
  418. return commitData;
  419. } catch (error) {
  420. console.log(error);
  421. // 表单验证未通过,不执行保存操作
  422. }
  423. },
  424. cancel() {
  425. this.$nextTick(() => {
  426. this.activeName = 'base';
  427. // 关闭后,销毁所有的表单数据
  428. this.$refs['otherForm'] && this.$refs['otherForm'].resetFields();
  429. this.$refs['formRef'] && this.$refs['formRef'].resetFields();
  430. this.form = copyObj(this.formDef);
  431. this.visible = false;
  432. });
  433. }
  434. }
  435. };
  436. </script>