PersonalizeEdit.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690
  1. /* eslint-disable no-trailing-spaces */
  2. import React, { Component } from 'react';
  3. import pathToRegexp from 'path-to-regexp';
  4. import { routerRedux } from 'dva/router';
  5. import { connect } from 'dva';
  6. import { Card, Table, Modal, Form, Tooltip, Popconfirm, Switch, Button, Input } from 'antd';
  7. import Selector from '../../../components/AXTableSelector/Selector';
  8. import AXDragSortTable from '../../../components/AXDragSortTable';
  9. import FooterToolbar from '../../../components/FooterToolbar';
  10. import { renderStatus, statusToBool, boolToStatus, renderProductType } from '../../../utils/utils';
  11. import { Hotax } from '../../../utils/config';
  12. import styles from './PersonalizeEdit.less';
  13. const formItemLayout = {
  14. labelCol: {
  15. xs: { span: 24 },
  16. sm: { span: 2 },
  17. md: { span: 2 },
  18. },
  19. wrapperCol: {
  20. xs: { span: 24 },
  21. sm: { span: 14 },
  22. md: { span: 22 },
  23. },
  24. };
  25. @Form.create()
  26. @connect(({ loading, terminal, shelves, tagType, tag }) => ({
  27. tag,
  28. shelves,
  29. terminal,
  30. tagType,
  31. loading,
  32. sLoading: loading.models.shelves,
  33. tLoading: loading.models.tagType,
  34. mtLoading: loading.models.tag,
  35. }))
  36. export default class PersonalizeEditPage extends Component {
  37. constructor(props) {
  38. super(props);
  39. const { location } = props;
  40. const { state } = location;
  41. const match = pathToRegexp('/frontend/personalize/edit/:id').exec(location.pathname);
  42. this.state = {
  43. uid: match[1],
  44. mid: state.merchantId,
  45. tagModalDestroy: true,
  46. tagModalName: '新建用户标签',
  47. tagTypeSelecting: false, // 标签类型处于选择状态
  48. recModalDestroy: true,
  49. productSelecting: false, // 产品处于选择状态
  50. merchantTagModalDestroy: true,
  51. targetRow: {},
  52. };
  53. }
  54. componentDidMount() {
  55. // 加载用户标签数据
  56. this.props.dispatch({
  57. type: 'terminal/fetchTerminalTagList',
  58. payload: { uid: this.state.uid },
  59. });
  60. // 加载用户推荐课程
  61. this.props.dispatch({
  62. type: 'terminal/fetchTerminalRecommendCourse',
  63. payload: { uid: this.state.uid },
  64. });
  65. // 加载全部
  66. this.props.dispatch({
  67. type: 'tagType/fetchTagTypeList',
  68. payload: { pageSize: 1000 },
  69. });
  70. }
  71. /**
  72. * 用户标签创建/编辑模态框展现及数据加载控制
  73. * @param userTagId
  74. */
  75. handleUserTagModalShow = (userTagId) => {
  76. // 展现模态框前清空下currentUserTagItem内容
  77. this.props.dispatch({
  78. type: 'terminal/resetUserTagItem',
  79. });
  80. // 展现模态框
  81. this.setState({ tagModalDestroy: false });
  82. // 如果编辑标签则发获取详情请求
  83. if (userTagId) {
  84. this.props.dispatch({
  85. type: 'terminal/fetchTerminalTagItem',
  86. payload: { userTagId },
  87. });
  88. this.setState({ tagModalName: '编辑用户标签' });
  89. }
  90. };
  91. /**
  92. * 用户推荐课程模态框的展现
  93. */
  94. handleUserRecModalShow = () => {
  95. this.setState({ recModalDestroy: false });
  96. };
  97. handleMerchantTagModalShow = (record) => {
  98. this.setState({
  99. targetRow: record,
  100. merchantTagModalDestroy: false,
  101. });
  102. this.props.dispatch({
  103. type: 'tag/fetchTagList',
  104. payload: { merchantId: this.state.mid },
  105. });
  106. };
  107. /**
  108. * 模态框的取消操作
  109. * @param modalName
  110. */
  111. handleModalHide = (modalName) => {
  112. this.setState({ [modalName]: true });
  113. };
  114. /**
  115. * 用户标签内卡片切换操作
  116. * @param name
  117. */
  118. handleCardSwitch = (name) => {
  119. if (name === 'tagType') {
  120. this.props.dispatch({
  121. type: 'tagType/fetchTagTypeList',
  122. payload: {},
  123. });
  124. this.setState({ tagTypeSelecting: true });
  125. }
  126. if (name === 'product') {
  127. this.props.dispatch({
  128. type: 'shelves/fetchItemList',
  129. payload: { merchantId: this.state.mid },
  130. });
  131. this.setState({ productSelecting: true });
  132. }
  133. if (name === 'course') {
  134. this.props.dispatch({
  135. type: 'shelves/fetchCourseItemList',
  136. payload: { merchantId: this.state.mid },
  137. });
  138. this.setState({ productSelecting: true });
  139. }
  140. };
  141. /**
  142. * 标签类型/关联产品取消选择操作
  143. * @param name
  144. */
  145. handleSelectingCardCancel = (name) => {
  146. if (name === 'tagType') {
  147. this.setState({ tagTypeSelecting: false });
  148. }
  149. if (name === 'product' || name === 'course') {
  150. this.setState({ productSelecting: false });
  151. }
  152. };
  153. /**
  154. * 标签类型/关联产品筛选操作
  155. * @param name
  156. * @param data
  157. */
  158. handleSelectingCardChange = (name, data) => {
  159. if (name === 'tagType') {
  160. this.props.dispatch({
  161. type: 'tagType/fetchTagTypeList',
  162. payload: data,
  163. });
  164. }
  165. if (name === 'product') {
  166. this.props.dispatch({
  167. type: 'shelves/fetchItemList',
  168. payload: { merchantId: this.state.mid, ...data },
  169. });
  170. }
  171. if (name === 'course') {
  172. this.props.dispatch({
  173. type: 'shelves/fetchCourseItemList',
  174. payload: { merchantId: this.state.mid, ...data },
  175. });
  176. }
  177. if (name === 'tag') {
  178. this.props.dispatch({
  179. type: 'tag/fetchTagList',
  180. payload: { merchantId: this.state.uid, ...data },
  181. });
  182. }
  183. };
  184. /**
  185. * 标签类型/关联产品完成操作
  186. * @param cardName
  187. * @param data
  188. */
  189. handleSelectingCardFinish = (cardName, data) => {
  190. if (cardName === 'tagType') {
  191. const tagType = data[0] || {};
  192. const { code, name } = tagType;
  193. this.props.dispatch({
  194. type: 'terminal/fixCurrentUserTagItem',
  195. payload: {
  196. typeCode: code,
  197. typeName: name,
  198. },
  199. });
  200. this.setState({ tagTypeSelecting: false });
  201. }
  202. if (cardName === 'product') {
  203. this.props.dispatch({
  204. type: 'terminal/fixCurrentUserTagItem',
  205. payload: { productList: data },
  206. });
  207. this.setState({ productSelecting: false });
  208. }
  209. if (cardName === 'course') {
  210. this.props.dispatch({
  211. type: 'terminal/fixUserRecCourse',
  212. payload: (data || []).slice(0, 5),
  213. });
  214. this.setState({ productSelecting: false });
  215. }
  216. };
  217. /**
  218. * 用户标签内关联产品/推荐课程排序操作
  219. * @param rows
  220. * @param tabName
  221. */
  222. handleDragSortTableChange = (rows, tabName) => {
  223. if (tabName === 'tag') {
  224. this.props.dispatch({
  225. type: 'terminal/fixCurrentUserTagItem',
  226. payload: { productList: rows },
  227. });
  228. }
  229. if (tabName === 'rec') {
  230. this.props.dispatch({
  231. type: 'terminal/fixUserRecCourse',
  232. payload: rows,
  233. });
  234. }
  235. };
  236. handlePageBack = () => {
  237. this.props.dispatch(routerRedux.push({
  238. pathname: '/frontend/personalize/list',
  239. state: this.props.location.state,
  240. }));
  241. };
  242. handleUserTagSubmit = () => {
  243. this.props.form.validateFieldsAndScroll((err, values) => {
  244. if (!err) {
  245. const { sort, name, status } = values;
  246. const { uid } = this.state;
  247. const { terminal } = this.props;
  248. const { currentUserTagItem } = terminal;
  249. const { id, typeCode, productList } = currentUserTagItem;
  250. const pidList = (productList || []).map(product => product.pid);
  251. const newStatus = boolToStatus(status);
  252. if (!id) {
  253. this.props.dispatch({
  254. type: 'terminal/createTerminalTagItem',
  255. payload: { sort, uid, name, typeCode, status: newStatus, productList: pidList },
  256. });
  257. } else {
  258. this.props.dispatch({
  259. type: 'terminal/updateTerminalTagItem',
  260. payload: { id, sort, uid, name, typeCode, status: newStatus, productList: pidList },
  261. });
  262. }
  263. this.setState({ tagModalDestroy: true });
  264. }
  265. });
  266. };
  267. handleUserTagCopyOperation = (data) => {
  268. const { targetRow } = this.state;
  269. const merchantTag = (data || [])[0] || {};
  270. this.props.dispatch({
  271. type: 'terminal/copyMerchantTagToUser',
  272. payload: { uid: this.state.uid, userTagId: targetRow.id, tagId: merchantTag.id },
  273. });
  274. this.setState({ merchantTagModalDestroy: true });
  275. };
  276. handleUserTagDelete = (userTagId) => {
  277. this.props.dispatch({
  278. type: 'terminal/deleteTerminalTagItem',
  279. payload: { uid: this.state.uid, userTagId },
  280. });
  281. };
  282. handleUserRecCourseSubmit = () => {
  283. const { uid } = this.state;
  284. const { terminal } = this.props;
  285. const { userRecCourse } = terminal;
  286. const idList = (userRecCourse || []).map(product => product.pid);
  287. this.props.dispatch({
  288. type: 'terminal/updateTerminalRecommendCourse',
  289. payload: { uid, idList },
  290. });
  291. this.setState({ recModalDestroy: true });
  292. };
  293. render() {
  294. const {
  295. tagModalDestroy,
  296. tagModalName,
  297. tagTypeSelecting,
  298. productSelecting,
  299. recModalDestroy,
  300. merchantTagModalDestroy,
  301. } = this.state;
  302. const {
  303. sLoading, tLoading, mtLoading, loading, terminal, shelves, tagType, tag, form,
  304. } = this.props;
  305. const { getFieldDecorator } = form;
  306. const { userTagList, userRecCourse, currentUserTagItem } = terminal;
  307. const { sort, name, status, productList, typeCode, typeName } = currentUserTagItem;
  308. // 用户标签列表表头
  309. const tagColumns = [{
  310. title: '标签位置',
  311. dataIndex: 'sort',
  312. width: '10%',
  313. }, {
  314. title: '标签名称',
  315. dataIndex: 'name',
  316. width: '25%',
  317. }, {
  318. title: '标签类型',
  319. dataIndex: 'typeCode',
  320. width: '25%',
  321. filters: (tagType.list || []).map(item => ({ text: item.name, value: item.code })),
  322. onFilter: (value, record) => record.typeCode.indexOf(value) === 0,
  323. }, {
  324. title: '标签状态',
  325. dataIndex: 'status',
  326. width: '15%',
  327. render: text => renderStatus(text),
  328. filters: [{
  329. text: '正常',
  330. value: Hotax.STATUS_NORMAL,
  331. }, {
  332. text: '删除',
  333. value: Hotax.STATUS_DELETE,
  334. }],
  335. onFilter: (value, record) => record.status === value,
  336. align: 'center',
  337. }, {
  338. title: '操作',
  339. width: '25%',
  340. align: 'right',
  341. render: (_, record) => (
  342. <div>
  343. <Button
  344. size="small"
  345. className="editBtn"
  346. onClick={() => this.handleUserTagModalShow(record.id)}
  347. >编辑
  348. </Button>
  349. <Tooltip title="复制渠道标签进行关联产品">
  350. <Button
  351. size="small"
  352. className="recBtn"
  353. onClick={() => this.handleMerchantTagModalShow(record)}
  354. >复制
  355. </Button>
  356. </Tooltip>
  357. <Popconfirm
  358. placement="top"
  359. title="确定要删除该用户标签?"
  360. okText="确定"
  361. cancelText="取消"
  362. onConfirm={() => this.handleUserTagDelete(record.id)}
  363. >
  364. <Button
  365. size="small"
  366. className="delBtn"
  367. >删除
  368. </Button>
  369. </Popconfirm>
  370. </div>
  371. ),
  372. }];
  373. // 推荐课程
  374. const courseColumns = [{
  375. title: '课程编号',
  376. dataIndex: 'code',
  377. key: 1,
  378. width: '40%',
  379. }, {
  380. title: '课程名称',
  381. dataIndex: 'name',
  382. key: 2,
  383. width: '40%',
  384. }, {
  385. title: '课程状态',
  386. dataIndex: 'status',
  387. key: 3,
  388. render: text => renderStatus(text),
  389. width: '20%',
  390. }];
  391. /* ************************ modal1: 用户标签创建 ************************* */
  392. const getTagCreateModal = () => {
  393. const tagTypeColumns = [{
  394. title: '标签类型编号',
  395. dataIndex: 'code',
  396. key: 1,
  397. width: '50%',
  398. }, {
  399. title: '标签类型名称',
  400. dataIndex: 'name',
  401. key: 2,
  402. width: '50%',
  403. }];
  404. const tagTypeData = typeCode ? [{ key: 'row-1', name: typeName || '-', code: typeCode }] : undefined;
  405. const tagTypeSelectCard = (
  406. <Card title="选择标签类型">
  407. <Selector
  408. multiple={false}
  409. loading={tLoading}
  410. selectorName="TagType"
  411. list={tagType.list}
  412. pageNo={tagType.pageNo}
  413. pageSize={tagType.pageSize}
  414. totalSize={tagType.totalSize}
  415. onCancel={() => this.handleSelectingCardCancel('tagType')}
  416. onChange={data => this.handleSelectingCardChange('tagType', data)}
  417. onFinish={data => this.handleSelectingCardFinish('tagType', data)}
  418. />
  419. </Card>
  420. );
  421. const tagTypeShowCard = (
  422. <Card title="标签类型信息">
  423. <Table
  424. pagination={false}
  425. dataSource={tagTypeData}
  426. columns={tagTypeColumns}
  427. className={styles.tagTable}
  428. onChange={this.handleUserTagTableChange}
  429. />
  430. </Card>
  431. );
  432. const productColumns = [{
  433. title: '产品编号',
  434. dataIndex: 'code',
  435. key: 1,
  436. width: '20%',
  437. }, {
  438. title: '产品名称',
  439. dataIndex: 'name',
  440. key: 2,
  441. width: '30%',
  442. }, {
  443. title: '产品类型',
  444. dataIndex: 'type',
  445. key: 3,
  446. render: text => renderProductType(text),
  447. }];
  448. const productSelectCard = (
  449. <Card title="选择产品">
  450. <Selector
  451. multiple
  452. loading={sLoading}
  453. selectorName="Product"
  454. list={shelves.list}
  455. pageNo={shelves.pageNo}
  456. pageSize={shelves.pageSize}
  457. totalSize={shelves.totalSize}
  458. selectedRows={productList}
  459. onCancel={() => this.handleSelectingCardCancel('product')}
  460. onChange={data => this.handleSelectingCardChange('product', data)}
  461. onFinish={data => this.handleSelectingCardFinish('product', data)}
  462. />
  463. </Card>
  464. );
  465. const productShowCard = (
  466. <Card title="已关联产品列表">
  467. <AXDragSortTable
  468. data={productList}
  469. columns={productColumns}
  470. onChange={rows => this.handleDragSortTableChange(rows, 'tag')}
  471. />
  472. </Card>
  473. );
  474. return (
  475. <Modal
  476. visible
  477. width={1100}
  478. title={tagModalName}
  479. maskClosable={false}
  480. cancelText="取消"
  481. okText="提交"
  482. onCancel={() => this.handleModalHide('tagModalDestroy')}
  483. onOk={this.handleUserTagSubmit}
  484. >
  485. <Form>
  486. <Form.Item hasFeedback label="标签名称" {...formItemLayout}>
  487. {getFieldDecorator('name', {
  488. rules: [{ required: true, message: '请填写标签名称' }],
  489. initialValue: name,
  490. })(
  491. <Input placeholder="请输入" />
  492. )}
  493. </Form.Item>
  494. <Form.Item hasFeedback label="标签位置" {...formItemLayout}>
  495. {getFieldDecorator('sort', {
  496. initialValue: sort,
  497. })(
  498. <Input placeholder="请输入" />
  499. )}
  500. </Form.Item>
  501. <Form.Item label="标签类型" {...formItemLayout}>
  502. <Button
  503. disabled={tagTypeSelecting}
  504. type="primary"
  505. size="small"
  506. icon="search"
  507. onClick={() => this.handleCardSwitch('tagType')}
  508. >选择
  509. </Button>
  510. </Form.Item>
  511. <Form.Item wrapperCol={{ offset: 2, span: 22 }}>
  512. {tagTypeSelecting ? tagTypeSelectCard : tagTypeShowCard}
  513. </Form.Item>
  514. <Form.Item label="关联产品" {...formItemLayout}>
  515. <Button
  516. disabled={productSelecting}
  517. type="primary"
  518. size="small"
  519. icon="form"
  520. onClick={() => this.handleCardSwitch('product')}
  521. >编辑
  522. </Button>
  523. </Form.Item>
  524. <Form.Item wrapperCol={{ offset: 2, span: 22 }}>
  525. {productSelecting ? productSelectCard : productShowCard}
  526. </Form.Item>
  527. <Form.Item label="标签状态" {...formItemLayout}>
  528. {getFieldDecorator('status', {
  529. valuePropName: 'checked',
  530. initialValue: statusToBool(status),
  531. })(
  532. <Switch checkedChildren="正常" unCheckedChildren="删除" />
  533. )}
  534. </Form.Item>
  535. </Form>
  536. </Modal>
  537. );
  538. };
  539. /* ************************ modal2: 用户推荐课程修改 ************************* */
  540. const getRecCourseModal = () => {
  541. const cColumns = [{
  542. title: '课程编号',
  543. dataIndex: 'code',
  544. key: 1,
  545. width: '30%',
  546. }, {
  547. title: '课程名称',
  548. dataIndex: 'name',
  549. key: 2,
  550. width: '30%',
  551. }];
  552. const recCourseSelectCard = (
  553. <Card title="选择课程">
  554. <Selector
  555. multiple
  556. loading={sLoading}
  557. selectorName="Course"
  558. list={shelves.list}
  559. pageNo={shelves.pageNo}
  560. pageSize={shelves.pageSize}
  561. totalSize={shelves.totalSize}
  562. selectedRows={userRecCourse}
  563. onCancel={() => this.handleSelectingCardCancel('course')}
  564. onChange={data => this.handleSelectingCardChange('course', data)}
  565. onFinish={data => this.handleSelectingCardFinish('course', data)}
  566. />
  567. </Card>
  568. );
  569. const recCourseShowCard = (
  570. <Card title="推荐课程列表">
  571. <AXDragSortTable
  572. data={userRecCourse}
  573. columns={cColumns}
  574. onChange={rows => this.handleDragSortTableChange(rows, 'rec')}
  575. />
  576. </Card>
  577. );
  578. return (
  579. <Modal
  580. visible
  581. width={1100}
  582. title="用户推荐课程"
  583. maskClosable={false}
  584. cancelText="取消"
  585. okText="提交"
  586. onCancel={() => this.handleModalHide('recModalDestroy')}
  587. onOk={this.handleUserRecCourseSubmit}
  588. >
  589. <Form>
  590. <Form.Item label="推荐课程" {...formItemLayout}>
  591. <Button
  592. disabled={productSelecting}
  593. type="primary"
  594. size="small"
  595. icon="search"
  596. onClick={() => this.handleCardSwitch('course')}
  597. >选择
  598. </Button>
  599. </Form.Item>
  600. <Form.Item wrapperCol={{ offset: 2, span: 22 }}>
  601. {productSelecting ? recCourseSelectCard : recCourseShowCard}
  602. </Form.Item>
  603. </Form>
  604. </Modal>
  605. );
  606. };
  607. /* ************************ modal3: 渠道标签选择 ************************* */
  608. const getMerchantTagModal = () => {
  609. return (
  610. <Modal
  611. width={1100}
  612. footer={null}
  613. visible
  614. title="渠道标签列表"
  615. maskClosable={false}
  616. onCancel={() => this.handleModalHide('merchantTagModalDestroy')}
  617. >
  618. <Selector
  619. multiple={false}
  620. loading={mtLoading}
  621. selectorName="Tag"
  622. list={tag.list}
  623. pageNo={tag.pageNo}
  624. pageSize={tag.pageSize}
  625. totalSize={tag.totalSize}
  626. onCancel={() => this.handleModalHide('merchantTagModalDestroy')}
  627. onChange={data => this.handleSelectingCardChange('tag', data)}
  628. onFinish={this.handleUserTagCopyOperation}
  629. />
  630. </Modal>
  631. );
  632. };
  633. return (
  634. <div>
  635. <Card
  636. title={
  637. <div>
  638. 用户标签
  639. <Button onClick={() => this.handleUserTagModalShow()} style={{ float: 'right' }} type="primary">新建标签</Button>
  640. </div>
  641. }
  642. loading={loading.effects['terminal/fetchTerminalTagList']}
  643. style={{ marginBottom: 16 }}
  644. className={styles.tagCard}
  645. >
  646. <Table
  647. pagination={{ pageSize: 15 }}
  648. dataSource={userTagList}
  649. columns={tagColumns}
  650. rowKey={record => record.id}
  651. className={styles.tagTable}
  652. />
  653. {!tagModalDestroy && getTagCreateModal()}
  654. {!merchantTagModalDestroy && getMerchantTagModal()}
  655. </Card>
  656. <Card
  657. title={
  658. <div>
  659. 推荐课程
  660. <Button onClick={this.handleUserRecModalShow} style={{ float: 'right' }} type="primary">更换课程</Button>
  661. </div>
  662. }
  663. loading={loading.effects['terminal/fetchTerminalRecommendCourse']}
  664. style={{ marginBottom: 70 }}
  665. className={styles.tagCard}
  666. >
  667. <Table
  668. pagination={false}
  669. dataSource={userRecCourse}
  670. columns={courseColumns}
  671. rowKey={record => record.id}
  672. className={styles.tagTable}
  673. />
  674. {!recModalDestroy && getRecCourseModal()}
  675. </Card>
  676. <FooterToolbar style={{ width: '100%' }}>
  677. <Button type="primary" onClick={this.handlePageBack}>返回上一页</Button>
  678. </FooterToolbar>
  679. </div>
  680. );
  681. }
  682. }