index.js 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. import React, { PureComponent } from 'react';
  2. import { connect } from 'dva';
  3. import { routerRedux } from 'dva/router';
  4. import { Card, Table, Form, Spin, Input, InputNumber, Button, Icon, message } from 'antd';
  5. import queryString from 'query-string';
  6. import PageHeaderLayout from '../../../layouts/PageHeaderLayout';
  7. import ProductSelectModal from './product';
  8. import { Codes, pageSize, productType } from '../../../utils/config';
  9. @Form.create()
  10. @connect(state => ({
  11. comboDetail: state.comboDetail,
  12. product: state.product,
  13. }))
  14. export default class PackageProfile extends PureComponent {
  15. state = { proType: [Codes.CODE_COURSE] }; // 产品类型[默认为课程资源]
  16. handleEditProuctClick = () => {
  17. const { proType } = this.state;
  18. this.props.dispatch({ type: 'comboDetail/showModal' });
  19. this.props.dispatch({
  20. type: 'product/query',
  21. payload: {
  22. type: proType[0],
  23. pageNo: 1,
  24. pageSize,
  25. }
  26. });
  27. }
  28. handleInputNumberChange = (value, record) => {
  29. const { currentItem } = this.props.comboDetail;
  30. const { products } = currentItem;
  31. const boundPriceProducts = [...products];
  32. boundPriceProducts.map(item => item.pid === record.pid ? item.cpPrice = value : null);
  33. this.props.dispatch({
  34. type: 'comboDetail/savePrice',
  35. payload: boundPriceProducts,
  36. });
  37. }
  38. handleProductModalOk = (data) => {
  39. this.props.dispatch({
  40. type: 'comboDetail/saveProducts',
  41. payload: data ,
  42. });
  43. }
  44. handleProductModalCancel = () => {
  45. this.props.dispatch({ type: 'comboDetail/hideModal' });
  46. }
  47. handleProductModalSearch = (data) => {
  48. const newData = { ...data };
  49. const { proType } = this.state;
  50. if (newData.keyword) {
  51. newData[newData.field] = newData.keyword;
  52. }
  53. delete newData.field;
  54. delete newData.keyword;
  55. this.props.dispatch({
  56. type: 'product/query',
  57. payload: {
  58. ...newData,
  59. type: proType[0],
  60. pageNo: 1,
  61. pageSize,
  62. }
  63. });
  64. }
  65. handleProductModalTableChange = (pagination, filterArgs, filters) => {
  66. // 待选资源列表中资源类型过滤选项
  67. const { type } = filterArgs;
  68. if (Array.isArray(type) && type.length) {
  69. this.setState({ proType: type.map(item => item) });
  70. } else {
  71. this.setState({ proType: [Codes.CODE_COURSE] });
  72. filterArgs.type = [Codes.CODE_COURSE];
  73. }
  74. const newFilters = { ...filters };
  75. if (newFilters.keyword) {
  76. newFilters[newFilters.field] = newFilters.keyword;
  77. }
  78. delete newFilters.field;
  79. delete newFilters.keyword;
  80. // table header filter
  81. const getValue = obj => Object.keys(obj).map(key => obj[key]).join(',');
  82. const tableFilters = Object.keys(filterArgs).reduce((obj, key) => {
  83. const newObj = { ...obj };
  84. newObj[key] = getValue(filterArgs[key]);
  85. return newObj;
  86. }, {});
  87. const data = { ...newFilters, ...tableFilters, pageNo: pagination.current, pageSize: pagination.pageSize };
  88. Object.keys(data).map(key => data[key] ? null : delete data[key]);
  89. this.props.dispatch({ type: 'product/query', payload: data });
  90. }
  91. handlePageCancel = () => {
  92. const { filters } = this.props.comboDetail;
  93. this.props.dispatch(routerRedux.push({
  94. pathname: '/product/package',
  95. search: queryString.stringify(filters),
  96. }));
  97. this.props.dispatch({ type: 'comboDetail/initState' });
  98. }
  99. handlePageSubmit = (e) => {
  100. e.preventDefault();
  101. const { dispatch, form, comboDetail } = this.props;
  102. const { currentItem, operType, filters } = comboDetail;
  103. const { products } = currentItem;
  104. const { validateFields, getFieldsValue } = form;
  105. validateFields((errors) => {
  106. if (errors) return;
  107. if (products && products.filter(item => !item.cpPrice).length > 0 ) {
  108. message.error('还有供应商价格未填写!');
  109. return;
  110. }
  111. // add params `code` and `name`
  112. const data = { ...getFieldsValue() };
  113. // add params `products`
  114. if (Array.isArray(products)) {
  115. data.products = products.map(item => ({
  116. pid: item.pid,
  117. type: item.type,
  118. cpPrice: item.cpPrice,
  119. }));
  120. }
  121. // add params `status`
  122. if (operType === 'create') {
  123. data.status = Codes.CODE_NORMAL;
  124. data.type = Codes.CODE_PACKAGE;
  125. }
  126. else if (operType === 'update') {
  127. data.status = currentItem.status;
  128. data.id = currentItem.id;
  129. }
  130. dispatch({
  131. type: `comboDetail/${operType}`,
  132. payload: data,
  133. callback: () => {
  134. dispatch(routerRedux.push({
  135. pathname: '/product/package',
  136. search: queryString.stringify(filters),
  137. }));
  138. }
  139. });
  140. });
  141. }
  142. render() {
  143. const { form, comboDetail, product } = this.props;
  144. const { proType } = this.state;
  145. const { modalShow, currentItem, itemLoading } = comboDetail;
  146. const { list, listLoading, pagination } = product;
  147. const { products } = currentItem;
  148. const { getFieldDecorator } = form;
  149. const formItemLayout = {
  150. labelCol: { span: 7 },
  151. wrapperCol: { span: 12 },
  152. };
  153. const submitFormLayout = {
  154. wrapperCol: {
  155. xs: { span: 24, offset: 0 },
  156. sm: { span: 10, offset: 7 },
  157. },
  158. };
  159. const tableFormLayout = {
  160. wrapperCol: { offset: 7, span: 12 },
  161. };
  162. const columns = [{
  163. // title: '封面',
  164. // dataIndex: 'coverUrl',
  165. // key: 'coverUrl',
  166. // render: (text, record) => {},
  167. // },{
  168. title: '编号',
  169. dataIndex: 'code',
  170. key: 'code',
  171. width: '20%',
  172. },{
  173. title: '名称',
  174. dataIndex: 'name',
  175. key: 'name',
  176. width: '20%',
  177. },{
  178. title: '类型',
  179. dataIndex: 'type',
  180. key: 'type',
  181. render: (text, record) => productType[text],
  182. width: '15%',
  183. },{
  184. title: '供应商',
  185. dataIndex: 'cpName',
  186. key: 'cpName',
  187. width: '17%',
  188. },{
  189. title: '供应商价格',
  190. dataIndex: 'cpPrice',
  191. key: 'cpPrice',
  192. render: (text, record) => (
  193. <InputNumber
  194. min={0}
  195. value={record.cpPrice}
  196. onChange={(value) => this.handleInputNumberChange(value, record)}
  197. formatter={value => `¥ ${value}`.replace(/\B(?=(\d{3})+(?!\d))/g, ',')}
  198. parser={value => value.replace(/\¥\s?|(,*)/g, '')}
  199. />
  200. ),
  201. width: '28%',
  202. }];
  203. return (
  204. <PageHeaderLayout>
  205. <Spin spinning={itemLoading}>
  206. <Card>
  207. <Form layout="horizontal" onSubmit={this.handlePageSubmit}>
  208. <Form.Item label="产品包编号" { ...formItemLayout }>
  209. {getFieldDecorator('code', {
  210. rules: [{ required: true, type: 'string', message: '编号为必填项!' }],
  211. })(<Input placeholder="请输入" />)}
  212. </Form.Item>
  213. <Form.Item label="产品包名称" { ...formItemLayout }>
  214. {getFieldDecorator('name', {
  215. rules: [{ required: true, type: 'string', message: '名称为必填项!' }],
  216. })(<Input placeholder="请输入" />)}
  217. </Form.Item>
  218. <Form.Item label="选择产品" { ...formItemLayout }>
  219. <Button onClick={this.handleEditProuctClick} type="primary" size="small" icon="select">选择</Button>
  220. </Form.Item>
  221. <Form.Item { ...tableFormLayout }>
  222. <Table
  223. simple
  224. bordered
  225. columns={columns}
  226. dataSource={products || []}
  227. pagination={false}
  228. rowKey={record => record.id}
  229. locale={{
  230. emptyText: <span style={{ color: "#C6D0D6" }}>&nbsp;&nbsp;<Icon type="frown-o"/>
  231. 该产品包内暂无相关内容!</span>
  232. }}
  233. />
  234. </Form.Item>
  235. <Form.Item { ...submitFormLayout } style={{ marginTop: 32 }}>
  236. <Button onClick={this.handlePageCancel}>取消</Button>
  237. <Button type="primary" style={{ marginLeft: 35 }} htmlType="submit">提交</Button>
  238. </Form.Item>
  239. </Form>
  240. {/*产品选择模态框*/}
  241. <ProductSelectModal
  242. rowKeyName="id"
  243. modalVisible={modalShow}
  244. selTableData={products || []}
  245. width={600}
  246. style={{ top: 20 }}
  247. fsTableDataSource={list}
  248. fsTableLoading={listLoading}
  249. fsTablePagination={pagination}
  250. fsFilteredValue={proType}
  251. onOk={this.handleProductModalOk}
  252. onCancel={this.handleProductModalCancel}
  253. onSearch={this.handleProductModalSearch}
  254. fsTableOnChange={this.handleProductModalTableChange}
  255. />
  256. </Card>
  257. </Spin>
  258. </PageHeaderLayout>
  259. );
  260. }
  261. }