feat(api, i18n): add admin report job functionalities and enhance locale support
- Introduced new API functions for managing admin report jobs, including download and post operations. - Updated English, Nepali, and Chinese locale files to include new messages related to report job actions and rollback confirmations. - Enhanced user experience by providing clearer instructions and feedback in the admin interface. - Refactored related components to integrate new functionalities and improve overall usability.
This commit is contained in:
65
src/api/admin-report-jobs.ts
Normal file
65
src/api/admin-report-jobs.ts
Normal file
@@ -0,0 +1,65 @@
|
||||
import { adminHttp, adminRequest } from "@/lib/admin-http";
|
||||
import { withAdminAuthHeader } from "@/lib/admin-auth";
|
||||
import { withAdminLocaleHeaders } from "@/lib/admin-locale";
|
||||
|
||||
import { API_V1_PREFIX } from "./paths";
|
||||
|
||||
import type {
|
||||
AdminReportJobCreatePayload,
|
||||
AdminReportJobCreateResult,
|
||||
AdminReportJobListData,
|
||||
AdminReportJobRow,
|
||||
} from "@/types/api/admin-report-jobs";
|
||||
|
||||
const A = `${API_V1_PREFIX}/admin/report-jobs`;
|
||||
|
||||
export async function getAdminReportJobs(params?: {
|
||||
page?: number;
|
||||
per_page?: number;
|
||||
}): Promise<AdminReportJobListData> {
|
||||
return adminRequest.get<AdminReportJobListData>(A, { params });
|
||||
}
|
||||
|
||||
export async function postAdminReportJob(
|
||||
payload: AdminReportJobCreatePayload,
|
||||
): Promise<AdminReportJobCreateResult> {
|
||||
return adminRequest.post<AdminReportJobCreateResult>(A, payload);
|
||||
}
|
||||
|
||||
export async function getAdminReportJob(id: number): Promise<AdminReportJobRow> {
|
||||
return adminRequest.get<AdminReportJobRow>(`${A}/${id}`);
|
||||
}
|
||||
|
||||
function filenameFromContentDisposition(header: string | undefined): string | null {
|
||||
if (!header) {
|
||||
return null;
|
||||
}
|
||||
const utf8 = /filename\*=UTF-8''([^;]+)/i.exec(header);
|
||||
if (utf8?.[1]) {
|
||||
try {
|
||||
return decodeURIComponent(utf8[1].trim());
|
||||
} catch {
|
||||
return utf8[1].trim();
|
||||
}
|
||||
}
|
||||
const plain = /filename="?([^";]+)"?/i.exec(header);
|
||||
return plain?.[1]?.trim() ?? null;
|
||||
}
|
||||
|
||||
export async function downloadAdminReportJob(
|
||||
id: number,
|
||||
): Promise<{ blob: Blob; filename: string | null }> {
|
||||
const res = await adminHttp.request<Blob>(
|
||||
withAdminAuthHeader(
|
||||
withAdminLocaleHeaders({
|
||||
url: `${A}/${id}/download`,
|
||||
method: "GET",
|
||||
responseType: "blob",
|
||||
}),
|
||||
),
|
||||
);
|
||||
const filename = filenameFromContentDisposition(
|
||||
typeof res.headers["content-disposition"] === "string" ? res.headers["content-disposition"] : undefined,
|
||||
);
|
||||
return { blob: res.data, filename };
|
||||
}
|
||||
@@ -20,6 +20,12 @@ export {
|
||||
getAdminReportPlayerWinLoss,
|
||||
getAdminReportRebateCommission,
|
||||
} from "@/api/admin-reports";
|
||||
export {
|
||||
downloadAdminReportJob,
|
||||
getAdminReportJob,
|
||||
getAdminReportJobs,
|
||||
postAdminReportJob,
|
||||
} from "@/api/admin-report-jobs";
|
||||
export {
|
||||
getAdminDraw,
|
||||
getAdminDrawFinanceSummary,
|
||||
|
||||
@@ -64,7 +64,14 @@
|
||||
"refresh": "Refresh versions",
|
||||
"newDraft": "New draft",
|
||||
"saveDraft": "Save draft",
|
||||
"saveFailed": "Failed to save configuration"
|
||||
"saveFailed": "Failed to save configuration",
|
||||
"rollbackSuccess": "Cloned v{{fromVersion}} into new draft v{{version}}",
|
||||
"rollbackFailed": "Rollback failed",
|
||||
"rollbackDialog": {
|
||||
"title": "Confirm rollback",
|
||||
"description": "A new draft will be cloned from version v{{version}}. The active version will not be overwritten directly.",
|
||||
"confirm": "Confirm rollback"
|
||||
}
|
||||
},
|
||||
"wallet": {
|
||||
"title": "Wallet transfer limit settings",
|
||||
@@ -327,7 +334,8 @@
|
||||
},
|
||||
"winEnjoy": {
|
||||
"label": "Apply rebate on winning tickets",
|
||||
"description": "Placeholder field. It can later be aligned with risk and settlement rules and persisted."
|
||||
"description": "Placeholder field. It can later be aligned with risk and settlement rules and persisted.",
|
||||
"pendingNote": "Required by the product spec, but the API has no field yet. This row is informational only—you cannot change it here."
|
||||
},
|
||||
"effectiveTime": "Effective time (current active odds version)"
|
||||
},
|
||||
@@ -372,12 +380,21 @@
|
||||
"actions": "Actions"
|
||||
},
|
||||
"occupancy": {
|
||||
"title": "All number occupancy",
|
||||
"description": "Placeholder view: filters and exports still need ticket-summary integration. Data below still comes from the current draft list.",
|
||||
"searchLabel": "Search number",
|
||||
"searchPlaceholder": "e.g. 8888",
|
||||
"filterPending": "Sold-out / high-risk preset filter is pending integration",
|
||||
"exportPending": "CSV export is pending integration"
|
||||
"searchPlaceholder": "e.g. 8888"
|
||||
},
|
||||
"runtime": {
|
||||
"title": "Per-draw occupancy (live)",
|
||||
"description": "Loaded from the draw risk-pool API, not from the version draft above. Select a draw to see used, remaining, and sold-out state.",
|
||||
"drawLabel": "Draw",
|
||||
"drawPlaceholder": "Select draw",
|
||||
"filterAll": "All",
|
||||
"filterSoldOut": "Sold out only",
|
||||
"filterHighRisk": "High usage",
|
||||
"manageHint": "Use the links above for full risk operations on this draw.",
|
||||
"noDraws": "No draws available; cannot load occupancy.",
|
||||
"soldYes": "Yes",
|
||||
"soldNo": "No"
|
||||
},
|
||||
"actions": {
|
||||
"update": "Update",
|
||||
|
||||
@@ -155,6 +155,7 @@
|
||||
"finance": "Draw finance",
|
||||
"review": "Review & publish",
|
||||
"riskOccupancy": "Risk occupancy",
|
||||
"riskLockLogs": "Lock logs",
|
||||
"riskHot": "Hot numbers",
|
||||
"riskSoldOut": "Sold-out numbers",
|
||||
"riskPools": "Risk pools"
|
||||
|
||||
@@ -10,8 +10,11 @@
|
||||
"dimension": "Dimension",
|
||||
"exportPending": "{{report}} {{format}} export API is not connected yet",
|
||||
"exportSuccess": "Exported {{report}} ({{format}})",
|
||||
"exportServerSuccess": "Job {{jobNo}} created and downloaded {{report}} ({{format}})",
|
||||
"exportFailed": "Export failed",
|
||||
"exportHint": "Once export APIs are connected, the current filters will generate the selected file format.",
|
||||
"exportHint": "Server exports use the current filters to build a full file. Completed jobs can be downloaded again below.",
|
||||
"exportServerHint": "Full export via server job (preview query not required)",
|
||||
"exportClientHint": "Export current preview page (run query first)",
|
||||
"validation": {
|
||||
"drawNoRequired": "Please enter a draw number",
|
||||
"drawNoNotFound": "Draw number «{{drawNo}}» was not found",
|
||||
@@ -19,7 +22,43 @@
|
||||
},
|
||||
"formats": {
|
||||
"csv": "CSV",
|
||||
"excel": "Excel"
|
||||
"excel": "Excel",
|
||||
"csvServer": "Export CSV (full)",
|
||||
"excelServer": "Export Excel (full)"
|
||||
},
|
||||
"tasks": {
|
||||
"refresh": "Refresh",
|
||||
"download": "Download",
|
||||
"loadFailed": "Failed to load export jobs",
|
||||
"downloadSuccess": "Downloaded {{jobNo}}",
|
||||
"downloadFailed": "Download failed",
|
||||
"columns": {
|
||||
"jobNo": "Job no.",
|
||||
"report": "Report",
|
||||
"format": "Format",
|
||||
"status": "Status",
|
||||
"createdAt": "Created",
|
||||
"actions": "Actions"
|
||||
},
|
||||
"status": {
|
||||
"pending": "Pending",
|
||||
"processing": "Processing",
|
||||
"completed": "Completed",
|
||||
"failed": "Failed"
|
||||
}
|
||||
},
|
||||
"jobTypes": {
|
||||
"draw_profit_summary": "Draw P&L",
|
||||
"daily_profit_summary": "Daily P&L",
|
||||
"player_win_loss": "Player win/loss",
|
||||
"wallet_transfer_report": "Wallet transfers",
|
||||
"wallet_txns_daily": "Wallet txns (daily)",
|
||||
"transfer_orders_daily": "Transfer orders (daily)",
|
||||
"hot_number_risk_report": "Hot number risk",
|
||||
"play_dimension_report": "Play dimension",
|
||||
"sold_out_number_report": "Sold-out numbers",
|
||||
"rebate_commission_report": "Rebate / commission",
|
||||
"audit_operation_report": "Admin audit"
|
||||
},
|
||||
"empty": "No matching reports",
|
||||
"backendPending": "This report is temporarily unavailable",
|
||||
|
||||
@@ -64,7 +64,14 @@
|
||||
"refresh": "संस्करण रिफ्रेस",
|
||||
"newDraft": "नयाँ ड्राफ्ट",
|
||||
"saveDraft": "ड्राफ्ट सेभ गर्नुहोस्",
|
||||
"saveFailed": "कन्फिगरेसन सुरक्षित गर्न असफल"
|
||||
"saveFailed": "कन्फिगरेसन सुरक्षित गर्न असफल",
|
||||
"rollbackSuccess": "v{{fromVersion}} बाट नयाँ ड्राफ्ट v{{version}} क्लोन गरियो",
|
||||
"rollbackFailed": "रोलब्याक असफल भयो",
|
||||
"rollbackDialog": {
|
||||
"title": "रोलब्याक पुष्टि गर्ने?",
|
||||
"description": "संस्करण v{{version}} बाट नयाँ ड्राफ्ट क्लोन हुनेछ। सक्रिय संस्करण सिधै अधिलेखन हुँदैन।",
|
||||
"confirm": "रोलब्याक पुष्टि"
|
||||
}
|
||||
},
|
||||
"wallet": {
|
||||
"title": "वालेट ट्रान्सफर सीमा सेटिङ",
|
||||
@@ -327,7 +334,8 @@
|
||||
},
|
||||
"winEnjoy": {
|
||||
"label": "जितेका टिकटहरूमा पनि रिबेट लागू गर्ने",
|
||||
"description": "यो placeholder field हो। पछि risk र settlement नियमसँग मिलाएर स्थायी रूपमा राख्न सकिन्छ।"
|
||||
"description": "यो placeholder field हो। पछि risk र settlement नियमसँग मिलाएर स्थायी रूपमा राख्न सकिन्छ।",
|
||||
"pendingNote": "उत्पादन विनिर्देशनले यो switch चाहिन्छ, तर API मा field छैन। यहाँ केवल जानकारी देखाइन्छ—यहाँबाट बदल्न मिल्दैन।"
|
||||
},
|
||||
"effectiveTime": "लागू समय (हाल सक्रिय अड्स संस्करण)"
|
||||
},
|
||||
@@ -372,12 +380,21 @@
|
||||
"actions": "कार्य"
|
||||
},
|
||||
"occupancy": {
|
||||
"title": "सबै नम्बर occupancy",
|
||||
"description": "यो placeholder दृश्य हो। filter र export ले ticket-summary एकीकरण अझै चाहिन्छ। तलको data अहिले पनि हालको ड्राफ्ट सूचीबाट आउँछ।",
|
||||
"searchLabel": "नम्बर खोज्नुहोस्",
|
||||
"searchPlaceholder": "जस्तै 8888",
|
||||
"filterPending": "Sold-out / high-risk preset filter अझै एकीकृत भएको छैन",
|
||||
"exportPending": "CSV export अझै एकीकृत भएको छैन"
|
||||
"searchPlaceholder": "जस्तै 8888"
|
||||
},
|
||||
"runtime": {
|
||||
"title": "ड्रअनुसार occupancy (लाइभ)",
|
||||
"description": "माथिको संस्करण ड्राफ्ट होइन—चयन गरिएको ड्रअको risk-pool API बाट लोड हुन्छ। प्रयोग, बाँकी र sold-out हेर्नुहोस्।",
|
||||
"drawLabel": "ड्र",
|
||||
"drawPlaceholder": "ड्र छान्नुहोस्",
|
||||
"filterAll": "सबै",
|
||||
"filterSoldOut": "मात्र sold-out",
|
||||
"filterHighRisk": "उच्च प्रयोग",
|
||||
"manageHint": "पूर्ण risk सञ्चालनका लागि माथिको लिङ्कबाट ड्र subpage खोल्नुहोस्।",
|
||||
"noDraws": "कुनै ड्र छैन; occupancy लोड गर्न सकिँदैन।",
|
||||
"soldYes": "हो",
|
||||
"soldNo": "होइन"
|
||||
},
|
||||
"actions": {
|
||||
"update": "अपडेट",
|
||||
|
||||
@@ -155,6 +155,7 @@
|
||||
"finance": "ड्रअ वित्त",
|
||||
"review": "समीक्षा र प्रकाशन",
|
||||
"riskOccupancy": "जोखिम अकुपेन्सी",
|
||||
"riskLockLogs": "लक लग",
|
||||
"riskHot": "हट नम्बर",
|
||||
"riskSoldOut": "बिक्री समाप्त नम्बर",
|
||||
"riskPools": "जोखिम पूल"
|
||||
|
||||
@@ -10,8 +10,11 @@
|
||||
"dimension": "आयाम",
|
||||
"exportPending": "{{report}} {{format}} निर्यात API अझै जोडिएको छैन",
|
||||
"exportSuccess": "{{report}} ({{format}}) निर्यात भयो",
|
||||
"exportServerSuccess": "कार्य {{jobNo}} सिर्जना भई {{report}} ({{format}}) डाउनलोड भयो",
|
||||
"exportFailed": "निर्यात असफल भयो",
|
||||
"exportHint": "निर्यात API जोडिएपछि हालका फिल्टरअनुसार छानिएको फाइल ढाँचा बनाइनेछ।",
|
||||
"exportHint": "सर्भर निर्यातले हालका फिल्टरअनुसार पूर्ण फाइल बनाउँछ। सम्पन्न कार्य तलबाट पुन: डाउनलोड गर्न सकिन्छ।",
|
||||
"exportServerHint": "पूर्ण निर्यात सर्भर कार्यबाट (पूर्वावलोकन क्वेरी अनिवार्य छैन)",
|
||||
"exportClientHint": "हालको पूर्वावलोकन पृष्ठ निर्यात (पहिले क्वेरी चलाउनुहोस्)",
|
||||
"validation": {
|
||||
"drawNoRequired": "कृपया ड्र नं. प्रविष्ट गर्नुहोस्",
|
||||
"drawNoNotFound": "ड्र नं. «{{drawNo}}» फेला परेन",
|
||||
@@ -19,7 +22,43 @@
|
||||
},
|
||||
"formats": {
|
||||
"csv": "CSV",
|
||||
"excel": "Excel"
|
||||
"excel": "Excel",
|
||||
"csvServer": "CSV निर्यात (पूर्ण)",
|
||||
"excelServer": "Excel निर्यात (पूर्ण)"
|
||||
},
|
||||
"tasks": {
|
||||
"refresh": "रिफ्रेस",
|
||||
"download": "डाउनलोड",
|
||||
"loadFailed": "कार्य सूची लोड असफल",
|
||||
"downloadSuccess": "{{jobNo}} डाउनलोड भयो",
|
||||
"downloadFailed": "डाउनलोड असफल",
|
||||
"columns": {
|
||||
"jobNo": "कार्य नं.",
|
||||
"report": "रिपोर्ट",
|
||||
"format": "ढाँचा",
|
||||
"status": "स्थिति",
|
||||
"createdAt": "सिर्जना",
|
||||
"actions": "कार्य"
|
||||
},
|
||||
"status": {
|
||||
"pending": "पर्खाइ",
|
||||
"processing": "प्रक्रियामा",
|
||||
"completed": "सम्पन्न",
|
||||
"failed": "असफल"
|
||||
}
|
||||
},
|
||||
"jobTypes": {
|
||||
"draw_profit_summary": "ड्र P&L",
|
||||
"daily_profit_summary": "दैनिक P&L",
|
||||
"player_win_loss": "खेलाडी जित/हार",
|
||||
"wallet_transfer_report": "वालेट ट्रान्सफर",
|
||||
"wallet_txns_daily": "वालेट लेनदेन (दैनिक)",
|
||||
"transfer_orders_daily": "ट्रान्सफर अर्डर (दैनिक)",
|
||||
"hot_number_risk_report": "लोकप्रिय नम्बर जोखिम",
|
||||
"play_dimension_report": "प्ले आयाम",
|
||||
"sold_out_number_report": "बिक्री समाप्त नम्बर",
|
||||
"rebate_commission_report": "रिबेट / कमिसन",
|
||||
"audit_operation_report": "प्रशासक अडिट"
|
||||
},
|
||||
"empty": "मिल्ने रिपोर्ट छैन",
|
||||
"backendPending": "यो रिपोर्ट अस्थायी रूपमा उपलब्ध छैन",
|
||||
|
||||
@@ -64,7 +64,14 @@
|
||||
"refresh": "刷新版本",
|
||||
"newDraft": "新建草稿",
|
||||
"saveDraft": "保存草稿",
|
||||
"saveFailed": "配置保存失败"
|
||||
"saveFailed": "配置保存失败",
|
||||
"rollbackSuccess": "已从 v{{fromVersion}} 克隆出新草稿 v{{version}}",
|
||||
"rollbackFailed": "回滚失败",
|
||||
"rollbackDialog": {
|
||||
"title": "确认回滚",
|
||||
"description": "系统会基于版本 v{{version}} 克隆出新的草稿,不会直接覆盖当前生效版本。",
|
||||
"confirm": "确认回滚"
|
||||
}
|
||||
},
|
||||
"wallet": {
|
||||
"title": "钱包转账限额配置",
|
||||
@@ -327,7 +334,8 @@
|
||||
},
|
||||
"winEnjoy": {
|
||||
"label": "中奖注单也应用回水",
|
||||
"description": "这是预留字段,后续可和风控、结算规则对齐后再真正落库存储。"
|
||||
"description": "这是预留字段,后续可和风控、结算规则对齐后再真正落库存储。",
|
||||
"pendingNote": "产品要求支持该开关,但后端尚未提供配置字段;当前仅展示说明,无法在此修改。"
|
||||
},
|
||||
"effectiveTime": "生效时间(当前赔率生效版本)"
|
||||
},
|
||||
@@ -372,12 +380,21 @@
|
||||
"actions": "操作"
|
||||
},
|
||||
"occupancy": {
|
||||
"title": "全号码占用视图",
|
||||
"description": "这里还是占位视图,筛选和导出后续还需要接入真实注单汇总;下方数据目前仍来自当前草稿列表。",
|
||||
"searchLabel": "搜索号码",
|
||||
"searchPlaceholder": "例如 8888",
|
||||
"filterPending": "售罄 / 高风险预设筛选尚未接入",
|
||||
"exportPending": "CSV 导出尚未接入"
|
||||
"searchPlaceholder": "例如 8888"
|
||||
},
|
||||
"runtime": {
|
||||
"title": "按期号查看占用(实时)",
|
||||
"description": "数据来自该期号风险池 API,与上方版本草稿无关。请选择期号后查看已占用、剩余额度与售罄状态。",
|
||||
"drawLabel": "期号",
|
||||
"drawPlaceholder": "选择期号",
|
||||
"filterAll": "全部",
|
||||
"filterSoldOut": "仅售罄",
|
||||
"filterHighRisk": "高占用",
|
||||
"manageHint": "完整风控操作请使用上方链接进入期号风控子页。",
|
||||
"noDraws": "暂无可用期号,无法加载占用数据。",
|
||||
"soldYes": "是",
|
||||
"soldNo": "否"
|
||||
},
|
||||
"actions": {
|
||||
"update": "更新",
|
||||
|
||||
@@ -155,6 +155,7 @@
|
||||
"finance": "期号收支",
|
||||
"review": "审核与发布",
|
||||
"riskOccupancy": "风控占用",
|
||||
"riskLockLogs": "占用流水",
|
||||
"riskHot": "热门号码",
|
||||
"riskSoldOut": "售罄号码",
|
||||
"riskPools": "风险池"
|
||||
|
||||
@@ -10,8 +10,11 @@
|
||||
"dimension": "维度",
|
||||
"exportPending": "{{report}} 的 {{format}} 导出接口待接入",
|
||||
"exportSuccess": "已导出 {{report}}({{format}})",
|
||||
"exportServerSuccess": "已生成任务 {{jobNo}} 并下载 {{report}}({{format}})",
|
||||
"exportFailed": "导出失败",
|
||||
"exportHint": "接入导出接口后,会按当前条件生成对应格式的文件。",
|
||||
"exportHint": "服务端导出会按当前筛选条件生成全量文件;任务完成后可在下方列表再次下载。",
|
||||
"exportServerHint": "全量导出走服务端任务(无需先查询预览)",
|
||||
"exportClientHint": "导出当前预览页数据(需先查询)",
|
||||
"validation": {
|
||||
"drawNoRequired": "请输入期号",
|
||||
"drawNoNotFound": "未找到期号「{{drawNo}}」",
|
||||
@@ -19,7 +22,43 @@
|
||||
},
|
||||
"formats": {
|
||||
"csv": "CSV",
|
||||
"excel": "Excel"
|
||||
"excel": "Excel",
|
||||
"csvServer": "导出 CSV(全量)",
|
||||
"excelServer": "导出 Excel(全量)"
|
||||
},
|
||||
"tasks": {
|
||||
"refresh": "刷新",
|
||||
"download": "下载",
|
||||
"loadFailed": "任务列表加载失败",
|
||||
"downloadSuccess": "已下载 {{jobNo}}",
|
||||
"downloadFailed": "下载失败",
|
||||
"columns": {
|
||||
"jobNo": "任务编号",
|
||||
"report": "报表",
|
||||
"format": "格式",
|
||||
"status": "状态",
|
||||
"createdAt": "创建时间",
|
||||
"actions": "操作"
|
||||
},
|
||||
"status": {
|
||||
"pending": "排队中",
|
||||
"processing": "处理中",
|
||||
"completed": "已完成",
|
||||
"failed": "失败"
|
||||
}
|
||||
},
|
||||
"jobTypes": {
|
||||
"draw_profit_summary": "期号盈亏",
|
||||
"daily_profit_summary": "每日盈亏汇总",
|
||||
"player_win_loss": "玩家输赢",
|
||||
"wallet_transfer_report": "玩家转入转出",
|
||||
"wallet_txns_daily": "钱包流水(日)",
|
||||
"transfer_orders_daily": "转账订单(日)",
|
||||
"hot_number_risk_report": "热门号码风险",
|
||||
"play_dimension_report": "玩法维度",
|
||||
"sold_out_number_report": "售罄号码",
|
||||
"rebate_commission_report": "佣金/回水",
|
||||
"audit_operation_report": "后台操作审计"
|
||||
},
|
||||
"empty": "没有匹配的报表",
|
||||
"backendPending": "该报表暂不可用",
|
||||
|
||||
@@ -56,7 +56,7 @@ const ROUTE_PATTERNS: RoutePattern[] = [
|
||||
},
|
||||
{
|
||||
test: (p) => /^\/admin\/draws\/\d+\/risk\/occupancy$/.test(p) || /^\/admin\/risk\/draws\/\d+\/occupancy$/.test(p),
|
||||
resolve: () => ({ ns: "draws", key: "subnav.riskOccupancy" }),
|
||||
resolve: () => ({ ns: "draws", key: "subnav.riskLockLogs" }),
|
||||
},
|
||||
{
|
||||
test: (p) => /^\/admin\/draws\/\d+\/risk\/hot$/.test(p) || /^\/admin\/risk\/draws\/\d+\/hot$/.test(p),
|
||||
|
||||
89
src/lib/report-export-map.ts
Normal file
89
src/lib/report-export-map.ts
Normal file
@@ -0,0 +1,89 @@
|
||||
export type ReportFilterSnapshot = {
|
||||
dateFrom: string;
|
||||
dateTo: string;
|
||||
playerId: number | null;
|
||||
play: string;
|
||||
operatorId: number | null;
|
||||
drawId: number | null;
|
||||
drawNo: string;
|
||||
number: string;
|
||||
};
|
||||
|
||||
/** UI report keys used in reports-console */
|
||||
export type ReportUiKey =
|
||||
| "draw_profit"
|
||||
| "daily_profit"
|
||||
| "player_win_loss"
|
||||
| "player_transfer"
|
||||
| "hot_number_risk"
|
||||
| "play_dimension"
|
||||
| "sold_out_number"
|
||||
| "rebate_commission"
|
||||
| "admin_audit";
|
||||
|
||||
/** Maps UI keys to POST /admin/report-jobs `report_type` */
|
||||
export const REPORT_UI_TO_JOB_TYPE: Record<ReportUiKey, string> = {
|
||||
draw_profit: "draw_profit_summary",
|
||||
daily_profit: "daily_profit_summary",
|
||||
player_win_loss: "player_win_loss",
|
||||
player_transfer: "wallet_transfer_report",
|
||||
hot_number_risk: "hot_number_risk_report",
|
||||
play_dimension: "play_dimension_report",
|
||||
sold_out_number: "sold_out_number_report",
|
||||
rebate_commission: "rebate_commission_report",
|
||||
admin_audit: "audit_operation_report",
|
||||
};
|
||||
|
||||
/** Report types with full server-side export (POST /admin/report-jobs). */
|
||||
export const REPORT_UI_SERVER_FULL_EXPORT = new Set<ReportUiKey>([
|
||||
"draw_profit",
|
||||
"daily_profit",
|
||||
"player_win_loss",
|
||||
"player_transfer",
|
||||
"hot_number_risk",
|
||||
"play_dimension",
|
||||
"sold_out_number",
|
||||
"rebate_commission",
|
||||
"admin_audit",
|
||||
]);
|
||||
|
||||
export function buildReportJobParameters(
|
||||
key: ReportUiKey,
|
||||
filters: ReportFilterSnapshot,
|
||||
): Record<string, unknown> {
|
||||
const params: Record<string, unknown> = {};
|
||||
|
||||
if (filters.dateFrom) {
|
||||
params.date_from = filters.dateFrom;
|
||||
}
|
||||
if (filters.dateTo) {
|
||||
params.date_to = filters.dateTo;
|
||||
}
|
||||
if (filters.playerId != null && filters.playerId > 0) {
|
||||
params.player_id = filters.playerId;
|
||||
}
|
||||
if (filters.operatorId != null && filters.operatorId > 0) {
|
||||
params.operator_id = filters.operatorId;
|
||||
}
|
||||
if (filters.play.trim()) {
|
||||
params.play_code = filters.play.trim();
|
||||
}
|
||||
|
||||
if (key === "draw_profit" || key === "hot_number_risk" || key === "sold_out_number") {
|
||||
if (filters.drawId != null && filters.drawId > 0) {
|
||||
params.draw_id = filters.drawId;
|
||||
}
|
||||
if (filters.drawNo.trim()) {
|
||||
params.draw_no = filters.drawNo.trim();
|
||||
}
|
||||
}
|
||||
|
||||
if (key === "hot_number_risk" && filters.number.trim()) {
|
||||
const digits = filters.number.trim();
|
||||
if (/^\d{4}$/.test(digits)) {
|
||||
params.normalized_number = digits;
|
||||
}
|
||||
}
|
||||
|
||||
return params;
|
||||
}
|
||||
@@ -151,6 +151,8 @@ export function PlayConfigDocScreen() {
|
||||
const [loadingDetail, setLoadingDetail] = useState(false);
|
||||
const [saving, setSaving] = useState(false);
|
||||
const [creatingDraftId, setCreatingDraftId] = useState<string | null>(null);
|
||||
const [rollbackOpen, setRollbackOpen] = useState(false);
|
||||
const [rollbackTarget, setRollbackTarget] = useState<ConfigVersionSummary | null>(null);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const detailRequestSeq = useRef(0);
|
||||
|
||||
@@ -400,6 +402,41 @@ export function PlayConfigDocScreen() {
|
||||
}
|
||||
}
|
||||
|
||||
function requestRollback(row: ConfigVersionSummary) {
|
||||
setRollbackTarget(row);
|
||||
setRollbackOpen(true);
|
||||
}
|
||||
|
||||
async function handleRollback() {
|
||||
if (!rollbackTarget) {
|
||||
return;
|
||||
}
|
||||
setSaving(true);
|
||||
try {
|
||||
const d = await postPlayConfigVersion({
|
||||
reason: `rollback from v${rollbackTarget.version_no}`,
|
||||
clone_from_version_id: rollbackTarget.id,
|
||||
});
|
||||
toast.success(
|
||||
t("versionActions.rollbackSuccess", {
|
||||
ns: "config",
|
||||
fromVersion: rollbackTarget.version_no,
|
||||
version: d.version_no,
|
||||
}),
|
||||
);
|
||||
await refreshList();
|
||||
setSelectedId(String(d.id));
|
||||
setDetail(d);
|
||||
setDraftRows(d.items.map((it) => ({ ...it })));
|
||||
setRollbackOpen(false);
|
||||
setRollbackTarget(null);
|
||||
} catch (e) {
|
||||
toast.error(e instanceof LotteryApiBizError ? e.message : t("versionActions.rollbackFailed", { ns: "config" }));
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
}
|
||||
|
||||
return (
|
||||
<ConfigDocPage
|
||||
title={t("nav.items.plays", { ns: "config" })}
|
||||
@@ -413,6 +450,8 @@ export function PlayConfigDocScreen() {
|
||||
loading={loadingList}
|
||||
sheetTitle={`${t("nav.items.plays", { ns: "config" })} ${t("versionSwitcher.sheetTitle", { ns: "config" })}`}
|
||||
onDeleteVersion={handleDeleteVersion}
|
||||
onRollbackVersion={requestRollback}
|
||||
rollbackBusy={saving}
|
||||
/>
|
||||
}
|
||||
actions={
|
||||
@@ -744,6 +783,29 @@ export function PlayConfigDocScreen() {
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
|
||||
<Dialog open={rollbackOpen} onOpenChange={setRollbackOpen}>
|
||||
<DialogContent showCloseButton className="sm:max-w-md">
|
||||
<DialogHeader>
|
||||
<DialogTitle>{t("versionActions.rollbackDialog.title", { ns: "config" })}</DialogTitle>
|
||||
<DialogDescription>
|
||||
{t("versionActions.rollbackDialog.description", {
|
||||
ns: "config",
|
||||
version: rollbackTarget?.version_no ?? "—",
|
||||
})}
|
||||
</DialogDescription>
|
||||
</DialogHeader>
|
||||
<DialogFooter>
|
||||
<Button type="button" variant="outline" onClick={() => setRollbackOpen(false)}>
|
||||
{t("actions.cancel", { ns: "adminUsers" })}
|
||||
</Button>
|
||||
<Button type="button" onClick={() => void handleRollback()} disabled={!rollbackTarget || saving}>
|
||||
{t("versionActions.rollbackDialog.confirm", { ns: "config" })}
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
|
||||
<ConfirmDialog />
|
||||
</ConfigDocPage>
|
||||
);
|
||||
|
||||
@@ -19,8 +19,16 @@ import {
|
||||
ConfigVersionToolbarMeta,
|
||||
ConfigVersionToolbarMetaEmphasis,
|
||||
} from "@/modules/config/config-version-toolbar-meta";
|
||||
import { AdminStatusBadge } from "@/components/admin/admin-status-badge";
|
||||
import { Alert, AlertDescription } from "@/components/ui/alert";
|
||||
import { Button } from "@/components/ui/button";
|
||||
import {
|
||||
Dialog,
|
||||
DialogContent,
|
||||
DialogDescription,
|
||||
DialogFooter,
|
||||
DialogHeader,
|
||||
DialogTitle,
|
||||
} from "@/components/ui/dialog";
|
||||
import { Input } from "@/components/ui/input";
|
||||
import { Label } from "@/components/ui/label";
|
||||
import { ConfigReadonlyValue } from "@/modules/config/config-readonly-value";
|
||||
@@ -116,6 +124,8 @@ export function RebateConfigDocScreen({
|
||||
const [p2, setP2] = useState("0");
|
||||
const [p3, setP3] = useState("0");
|
||||
const [p4, setP4] = useState("0");
|
||||
const [rollbackOpen, setRollbackOpen] = useState(false);
|
||||
const [rollbackTarget, setRollbackTarget] = useState<ConfigVersionSummary | null>(null);
|
||||
|
||||
const refreshTypes = useCallback(async () => {
|
||||
try {
|
||||
@@ -328,6 +338,45 @@ export function RebateConfigDocScreen({
|
||||
|
||||
const activeHead = listRows.find((x) => x.status === "active");
|
||||
|
||||
function requestRollback(row: ConfigVersionSummary) {
|
||||
setRollbackTarget(row);
|
||||
setRollbackOpen(true);
|
||||
}
|
||||
|
||||
async function handleRollback() {
|
||||
if (!rollbackTarget) {
|
||||
return;
|
||||
}
|
||||
setSaving(true);
|
||||
try {
|
||||
const d = await postOddsVersion({
|
||||
reason: `rollback from v${rollbackTarget.version_no}`,
|
||||
clone_from_version_id: rollbackTarget.id,
|
||||
});
|
||||
toast.success(
|
||||
t("versionActions.rollbackSuccess", {
|
||||
ns: "config",
|
||||
fromVersion: rollbackTarget.version_no,
|
||||
version: d.version_no,
|
||||
}),
|
||||
);
|
||||
await refreshList();
|
||||
setSelectedId(String(d.id));
|
||||
const rows = d.items.map((it) => ({ ...it }));
|
||||
setDetail(d);
|
||||
setDraftRows(rows);
|
||||
setP2(inferPercentFrom(2, rows, types));
|
||||
setP3(inferPercentFrom(3, rows, types));
|
||||
setP4(inferPercentFrom(4, rows, types));
|
||||
setRollbackOpen(false);
|
||||
setRollbackTarget(null);
|
||||
} catch (e) {
|
||||
toast.error(e instanceof LotteryApiBizError ? e.message : t("versionActions.rollbackFailed", { ns: "config" }));
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
}
|
||||
|
||||
async function handleDeleteVersion(row: ConfigVersionSummary) {
|
||||
try {
|
||||
await deleteOddsVersion(row.id);
|
||||
@@ -350,6 +399,8 @@ export function RebateConfigDocScreen({
|
||||
sheetTitle={`${t("nav.items.rebate", { ns: "config" })} ${t("versionSwitcher.sheetTitle", { ns: "config" })}`}
|
||||
sheetDescription={t("rebate.sheetDescription", { ns: "config" })}
|
||||
onDeleteVersion={handleDeleteVersion}
|
||||
onRollbackVersion={requestRollback}
|
||||
rollbackBusy={saving}
|
||||
/>
|
||||
}
|
||||
actions={
|
||||
@@ -460,12 +511,13 @@ export function RebateConfigDocScreen({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="flex items-center justify-between gap-3 rounded-xl border border-border/60 px-4 py-3">
|
||||
<p className="text-sm font-medium">{t("rebate.winEnjoy.label", { ns: "config" })}</p>
|
||||
<AdminStatusBadge status="enabled">
|
||||
{t("system.states.enabled", { ns: "config" })}
|
||||
</AdminStatusBadge>
|
||||
</div>
|
||||
<Alert className="border-border/80 bg-muted/30">
|
||||
<AlertDescription className="text-sm leading-relaxed">
|
||||
<span className="font-medium text-foreground">{t("rebate.winEnjoy.label", { ns: "config" })}</span>
|
||||
{" — "}
|
||||
{t("rebate.winEnjoy.pendingNote", { ns: "config" })}
|
||||
</AlertDescription>
|
||||
</Alert>
|
||||
|
||||
{!embedded ? (
|
||||
<div className="grid gap-1 text-sm">
|
||||
@@ -482,10 +534,35 @@ export function RebateConfigDocScreen({
|
||||
</>
|
||||
);
|
||||
|
||||
const rollbackDialog = (
|
||||
<Dialog open={rollbackOpen} onOpenChange={setRollbackOpen}>
|
||||
<DialogContent showCloseButton className="sm:max-w-md">
|
||||
<DialogHeader>
|
||||
<DialogTitle>{t("versionActions.rollbackDialog.title", { ns: "config" })}</DialogTitle>
|
||||
<DialogDescription>
|
||||
{t("versionActions.rollbackDialog.description", {
|
||||
ns: "config",
|
||||
version: rollbackTarget?.version_no ?? "—",
|
||||
})}
|
||||
</DialogDescription>
|
||||
</DialogHeader>
|
||||
<DialogFooter>
|
||||
<Button type="button" variant="outline" onClick={() => setRollbackOpen(false)}>
|
||||
{t("actions.cancel", { ns: "adminUsers" })}
|
||||
</Button>
|
||||
<Button type="button" onClick={() => void handleRollback()} disabled={!rollbackTarget || saving}>
|
||||
{t("versionActions.rollbackDialog.confirm", { ns: "config" })}
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
);
|
||||
|
||||
if (embedded) {
|
||||
return (
|
||||
<div className="space-y-4">
|
||||
{fieldsBlock}
|
||||
{rollbackDialog}
|
||||
<ConfirmDialog />
|
||||
</div>
|
||||
);
|
||||
@@ -497,6 +574,7 @@ export function RebateConfigDocScreen({
|
||||
toolbar={toolbarBlock}
|
||||
>
|
||||
{fieldsBlock}
|
||||
{rollbackDialog}
|
||||
<ConfirmDialog />
|
||||
</ConfigDocPage>
|
||||
);
|
||||
|
||||
@@ -31,6 +31,7 @@ import {
|
||||
import { Input } from "@/components/ui/input";
|
||||
import { Label } from "@/components/ui/label";
|
||||
import { ConfigVersionSwitcher } from "@/modules/config/config-version-switcher";
|
||||
import { RiskCapRuntimePanel } from "@/modules/config/risk-cap-runtime-panel";
|
||||
import {
|
||||
Table,
|
||||
TableBody,
|
||||
@@ -98,8 +99,9 @@ export function RiskCapDocScreen() {
|
||||
|
||||
const [defaultCapStr, setDefaultCapStr] = useState("");
|
||||
const [syncOpen, setSyncOpen] = useState(false);
|
||||
const [rollbackOpen, setRollbackOpen] = useState(false);
|
||||
const [rollbackTarget, setRollbackTarget] = useState<ConfigVersionSummary | null>(null);
|
||||
|
||||
const [occSearch, setOccSearch] = useState("");
|
||||
const amountCurrencyCode = "NPR";
|
||||
|
||||
const refreshList = useCallback(async () => {
|
||||
@@ -315,7 +317,7 @@ export function RiskCapDocScreen() {
|
||||
|
||||
function applyDefaultCap() {
|
||||
const n = parseAdminMajorToMinor(defaultCapStr, amountCurrencyCode);
|
||||
if (!Number.isFinite(n) || n <= 0) {
|
||||
if (n == null || !Number.isFinite(n) || n <= 0) {
|
||||
toast.error(t("riskCap.validation.enterValidCapAmount", { ns: "config" }));
|
||||
return;
|
||||
}
|
||||
@@ -327,14 +329,6 @@ export function RiskCapDocScreen() {
|
||||
toast.message(t("riskCap.savedLocalDraft", { ns: "config" }));
|
||||
}
|
||||
|
||||
const occFiltered = useMemo(() => {
|
||||
const q = occSearch.trim();
|
||||
if (!q) {
|
||||
return draftRows.filter((row) => !isDefaultRiskRow(row));
|
||||
}
|
||||
return draftRows.filter((r) => !isDefaultRiskRow(r) && r.normalized_number.includes(q));
|
||||
}, [draftRows, occSearch]);
|
||||
|
||||
const specialRows = useMemo(
|
||||
() => draftRows.map((row, index) => ({ row, index })).filter(({ row }) => !isDefaultRiskRow(row)),
|
||||
[draftRows],
|
||||
@@ -351,6 +345,49 @@ export function RiskCapDocScreen() {
|
||||
}
|
||||
}
|
||||
|
||||
function requestRollback(row: ConfigVersionSummary) {
|
||||
setRollbackTarget(row);
|
||||
setRollbackOpen(true);
|
||||
}
|
||||
|
||||
async function handleRollback() {
|
||||
if (!rollbackTarget) {
|
||||
return;
|
||||
}
|
||||
setSaving(true);
|
||||
try {
|
||||
const d = await postRiskCapVersion({
|
||||
reason: `rollback from v${rollbackTarget.version_no}`,
|
||||
clone_from_version_id: rollbackTarget.id,
|
||||
});
|
||||
toast.success(
|
||||
t("versionActions.rollbackSuccess", {
|
||||
ns: "config",
|
||||
fromVersion: rollbackTarget.version_no,
|
||||
version: d.version_no,
|
||||
}),
|
||||
);
|
||||
await refreshList();
|
||||
setSelectedId(String(d.id));
|
||||
setDetail(d);
|
||||
const mapped = d.items.map((it) => ({
|
||||
clientKey: `srv-${it.id}`,
|
||||
draw_id: it.draw_id,
|
||||
normalized_number: it.normalized_number,
|
||||
cap_amount: it.cap_amount,
|
||||
cap_type: it.cap_type,
|
||||
}));
|
||||
setDraftRows(mapped);
|
||||
syncDefaultCapFromRows(mapped);
|
||||
setRollbackOpen(false);
|
||||
setRollbackTarget(null);
|
||||
} catch (e) {
|
||||
toast.error(e instanceof LotteryApiBizError ? e.message : t("versionActions.rollbackFailed", { ns: "config" }));
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
}
|
||||
|
||||
return (
|
||||
<ConfigDocPage
|
||||
title={t("nav.items.risk-cap", { ns: "config" })}
|
||||
@@ -365,6 +402,8 @@ export function RiskCapDocScreen() {
|
||||
loading={loadingList}
|
||||
sheetTitle={`${t("nav.items.risk-cap", { ns: "config" })} ${t("versionSwitcher.sheetTitle", { ns: "config" })}`}
|
||||
onDeleteVersion={handleDeleteVersion}
|
||||
onRollbackVersion={requestRollback}
|
||||
rollbackBusy={saving}
|
||||
/>
|
||||
}
|
||||
actions={
|
||||
@@ -466,9 +505,6 @@ export function RiskCapDocScreen() {
|
||||
<TableRow>
|
||||
<TableHead className="w-[110px]">{t("riskCap.table.number", { ns: "config" })}</TableHead>
|
||||
<TableHead className="w-[140px]">{t("riskCap.table.capAmount", { ns: "config" })}</TableHead>
|
||||
<TableHead className="w-[90px] text-center">{t("riskCap.table.used", { ns: "config" })}</TableHead>
|
||||
<TableHead className="w-[90px] text-center">{t("riskCap.table.remaining", { ns: "config" })}</TableHead>
|
||||
<TableHead className="w-[72px] text-center">{t("riskCap.table.soldOut", { ns: "config" })}</TableHead>
|
||||
<TableHead className="w-[160px]">{t("riskCap.table.actions", { ns: "config" })}</TableHead>
|
||||
</TableRow>
|
||||
</TableHeader>
|
||||
@@ -513,9 +549,6 @@ export function RiskCapDocScreen() {
|
||||
</ConfigReadonlyValue>
|
||||
)}
|
||||
</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground tabular-nums text-sm">—</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground tabular-nums text-sm">—</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground text-sm">—</TableCell>
|
||||
<TableCell>
|
||||
{canEditDraft ? (
|
||||
<Button
|
||||
@@ -538,54 +571,7 @@ export function RiskCapDocScreen() {
|
||||
)}
|
||||
</ConfigSection>
|
||||
|
||||
<ConfigSection title={t("riskCap.occupancy.title", { ns: "config" })}>
|
||||
<div className="flex flex-wrap gap-3 items-end">
|
||||
<div className="grid gap-1">
|
||||
<Label htmlFor="occ-search">{t("riskCap.occupancy.searchLabel", { ns: "config" })}</Label>
|
||||
<Input
|
||||
id="occ-search"
|
||||
className="w-[140px] font-mono"
|
||||
placeholder={t("riskCap.occupancy.searchPlaceholder", { ns: "config" })}
|
||||
value={occSearch}
|
||||
onChange={(e) => setOccSearch(e.target.value)}
|
||||
/>
|
||||
</div>
|
||||
<Button type="button" variant="outline" onClick={() => toast.message(t("riskCap.occupancy.filterPending", { ns: "config" }))}>
|
||||
{t("riskCap.actions.filterPresets", { ns: "config" })}
|
||||
</Button>
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
onClick={() => toast.message(t("riskCap.occupancy.exportPending", { ns: "config" }))}
|
||||
>
|
||||
{t("riskCap.actions.exportCsv", { ns: "config" })}
|
||||
</Button>
|
||||
</div>
|
||||
<Table>
|
||||
<TableHeader>
|
||||
<TableRow>
|
||||
<TableHead>{t("riskCap.table.number", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.used", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.remaining", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.ratio", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.soldOut", { ns: "config" })}</TableHead>
|
||||
<TableHead className="w-[140px]">{t("riskCap.table.actions", { ns: "config" })}</TableHead>
|
||||
</TableRow>
|
||||
</TableHeader>
|
||||
<TableBody>
|
||||
{occFiltered.map((r) => (
|
||||
<TableRow key={`occ-${r.clientKey}`}>
|
||||
<TableCell className="font-mono text-sm">{r.normalized_number}</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground">—</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground">—</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground">—</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground">—</TableCell>
|
||||
<TableCell className="text-center text-muted-foreground">—</TableCell>
|
||||
</TableRow>
|
||||
))}
|
||||
</TableBody>
|
||||
</Table>
|
||||
</ConfigSection>
|
||||
<RiskCapRuntimePanel />
|
||||
|
||||
<Dialog open={syncOpen} onOpenChange={setSyncOpen}>
|
||||
<DialogContent showCloseButton className="sm:max-w-md">
|
||||
@@ -605,6 +591,29 @@ export function RiskCapDocScreen() {
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
|
||||
<Dialog open={rollbackOpen} onOpenChange={setRollbackOpen}>
|
||||
<DialogContent showCloseButton className="sm:max-w-md">
|
||||
<DialogHeader>
|
||||
<DialogTitle>{t("versionActions.rollbackDialog.title", { ns: "config" })}</DialogTitle>
|
||||
<DialogDescription>
|
||||
{t("versionActions.rollbackDialog.description", {
|
||||
ns: "config",
|
||||
version: rollbackTarget?.version_no ?? "—",
|
||||
})}
|
||||
</DialogDescription>
|
||||
</DialogHeader>
|
||||
<DialogFooter>
|
||||
<Button type="button" variant="outline" onClick={() => setRollbackOpen(false)}>
|
||||
{t("actions.cancel", { ns: "adminUsers" })}
|
||||
</Button>
|
||||
<Button type="button" onClick={() => void handleRollback()} disabled={!rollbackTarget || saving}>
|
||||
{t("versionActions.rollbackDialog.confirm", { ns: "config" })}
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
|
||||
<ConfirmDialog />
|
||||
</ConfigDocPage>
|
||||
);
|
||||
|
||||
275
src/modules/config/risk-cap-runtime-panel.tsx
Normal file
275
src/modules/config/risk-cap-runtime-panel.tsx
Normal file
@@ -0,0 +1,275 @@
|
||||
"use client";
|
||||
|
||||
import Link from "next/link";
|
||||
import { useCallback, useEffect, useMemo, useState } from "react";
|
||||
import { useTranslation } from "react-i18next";
|
||||
import { toast } from "sonner";
|
||||
|
||||
import { getAdminDraws } from "@/api/admin-draws";
|
||||
import { getAdminRiskPools } from "@/api/admin-risk";
|
||||
import { AdminTableExportButton } from "@/components/admin/admin-table-export-button";
|
||||
import { Button, buttonVariants } from "@/components/ui/button";
|
||||
import { Label } from "@/components/ui/label";
|
||||
import {
|
||||
Select,
|
||||
SelectContent,
|
||||
SelectItem,
|
||||
SelectTrigger,
|
||||
SelectValue,
|
||||
} from "@/components/ui/select";
|
||||
import {
|
||||
Table,
|
||||
TableBody,
|
||||
TableCell,
|
||||
TableHead,
|
||||
TableHeader,
|
||||
TableRow,
|
||||
} from "@/components/ui/table";
|
||||
import { Input } from "@/components/ui/input";
|
||||
import { ConfigSection } from "@/modules/config/config-section";
|
||||
import { formatAdminMinorUnits } from "@/lib/money";
|
||||
import { cn } from "@/lib/utils";
|
||||
import { LotteryApiBizError } from "@/types/api/errors";
|
||||
import type { AdminDrawListItem } from "@/types/api/admin-draws";
|
||||
import type { AdminRiskPoolRow } from "@/types/api/admin-risk";
|
||||
|
||||
type PoolFilter = "all" | "sold_out" | "high_risk";
|
||||
|
||||
export function RiskCapRuntimePanel() {
|
||||
const { t } = useTranslation(["config", "risk", "draws", "common"]);
|
||||
const [draws, setDraws] = useState<AdminDrawListItem[]>([]);
|
||||
const [drawsLoading, setDrawsLoading] = useState(true);
|
||||
const [drawId, setDrawId] = useState<string>("");
|
||||
|
||||
const [numberQ, setNumberQ] = useState("");
|
||||
const [appliedNumber, setAppliedNumber] = useState("");
|
||||
const [poolFilter, setPoolFilter] = useState<PoolFilter>("all");
|
||||
|
||||
const [pools, setPools] = useState<AdminRiskPoolRow[]>([]);
|
||||
const [currencyCode, setCurrencyCode] = useState<string | null>(null);
|
||||
const [poolsLoading, setPoolsLoading] = useState(false);
|
||||
const [poolsError, setPoolsError] = useState<string | null>(null);
|
||||
|
||||
const selectedDraw = useMemo(
|
||||
() => draws.find((d) => String(d.id) === drawId) ?? null,
|
||||
[draws, drawId],
|
||||
);
|
||||
|
||||
const loadDraws = useCallback(async () => {
|
||||
setDrawsLoading(true);
|
||||
try {
|
||||
const data = await getAdminDraws({ page: 1, per_page: 50 });
|
||||
setDraws(data.items);
|
||||
if (data.items.length > 0) {
|
||||
setDrawId((prev) => (prev === "" ? String(data.items[0].id) : prev));
|
||||
}
|
||||
} catch (e) {
|
||||
toast.error(e instanceof LotteryApiBizError ? e.message : t("errors.loadFailed", { ns: "common" }));
|
||||
setDraws([]);
|
||||
} finally {
|
||||
setDrawsLoading(false);
|
||||
}
|
||||
}, [t]);
|
||||
|
||||
const loadPools = useCallback(async () => {
|
||||
if (!drawId) {
|
||||
setPools([]);
|
||||
return;
|
||||
}
|
||||
const id = Number(drawId);
|
||||
if (!Number.isFinite(id)) {
|
||||
return;
|
||||
}
|
||||
setPoolsLoading(true);
|
||||
setPoolsError(null);
|
||||
try {
|
||||
const data = await getAdminRiskPools(id, {
|
||||
page: 1,
|
||||
per_page: 200,
|
||||
normalized_number: appliedNumber.trim() || undefined,
|
||||
sold_out_only: poolFilter === "sold_out",
|
||||
high_risk_only: poolFilter === "high_risk",
|
||||
sort: poolFilter === "high_risk" ? "usage_desc" : "number_asc",
|
||||
});
|
||||
setPools(data.items);
|
||||
setCurrencyCode(data.currency_code);
|
||||
} catch (e) {
|
||||
setPoolsError(e instanceof LotteryApiBizError ? e.message : t("errors.loadFailed", { ns: "common" }));
|
||||
setPools([]);
|
||||
} finally {
|
||||
setPoolsLoading(false);
|
||||
}
|
||||
}, [appliedNumber, drawId, poolFilter, t]);
|
||||
|
||||
useEffect(() => {
|
||||
void loadDraws();
|
||||
}, [loadDraws]);
|
||||
|
||||
useEffect(() => {
|
||||
void loadPools();
|
||||
}, [loadPools]);
|
||||
|
||||
const riskBase = drawId ? `/admin/draws/${drawId}/risk` : null;
|
||||
|
||||
return (
|
||||
<ConfigSection
|
||||
title={t("riskCap.runtime.title", { ns: "config" })}
|
||||
description={t("riskCap.runtime.description", { ns: "config" })}
|
||||
>
|
||||
<div className="flex flex-wrap items-end gap-3">
|
||||
<div className="grid min-w-[12rem] flex-1 gap-1.5">
|
||||
<Label htmlFor="risk-cap-draw">{t("riskCap.runtime.drawLabel", { ns: "config" })}</Label>
|
||||
<Select
|
||||
value={drawId || undefined}
|
||||
onValueChange={(v) => setDrawId(v ?? "")}
|
||||
disabled={drawsLoading || draws.length === 0}
|
||||
>
|
||||
<SelectTrigger id="risk-cap-draw" className="font-mono">
|
||||
<SelectValue placeholder={t("riskCap.runtime.drawPlaceholder", { ns: "config" })} />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
{draws.map((d) => (
|
||||
<SelectItem key={d.id} value={String(d.id)}>
|
||||
{d.draw_no} · {d.status}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div>
|
||||
{riskBase ? (
|
||||
<div className="flex flex-wrap gap-2 pb-0.5">
|
||||
<Link href={`${riskBase}/pools`} className={cn(buttonVariants({ variant: "outline", size: "sm" }))}>
|
||||
{t("subnav.riskPools", { ns: "draws" })}
|
||||
</Link>
|
||||
<Link href={`${riskBase}/hot`} className={cn(buttonVariants({ variant: "outline", size: "sm" }))}>
|
||||
{t("subnav.riskHot", { ns: "draws" })}
|
||||
</Link>
|
||||
<Link href={`${riskBase}/sold-out`} className={cn(buttonVariants({ variant: "outline", size: "sm" }))}>
|
||||
{t("subnav.riskSoldOut", { ns: "draws" })}
|
||||
</Link>
|
||||
<Link href={`${riskBase}/occupancy`} className={cn(buttonVariants({ variant: "outline", size: "sm" }))}>
|
||||
{t("subnav.riskLockLogs", { ns: "draws" })}
|
||||
</Link>
|
||||
</div>
|
||||
) : null}
|
||||
</div>
|
||||
|
||||
{drawId ? (
|
||||
<>
|
||||
<div className="flex flex-wrap items-end gap-3">
|
||||
<div className="grid gap-1.5">
|
||||
<Label htmlFor="risk-cap-number-q">{t("riskCap.occupancy.searchLabel", { ns: "config" })}</Label>
|
||||
<Input
|
||||
id="risk-cap-number-q"
|
||||
className="w-[140px] font-mono"
|
||||
placeholder={t("riskCap.occupancy.searchPlaceholder", { ns: "config" })}
|
||||
value={numberQ}
|
||||
onChange={(e) => setNumberQ(e.target.value)}
|
||||
/>
|
||||
</div>
|
||||
<div className="flex flex-wrap gap-2">
|
||||
{(
|
||||
[
|
||||
{ id: "all", label: t("riskCap.runtime.filterAll", { ns: "config" }) },
|
||||
{ id: "sold_out", label: t("riskCap.runtime.filterSoldOut", { ns: "config" }) },
|
||||
{ id: "high_risk", label: t("riskCap.runtime.filterHighRisk", { ns: "config" }) },
|
||||
] as const
|
||||
).map((f) => (
|
||||
<Button
|
||||
key={f.id}
|
||||
type="button"
|
||||
size="sm"
|
||||
variant={poolFilter === f.id ? "default" : "outline"}
|
||||
onClick={() => setPoolFilter(f.id)}
|
||||
>
|
||||
{f.label}
|
||||
</Button>
|
||||
))}
|
||||
</div>
|
||||
<Button
|
||||
type="button"
|
||||
size="sm"
|
||||
variant="secondary"
|
||||
onClick={() => {
|
||||
setAppliedNumber(numberQ.trim());
|
||||
}}
|
||||
>
|
||||
{t("actions.search", { ns: "common" })}
|
||||
</Button>
|
||||
<Button type="button" size="sm" variant="outline" disabled={poolsLoading} onClick={() => void loadPools()}>
|
||||
{t("versionActions.refresh", { ns: "config" })}
|
||||
</Button>
|
||||
{pools.length > 0 ? (
|
||||
<AdminTableExportButton
|
||||
tableId="risk-cap-runtime-pools"
|
||||
filename={`risk-pools-${selectedDraw?.draw_no ?? drawId}`}
|
||||
/>
|
||||
) : null}
|
||||
</div>
|
||||
|
||||
{poolsError ? <p className="text-sm text-destructive">{poolsError}</p> : null}
|
||||
|
||||
<div className="admin-table-shell">
|
||||
<Table id="risk-cap-runtime-pools">
|
||||
<TableHeader>
|
||||
<TableRow>
|
||||
<TableHead>{t("riskCap.table.number", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.used", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.remaining", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.ratio", { ns: "config" })}</TableHead>
|
||||
<TableHead className="text-center">{t("riskCap.table.soldOut", { ns: "config" })}</TableHead>
|
||||
</TableRow>
|
||||
</TableHeader>
|
||||
<TableBody>
|
||||
{poolsLoading ? (
|
||||
<TableRow>
|
||||
<TableCell colSpan={5} className="text-muted-foreground">
|
||||
{t("states.loading", { ns: "common" })}
|
||||
</TableCell>
|
||||
</TableRow>
|
||||
) : pools.length === 0 ? (
|
||||
<TableRow>
|
||||
<TableCell colSpan={5} className="text-muted-foreground">
|
||||
{t("states.noData", { ns: "common" })}
|
||||
</TableCell>
|
||||
</TableRow>
|
||||
) : (
|
||||
pools.map((row) => (
|
||||
<TableRow
|
||||
key={row.normalized_number}
|
||||
className={cn(
|
||||
row.is_sold_out && "bg-destructive/5",
|
||||
!row.is_sold_out && (row.usage_ratio ?? 0) >= 0.8 && "bg-amber-500/10",
|
||||
)}
|
||||
>
|
||||
<TableCell className="font-mono text-sm">{row.normalized_number}</TableCell>
|
||||
<TableCell className="text-center text-xs tabular-nums">
|
||||
{formatAdminMinorUnits(row.locked_amount, currencyCode ?? undefined)}
|
||||
</TableCell>
|
||||
<TableCell className="text-center text-xs tabular-nums">
|
||||
{formatAdminMinorUnits(row.remaining_amount, currencyCode ?? undefined)}
|
||||
</TableCell>
|
||||
<TableCell className="text-center text-xs tabular-nums">
|
||||
{row.usage_ratio != null ? `${Math.round(row.usage_ratio * 100)}%` : "—"}
|
||||
</TableCell>
|
||||
<TableCell className="text-center text-xs">
|
||||
{row.is_sold_out
|
||||
? t("riskCap.runtime.soldYes", { ns: "config" })
|
||||
: t("riskCap.runtime.soldNo", { ns: "config" })}
|
||||
</TableCell>
|
||||
</TableRow>
|
||||
))
|
||||
)}
|
||||
</TableBody>
|
||||
</Table>
|
||||
</div>
|
||||
<p className="text-xs text-muted-foreground">
|
||||
{t("riskCap.runtime.manageHint", { ns: "config" })}
|
||||
</p>
|
||||
</>
|
||||
) : (
|
||||
<p className="text-sm text-muted-foreground">{t("riskCap.runtime.noDraws", { ns: "config" })}</p>
|
||||
)}
|
||||
</ConfigSection>
|
||||
);
|
||||
}
|
||||
@@ -12,7 +12,7 @@ const segments = [
|
||||
{ suffix: "/results", key: "results", label: "subnav.results" },
|
||||
{ suffix: "/finance", key: "finance", label: "subnav.finance" },
|
||||
{ suffix: "/review", key: "review", label: "subnav.review" },
|
||||
{ suffix: "/risk/occupancy", key: "riskOccupancy", label: "subnav.riskOccupancy" },
|
||||
{ suffix: "/risk/occupancy", key: "riskLockLogs", label: "subnav.riskLockLogs" },
|
||||
{ suffix: "/risk/hot", key: "riskHot", label: "subnav.riskHot" },
|
||||
{ suffix: "/risk/sold-out", key: "riskSoldOut", label: "subnav.riskSoldOut" },
|
||||
{ suffix: "/risk/pools", key: "riskPools", label: "subnav.riskPools" },
|
||||
|
||||
156
src/modules/reports/report-jobs-panel.tsx
Normal file
156
src/modules/reports/report-jobs-panel.tsx
Normal file
@@ -0,0 +1,156 @@
|
||||
"use client";
|
||||
|
||||
import { useCallback, useEffect, useState } from "react";
|
||||
import { useTranslation } from "react-i18next";
|
||||
import { toast } from "sonner";
|
||||
import { Download, RefreshCw } from "lucide-react";
|
||||
|
||||
import { downloadAdminReportJob, getAdminReportJobs } from "@/api/admin-report-jobs";
|
||||
import { AdminStatusBadge } from "@/components/admin/admin-status-badge";
|
||||
import { Button } from "@/components/ui/button";
|
||||
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
||||
import {
|
||||
Table,
|
||||
TableBody,
|
||||
TableCell,
|
||||
TableHead,
|
||||
TableHeader,
|
||||
TableRow,
|
||||
} from "@/components/ui/table";
|
||||
import { useAdminDateTimeFormatter } from "@/hooks/use-admin-datetime-formatter";
|
||||
import { LotteryApiBizError } from "@/types/api/errors";
|
||||
import type { AdminReportJobRow } from "@/types/api/admin-report-jobs";
|
||||
|
||||
function downloadBlob(blob: Blob, filename: string): void {
|
||||
const url = URL.createObjectURL(blob);
|
||||
const anchor = document.createElement("a");
|
||||
anchor.href = url;
|
||||
anchor.download = filename;
|
||||
document.body.appendChild(anchor);
|
||||
anchor.click();
|
||||
anchor.remove();
|
||||
URL.revokeObjectURL(url);
|
||||
}
|
||||
|
||||
type ReportJobsPanelProps = {
|
||||
canExport: boolean;
|
||||
refreshToken?: number;
|
||||
};
|
||||
|
||||
export function ReportJobsPanel({ canExport, refreshToken = 0 }: ReportJobsPanelProps) {
|
||||
const { t } = useTranslation(["reports", "common"]);
|
||||
const formatTs = useAdminDateTimeFormatter();
|
||||
const [jobs, setJobs] = useState<AdminReportJobRow[]>([]);
|
||||
const [loading, setLoading] = useState(true);
|
||||
const [downloadingId, setDownloadingId] = useState<number | null>(null);
|
||||
|
||||
const loadJobs = useCallback(async () => {
|
||||
setLoading(true);
|
||||
try {
|
||||
const data = await getAdminReportJobs({ page: 1, per_page: 10 });
|
||||
setJobs(data.items);
|
||||
} catch (e) {
|
||||
toast.error(e instanceof LotteryApiBizError ? e.message : t("tasks.loadFailed"));
|
||||
setJobs([]);
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
}, [t]);
|
||||
|
||||
useEffect(() => {
|
||||
void loadJobs();
|
||||
}, [loadJobs, refreshToken]);
|
||||
|
||||
async function handleDownload(job: AdminReportJobRow): Promise<void> {
|
||||
if (!canExport || job.status !== "completed") {
|
||||
return;
|
||||
}
|
||||
setDownloadingId(job.id);
|
||||
try {
|
||||
const { blob, filename } = await downloadAdminReportJob(job.id);
|
||||
const fallback = `${job.job_no}.${job.export_format}`;
|
||||
downloadBlob(blob, filename ?? fallback);
|
||||
toast.success(t("tasks.downloadSuccess", { jobNo: job.job_no }));
|
||||
} catch (e) {
|
||||
toast.error(e instanceof LotteryApiBizError ? e.message : t("tasks.downloadFailed"));
|
||||
} finally {
|
||||
setDownloadingId(null);
|
||||
}
|
||||
}
|
||||
|
||||
function reportTypeLabel(reportType: string): string {
|
||||
const key = `jobTypes.${reportType}`;
|
||||
const label = t(key);
|
||||
return label === key ? reportType : label;
|
||||
}
|
||||
|
||||
return (
|
||||
<Card className="admin-list-card">
|
||||
<CardHeader className="admin-list-header flex flex-row items-center justify-between gap-3 pb-4">
|
||||
<CardTitle className="admin-list-title">{t("recentTasks")}</CardTitle>
|
||||
<Button type="button" variant="outline" size="sm" disabled={loading} onClick={() => void loadJobs()}>
|
||||
<RefreshCw data-icon="inline-start" className={loading ? "animate-spin" : undefined} />
|
||||
{t("tasks.refresh")}
|
||||
</Button>
|
||||
</CardHeader>
|
||||
<CardContent className="pt-2">
|
||||
<p className="mb-3 text-xs text-muted-foreground">{t("exportHint")}</p>
|
||||
<Table>
|
||||
<TableHeader>
|
||||
<TableRow>
|
||||
<TableHead>{t("tasks.columns.jobNo")}</TableHead>
|
||||
<TableHead>{t("tasks.columns.report")}</TableHead>
|
||||
<TableHead>{t("tasks.columns.format")}</TableHead>
|
||||
<TableHead>{t("tasks.columns.status")}</TableHead>
|
||||
<TableHead>{t("tasks.columns.createdAt")}</TableHead>
|
||||
<TableHead>{t("tasks.columns.actions")}</TableHead>
|
||||
</TableRow>
|
||||
</TableHeader>
|
||||
<TableBody>
|
||||
{loading ? (
|
||||
<TableRow>
|
||||
<TableCell colSpan={6} className="text-muted-foreground">
|
||||
{t("states.loading", { ns: "common" })}
|
||||
</TableCell>
|
||||
</TableRow>
|
||||
) : jobs.length === 0 ? (
|
||||
<TableRow>
|
||||
<TableCell colSpan={6} className="text-muted-foreground">
|
||||
{t("taskEmpty")}
|
||||
</TableCell>
|
||||
</TableRow>
|
||||
) : (
|
||||
jobs.map((job) => (
|
||||
<TableRow key={job.id}>
|
||||
<TableCell className="font-mono text-xs">{job.job_no}</TableCell>
|
||||
<TableCell className="text-sm">{reportTypeLabel(job.report_type)}</TableCell>
|
||||
<TableCell className="uppercase">{job.export_format}</TableCell>
|
||||
<TableCell>
|
||||
<AdminStatusBadge status={job.status}>
|
||||
{t(`tasks.status.${job.status}`, { defaultValue: job.status })}
|
||||
</AdminStatusBadge>
|
||||
</TableCell>
|
||||
<TableCell className="text-xs text-muted-foreground">
|
||||
{formatTs(job.created_at ?? job.finished_at)}
|
||||
</TableCell>
|
||||
<TableCell>
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
size="sm"
|
||||
disabled={!canExport || job.status !== "completed" || downloadingId === job.id}
|
||||
onClick={() => void handleDownload(job)}
|
||||
>
|
||||
<Download data-icon="inline-start" />
|
||||
{t("tasks.download")}
|
||||
</Button>
|
||||
</TableCell>
|
||||
</TableRow>
|
||||
))
|
||||
)}
|
||||
</TableBody>
|
||||
</Table>
|
||||
</CardContent>
|
||||
</Card>
|
||||
);
|
||||
}
|
||||
@@ -29,12 +29,20 @@ import {
|
||||
} from "@/lib/admin-play-types";
|
||||
import { getAdminDraws, getAdminDrawFinanceSummary } from "@/api/admin-draws";
|
||||
import { getAdminPlayers } from "@/api/admin-player";
|
||||
import { downloadAdminReportJob, postAdminReportJob } from "@/api/admin-report-jobs";
|
||||
import {
|
||||
getAdminReportDailyProfit,
|
||||
getAdminReportPlayDimension,
|
||||
getAdminReportPlayerWinLoss,
|
||||
getAdminReportRebateCommission,
|
||||
} from "@/api/admin-reports";
|
||||
import {
|
||||
buildReportJobParameters,
|
||||
REPORT_UI_SERVER_FULL_EXPORT,
|
||||
REPORT_UI_TO_JOB_TYPE,
|
||||
type ReportUiKey,
|
||||
} from "@/lib/report-export-map";
|
||||
import { ReportJobsPanel } from "@/modules/reports/report-jobs-panel";
|
||||
import { getAdminRiskPoolDetail, getAdminRiskPools } from "@/api/admin-risk";
|
||||
import { getAdminUsers } from "@/api/admin-users";
|
||||
import { getAdminTransferOrders } from "@/api/admin-wallet";
|
||||
@@ -387,6 +395,7 @@ export function ReportsConsole() {
|
||||
const [page, setPage] = useState(1);
|
||||
const [perPage, setPerPage] = useState(20);
|
||||
const [exporting, setExporting] = useState<ExportFormat | null>(null);
|
||||
const [jobRefreshToken, setJobRefreshToken] = useState(0);
|
||||
const [search, setSearch] = useState<SearchState>(emptySearch);
|
||||
const [playOptions, setPlayOptions] = useState<PlayOption[]>([]);
|
||||
|
||||
@@ -778,10 +787,55 @@ export function ReportsConsole() {
|
||||
setPage(1);
|
||||
}
|
||||
|
||||
const usesServerExport = REPORT_UI_SERVER_FULL_EXPORT.has(selectedReport.key as ReportUiKey);
|
||||
|
||||
async function exportViaServer(format: ExportFormat): Promise<void> {
|
||||
if (!canExportReports) {
|
||||
return;
|
||||
}
|
||||
setExporting(format);
|
||||
try {
|
||||
const parameters = buildReportJobParameters(selectedReport.key as ReportUiKey, {
|
||||
dateFrom: filters.dateFrom,
|
||||
dateTo: filters.dateTo,
|
||||
playerId: filters.playerId,
|
||||
play: filters.play,
|
||||
operatorId: filters.operatorId,
|
||||
drawId: filters.drawId,
|
||||
drawNo: filters.drawNo,
|
||||
number: filters.number,
|
||||
});
|
||||
const job = await postAdminReportJob({
|
||||
report_type: REPORT_UI_TO_JOB_TYPE[selectedReport.key as ReportUiKey],
|
||||
export_format: format === "excel" ? "xlsx" : "csv",
|
||||
parameters,
|
||||
});
|
||||
setJobRefreshToken((n) => n + 1);
|
||||
const { blob, filename } = await downloadAdminReportJob(job.id);
|
||||
const ext = job.export_format === "xlsx" ? "xlsx" : "csv";
|
||||
downloadBlob(blob, filename ?? `${exportFileBase}.${ext}`);
|
||||
toast.success(
|
||||
t("exportServerSuccess", {
|
||||
report: t(`items.${selectedReport.key}.title`),
|
||||
format: t(`formats.${format}`),
|
||||
jobNo: job.job_no,
|
||||
}),
|
||||
);
|
||||
} catch (err) {
|
||||
toast.error(err instanceof LotteryApiBizError ? err.message : t("exportFailed"));
|
||||
} finally {
|
||||
setExporting(null);
|
||||
}
|
||||
}
|
||||
|
||||
function exportReport(format: ExportFormat): void {
|
||||
if (!canExportReports) {
|
||||
return;
|
||||
}
|
||||
if (usesServerExport) {
|
||||
void exportViaServer(format);
|
||||
return;
|
||||
}
|
||||
if (!result || result.rows.length === 0) {
|
||||
toast.info(t("empty"));
|
||||
return;
|
||||
@@ -1273,24 +1327,39 @@ export function ReportsConsole() {
|
||||
<div>
|
||||
<CardTitle className="admin-list-title">{t("preview.title")}</CardTitle>
|
||||
</div>
|
||||
<div className="flex shrink-0 gap-2">
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
disabled={!canExportReports || !result || exporting !== null}
|
||||
onClick={() => exportReport("csv")}
|
||||
>
|
||||
<FileDown data-icon="inline-start" />
|
||||
{t("formats.csv")}
|
||||
</Button>
|
||||
<Button
|
||||
type="button"
|
||||
disabled={!canExportReports || !result || exporting !== null}
|
||||
onClick={() => exportReport("excel")}
|
||||
>
|
||||
<FileSpreadsheet data-icon="inline-start" />
|
||||
{t("formats.excel")}
|
||||
</Button>
|
||||
<div className="flex flex-col items-end gap-2 sm:flex-row sm:items-center">
|
||||
{usesServerExport ? (
|
||||
<p className="text-xs text-muted-foreground sm:mr-2">{t("exportServerHint")}</p>
|
||||
) : (
|
||||
<p className="text-xs text-muted-foreground sm:mr-2">{t("exportClientHint")}</p>
|
||||
)}
|
||||
<div className="flex shrink-0 gap-2">
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
disabled={
|
||||
!canExportReports ||
|
||||
exporting !== null ||
|
||||
(!usesServerExport && (!result || result.rows.length === 0))
|
||||
}
|
||||
onClick={() => exportReport("csv")}
|
||||
>
|
||||
<FileDown data-icon="inline-start" />
|
||||
{usesServerExport ? t("formats.csvServer") : t("formats.csv")}
|
||||
</Button>
|
||||
<Button
|
||||
type="button"
|
||||
disabled={
|
||||
!canExportReports ||
|
||||
exporting !== null ||
|
||||
(!usesServerExport && (!result || result.rows.length === 0))
|
||||
}
|
||||
onClick={() => exportReport("excel")}
|
||||
>
|
||||
<FileSpreadsheet data-icon="inline-start" />
|
||||
{usesServerExport ? t("formats.excelServer") : t("formats.excel")}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
</CardHeader>
|
||||
<CardContent className="space-y-4 pt-4">
|
||||
@@ -1329,6 +1398,8 @@ export function ReportsConsole() {
|
||||
</Card>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<ReportJobsPanel canExport={canExportReports} refreshToken={jobRefreshToken} />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
39
src/types/api/admin-report-jobs.ts
Normal file
39
src/types/api/admin-report-jobs.ts
Normal file
@@ -0,0 +1,39 @@
|
||||
export type AdminReportJobExportFormat = "csv" | "xlsx";
|
||||
|
||||
export type AdminReportJobStatus = "pending" | "processing" | "completed" | "failed" | string;
|
||||
|
||||
export type AdminReportJobRow = {
|
||||
id: number;
|
||||
job_no: string;
|
||||
admin_user_id: number | null;
|
||||
report_type: string;
|
||||
export_format: AdminReportJobExportFormat;
|
||||
status: AdminReportJobStatus;
|
||||
finished_at: string | null;
|
||||
created_at: string | null;
|
||||
};
|
||||
|
||||
export type AdminReportJobListData = {
|
||||
items: AdminReportJobRow[];
|
||||
meta: {
|
||||
current_page: number;
|
||||
per_page: number;
|
||||
total: number;
|
||||
last_page: number;
|
||||
};
|
||||
};
|
||||
|
||||
export type AdminReportJobCreatePayload = {
|
||||
report_type: string;
|
||||
export_format?: AdminReportJobExportFormat;
|
||||
parameters?: Record<string, unknown>;
|
||||
filter_json?: Record<string, unknown>;
|
||||
};
|
||||
|
||||
export type AdminReportJobCreateResult = {
|
||||
id: number;
|
||||
job_no: string;
|
||||
report_type: string;
|
||||
export_format: AdminReportJobExportFormat;
|
||||
status: AdminReportJobStatus;
|
||||
};
|
||||
Reference in New Issue
Block a user