123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505 |
- import React, { Component } from 'react';
- import pathToRegexp from 'path-to-regexp';
- import { connect } from 'dva';
- import { routerRedux } from 'dva/router';
- import { Card, Table, Modal, Popconfirm, Radio, Switch, Button, Input } from 'antd';
- import Selector from '../../../components/AXTableSelector/Selector';
- import FooterToolbar from '../../../components/FooterToolbar';
- import {
- boolToStatus, genAbsolutePicUrl, renderProductType, renderStatus, statusToBool,
- } from '../../../utils/utils';
- import bitmap from '../../../assets/bitmap.png';
- import styles from './RecommendPoster.less';
- @connect(({ loading, merchant, shelves, resource }) => ({
- shelves,
- merchant,
- resource,
- rLoading: loading.models.resource,
- sLoading: loading.models.shelves,
- mLoading: loading.models.merchant,
- }))
- export default class RecommendPosterEditPage extends Component {
- state = {
- productSelectorDestroy: true,
- resourceSelectorDestroy: true,
- productType: 'Course',
- currentEditPosterId: '',
- };
- componentDidMount() {
- this.props.dispatch({
- type: 'merchant/fetchMerchantPoster',
- payload: { merchantId: this.getMerchantId() },
- });
- }
- /**
- * 1.从URL中提取merchantId
- * @returns {String}
- */
- getMerchantId = () => {
- const match = pathToRegexp('/frontend/recommend/poster-edit/:id')
- .exec(this.props.location.pathname);
- return match[1];
- };
- /**
- * 2.响应产品类型变化
- * @param e
- */
- handleRadioChange = (e) => {
- this.setState({ productType: e.target.value });
- this.props.dispatch({
- type: `shelves/fetch${e.target.value}ItemList`,
- payload: { merchantId: this.getMerchantId() },
- });
- };
- /**
- * 3.新建一条海报
- */
- handlePosterItemCreate = () => {
- const newData = [...this.props.merchant.posterList];
- newData.push({
- id: `new-poster-${newData.length + 1}`,
- isNew: true,
- isEdit: true,
- });
- this.props.dispatch({
- type: 'merchant/fixPosterList',
- payload: newData,
- });
- };
- /**
- * 4.删除一条海报
- * @param posterId
- * @param isNew
- */
- handlePosterItemDelete = (posterId, isNew) => {
- if (isNew) {
- const originalData = [...this.props.merchant.posterList];
- const newData = originalData.filter(data => data.id !== posterId);
- this.props.dispatch({
- type: 'merchant/fixPosterList',
- payload: newData,
- });
- return;
- }
- this.props.dispatch({
- type: 'merchant/deleteMerchantPosterItem',
- payload: { posterId },
- });
- };
- /**
- * 5.编辑一条海报
- * @param posterId
- */
- handlePosterItemEdit = (posterId) => {
- const newData = [...this.props.merchant.posterList];
- for (const index in newData) {
- if (newData[index].id === posterId) {
- newData[index].isEdit = true;
- }
- }
- this.props.dispatch({
- type: 'merchant/fixPosterList',
- payload: newData,
- });
- };
- /**
- * 6.控制模态框的展现
- * @param {String} flag
- * @param {String} posterId
- */
- handleSelectorModalShow = (flag, posterId) => {
- this.setState({
- [`${flag}SelectorDestroy`]: false,
- currentEditPosterId: posterId,
- });
- if (flag !== 'resource') {
- const { productType } = this.state;
- this.props.dispatch({
- type: `shelves/fetch${productType}ItemList`,
- payload: { merchantId: this.getMerchantId() },
- });
- return;
- }
- this.props.dispatch({
- type: 'resource/fetchImageList',
- payload: {},
- });
- };
- /**
- * 7.控制模态框的销毁
- */
- handleSelectorCancel = (flag) => {
- this.setState({ [`${flag}SelectorDestroy`]: true });
- };
- /**
- * 8.响应模态框内查询操作
- * @param {String} flag
- * @param {Object} params
- */
- handleSelectorChange = (flag, params) => {
- if (flag !== 'resource') {
- const { productType } = this.state;
- this.props.dispatch({
- type: `shelves/fetch${productType}ItemList`,
- payload: { ...params, merchantId: this.getMerchantId() },
- });
- return;
- }
- this.props.dispatch({
- type: 'resource/fetchImageList',
- payload: params,
- });
- };
- /**
- * 9.响应选择完成操作
- * @param {String} flag
- * @param {Array} rows
- */
- handleSelectorFinish = (flag, rows) => {
- this.setState({ [`${flag}SelectorDestroy`]: true });
- const { currentEditPosterId } = this.state;
- const originalData = [...this.props.merchant.posterList];
- const newData = originalData.map((data) => {
- if (flag === 'product' && data.id === currentEditPosterId) {
- const { pid, name, code, type } = rows[0];
- return { ...data, type, name, code, pid };
- }
- if (flag === 'resource' && data.id === currentEditPosterId) {
- const { path } = rows[0];
- return { ...data, img: path };
- }
- return { ...data };
- });
- this.props.dispatch({
- type: 'merchant/fixPosterList',
- payload: newData,
- });
- };
- /**
- * 10.修改排序值
- * @param e
- * @param posterId
- */
- handleSortInputChange = (e, posterId) => {
- const originalData = [...this.props.merchant.posterList];
- const newData = originalData.map((data) => {
- if (data.id === posterId) {
- return { ...data, sort: parseInt(e.target.value, 10) || 0 };
- }
- return { ...data };
- });
- this.props.dispatch({
- type: 'merchant/fixPosterList',
- payload: newData,
- });
- };
- /**
- * 11.修改状态
- * @param checked
- * @param posterId
- */
- handleStatusSwitchChange = (checked, posterId) => {
- const originalData = [...this.props.merchant.posterList];
- const newData = originalData.map((data) => {
- if (data.id === posterId) {
- return { ...data, status: boolToStatus(checked) };
- }
- return { ...data };
- });
- this.props.dispatch({
- type: 'merchant/fixPosterList',
- payload: newData,
- });
- };
- /**
- * 12.提交海报内容
- * @param posterId
- */
- handleSaveOperation = (posterId) => {
- const originalData = [...this.props.merchant.posterList];
- const targetData = originalData.filter(data => data.id === posterId)[0];
- const { id, pid, img, sort, type, status, isNew } = targetData;
- if (isNew) {
- this.props.dispatch({
- type: 'merchant/createMerchantPosterItem',
- payload: {
- img, pid, type, sort, status: boolToStatus(status), merchantId: this.getMerchantId(),
- },
- posterId: id,
- });
- return;
- }
- this.props.dispatch({
- type: 'merchant/updateMerchantPosterItem',
- payload: {
- id, img, pid, type, sort, status: boolToStatus(status), merchantId: this.getMerchantId(),
- },
- posterId: id,
- });
- };
- /**
- * 13.返回上一页
- */
- handlePageBack = () => {
- this.props.dispatch(routerRedux.push({
- pathname: '/frontend/recommend',
- state: this.props.location.state,
- }));
- };
- render() {
- const { productType, productSelectorDestroy, resourceSelectorDestroy } = this.state;
- const { merchant, shelves, resource, rLoading, sLoading, mLoading } = this.props;
- const { posterList } = merchant;
- /* 海报列表格式设定 */
- const posterColumns = [{
- title: '位置',
- key: 1,
- dataIndex: 'sort',
- width: '15%',
- render: (text, record) => {
- // 编辑状态下排序可自由填写
- const { id, isEdit } = record;
- if (isEdit) {
- return (
- <Input
- value={text}
- onChange={e => this.handleSortInputChange(e, id)}
- placeholder="请填写"
- style={{ width: 100 }}
- />
- );
- }
- return text;
- },
- align: 'center',
- }, {
- title: '海报封面',
- key: 2,
- dataIndex: 'img',
- width: '27%',
- render: (text, record) => {
- // 将海报封面渲染为一张图片,编辑状态下可更换
- const { id, isNew, isEdit } = record;
- return (
- <div className={styles.cover}>
- {isEdit && (
- <div className={styles.mongolian}>
- <a onClick={() => this.handleSelectorModalShow('resource', id)}>{isNew ? '选择' : '更换'}</a>
- </div>
- )}
- {(isNew && !text) ? <img src={bitmap} alt="" /> : <img src={genAbsolutePicUrl(text)} alt="海报封面" />}
- </div>
- );
- },
- align: 'center',
- }, {
- title: '关联产品信息',
- key: 3,
- render: (_, record) => {
- // 将产品信息渲染成一个小表格
- const { id, code, name, type, product, isEdit, isNew } = record;
- const columns = [{
- dataIndex: 'label',
- width: '40%',
- }, {
- dataIndex: 'value',
- width: '60%',
- }];
- const data = [{
- key: 'name',
- label: '产品名称',
- value: name || (product || {}).name,
- }, {
- key: 'code',
- label: '产品编号',
- value: code || (product || {}).code,
- }, {
- key: 'type',
- label: '产品类型',
- value: renderProductType(type || (product || {}).type),
- }];
- return (
- <div className={styles.product}>
- {isEdit && (
- <div className={styles.mongolian}>
- <a onClick={() => this.handleSelectorModalShow('product', id)}>{isNew ? '选择' : '更换'}</a>
- </div>
- )}
- <Table
- bordered
- size="small"
- showHeader={false}
- pagination={false}
- columns={columns}
- dataSource={data}
- />
- </div>
- );
- },
- width: '30%',
- align: 'center',
- }, {
- title: '删除状态',
- key: 4,
- dataIndex: 'status',
- width: '13%',
- render: (text, record) => {
- const { id, isEdit } = record;
- if (isEdit) {
- return (
- <Switch
- checked={statusToBool(text)}
- checkedChildren="正常"
- unCheckedChildren="删除"
- onChange={checked => this.handleStatusSwitchChange(checked, id)}
- />
- );
- }
- return renderStatus(text);
- },
- align: 'center',
- }, {
- title: '相关操作',
- key: 5,
- width: '15%',
- render: (_, record) => {
- const { id, isNew, isEdit } = record;
- const getPopconfirmBtn = () => {
- return (
- <Popconfirm
- placement="top"
- title="确定要删除该海报?"
- okText="确定"
- cancelText="取消"
- onConfirm={() => this.handlePosterItemDelete(id, isNew)}
- >
- <Button
- size="small"
- className="delBtn"
- >删除
- </Button>
- </Popconfirm>
- );
- };
- if (isEdit) {
- return (
- <div>
- <Button
- size="small"
- className="editBtn"
- onClick={() => this.handleSaveOperation(id)}
- >保存
- </Button>
- {getPopconfirmBtn()}
- </div>
- );
- }
- return (
- <div>
- <Button
- size="small"
- className="editBtn"
- onClick={() => this.handlePosterItemEdit(id)}
- >编辑
- </Button>
- {getPopconfirmBtn()}
- </div>
- );
- },
- align: 'right',
- }];
- /* 产品模态框选择器 */
- const renderModalTitle = () => {
- return (
- <Radio.Group
- value={productType}
- onChange={this.handleRadioChange}
- >
- <Radio.Button value="Course">课程</Radio.Button>
- <Radio.Button value="Support">配套</Radio.Button>
- <Radio.Button value="Training">师训</Radio.Button>
- <Radio.Button value="Package">套餐包</Radio.Button>
- </Radio.Group>
- );
- };
- const getProductModal = () => {
- return (
- <Modal
- visible
- width={1100}
- footer={null}
- title={renderModalTitle()}
- maskClosable={false}
- onCancel={() => this.handleSelectorCancel('product')}
- >
- <Selector
- multiple={false}
- loading={sLoading}
- selectorName={productType}
- fixedName="Product"
- list={shelves.list}
- pageNo={shelves.pageNo}
- pageSize={shelves.pageSize}
- totalSize={shelves.totalSize}
- onCancel={() => this.handleSelectorCancel('product')}
- onChange={data => this.handleSelectorChange('product', data)}
- onFinish={data => this.handleSelectorFinish('product', data)}
- />
- </Modal>
- );
- };
- /* 封面模态框选择器 */
- const getResourceModal = () => {
- return (
- <Modal
- width={1100}
- footer={null}
- visible
- title="图片资源"
- maskClosable={false}
- onCancel={() => this.handleSelectorCancel('resource')}
- >
- <Selector
- multiple={false}
- loading={rLoading}
- selectorName="PictureSingle"
- list={resource.list}
- pageNo={resource.pageNo}
- pageSize={resource.pageSize}
- totalSize={resource.totalSize}
- onCancel={() => this.handleSelectorCancel('resource')}
- onChange={data => this.handleSelectorChange('resource', data)}
- onFinish={rows => this.handleSelectorFinish('resource', rows)}
- />
- </Modal>
- );
- };
- return (
- <div>
- <Card style={{ marginBottom: 70 }}>
- <Table
- pagination={false}
- loading={mLoading}
- dataSource={posterList}
- columns={posterColumns}
- rowKey={record => record.id}
- className={styles.posterTable}
- />
- <Button
- type="dashed"
- icon="plus"
- style={{ width: '100%', marginTop: 16, marginBottom: 8 }}
- onClick={this.handlePosterItemCreate}
- >新建
- </Button>
- {!resourceSelectorDestroy && getResourceModal()}
- {!productSelectorDestroy && getProductModal()}
- </Card>
- <FooterToolbar style={{ width: '100%' }}>
- <Button type="primary" onClick={this.handlePageBack}>返回上一页</Button>
- </FooterToolbar>
- </div>
- );
- }
- }
|