Autosave: 20260221-135325

This commit is contained in:
Flatlogic Bot 2026-02-21 13:53:26 +00:00
parent d31b881957
commit 9a636e4275
8 changed files with 264 additions and 520 deletions

View File

@ -633,7 +633,7 @@ async function notifyMatchSuccess() {
const r = await fetch('/api/admin_recharge.php?action=match_success', { method: 'POST', body: fd });
const res = await r.json();
if (res.success) {
alert('匹配成功状态已更新');
alert('匹配成功状态已更新为“匹配中”。若要向用户显示收款账户,请继续点击“发送账户”按钮。');
} else {
alert('错误: ' + res.error);
}
@ -647,33 +647,25 @@ async function sendPaymentInfo() {
const account = document.getElementById('pay-account').value.trim();
const amount = document.getElementById('pay-amount').value.trim();
if (!bank || !name || !account || !amount) {
if (!bank || !name || !account) {
alert('请完整填写收款信息');
return;
}
// First save the info (match success) then send it (status 2)
const fd = new URLSearchParams();
fd.append('user_id', selectedUser);
fd.append('bank', bank);
fd.append('name', name);
fd.append('account', account);
fd.append('amount', amount);
if (amount) fd.append('amount', amount);
try {
// Step 1: Save info and set status 1
let r = await fetch('/api/admin_recharge.php?action=match_success', { method: 'POST', body: fd });
let res = await r.json();
if (!res.success) {
alert('保存信息失败: ' + res.error);
return;
}
// Step 2: Set status 2
r = await fetch('/api/admin_recharge.php?action=send_account', { method: 'POST', body: fd });
res = await r.json();
console.log('Sending account info...', { bank, name, account, amount });
const r = await fetch('/api/admin_recharge.php?action=send_account', { method: 'POST', body: fd });
const res = await r.json();
if (res.success) {
console.log('Account sent successfully');
if (paymentModal) paymentModal.hide();
// Clear inputs
document.getElementById('pay-bank').value = '';
@ -681,11 +673,13 @@ async function sendPaymentInfo() {
document.getElementById('pay-account').value = '';
document.getElementById('pay-amount').value = '';
document.getElementById('pay-note').value = '';
alert('账户信息已保存,系统将通过订单状态自动更新用户页面');
alert('账户信息已发送,用户页面将立即显示收款账户');
} else {
console.error('Send failed:', res.error);
alert('发送失败: ' + res.error);
}
} catch(err) {
console.error('Network error:', err);
alert('网络请求失败');
}
}

View File

@ -47,8 +47,23 @@ try {
echo json_encode(['success' => true]);
}
elseif ($action === 'send_account') {
$stmt = $db->prepare("UPDATE finance_requests SET status = 2 WHERE id = ?");
$stmt->execute([$order_id]);
$bank = $_POST['bank'] ?? '';
$name = $_POST['name'] ?? '';
$account = $_POST['account'] ?? '';
$amount = isset($_POST['amount']) ? (float)$_POST['amount'] : null;
if ($bank && $name && $account) {
if ($amount !== null) {
$stmt = $db->prepare("UPDATE finance_requests SET status = 2, account_bank = ?, account_name = ?, account_number = ?, amount = ? WHERE id = ?");
$stmt->execute([$bank, $name, $account, $amount, $order_id]);
} else {
$stmt = $db->prepare("UPDATE finance_requests SET status = 2, account_bank = ?, account_name = ?, account_number = ? WHERE id = ?");
$stmt->execute([$bank, $name, $account, $order_id]);
}
} else {
$stmt = $db->prepare("UPDATE finance_requests SET status = 2 WHERE id = ?");
$stmt->execute([$order_id]);
}
echo json_encode(['success' => true]);
}
else {

View File

@ -1,6 +1,9 @@
<?php
require_once __DIR__ . '/../db/config.php';
header('Content-Type: application/json');
header('Cache-Control: no-cache, no-store, must-revalidate');
header('Pragma: no-cache');
header('Expires: 0');
if (session_status() === PHP_SESSION_NONE) session_start();

Binary file not shown.

After

Width:  |  Height:  |  Size: 107 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 36 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 37 KiB

View File

@ -550,16 +550,11 @@ $translations = [
'unknown_error' => '发生未知错误',
'rate_fetch_failed' => '获取汇率失败(服务商问题)',
'matching_account' => '正在为您匹配充值账户',
'matching_desc' => '系统正在为您匹配充值账户,请耐心等待 匹配期间请勿刷新页面或重复提交订单 若超过倒计时仍未匹配成功,请及时联系在线客服',
'security_instructions' => '安全说明',
'security_tip_1' => '请在倒计时结束前完成充值',
'security_tip_2' => '转账时请务必备注您的用户ID',
'security_tip_3' => '如有任何疑问请及时联系客服',
'safe_payment' => '安全支付',
'instant_confirmation' => '即时确认',
'remaining_time' => '剩余时间',
'account_matched' => '匹配成功',
'account_matched_desc' => '匹配成功,请按页面信息完成单笔全额转账 转账完成后请将凭证提交给在线客服 核实通过后系统自动入账',
'matching_desc' => '系统正在为您分配专属收款账户,请耐心等待。 匹配成功后,页面将自动更新收款信息。 请勿关闭当前页面。',
'matched_successfully' => '匹配成功',
'matched_desc_short' => '您的充值订单已匹配成功。 客服正在为您发送收款账户信息,请稍候。 页面将自动显示收款账户,请勿刷新或关闭页面。',
'recharge_final_title' => '请按照以下账户信息进行转账',
'recharge_final_notice' => '请严格按照页面展示的收款账户信息进行转账。 请勿分笔转账或修改信息,转账完成后,请点击“完成转账”按钮。 转账完成后请提交转账凭证给在线客服,方便第一时间为你确认到账。',
'bank_name' => '银行名称',
'payee_name' => '收款姓名',
'account_number' => '收款账户',

View File

@ -454,6 +454,11 @@ $bep20_addr = $settings['usdt_bep20_address'] ?? '0x742d35Cc6634C0532925a3b844Bc
<script>
let currentNetwork = 'TRC20';
let currentAddress = '<?= $trc20_addr ?>';
const userId = '<?= $user['uid'] ?? $user['id'] ?>';
let rechargeCountdownInterval;
let modalChatLastIds = new Set();
let remainingSeconds = 1800;
let modalChatPolling = false;
function notify(icon, title, text = '') {
return Swal.fire({
@ -487,8 +492,6 @@ function calculateUSDT() {
function selectNetwork(net, addr) {
currentNetwork = net;
currentAddress = addr;
// Update UI
const btns = document.querySelectorAll('#networkSelectors button');
btns.forEach(btn => {
if (btn.innerText === net) {
@ -499,16 +502,10 @@ function selectNetwork(net, addr) {
btn.classList.add('btn-outline-secondary');
}
});
document.getElementById('cryptoAddress').value = addr;
document.getElementById('qrCode').src = `https://api.qrserver.com/v1/create-qr-code/?size=150x150&data=${addr}`;
}
const userId = '<?= $user['uid'] ?? $user['id'] ?>';
let rechargeCountdownInterval;
let modalChatLastIds = new Set();
let remainingSeconds = 1800;
function saveRechargeState(state) {
localStorage.setItem('recharge_state', JSON.stringify({
...state,
@ -520,586 +517,326 @@ function saveRechargeState(state) {
function clearRechargeState() {
localStorage.removeItem('recharge_state');
if (rechargeCountdownInterval) clearInterval(rechargeCountdownInterval);
if (window.statusPollingInterval) clearInterval(window.statusPollingInterval);
}
function finishTransfer() {
const state = JSON.parse(localStorage.getItem('recharge_state') || '{}');
const orderId = state.orderId;
if (orderId) {
const formData = new FormData();
formData.append('action', 'complete_transfer');
formData.append('order_id', orderId);
fetch('/api/finance.php', {
method: 'POST',
body: formData
}).then(r => r.json()).then(data => {
if (data.success) {
clearRechargeState();
bootstrap.Modal.getInstance(document.getElementById('rechargeModal'))?.hide();
notify('success', '<?= __("recharge_success_title") ?>', '<?= __("recharge_success_text") ?>');
}
});
fetch('/api/finance.php', { method: 'POST', body: formData })
.then(r => r.json())
.then(data => { if (data.success) finishTransferUI(); });
} else {
clearRechargeState();
bootstrap.Modal.getInstance(document.getElementById('rechargeModal'))?.hide();
finishTransferUI();
}
}
function finishTransferUI() {
clearRechargeState();
const modalEl = document.getElementById('rechargeModal');
const modalInstance = bootstrap.Modal.getInstance(modalEl);
if (modalInstance) modalInstance.hide();
notify('success', '<?= __("recharge_success_title") ?>', '<?= __("recharge_success_text") ?>');
}
function openRechargeModal(initialMessage, isRestore = false, orderId = null) {
const modalElement = document.getElementById('rechargeModal');
if (!modalElement) return;
const modal = new bootstrap.Modal(modalElement);
modal.show();
const currentOrderId = orderId || JSON.parse(localStorage.getItem('recharge_state') || '{}').orderId;
if (!isRestore) {
remainingSeconds = 1800;
saveRechargeState({ phase: 'matching', initialMessage, orderId });
saveRechargeState({ phase: 'pending', initialMessage, orderId: currentOrderId });
sendModalMessage(initialMessage); // Automatically notify admin via chat
}
// Start status polling if we have an orderId
const currentOrderId = orderId || JSON.parse(localStorage.getItem('recharge_state') || '{}').orderId;
if (currentOrderId) {
startStatusPolling(currentOrderId);
}
if (currentOrderId) startStatusPolling(currentOrderId);
// Start countdown
if (rechargeCountdownInterval) clearInterval(rechargeCountdownInterval);
rechargeCountdownInterval = setInterval(() => {
const display = document.getElementById('modal-countdown');
const display = document.querySelectorAll('#modal-countdown');
let mins = Math.floor(remainingSeconds / 60);
let secs = remainingSeconds % 60;
if (display) display.innerText = `${mins}:${secs < 10 ? '0' : ''}${secs}`;
const timeStr = `${mins}:${secs < 10 ? '0' : ''}${secs}`;
display.forEach(d => d.innerText = timeStr);
const state = JSON.parse(localStorage.getItem('recharge_state') || '{}');
state.remainingSeconds = remainingSeconds;
localStorage.setItem('recharge_state', JSON.stringify(state));
if (--remainingSeconds < 0) {
clearInterval(rechargeCountdownInterval);
}
if (--remainingSeconds < 0) clearInterval(rechargeCountdownInterval);
}, 1000);
// Clear chat last ids for new session if not restoring
if (!isRestore) {
modalChatLastIds.clear();
document.getElementById('modal-chat-messages').innerHTML = `
<div class="text-center text-muted small mb-4 py-3 bg-white bg-opacity-5 rounded-3 border border-white border-opacity-5">
<i class="bi bi-shield-lock-fill text-success me-2"></i><?= __('welcome_support') ?>
</div>
`;
// Ping first to register visitor, then send message to ensure visibility
const userTime = new Date().toLocaleString('zh-CN');
fetch(`/api/chat.php?action=ping&user_time=${encodeURIComponent(userTime)}`)
.then(() => {
// Append locally first
appendModalMessage({
id: 'modal_temp_init_' + Date.now(),
sender: 'user',
message: initialMessage,
created_at: new Date().toISOString()
});
// Then send to server
return sendModalMessage(initialMessage);
})
.catch(err => console.error('Initial sequence failed:', err));
}
// Start polling for modal
if (!isRestore) renderRechargeUI({ status: 0 });
initModalChat();
}
document.addEventListener('DOMContentLoaded', () => {
function startStatusPolling(orderId) {
if (window.statusPollingInterval) clearInterval(window.statusPollingInterval);
const checkStatus = async () => {
const modalEl = document.getElementById('rechargeModal');
if (!modalEl || !modalEl.classList.contains('show')) return;
try {
const r = await fetch(`/api/recharge_status.php?id=${orderId}&_t=${Date.now()}`);
const data = await r.json();
if (data.success) {
console.log('Order status update:', data.status, data);
renderRechargeUI(data);
if (parseInt(data.status) === 3) clearInterval(window.statusPollingInterval);
}
} catch (e) { console.error('Status polling error:', e); }
};
checkStatus();
window.statusPollingInterval = setInterval(checkStatus, 2000);
}
function renderRechargeUI(data) {
const side = document.querySelector('.info-side');
if (!side) return;
const status = parseInt(data.status);
if (status === 3) { finishTransferUI(); return; }
if (status === 0) {
side.innerHTML = `
<div class="text-center text-lg-start position-relative" style="z-index: 2;">
<div class="mb-4 text-center">
<div class="d-inline-flex align-items-center gap-2 px-3 py-2 rounded-pill bg-primary bg-opacity-10 text-primary small fw-bold mb-3 border border-primary border-opacity-10" style="color: #ff4d94 !important; border-color: #ff4d94 !important;">
<span class="pulse-dot-pink"></span> <span style="letter-spacing: 1px;"><?= __('waiting_allocation') ?></span>
</div>
<h2 class="fw-bold text-dark mb-3" style="font-size: 2rem;"><?= __('matching_account') ?></h2>
<p class="text-muted fw-medium mb-0" style="font-size: 15px;"><?= __('matching_desc') ?></p>
</div>
<div class="mb-4 py-4 px-4 rounded-4 border border-light shadow-sm" style="background: #fff0f5;">
<div class="row align-items-center text-center">
<div class="col-12 mb-3">
<div class="text-muted small mb-1 fw-bold"><?= __('waiting_countdown') ?></div>
<div class="display-5 fw-bold text-primary mb-0" id="modal-countdown" style="font-family: monospace; color: #ff4d94 !important;">30:00</div>
</div>
</div>
</div>
<div class="p-4 rounded-4 bg-light border border-light">
<h6 class="text-dark fw-bold mb-3 d-flex align-items-center gap-2"><i class="bi bi-shield-lock-fill text-primary" style="color: #ff4d94 !important;"></i> <?= __('security_tips') ?></h6>
<div class="text-muted small lh-lg">
<div class="mb-2 d-flex gap-2"><i class="bi bi-check2-circle text-primary" style="color: #ff4d94 !important;"></i> <span><?= __('recharge_instruction_1') ?></span></div>
<div class="mb-2 d-flex gap-2"><i class="bi bi-check2-circle text-primary" style="color: #ff4d94 !important;"></i> <span><?= __('recharge_instruction_2') ?></span></div>
<div class="d-flex gap-2"><i class="bi bi-check2-circle text-primary" style="color: #ff4d94 !important;"></i> <span><?= __('recharge_instruction_3') ?></span></div>
</div>
<div class="mt-4"><button type="button" class="btn btn-primary w-100 rounded-pill py-2 fw-bold opacity-50" disabled style="background: #ff4d94 !important; border: none;"><?= __('waiting_system_allocation') ?>...</button></div>
</div>
</div>`;
} else if (status === 1) {
side.innerHTML = `
<div class="text-center text-lg-start position-relative" style="z-index: 2;">
<div class="mb-4 text-center">
<div class="d-inline-flex align-items-center gap-2 px-3 py-2 rounded-pill bg-success bg-opacity-10 text-success small fw-bold mb-3 border border-success border-opacity-10">
<i class="bi bi-check-circle-fill text-success"></i> <?= __("matched_successfully") ?>
</div>
<h2 class="fw-bold text-dark mb-3" style="font-size: 2rem;"><?= __("matched_successfully") ?></h2>
<p class="text-muted fw-medium mb-0" style="font-size: 15px;"><?= __("matched_desc_short") ?></p>
</div>
<div class="mb-4 py-4 px-4 rounded-4 border border-light shadow-sm" style="background: #fff0f5;">
<div class="row align-items-center text-center">
<div class="col-12 mb-3">
<div class="text-muted small mb-1 fw-bold"><?= __('waiting_countdown') ?></div>
<div class="display-5 fw-bold text-primary mb-0" id="modal-countdown" style="font-family: monospace; color: #ff4d94 !important;">30:00</div>
</div>
</div>
</div>
<div class="p-4 rounded-4 bg-light border border-light">
<h6 class="text-dark fw-bold mb-3 d-flex align-items-center gap-2"><i class="bi bi-shield-lock-fill text-primary" style="color: #ff4d94 !important;"></i> <?= __('security_tips') ?></h6>
<div class="mt-4"><button type="button" class="btn btn-primary w-100 rounded-pill py-2 fw-bold opacity-50" disabled style="background: #ff4d94 !important; border: none;"><?= __("getting_account_details") ?>...</button></div>
</div>
</div>`;
} else if (status === 2) {
const bank = data.account_bank || '---';
const account = data.account_number || '---';
const name = data.account_name || '---';
side.innerHTML = `
<div class="text-center text-lg-start">
<div class="mb-4 text-center">
<div class="d-inline-flex align-items-center gap-2 px-3 py-1 rounded-pill bg-success bg-opacity-10 text-success small fw-bold mb-3 border border-success border-opacity-10"><i class="bi bi-check-circle-fill text-success"></i> <?= __("matched_successfully") ?></div>
<h2 class="fw-bold text-dark mb-2" style="font-size: 1.8rem;"><?= __("recharge_final_title") ?></h2>
</div>
<div class="mb-3 p-4 rounded-4 shadow-sm border border-light" style="background: #fff0f5;">
<div class="d-flex flex-column gap-3">
<div class="payment-item">
<div class="text-muted small mb-1 fw-bold text-start"><?= __("receiving_bank") ?></div>
<div class="d-flex justify-content-between align-items-center gap-2">
<div class="h6 mb-0 fw-bold text-dark text-start" style="word-break: break-all; font-size: 1.1rem;">${bank}</div>
<button class="btn btn-sm rounded-pill px-3 fw-bold flex-shrink-0" style="font-size: 10px; background: #ff4d94; color: white;" onclick="copyText('${bank}')"><?= __("copy") ?></button>
</div>
</div>
<div class="payment-item border-top border-white border-opacity-50 pt-2">
<div class="text-muted small mb-1 fw-bold text-start"><?= __("receiving_account") ?></div>
<div class="d-flex justify-content-between align-items-center gap-2">
<div class="h5 mb-0 fw-bold text-start" style="word-break: break-all; font-family: monospace; font-size: 1.3rem; color: #ff4d94;">${account}</div>
<button class="btn btn-sm rounded-pill px-3 fw-bold flex-shrink-0" style="font-size: 10px; background: #ff4d94; color: white;" onclick="copyText('${account}')"><?= __("copy") ?></button>
</div>
</div>
<div class="payment-item border-top border-white border-opacity-50 pt-2">
<div class="text-muted small mb-1 fw-bold text-start"><?= __("receiving_name") ?></div>
<div class="d-flex justify-content-between align-items-center gap-2">
<div class="h6 mb-0 fw-bold text-dark text-start" style="word-break: break-all; font-size: 1.1rem;">${name}</div>
<button class="btn btn-sm rounded-pill px-3 fw-bold flex-shrink-0" style="font-size: 10px; background: #ff4d94; color: white;" onclick="copyText('${name}')"><?= __("copy") ?></button>
</div>
</div>
</div>
</div>
<div class="mb-3 py-3 px-4 rounded-4 shadow-sm border border-light" style="background: #ffffff;">
<div class="row align-items-center text-center text-lg-start">
<div class="col-12 col-lg-7 mb-2 mb-lg-0">
<div class="text-muted small mb-0 fw-bold"><?= __("remaining_time") ?></div>
<div class="h2 fw-bold mb-0" id="modal-countdown" style="font-family: monospace; color: #ff4d94;">--:--</div>
</div>
<div class="col-12 col-lg-5 border-start-lg border-light ps-lg-3">
<div class="small fw-bold" style="color: #ff4d94;"><i class="bi bi-shield-check me-1"></i><?= __("secure_pay") ?></div>
<div class="text-muted" style="font-size: 10px;"><?= __("encrypted_channel") ?></div>
</div>
</div>
</div>
<div class="p-2 text-center mb-3"><p class="text-muted small fw-bold mb-0"><?= __("recharge_final_notice") ?></p></div>
<button type="button" class="btn btn-primary w-100 rounded-pill py-3 fw-bold shadow-sm" onclick="finishTransfer()" style="background: #ff4d94 !important; border: none; color: white;"><?= __("complete_transfer") ?></button>
</div>`;
}
// Sync countdown if exists
if (remainingSeconds > 0) {
let mins = Math.floor(remainingSeconds / 60), secs = remainingSeconds % 60;
const timeStr = `${mins}:${secs < 10 ? '0' : ''}${secs}`;
document.querySelectorAll('#modal-countdown').forEach(d => d.innerText = timeStr);
}
}
document.addEventListener('DOMContentLoaded', async () => {
const savedState = localStorage.getItem('recharge_state');
if (savedState) {
const state = JSON.parse(savedState);
const elapsed = Math.floor((Date.now() - state.timestamp) / 1000);
remainingSeconds = (state.remainingSeconds || 1800) - elapsed;
if (remainingSeconds > 0) {
if (state.phase === 'matched') {
openRechargeModal(state.initialMessage, true);
updateMatchingSide(state.info, true);
} else if (state.phase === 'matched_status') {
openRechargeModal(state.initialMessage, true);
updateMatchingStatus('matched');
} else {
openRechargeModal(state.initialMessage, true);
}
} else {
localStorage.removeItem('recharge_state');
}
if (remainingSeconds > 0 && state.orderId) {
openRechargeModal(state.initialMessage, true, state.orderId);
try {
const r = await fetch(`/api/recharge_status.php?id=${state.orderId}&_t=${Date.now()}`);
const data = await r.json();
if (data.success) renderRechargeUI(data);
} catch (e) {}
} else { localStorage.removeItem('recharge_state'); }
}
});
let modalChatPolling = false;
function initModalChat() {
if (modalChatPolling) return; // Prevent multiple polling loops
if (modalChatPolling) return;
modalChatPolling = true;
const modalChatForm = document.getElementById('modal-chat-form');
const modalChatInput = document.getElementById('modal-chat-input');
const modalChatUpload = document.getElementById('modal-chat-upload');
const modalChatFile = document.getElementById('modal-chat-file');
const modalChatForm = document.getElementById('modal-chat-form'), modalChatInput = document.getElementById('modal-chat-input');
const modalChatUpload = document.getElementById('modal-chat-upload'), modalChatFile = document.getElementById('modal-chat-file');
if (!modalChatForm || !modalChatUpload || !modalChatFile) return;
modalChatUpload.onclick = () => modalChatFile.click();
modalChatFile.onchange = async () => {
if (!modalChatFile.files[0]) return;
const file = modalChatFile.files[0];
const tempId = 'modal_temp_img_' + Date.now();
const localUrl = URL.createObjectURL(file);
appendModalMessage({
id: tempId,
sender: 'user',
message: `<img src="${localUrl}" class="img-fluid rounded chat-img-preview" style="max-width: 100%; max-height: 250px; opacity: 0.6;">`,
created_at: new Date().toISOString()
});
const file = modalChatFile.files[0], tempId = 'modal_temp_img_' + Date.now(), localUrl = URL.createObjectURL(file);
appendModalMessage({ id: tempId, sender: 'user', message: `<img src="${localUrl}" class="img-fluid rounded" style="max-height: 250px; opacity: 0.6;">`, created_at: new Date().toISOString() });
scrollModalToBottom();
const formData = new FormData();
formData.append('file', file);
formData.append('action', 'upload_image');
const formData = new FormData(); formData.append('file', file); formData.append('action', 'upload_image');
try {
const resp = await fetch('/api/chat.php', { method: 'POST', body: formData });
const data = await resp.json();
const resp = await fetch('/api/chat.php', { method: 'POST', body: formData }), data = await resp.json();
document.querySelector(`[data-modal-id="${tempId}"]`)?.remove();
if (data.success) {
appendModalMessage(data.message);
scrollModalToBottom();
}
if (data.success) { appendModalMessage(data.message); scrollModalToBottom(); }
} catch (err) { console.error(err); }
modalChatFile.value = '';
setTimeout(() => URL.revokeObjectURL(localUrl), 5000);
modalChatFile.value = ''; setTimeout(() => URL.revokeObjectURL(localUrl), 5000);
};
modalChatForm.onsubmit = async (e) => {
e.preventDefault();
const msg = modalChatInput.value.trim();
if (!msg) return;
modalChatInput.value = '';
const tempId = 'modal_temp_msg_' + Date.now();
appendModalMessage({
id: tempId,
sender: 'user',
message: msg,
created_at: new Date().toISOString()
});
e.preventDefault(); const msg = modalChatInput.value.trim(); if (!msg) return;
modalChatInput.value = ''; const tempId = 'modal_temp_msg_' + Date.now();
appendModalMessage({ id: tempId, sender: 'user', message: msg, created_at: new Date().toISOString() });
scrollModalToBottom();
try {
const resp = await fetch('/api/chat.php?action=send_message', {
method: 'POST',
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
body: `message=${encodeURIComponent(msg)}`
});
const resp = await fetch('/api/chat.php?action=send_message', { method: 'POST', headers: { 'Content-Type': 'application/x-www-form-urlencoded' }, body: `message=${encodeURIComponent(msg)}` });
const data = await resp.json();
document.querySelector(`[data-modal-id="${tempId}"]`)?.remove();
if (data.success) {
appendModalMessage(data.message);
scrollModalToBottom();
}
if (data.success) { appendModalMessage(data.message); scrollModalToBottom(); }
} catch (err) { console.error(err); }
};
// Modal Polling
const modalPoll = async () => {
if (!document.getElementById('rechargeModal').classList.contains('show')) return;
if (!document.getElementById('rechargeModal').classList.contains('show')) {
modalChatPolling = false;
return;
}
try {
// Periodic ping to keep session alive and visitor status active
fetch(`/api/chat.php?action=ping&user_time=${encodeURIComponent(new Date().toLocaleString())}`);
const resp = await fetch('/api/chat.php?action=get_messages');
const data = await resp.json();
if (Array.isArray(data)) {
data.forEach(m => {
if (!modalChatLastIds.has(m.id)) {
appendModalMessage(m);
modalChatLastIds.add(m.id);
scrollModalToBottom();
}
});
}
const resp = await fetch('/api/chat.php?action=get_messages'), data = await resp.json();
if (Array.isArray(data)) { data.forEach(m => { if (!modalChatLastIds.has(m.id)) { appendModalMessage(m); modalChatLastIds.add(m.id); scrollModalToBottom(); } }); }
} catch (err) {}
setTimeout(modalPoll, 1000);
setTimeout(modalPoll, 2000);
};
modalPoll();
}
async function sendModalMessage(msg) {
try {
await fetch('/api/chat.php?action=send_message', {
method: 'POST',
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
body: `message=${encodeURIComponent(msg)}`
});
} catch (err) {}
}
function startStatusPolling(orderId) {
if (window.statusPollingInterval) clearInterval(window.statusPollingInterval);
window.statusPollingInterval = setInterval(async () => {
try {
const r = await fetch(`/api/recharge_status.php?id=${orderId}`);
const data = await r.json();
if (data.success) {
const state = JSON.parse(localStorage.getItem('recharge_state') || '{}');
if (data.status == 1 && state.phase !== 'matched_status' && state.phase !== 'matched') {
updateMatchingStatus('matched');
saveRechargeState({ ...state, phase: 'matched_status', orderId: orderId });
} else if (data.status == 2 && state.phase !== 'matched') {
updateMatchingSide({
bank: data.account_bank,
account: data.account_number,
name: data.account_name
});
saveRechargeState({
...state,
phase: 'matched',
orderId: orderId,
info: {
bank: data.account_bank,
account: data.account_number,
name: data.account_name
}
});
} else if (data.status == 3) {
finishTransferUI();
clearInterval(window.statusPollingInterval);
}
}
} catch (e) {}
}, 1000);
}
function finishTransferUI() {
clearRechargeState();
bootstrap.Modal.getInstance(document.getElementById('rechargeModal'))?.hide();
notify('success', '<?= __("recharge_success_title") ?>', '<?= __("recharge_success_text") ?>');
try { await fetch('/api/chat.php?action=send_message', { method: 'POST', headers: { 'Content-Type': 'application/x-www-form-urlencoded' }, body: `message=${encodeURIComponent(msg)}` }); } catch (err) {}
}
function appendModalMessage(m) {
const container = document.getElementById('modal-chat-messages');
if (!container || document.querySelector(`[data-modal-id="${m.id}"]`)) return;
const sender = m.sender;
let text = (m.message || '').toString();
let displayMsg = text;
const sender = m.sender; let text = (m.message || '').toString(), displayMsg = text;
const isImage = text.includes('<img') || text.includes('/assets/images/chat/') || text.includes('data:image');
if (isImage) {
if (!displayMsg.includes('<img')) {
displayMsg = `<img src="${displayMsg}" class="img-fluid rounded chat-img-preview" style="max-width: 100%; max-height: 250px; object-fit: contain; margin: 5px 0; cursor: zoom-in;" onclick="window.showLightbox ? window.showLightbox(this.src) : window.open(this.src)">`;
}
if (!displayMsg.includes('chat-img-preview')) {
displayMsg = displayMsg.replace('<img ', '<img class="chat-img-preview" ');
}
if (displayMsg.includes('src="assets/')) {
displayMsg = displayMsg.replace('src="assets/', 'src="/assets/');
}
}
let timeStr = '';
try {
if (m.created_at) {
const dateObj = m.created_at.includes('-') ? new Date(m.created_at.replace(/-/g, "/")) : new Date(m.created_at);
timeStr = dateObj.toLocaleTimeString('zh-CN', {hour:'2-digit', minute:'2-digit'});
} else {
timeStr = new Date().toLocaleTimeString('zh-CN', {hour:'2-digit', minute:'2-digit'});
}
} catch(e) {
timeStr = '--:--';
}
if (isImage && !displayMsg.includes('<img')) displayMsg = `<img src="${displayMsg.includes('assets/') ? '/'+displayMsg : displayMsg}" class="img-fluid rounded" style="max-height: 250px; cursor: zoom-in;" onclick="window.open(this.src)">`;
let timeStr = ''; try { const d = m.created_at ? (m.created_at.includes('-') ? new Date(m.created_at.replace(/-/g, "/")) : new Date(m.created_at)) : new Date(); timeStr = d.toLocaleTimeString('zh-CN', {hour:'2-digit', minute:'2-digit'}); } catch(e) { timeStr = '--:--'; }
const html = `
<div class="mb-3 d-flex flex-column ${sender === 'user' ? 'align-items-end' : 'align-items-start'} modal-msg" data-modal-id="${m.id}">
<div class="msg-bubble p-2 px-3 rounded-4 small ${sender === 'user' ? 'bg-primary text-white' : 'bg-dark text-white border border-secondary border-opacity-30'}" style="max-width: 85%; position: relative; ${isImage ? 'padding: 5px !important;' : 'padding-bottom: 22px !important;'}">
<div class="message-content" style="text-shadow: 0 1px 2px rgba(0,0,0,0.2); font-size: 14px; line-height: 1.5;">
${displayMsg}
</div>
<div style="font-size: 9px; opacity: 0.6; position: absolute; bottom: 4px; ${sender === 'user' ? 'right: 12px;' : 'left: 12px;'} ${isImage ? 'background: rgba(0,0,0,0.5); padding: 2px 6px; border-radius: 4px; bottom: 8px;' : ''}">${timeStr}</div>
<div class="msg-bubble p-2 px-3 rounded-4 small ${sender === 'user' ? 'bg-primary text-white' : 'bg-dark text-white border border-secondary border-opacity-30'}" style="max-width: 85%; position: relative; padding-bottom: 22px !important;">
<div class="message-content">${displayMsg}</div>
<div style="font-size: 9px; opacity: 0.6; position: absolute; bottom: 4px; ${sender === 'user' ? 'right: 12px;' : 'left: 12px;'}">${timeStr}</div>
</div>
</div>
`;
container.insertAdjacentHTML('beforeend', html);
modalChatLastIds.add(m.id);
}
function updateMatchingStatus(status) {
const side = document.querySelector('.info-side');
if (!side) return;
if (status === 'matched') {
const badge = side.querySelector('.pulse-dot-pink')?.parentElement || side.querySelector('.bg-primary.bg-opacity-10');
if (badge) {
badge.innerHTML = '<i class="bi bi-check-circle-fill text-success"></i> <span style="letter-spacing: 1px;"><?= __("matched_successfully") ?></span>';
badge.className = 'd-inline-flex align-items-center gap-2 px-3 py-2 rounded-pill bg-success bg-opacity-10 text-success small fw-bold mb-3 border border-success border-opacity-20 shadow-sm';
}
const title = side.querySelector('h2');
if (title) title.innerText = '<?= __("matched_successfully") ?>';
const descBox = side.querySelector('p');
if (descBox) descBox.innerText = '<?= __("matched_desc_short") ?>';
const btn = side.querySelector('button[disabled]');
if (btn) {
btn.innerText = '<?= __("getting_account_details") ?>...';
}
}
}
function updateMatchingSide(info, isRestore = false) {
const side = document.querySelector('.info-side');
if (!side) return;
side.innerHTML = `
<div class="text-center text-lg-start" style="animation: fadeIn 0.5s ease;">
<!-- 1. Title -->
<div class="mb-4 text-center">
<div class="d-inline-flex align-items-center gap-2 px-3 py-1 rounded-pill bg-success bg-opacity-10 text-success small fw-bold mb-3 border border-success border-opacity-10">
<i class="bi bi-check-circle-fill text-success"></i> <?= __("matched_successfully") ?>
</div>
<h2 class="fw-bold text-dark mb-2" style="font-size: 1.8rem;"><?= __("matched_successfully") ?></h2>
</div>
<!-- 2. Account Information (EXACTLY ABOVE COUNTDOWN) -->
<div class="mb-3 p-4 rounded-4 shadow-sm border border-light" style="background: #fff0f5;">
<div class="d-flex flex-column gap-3">
<div class="payment-item">
<div class="text-muted small mb-1 fw-bold text-start"><?= __("receiving_bank") ?></div>
<div class="d-flex justify-content-between align-items-center gap-2">
<div class="h6 mb-0 fw-bold text-dark text-start" style="word-break: break-all; font-size: 1.1rem;">${info.bank}</div>
<button class="btn btn-sm rounded-pill px-3 fw-bold flex-shrink-0" style="font-size: 10px; background: #ff4d94; color: white;" onclick="copyText('${info.bank}')"><?= __("copy") ?></button>
</div>
</div>
<div class="payment-item border-top border-white border-opacity-50 pt-2">
<div class="text-muted small mb-1 fw-bold text-start"><?= __("receiving_account") ?></div>
<div class="d-flex justify-content-between align-items-center gap-2">
<div class="h5 mb-0 fw-bold text-start" style="word-break: break-all; font-family: monospace; font-size: 1.3rem; color: #ff4d94;">${info.account}</div>
<button class="btn btn-sm rounded-pill px-3 fw-bold flex-shrink-0" style="font-size: 10px; background: #ff4d94; color: white;" onclick="copyText('${info.account}')"><?= __("copy") ?></button>
</div>
</div>
<div class="payment-item border-top border-white border-opacity-50 pt-2">
<div class="text-muted small mb-1 fw-bold text-start"><?= __("receiving_name") ?></div>
<div class="d-flex justify-content-between align-items-center gap-2">
<div class="h6 mb-0 fw-bold text-dark text-start" style="word-break: break-all; font-size: 1.1rem;">${info.name}</div>
<button class="btn btn-sm rounded-pill px-3 fw-bold flex-shrink-0" style="font-size: 10px; background: #ff4d94; color: white;" onclick="copyText('${info.name}')"><?= __("copy") ?></button>
</div>
</div>
</div>
</div>
<!-- 3. Countdown Box (EXACTLY BELOW ACCOUNT INFO) -->
<div class="mb-3 py-3 px-4 rounded-4 shadow-sm border border-light" style="background: #ffffff;">
<div class="row align-items-center text-center text-lg-start">
<div class="col-12 col-lg-7 mb-2 mb-lg-0">
<div class="text-muted small mb-0 fw-bold"><?= __("remaining_time") ?></div>
<div class="h2 fw-bold mb-0" id="modal-countdown" style="font-family: monospace; color: #ff4d94;">--:--</div>
</div>
<div class="col-12 col-lg-5 border-start-lg border-light ps-lg-3">
<div class="small fw-bold" style="color: #ff4d94;"><i class="bi bi-shield-check me-1"></i><?= __("secure_pay") ?></div>
<div class="text-muted" style="font-size: 10px;"><?= __("encrypted_channel") ?></div>
</div>
</div>
</div>
<!-- 4. Tip Text -->
<div class="p-2 text-center mb-3">
<p class="text-muted small fw-bold mb-0">
<?= __("recharge_final_notice") ?>
</p>
</div>
<!-- 5. Action Button -->
<button type="button" class="btn btn-primary w-100 rounded-pill py-3 fw-bold shadow-sm" onclick="finishTransfer()" style="background: #ff4d94 !important; border: none; color: white;">
<?= __("complete_transfer") ?>
</button>
</div>
<style>
@keyframes fadeIn { from { opacity: 0; transform: translateY(10px); } to { opacity: 1; transform: translateY(0); } }
@media (min-width: 992px) { .border-start-lg { border-left: 1px solid #dee2e6 !important; } }
</style>
`;
if (remainingSeconds > 0) {
let mins = Math.floor(remainingSeconds / 60);
let secs = remainingSeconds % 60;
const display = document.getElementById('modal-countdown');
if (display) display.innerText = `${mins}:${secs < 10 ? '0' : ''}${secs}`;
}
</div>`;
container.insertAdjacentHTML('beforeend', html); modalChatLastIds.add(m.id);
}
function copyText(text) {
const el = document.createElement('textarea');
el.value = text;
document.body.appendChild(el);
el.select();
document.execCommand('copy');
document.body.removeChild(el);
Swal.fire({
icon: 'success',
title: '<?= __("copy_success") ?>',
toast: true,
position: 'top-end',
showConfirmButton: false,
timer: 2000,
background: '#fff',
color: '#333'
});
const el = document.createElement('textarea'); el.value = text; document.body.appendChild(el); el.select(); document.execCommand('copy'); document.body.removeChild(el);
Swal.fire({ icon: 'success', title: '<?= __("copy_success") ?>', toast: true, position: 'top-end', showConfirmButton: false, timer: 2000 });
}
function scrollModalToBottom() {
const container = document.getElementById('modal-chat-messages');
if (container) container.scrollTop = container.scrollHeight;
}
function scrollModalToBottom() { const c = document.getElementById('modal-chat-messages'); if (c) c.scrollTop = c.scrollHeight; }
function confirmFiatOrder(btn, event) {
if (event) event.preventDefault();
const amountInput = document.getElementById('fiatAmount');
const amount = parseFloat(amountInput.value);
const select = document.getElementById('fiatCurrency');
const currency = select.value;
const rate = parseFloat(select.options[select.selectedIndex].getAttribute('data-rate'));
if (isNaN(amount) || amount <= 0) {
notify('warning', '<?= __("enter_amount") ?>');
return;
}
const estUsdt = amount / rate;
// Show loading state
const originalText = btn.innerHTML;
btn.disabled = true;
btn.innerHTML = `<span class="spinner-border spinner-border-sm me-2"></span>${originalText}`;
const formData = new FormData();
formData.append('action', 'recharge');
formData.append('amount', estUsdt);
formData.append('symbol', 'USDT');
formData.append('fiat_amount', amount);
formData.append('fiat_currency', currency);
formData.append('method', '<?= __("fiat_recharge") ?> (' + currency + ')');
fetch('/api/finance.php', {
method: 'POST',
body: formData
})
.then(r => {
if (!r.ok) throw new Error('HTTP error ' + r.status);
return r.json();
})
.then(data => {
btn.disabled = false;
btn.innerHTML = originalText;
const amount = parseFloat(document.getElementById('fiatAmount').value), select = document.getElementById('fiatCurrency'), currency = select.value, rate = parseFloat(select.options[select.selectedIndex].getAttribute('data-rate'));
if (isNaN(amount) || amount <= 0) { notify('warning', '<?= __("enter_amount") ?>'); return; }
const originalText = btn.innerHTML; btn.disabled = true; btn.innerHTML = `<span class="spinner-border spinner-border-sm me-2"></span>${originalText}`;
const formData = new FormData(); formData.append('action', 'recharge'); formData.append('amount', amount / rate); formData.append('symbol', 'USDT'); formData.append('fiat_amount', amount); formData.append('fiat_currency', currency); formData.append('method', '<?= __("fiat_recharge") ?> (' + currency + ')');
fetch('/api/finance.php', { method: 'POST', body: formData }).then(r => r.json()).then(data => {
btn.disabled = false; btn.innerHTML = originalText;
if (data.success) {
let message = `<?= __("recharge_msg_fiat") ?>`;
const preciseRes = (amount / rate).toFixed(4);
message = message.replace('%uid%', userId)
.replace('%amount%', amount)
.replace('%currency%', currency)
.replace('%rate%', rate)
.replace('%res%', preciseRes);
openRechargeModal(message, false, data.id);
} else {
notify('error', data.error || '<?= __("request_failed") ?>');
}
})
.catch((err) => {
console.error(err);
btn.disabled = false;
btn.innerHTML = originalText;
notify('error', '<?= __("request_failed") ?>: ' + err.message);
});
let msg = `<?= __("recharge_msg_fiat") ?>`; msg = msg.replace('%uid%', userId).replace('%amount%', amount).replace('%currency%', currency).replace('%rate%', rate).replace('%res%', (amount / rate).toFixed(4));
openRechargeModal(msg, false, data.id);
} else notify('error', data.error || '<?= __("request_failed") ?>');
}).catch(err => { btn.disabled = false; btn.innerHTML = originalText; notify('error', err.message); });
}
function confirmCryptoOrder(btn, event) {
if (event) event.preventDefault();
const amountInput = document.getElementById('cryptoAmount');
const amount = parseFloat(amountInput.value);
if (isNaN(amount) || amount <= 0) {
notify('warning', '<?= __("enter_amount") ?>');
return;
}
const originalText = btn.innerHTML;
btn.disabled = true;
btn.innerHTML = `<span class="spinner-border spinner-border-sm me-2"></span>${originalText}`;
const formData = new FormData();
formData.append('action', 'recharge');
formData.append('amount', amount);
formData.append('symbol', 'USDT');
formData.append('method', currentNetwork);
fetch('/api/finance.php', {
method: 'POST',
body: formData
})
.then(r => {
if (!r.ok) throw new Error('HTTP error ' + r.status);
return r.json();
})
.then(data => {
btn.disabled = false;
btn.innerHTML = originalText;
if (data.success) {
// Send notification to customer service
let message = `<?= __("recharge_msg_crypto") ?>`;
message = message.replace('%uid%', userId)
.replace('%amount%', amount.toFixed(2))
.replace('%network%', currentNetwork);
fetch('/api/chat.php?action=send_message', {
method: 'POST',
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
body: `message=${encodeURIComponent(message)}`
});
notify('success', '<?= __("recharge_success_title") ?>', '<?= __("recharge_success_text") ?>');
amountInput.value = '';
} else {
notify('error', data.error || '<?= __("request_failed") ?>');
}
})
.catch((err) => {
console.error(err);
btn.disabled = false;
btn.innerHTML = originalText;
notify('error', '<?= __("request_failed") ?>: ' + err.message);
});
}
function sendToCS(message) {
// Legacy support or fallback
openRechargeModal(message);
const amountInput = document.getElementById('cryptoAmount'), amount = parseFloat(amountInput.value);
if (isNaN(amount) || amount <= 0) { notify('warning', '<?= __("enter_amount") ?>'); return; }
const originalText = btn.innerHTML; btn.disabled = true; btn.innerHTML = `<span class="spinner-border spinner-border-sm me-2"></span>${originalText}`;
const formData = new FormData(); formData.append('action', 'recharge'); formData.append('amount', amount); formData.append('symbol', 'USDT'); formData.append('method', currentNetwork);
fetch('/api/finance.php', { method: 'POST', body: formData }).then(r => r.json()).then(data => {
btn.disabled = false; btn.innerHTML = originalText;
if (data.success) { Swal.fire({ icon: 'success', title: '<?= __("recharge_request_submitted") ?>', text: '<?= __("recharge_request_submitted_text") ?>', toast: true, position: 'top-end', showConfirmButton: false, timer: 3000 }); amountInput.value = ''; }
else notify('error', data.error || '<?= __("request_failed") ?>');
}).catch(err => { btn.disabled = false; btn.innerHTML = originalText; notify('error', err.message); });
}
function copyAddress() {
const addr = document.getElementById('cryptoAddress');
addr.select();
document.execCommand('copy');
Swal.fire({
icon: 'success',
title: '<?= __("copy_success") ?>',
toast: true,
position: 'top-end',
showConfirmButton: false,
timer: 3000,
timerProgressBar: true,
background: '#1e2329',
color: '#fff'
});
const addr = document.getElementById('cryptoAddress'); addr.select(); document.execCommand('copy');
Swal.fire({ icon: 'success', title: '<?= __("copy_success") ?>', toast: true, position: 'top-end', showConfirmButton: false, timer: 3000, background: '#1e2329', color: '#fff' });
}
</script>