MonthSaleGoal.vue 45 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092
  1. <template>
  2. <div class="app-container">
  3. <el-form :model="queryParams" ref="queryForm" size="small" :inline="true" label-width="68px">
  4. <el-row :gutter="20">
  5. <el-col :span="6">
  6. <el-form-item label="编码" prop="code">
  7. <el-input
  8. v-model="queryParams.code"
  9. placeholder="请输入编码"
  10. clearable
  11. @keyup.enter.native="handleQuery"
  12. />
  13. </el-form-item>
  14. </el-col>
  15. <el-col :span="6">
  16. <el-form-item label="目标名称" prop="goalName">
  17. <el-input
  18. v-model="queryParams.goalName"
  19. placeholder="请输入目标名称"
  20. clearable
  21. @keyup.enter.native="handleQuery"
  22. />
  23. </el-form-item>
  24. </el-col>
  25. <el-col :span="8">
  26. <el-form-item label="单据日期" prop="documentDate">
  27. <el-date-picker
  28. v-model="documentDateRange"
  29. value-format="yyyy-MM-dd"
  30. @change="setBeginAndEnd"
  31. type="daterange"
  32. align="right"
  33. range-separator="至"
  34. start-placeholder="开始日期"
  35. end-placeholder="结束日期"
  36. :picker-options="pickerOptions">
  37. </el-date-picker>
  38. </el-form-item>
  39. </el-col>
  40. <el-col :span="4">
  41. <el-form-item>
  42. <el-button type="primary" icon="el-icon-search" size="mini" @click="handleQuery">搜索</el-button>
  43. <el-button icon="el-icon-refresh" size="mini" @click="resetQuery">重置</el-button>
  44. <el-tooltip class="item" effect="dark" :content="showSearch?'隐藏搜索':'显示搜索'" placement="top">
  45. <el-button type="warning" :icon="showSearch?'el-icon-caret-top':'el-icon-caret-bottom'" circle @click="showSearch = !showSearch"></el-button>
  46. </el-tooltip>
  47. </el-form-item>
  48. </el-col>
  49. </el-row>
  50. </el-form>
  51. <el-form :model="queryParams" ref="queryForm" size="small" :inline="true" v-show="showSearch" label-width="68px">
  52. <el-row :gutter="20">
  53. <el-col :span="6">
  54. <el-form-item label="年度" prop="annual">
  55. <el-date-picker
  56. v-model="queryParams.annual"
  57. type="year"
  58. value-format="yyyy"
  59. placeholder="选择年度"
  60. clearable
  61. @keyup.enter.native="handleQuery">
  62. </el-date-picker>
  63. </el-form-item>
  64. </el-col>
  65. <el-col :span="6">
  66. <el-form-item label="客户" prop="custom">
  67. <el-popover-select-v2 v-model="queryParams.custom" title="客户" valueKey="name"
  68. referName="CUSTOMER_PARAM"
  69. :dataMapping="{ customCode: 'code', custom: 'name'}"
  70. :source.sync="queryParams" placeholder="请输入客户" @keyup.enter.native="handleQuery">
  71. </el-popover-select-v2>
  72. </el-form-item>
  73. </el-col>
  74. <el-col :span="6">
  75. <el-form-item label="销售区域" prop="saleZone">
  76. <el-popover-select-v2 v-model="queryParams.saleZone" title="销售区域" valueKey="name"
  77. referName="MK_SALESAREA_PARAM"
  78. :dataMapping="{ saleZoneCode: 'code', saleZone: 'name'}"
  79. :source.sync="queryParams" placeholder="请输入销售区域" @keyup.enter.native="handleQuery">
  80. </el-popover-select-v2>
  81. </el-form-item>
  82. </el-col>
  83. <el-col :span="6">
  84. <el-form-item label="制单人" prop="creator">
  85. <el-popover-select-v2 v-model="queryParams.creator" title="制单人" valueKey="name"
  86. referName="CONTACTS_PARAM"
  87. :dataMapping="{ creatorCode: 'code', creator: 'name'}"
  88. :source.sync="queryParams" placeholder="请输入制单人" @keyup.enter.native="handleQuery">
  89. </el-popover-select-v2>
  90. </el-form-item>
  91. </el-col>
  92. </el-row>
  93. <el-row :gutter="20">
  94. <el-col :span="6">
  95. <el-form-item label="部门" prop="dept">
  96. <el-popover-select-v2 v-model="queryParams.dept" title="部门" valueKey="name"
  97. referName="DEPT_PARAM"
  98. :dataMapping="{ deptCode: 'code', dept: 'name'}"
  99. :source.sync="queryParams" placeholder="请输入部门" @keyup.enter.native="handleQuery">
  100. </el-popover-select-v2>
  101. </el-form-item>
  102. </el-col>
  103. <el-col :span="6">
  104. <el-form-item label="单据状态" prop="documentStatus">
  105. <el-select v-model="queryParams.documentStatus" placeholder="请选择">
  106. <el-option
  107. v-for="item in [{value: '未提交', label: '未提交'}, {value: '审核中', label: '审核中'}, {value: '已审核', label: '已审核'}]"
  108. :key="item.value"
  109. :label="item.label"
  110. :value="item.value">
  111. </el-option>
  112. </el-select>
  113. </el-form-item>
  114. </el-col>
  115. </el-row>
  116. </el-form>
  117. <el-row :gutter="10" class="mb8" style="float: right">
  118. <el-col :span="1.5">
  119. <el-button
  120. type="primary"
  121. plain
  122. icon="el-icon-plus"
  123. size="mini"
  124. @click="handleAdd"
  125. >新增
  126. </el-button>
  127. </el-col>
  128. <el-col :span="1.5">
  129. <el-button
  130. type="danger"
  131. plain
  132. icon="el-icon-delete"
  133. size="mini"
  134. :disabled="multiple"
  135. @click="handleDelete"
  136. >删除
  137. </el-button>
  138. </el-col>
  139. <el-col :span="1.5">
  140. <el-button
  141. type="primary"
  142. plain
  143. icon="el-icon-grape"
  144. size="mini"
  145. :disabled="multiple"
  146. >提交
  147. </el-button>
  148. </el-col>
  149. <el-col :span="1.5">
  150. <el-button
  151. type="info"
  152. plain
  153. icon="el-icon-upload2"
  154. size="mini"
  155. @click="handleImport"
  156. >导入</el-button>
  157. </el-col>
  158. <el-col :span="1.5">
  159. <el-dropdown @command="handleCommand">
  160. <el-button type="warning" plain icon="el-icon-download" size="mini">
  161. 导出<i class="el-icon-arrow-down el-icon--right"></i>
  162. </el-button>
  163. <el-dropdown-menu slot="dropdown">
  164. <el-dropdown-item command="export">导出</el-dropdown-item>
  165. <el-dropdown-item command="exportDetails">导出明细</el-dropdown-item>
  166. </el-dropdown-menu>
  167. </el-dropdown>
  168. </el-col>
  169. </el-row>
  170. <el-table v-loading="loading" :data="monthSaleGoalList" @selection-change="handleSelectionChange" v-horizontal-scroll>
  171. <el-table-column type="selection" width="55" align="center"/>
  172. <el-table-column label="编码" align="center" prop="code"/>
  173. <el-table-column label="目标名称" align="center" prop="goalName"/>
  174. <el-table-column label="单据日期" align="center" prop="documentDate" width="180">
  175. <template slot-scope="scope">
  176. <span>{{ parseTime(scope.row.documentDate, '{y}-{m}-{d}') }}</span>
  177. </template>
  178. </el-table-column>
  179. <el-table-column label="年度" align="center" prop="annual"/>
  180. <el-table-column label="客户" align="center" prop="custom"/>
  181. <el-table-column label="销售区域" align="center" prop="saleZone"/>
  182. <el-table-column label="制单人" align="center" prop="creator"/>
  183. <el-table-column label="部门" align="center" prop="dept"/>
  184. <el-table-column label="目标合计" align="center" prop="goalSum"/>
  185. <el-table-column label="备注" align="center" prop="notes"/>
  186. <el-table-column label="单据状态" align="center" prop="documentStatus"/>
  187. <el-table-column label="操作" align="center" class-name="small-padding fixed-width" width="180" fixed="right">
  188. <template slot-scope="scope">
  189. <el-button
  190. size="mini"
  191. type="text"
  192. icon="el-icon-document-copy"
  193. @click="handleCopy(scope.row.id)"
  194. >复制
  195. </el-button>
  196. <el-button
  197. size="mini"
  198. type="text"
  199. icon="el-icon-edit"
  200. @click="handleUpdate(scope.row)"
  201. >修改
  202. </el-button>
  203. <el-button
  204. size="mini"
  205. type="text"
  206. icon="el-icon-delete"
  207. @click="handleDelete(scope.row)"
  208. >删除
  209. </el-button>
  210. </template>
  211. </el-table-column>
  212. </el-table>
  213. <pagination
  214. v-show="total>0"
  215. :total="total"
  216. :page.sync="queryParams.pageNum"
  217. :limit.sync="queryParams.pageSize"
  218. @pagination="getList"
  219. />
  220. <el-drawer :title="title" :visible.sync="open" direction="rtl" :before-close="handleClose" size="100%">
  221. <el-form ref="form" :model="form" :rules="rules" label-width="120px">
  222. <el-row :gutter="20">
  223. <el-col :span="6">
  224. <el-form-item label="编码" prop="code">
  225. <el-input v-model="form.code" placeholder="编码后端自动生成" disabled/>
  226. </el-form-item>
  227. </el-col>
  228. <el-col :span="6">
  229. <el-form-item label="目标名称" prop="goalName">
  230. <el-input v-model="form.goalName" placeholder="目标名称后端自动生成" disabled/>
  231. </el-form-item>
  232. </el-col>
  233. <el-col :span="6">
  234. <el-form-item label="单据日期" prop="documentDate">
  235. <el-date-picker clearable
  236. v-model="form.documentDate"
  237. type="date"
  238. value-format="yyyy-MM-dd"
  239. placeholder="请选择单据日期">
  240. </el-date-picker>
  241. </el-form-item>
  242. </el-col>
  243. <el-col :span="6">
  244. <el-form-item label="年度" prop="annual">
  245. <el-date-picker
  246. v-model="form.annual"
  247. type="year"
  248. value-format="yyyy"
  249. placeholder="选择年度">
  250. </el-date-picker>
  251. </el-form-item>
  252. </el-col>
  253. </el-row>
  254. <el-row :gutter="20">
  255. <el-col :span="6">
  256. <el-form-item label="客户" prop="custom">
  257. <el-popover-select-v2 v-model="form.custom" title="客户" valueKey="name"
  258. referName="CUSTOMER_PARAM"
  259. :dataMapping="{ customCode: 'code', custom: 'name'}"
  260. :source.sync="form" placeholder="请输入客户">
  261. </el-popover-select-v2>
  262. </el-form-item>
  263. </el-col>
  264. <el-col :span="6">
  265. <el-form-item label="销售区域" prop="saleZone">
  266. <el-popover-select-v2 v-model="form.saleZone" title="销售区域" valueKey="name"
  267. referName="MK_SALESAREA_PARAM"
  268. :dataMapping="{ saleZoneCode: 'code', saleZone: 'name'}"
  269. :source.sync="form" placeholder="请输入销售区域">
  270. </el-popover-select-v2>
  271. </el-form-item></el-col>
  272. <el-col :span="6">
  273. <el-form-item label="制单人" prop="creator">
  274. <el-popover-select-v2 v-model="form.creator" title="制单人" valueKey="name"
  275. referName="CONTACTS_PARAM"
  276. :dataMapping="{ creatorCode: 'code', creator: 'name'}"
  277. :source.sync="form" placeholder="请输入制单人">
  278. </el-popover-select-v2>
  279. </el-form-item>
  280. </el-col>
  281. <el-col :span="6">
  282. <el-form-item label="部门" prop="dept">
  283. <el-popover-select-v2 v-model="form.dept" title="部门" valueKey="name"
  284. referName="DEPT_PARAM"
  285. :dataMapping="{ deptCode: 'code', dept: 'name'}"
  286. :source.sync="form" placeholder="请输入部门">
  287. </el-popover-select-v2>
  288. </el-form-item>
  289. </el-col>
  290. </el-row>
  291. <el-row :gutter="20">
  292. <el-col :span="6">
  293. <el-form-item label="目标合计" prop="goalSum">
  294. <el-input v-model="form.goalSum" placeholder="目标合计自动计算" disabled />
  295. </el-form-item>
  296. </el-col>
  297. <el-col :span="6">
  298. <el-form-item label="备注" prop="notes">
  299. <el-input v-model="form.notes" placeholder="请输入备注"/>
  300. </el-form-item>
  301. </el-col>
  302. <el-col :span="6">
  303. <el-form-item label="单据状态" prop="documentStatus">
  304. <el-input v-model="form.documentStatus" placeholder="未提交" disabled/>
  305. </el-form-item>
  306. </el-col>
  307. </el-row>
  308. </el-form>
  309. <div>
  310. <el-row :gutter="10" class="mb8" style="margin-left: 80%">
  311. <el-col :span="1.5">
  312. <el-button type="info" plain icon="el-icon-upload2" size="mini" @click="handleImport">导入明细</el-button>
  313. </el-col>
  314. <el-col :span="1.5">
  315. <el-button type="primary" plain icon="el-icon-plus" size="mini" @click="handleAddDetails">增行</el-button>
  316. </el-col>
  317. <el-col :span="1.5">
  318. <el-button type="primary" plain icon="el-icon-edit-outline" size="mini" @click="dialogUpdateMore.dialogFormVisible = true">批量修改</el-button>
  319. </el-col>
  320. </el-row>
  321. <el-tabs v-model="activeName">
  322. <el-tab-pane label="月销售目标明细" name="monthSaleGoalDetails">
  323. <el-table max-height="300" show-summary :summary-method="getSummaries" v-loading="loading" :data="monthSaleGoalDetailsList" @selection-change="handleSelectionChange" v-horizontal-scroll>
  324. <el-table-column label="序号" type="index" width="70" align="center" fixed />
  325. <el-table-column label="销售组织" align="center" width="180" :render-header="addRedStar">
  326. <template slot-scope="scope">
  327. <el-popover-select-v2 v-model="monthSaleGoalDetailsList[scope.$index].saleOrg" title="销售组织" valueKey="name"
  328. referName="ORG_PARAM"
  329. :dataMapping="{ saleOrgCode: 'code', saleOrg: 'name'}"
  330. :source.sync="monthSaleGoalDetailsList[scope.$index]" placeholder="请输入销售组织">
  331. </el-popover-select-v2>
  332. </template>
  333. </el-table-column>
  334. <el-table-column label="销售区域" align="center" width="180" :render-header="addRedStar">
  335. <template slot-scope="scope">
  336. <el-popover-select-v2 v-model="monthSaleGoalDetailsList[scope.$index].saleZone" title="销售区域" valueKey="name"
  337. referName="MK_SALESAREA_PARAM"
  338. :dataMapping="{ saleZoneCode: 'code', saleZone: 'name'}"
  339. :source.sync="monthSaleGoalDetailsList[scope.$index]" placeholder="请输入销售区域">
  340. </el-popover-select-v2>
  341. </template>
  342. </el-table-column>
  343. <el-table-column label="客户" align="center" width="180" :render-header="addRedStar">
  344. <template slot-scope="scope">
  345. <el-popover-select-v2 v-model="monthSaleGoalDetailsList[scope.$index].custom" title="客户" valueKey="name"
  346. referName="CUSTOMER_PARAM"
  347. :dataMapping="{ customCode: 'code', custom: 'name'}"
  348. :source.sync="monthSaleGoalDetailsList[scope.$index]" placeholder="请输入客户">
  349. </el-popover-select-v2>
  350. </template>
  351. </el-table-column>
  352. <el-table-column label="部门" align="center" width="180" :render-header="addRedStar">
  353. <template slot-scope="scope">
  354. <el-popover-select-v2 v-model="monthSaleGoalDetailsList[scope.$index].dept" title="部门" valueKey="name"
  355. referName="DEPT_PARAM"
  356. :dataMapping="{ deptCode: 'code', dept: 'name'}"
  357. :source.sync="monthSaleGoalDetailsList[scope.$index]" placeholder="请输入部门">
  358. </el-popover-select-v2>
  359. </template>
  360. </el-table-column>
  361. <el-table-column label="制单人" align="center" width="180" :render-header="addRedStar">
  362. <template slot-scope="scope">
  363. <el-popover-select-v2 v-model="monthSaleGoalDetailsList[scope.$index].creator" title="制单人" valueKey="name"
  364. referName="CONTACTS_PARAM"
  365. :dataMapping="{ creatorCode: 'code', creator: 'name'}"
  366. :source.sync="monthSaleGoalDetailsList[scope.$index]" placeholder="请输入制单人">
  367. </el-popover-select-v2>
  368. </template>
  369. </el-table-column>
  370. <el-table-column label="一级分类" align="center" width="180" :render-header="addRedStar">
  371. <template slot-scope="scope">
  372. <el-input v-model="monthSaleGoalDetailsList[scope.$index].oneLevelClassify" placeholder="请输入一级分类" disabled></el-input>
  373. </template>
  374. </el-table-column>
  375. <el-table-column label="二级分类" align="center" width="180" :render-header="addRedStar">
  376. <template slot-scope="scope">
  377. <el-input v-model="monthSaleGoalDetailsList[scope.$index].twoLevelClassify" placeholder="请输入二级分类" disabled></el-input>
  378. </template>
  379. </el-table-column>
  380. <el-table-column label="物料" align="center" width="180" :render-header="addRedStar">
  381. <template slot-scope="scope">
  382. <el-popover-select-v2 v-model="monthSaleGoalDetailsList[scope.$index].material" title="物料" valueKey="name"
  383. referName="MATERIAL_PARAM"
  384. @change="setClassify(scope.row.oneLevelClassify, scope.row.twoLevelClassify, monthSaleGoalDetailsList[scope.$index])"
  385. :dataMapping="{ materialCode: 'code', material: 'name', oneLevelClassify: 'oneClass', twoLevelClassify: 'twoClass'}"
  386. :source.sync="monthSaleGoalDetailsList[scope.$index]" placeholder="请输入物料">
  387. </el-popover-select-v2>
  388. </template>
  389. </el-table-column>
  390. <el-table-column label="科室" align="center" width="180">
  391. <template slot-scope="scope">
  392. <el-input v-model="monthSaleGoalDetailsList[scope.$index].department" placeholder="请输入科室"></el-input>
  393. </template>
  394. </el-table-column>
  395. <el-table-column label="数量" align="center" prop="num" width="220">
  396. <template slot-scope="scope">
  397. <el-input-number v-model="monthSaleGoalDetailsList[scope.$index].num" :min="0"></el-input-number>
  398. </template>
  399. </el-table-column>
  400. <el-table-column label="月份" align="center" width="250" :render-header="addRedStar">
  401. <template slot-scope="scope">
  402. <el-date-picker v-model="monthSaleGoalDetailsList[scope.$index].monthly" value-format="yyyy-MM" type="month" placeholder="选择月份">
  403. </el-date-picker>
  404. </template>
  405. </el-table-column>
  406. <el-table-column label="目标值" align="center" prop="goalValue" width="220" :render-header="addRedStar">
  407. <template slot-scope="scope">
  408. <el-input-number @change="computeTotal" v-model="monthSaleGoalDetailsList[scope.$index].goalValue" :precision="2" :step="1" :min="0"></el-input-number>
  409. </template>
  410. </el-table-column>
  411. <el-table-column label="操作" align="center" class-name="small-padding fixed-width" width="120px" fixed="right">
  412. <template slot-scope="scope">
  413. <el-button
  414. size="mini"
  415. type="text"
  416. icon="el-icon-delete"
  417. @click="handleDeleteDetails(scope.$index, scope.row)"
  418. >删除</el-button>
  419. <el-button
  420. size="mini"
  421. type="text"
  422. icon="el-icon-delete"
  423. @click="handleCopyDetails(scope.row)"
  424. >复制</el-button>
  425. </template>
  426. </el-table-column>
  427. </el-table>
  428. </el-tab-pane>
  429. </el-tabs>
  430. <div slot="footer" class="dialog-footer" style="margin-left: 88%; margin-top: 1%">
  431. <el-button type="primary" @click="submitForm" size="medium">确 定</el-button>
  432. <el-button @click="cancel" size="medium">返 回</el-button>
  433. </div>
  434. </div>
  435. </el-drawer>
  436. <!-- 用户导入对话框 -->
  437. <el-dialog :title="upload.title" :visible.sync="upload.open" width="400px" append-to-body>
  438. <el-upload
  439. ref="upload"
  440. :limit="1"
  441. accept=".xlsx, .xls"
  442. :headers="upload.headers"
  443. :action="upload.url + '?updateSupport=' + upload.updateSupport"
  444. :disabled="upload.isUploading"
  445. :on-progress="handleFileUploadProgress"
  446. :on-success="handleFileSuccess"
  447. :auto-upload="false"
  448. drag
  449. >
  450. <i class="el-icon-upload"></i>
  451. <div class="el-upload__text">将文件拖到此处,或<em>点击上传</em></div>
  452. <div class="el-upload__tip text-center" slot="tip">
  453. <div class="el-upload__tip" slot="tip">
  454. <el-checkbox v-model="upload.updateSupport" /> 是否更新已经存在的用户数据
  455. </div>
  456. <span>仅允许导入xls、xlsx格式文件。</span>
  457. <el-link type="primary" :underline="false" style="font-size:12px;vertical-align: baseline;" @click="importTemplate">下载模板</el-link>
  458. </div>
  459. </el-upload>
  460. <div slot="footer">
  461. <el-button type="primary" @click="submitFileForm">确 定</el-button>
  462. <el-button @click="upload.open = false">取 消</el-button>
  463. </div>
  464. </el-dialog>
  465. <!-- 批量修改对话框-->
  466. <el-dialog title="批量修改" width="30%" :visible.sync="dialogUpdateMore.dialogFormVisible" append-to-body @closed="resetDialogUpdateMore">
  467. <el-form>
  468. <el-form-item label="修改项" label-width="100px">
  469. <el-select @change="changeDialogData" v-model="dialogUpdateMore.updateName" placeholder="请选择需要批量修改的字段">
  470. <el-option v-for="item in dialogUpdateMore.optionList" :key="item.value" :label="item.label" :value="item.value"></el-option>
  471. </el-select>
  472. </el-form-item>
  473. <el-form-item label="修改值" label-width="100px">
  474. <el-popover-select-v2 v-if="dialogUpdateMore.updateName === 1"
  475. v-model="dialogUpdateMore.updateData" title="销售组织" valueKey="name"
  476. referName="ORG_PARAM"
  477. :dataMapping="{updateData: 'name'}"
  478. :source.sync="dialogUpdateMore" placeholder="请输入销售组织">
  479. </el-popover-select-v2>
  480. <el-popover-select-v2 v-else-if="dialogUpdateMore.updateName === 2"
  481. v-model="dialogUpdateMore.updateData" title="销售区域" valueKey="name"
  482. referName="MK_SALESAREA_PARAM"
  483. :dataMapping="{updateData: 'name'}"
  484. :source.sync="dialogUpdateMore" placeholder="请输入销售区域">
  485. </el-popover-select-v2>
  486. <el-popover-select-v2 v-else-if="dialogUpdateMore.updateName === 3"
  487. v-model="dialogUpdateMore.updateData" title="客户" valueKey="name"
  488. referName="CUSTOMER_PARAM"
  489. :dataMapping="{updateData: 'name'}"
  490. :source.sync="dialogUpdateMore" placeholder="请输入客户">
  491. </el-popover-select-v2>
  492. <el-popover-select-v2 v-else-if="dialogUpdateMore.updateName === 4"
  493. v-model="dialogUpdateMore.updateData" title="部门" valueKey="name"
  494. referName="DEPT_PARAM"
  495. :dataMapping="{updateData: 'name'}"
  496. :source.sync="dialogUpdateMore" placeholder="请输入部门">
  497. </el-popover-select-v2>
  498. <el-popover-select-v2 v-else-if="dialogUpdateMore.updateName === 5"
  499. v-model="dialogUpdateMore.updateData" title="制单人" valueKey="name"
  500. referName="CONTACTS_PARAM"
  501. :dataMapping="{updateData: 'name'}"
  502. :source.sync="dialogUpdateMore" placeholder="请输入制单人">
  503. </el-popover-select-v2>
  504. <el-input v-else-if="dialogUpdateMore.updateName === 6"
  505. v-model="dialogUpdateMore.updateData" placeholder="请输入科室"
  506. ></el-input>
  507. <el-date-picker v-else-if="dialogUpdateMore.updateName === 7"
  508. v-model="dialogUpdateMore.updateData" value-format="yyyy-MM"
  509. type="month"
  510. placeholder="选择月">
  511. </el-date-picker>
  512. <el-input v-else placeholder="请输入修改项" disabled></el-input>
  513. </el-form-item>
  514. </el-form>
  515. <div slot="footer" class="dialog-footer">
  516. <el-button type="primary" @click="dialogUpdateMoreSave">确 定</el-button>
  517. </div>
  518. </el-dialog>
  519. </div>
  520. </template>
  521. <script>
  522. import {
  523. listMonthSaleGoal,
  524. getMonthSaleGoal,
  525. delMonthSaleGoal,
  526. addMonthSaleGoal,
  527. updateMonthSaleGoal
  528. } from "@/api/business/spd/goal_management/monthSaleGoal";
  529. import {
  530. getMonthSaleGoalDetails,
  531. delMonthSaleGoalDetails
  532. } from "@/api/business/spd/goal_management/monthSaleGoalDetails"
  533. import { getToken } from "@/utils/auth";
  534. import { getSummary } from "../../../../api/business/spd/goal_management/commonWays";
  535. // 树形参照
  536. import TreeRefers from '@/components/Refers/treeRefer.vue'
  537. import ElPopoverSelectV2 from "@/components/popover-select-v2"
  538. export default {
  539. name: "MonthSaleGoal",
  540. components: {
  541. TreeRefers, ElPopoverSelectV2
  542. },
  543. data() {
  544. return {
  545. // 遮罩层
  546. loading: true,
  547. // 选中数组
  548. ids: [],
  549. // 非单个禁用
  550. single: true,
  551. // 非多个禁用
  552. multiple: true,
  553. // 显示搜索条件
  554. showSearch: false,
  555. // 总条数
  556. total: 0,
  557. // 月销售目标填报表格数据
  558. monthSaleGoalList: [],
  559. monthSaleGoalDetailsList: [],
  560. // 弹出层标题
  561. title: "",
  562. // 是否显示弹出层
  563. open: false,
  564. // 查询参数
  565. queryParams: {
  566. pageNum: 1,
  567. pageSize: 10,
  568. code: null,
  569. goalName: null,
  570. documentDate: null,
  571. annual: null,
  572. customCode: null,
  573. custom: null,
  574. saleZoneCode: null,
  575. saleZone: null,
  576. creatorCode: null,
  577. creator: null,
  578. deptCode: null,
  579. dept: null,
  580. goalSum: null,
  581. notes: null,
  582. documentStatus: null,
  583. params: { beginTime: null, endTime: null }
  584. },
  585. documentDateRange: null,
  586. pickerOptions: {
  587. shortcuts: [{
  588. text: '最近一周',
  589. onClick(picker) {
  590. const end = new Date();
  591. const start = new Date();
  592. start.setTime(start.getTime() - 3600 * 1000 * 24 * 7);
  593. picker.$emit('pick', [start, end]);
  594. }
  595. }, {
  596. text: '最近一个月',
  597. onClick(picker) {
  598. const end = new Date();
  599. const start = new Date();
  600. start.setTime(start.getTime() - 3600 * 1000 * 24 * 30);
  601. picker.$emit('pick', [start, end]);
  602. }
  603. }, {
  604. text: '最近三个月',
  605. onClick(picker) {
  606. const end = new Date();
  607. const start = new Date();
  608. start.setTime(start.getTime() - 3600 * 1000 * 24 * 90);
  609. picker.$emit('pick', [start, end]);
  610. }
  611. }]
  612. },
  613. // 表单参数
  614. form: {
  615. id: null,
  616. code: null,
  617. goalName: null,
  618. documentDate: null,
  619. annual: null,
  620. customCode: null,
  621. custom: null,
  622. saleZoneCode: null,
  623. saleZone: null,
  624. creatorCode: null,
  625. creator: null,
  626. deptCode: null,
  627. dept: null,
  628. goalSum: null,
  629. notes: null,
  630. documentStatus: null,
  631. monthDetailsList: null
  632. },
  633. formDetails: {
  634. id: null,
  635. code: null,
  636. saleOrg: null,
  637. saleZone: null,
  638. custom: null,
  639. dept: null,
  640. creator: null,
  641. oneLevelClassifyCode: null,
  642. oneLevelClassify: null,
  643. twoLevelClassifyCode: null,
  644. twoLevelClassify: null,
  645. materialCode: null,
  646. material: null,
  647. department: null,
  648. num: null,
  649. monthly: null,
  650. goalValue: null
  651. },
  652. // 表单校验
  653. rules: {
  654. documentDate: [{ required: true, message: '单据日期不能为空', trigger: 'blur' }],
  655. annual: [{ required: true, message: '年度不能为空', trigger: 'blur' }],
  656. custom: [{ required: true, message: '客户不能为空', trigger: 'blur' }],
  657. saleZone: [{ required: true, message: '销售区域不能为空', trigger: 'blur' }],
  658. creator: [{ required: true, message: '制单人不能为空', trigger: 'blur' }],
  659. dept: [{ required: true, message: '部门不能为空', trigger: 'blur' }]
  660. },
  661. // 子表导航名
  662. activeName: 'monthSaleGoalDetails',
  663. // 用户导入参数
  664. upload: {
  665. // 是否显示弹出层(用户导入)
  666. open: false,
  667. // 弹出层标题(用户导入)
  668. title: "",
  669. // 是否禁用上传
  670. isUploading: false,
  671. // 是否更新已经存在的用户数据
  672. updateSupport: 0,
  673. // 设置上传的请求头部
  674. headers: { Authorization: "Bearer " + getToken() },
  675. // 上传的地址
  676. url: process.env.VUE_APP_BASE_API + "/goal_management/monthSaleGoal/importData"
  677. },
  678. dialogUpdateMore: {
  679. updateName: null,
  680. dialogFormVisible: false,
  681. updateData: null,
  682. optionList:[{value: 1, label: '销售组织'}, {value: 2, label: '销售区域'}, {value: 3, label: '客户'}, {value: 4, label: '部门'}, {value: 5, label: '制单人'}, {value: 6, label: '科室'}, {value: 7, label: '月份'}]
  683. }
  684. };
  685. },
  686. created() {
  687. this.getList();
  688. },
  689. methods: {
  690. /** 查询月销售目标填报列表 */
  691. getList() {
  692. this.loading = true;
  693. listMonthSaleGoal(this.queryParams).then(response => {
  694. this.monthSaleGoalList = response.rows;
  695. this.total = response.total;
  696. this.loading = false;
  697. });
  698. },
  699. getListDetails() {
  700. this.loading = true
  701. getMonthSaleGoalDetails(this.form.id).then(response => {
  702. this.monthSaleGoalDetailsList = response.data
  703. this.computeTotal()
  704. this.form.monthDetailsList = this.monthSaleGoalDetailsList
  705. updateMonthSaleGoal(this.form).then(response => {
  706. })
  707. this.loading = false
  708. })
  709. },
  710. // 取消按钮
  711. cancel() {
  712. this.open = false;
  713. this.reset();
  714. },
  715. // 表单重置
  716. reset() {
  717. this.form = {
  718. id: null,
  719. code: null,
  720. goalName: null,
  721. documentDate: null,
  722. annual: null,
  723. customCode: null,
  724. custom: null,
  725. saleZoneCode: null,
  726. saleZone: null,
  727. creatorCode: null,
  728. creator: null,
  729. deptCode: null,
  730. dept: null,
  731. goalSum: null,
  732. notes: null,
  733. documentStatus: null,
  734. monthDetailsList: null
  735. };
  736. this.resetForm("form");
  737. },
  738. /** 搜索按钮操作 */
  739. handleQuery() {
  740. this.queryParams.pageNum = 1;
  741. this.getList();
  742. },
  743. /** 重置按钮操作 */
  744. resetQuery() {
  745. this.queryParams = {
  746. pageNum: 1,
  747. pageSize: 10,
  748. code: null,
  749. goalName: null,
  750. documentDate: null,
  751. annual: null,
  752. customCode: null,
  753. custom: null,
  754. saleZoneCode: null,
  755. saleZone: null,
  756. creatorCode: null,
  757. creator: null,
  758. deptCode: null,
  759. dept: null,
  760. goalSum: null,
  761. notes: null,
  762. documentStatus: null,
  763. params: { beginTime: null, endTime: null }
  764. }
  765. this.documentDateRange = null
  766. this.resetForm("queryForm");
  767. this.handleQuery();
  768. },
  769. // 多选框选中数据
  770. handleSelectionChange(selection) {
  771. this.ids = selection.map(item => item.id)
  772. this.single = selection.length !== 1
  773. this.multiple = !selection.length
  774. },
  775. /** 新增按钮操作 */
  776. handleAdd() {
  777. this.reset();
  778. this.monthSaleGoalDetailsList = []
  779. this.open = true;
  780. this.title = "添加--月销售目标填报";
  781. this.form.documentDate = new Date().getFullYear().toString() + '-' + (new Date().getMonth() + 1).toString().padStart(2, '0') + '-' + new Date().getDate().toString().padStart(2, '0')
  782. this.form.annual = new Date().getFullYear().toString()
  783. this.form.creator = this.$store.state.user.nickName
  784. this.form.dept = this.$store.state.user.deptName
  785. },
  786. handleAddDetails() {
  787. let list = {
  788. id: null,
  789. code: null,
  790. saleOrg: this.$store.state.user.orgName,
  791. saleZone: this.form.saleZone,
  792. custom: this.form.custom,
  793. dept: this.form.dept,
  794. creator: this.form.creator,
  795. oneLevelClassifyCode: null,
  796. oneLevelClassify: null,
  797. twoLevelClassifyCode: null,
  798. twoLevelClassify: null,
  799. materialCode: null,
  800. material: null,
  801. department: null,
  802. num: null,
  803. monthly: new Date().getFullYear().toString() + '-' + (new Date().getMonth() + 1).toString().padStart(2, '0'),
  804. goalValue: 0
  805. }
  806. this.monthSaleGoalDetailsList.push(list)
  807. this.computeTotal()
  808. },
  809. /** 修改按钮操作 */
  810. handleUpdate(row) {
  811. this.reset();
  812. const id = row.id || this.ids
  813. getMonthSaleGoal(id).then(response => {
  814. this.form = response.data;
  815. this.form.goalSum = this.form.goalSum.toFixed(2)
  816. this.monthSaleGoalDetailsList = JSON.parse(JSON.stringify(this.form.monthDetailsList))
  817. this.open = true;
  818. this.title = "修改--月销售目标填报";
  819. });
  820. },
  821. // 复制按钮
  822. handleCopy(id) {
  823. this.reset();
  824. getMonthSaleGoal(id).then(response => {
  825. this.form = response.data;
  826. this.form.id = null
  827. this.form.code = null
  828. this.form.goalSum = this.form.goalSum.toFixed(2)
  829. this.form.documentDate = new Date().getFullYear().toString() + '-' + (new Date().getMonth() + 1).toString().padStart(2, '0') + '-' + new Date().getDate().toString().padStart(2, '0')
  830. this.form.annual = new Date().getFullYear().toString()
  831. this.monthSaleGoalDetailsList = JSON.parse(JSON.stringify(this.form.monthDetailsList))
  832. for (const element of this.monthSaleGoalDetailsList) {
  833. element.id = null
  834. element.code = null
  835. }
  836. this.open = true;
  837. this.title = "新增--月销售目标填报";
  838. })
  839. },
  840. /** 提交按钮 */
  841. submitForm() {
  842. if (!this.justiceDetailsList()) {
  843. return this.$message.error('子表有必填字段未填,或者目标值为0')
  844. }
  845. this.$refs["form"].validate(valid => {
  846. if (valid) {
  847. if (this.form.id != null) {
  848. this.form.monthDetailsList = JSON.parse(JSON.stringify(this.monthSaleGoalDetailsList))
  849. updateMonthSaleGoal(this.form).then(response => {
  850. this.$modal.msgSuccess("修改成功");
  851. this.open = false;
  852. this.getList();
  853. });
  854. } else {
  855. this.form.documentStatus = '未提交'
  856. this.form.monthDetailsList = JSON.parse(JSON.stringify(this.monthSaleGoalDetailsList))
  857. console.log(this.form);
  858. addMonthSaleGoal(this.form).then(response => {
  859. this.$modal.msgSuccess("新增成功");
  860. this.open = false;
  861. this.getList();
  862. });
  863. }
  864. }
  865. });
  866. },
  867. /** 删除按钮操作 */
  868. handleDelete(row) {
  869. const ids = row.id || this.ids;
  870. this.$modal.confirm('是否确认删除月销售目标填报单据编号为"' + row.code + '"的数据项?').then(function () {
  871. return delMonthSaleGoal(ids);
  872. }).then(() => {
  873. this.getList();
  874. this.$modal.msgSuccess("删除成功");
  875. }).catch(() => {
  876. });
  877. },
  878. handleDeleteDetails(index, row) {
  879. if (this.form.id === null) {
  880. this.monthSaleGoalDetailsList.splice(index, 1)
  881. this.computeTotal()
  882. } else {
  883. if (row.id !== null) {
  884. this.$modal.confirm('是否确认删除月销售目标填报明细序号为"' + (index + 1) + '"的数据项?').then(function () {
  885. return delMonthSaleGoalDetails(row.id);
  886. }).then(() => {
  887. this.getListDetails();
  888. this.$modal.msgSuccess("删除成功");
  889. }).catch(() => {
  890. });
  891. } else {
  892. this.monthSaleGoalDetailsList.splice(index, 1)
  893. this.$message.success('删除成功')
  894. this.computeTotal()
  895. }
  896. }
  897. },
  898. /** 导出按钮操作 */
  899. handleExport() {
  900. this.download('goal_management/monthSaleGoal/export', {
  901. ...this.queryParams
  902. }, `monthSaleGoal_${new Date().getTime()}.xlsx`)
  903. },
  904. handleExportDetails() {
  905. this.download('goal_management/monthSaleGoalDetails/export', {
  906. ...this.queryParams
  907. }, `monthSaleGoalDetails_${new Date().getTime()}.xlsx`)
  908. },
  909. handleClose(done) {
  910. this.$confirm('确认关闭?')
  911. .then(_ => {
  912. done();
  913. })
  914. .catch(_ => {
  915. });
  916. },
  917. // 复制明细
  918. handleCopyDetails(row) {
  919. let list = {
  920. id: null,
  921. code: row.code,
  922. saleOrg: row.saleOrg,
  923. saleZone: row.saleZone,
  924. custom: row.custom,
  925. dept: row.dept,
  926. creator: row.creator,
  927. oneLevelClassifyCode: row.oneLevelClassifyCode,
  928. oneLevelClassify: row.oneLevelClassify,
  929. twoLevelClassifyCode: row.twoLevelClassifyCode,
  930. twoLevelClassify: row.twoLevelClassify,
  931. materialCode: row.materialCode,
  932. material: row.material,
  933. department: row.department,
  934. num: row.num,
  935. monthly: row.monthly,
  936. goalValue: row.goalValue
  937. }
  938. this.monthSaleGoalDetailsList.push(list)
  939. this.computeTotal()
  940. },
  941. // 计算主表合计
  942. computeTotal() {
  943. let list = this.monthSaleGoalDetailsList
  944. let sum = 0
  945. for (const listElement of list) {
  946. sum = (sum * 1000000 + listElement.goalValue * 1000000) / 1000000
  947. }
  948. this.form.goalSum = sum.toFixed(2)
  949. },
  950. getSummaries(param) { return getSummary(param) },
  951. // 子表table加必填标志
  952. addRedStar(h, { column }) {
  953. return [
  954. h('span', { style: 'color: #F56C6C' }, '*'),
  955. h('span', '' + column.label)
  956. ]
  957. },
  958. justiceDetailsList() {
  959. const arr = JSON.parse(JSON.stringify(this.monthSaleGoalDetailsList))
  960. for (const element of arr) {
  961. const flag1 = (element.saleOrg !== null) && (element.saleZone !== null) && (element.custom !== null) && (element.dept !== null) && (element.creator !== null) && (element.materialCode !== null) && (element.material !== null) && (element.monthly !== null) && (element.goalValue !== 0)
  962. const flag2 = (element.saleOrg !== undefined) && (element.saleZone !== undefined) && (element.custom !== undefined) && (element.dept !== undefined) && (element.creator !== undefined) && (element.materialCode !== undefined) && (element.material !== undefined) && (element.monthly !== undefined)
  963. if (flag1 && flag2) {
  964. return true
  965. }
  966. }
  967. return false
  968. },
  969. setBeginAndEnd() {
  970. let array = this.documentDateRange
  971. if (this.documentDateRange !== null) {
  972. this.queryParams.params.beginTime = array[0]
  973. this.queryParams.params.endTime = array[1]
  974. } else {
  975. this.queryParams.params.beginTime = null
  976. this.queryParams.params.endTime = null
  977. }
  978. },
  979. handleCommand(command) {
  980. if (command === 'export') {
  981. console.log('导出主表');
  982. this.handleExport()
  983. } else if (command === 'exportDetails') {
  984. console.log('导出明细');
  985. this.handleExportDetails()
  986. }
  987. },
  988. /** 导入按钮操作 */
  989. handleImport() {
  990. this.upload.open = true;
  991. if (this.open) {
  992. this.upload.title = "月销售目标填报明细导入"
  993. this.upload.url = process.env.VUE_APP_BASE_API + "goal_management/monthSaleGoalDetails/importData/" + this.form.id
  994. } else {
  995. this.upload.title = "月销售目标填报导入";
  996. this.upload.url = process.env.VUE_APP_BASE_API + "goal_management/monthSaleGoal/importData"
  997. }
  998. },
  999. /** 下载模板操作 */
  1000. importTemplate() {
  1001. if (this.open) {
  1002. this.download('goal_management/monthSaleGoalDetails/importTemplate', {
  1003. }, `monthSaleGoalDetails_${new Date().getTime()}.xlsx`)
  1004. } else {
  1005. this.download('goal_management/monthSaleGoal/importTemplate', {
  1006. }, `monthSaleGoal_${new Date().getTime()}.xlsx`)
  1007. }
  1008. },
  1009. // 文件上传中处理
  1010. handleFileUploadProgress(event, file, fileList) {
  1011. this.upload.isUploading = true;
  1012. },
  1013. // 文件上传成功处理
  1014. handleFileSuccess(response, file, fileList) {
  1015. this.upload.open = false;
  1016. this.upload.isUploading = false;
  1017. this.$refs.upload.clearFiles();
  1018. this.$alert("<div style='overflow: auto;overflow-x: hidden;max-height: 70vh;padding: 10px 20px 0;'>" + response.msg + "</div>", "导入结果", { dangerouslyUseHTMLString: true });
  1019. if (this.open) {
  1020. let array = response.data
  1021. for (const arrayElement of array) {
  1022. arrayElement.id = null
  1023. this.monthSaleGoalDetailsList.push(arrayElement)
  1024. }
  1025. this.computeTotal()
  1026. } else {
  1027. this.getList();
  1028. }
  1029. },
  1030. // 提交上传文件
  1031. submitFileForm() {
  1032. this.$refs.upload.submit();
  1033. },
  1034. // 选择物料后,给一级分类和二级分类复赋值
  1035. setClassify(one, two, obj) {
  1036. const oneArray = one.split("&")
  1037. const twoArray = two.split("&")
  1038. obj.oneLevelClassifyCode = oneArray[1]
  1039. obj.oneLevelClassify = oneArray[0]
  1040. obj.twoLevelClassifyCode = twoArray[1]
  1041. obj.twoLevelClassify = twoArray[0]
  1042. },
  1043. // 批量修改对话框的方法
  1044. dialogUpdateMoreSave() {
  1045. console.log(this.dialogUpdateMore);
  1046. if (this.dialogUpdateMore.updateData === null) {
  1047. return this.$message.error('请输入修改值')
  1048. }
  1049. const array = JSON.parse(JSON.stringify(this.monthSaleGoalDetailsList))
  1050. if (array.length !== 0) {
  1051. let condition = this.dialogUpdateMore.updateName
  1052. for (const element of array) {
  1053. if (condition === 1) {
  1054. element.saleOrg = JSON.parse(JSON.stringify(this.dialogUpdateMore.updateData))
  1055. } else if (condition === 2) {
  1056. element.saleZone = JSON.parse(JSON.stringify(this.dialogUpdateMore.updateData))
  1057. } else if (condition === 3) {
  1058. element.custom = JSON.parse(JSON.stringify(this.dialogUpdateMore.updateData))
  1059. } else if (condition === 4) {
  1060. element.dept = JSON.parse(JSON.stringify(this.dialogUpdateMore.updateData))
  1061. } else if (condition === 5) {
  1062. element.creator = JSON.parse(JSON.stringify(this.dialogUpdateMore.updateData))
  1063. } else if (condition === 6) {
  1064. element.department = JSON.parse(JSON.stringify(this.dialogUpdateMore.updateData))
  1065. } else if (condition === 7) {
  1066. element.monthly = JSON.parse(JSON.stringify(this.dialogUpdateMore.updateData))
  1067. }
  1068. }
  1069. }
  1070. this.monthSaleGoalDetailsList = array
  1071. this.dialogUpdateMore.dialogFormVisible = !this.dialogUpdateMore.dialogFormVisible
  1072. this.dialogUpdateMore.updateData = null
  1073. this.dialogUpdateMore.updateName = null
  1074. },
  1075. changeDialogData() {
  1076. this.dialogUpdateMore.updateData = null
  1077. },
  1078. resetDialogUpdateMore() {
  1079. this.dialogUpdateMore.updateName = null
  1080. this.dialogUpdateMore.updateData = null
  1081. }
  1082. }
  1083. }
  1084. </script>