87 lines
3.3 KiB
JavaScript
87 lines
3.3 KiB
JavaScript
'use client';
|
||
|
||
import { motion } from 'framer-motion';
|
||
import { CloseIcon } from './Icons';
|
||
|
||
export default function ScanImportConfirmModal({
|
||
scannedFunds,
|
||
selectedScannedCodes,
|
||
onClose,
|
||
onToggle,
|
||
onConfirm
|
||
}) {
|
||
return (
|
||
<motion.div
|
||
className="modal-overlay"
|
||
role="dialog"
|
||
aria-modal="true"
|
||
aria-label="确认导入基金"
|
||
onClick={onClose}
|
||
initial={{ opacity: 0 }}
|
||
animate={{ opacity: 1 }}
|
||
exit={{ opacity: 0 }}
|
||
>
|
||
<motion.div
|
||
initial={{ opacity: 0, scale: 0.95, y: 20 }}
|
||
animate={{ opacity: 1, scale: 1, y: 0 }}
|
||
exit={{ opacity: 0, scale: 0.95, y: 20 }}
|
||
className="glass card modal"
|
||
onClick={(e) => e.stopPropagation()}
|
||
style={{ width: 460, maxWidth: '90vw' }}
|
||
>
|
||
<div className="title" style={{ marginBottom: 12, justifyContent: 'space-between' }}>
|
||
<span>确认导入基金</span>
|
||
<button className="icon-button" onClick={onClose} style={{ border: 'none', background: 'transparent' }}>
|
||
<CloseIcon width="20" height="20" />
|
||
</button>
|
||
</div>
|
||
{scannedFunds.length === 0 ? (
|
||
<div className="muted" style={{ fontSize: 13, lineHeight: 1.6 }}>
|
||
未识别到有效的基金代码,请尝试更清晰的截图或手动搜索。
|
||
</div>
|
||
) : (
|
||
<div className="search-results pending-list" style={{ maxHeight: 320, overflowY: 'auto' }}>
|
||
{scannedFunds.map((item) => {
|
||
const isSelected = selectedScannedCodes.has(item.code);
|
||
const isAlreadyAdded = item.status === 'added';
|
||
const isInvalid = item.status === 'invalid';
|
||
const isDisabled = isAlreadyAdded || isInvalid;
|
||
const displayName = item.name || (isInvalid ? '未找到基金' : '未知基金');
|
||
return (
|
||
<div
|
||
key={item.code}
|
||
className={`search-item ${isSelected ? 'selected' : ''} ${isAlreadyAdded ? 'added' : ''}`}
|
||
onMouseDown={(e) => e.preventDefault()}
|
||
onClick={() => {
|
||
if (isDisabled) return;
|
||
onToggle(item.code);
|
||
}}
|
||
style={{ cursor: isDisabled ? 'not-allowed' : 'pointer' }}
|
||
>
|
||
<div className="fund-info">
|
||
<span className="fund-name">{displayName}</span>
|
||
<span className="fund-code muted">#{item.code}</span>
|
||
</div>
|
||
{isAlreadyAdded ? (
|
||
<span className="added-label">已添加</span>
|
||
) : isInvalid ? (
|
||
<span className="added-label">未找到</span>
|
||
) : (
|
||
<div className="checkbox">
|
||
{isSelected && <div className="checked-mark" />}
|
||
</div>
|
||
)}
|
||
</div>
|
||
);
|
||
})}
|
||
</div>
|
||
)}
|
||
<div style={{ display: 'flex', gap: 8, justifyContent: 'flex-end', marginTop: 12 }}>
|
||
<button className="button secondary" onClick={onClose}>取消</button>
|
||
<button className="button" onClick={onConfirm} disabled={selectedScannedCodes.size === 0}>确认导入</button>
|
||
</div>
|
||
</motion.div>
|
||
</motion.div>
|
||
);
|
||
}
|