|
@@ -0,0 +1,341 @@
|
|
|
+import React, { useCallback, useMemo, useRef, useState } from 'react';
|
|
|
+
|
|
|
+import {
|
|
|
+ createExchangeItem,
|
|
|
+ deleteExchangeItem,
|
|
|
+ getExchangeItems,
|
|
|
+ updateExchangeItem,
|
|
|
+} from '@/services/api';
|
|
|
+import { PlusOutlined } from '@ant-design/icons';
|
|
|
+import {
|
|
|
+ ActionType,
|
|
|
+ ModalForm,
|
|
|
+ ProColumns,
|
|
|
+ ProForm,
|
|
|
+ ProFormDigit,
|
|
|
+ ProFormSelect,
|
|
|
+ ProFormUploadButton,
|
|
|
+ ProTable,
|
|
|
+} from '@ant-design/pro-components';
|
|
|
+import { Button, FormInstance, message, Popconfirm } from 'antd';
|
|
|
+
|
|
|
+import ReactQuill from 'react-quill';
|
|
|
+import 'react-quill/dist/quill.snow.css';
|
|
|
+
|
|
|
+const modules = {
|
|
|
+ toolbar: [
|
|
|
+ [{ header: [1, 2, false] }],
|
|
|
+ ['bold', 'italic', 'underline', 'strike', 'blockquote'],
|
|
|
+ [{ list: 'ordered' }, { list: 'bullet' }, { indent: '-1' }, { indent: '+1' }],
|
|
|
+ ['link', 'image'],
|
|
|
+ [{ align: [] }],
|
|
|
+ [{ color: [] }, { background: [] }],
|
|
|
+ ],
|
|
|
+};
|
|
|
+
|
|
|
+const stripHtml = (html: string) => {
|
|
|
+ return html?.replace(/<[^>]+>/g, '') || '';
|
|
|
+};
|
|
|
+
|
|
|
+const ExchangeItemManagement: React.FC = () => {
|
|
|
+ const [createModalVisible, setCreateModalVisible] = useState<boolean>(false);
|
|
|
+ const [editModalVisible, setEditModalVisible] = useState<boolean>(false);
|
|
|
+ const [currentItem, setCurrentItem] = useState<API.ExchangeItem | null>(null);
|
|
|
+ const actionRef = useRef<ActionType>();
|
|
|
+ const formRef = useRef<FormInstance>(null);
|
|
|
+
|
|
|
+ const [logoUrl, setLogoUrl] = useState<string>('');
|
|
|
+
|
|
|
+ const columns: ProColumns<API.ExchangeItem>[] = [
|
|
|
+ {
|
|
|
+ title: '类型',
|
|
|
+ dataIndex: 'type',
|
|
|
+ valueEnum: {
|
|
|
+ 彩金: { text: '彩金' },
|
|
|
+ 优惠券: { text: '优惠券' },
|
|
|
+ 实物: { text: '实物' },
|
|
|
+ },
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '标题',
|
|
|
+ dataIndex: 'title',
|
|
|
+ copyable: true,
|
|
|
+ ellipsis: true,
|
|
|
+ render: (_, record) => {
|
|
|
+ return stripHtml(record.title);
|
|
|
+ },
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '所需积分',
|
|
|
+ dataIndex: 'points',
|
|
|
+ sorter: true,
|
|
|
+ },
|
|
|
+ // {
|
|
|
+ // title: '图标',
|
|
|
+ // dataIndex: 'logo',
|
|
|
+ // valueType: 'image',
|
|
|
+ // },
|
|
|
+ {
|
|
|
+ title: '操作',
|
|
|
+ valueType: 'option',
|
|
|
+ key: 'option',
|
|
|
+ render: (text, record) => [
|
|
|
+ <a
|
|
|
+ key="edit"
|
|
|
+ onClick={() => {
|
|
|
+ setCurrentItem(record);
|
|
|
+ setLogoUrl(record.logo);
|
|
|
+ setEditModalVisible(true);
|
|
|
+ }}
|
|
|
+ >
|
|
|
+ 编辑
|
|
|
+ </a>,
|
|
|
+ <Popconfirm
|
|
|
+ key="delete"
|
|
|
+ title="确认删除"
|
|
|
+ description={`您确定要删除兑换项目 "${record.title}" 吗?`}
|
|
|
+ onConfirm={() => handleDelete(record)}
|
|
|
+ okText="确定"
|
|
|
+ cancelText="取消"
|
|
|
+ >
|
|
|
+ <a>删除</a>
|
|
|
+ </Popconfirm>,
|
|
|
+ ],
|
|
|
+ },
|
|
|
+ ];
|
|
|
+
|
|
|
+ const handleAdd = async (fields: API.ExchangeItem) => {
|
|
|
+ try {
|
|
|
+ await createExchangeItem({ ...fields });
|
|
|
+ message.success('创建成功');
|
|
|
+ return true;
|
|
|
+ } catch (error) {
|
|
|
+ message.error('创建失败,请重试!');
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ const handleEdit = async (updateData: API.ExchangeItem) => {
|
|
|
+ try {
|
|
|
+ const response = await updateExchangeItem(currentItem._id, updateData);
|
|
|
+
|
|
|
+ if (response.success) {
|
|
|
+ message.success('更新成功');
|
|
|
+ return true;
|
|
|
+ } else {
|
|
|
+ message.error(response.message || '更新失败,请重试!');
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error('更新兑换项目时出错:', error);
|
|
|
+ message.error('更新失败,请重试!');
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ const handleDelete = async (record: API.ExchangeItem) => {
|
|
|
+ const res = await deleteExchangeItem(record._id);
|
|
|
+ if (res.success) {
|
|
|
+ message.success('兑换项目已成功删除');
|
|
|
+ actionRef.current?.reload();
|
|
|
+ } else {
|
|
|
+ message.error('删除失败,请重试');
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ const handleUpload = useCallback(async (file) => {
|
|
|
+ const formData = new FormData();
|
|
|
+ formData.append('file', file);
|
|
|
+
|
|
|
+ try {
|
|
|
+ const uploadResponse = await fetch(`${API_URL}/api/upload`, {
|
|
|
+ method: 'POST',
|
|
|
+ body: formData,
|
|
|
+ });
|
|
|
+
|
|
|
+ if (!uploadResponse.ok) {
|
|
|
+ throw new Error(`Upload failed with status ${uploadResponse.status}`);
|
|
|
+ }
|
|
|
+
|
|
|
+ const uploadData = await uploadResponse.json();
|
|
|
+
|
|
|
+ if (uploadData.success) {
|
|
|
+ setLogoUrl(uploadData.url);
|
|
|
+ return {
|
|
|
+ status: 'done',
|
|
|
+ url: uploadData.url,
|
|
|
+ };
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error('Upload error:', error);
|
|
|
+ message.error(`文件上传失败: ${error.message}`);
|
|
|
+ return {
|
|
|
+ status: 'error',
|
|
|
+ error: error.message,
|
|
|
+ };
|
|
|
+ }
|
|
|
+ }, []);
|
|
|
+
|
|
|
+ return (
|
|
|
+ <>
|
|
|
+ <ProTable<API.ExchangeItem>
|
|
|
+ headerTitle="兑换项目"
|
|
|
+ actionRef={actionRef}
|
|
|
+ rowKey="id"
|
|
|
+ search={false}
|
|
|
+ toolBarRender={() => [
|
|
|
+ <Button type="primary" key="primary" onClick={() => setCreateModalVisible(true)}>
|
|
|
+ <PlusOutlined /> 新建兑换项目
|
|
|
+ </Button>,
|
|
|
+ ]}
|
|
|
+ request={getExchangeItems}
|
|
|
+ columns={columns}
|
|
|
+ />
|
|
|
+ {/* 新建兑换项目 */}
|
|
|
+ <ModalForm
|
|
|
+ title="新建兑换项目"
|
|
|
+ width="400px"
|
|
|
+ formRef={formRef}
|
|
|
+ open={createModalVisible}
|
|
|
+ onOpenChange={setCreateModalVisible}
|
|
|
+ onFinish={async (value) => {
|
|
|
+ const formData = {
|
|
|
+ ...value,
|
|
|
+ logo: logoUrl,
|
|
|
+ };
|
|
|
+ const success = await handleAdd(formData as API.ExchangeItem);
|
|
|
+ if (success) {
|
|
|
+ setCreateModalVisible(false);
|
|
|
+ if (actionRef.current) {
|
|
|
+ actionRef.current.reload();
|
|
|
+ }
|
|
|
+ formRef.current?.resetFields();
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ >
|
|
|
+ <ProFormSelect
|
|
|
+ name="type"
|
|
|
+ label="类型"
|
|
|
+ options={[
|
|
|
+ { label: '彩金', value: '彩金' },
|
|
|
+ { label: '优惠券', value: '优惠券' },
|
|
|
+ { label: '实物', value: '实物' },
|
|
|
+ ]}
|
|
|
+ rules={[{ required: true, message: '请选择类型' }]}
|
|
|
+ />
|
|
|
+ {/* <ProFormText
|
|
|
+ name="title"
|
|
|
+ label="标题"
|
|
|
+ rules={[{ required: true, message: '请输入标题' }]}
|
|
|
+ /> */}
|
|
|
+ <ProForm.Item name="title" label="标题" rules={[{ required: true, message: '请输入标题' }]}>
|
|
|
+ <ReactQuill
|
|
|
+ theme="snow"
|
|
|
+ modules={modules}
|
|
|
+ style={{
|
|
|
+ height: '150px',
|
|
|
+ paddingBottom: '60px',
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </ProForm.Item>
|
|
|
+ <ProFormDigit
|
|
|
+ name="points"
|
|
|
+ label="所需积分"
|
|
|
+ rules={[{ required: true, message: '请输入所需积分' }]}
|
|
|
+ />
|
|
|
+ <ProFormUploadButton
|
|
|
+ name="logo"
|
|
|
+ label="图标"
|
|
|
+ max={1}
|
|
|
+ action={handleUpload}
|
|
|
+ fieldProps={{
|
|
|
+ name: 'file',
|
|
|
+ listType: 'picture-card',
|
|
|
+ }}
|
|
|
+ onChange={(info) => {
|
|
|
+ const { status } = info.file;
|
|
|
+ if (status === 'done') {
|
|
|
+ message.success('图片上传成功');
|
|
|
+ } else if (status === 'error') {
|
|
|
+ message.error('图片上传失败');
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </ModalForm>
|
|
|
+
|
|
|
+ {/* 编辑兑换项目 */}
|
|
|
+ <ModalForm
|
|
|
+ key={currentItem?._id}
|
|
|
+ title="编辑兑换项目"
|
|
|
+ width="400px"
|
|
|
+ open={editModalVisible}
|
|
|
+ onOpenChange={setEditModalVisible}
|
|
|
+ onFinish={async (value) => {
|
|
|
+ const formData = {
|
|
|
+ ...value,
|
|
|
+ logo: logoUrl,
|
|
|
+ };
|
|
|
+ const success = await handleEdit(formData as API.ExchangeItem);
|
|
|
+ if (success) {
|
|
|
+ setEditModalVisible(false);
|
|
|
+ if (actionRef.current) {
|
|
|
+ actionRef.current.reload();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ initialValues={currentItem}
|
|
|
+ >
|
|
|
+ <ProFormSelect
|
|
|
+ name="type"
|
|
|
+ label="类型"
|
|
|
+ options={[
|
|
|
+ { label: '彩金', value: '彩金' },
|
|
|
+ { label: '优惠券', value: '优惠券' },
|
|
|
+ { label: '实物', value: '实物' },
|
|
|
+ ]}
|
|
|
+ rules={[{ required: true, message: '请选择类型' }]}
|
|
|
+ />
|
|
|
+ {/* <ProFormText
|
|
|
+ name="title"
|
|
|
+ label="标题"
|
|
|
+ rules={[{ required: true, message: '请输入标题' }]}
|
|
|
+ /> */}
|
|
|
+ <ProForm.Item name="title" label="标题" rules={[{ required: true, message: '请输入标题' }]}>
|
|
|
+ <ReactQuill
|
|
|
+ theme="snow"
|
|
|
+ modules={modules}
|
|
|
+ style={{
|
|
|
+ height: '150px', // 增加高度
|
|
|
+ paddingBottom: '60px',
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </ProForm.Item>
|
|
|
+ <ProFormDigit
|
|
|
+ name="points"
|
|
|
+ label="所需积分"
|
|
|
+ rules={[{ required: true, message: '请输入所需积分' }]}
|
|
|
+ />
|
|
|
+ <ProFormUploadButton
|
|
|
+ name="logo"
|
|
|
+ label="图标"
|
|
|
+ max={1}
|
|
|
+ action={handleUpload}
|
|
|
+ fieldProps={{
|
|
|
+ name: 'file',
|
|
|
+ listType: 'picture-card',
|
|
|
+ }}
|
|
|
+ fileList={useMemo(() => {
|
|
|
+ return logoUrl ? [{ url: `${API_URL}${logoUrl}` }] : [];
|
|
|
+ }, [logoUrl])}
|
|
|
+ onChange={({ fileList }) => {
|
|
|
+ if (fileList.length === 0) {
|
|
|
+ setLogoUrl('');
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </ModalForm>
|
|
|
+ </>
|
|
|
+ );
|
|
|
+};
|
|
|
+
|
|
|
+export default ExchangeItemManagement;
|