dec3a32397
Implement comprehensive topup billing system with user history viewing and admin management capabilities.
## Features Added
### Frontend
- Add topup history modal with paginated billing records
- Display order details: trade number, payment method, amount, money, status, create time
- Implement empty state with proper illustrations
- Add payment method column with localized display (Stripe, Alipay, WeChat)
- Add admin manual completion feature for pending orders
- Add Coins icon for recharge amount display
- Integrate "Bills" button in RechargeCard header
- Optimize code quality by using shared utility functions (isAdmin)
- Extract constants for status and payment method mappings
- Use React.useMemo for performance optimization
### Backend
- Create GET `/api/user/topup/self` endpoint for user topup history with pagination
- Create POST `/api/user/topup/complete` endpoint for admin manual order completion
- Add `payment_method` field to TopUp model for tracking payment types
- Implement `GetUserTopUps` method with proper pagination and ordering
- Implement `ManualCompleteTopUp` with transaction safety and row-level locking
- Add application-level mutex locks to prevent concurrent order processing
- Record payment method in Epay and Stripe payment flows
- Ensure idempotency and data consistency with proper error handling
### Internationalization
- Add i18n keys for Chinese (zh), English (en), and French (fr)
- Support for billing-related UI text and status messages
## Technical Improvements
- Use database transactions with FOR UPDATE row-level locking
- Implement sync.Map-based mutex for order-level concurrency control
- Proper error handling and user-friendly toast notifications
- Follow existing codebase patterns for empty states and modals
- Maintain code quality with extracted render functions and constants
## Files Changed
- Backend: controller/topup.go, controller/topup_stripe.go, model/topup.go, router/api-router.go
- Frontend: web/src/components/topup/modals/TopupHistoryModal.jsx (new), web/src/components/topup/RechargeCard.jsx, web/src/components/topup/index.jsx
- i18n: web/src/i18n/locales/{zh,en,fr}.json
114 lines
3.5 KiB
React
114 lines
3.5 KiB
React
/*
|
|
Copyright (C) 2025 QuantumNous
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Affero General Public License as
|
|
published by the Free Software Foundation, either version 3 of the
|
|
License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
For commercial licensing, please contact support@quantumnous.com
|
|
*/
|
|
|
|
import React, { useState } from 'react';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { Button, Modal } from '@douyinfe/semi-ui';
|
|
import { useSecureVerification } from '../../../hooks/common/useSecureVerification';
|
|
import { createApiCalls } from '../../../services/secureVerification';
|
|
import SecureVerificationModal from '../modals/SecureVerificationModal';
|
|
import ChannelKeyDisplay from '../ui/ChannelKeyDisplay';
|
|
|
|
/**
|
|
* 渠道密钥查看组件使用示例
|
|
* 展示如何使用通用安全验证系统
|
|
*/
|
|
const ChannelKeyViewExample = ({ channelId }) => {
|
|
const { t } = useTranslation();
|
|
const [keyData, setKeyData] = useState('');
|
|
const [showKeyModal, setShowKeyModal] = useState(false);
|
|
|
|
// 使用通用安全验证 Hook
|
|
const {
|
|
isModalVisible,
|
|
verificationMethods,
|
|
verificationState,
|
|
startVerification,
|
|
executeVerification,
|
|
cancelVerification,
|
|
setVerificationCode,
|
|
switchVerificationMethod,
|
|
} = useSecureVerification({
|
|
onSuccess: (result) => {
|
|
// 验证成功后处理结果
|
|
if (result.success && result.data?.key) {
|
|
setKeyData(result.data.key);
|
|
setShowKeyModal(true);
|
|
}
|
|
},
|
|
successMessage: t('密钥获取成功'),
|
|
});
|
|
|
|
// 开始查看密钥流程
|
|
const handleViewKey = async () => {
|
|
const apiCall = createApiCalls.viewChannelKey(channelId);
|
|
|
|
await startVerification(apiCall, {
|
|
title: t('查看渠道密钥'),
|
|
description: t('为了保护账户安全,请验证您的身份。'),
|
|
preferredMethod: 'passkey', // 可以指定首选验证方式
|
|
});
|
|
};
|
|
|
|
return (
|
|
<>
|
|
{/* 查看密钥按钮 */}
|
|
<Button type='primary' theme='outline' onClick={handleViewKey}>
|
|
{t('查看密钥')}
|
|
</Button>
|
|
|
|
{/* 安全验证模态框 */}
|
|
<SecureVerificationModal
|
|
visible={isModalVisible}
|
|
verificationMethods={verificationMethods}
|
|
verificationState={verificationState}
|
|
onVerify={executeVerification}
|
|
onCancel={cancelVerification}
|
|
onCodeChange={setVerificationCode}
|
|
onMethodSwitch={switchVerificationMethod}
|
|
title={verificationState.title}
|
|
description={verificationState.description}
|
|
/>
|
|
|
|
{/* 密钥显示模态框 */}
|
|
<Modal
|
|
title={t('渠道密钥信息')}
|
|
visible={showKeyModal}
|
|
onCancel={() => setShowKeyModal(false)}
|
|
footer={
|
|
<Button type='primary' onClick={() => setShowKeyModal(false)}>
|
|
{t('完成')}
|
|
</Button>
|
|
}
|
|
width={700}
|
|
style={{ maxWidth: '90vw' }}
|
|
>
|
|
<ChannelKeyDisplay
|
|
keyData={keyData}
|
|
showSuccessIcon={true}
|
|
successText={t('密钥获取成功')}
|
|
showWarning={true}
|
|
/>
|
|
</Modal>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default ChannelKeyViewExample;
|