群成员消息导出放在消息查看里面

This commit is contained in:
xuncha
2026-03-16 18:19:49 +08:00
parent 79e40f6a53
commit 7fad75fad0
3 changed files with 315 additions and 296 deletions

View File

@@ -4453,6 +4453,7 @@ class ExportService {
const cleanedMyWxid = conn.cleanedWxid const cleanedMyWxid = conn.cleanedWxid
const isGroup = sessionId.includes('@chatroom') const isGroup = sessionId.includes('@chatroom')
const rawMyWxid = String(this.configService.get('myWxid') || '').trim()
const sessionInfo = await this.getContactInfo(sessionId) const sessionInfo = await this.getContactInfo(sessionId)
const myInfo = await this.getContactInfo(cleanedMyWxid) const myInfo = await this.getContactInfo(cleanedMyWxid)
@@ -5650,6 +5651,7 @@ class ExportService {
const cleanedMyWxid = conn.cleanedWxid const cleanedMyWxid = conn.cleanedWxid
const isGroup = sessionId.includes('@chatroom') const isGroup = sessionId.includes('@chatroom')
const rawMyWxid = String(this.configService.get('myWxid') || '').trim()
const sessionInfo = await this.getContactInfo(sessionId) const sessionInfo = await this.getContactInfo(sessionId)
const myInfo = await this.getContactInfo(cleanedMyWxid) const myInfo = await this.getContactInfo(cleanedMyWxid)
const contactCache = new Map<string, { success: boolean; contact?: any; error?: string }>() const contactCache = new Map<string, { success: boolean; contact?: any; error?: string }>()

View File

@@ -480,6 +480,12 @@
overflow: hidden; overflow: hidden;
} }
.detail-drag-region {
height: 16px;
flex-shrink: 0;
-webkit-app-region: drag;
}
.resize-handle { .resize-handle {
width: 4px; width: 4px;
cursor: col-resize; cursor: col-resize;
@@ -1177,14 +1183,36 @@
.member-message-toolbar { .member-message-toolbar {
display: grid; display: grid;
gap: 12px; gap: 12px;
grid-template-columns: minmax(240px, 360px) minmax(0, 1fr); grid-template-columns: minmax(240px, 360px) minmax(160px, 1fr);
align-items: start; align-items: end;
@media (max-width: 900px) { @media (max-width: 900px) {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
} }
.member-message-toolbar-actions {
display: flex;
justify-content: flex-end;
align-items: center;
@media (max-width: 900px) {
justify-content: flex-start;
}
}
.member-message-select-trigger {
border-radius: 12px;
}
.member-message-summary-text {
align-self: flex-start;
font-size: 14px;
font-weight: 600;
color: var(--text-primary);
line-height: 1.2;
}
.member-message-summary-card { .member-message-summary-card {
min-height: 48px; min-height: 48px;
display: flex; display: flex;
@@ -1573,6 +1601,11 @@
background: rgba(30, 30, 30, 0.95); background: rgba(30, 30, 30, 0.95);
border: 1px solid rgba(255, 255, 255, 0.1); border: 1px solid rgba(255, 255, 255, 0.1);
} }
.member-export-modal {
background: rgba(30, 30, 30, 0.95);
border: 1px solid rgba(255, 255, 255, 0.1);
}
} }
// 成员详情弹框 // 成员详情弹框
@@ -1733,3 +1766,59 @@
} }
} }
} }
.member-export-modal {
background: rgba(255, 255, 255, 0.97);
border-radius: 20px;
padding: 28px;
width: min(720px, calc(100vw - 32px));
max-height: min(760px, calc(100vh - 32px));
overflow-y: auto;
position: relative;
backdrop-filter: blur(20px);
box-shadow: 0 20px 60px rgba(0, 0, 0, 0.3);
.modal-close {
position: absolute;
top: 16px;
right: 16px;
background: var(--bg-tertiary);
border: none;
width: 32px;
height: 32px;
border-radius: 50%;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
color: var(--text-secondary);
transition: all 0.15s;
&:hover {
background: var(--bg-hover);
color: var(--text-primary);
}
}
.member-export-modal-header {
margin-bottom: 18px;
padding-right: 40px;
h3 {
margin: 0;
font-size: 20px;
font-weight: 600;
color: var(--text-primary);
}
p {
margin: 6px 0 0;
font-size: 13px;
color: var(--text-secondary);
}
}
.member-export-panel {
gap: 18px;
}
}

View File

@@ -37,7 +37,7 @@ interface GroupMessageRank {
messageCount: number messageCount: number
} }
type AnalysisFunction = 'members' | 'memberMessages' | 'memberExport' | 'ranking' | 'activeHours' | 'mediaStats' type AnalysisFunction = 'members' | 'memberMessages' | 'ranking' | 'activeHours' | 'mediaStats'
type MemberExportFormat = 'chatlab' | 'chatlab-jsonl' | 'json' | 'arkme-json' | 'html' | 'txt' | 'excel' | 'weclone' type MemberExportFormat = 'chatlab' | 'chatlab-jsonl' | 'json' | 'arkme-json' | 'html' | 'txt' | 'excel' | 'weclone'
interface MemberMessageExportOptions { interface MemberMessageExportOptions {
@@ -170,7 +170,6 @@ function GroupAnalyticsPage() {
const [memberMessagesHasMore, setMemberMessagesHasMore] = useState(false) const [memberMessagesHasMore, setMemberMessagesHasMore] = useState(false)
const [memberMessagesCursor, setMemberMessagesCursor] = useState(0) const [memberMessagesCursor, setMemberMessagesCursor] = useState(0)
const [memberMessagesLoadingMore, setMemberMessagesLoadingMore] = useState(false) const [memberMessagesLoadingMore, setMemberMessagesLoadingMore] = useState(false)
const [selectedExportMemberUsername, setSelectedExportMemberUsername] = useState('')
const [selectedMessageMemberUsername, setSelectedMessageMemberUsername] = useState('') const [selectedMessageMemberUsername, setSelectedMessageMemberUsername] = useState('')
const [exportFolder, setExportFolder] = useState('') const [exportFolder, setExportFolder] = useState('')
const [memberExportOptions, setMemberExportOptions] = useState<MemberMessageExportOptions>({ const [memberExportOptions, setMemberExportOptions] = useState<MemberMessageExportOptions>({
@@ -188,14 +187,12 @@ function GroupAnalyticsPage() {
// 成员详情弹框 // 成员详情弹框
const [selectedMember, setSelectedMember] = useState<GroupMember | null>(null) const [selectedMember, setSelectedMember] = useState<GroupMember | null>(null)
const [copiedField, setCopiedField] = useState<string | null>(null) const [copiedField, setCopiedField] = useState<string | null>(null)
const [showMemberExportModal, setShowMemberExportModal] = useState(false)
const [showMessageMemberSelect, setShowMessageMemberSelect] = useState(false) const [showMessageMemberSelect, setShowMessageMemberSelect] = useState(false)
const [showMemberSelect, setShowMemberSelect] = useState(false)
const [showFormatSelect, setShowFormatSelect] = useState(false) const [showFormatSelect, setShowFormatSelect] = useState(false)
const [showDisplayNameSelect, setShowDisplayNameSelect] = useState(false) const [showDisplayNameSelect, setShowDisplayNameSelect] = useState(false)
const [messageMemberSearchKeyword, setMessageMemberSearchKeyword] = useState('') const [messageMemberSearchKeyword, setMessageMemberSearchKeyword] = useState('')
const [memberSearchKeyword, setMemberSearchKeyword] = useState('')
const messageMemberSelectDropdownRef = useRef<HTMLDivElement>(null) const messageMemberSelectDropdownRef = useRef<HTMLDivElement>(null)
const memberSelectDropdownRef = useRef<HTMLDivElement>(null)
const formatDropdownRef = useRef<HTMLDivElement>(null) const formatDropdownRef = useRef<HTMLDivElement>(null)
const displayNameDropdownRef = useRef<HTMLDivElement>(null) const displayNameDropdownRef = useRef<HTMLDivElement>(null)
@@ -241,10 +238,6 @@ function GroupAnalyticsPage() {
{ value: 'remark', label: '备注优先', desc: '有备注显示备注,否则显示昵称' }, { value: 'remark', label: '备注优先', desc: '有备注显示备注,否则显示昵称' },
{ value: 'nickname', label: '微信昵称', desc: '始终显示微信昵称' } { value: 'nickname', label: '微信昵称', desc: '始终显示微信昵称' }
]), []) ]), [])
const selectedExportMember = useMemo(
() => members.find(member => member.username === selectedExportMemberUsername) || null,
[members, selectedExportMemberUsername]
)
const selectedMessageMember = useMemo( const selectedMessageMember = useMemo(
() => members.find(member => member.username === selectedMessageMemberUsername) || null, () => members.find(member => member.username === selectedMessageMemberUsername) || null,
[members, selectedMessageMemberUsername] [members, selectedMessageMemberUsername]
@@ -257,9 +250,6 @@ function GroupAnalyticsPage() {
() => displayNameOptions.find(option => option.value === memberExportOptions.displayNamePreference) || displayNameOptions[0], () => displayNameOptions.find(option => option.value === memberExportOptions.displayNamePreference) || displayNameOptions[0],
[displayNameOptions, memberExportOptions.displayNamePreference] [displayNameOptions, memberExportOptions.displayNamePreference]
) )
const filteredMemberOptions = useMemo(() => {
return filterMembersByKeyword(members, memberSearchKeyword)
}, [memberSearchKeyword, members])
const filteredMessageMemberOptions = useMemo(() => { const filteredMessageMemberOptions = useMemo(() => {
return filterMembersByKeyword(members, messageMemberSearchKeyword) return filterMembersByKeyword(members, messageMemberSearchKeyword)
}, [members, messageMemberSearchKeyword]) }, [members, messageMemberSearchKeyword])
@@ -353,19 +343,14 @@ function GroupAnalyticsPage() {
useEffect(() => { useEffect(() => {
if (members.length === 0) { if (members.length === 0) {
setSelectedExportMemberUsername('')
setSelectedMessageMemberUsername('') setSelectedMessageMemberUsername('')
return return
} }
const exportExists = members.some(member => member.username === selectedExportMemberUsername)
if (!exportExists) {
setSelectedExportMemberUsername(members[0].username)
}
const messageExists = members.some(member => member.username === selectedMessageMemberUsername) const messageExists = members.some(member => member.username === selectedMessageMemberUsername)
if (!messageExists) { if (!messageExists) {
setSelectedMessageMemberUsername(members[0].username) setSelectedMessageMemberUsername(members[0].username)
} }
}, [members, selectedExportMemberUsername, selectedMessageMemberUsername]) }, [members, selectedMessageMemberUsername])
useEffect(() => { useEffect(() => {
const handleClickOutside = (event: MouseEvent) => { const handleClickOutside = (event: MouseEvent) => {
@@ -373,9 +358,6 @@ function GroupAnalyticsPage() {
if (showMessageMemberSelect && messageMemberSelectDropdownRef.current && !messageMemberSelectDropdownRef.current.contains(target)) { if (showMessageMemberSelect && messageMemberSelectDropdownRef.current && !messageMemberSelectDropdownRef.current.contains(target)) {
setShowMessageMemberSelect(false) setShowMessageMemberSelect(false)
} }
if (showMemberSelect && memberSelectDropdownRef.current && !memberSelectDropdownRef.current.contains(target)) {
setShowMemberSelect(false)
}
if (showFormatSelect && formatDropdownRef.current && !formatDropdownRef.current.contains(target)) { if (showFormatSelect && formatDropdownRef.current && !formatDropdownRef.current.contains(target)) {
setShowFormatSelect(false) setShowFormatSelect(false)
} }
@@ -385,7 +367,7 @@ function GroupAnalyticsPage() {
} }
document.addEventListener('mousedown', handleClickOutside) document.addEventListener('mousedown', handleClickOutside)
return () => document.removeEventListener('mousedown', handleClickOutside) return () => document.removeEventListener('mousedown', handleClickOutside)
}, [showDisplayNameSelect, showFormatSelect, showMemberSelect, showMessageMemberSelect]) }, [showDisplayNameSelect, showFormatSelect, showMessageMemberSelect])
useEffect(() => { useEffect(() => {
if (preselectAppliedRef.current) return if (preselectAppliedRef.current) return
@@ -422,7 +404,7 @@ function GroupAnalyticsPage() {
// 日期范围变化时自动刷新 // 日期范围变化时自动刷新
useEffect(() => { useEffect(() => {
if (dateRangeReady && selectedGroup && selectedFunction && selectedFunction !== 'members' && selectedFunction !== 'memberExport') { if (dateRangeReady && selectedGroup && selectedFunction && selectedFunction !== 'members') {
setDateRangeReady(false) setDateRangeReady(false)
loadFunctionData(selectedFunction) loadFunctionData(selectedFunction)
} }
@@ -436,6 +418,7 @@ function GroupAnalyticsPage() {
setSelectedFunction(null) setSelectedFunction(null)
setMembers([]) setMembers([])
resetMemberMessageState() resetMemberMessageState()
setShowMemberExportModal(false)
setRankings([]) setRankings([])
setActiveHours({}) setActiveHours({})
setMediaStats(null) setMediaStats(null)
@@ -450,10 +433,8 @@ function GroupAnalyticsPage() {
setSelectedGroupId(group.username) setSelectedGroupId(group.username)
setSelectedFunction(null) setSelectedFunction(null)
setSelectedMember(null) setSelectedMember(null)
setShowMemberExportModal(false)
resetMemberMessageState() resetMemberMessageState()
setSelectedExportMemberUsername('')
setMemberSearchKeyword('')
setShowMemberSelect(false)
setShowFormatSelect(false) setShowFormatSelect(false)
setShowDisplayNameSelect(false) setShowDisplayNameSelect(false)
} }
@@ -580,23 +561,6 @@ function GroupAnalyticsPage() {
}) })
break break
} }
case 'memberExport': {
updateBackgroundTask(taskId, {
detail: '正在读取导出成员列表',
progressText: '成员导出'
})
const result = await window.electronAPI.groupAnalytics.getGroupMembers(targetGroup.username)
if (isBackgroundTaskCancelRequested(taskId)) {
finishBackgroundTask(taskId, 'canceled', { detail: '已停止后续加载,成员导出列表未继续写入' })
return
}
if (result.success && result.data) setMembers(result.data)
finishBackgroundTask(taskId, result.success ? 'completed' : 'failed', {
detail: result.success ? `成员导出列表加载完成,共 ${result.data?.length || 0}` : (result.error || '读取成员导出列表失败'),
progressText: result.success ? `${result.data?.length || 0}` : '失败'
})
break
}
case 'ranking': { case 'ranking': {
updateBackgroundTask(taskId, { updateBackgroundTask(taskId, {
detail: '正在计算群消息排行', detail: '正在计算群消息排行',
@@ -731,7 +695,6 @@ function GroupAnalyticsPage() {
} }
const handleDateRangeComplete = () => { const handleDateRangeComplete = () => {
if (selectedFunction === 'memberExport') return
setDateRangeReady(true) setDateRangeReady(true)
} }
@@ -796,6 +759,13 @@ function GroupAnalyticsPage() {
await loadFunctionData('memberMessages', selectedGroup, member.username) await loadFunctionData('memberMessages', selectedGroup, member.username)
} }
const handleOpenMemberExportModal = () => {
setShowMessageMemberSelect(false)
setShowFormatSelect(false)
setShowDisplayNameSelect(false)
setShowMemberExportModal(true)
}
const handleExportMembers = async () => { const handleExportMembers = async () => {
if (!selectedGroup || isExportingMembers) return if (!selectedGroup || isExportingMembers) return
setIsExportingMembers(true) setIsExportingMembers(true)
@@ -858,8 +828,8 @@ function GroupAnalyticsPage() {
} }
const handleExportMemberMessages = async () => { const handleExportMemberMessages = async () => {
if (!selectedGroup || !selectedExportMemberUsername || !exportFolder || isExportingMemberMessages) return if (!selectedGroup || !selectedMessageMemberUsername || !exportFolder || isExportingMemberMessages) return
const member = members.find(item => item.username === selectedExportMemberUsername) const member = members.find(item => item.username === selectedMessageMemberUsername)
if (!member) { if (!member) {
alert('请先选择成员') alert('请先选择成员')
return return
@@ -893,6 +863,7 @@ function GroupAnalyticsPage() {
} }
) )
if (result.success && (result.successCount ?? 0) > 0) { if (result.success && (result.successCount ?? 0) > 0) {
setShowMemberExportModal(false)
alert(`导出成功:${member.displayName || member.username}`) alert(`导出成功:${member.displayName || member.username}`)
} else { } else {
alert(`导出失败:${result.error || '未知错误'}`) alert(`导出失败:${result.error || '未知错误'}`)
@@ -1080,11 +1051,6 @@ function GroupAnalyticsPage() {
<span></span> <span></span>
<small></small> <small></small>
</div> </div>
<div className="function-card" onClick={() => handleFunctionSelect('memberExport')}>
<Download size={32} />
<span></span>
<small></small>
</div>
<div className="function-card" onClick={() => handleFunctionSelect('ranking')}> <div className="function-card" onClick={() => handleFunctionSelect('ranking')}>
<BarChart3 size={32} /> <BarChart3 size={32} />
<span></span> <span></span>
@@ -1109,7 +1075,6 @@ function GroupAnalyticsPage() {
switch (selectedFunction) { switch (selectedFunction) {
case 'members': return '群成员查看' case 'members': return '群成员查看'
case 'memberMessages': return '成员消息查看' case 'memberMessages': return '成员消息查看'
case 'memberExport': return '成员消息导出'
case 'ranking': return '群聊发言排行' case 'ranking': return '群聊发言排行'
case 'activeHours': return '群聊活跃时段' case 'activeHours': return '群聊活跃时段'
case 'mediaStats': return '媒体内容统计' case 'mediaStats': return '媒体内容统计'
@@ -1177,15 +1142,16 @@ function GroupAnalyticsPage() {
<div className="member-message-empty"></div> <div className="member-message-empty"></div>
) : ( ) : (
<> <>
<div className="member-message-summary-text"> {memberMessages.length} </div>
<div className="member-message-toolbar"> <div className="member-message-toolbar">
<div className="member-export-field" ref={messageMemberSelectDropdownRef}> <div className="member-export-field" ref={messageMemberSelectDropdownRef}>
<span></span> <span></span>
<button <button
type="button" type="button"
className={`select-trigger ${showMessageMemberSelect ? 'open' : ''}`} className={`select-trigger member-message-select-trigger ${showMessageMemberSelect ? 'open' : ''}`}
onClick={() => { onClick={() => {
setShowMessageMemberSelect(prev => !prev) setShowMessageMemberSelect(prev => !prev)
setShowMemberSelect(false)
setShowFormatSelect(false) setShowFormatSelect(false)
setShowDisplayNameSelect(false) setShowDisplayNameSelect(false)
}} }}
@@ -1238,11 +1204,15 @@ function GroupAnalyticsPage() {
</div> </div>
)} )}
</div> </div>
<div className="member-message-summary-card"> <div className="member-message-toolbar-actions">
<span className="summary-title"> {memberMessages.length} </span> <button
<span className="summary-desc"> className="member-export-start-btn"
{selectedMessageMember?.displayName || selectedMessageMember?.username || '未选择成员'} onClick={handleOpenMemberExportModal}
</span> disabled={!selectedMessageMemberUsername}
>
<Download size={16} />
<span></span>
</button>
</div> </div>
</div> </div>
@@ -1283,234 +1253,6 @@ function GroupAnalyticsPage() {
)} )}
</div> </div>
)} )}
{selectedFunction === 'memberExport' && (
<div className="member-export-panel">
{members.length === 0 ? (
<div className="member-export-empty"></div>
) : (
<>
<div className="member-export-grid">
<div className="member-export-field" ref={memberSelectDropdownRef}>
<span></span>
<button
type="button"
className={`select-trigger ${showMemberSelect ? 'open' : ''}`}
onClick={() => {
setShowMemberSelect(prev => !prev)
setShowFormatSelect(false)
setShowDisplayNameSelect(false)
}}
>
<div className="member-select-trigger-value">
<Avatar
src={selectedExportMember?.avatarUrl}
name={selectedExportMember?.displayName || selectedExportMember?.username || '?'}
size={24}
/>
<span className="select-value">{selectedExportMember?.displayName || selectedExportMember?.username || '请选择成员'}</span>
</div>
<ChevronDown size={16} />
</button>
{showMemberSelect && (
<div className="select-dropdown member-select-dropdown">
<div className="member-select-search">
<Search size={14} />
<input
type="text"
value={memberSearchKeyword}
onChange={e => setMemberSearchKeyword(e.target.value)}
placeholder="搜索 wxid / 昵称 / 备注 / 微信号"
/>
</div>
<div className="member-select-options">
{filteredMemberOptions.length === 0 ? (
<div className="member-select-empty"></div>
) : (
filteredMemberOptions.map(member => (
<button
key={member.username}
type="button"
className={`select-option member-select-option ${selectedExportMemberUsername === member.username ? 'active' : ''}`}
onClick={() => {
setSelectedExportMemberUsername(member.username)
setShowMemberSelect(false)
}}
>
<Avatar src={member.avatarUrl} name={member.displayName} size={28} />
<span className="member-option-main">{member.displayName || member.username}</span>
<span className="member-option-meta">
wxid: {member.username}
{member.alias ? ` · 微信号: ${member.alias}` : ''}
{member.remark ? ` · 备注: ${member.remark}` : ''}
{member.nickname ? ` · 昵称: ${member.nickname}` : ''}
</span>
</button>
))
)}
</div>
</div>
)}
</div>
<div className="member-export-field" ref={formatDropdownRef}>
<span></span>
<button
type="button"
className={`select-trigger ${showFormatSelect ? 'open' : ''}`}
onClick={() => {
setShowFormatSelect(prev => !prev)
setShowMemberSelect(false)
setShowDisplayNameSelect(false)
}}
>
<span className="select-value">{selectedFormatOption.label}</span>
<ChevronDown size={16} />
</button>
{showFormatSelect && (
<div className="select-dropdown">
{memberExportFormatOptions.map(option => (
<button
key={option.value}
type="button"
className={`select-option ${memberExportOptions.format === option.value ? 'active' : ''}`}
onClick={() => {
handleMemberExportFormatChange(option.value)
setShowFormatSelect(false)
}}
>
<span className="option-label">{option.label}</span>
<span className="option-desc">{option.desc}</span>
</button>
))}
</div>
)}
</div>
<div className="member-export-field member-export-folder">
<span></span>
<div className="member-export-folder-row">
<input value={exportFolder} readOnly placeholder="请选择导出目录" />
<button type="button" onClick={handleChooseExportFolder}>
</button>
</div>
</div>
</div>
<div className="member-export-options">
<div className="member-export-chip-group">
<span className="chip-group-label"></span>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportMedia ? 'active' : ''}`}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportMedia: !prev.exportMedia }))}
>
</button>
</div>
<div className="member-export-chip-group">
<span className="chip-group-label"></span>
<div className="member-export-chip-list">
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportImages ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportImages: !prev.exportImages }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportVoices ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportVoices: !prev.exportVoices }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportVideos ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportVideos: !prev.exportVideos }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportEmojis ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportEmojis: !prev.exportEmojis }))}
>
</button>
</div>
</div>
<div className="member-export-chip-group">
<span className="chip-group-label"></span>
<div className="member-export-chip-list">
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportVoiceAsText ? 'active' : ''}`}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportVoiceAsText: !prev.exportVoiceAsText }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportAvatars ? 'active' : ''}`}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportAvatars: !prev.exportAvatars }))}
>
</button>
</div>
</div>
<div className="member-export-field" ref={displayNameDropdownRef}>
<span></span>
<button
type="button"
className={`select-trigger ${showDisplayNameSelect ? 'open' : ''}`}
onClick={() => {
setShowDisplayNameSelect(prev => !prev)
setShowMemberSelect(false)
setShowFormatSelect(false)
}}
>
<span className="select-value">{selectedDisplayNameOption.label}</span>
<ChevronDown size={16} />
</button>
{showDisplayNameSelect && (
<div className="select-dropdown">
{displayNameOptions.map(option => (
<button
key={option.value}
type="button"
className={`select-option ${memberExportOptions.displayNamePreference === option.value ? 'active' : ''}`}
onClick={() => {
setMemberExportOptions(prev => ({ ...prev, displayNamePreference: option.value }))
setShowDisplayNameSelect(false)
}}
>
<span className="option-label">{option.label}</span>
<span className="option-desc">{option.desc}</span>
</button>
))}
</div>
)}
</div>
</div>
<div className="member-export-actions">
<button
className="member-export-start-btn"
onClick={handleExportMemberMessages}
disabled={isExportingMemberMessages || !selectedExportMemberUsername || !exportFolder}
>
{isExportingMemberMessages ? <Loader2 size={16} className="spin" /> : <Download size={16} />}
<span>{isExportingMemberMessages ? '导出中...' : '开始导出'}</span>
</button>
</div>
</>
)}
</div>
)}
{selectedFunction === 'ranking' && ( {selectedFunction === 'ranking' && (
<div className="rankings-list"> <div className="rankings-list">
{rankings.map((item, index) => ( {rankings.map((item, index) => (
@@ -1572,18 +1314,203 @@ function GroupAnalyticsPage() {
const renderDetailPanel = () => { const renderDetailPanel = () => {
if (selectedFunction) {
return renderFunctionContent()
}
if (!selectedGroup) { if (!selectedGroup) {
return ( return (
<div className="placeholder"> <>
<Users size={64} /> <div className="detail-drag-region" aria-hidden="true" />
<div className="placeholder">
<Users size={64} />
<p></p> <p></p>
</div> </div>
</>
) )
} }
if (!selectedFunction) { return (
return renderFunctionMenu() <>
} <div className="detail-drag-region" aria-hidden="true" />
return renderFunctionContent() {renderFunctionMenu()}
</>
)
}
const renderMemberExportModal = () => {
if (!showMemberExportModal) return null
return (
<div className="member-modal-overlay" onClick={() => setShowMemberExportModal(false)}>
<div className="member-export-modal" onClick={e => e.stopPropagation()}>
<button className="modal-close" onClick={() => setShowMemberExportModal(false)}>
<X size={20} />
</button>
<div className="member-export-modal-header">
<h3></h3>
<p>{selectedMessageMember?.displayName || selectedMessageMember?.username || '未选择成员'}</p>
</div>
<div className="member-export-panel">
<div className="member-export-grid">
<div className="member-export-field" ref={formatDropdownRef}>
<span></span>
<button
type="button"
className={`select-trigger ${showFormatSelect ? 'open' : ''}`}
onClick={() => {
setShowFormatSelect(prev => !prev)
setShowDisplayNameSelect(false)
}}
>
<span className="select-value">{selectedFormatOption.label}</span>
<ChevronDown size={16} />
</button>
{showFormatSelect && (
<div className="select-dropdown">
{memberExportFormatOptions.map(option => (
<button
key={option.value}
type="button"
className={`select-option ${memberExportOptions.format === option.value ? 'active' : ''}`}
onClick={() => {
handleMemberExportFormatChange(option.value)
setShowFormatSelect(false)
}}
>
<span className="option-label">{option.label}</span>
<span className="option-desc">{option.desc}</span>
</button>
))}
</div>
)}
</div>
<div className="member-export-field member-export-folder">
<span></span>
<div className="member-export-folder-row">
<input value={exportFolder} readOnly placeholder="请选择导出目录" />
<button type="button" onClick={handleChooseExportFolder}>
</button>
</div>
</div>
</div>
<div className="member-export-options">
<div className="member-export-chip-group">
<span className="chip-group-label"></span>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportMedia ? 'active' : ''}`}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportMedia: !prev.exportMedia }))}
>
</button>
</div>
<div className="member-export-chip-group">
<span className="chip-group-label"></span>
<div className="member-export-chip-list">
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportImages ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportImages: !prev.exportImages }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportVoices ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportVoices: !prev.exportVoices }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportVideos ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportVideos: !prev.exportVideos }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportEmojis ? 'active' : ''} ${!memberExportOptions.exportMedia ? 'disabled' : ''}`}
disabled={!memberExportOptions.exportMedia}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportEmojis: !prev.exportEmojis }))}
>
</button>
</div>
</div>
<div className="member-export-chip-group">
<span className="chip-group-label"></span>
<div className="member-export-chip-list">
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportVoiceAsText ? 'active' : ''}`}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportVoiceAsText: !prev.exportVoiceAsText }))}
>
</button>
<button
type="button"
className={`export-filter-chip ${memberExportOptions.exportAvatars ? 'active' : ''}`}
onClick={() => setMemberExportOptions(prev => ({ ...prev, exportAvatars: !prev.exportAvatars }))}
>
</button>
</div>
</div>
<div className="member-export-field" ref={displayNameDropdownRef}>
<span></span>
<button
type="button"
className={`select-trigger ${showDisplayNameSelect ? 'open' : ''}`}
onClick={() => {
setShowDisplayNameSelect(prev => !prev)
setShowFormatSelect(false)
}}
>
<span className="select-value">{selectedDisplayNameOption.label}</span>
<ChevronDown size={16} />
</button>
{showDisplayNameSelect && (
<div className="select-dropdown">
{displayNameOptions.map(option => (
<button
key={option.value}
type="button"
className={`select-option ${memberExportOptions.displayNamePreference === option.value ? 'active' : ''}`}
onClick={() => {
setMemberExportOptions(prev => ({ ...prev, displayNamePreference: option.value }))
setShowDisplayNameSelect(false)
}}
>
<span className="option-label">{option.label}</span>
<span className="option-desc">{option.desc}</span>
</button>
))}
</div>
)}
</div>
</div>
<div className="member-export-actions">
<button
className="member-export-start-btn"
onClick={handleExportMemberMessages}
disabled={isExportingMemberMessages || !selectedMessageMemberUsername || !exportFolder}
>
{isExportingMemberMessages ? <Loader2 size={16} className="spin" /> : <Download size={16} />}
<span>{isExportingMemberMessages ? '导出中...' : '开始导出'}</span>
</button>
</div>
</div>
</div>
</div>
)
} }
return ( return (
@@ -1597,6 +1524,7 @@ function GroupAnalyticsPage() {
</div> </div>
</div> </div>
{renderMemberModal()} {renderMemberModal()}
{renderMemberExportModal()}
</div> </div>
) )
} }