feat: 渠道展示、订阅套餐、系统配置全功能

- 新增 Channel / SubscriptionPlan / SystemConfig 三个数据模型
- Order 模型扩展支持订阅订单(order_type, plan_id, subscription_group_id)
- Sub2API client 新增分组查询、订阅分配/续期、用户订阅查询
- 订单服务支持订阅履约流程(CAS 锁 + 分组消失安全处理)
- 管理后台:渠道管理、订阅套餐管理、系统配置、Sub2API 分组同步
- 用户页面:双 Tab UI(按量付费/包月订阅)、渠道卡片、充值弹窗、订阅确认
- PaymentForm 支持 fixedAmount 固定金额模式
- 订单状态 API 返回 failedReason 用于订阅异常展示
- 数据库迁移脚本
This commit is contained in:
erio
2026-03-13 19:06:25 +08:00
parent 9f621713c3
commit eafb7e49fa
38 changed files with 5376 additions and 289 deletions

View File

@@ -0,0 +1,958 @@
'use client';
import { useSearchParams } from 'next/navigation';
import { useState, useEffect, useCallback, Suspense } from 'react';
import PayPageLayout from '@/components/PayPageLayout';
import { resolveLocale, type Locale } from '@/lib/locale';
// ── Types ──
interface Channel {
id: string;
groupId: number;
name: string;
platform: string;
rateMultiplier: number;
description: string | null;
models: string | null;
features: string | null;
sortOrder: number;
enabled: boolean;
groupExists: boolean;
createdAt: string;
updatedAt: string;
}
interface Sub2ApiGroup {
id: number;
name: string;
description: string;
platform: string;
status: string;
rate_multiplier: number;
}
interface ChannelFormData {
group_id: number | '';
name: string;
platform: string;
rate_multiplier: string;
description: string;
models: string;
features: string;
sort_order: string;
enabled: boolean;
}
const PLATFORMS = ['claude', 'openai', 'gemini', 'codex', 'sora'] as const;
const PLATFORM_COLORS: Record<string, { bg: string; text: string }> = {
claude: { bg: 'bg-orange-100 dark:bg-orange-900/40', text: 'text-orange-700 dark:text-orange-300' },
openai: { bg: 'bg-green-100 dark:bg-green-900/40', text: 'text-green-700 dark:text-green-300' },
gemini: { bg: 'bg-blue-100 dark:bg-blue-900/40', text: 'text-blue-700 dark:text-blue-300' },
codex: { bg: 'bg-purple-100 dark:bg-purple-900/40', text: 'text-purple-700 dark:text-purple-300' },
sora: { bg: 'bg-pink-100 dark:bg-pink-900/40', text: 'text-pink-700 dark:text-pink-300' },
};
// ── i18n ──
function getTexts(locale: Locale) {
return locale === 'en'
? {
missingToken: 'Missing admin token',
missingTokenHint: 'Please access the admin page from the Sub2API platform.',
invalidToken: 'Invalid admin token',
title: 'Channel Management',
subtitle: 'Configure and manage subscription channels',
orders: 'Orders',
refresh: 'Refresh',
loading: 'Loading...',
noChannels: 'No channels found',
noChannelsHint: 'Click "Sync from Sub2API" or "New Channel" to get started.',
syncFromSub2Api: 'Sync from Sub2API',
newChannel: 'New Channel',
editChannel: 'Edit Channel',
colName: 'Name',
colPlatform: 'Platform',
colRate: 'Rate',
colSub2ApiStatus: 'Sub2API Status',
colSortOrder: 'Sort',
colEnabled: 'Enabled',
colActions: 'Actions',
edit: 'Edit',
delete: 'Delete',
deleteConfirm: 'Are you sure you want to delete this channel?',
fieldName: 'Channel Name',
fieldPlatform: 'Category',
fieldRate: 'Rate Multiplier',
fieldRateHint: 'e.g. 0.15 means 0.15x',
fieldDescription: 'Description',
fieldModels: 'Supported Models (one per line)',
fieldFeatures: 'Features (one per line)',
fieldSortOrder: 'Sort Order',
fieldEnabled: 'Enable Channel',
fieldGroupId: 'Sub2API Group ID',
cancel: 'Cancel',
save: 'Save',
saving: 'Saving...',
syncTitle: 'Sync from Sub2API',
syncHint: 'Select groups to import as channels',
syncLoading: 'Loading groups...',
syncNoGroups: 'No groups found in Sub2API',
syncAlreadyExists: 'Already imported',
syncImport: 'Import Selected',
syncImporting: 'Importing...',
loadFailed: 'Failed to load channels',
saveFailed: 'Failed to save channel',
deleteFailed: 'Failed to delete channel',
syncFetchFailed: 'Failed to fetch Sub2API groups',
syncImportFailed: 'Failed to import groups',
syncImportSuccess: (n: number) => `Successfully imported ${n} channel(s)`,
yes: 'Yes',
no: 'No',
}
: {
missingToken: '缺少管理员凭证',
missingTokenHint: '请从 Sub2API 平台正确访问管理页面',
invalidToken: '管理员凭证无效',
title: '渠道管理',
subtitle: '配置和管理订阅渠道',
orders: '订单管理',
refresh: '刷新',
loading: '加载中...',
noChannels: '暂无渠道',
noChannelsHint: '点击「从 Sub2API 同步」或「新建渠道」开始创建。',
syncFromSub2Api: '从 Sub2API 同步',
newChannel: '新建渠道',
editChannel: '编辑渠道',
colName: '名称',
colPlatform: '平台',
colRate: '倍率',
colSub2ApiStatus: 'Sub2API 状态',
colSortOrder: '排序',
colEnabled: '启用',
colActions: '操作',
edit: '编辑',
delete: '删除',
deleteConfirm: '确定要删除该渠道吗?',
fieldName: '渠道名称',
fieldPlatform: '分类',
fieldRate: '倍率',
fieldRateHint: '如 0.15 表示 0.15 倍',
fieldDescription: '描述',
fieldModels: '支持模型(每行一个)',
fieldFeatures: '功能特性(每行一个)',
fieldSortOrder: '排序',
fieldEnabled: '启用渠道',
fieldGroupId: 'Sub2API 分组 ID',
cancel: '取消',
save: '保存',
saving: '保存中...',
syncTitle: '从 Sub2API 同步',
syncHint: '选择要导入为渠道的分组',
syncLoading: '加载分组中...',
syncNoGroups: 'Sub2API 中没有找到分组',
syncAlreadyExists: '已导入',
syncImport: '导入所选',
syncImporting: '导入中...',
loadFailed: '加载渠道列表失败',
saveFailed: '保存渠道失败',
deleteFailed: '删除渠道失败',
syncFetchFailed: '获取 Sub2API 分组列表失败',
syncImportFailed: '导入分组失败',
syncImportSuccess: (n: number) => `成功导入 ${n} 个渠道`,
yes: '是',
no: '否',
};
}
// ── Helpers ──
function parseJsonArray(value: string | null): string[] {
if (!value) return [];
try {
const parsed = JSON.parse(value);
return Array.isArray(parsed) ? parsed : [];
} catch {
return [];
}
}
function arrayToLines(value: string | null): string {
return parseJsonArray(value).join('\n');
}
function linesToJsonString(lines: string): string {
const arr = lines
.split('\n')
.map((l) => l.trim())
.filter(Boolean);
return JSON.stringify(arr);
}
const emptyForm: ChannelFormData = {
group_id: '',
name: '',
platform: 'claude',
rate_multiplier: '1',
description: '',
models: '',
features: '',
sort_order: '0',
enabled: true,
};
// ── Main Content ──
function ChannelsContent() {
const searchParams = useSearchParams();
const token = searchParams.get('token') || '';
const theme = searchParams.get('theme') === 'dark' ? 'dark' : 'light';
const uiMode = searchParams.get('ui_mode') || 'standalone';
const locale = resolveLocale(searchParams.get('lang'));
const isDark = theme === 'dark';
const isEmbedded = uiMode === 'embedded';
const t = getTexts(locale);
const [channels, setChannels] = useState<Channel[]>([]);
const [loading, setLoading] = useState(true);
const [error, setError] = useState('');
// Edit modal state
const [editModalOpen, setEditModalOpen] = useState(false);
const [editingChannel, setEditingChannel] = useState<Channel | null>(null);
const [form, setForm] = useState<ChannelFormData>(emptyForm);
const [saving, setSaving] = useState(false);
// Sync modal state
const [syncModalOpen, setSyncModalOpen] = useState(false);
const [syncGroups, setSyncGroups] = useState<Sub2ApiGroup[]>([]);
const [syncLoading, setSyncLoading] = useState(false);
const [syncSelected, setSyncSelected] = useState<Set<number>>(new Set());
const [syncImporting, setSyncImporting] = useState(false);
// ── Fetch channels ──
const fetchChannels = useCallback(async () => {
if (!token) return;
setLoading(true);
try {
const res = await fetch(`/api/admin/channels?token=${encodeURIComponent(token)}`);
if (!res.ok) {
if (res.status === 401) {
setError(t.invalidToken);
return;
}
throw new Error();
}
const data = await res.json();
setChannels(data.channels);
} catch {
setError(t.loadFailed);
} finally {
setLoading(false);
}
}, [token]);
useEffect(() => {
fetchChannels();
}, [fetchChannels]);
// ── Missing token ──
if (!token) {
return (
<div className={`flex min-h-screen items-center justify-center p-4 ${isDark ? 'bg-slate-950' : 'bg-slate-50'}`}>
<div className="text-center text-red-500">
<p className="text-lg font-medium">{t.missingToken}</p>
<p className="mt-2 text-sm text-gray-500">{t.missingTokenHint}</p>
</div>
</div>
);
}
// ── Edit modal handlers ──
const openCreateModal = () => {
setEditingChannel(null);
setForm(emptyForm);
setEditModalOpen(true);
};
const openEditModal = (channel: Channel) => {
setEditingChannel(channel);
setForm({
group_id: channel.groupId,
name: channel.name,
platform: channel.platform,
rate_multiplier: String(channel.rateMultiplier),
description: channel.description ?? '',
models: arrayToLines(channel.models),
features: arrayToLines(channel.features),
sort_order: String(channel.sortOrder),
enabled: channel.enabled,
});
setEditModalOpen(true);
};
const closeEditModal = () => {
setEditModalOpen(false);
setEditingChannel(null);
};
const handleSave = async () => {
if (!form.name.trim() || form.group_id === '' || !form.rate_multiplier) return;
setSaving(true);
setError('');
const body = {
group_id: Number(form.group_id),
name: form.name.trim(),
platform: form.platform,
rate_multiplier: parseFloat(form.rate_multiplier),
description: form.description.trim() || null,
models: form.models.trim() ? linesToJsonString(form.models) : null,
features: form.features.trim() ? linesToJsonString(form.features) : null,
sort_order: parseInt(form.sort_order, 10) || 0,
enabled: form.enabled,
};
try {
const url = editingChannel ? `/api/admin/channels/${editingChannel.id}` : '/api/admin/channels';
const method = editingChannel ? 'PUT' : 'POST';
const res = await fetch(url, {
method,
headers: {
'Content-Type': 'application/json',
Authorization: `Bearer ${token}`,
},
body: JSON.stringify(body),
});
if (!res.ok) {
const data = await res.json().catch(() => ({}));
setError(data.error || t.saveFailed);
return;
}
closeEditModal();
fetchChannels();
} catch {
setError(t.saveFailed);
} finally {
setSaving(false);
}
};
// ── Delete handler ──
const handleDelete = async (channel: Channel) => {
if (!confirm(t.deleteConfirm)) return;
try {
const res = await fetch(`/api/admin/channels/${channel.id}`, {
method: 'DELETE',
headers: { Authorization: `Bearer ${token}` },
});
if (!res.ok) {
const data = await res.json().catch(() => ({}));
setError(data.error || t.deleteFailed);
return;
}
fetchChannels();
} catch {
setError(t.deleteFailed);
}
};
// ── Toggle enabled ──
const handleToggleEnabled = async (channel: Channel) => {
try {
const res = await fetch(`/api/admin/channels/${channel.id}`, {
method: 'PUT',
headers: {
'Content-Type': 'application/json',
Authorization: `Bearer ${token}`,
},
body: JSON.stringify({ enabled: !channel.enabled }),
});
if (res.ok) {
setChannels((prev) => prev.map((c) => (c.id === channel.id ? { ...c, enabled: !c.enabled } : c)));
}
} catch {
/* ignore */
}
};
// ── Sync modal handlers ──
const openSyncModal = async () => {
setSyncModalOpen(true);
setSyncLoading(true);
setSyncSelected(new Set());
try {
const res = await fetch(`/api/admin/sub2api/groups?token=${encodeURIComponent(token)}`);
if (!res.ok) throw new Error();
const data = await res.json();
setSyncGroups(data.groups ?? []);
} catch {
setError(t.syncFetchFailed);
setSyncModalOpen(false);
} finally {
setSyncLoading(false);
}
};
const closeSyncModal = () => {
setSyncModalOpen(false);
setSyncGroups([]);
setSyncSelected(new Set());
};
const existingGroupIds = new Set(channels.map((c) => c.groupId));
const toggleSyncGroup = (id: number) => {
setSyncSelected((prev) => {
const next = new Set(prev);
if (next.has(id)) next.delete(id);
else next.add(id);
return next;
});
};
const handleSyncImport = async () => {
if (syncSelected.size === 0) return;
setSyncImporting(true);
setError('');
let successCount = 0;
for (const groupId of syncSelected) {
const group = syncGroups.find((g) => g.id === groupId);
if (!group) continue;
try {
const res = await fetch('/api/admin/channels', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
Authorization: `Bearer ${token}`,
},
body: JSON.stringify({
group_id: group.id,
name: group.name,
platform: group.platform || 'claude',
rate_multiplier: group.rate_multiplier ?? 1,
description: group.description || null,
sort_order: 0,
enabled: true,
}),
});
if (res.ok) successCount++;
} catch {
/* continue with remaining */
}
}
setSyncImporting(false);
closeSyncModal();
if (successCount > 0) {
fetchChannels();
} else {
setError(t.syncImportFailed);
}
};
// ── Nav params ──
const navParams = new URLSearchParams();
if (token) navParams.set('token', token);
if (locale === 'en') navParams.set('lang', 'en');
if (isDark) navParams.set('theme', 'dark');
if (isEmbedded) navParams.set('ui_mode', 'embedded');
const btnBase = [
'inline-flex items-center rounded-lg border px-3 py-1.5 text-xs font-medium transition-colors',
isDark
? 'border-slate-600 text-slate-200 hover:bg-slate-800'
: 'border-slate-300 text-slate-700 hover:bg-slate-100',
].join(' ');
// ── Shared input classes ──
const inputCls = [
'w-full rounded-lg border px-3 py-2 text-sm transition-colors focus:outline-none focus:ring-2 focus:ring-emerald-500/50',
isDark
? 'border-slate-600 bg-slate-700 text-slate-100 placeholder-slate-400'
: 'border-slate-300 bg-white text-slate-900 placeholder-slate-400',
].join(' ');
const labelCls = ['block text-sm font-medium mb-1', isDark ? 'text-slate-300' : 'text-slate-700'].join(' ');
// ── Render ──
return (
<PayPageLayout
isDark={isDark}
isEmbedded={isEmbedded}
maxWidth="full"
title={t.title}
subtitle={t.subtitle}
locale={locale}
actions={
<>
<a href={`/admin?${navParams}`} className={btnBase}>
{t.orders}
</a>
<button type="button" onClick={fetchChannels} className={btnBase}>
{t.refresh}
</button>
<button
type="button"
onClick={openSyncModal}
className="inline-flex items-center rounded-lg border border-indigo-500 bg-indigo-500 px-3 py-1.5 text-xs font-medium text-white transition-colors hover:bg-indigo-600"
>
{t.syncFromSub2Api}
</button>
<button
type="button"
onClick={openCreateModal}
className="inline-flex items-center rounded-lg border border-emerald-500 bg-emerald-500 px-3 py-1.5 text-xs font-medium text-white transition-colors hover:bg-emerald-600"
>
{t.newChannel}
</button>
</>
}
>
{/* Error banner */}
{error && (
<div
className={`mb-4 rounded-lg border p-3 text-sm ${isDark ? 'border-red-800 bg-red-950/50 text-red-400' : 'border-red-200 bg-red-50 text-red-600'}`}
>
{error}
<button onClick={() => setError('')} className="ml-2 opacity-60 hover:opacity-100">
</button>
</div>
)}
{/* Channel table */}
<div
className={[
'overflow-x-auto rounded-xl border',
isDark ? 'border-slate-700 bg-slate-800/70' : 'border-slate-200 bg-white shadow-sm',
].join(' ')}
>
{loading ? (
<div className={`py-12 text-center ${isDark ? 'text-slate-400' : 'text-gray-500'}`}>{t.loading}</div>
) : channels.length === 0 ? (
<div className={`py-12 text-center ${isDark ? 'text-slate-400' : 'text-gray-500'}`}>
<p className="text-base font-medium">{t.noChannels}</p>
<p className="mt-1 text-sm opacity-70">{t.noChannelsHint}</p>
</div>
) : (
<table className="w-full text-sm">
<thead>
<tr className={isDark ? 'border-b border-slate-700 text-slate-400' : 'border-b border-slate-200 text-slate-500'}>
<th className="px-4 py-3 text-left font-medium">{t.colName}</th>
<th className="px-4 py-3 text-left font-medium">{t.colPlatform}</th>
<th className="px-4 py-3 text-left font-medium">{t.colRate}</th>
<th className="px-4 py-3 text-center font-medium">{t.colSub2ApiStatus}</th>
<th className="px-4 py-3 text-center font-medium">{t.colSortOrder}</th>
<th className="px-4 py-3 text-center font-medium">{t.colEnabled}</th>
<th className="px-4 py-3 text-right font-medium">{t.colActions}</th>
</tr>
</thead>
<tbody>
{channels.map((channel) => {
const pc = PLATFORM_COLORS[channel.platform] ?? PLATFORM_COLORS.claude;
return (
<tr
key={channel.id}
className={[
'border-b transition-colors',
isDark ? 'border-slate-700/50 hover:bg-slate-700/30' : 'border-slate-100 hover:bg-slate-50',
].join(' ')}
>
<td className={`px-4 py-3 font-medium ${isDark ? 'text-slate-100' : 'text-slate-900'}`}>
<div>{channel.name}</div>
<div className={`text-xs ${isDark ? 'text-slate-500' : 'text-slate-400'}`}>
Group #{channel.groupId}
</div>
</td>
<td className="px-4 py-3">
<span
className={`inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium ${isDark ? pc.bg.replace('dark:', '') : pc.bg.split(' ')[0]} ${isDark ? pc.text.replace('dark:', '') : pc.text.split(' ')[0]}`}
>
{channel.platform}
</span>
</td>
<td className={`px-4 py-3 ${isDark ? 'text-slate-300' : 'text-slate-700'}`}>
{channel.rateMultiplier}x
</td>
<td className="px-4 py-3 text-center">
{channel.groupExists ? (
<span className="inline-flex h-5 w-5 items-center justify-center rounded-full bg-emerald-100 text-emerald-600 dark:bg-emerald-900/40 dark:text-emerald-400">
<svg className="h-3.5 w-3.5" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2.5}>
<path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
</svg>
</span>
) : (
<span className="inline-flex h-5 w-5 items-center justify-center rounded-full bg-red-100 text-red-600 dark:bg-red-900/40 dark:text-red-400">
<svg className="h-3.5 w-3.5" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2.5}>
<path strokeLinecap="round" strokeLinejoin="round" d="M6 18L18 6M6 6l12 12" />
</svg>
</span>
)}
</td>
<td className={`px-4 py-3 text-center ${isDark ? 'text-slate-400' : 'text-slate-500'}`}>
{channel.sortOrder}
</td>
<td className="px-4 py-3 text-center">
<button
type="button"
onClick={() => handleToggleEnabled(channel)}
className={[
'relative inline-flex h-5 w-9 items-center rounded-full transition-colors',
channel.enabled ? 'bg-emerald-500' : isDark ? 'bg-slate-600' : 'bg-slate-300',
].join(' ')}
>
<span
className={[
'inline-block h-3.5 w-3.5 rounded-full bg-white transition-transform',
channel.enabled ? 'translate-x-4.5' : 'translate-x-0.5',
].join(' ')}
/>
</button>
</td>
<td className="px-4 py-3 text-right">
<div className="inline-flex gap-1">
<button
type="button"
onClick={() => openEditModal(channel)}
className={[
'rounded-md px-2 py-1 text-xs font-medium transition-colors',
isDark ? 'text-indigo-400 hover:bg-indigo-500/20' : 'text-indigo-600 hover:bg-indigo-50',
].join(' ')}
>
{t.edit}
</button>
<button
type="button"
onClick={() => handleDelete(channel)}
className={[
'rounded-md px-2 py-1 text-xs font-medium transition-colors',
isDark ? 'text-red-400 hover:bg-red-500/20' : 'text-red-600 hover:bg-red-50',
].join(' ')}
>
{t.delete}
</button>
</div>
</td>
</tr>
);
})}
</tbody>
</table>
)}
</div>
{/* ── Edit / Create Modal ── */}
{editModalOpen && (
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/50 p-4" onClick={closeEditModal}>
<div
className={[
'relative w-full max-w-lg overflow-y-auto rounded-2xl border p-6 shadow-2xl',
isDark ? 'border-slate-700 bg-slate-800' : 'border-slate-200 bg-white',
].join(' ')}
style={{ maxHeight: '90vh' }}
onClick={(e) => e.stopPropagation()}
>
<h2 className={`mb-5 text-lg font-semibold ${isDark ? 'text-slate-100' : 'text-slate-900'}`}>
{editingChannel ? t.editChannel : t.newChannel}
</h2>
<div className="space-y-4">
{/* Group ID (only for create) */}
{!editingChannel && (
<div>
<label className={labelCls}>{t.fieldGroupId}</label>
<input
type="number"
value={form.group_id}
onChange={(e) => setForm({ ...form, group_id: e.target.value ? Number(e.target.value) : '' })}
className={inputCls}
required
/>
</div>
)}
{/* Name */}
<div>
<label className={labelCls}>{t.fieldName}</label>
<input
type="text"
value={form.name}
onChange={(e) => setForm({ ...form, name: e.target.value })}
className={inputCls}
required
/>
</div>
{/* Platform */}
<div>
<label className={labelCls}>{t.fieldPlatform}</label>
<select
value={form.platform}
onChange={(e) => setForm({ ...form, platform: e.target.value })}
className={inputCls}
>
{PLATFORMS.map((p) => (
<option key={p} value={p}>
{p}
</option>
))}
</select>
</div>
{/* Rate Multiplier */}
<div>
<label className={labelCls}>{t.fieldRate}</label>
<input
type="number"
step="0.0001"
min="0"
value={form.rate_multiplier}
onChange={(e) => setForm({ ...form, rate_multiplier: e.target.value })}
className={inputCls}
required
/>
<p className={`mt-1 text-xs ${isDark ? 'text-slate-500' : 'text-slate-400'}`}>{t.fieldRateHint}</p>
</div>
{/* Description */}
<div>
<label className={labelCls}>{t.fieldDescription}</label>
<textarea
value={form.description}
onChange={(e) => setForm({ ...form, description: e.target.value })}
rows={2}
className={inputCls}
/>
</div>
{/* Models */}
<div>
<label className={labelCls}>{t.fieldModels}</label>
<textarea
value={form.models}
onChange={(e) => setForm({ ...form, models: e.target.value })}
rows={4}
className={[inputCls, 'font-mono text-xs'].join(' ')}
placeholder="claude-sonnet-4-20250514&#10;claude-opus-4-20250514"
/>
</div>
{/* Features */}
<div>
<label className={labelCls}>{t.fieldFeatures}</label>
<textarea
value={form.features}
onChange={(e) => setForm({ ...form, features: e.target.value })}
rows={3}
className={[inputCls, 'font-mono text-xs'].join(' ')}
placeholder="Extended thinking&#10;Vision&#10;Tool use"
/>
</div>
{/* Sort Order */}
<div>
<label className={labelCls}>{t.fieldSortOrder}</label>
<input
type="number"
value={form.sort_order}
onChange={(e) => setForm({ ...form, sort_order: e.target.value })}
className={inputCls}
/>
</div>
{/* Enabled */}
<div className="flex items-center gap-3">
<button
type="button"
onClick={() => setForm({ ...form, enabled: !form.enabled })}
className={[
'relative inline-flex h-6 w-11 items-center rounded-full transition-colors',
form.enabled ? 'bg-emerald-500' : isDark ? 'bg-slate-600' : 'bg-slate-300',
].join(' ')}
>
<span
className={[
'inline-block h-4 w-4 rounded-full bg-white transition-transform',
form.enabled ? 'translate-x-6' : 'translate-x-1',
].join(' ')}
/>
</button>
<span className={`text-sm ${isDark ? 'text-slate-300' : 'text-slate-700'}`}>{t.fieldEnabled}</span>
</div>
</div>
{/* Actions */}
<div className="mt-6 flex justify-end gap-3">
<button
type="button"
onClick={closeEditModal}
className={[
'rounded-lg px-4 py-2 text-sm font-medium transition-colors',
isDark ? 'text-slate-400 hover:bg-slate-700' : 'text-slate-600 hover:bg-slate-100',
].join(' ')}
>
{t.cancel}
</button>
<button
type="button"
onClick={handleSave}
disabled={saving || !form.name.trim() || form.group_id === '' || !form.rate_multiplier}
className="rounded-lg bg-emerald-500 px-4 py-2 text-sm font-medium text-white transition-colors hover:bg-emerald-600 disabled:opacity-50 disabled:cursor-not-allowed"
>
{saving ? t.saving : t.save}
</button>
</div>
</div>
</div>
)}
{/* ── Sync from Sub2API Modal ── */}
{syncModalOpen && (
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/50 p-4" onClick={closeSyncModal}>
<div
className={[
'relative w-full max-w-lg overflow-y-auto rounded-2xl border p-6 shadow-2xl',
isDark ? 'border-slate-700 bg-slate-800' : 'border-slate-200 bg-white',
].join(' ')}
style={{ maxHeight: '80vh' }}
onClick={(e) => e.stopPropagation()}
>
<h2 className={`mb-1 text-lg font-semibold ${isDark ? 'text-slate-100' : 'text-slate-900'}`}>
{t.syncTitle}
</h2>
<p className={`mb-4 text-sm ${isDark ? 'text-slate-400' : 'text-slate-500'}`}>{t.syncHint}</p>
{syncLoading ? (
<div className={`py-8 text-center ${isDark ? 'text-slate-400' : 'text-gray-500'}`}>{t.syncLoading}</div>
) : syncGroups.length === 0 ? (
<div className={`py-8 text-center ${isDark ? 'text-slate-400' : 'text-gray-500'}`}>{t.syncNoGroups}</div>
) : (
<div className="space-y-2">
{syncGroups.map((group) => {
const alreadyImported = existingGroupIds.has(group.id);
return (
<label
key={group.id}
className={[
'flex items-start gap-3 rounded-lg border p-3 transition-colors',
alreadyImported
? isDark
? 'border-slate-700 bg-slate-700/30 opacity-60'
: 'border-slate-200 bg-slate-50 opacity-60'
: syncSelected.has(group.id)
? isDark
? 'border-indigo-500/50 bg-indigo-500/10'
: 'border-indigo-300 bg-indigo-50'
: isDark
? 'border-slate-700 hover:border-slate-600'
: 'border-slate-200 hover:border-slate-300',
alreadyImported ? 'cursor-not-allowed' : 'cursor-pointer',
].join(' ')}
>
<input
type="checkbox"
disabled={alreadyImported}
checked={syncSelected.has(group.id)}
onChange={() => toggleSyncGroup(group.id)}
className="mt-0.5 h-4 w-4 rounded border-slate-300 text-indigo-500 focus:ring-indigo-500"
/>
<div className="flex-1 min-w-0">
<div className="flex items-center gap-2">
<span className={`text-sm font-medium ${isDark ? 'text-slate-100' : 'text-slate-900'}`}>
{group.name}
</span>
<span className={`text-xs ${isDark ? 'text-slate-500' : 'text-slate-400'}`}>
#{group.id}
</span>
{(() => {
const gpc = PLATFORM_COLORS[group.platform] ?? PLATFORM_COLORS.claude;
return (
<span
className={`inline-flex items-center rounded-full px-2 py-0.5 text-[10px] font-medium ${isDark ? gpc.bg.replace('dark:', '') : gpc.bg.split(' ')[0]} ${isDark ? gpc.text.replace('dark:', '') : gpc.text.split(' ')[0]}`}
>
{group.platform}
</span>
);
})()}
{alreadyImported && (
<span className="text-[10px] text-amber-500 font-medium">{t.syncAlreadyExists}</span>
)}
</div>
{group.description && (
<p className={`mt-0.5 text-xs truncate ${isDark ? 'text-slate-400' : 'text-slate-500'}`}>
{group.description}
</p>
)}
<p className={`mt-0.5 text-xs ${isDark ? 'text-slate-500' : 'text-slate-400'}`}>
{t.colRate}: {group.rate_multiplier}x
</p>
</div>
</label>
);
})}
</div>
)}
{/* Sync actions */}
<div className="mt-5 flex justify-end gap-3">
<button
type="button"
onClick={closeSyncModal}
className={[
'rounded-lg px-4 py-2 text-sm font-medium transition-colors',
isDark ? 'text-slate-400 hover:bg-slate-700' : 'text-slate-600 hover:bg-slate-100',
].join(' ')}
>
{t.cancel}
</button>
<button
type="button"
onClick={handleSyncImport}
disabled={syncImporting || syncSelected.size === 0}
className="rounded-lg bg-indigo-500 px-4 py-2 text-sm font-medium text-white transition-colors hover:bg-indigo-600 disabled:opacity-50 disabled:cursor-not-allowed"
>
{syncImporting ? t.syncImporting : `${t.syncImport} (${syncSelected.size})`}
</button>
</div>
</div>
</div>
)}
</PayPageLayout>
);
}
function ChannelsPageFallback() {
const searchParams = useSearchParams();
const locale = resolveLocale(searchParams.get('lang'));
return (
<div className="flex min-h-screen items-center justify-center">
<div className="text-gray-500">{locale === 'en' ? 'Loading...' : '加载中...'}</div>
</div>
);
}
export default function ChannelsPage() {
return (
<Suspense fallback={<ChannelsPageFallback />}>
<ChannelsContent />
</Suspense>
);
}