NewPriceTable.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513
  1. <template>
  2. <div
  3. class="wrap"
  4. :style="{ marginBottom: marginBottom + 'px' }">
  5. <p
  6. class="tb-title"
  7. v-if="tableType != 'Addon'"
  8. v-html="comTitle"></p>
  9. <div :class="getTableContainerClass">
  10. <el-table
  11. :show-header="tableType != 'Addon'"
  12. ref="elTable"
  13. :data="comTableData"
  14. style="width: 100%"
  15. :header-cell-style="{
  16. backgroundColor: '#fff',
  17. fontWeight: 'normal',
  18. fontSize: '14px',
  19. fontFamily: 'Proxima Nova',
  20. color: '#000',
  21. '--comDisplayCss': comDisplayCss,
  22. }"
  23. @selection-change="handleSelectionChange"
  24. @select="dialogCheck"
  25. row-key="id"
  26. :span-method="arraySpanMethod"
  27. :row-class-name="tableRowClassName">
  28. <el-table-column
  29. v-if="tableType != 'Comparison'"
  30. type="selection"
  31. width="50"
  32. align="center"
  33. :reserve-selection="true"
  34. :selectable="handleSelectable">
  35. </el-table-column>
  36. <template v-for="item in tableColumns">
  37. <el-table-column
  38. :align="item.align ? item.align : 'left'"
  39. v-if="item.type"
  40. :key="item.prop"
  41. :prop="item.prop"
  42. :label="item.label"
  43. :formatter="item.formatter"
  44. :width="item.width"
  45. :sortable="item.sortable">
  46. <template #default="{ row }">
  47. <template v-if="item.type == 'radioInput'">
  48. <div class="flex between">
  49. <span>{{ row.decName }}</span>
  50. <el-input
  51. min="0" :max="row.max_num"
  52. v-model="row.num"
  53. type="number"
  54. class="edit-input"
  55. size="small" @input="getNumber($event, row, row.max_num)"/>
  56. </div>
  57. </template>
  58. <div v-else>
  59. <template v-if="row.filter_boolean">
  60. <div class="triangle"></div>
  61. <div class="text">Eco</div>
  62. </template>
  63. {{ row[item.prop] }}
  64. <el-popover
  65. v-if="tableType == 'Undecorated'"
  66. placement="top-start"
  67. trigger="hover">
  68. <i class="el-icon-warning" slot="reference"></i>
  69. <p class="popover-text">{{ comTitle }}</p>
  70. </el-popover>
  71. </div>
  72. </template>
  73. </el-table-column>
  74. <el-table-column
  75. :align="item.align ? item.align : 'center'"
  76. v-if="!item.type"
  77. :key="item.prop"
  78. :prop="item.prop"
  79. :label="item.label"
  80. :formatter="item.formatter"
  81. :sortable="item.sortable"
  82. show-overflow-tooltip
  83. :width="(798 - tableColumns[0].width - tableColumns[0].extraWidth) /
  84. (tableColumns.length - 1)">
  85. <template #default="{ row, $index }">
  86. <div v-if="item.prop === 'website_setup'">
  87. <!-- 打印表用supplier -->
  88. <div v-if="$index === 0 && tableType == 'Decoration'">
  89. <span v-if="row['supplier_setup_id'] < 5 || row['supplier_setup_id'] == 7">$ {{ row['supplier_setup']? parseFloat(row['supplier_setup']).toFixed(2) : "0.00" }}</span>
  90. <span>{{
  91. $_setupPriceFilter(row['supplier_setup_id'])
  92. }}</span>
  93. </div>
  94. <div v-else>
  95. <span v-if="row['website_setup_id'] < 5 || row['website_setup_id'] == 7">$ {{ row['website_setup']? parseFloat(row['website_setup']).toFixed(2) : "0.00" }}</span>
  96. <span>{{
  97. $_setupPriceFilter(row['website_setup_id'])
  98. }}</span>
  99. </div>
  100. </div>
  101. <div
  102. v-else-if="
  103. judgeProp(row, item.prop, $index) == '111' ||
  104. judgeProp(row, item.prop, $index) == '111.00'
  105. ">
  106. -
  107. </div>
  108. <div
  109. v-else-if="
  110. judgeProp(row, item.prop, $index) == '999' ||
  111. judgeProp(row, item.prop, $index) === '999.00'
  112. ">
  113. POA
  114. </div>
  115. <div v-else-if="judgeProp(row, item.prop, $index) === ''">0</div>
  116. <div v-else>
  117. <span
  118. v-if="
  119. isNumber(judgeProp(row, item.prop, $index)) &&
  120. !item.isFirstColumn
  121. "
  122. >$</span
  123. >
  124. <span>{{ judgeProp(row, item.prop, $index) }}</span>
  125. </div>
  126. </template>
  127. </el-table-column>
  128. </template>
  129. </el-table>
  130. </div>
  131. </div>
  132. </template>
  133. <script>
  134. import { times as npTimes,plus as npPlus, minus as npMinus } from 'number-precision';
  135. export default {
  136. filters: {
  137. // Decoration Table:打印价格+附加价格
  138. multiplyPrice(a1, row, prop) {
  139. const rep = /^[0-9]+.?[0-9]*$/
  140. if (!rep.test(a1)) {
  141. return a1
  142. } else {
  143. return npTimes(a1, row.num)
  144. }
  145. },
  146. // Undecorated Table:基础价格+Decoration所选价格(包含附加价格)
  147. plusPrice(a1, a2) {
  148. return a2 ? npPlus(a1, a2) : a1
  149. },
  150. // Decoration Table:计算差价
  151. differencePrice(a1, row, prop, selectRow) {
  152. const rep = /^[0-9]+.?[0-9]*$/
  153. if (rep.test(a1)) {
  154. const temp = prop.split('_')
  155. const supplier = +row[`supplier_${temp[1]}`]
  156. const suppliers = npTimes(row.num - 1, supplier)
  157. const noIncludePrice = npPlus(a1, suppliers)
  158. const selSupplier = +selectRow[`supplier_${temp[1]}`]
  159. const selSuppliers = npTimes(selectRow.num - 1, selSupplier)
  160. const isIncludePrice = npPlus(+selectRow[prop], selSuppliers)
  161. return noIncludePrice > isIncludePrice
  162. ? `+${npMinus(noIncludePrice, isIncludePrice)}`
  163. : npMinus(noIncludePrice, isIncludePrice)
  164. } else {
  165. return a1
  166. }
  167. },
  168. },
  169. props: {
  170. hasDecAdd: {
  171. type: Number,
  172. default: 0,
  173. },
  174. tableData: {
  175. type: Array,
  176. default: () => {
  177. return []
  178. },
  179. },
  180. tableColumns: {
  181. type: Array,
  182. default: () => {
  183. return []
  184. },
  185. },
  186. operateWith: {
  187. type: String,
  188. default: '100%',
  189. },
  190. isCheckBox: {
  191. type: Boolean,
  192. default: false,
  193. },
  194. tableType: {
  195. type: String,
  196. default: 'Undecorated',
  197. },
  198. decorationObj: {
  199. type: Object,
  200. default: () => {
  201. return {}
  202. },
  203. },
  204. selectId: {
  205. type: Number,
  206. default: null
  207. },
  208. selectRow: {
  209. type: Object,
  210. default: () => {
  211. return {}
  212. },
  213. },
  214. multipleSelection: {
  215. type: Array,
  216. default: () => {
  217. return []
  218. },
  219. },
  220. selectDecoration: {
  221. type: Object,
  222. default: () => {
  223. return {}
  224. },
  225. },
  226. curPrint: {
  227. type: String,
  228. default: '',
  229. },
  230. curModel: {
  231. type: String,
  232. default: '',
  233. },
  234. marginBottom: {
  235. type: Number,
  236. default: 20,
  237. },
  238. decorationMultiple: {
  239. type: Number,
  240. default: 1,
  241. },
  242. headerShow: {
  243. type: Boolean,
  244. default: true,
  245. },
  246. },
  247. data() {
  248. return {
  249. setupConfigList: [
  250. { id: 1, name: '/col' },
  251. { id: 2, name: '/desgin' },
  252. { id: 3, name: '/pos' },
  253. { id: 4, name: '/pos desgin' },
  254. { id: 5, name: 'POA' },
  255. { id: 6, name: 'Waived' },
  256. ],
  257. setupTitleList: ['1st', '2nd', '3rd'],
  258. }
  259. },
  260. computed: {
  261. getTableContainerClass() {
  262. // 根据tableType的值返回不同的class
  263. return {
  264. 'table-container': true,
  265. 'border-Decoration': this.tableType === 'Decoration' && this.hasDecAdd
  266. };
  267. },
  268. comTitle() {
  269. if (this.tableType === 'Undecorated') {
  270. return `Includes ${this.curModel}`
  271. } else if (this.tableType === 'Decoration') {
  272. return `Add-Ons`
  273. } else {
  274. return this.curPrint
  275. }
  276. },
  277. comDisplayCss() {
  278. return this.tableType === 'Comparison' ? '1px solid #efefef' : 'none'
  279. },
  280. comMultipleSelID() {
  281. return this.multipleSelection.map(i => i.id)
  282. },
  283. comTableData() {
  284. if (this.tableType === 'Decoration') {
  285. return this.tableData
  286. } else {
  287. return this.tableData
  288. }
  289. },
  290. },
  291. mounted() {
  292. // 初始勾上默认选择打印项
  293. if (JSON.stringify(this.selectRow) !== '{}') {
  294. this.$refs.elTable.toggleRowSelection(this.selectRow)
  295. }
  296. },
  297. methods: {
  298. judgeProp(row, prop, $index) {
  299. if (
  300. $index === 0 && this.tableType === 'Decoration'
  301. ) {
  302. const temp = prop.split('_')
  303. return row[`supplier_${temp[1]}`]
  304. } else {
  305. return row[prop]
  306. }
  307. },
  308. $_setupPriceFilter(setupPriceID) {
  309. const result = this.setupConfigList.filter(
  310. item => item.id === setupPriceID
  311. )
  312. return result.length > 0 ? result[0].name : ''
  313. },
  314. isNumber(val) {
  315. const rep = /^[0-9]+.?[0-9]*$/
  316. return rep.test(val)
  317. },
  318. /* 暂不实现此功能
  319. //点击table栏时选中按钮
  320. handleCurrentChange(currentRow, oldCurrentRow) {
  321. this.$refs.elTable.clearSelection()
  322. this.$emit('update:selectId', currentRow.id)
  323. this.addCar(currentRow)
  324. },
  325. */
  326. // table全选
  327. handleSelectionChange(val) {
  328. this.$emit('update:multipleSelection', val)
  329. },
  330. dialogCheck(selection, row) {
  331. if (this.isCheckBox) return // 是多选框返回
  332. this.$refs.elTable.clearSelection()
  333. // 初始化
  334. if (selection.length === 0) {
  335. this.$emit('update:selectId', null)
  336. this.$emit('update:selectRow', {})
  337. return
  338. }
  339. if (row) {
  340. this.$emit('update:selectId', row.id)
  341. this.$emit('update:selectRow', row)
  342. this.$refs.elTable.toggleRowSelection(row, true)
  343. }
  344. },
  345. addCar(currentRow) {
  346. // 加入购物车需要parentId
  347. const selrow = Object.assign({}, currentRow)
  348. selrow.parentId = this.decorationObj.parentId
  349. this.$emit('update:selectRow', selrow)
  350. this.$refs.elTable.toggleRowSelection(currentRow)
  351. },
  352. getRowKey(row) {
  353. return row.id
  354. },
  355. arraySpanMethod({ row, column, rowIndex, columnIndex }) {
  356. // 隐藏'Decoration'选项框
  357. if (this.tableType === 'Decoration' && columnIndex === 0) {
  358. return [0, 1]
  359. }
  360. if (this.tableType === 'Decoration' && columnIndex === 1) {
  361. return [1, 2]
  362. }
  363. },
  364. tableRowClassName({ row }) {
  365. if (this.tableType === 'Decoration') {
  366. return this.decorationMultiple ? 'borderNone' : 'hiddenRow'
  367. }
  368. return 'fontBold'
  369. },
  370. handleSelectable(row, index) {
  371. // website_setup_id 5为POA,不可选
  372. return row.website_setup_id !== 5;
  373. },
  374. getNumber(e, row, max) {
  375. if (parseInt(e) > max) {
  376. row.num = max
  377. }
  378. }
  379. },
  380. }
  381. </script>
  382. <style lang="scss" scoped>
  383. $deep-grey: #63676b;
  384. $deep-blue: #1d4992;
  385. .wrap {
  386. font-family: Arial;
  387. }
  388. .table-container {
  389. border-width: 1px 1px 4px;
  390. border-style: solid;
  391. border-bottom-left-radius: 4px;
  392. border-bottom-right-radius: 4px;
  393. border-color: transparent $deep-blue $deep-blue;
  394. }
  395. .border-Decoration {
  396. border-width: 1px 1px 1px !important;
  397. border-style: solid;
  398. border-radius: 0;
  399. border-color: transparent $deep-blue $deep-blue;
  400. }
  401. .tb-title {
  402. background-color: $deep-blue;
  403. color: #fff;
  404. font-size: 14px;
  405. text-align: center;
  406. padding: 9px 0;
  407. font-weight: bold;
  408. }
  409. .popover-text{
  410. font-size: 18px;
  411. padding: 10px;
  412. color: #000;
  413. font-weight: 600;
  414. }
  415. // 首行样式
  416. :deep(th.is-leaf) {
  417. padding: 0 !important;
  418. .cell {
  419. padding: 4px 0 !important;
  420. }
  421. }
  422. // :deep(.is-left:nth-child(2)) {
  423. // border-right: 1px solid #efefef;
  424. // }
  425. // :deep(.is-center:nth-child(2)) {
  426. // border-right: 1px solid #efefef;
  427. // }
  428. // 无数据不显示No Data
  429. :deep(.el-table__empty-block){
  430. display: none;
  431. }
  432. // 非首行样式
  433. :deep(.el-table__header-wrapper) {
  434. .el-checkbox{
  435. display: none;
  436. }
  437. }
  438. :deep(.el-table__body) {
  439. color: #000;
  440. font-size: 14px;
  441. td {
  442. padding: 4px 0 !important;
  443. }
  444. .el-table__cell{
  445. padding: 8px 0 !important;
  446. // 选择框颜色
  447. .el-checkbox__inner,.el-input__inner{
  448. border: 1px solid $deep-blue;
  449. }
  450. }
  451. .el-table__cell:not(:first-child) {
  452. .cell {
  453. word-break: normal !important;
  454. padding: 0;
  455. }
  456. }
  457. // 打印表格首列
  458. .el-table__cell:first-child {
  459. font-weight: bold;
  460. }
  461. // el-table 内置函数返回样式
  462. .borderNone {
  463. td:nth-child(1) {
  464. // border-right: 1px solid #efefef;
  465. .edit-input {
  466. .el-input__inner {
  467. padding-left: 5px;
  468. padding-right: 0;
  469. }
  470. }
  471. }
  472. td:nth-child(2) {
  473. border-right: none;
  474. }
  475. }
  476. .fontBold {
  477. .el-table__cell:nth-child(2) {
  478. font-weight: bold;
  479. }
  480. }
  481. .hiddenRow {
  482. display: none;
  483. }
  484. }
  485. .edit-input {
  486. max-width: 42px;
  487. }
  488. .triangle{
  489. position: absolute;
  490. left:-50px;
  491. top: 0;
  492. width: 0;
  493. height: 0;
  494. border-color: transparent #33AD60;
  495. border-width: 0 0 25px 50px;
  496. border-style: solid;
  497. }
  498. .text{
  499. position: absolute;
  500. left:-47px;
  501. top: -2px;
  502. color: #fff;
  503. font-size: 10px;
  504. }
  505. </style>