|
@@ -0,0 +1,311 @@
|
|
|
+import {
|
|
|
+ createNewActivity,
|
|
|
+ deleteNewActivity,
|
|
|
+ getNewActivities,
|
|
|
+ updateNewActivity,
|
|
|
+} from '@/services/api';
|
|
|
+import { PlusOutlined } from '@ant-design/icons';
|
|
|
+import {
|
|
|
+ ActionType,
|
|
|
+ ModalForm,
|
|
|
+ ProColumns,
|
|
|
+ ProForm,
|
|
|
+ ProFormDigit,
|
|
|
+ ProFormSwitch,
|
|
|
+ ProFormText,
|
|
|
+ ProFormUploadButton,
|
|
|
+ ProTable,
|
|
|
+} from '@ant-design/pro-components';
|
|
|
+import { Button, FormInstance, message, Popconfirm } from 'antd';
|
|
|
+import React, { useCallback, useMemo, useRef, useState } from 'react';
|
|
|
+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 ActivityManagement: React.FC = () => {
|
|
|
+ const [createModalVisible, setCreateModalVisible] = useState<boolean>(false);
|
|
|
+ const [editModalVisible, setEditModalVisible] = useState<boolean>(false);
|
|
|
+ const [currentActivity, setCurrentActivity] = useState<API.NewActivityItem | null>(null);
|
|
|
+ const actionRef = useRef<ActionType>();
|
|
|
+ const formRef = useRef<FormInstance>(null);
|
|
|
+
|
|
|
+ const [iconUrl, setIconUrl] = useState<string>('');
|
|
|
+
|
|
|
+ console.log('currentActivity', currentActivity);
|
|
|
+
|
|
|
+ const columns: ProColumns<API.NewActivityItem>[] = [
|
|
|
+ {
|
|
|
+ title: '活动标题',
|
|
|
+ dataIndex: 'title',
|
|
|
+ copyable: true,
|
|
|
+ ellipsis: true,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '链接',
|
|
|
+ dataIndex: 'link',
|
|
|
+ ellipsis: true,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '排序',
|
|
|
+ dataIndex: 'order',
|
|
|
+ sorter: true,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '是否激活',
|
|
|
+ dataIndex: 'isActive',
|
|
|
+ valueEnum: {
|
|
|
+ true: { text: '是', status: 'Success' },
|
|
|
+ false: { text: '否', status: 'Default' },
|
|
|
+ },
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '操作',
|
|
|
+ valueType: 'option',
|
|
|
+ key: 'option',
|
|
|
+ render: (text, record) => [
|
|
|
+ <a
|
|
|
+ key="edit"
|
|
|
+ onClick={() => {
|
|
|
+ console.log('record', record);
|
|
|
+
|
|
|
+ setCurrentActivity(record);
|
|
|
+ setIconUrl(record.icon);
|
|
|
+ setEditModalVisible(true);
|
|
|
+ }}
|
|
|
+ >
|
|
|
+ 编辑
|
|
|
+ </a>,
|
|
|
+ <Popconfirm
|
|
|
+ key="delete"
|
|
|
+ title="确认删除"
|
|
|
+ description={`您确定要删除活动 "${record.title}" 吗?`}
|
|
|
+ onConfirm={() => handleDelete(record)}
|
|
|
+ okText="确定"
|
|
|
+ cancelText="取消"
|
|
|
+ >
|
|
|
+ <a>删除</a>
|
|
|
+ </Popconfirm>,
|
|
|
+ ],
|
|
|
+ },
|
|
|
+ ];
|
|
|
+
|
|
|
+ const handleAdd = async (fields: API.NewActivityItem) => {
|
|
|
+ try {
|
|
|
+ const response = await createNewActivity(fields);
|
|
|
+ if (response.success) {
|
|
|
+ message.success('创建成功');
|
|
|
+ return true;
|
|
|
+ } else {
|
|
|
+ message.error(response.message || '创建失败,请重试!');
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error('创建活动时出错:', error);
|
|
|
+ message.error('创建失败,请重试!');
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ const handleEdit = async (updateData: API.NewActivityItem) => {
|
|
|
+ try {
|
|
|
+ const response = await updateNewActivity(currentActivity._id, updateData);
|
|
|
+
|
|
|
+ if (response.success) {
|
|
|
+ message.success('更新成功');
|
|
|
+ return true;
|
|
|
+ } else {
|
|
|
+ message.error(response.message || '更新失败,请重试!');
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error('更新活动时出错:', error);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ const handleDelete = async (record: API.NewActivityItem) => {
|
|
|
+ try {
|
|
|
+ const res = await deleteNewActivity(record._id);
|
|
|
+ if (res.success) {
|
|
|
+ message.success('活动已成功删除');
|
|
|
+ actionRef.current?.reload();
|
|
|
+ } else {
|
|
|
+ message.error(res.message || '删除失败,请重试');
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error('删除活动时出错:', error);
|
|
|
+ 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();
|
|
|
+
|
|
|
+ console.log('uploadData', uploadData);
|
|
|
+
|
|
|
+ if (uploadData.success) {
|
|
|
+ setIconUrl(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.NewActivityItem>
|
|
|
+ headerTitle="最新活动"
|
|
|
+ actionRef={actionRef}
|
|
|
+ rowKey="_id"
|
|
|
+ search={false}
|
|
|
+ toolBarRender={() => [
|
|
|
+ <Button type="primary" key="primary" onClick={() => setCreateModalVisible(true)}>
|
|
|
+ <PlusOutlined /> 新建活动
|
|
|
+ </Button>,
|
|
|
+ ]}
|
|
|
+ request={getNewActivities}
|
|
|
+ columns={columns}
|
|
|
+ />
|
|
|
+
|
|
|
+ <ModalForm
|
|
|
+ title="新建活动"
|
|
|
+ width="400px"
|
|
|
+ formRef={formRef}
|
|
|
+ open={createModalVisible}
|
|
|
+ onOpenChange={setCreateModalVisible}
|
|
|
+ onFinish={async (value) => {
|
|
|
+ const formData = {
|
|
|
+ ...value,
|
|
|
+ icon: iconUrl,
|
|
|
+ };
|
|
|
+ console.log('formData', formData);
|
|
|
+ const success = await handleAdd(formData as API.NewActivityItem);
|
|
|
+ if (success) {
|
|
|
+ setCreateModalVisible(false);
|
|
|
+ actionRef.current?.reload();
|
|
|
+ formRef.current?.resetFields();
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ >
|
|
|
+ <ProForm.Item name="title" label="活动标题">
|
|
|
+ <ReactQuill
|
|
|
+ theme="snow"
|
|
|
+ modules={modules}
|
|
|
+ style={{
|
|
|
+ height: '150px',
|
|
|
+ paddingBottom: '60px',
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </ProForm.Item>
|
|
|
+ <ProFormText name="link" label="链接" />
|
|
|
+ <ProFormUploadButton
|
|
|
+ name="icon"
|
|
|
+ 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('图标上传失败');
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ <ProFormDigit name="order" label="排序" min={0} initialValue={0} />
|
|
|
+ <ProFormSwitch name="isActive" label="是否激活" initialValue={true} />
|
|
|
+ </ModalForm>
|
|
|
+
|
|
|
+ <ModalForm
|
|
|
+ key={currentActivity?._id}
|
|
|
+ title="编辑活动"
|
|
|
+ width="400px"
|
|
|
+ open={editModalVisible}
|
|
|
+ onOpenChange={setEditModalVisible}
|
|
|
+ onFinish={async (value) => {
|
|
|
+ console.log('edit value', value);
|
|
|
+ const formData = {
|
|
|
+ ...value,
|
|
|
+ icon: iconUrl,
|
|
|
+ };
|
|
|
+ const success = await handleEdit(formData as API.NewActivityItem);
|
|
|
+ if (success) {
|
|
|
+ setEditModalVisible(false);
|
|
|
+ actionRef.current?.reload();
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ initialValues={currentActivity}
|
|
|
+ >
|
|
|
+ <ProForm.Item name="title" label="活动标题">
|
|
|
+ <ReactQuill
|
|
|
+ theme="snow"
|
|
|
+ modules={modules}
|
|
|
+ style={{
|
|
|
+ height: '150px',
|
|
|
+ paddingBottom: '60px',
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </ProForm.Item>
|
|
|
+ <ProFormText name="link" label="链接" />
|
|
|
+ <ProFormUploadButton
|
|
|
+ name="icon"
|
|
|
+ label="图标"
|
|
|
+ max={1}
|
|
|
+ action={handleUpload}
|
|
|
+ fieldProps={{
|
|
|
+ name: 'file',
|
|
|
+ listType: 'picture-card',
|
|
|
+ }}
|
|
|
+ fileList={useMemo(() => {
|
|
|
+ return iconUrl ? [{ url: `${API_URL}${iconUrl}` }] : [];
|
|
|
+ }, [iconUrl])}
|
|
|
+ onChange={({ fileList }) => {
|
|
|
+ if (fileList.length === 0) {
|
|
|
+ setIconUrl('');
|
|
|
+ }
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ <ProFormDigit name="order" label="排序" min={0} />
|
|
|
+ <ProFormSwitch name="isActive" label="是否激活" />
|
|
|
+ </ModalForm>
|
|
|
+ </>
|
|
|
+ );
|
|
|
+};
|
|
|
+
|
|
|
+export default ActivityManagement;
|