♻️Refactor: Redemptions Page

This commit is contained in:
Apple\Apple
2025-05-23 16:58:19 +08:00
parent 0befa28e8e
commit 9a6c540013
4 changed files with 246 additions and 169 deletions

View File

@@ -11,17 +11,33 @@ import { ITEMS_PER_PAGE } from '../constants';
import { renderQuota } from '../helpers/render'; import { renderQuota } from '../helpers/render';
import { import {
Button, Button,
Card,
Divider, Divider,
Form, Dropdown,
Input,
Modal, Modal,
Popconfirm,
Popover, Popover,
Space,
Table, Table,
Tag, Tag,
Typography,
} from '@douyinfe/semi-ui'; } from '@douyinfe/semi-ui';
import {
IconPlus,
IconCopy,
IconSearch,
IconEyeOpened,
IconEdit,
IconDelete,
IconStop,
IconPlay,
IconMore,
} from '@douyinfe/semi-icons';
import EditRedemption from '../pages/Redemption/EditRedemption'; import EditRedemption from '../pages/Redemption/EditRedemption';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
const { Text } = Typography;
function renderTimestamp(timestamp) { function renderTimestamp(timestamp) {
return <>{timestamp2string(timestamp)}</>; return <>{timestamp2string(timestamp)}</>;
} }
@@ -33,25 +49,25 @@ const RedemptionsTable = () => {
switch (status) { switch (status) {
case 1: case 1:
return ( return (
<Tag color='green' size='large'> <Tag color='green' size='large' shape='circle'>
{t('未使用')} {t('未使用')}
</Tag> </Tag>
); );
case 2: case 2:
return ( return (
<Tag color='red' size='large'> <Tag color='red' size='large' shape='circle'>
{t('已禁用')} {t('已禁用')}
</Tag> </Tag>
); );
case 3: case 3:
return ( return (
<Tag color='grey' size='large'> <Tag color='grey' size='large' shape='circle'>
{t('已使用')} {t('已使用')}
</Tag> </Tag>
); );
default: default:
return ( return (
<Tag color='black' size='large'> <Tag color='black' size='large' shape='circle'>
{t('未知状态')} {t('未知状态')}
</Tag> </Tag>
); );
@@ -99,66 +115,83 @@ const RedemptionsTable = () => {
{ {
title: '', title: '',
dataIndex: 'operate', dataIndex: 'operate',
render: (text, record, index) => ( render: (text, record, index) => {
<div> // 创建更多操作的下拉菜单项
const moreMenuItems = [
{
node: 'item',
name: t('删除'),
icon: <IconDelete />,
type: 'danger',
onClick: () => {
Modal.confirm({
title: t('确定是否要删除此兑换码?'),
content: t('此修改将不可逆'),
onOk: () => {
manageRedemption(record.id, 'delete', record).then(() => {
removeRecord(record.key);
});
},
});
},
}
];
// 动态添加启用/禁用按钮
if (record.status === 1) {
moreMenuItems.push({
node: 'item',
name: t('禁用'),
icon: <IconStop />,
type: 'warning',
onClick: () => {
manageRedemption(record.id, 'disable', record);
},
});
} else {
moreMenuItems.push({
node: 'item',
name: t('启用'),
icon: <IconPlay />,
type: 'secondary',
onClick: () => {
manageRedemption(record.id, 'enable', record);
},
disabled: record.status === 3,
});
}
return (
<Space>
<Popover content={record.key} style={{ padding: 20 }} position='top'> <Popover content={record.key} style={{ padding: 20 }} position='top'>
<Button theme='light' type='tertiary' style={{ marginRight: 1 }}> <Button
icon={<IconEyeOpened />}
theme='light'
type='tertiary'
size="small"
className="!rounded-full"
>
{t('查看')} {t('查看')}
</Button> </Button>
</Popover> </Popover>
<Button <Button
icon={<IconCopy />}
theme='light' theme='light'
type='secondary' type='secondary'
style={{ marginRight: 1 }} size="small"
onClick={async (text) => { className="!rounded-full"
onClick={async () => {
await copyText(record.key); await copyText(record.key);
}} }}
> >
{t('复制')} {t('复制')}
</Button> </Button>
<Popconfirm
title={t('确定是否要删除此兑换码?')}
content={t('此修改将不可逆')}
okType={'danger'}
position={'left'}
onConfirm={() => {
manageRedemption(record.id, 'delete', record).then(() => {
removeRecord(record.key);
});
}}
>
<Button theme='light' type='danger' style={{ marginRight: 1 }}>
{t('删除')}
</Button>
</Popconfirm>
{record.status === 1 ? (
<Button
theme='light'
type='warning'
style={{ marginRight: 1 }}
onClick={async () => {
manageRedemption(record.id, 'disable', record);
}}
>
{t('禁用')}
</Button>
) : (
<Button
theme='light'
type='secondary'
style={{ marginRight: 1 }}
onClick={async () => {
manageRedemption(record.id, 'enable', record);
}}
disabled={record.status === 3}
>
{t('启用')}
</Button>
)}
<Button <Button
icon={<IconEdit />}
theme='light' theme='light'
type='tertiary' type='tertiary'
style={{ marginRight: 1 }} size="small"
className="!rounded-full"
onClick={() => { onClick={() => {
setEditingRedemption(record); setEditingRedemption(record);
setShowEdit(true); setShowEdit(true);
@@ -167,8 +200,22 @@ const RedemptionsTable = () => {
> >
{t('编辑')} {t('编辑')}
</Button> </Button>
</div> <Dropdown
), trigger='click'
position='bottomRight'
menu={moreMenuItems}
>
<Button
icon={<IconMore />}
theme='light'
type='tertiary'
size="small"
className="!rounded-full"
/>
</Dropdown>
</Space>
);
},
}, },
]; ];
@@ -187,6 +234,11 @@ const RedemptionsTable = () => {
const closeEdit = () => { const closeEdit = () => {
setShowEdit(false); setShowEdit(false);
setTimeout(() => {
setEditingRedemption({
id: undefined,
});
}, 500);
}; };
const setRedemptionFormat = (redeptions) => { const setRedemptionFormat = (redeptions) => {
@@ -225,8 +277,11 @@ const RedemptionsTable = () => {
if (await copy(text)) { if (await copy(text)) {
showSuccess(t('已复制到剪贴板!')); showSuccess(t('已复制到剪贴板!'));
} else { } else {
// setSearchKeyword(text); Modal.error({
Modal.error({ title: t('无法复制到剪贴板,请手动复制'), content: text }); title: t('无法复制到剪贴板,请手动复制'),
content: text,
size: 'large'
});
} }
}; };
@@ -245,13 +300,14 @@ const RedemptionsTable = () => {
.catch((reason) => { .catch((reason) => {
showError(reason); showError(reason);
}); });
}, []); }, [pageSize]);
const refresh = async () => { const refresh = async () => {
await loadRedemptions(activePage - 1, pageSize); await loadRedemptions(activePage - 1, pageSize);
}; };
const manageRedemption = async (id, action, record) => { const manageRedemption = async (id, action, record) => {
setLoading(true);
let data = { id }; let data = { id };
let res; let res;
switch (action) { switch (action) {
@@ -272,7 +328,6 @@ const RedemptionsTable = () => {
showSuccess(t('操作成功完成!')); showSuccess(t('操作成功完成!'));
let redemption = res.data.data; let redemption = res.data.data;
let newRedemptions = [...redemptions]; let newRedemptions = [...redemptions];
// let realIdx = (activePage - 1) * ITEMS_PER_PAGE + idx;
if (action === 'delete') { if (action === 'delete') {
} else { } else {
record.status = redemption.status; record.status = redemption.status;
@@ -281,6 +336,7 @@ const RedemptionsTable = () => {
} else { } else {
showError(message); showError(message);
} }
setLoading(false);
}; };
const searchRedemptions = async (keyword, page, pageSize) => { const searchRedemptions = async (keyword, page, pageSize) => {
@@ -352,36 +408,24 @@ const RedemptionsTable = () => {
} }
}; };
return ( const renderHeader = () => (
<> <div className="flex flex-col w-full">
<EditRedemption <div className="mb-2">
refresh={refresh} <div className="flex items-center text-orange-500">
editingRedemption={editingRedemption} <IconEyeOpened className="mr-2" />
visiable={showEdit} <Text>{t('兑换码可以批量生成和分发,适合用于推广活动或批量充值。')}</Text>
handleClose={closeEdit} </div>
></EditRedemption> </div>
<Form
onSubmit={() => { <Divider margin="12px" />
searchRedemptions(searchKeyword, activePage, pageSize).then();
}} <div className="flex flex-col md:flex-row justify-between items-center gap-4 w-full">
> <div className="flex gap-2 w-full md:w-auto order-2 md:order-1">
<Form.Input
label={t('搜索关键字')}
field='keyword'
icon='search'
iconPosition='left'
placeholder={t('关键字(id或者名称)')}
value={searchKeyword}
loading={searching}
onChange={handleKeywordChange}
/>
</Form>
<Divider style={{ margin: '5px 0 15px 0' }} />
<div>
<Button <Button
theme='light' theme='light'
type='primary' type='primary'
style={{ marginRight: 8 }} icon={<IconPlus />}
className="!rounded-full w-full md:w-auto"
onClick={() => { onClick={() => {
setEditingRedemption({ setEditingRedemption({
id: undefined, id: undefined,
@@ -392,8 +436,9 @@ const RedemptionsTable = () => {
{t('添加兑换码')} {t('添加兑换码')}
</Button> </Button>
<Button <Button
label={t('复制所选兑换码')}
type='warning' type='warning'
icon={<IconCopy />}
className="!rounded-full w-full md:w-auto"
onClick={async () => { onClick={async () => {
if (selectedKeys.length === 0) { if (selectedKeys.length === 0) {
showError(t('请至少选择一个兑换码!')); showError(t('请至少选择一个兑换码!'));
@@ -411,8 +456,47 @@ const RedemptionsTable = () => {
</Button> </Button>
</div> </div>
<div className="flex flex-col md:flex-row items-center gap-4 w-full md:w-auto order-1 md:order-2">
<div className="relative w-full md:w-64">
<Input
prefix={<IconSearch />}
placeholder={t('关键字(id或者名称)')}
value={searchKeyword}
onChange={handleKeywordChange}
className="!rounded-full"
showClear
/>
</div>
<Button
type="primary"
onClick={() => {
searchRedemptions(searchKeyword, 1, pageSize).then();
}}
loading={searching}
className="!rounded-full w-full md:w-auto"
>
{t('查询')}
</Button>
</div>
</div>
</div>
);
return (
<>
<EditRedemption
refresh={refresh}
editingRedemption={editingRedemption}
visiable={showEdit}
handleClose={closeEdit}
></EditRedemption>
<Card
className="!rounded-2xl overflow-hidden"
title={renderHeader()}
shadows='hover'
>
<Table <Table
style={{ marginTop: 20 }}
columns={columns} columns={columns}
dataSource={pageData} dataSource={pageData}
pagination={{ pagination={{
@@ -420,7 +504,7 @@ const RedemptionsTable = () => {
pageSize: pageSize, pageSize: pageSize,
total: tokenCount, total: tokenCount,
showSizeChanger: true, showSizeChanger: true,
pageSizeOpts: [10, 20, 50, 100], pageSizeOptions: [10, 20, 50, 100],
formatPageText: (page) => formatPageText: (page) =>
t('第 {{start}} - {{end}} 条,共 {{total}} 条', { t('第 {{start}} - {{end}} 条,共 {{total}} 条', {
start: page.currentStart, start: page.currentStart,
@@ -441,7 +525,10 @@ const RedemptionsTable = () => {
loading={loading} loading={loading}
rowSelection={rowSelection} rowSelection={rowSelection}
onRow={handleRow} onRow={handleRow}
className="rounded-xl overflow-hidden"
size="middle"
></Table> ></Table>
</Card>
</> </>
); );
}; };

View File

@@ -1432,5 +1432,6 @@
"30个": "30 items", "30个": "30 items",
"100个": "100 items", "100个": "100 items",
"Midjourney 任务记录": "Midjourney Task Records", "Midjourney 任务记录": "Midjourney Task Records",
"任务记录": "Task Records" "任务记录": "Task Records",
"兑换码可以批量生成和分发,适合用于推广活动或批量充值。": "Redemption codes can be batch generated and distributed, suitable for promotion activities or bulk recharge."
} }

View File

@@ -1,20 +1,10 @@
import React from 'react'; import React from 'react';
import RedemptionsTable from '../../components/RedemptionsTable'; import RedemptionsTable from '../../components/RedemptionsTable';
import { Layout } from '@douyinfe/semi-ui';
import { useTranslation } from 'react-i18next';
const Redemption = () => { const Redemption = () => {
const { t } = useTranslation();
return ( return (
<> <>
<Layout>
<Layout.Header>
<h3>{t('管理兑换码')}</h3>
</Layout.Header>
<Layout.Content>
<RedemptionsTable /> <RedemptionsTable />
</Layout.Content>
</Layout>
</> </>
); );
}; };

View File

@@ -1,8 +1,7 @@
import React from 'react'; import React from 'react';
import TokensTable from '../../components/TokensTable'; import TokensTable from '../../components/TokensTable';
import { useTranslation } from 'react-i18next';
const Token = () => { const Token = () => {
const { t } = useTranslation();
return ( return (
<> <>
<TokensTable /> <TokensTable />