mirror of
https://github.com/hicccc77/WeFlow.git
synced 2026-03-24 23:06:51 +00:00
feat: 一些更新
This commit is contained in:
@@ -52,7 +52,7 @@ WeFlow 是一个**完全本地**的微信**实时**聊天记录查看、分析
|
|||||||
- 本地解密与数据库管理
|
- 本地解密与数据库管理
|
||||||
|
|
||||||
> [!NOTE]
|
> [!NOTE]
|
||||||
> ⚠️ 本工具仅适配微信 **4.0 及以上**版本,请确保你的微信版本符合要求
|
> 本工具仅适配微信 **4.0 及以上**版本,请确保你的微信版本符合要求
|
||||||
|
|
||||||
## 快速开始
|
## 快速开始
|
||||||
|
|
||||||
|
|||||||
@@ -27,6 +27,11 @@ interface ConfigSchema {
|
|||||||
autoTranscribeVoice: boolean
|
autoTranscribeVoice: boolean
|
||||||
transcribeLanguages: string[]
|
transcribeLanguages: string[]
|
||||||
exportDefaultConcurrency: number
|
exportDefaultConcurrency: number
|
||||||
|
|
||||||
|
// 安全相关
|
||||||
|
authEnabled: boolean
|
||||||
|
authPassword: string // SHA-256 hash
|
||||||
|
authUseHello: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ConfigService {
|
export class ConfigService {
|
||||||
@@ -56,7 +61,11 @@ export class ConfigService {
|
|||||||
whisperDownloadSource: 'tsinghua',
|
whisperDownloadSource: 'tsinghua',
|
||||||
autoTranscribeVoice: false,
|
autoTranscribeVoice: false,
|
||||||
transcribeLanguages: ['zh'],
|
transcribeLanguages: ['zh'],
|
||||||
exportDefaultConcurrency: 2
|
exportDefaultConcurrency: 2,
|
||||||
|
|
||||||
|
authEnabled: false,
|
||||||
|
authPassword: '',
|
||||||
|
authUseHello: false
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -292,7 +292,7 @@ class ExportService {
|
|||||||
extBuffer = Buffer.from(extBuffer, 'base64')
|
extBuffer = Buffer.from(extBuffer, 'base64')
|
||||||
} else {
|
} else {
|
||||||
// 默认尝试hex
|
// 默认尝试hex
|
||||||
console.log('⚠️ 无法判断编码格式,默认尝试hex')
|
console.log(' 无法判断编码格式,默认尝试hex')
|
||||||
try {
|
try {
|
||||||
extBuffer = Buffer.from(extBuffer, 'hex')
|
extBuffer = Buffer.from(extBuffer, 'hex')
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|||||||
@@ -47,11 +47,11 @@ ManifestDPIAware true
|
|||||||
DetailPrint "Visual C++ Redistributable 安装成功"
|
DetailPrint "Visual C++ Redistributable 安装成功"
|
||||||
MessageBox MB_OK|MB_ICONINFORMATION "Visual C++ 运行库安装成功!"
|
MessageBox MB_OK|MB_ICONINFORMATION "Visual C++ 运行库安装成功!"
|
||||||
${Else}
|
${Else}
|
||||||
MessageBox MB_OK|MB_ICONEXCLAMATION "Visual C++ 运行库安装失败,您可能需要手动安装。"
|
MessageBox MB_OK|MB_ICONEXCLAMATION "Visual C++ 运行库安装失败,你可能需要手动安装。"
|
||||||
${EndIf}
|
${EndIf}
|
||||||
Delete "$TEMP\vc_redist.x64.exe"
|
Delete "$TEMP\vc_redist.x64.exe"
|
||||||
${Else}
|
${Else}
|
||||||
MessageBox MB_OK|MB_ICONEXCLAMATION "下载失败:$0$\n$\n您可以稍后手动下载安装 Visual C++ Redistributable。"
|
MessageBox MB_OK|MB_ICONEXCLAMATION "下载失败:$0$\n$\n你可以稍后手动下载安装 Visual C++ Redistributable。"
|
||||||
${EndIf}
|
${EndIf}
|
||||||
Goto doneVC
|
Goto doneVC
|
||||||
|
|
||||||
|
|||||||
39
src/App.tsx
39
src/App.tsx
@@ -27,6 +27,7 @@ import './App.scss'
|
|||||||
|
|
||||||
import UpdateDialog from './components/UpdateDialog'
|
import UpdateDialog from './components/UpdateDialog'
|
||||||
import UpdateProgressCapsule from './components/UpdateProgressCapsule'
|
import UpdateProgressCapsule from './components/UpdateProgressCapsule'
|
||||||
|
import LockScreen from './components/LockScreen'
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const navigate = useNavigate()
|
const navigate = useNavigate()
|
||||||
@@ -50,6 +51,10 @@ function App() {
|
|||||||
const isVideoPlayerWindow = location.pathname === '/video-player-window'
|
const isVideoPlayerWindow = location.pathname === '/video-player-window'
|
||||||
const [themeHydrated, setThemeHydrated] = useState(false)
|
const [themeHydrated, setThemeHydrated] = useState(false)
|
||||||
|
|
||||||
|
// 锁定状态
|
||||||
|
const [isLocked, setIsLocked] = useState(false)
|
||||||
|
const [lockAvatar, setLockAvatar] = useState('')
|
||||||
|
|
||||||
// 协议同意状态
|
// 协议同意状态
|
||||||
const [showAgreement, setShowAgreement] = useState(false)
|
const [showAgreement, setShowAgreement] = useState(false)
|
||||||
const [agreementChecked, setAgreementChecked] = useState(false)
|
const [agreementChecked, setAgreementChecked] = useState(false)
|
||||||
@@ -250,6 +255,28 @@ function App() {
|
|||||||
autoConnect()
|
autoConnect()
|
||||||
}, [isAgreementWindow, isOnboardingWindow, navigate, setDbConnected])
|
}, [isAgreementWindow, isOnboardingWindow, navigate, setDbConnected])
|
||||||
|
|
||||||
|
// 检查应用锁
|
||||||
|
useEffect(() => {
|
||||||
|
if (isAgreementWindow || isOnboardingWindow || isVideoPlayerWindow) return
|
||||||
|
|
||||||
|
const checkLock = async () => {
|
||||||
|
const enabled = await configService.getAuthEnabled()
|
||||||
|
if (enabled) {
|
||||||
|
setIsLocked(true)
|
||||||
|
// 尝试获取头像
|
||||||
|
try {
|
||||||
|
const result = await window.electronAPI.chat.getMyAvatarUrl()
|
||||||
|
if (result && result.success && result.avatarUrl) {
|
||||||
|
setLockAvatar(result.avatarUrl)
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error('获取锁屏头像失败', e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
checkLock()
|
||||||
|
}, [isAgreementWindow, isOnboardingWindow, isVideoPlayerWindow])
|
||||||
|
|
||||||
// 独立协议窗口
|
// 独立协议窗口
|
||||||
if (isAgreementWindow) {
|
if (isAgreementWindow) {
|
||||||
return <AgreementPage />
|
return <AgreementPage />
|
||||||
@@ -267,6 +294,12 @@ function App() {
|
|||||||
// 主窗口 - 完整布局
|
// 主窗口 - 完整布局
|
||||||
return (
|
return (
|
||||||
<div className="app-container">
|
<div className="app-container">
|
||||||
|
{isLocked && (
|
||||||
|
<LockScreen
|
||||||
|
onUnlock={() => setIsLocked(false)}
|
||||||
|
avatar={lockAvatar}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
<TitleBar />
|
<TitleBar />
|
||||||
|
|
||||||
{/* 全局悬浮进度胶囊 (处理:新版本提示、下载进度、错误提示) */}
|
{/* 全局悬浮进度胶囊 (处理:新版本提示、下载进度、错误提示) */}
|
||||||
@@ -293,13 +326,13 @@ function App() {
|
|||||||
</div>
|
</div>
|
||||||
<div className="agreement-text">
|
<div className="agreement-text">
|
||||||
<h4>1. 数据安全</h4>
|
<h4>1. 数据安全</h4>
|
||||||
<p>本软件所有数据处理均在本地完成,不会上传任何聊天记录、个人信息到服务器。您的数据完全由您自己掌控。</p>
|
<p>本软件所有数据处理均在本地完成,不会上传任何聊天记录、个人信息到服务器。你的数据完全由你自己掌控。</p>
|
||||||
|
|
||||||
<h4>2. 使用须知</h4>
|
<h4>2. 使用须知</h4>
|
||||||
<p>本软件仅供个人学习研究使用,请勿用于任何非法用途。使用本软件解密、查看、分析的数据应为您本人所有或已获得授权。</p>
|
<p>本软件仅供个人学习研究使用,请勿用于任何非法用途。使用本软件解密、查看、分析的数据应为你本人所有或已获得授权。</p>
|
||||||
|
|
||||||
<h4>3. 免责声明</h4>
|
<h4>3. 免责声明</h4>
|
||||||
<p>因使用本软件产生的任何直接或间接损失,开发者不承担任何责任。请确保您的使用行为符合当地法律法规。</p>
|
<p>因使用本软件产生的任何直接或间接损失,开发者不承担任何责任。请确保你的使用行为符合当地法律法规。</p>
|
||||||
|
|
||||||
<h4>4. 隐私保护</h4>
|
<h4>4. 隐私保护</h4>
|
||||||
<p>本软件不收集任何用户数据。软件更新检测仅获取版本信息,不涉及任何个人隐私。</p>
|
<p>本软件不收集任何用户数据。软件更新检测仅获取版本信息,不涉及任何个人隐私。</p>
|
||||||
|
|||||||
166
src/components/LockScreen.scss
Normal file
166
src/components/LockScreen.scss
Normal file
@@ -0,0 +1,166 @@
|
|||||||
|
.lock-screen {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100vw;
|
||||||
|
height: 100vh;
|
||||||
|
background-color: var(--bg-primary);
|
||||||
|
z-index: 9999;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
user-select: none;
|
||||||
|
-webkit-app-region: drag;
|
||||||
|
|
||||||
|
.lock-content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
width: 320px;
|
||||||
|
-webkit-app-region: no-drag;
|
||||||
|
animation: fadeIn 0.5s ease-out;
|
||||||
|
|
||||||
|
.lock-avatar {
|
||||||
|
width: 100px;
|
||||||
|
height: 100px;
|
||||||
|
border-radius: 50%;
|
||||||
|
margin-bottom: 24px;
|
||||||
|
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
|
||||||
|
border: 4px solid var(--bg-total);
|
||||||
|
background-color: var(--bg-secondary);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.lock-title {
|
||||||
|
font-size: 24px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-primary);
|
||||||
|
margin-bottom: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.lock-form {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 16px;
|
||||||
|
|
||||||
|
.input-group {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
input {
|
||||||
|
width: 100%;
|
||||||
|
height: 48px;
|
||||||
|
padding: 0 16px;
|
||||||
|
padding-right: 48px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
background-color: var(--bg-input);
|
||||||
|
color: var(--text-primary);
|
||||||
|
font-size: 16px;
|
||||||
|
outline: none;
|
||||||
|
transition: all 0.2s;
|
||||||
|
|
||||||
|
&:focus {
|
||||||
|
border-color: var(--primary-color);
|
||||||
|
box-shadow: 0 0 0 2px var(--primary-color-alpha);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.submit-btn {
|
||||||
|
position: absolute;
|
||||||
|
right: 8px;
|
||||||
|
top: 8px;
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: 8px;
|
||||||
|
border: none;
|
||||||
|
background: var(--primary-color);
|
||||||
|
color: white;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: opacity 0.2s;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
opacity: 0.9;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.hello-btn {
|
||||||
|
width: 100%;
|
||||||
|
height: 48px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 8px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
background-color: var(--bg-secondary);
|
||||||
|
color: var(--text-primary);
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--bg-hover);
|
||||||
|
transform: translateY(-1px);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.loading {
|
||||||
|
opacity: 0.7;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.lock-error {
|
||||||
|
margin-top: 16px;
|
||||||
|
color: #ff4d4f;
|
||||||
|
font-size: 14px;
|
||||||
|
animation: shake 0.5s ease-in-out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fadeIn {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(20px);
|
||||||
|
}
|
||||||
|
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateY(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes shake {
|
||||||
|
|
||||||
|
0%,
|
||||||
|
100% {
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
10%,
|
||||||
|
30%,
|
||||||
|
50%,
|
||||||
|
70%,
|
||||||
|
90% {
|
||||||
|
transform: translateX(-4px);
|
||||||
|
}
|
||||||
|
|
||||||
|
20%,
|
||||||
|
40%,
|
||||||
|
60%,
|
||||||
|
80% {
|
||||||
|
transform: translateX(4px);
|
||||||
|
}
|
||||||
|
}
|
||||||
149
src/components/LockScreen.tsx
Normal file
149
src/components/LockScreen.tsx
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
import { useState, useEffect, useRef } from 'react'
|
||||||
|
import * as configService from '../services/config'
|
||||||
|
import { ArrowRight, Fingerprint, Lock, ShieldCheck } from 'lucide-react'
|
||||||
|
import './LockScreen.scss'
|
||||||
|
|
||||||
|
interface LockScreenProps {
|
||||||
|
onUnlock: () => void
|
||||||
|
avatar?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
async function sha256(message: string) {
|
||||||
|
const msgBuffer = new TextEncoder().encode(message)
|
||||||
|
const hashBuffer = await crypto.subtle.digest('SHA-256', msgBuffer)
|
||||||
|
const hashArray = Array.from(new Uint8Array(hashBuffer))
|
||||||
|
const hashHex = hashArray.map(b => b.toString(16).padStart(2, '0')).join('')
|
||||||
|
return hashHex
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function LockScreen({ onUnlock, avatar }: LockScreenProps) {
|
||||||
|
const [password, setPassword] = useState('')
|
||||||
|
const [error, setError] = useState('')
|
||||||
|
const [isVerifying, setIsVerifying] = useState(false)
|
||||||
|
const [showHello, setShowHello] = useState(false)
|
||||||
|
const [helloAvailable, setHelloAvailable] = useState(false)
|
||||||
|
|
||||||
|
const inputRef = useRef<HTMLInputElement>(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
checkHelloAvailability()
|
||||||
|
// Auto focus input
|
||||||
|
inputRef.current?.focus()
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const checkHelloAvailability = async () => {
|
||||||
|
try {
|
||||||
|
const useHello = await configService.getAuthUseHello()
|
||||||
|
if (useHello && window.PublicKeyCredential) {
|
||||||
|
// Simple check if WebAuthn is supported
|
||||||
|
const available = await PublicKeyCredential.isUserVerifyingPlatformAuthenticatorAvailable()
|
||||||
|
setHelloAvailable(available)
|
||||||
|
if (available) {
|
||||||
|
setShowHello(true)
|
||||||
|
verifyHello()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Failed to check Hello availability', e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const verifyHello = async () => {
|
||||||
|
setIsVerifying(true)
|
||||||
|
setError('')
|
||||||
|
try {
|
||||||
|
// Use WebAuthn for authentication
|
||||||
|
// We use a dummy challenge because we are just verifying local presence
|
||||||
|
const challenge = new Uint8Array(32)
|
||||||
|
window.crypto.getRandomValues(challenge)
|
||||||
|
|
||||||
|
const credential = await navigator.credentials.get({
|
||||||
|
publicKey: {
|
||||||
|
challenge,
|
||||||
|
rpId: window.location.hostname, // 'localhost' or empty for file://
|
||||||
|
userVerification: 'required',
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (credential) {
|
||||||
|
onUnlock()
|
||||||
|
}
|
||||||
|
} catch (e: any) {
|
||||||
|
// NotAllowedError is common if user cancels
|
||||||
|
if (e.name !== 'NotAllowedError') {
|
||||||
|
console.error('Hello verification failed', e)
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
setIsVerifying(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handlePasswordSubmit = async (e?: React.FormEvent) => {
|
||||||
|
e?.preventDefault()
|
||||||
|
if (!password) return
|
||||||
|
|
||||||
|
setIsVerifying(true)
|
||||||
|
setError('')
|
||||||
|
|
||||||
|
try {
|
||||||
|
const storedHash = await configService.getAuthPassword()
|
||||||
|
const inputHash = await sha256(password)
|
||||||
|
|
||||||
|
if (inputHash === storedHash) {
|
||||||
|
onUnlock()
|
||||||
|
} else {
|
||||||
|
setError('密码错误')
|
||||||
|
setPassword('')
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
setError('验证失败')
|
||||||
|
} finally {
|
||||||
|
setIsVerifying(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="lock-screen">
|
||||||
|
<div className="lock-content">
|
||||||
|
<div className="lock-avatar">
|
||||||
|
{avatar ? (
|
||||||
|
<img src={avatar} alt="User" style={{ width: '100%', height: '100%', borderRadius: '50%' }} />
|
||||||
|
) : (
|
||||||
|
<Lock size={40} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h2 className="lock-title">WeFlow 已锁定</h2>
|
||||||
|
|
||||||
|
<form className="lock-form" onSubmit={handlePasswordSubmit}>
|
||||||
|
<div className="input-group">
|
||||||
|
<input
|
||||||
|
ref={inputRef}
|
||||||
|
type="password"
|
||||||
|
placeholder="输入应用密码"
|
||||||
|
value={password}
|
||||||
|
onChange={(e) => setPassword(e.target.value)}
|
||||||
|
disabled={isVerifying}
|
||||||
|
/>
|
||||||
|
<button type="submit" className="submit-btn" disabled={!password || isVerifying}>
|
||||||
|
<ArrowRight size={18} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{helloAvailable && (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={`hello-btn ${isVerifying ? 'loading' : ''}`}
|
||||||
|
onClick={verifyHello}
|
||||||
|
>
|
||||||
|
<Fingerprint size={20} />
|
||||||
|
{isVerifying ? '验证中...' : '使用 Windows Hello 解锁'}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</form>
|
||||||
|
|
||||||
|
{error && <div className="lock-error">{error}</div>}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -11,7 +11,7 @@ function AgreementPage() {
|
|||||||
<h2>用户协议</h2>
|
<h2>用户协议</h2>
|
||||||
|
|
||||||
<h3>一、总则</h3>
|
<h3>一、总则</h3>
|
||||||
<p>欢迎使用WeFlow(WeFlow)软件。请在使用本软件前仔细阅读本协议。一旦您开始使用本软件,即表示您已充分理解并同意本协议的全部内容。</p>
|
<p>欢迎使用WeFlow(WeFlow)软件。请在使用本软件前仔细阅读本协议。一旦你开始使用本软件,即表示你已充分理解并同意本协议的全部内容。</p>
|
||||||
|
|
||||||
<h3>二、软件说明</h3>
|
<h3>二、软件说明</h3>
|
||||||
<p>WeFlow是一款本地化的微信聊天记录查看与分析工具,所有数据处理均在用户本地设备上完成。</p>
|
<p>WeFlow是一款本地化的微信聊天记录查看与分析工具,所有数据处理均在用户本地设备上完成。</p>
|
||||||
@@ -35,7 +35,7 @@ function AgreementPage() {
|
|||||||
<p>本软件不收集、不上传、不存储任何用户个人信息或聊天数据。所有数据处理均在本地完成。</p>
|
<p>本软件不收集、不上传、不存储任何用户个人信息或聊天数据。所有数据处理均在本地完成。</p>
|
||||||
|
|
||||||
<h3>二、数据安全</h3>
|
<h3>二、数据安全</h3>
|
||||||
<p>您的聊天记录和个人数据完全存储在您的本地设备上,本软件不会将任何数据传输至外部服务器。</p>
|
<p>你的聊天记录和个人数据完全存储在你的本地设备上,本软件不会将任何数据传输至外部服务器。</p>
|
||||||
|
|
||||||
<h3>三、网络请求</h3>
|
<h3>三、网络请求</h3>
|
||||||
<p>本软件仅在检查更新时会访问更新服务器获取版本信息,不涉及任何用户数据的传输。</p>
|
<p>本软件仅在检查更新时会访问更新服务器获取版本信息,不涉及任何用户数据的传输。</p>
|
||||||
|
|||||||
@@ -34,8 +34,8 @@ function AnalyticsWelcomePage() {
|
|||||||
</div>
|
</div>
|
||||||
<h1>私聊数据分析</h1>
|
<h1>私聊数据分析</h1>
|
||||||
<p>
|
<p>
|
||||||
WeFlow 可以分析您的聊天记录,生成详细的统计报表。<br />
|
WeFlow 可以分析你的聊天记录,生成详细的统计报表。<br />
|
||||||
您可以选择加载上次的分析结果(速度快),或者开始新的分析(数据最新)。
|
你可以选择加载上次的分析结果(速度快),或者开始新的分析(数据最新)。
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<div className="action-cards">
|
<div className="action-cards">
|
||||||
|
|||||||
@@ -338,61 +338,33 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.time-options {
|
.time-range-picker-item {
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.checkbox-item {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 10px;
|
justify-content: space-between;
|
||||||
|
padding: 14px 16px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 14px;
|
transition: background 0.2s;
|
||||||
color: var(--text-primary);
|
background: transparent;
|
||||||
|
|
||||||
input[type="checkbox"] {
|
|
||||||
width: 18px;
|
|
||||||
height: 18px;
|
|
||||||
accent-color: var(--primary);
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
|
|
||||||
svg {
|
|
||||||
color: var(--text-secondary);
|
|
||||||
}
|
|
||||||
|
|
||||||
&.main-toggle {
|
|
||||||
padding: 12px 16px;
|
|
||||||
background: var(--bg-secondary);
|
|
||||||
border-radius: 10px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.date-range {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 10px;
|
|
||||||
padding: 12px 16px;
|
|
||||||
background: var(--bg-secondary);
|
|
||||||
border-radius: 10px;
|
|
||||||
font-size: 14px;
|
|
||||||
color: var(--text-primary);
|
|
||||||
cursor: pointer;
|
|
||||||
transition: all 0.2s;
|
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background: var(--bg-hover);
|
background: var(--bg-hover);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.time-picker-info {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
font-size: 14px;
|
||||||
|
color: var(--text-primary);
|
||||||
|
|
||||||
svg {
|
svg {
|
||||||
color: var(--text-tertiary);
|
color: var(--primary);
|
||||||
flex-shrink: 0;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
span {
|
svg {
|
||||||
flex: 1;
|
color: var(--text-tertiary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1184,50 +1156,4 @@
|
|||||||
color: var(--text-tertiary);
|
color: var(--text-tertiary);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Switch 开关样式
|
// 全局样式已在 main.scss 中定义
|
||||||
.switch {
|
|
||||||
position: relative;
|
|
||||||
display: inline-block;
|
|
||||||
width: 44px;
|
|
||||||
height: 24px;
|
|
||||||
flex-shrink: 0;
|
|
||||||
|
|
||||||
input {
|
|
||||||
opacity: 0;
|
|
||||||
width: 0;
|
|
||||||
height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.slider {
|
|
||||||
position: absolute;
|
|
||||||
cursor: pointer;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
bottom: 0;
|
|
||||||
background-color: var(--bg-tertiary);
|
|
||||||
transition: 0.3s;
|
|
||||||
border-radius: 24px;
|
|
||||||
|
|
||||||
&::before {
|
|
||||||
position: absolute;
|
|
||||||
content: "";
|
|
||||||
height: 18px;
|
|
||||||
width: 18px;
|
|
||||||
left: 3px;
|
|
||||||
bottom: 3px;
|
|
||||||
background-color: white;
|
|
||||||
transition: 0.3s;
|
|
||||||
border-radius: 50%;
|
|
||||||
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
input:checked+.slider {
|
|
||||||
background-color: var(--primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
input:checked+.slider::before {
|
|
||||||
transform: translateX(20px);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -537,21 +537,34 @@ function ExportPage() {
|
|||||||
|
|
||||||
<div className="setting-section">
|
<div className="setting-section">
|
||||||
<h3>时间范围</h3>
|
<h3>时间范围</h3>
|
||||||
<div className="time-options">
|
<p className="setting-subtitle">选择要导出的消息时间区间</p>
|
||||||
<label className="checkbox-item">
|
<div className="media-options-card">
|
||||||
|
<div className="media-switch-row">
|
||||||
|
<div className="media-switch-info">
|
||||||
|
<span className="media-switch-title">导出全部时间</span>
|
||||||
|
<span className="media-switch-desc">关闭此项以选择特定的起止日期</span>
|
||||||
|
</div>
|
||||||
|
<label className="switch">
|
||||||
<input
|
<input
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
checked={options.useAllTime}
|
checked={options.useAllTime}
|
||||||
onChange={e => setOptions({ ...options, useAllTime: e.target.checked })}
|
onChange={e => setOptions({ ...options, useAllTime: e.target.checked })}
|
||||||
/>
|
/>
|
||||||
<span>导出全部时间</span>
|
<span className="switch-slider"></span>
|
||||||
</label>
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
{!options.useAllTime && options.dateRange && (
|
{!options.useAllTime && options.dateRange && (
|
||||||
<div className="date-range" onClick={() => setShowDatePicker(true)}>
|
<>
|
||||||
|
<div className="media-option-divider"></div>
|
||||||
|
<div className="time-range-picker-item" onClick={() => setShowDatePicker(true)}>
|
||||||
|
<div className="time-picker-info">
|
||||||
<Calendar size={16} />
|
<Calendar size={16} />
|
||||||
<span>{formatDate(options.dateRange.start)} - {formatDate(options.dateRange.end)}</span>
|
<span>{formatDate(options.dateRange.start)} - {formatDate(options.dateRange.end)}</span>
|
||||||
|
</div>
|
||||||
<ChevronDown size={14} />
|
<ChevronDown size={14} />
|
||||||
</div>
|
</div>
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -609,7 +622,7 @@ function ExportPage() {
|
|||||||
checked={options.exportMedia}
|
checked={options.exportMedia}
|
||||||
onChange={e => setOptions({ ...options, exportMedia: e.target.checked })}
|
onChange={e => setOptions({ ...options, exportMedia: e.target.checked })}
|
||||||
/>
|
/>
|
||||||
<span className="slider"></span>
|
<span className="switch-slider"></span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -689,7 +702,7 @@ function ExportPage() {
|
|||||||
checked={options.exportAvatars}
|
checked={options.exportAvatars}
|
||||||
onChange={e => setOptions({ ...options, exportAvatars: e.target.checked })}
|
onChange={e => setOptions({ ...options, exportAvatars: e.target.checked })}
|
||||||
/>
|
/>
|
||||||
<span className="slider"></span>
|
<span className="switch-slider"></span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -603,54 +603,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.switch {
|
// 全局样式已在 main.scss 中定义
|
||||||
position: relative;
|
|
||||||
width: 46px;
|
|
||||||
height: 24px;
|
|
||||||
display: inline-block;
|
|
||||||
user-select: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.switch-input {
|
|
||||||
opacity: 0;
|
|
||||||
width: 0;
|
|
||||||
height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.switch-slider {
|
|
||||||
position: absolute;
|
|
||||||
cursor: pointer;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
bottom: 0;
|
|
||||||
background: var(--bg-tertiary);
|
|
||||||
border: 1px solid var(--border-color);
|
|
||||||
border-radius: 999px;
|
|
||||||
transition: all 0.2s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.switch-slider::before {
|
|
||||||
content: '';
|
|
||||||
position: absolute;
|
|
||||||
height: 18px;
|
|
||||||
width: 18px;
|
|
||||||
left: 3px;
|
|
||||||
top: 2px;
|
|
||||||
background: var(--text-tertiary);
|
|
||||||
border-radius: 50%;
|
|
||||||
transition: all 0.2s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.switch-input:checked+.switch-slider {
|
|
||||||
background: var(--primary);
|
|
||||||
border-color: var(--primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
.switch-input:checked+.switch-slider::before {
|
|
||||||
transform: translateX(22px);
|
|
||||||
background: #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.log-actions {
|
.log-actions {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@@ -8,11 +8,12 @@ import * as configService from '../services/config'
|
|||||||
import {
|
import {
|
||||||
Eye, EyeOff, FolderSearch, FolderOpen, Search, Copy,
|
Eye, EyeOff, FolderSearch, FolderOpen, Search, Copy,
|
||||||
RotateCcw, Trash2, Save, Plug, Check, Sun, Moon,
|
RotateCcw, Trash2, Save, Plug, Check, Sun, Moon,
|
||||||
Palette, Database, Download, HardDrive, Info, RefreshCw, ChevronDown, Mic
|
Palette, Database, Download, HardDrive, Info, RefreshCw, ChevronDown, Mic,
|
||||||
|
ShieldCheck, Fingerprint, Lock, KeyRound
|
||||||
} from 'lucide-react'
|
} from 'lucide-react'
|
||||||
import './SettingsPage.scss'
|
import './SettingsPage.scss'
|
||||||
|
|
||||||
type SettingsTab = 'appearance' | 'database' | 'whisper' | 'export' | 'cache' | 'about'
|
type SettingsTab = 'appearance' | 'database' | 'whisper' | 'export' | 'cache' | 'security' | 'about'
|
||||||
|
|
||||||
const tabs: { id: SettingsTab; label: string; icon: React.ElementType }[] = [
|
const tabs: { id: SettingsTab; label: string; icon: React.ElementType }[] = [
|
||||||
{ id: 'appearance', label: '外观', icon: Palette },
|
{ id: 'appearance', label: '外观', icon: Palette },
|
||||||
@@ -20,6 +21,7 @@ const tabs: { id: SettingsTab; label: string; icon: React.ElementType }[] = [
|
|||||||
{ id: 'whisper', label: '语音识别模型', icon: Mic },
|
{ id: 'whisper', label: '语音识别模型', icon: Mic },
|
||||||
{ id: 'export', label: '导出', icon: Download },
|
{ id: 'export', label: '导出', icon: Download },
|
||||||
{ id: 'cache', label: '缓存', icon: HardDrive },
|
{ id: 'cache', label: '缓存', icon: HardDrive },
|
||||||
|
{ id: 'security', label: '安全', icon: ShieldCheck },
|
||||||
{ id: 'about', label: '关于', icon: Info }
|
{ id: 'about', label: '关于', icon: Info }
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -95,8 +97,31 @@ function SettingsPage() {
|
|||||||
const [isClearingImageCache, setIsClearingImageCache] = useState(false)
|
const [isClearingImageCache, setIsClearingImageCache] = useState(false)
|
||||||
const [isClearingAllCache, setIsClearingAllCache] = useState(false)
|
const [isClearingAllCache, setIsClearingAllCache] = useState(false)
|
||||||
|
|
||||||
|
// 安全设置 state
|
||||||
|
const [authEnabled, setAuthEnabled] = useState(false)
|
||||||
|
const [authUseHello, setAuthUseHello] = useState(false)
|
||||||
|
const [helloAvailable, setHelloAvailable] = useState(false)
|
||||||
|
const [newPassword, setNewPassword] = useState('')
|
||||||
|
const [confirmPassword, setConfirmPassword] = useState('')
|
||||||
|
const [isSettingHello, setIsSettingHello] = useState(false)
|
||||||
|
|
||||||
const isClearingCache = isClearingAnalyticsCache || isClearingImageCache || isClearingAllCache
|
const isClearingCache = isClearingAnalyticsCache || isClearingImageCache || isClearingAllCache
|
||||||
|
|
||||||
|
// 检查 Hello 可用性
|
||||||
|
useEffect(() => {
|
||||||
|
if (window.PublicKeyCredential) {
|
||||||
|
void PublicKeyCredential.isUserVerifyingPlatformAuthenticatorAvailable().then(setHelloAvailable)
|
||||||
|
}
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
async function sha256(message: string) {
|
||||||
|
const msgBuffer = new TextEncoder().encode(message)
|
||||||
|
const hashBuffer = await crypto.subtle.digest('SHA-256', msgBuffer)
|
||||||
|
const hashArray = Array.from(new Uint8Array(hashBuffer))
|
||||||
|
const hashHex = hashArray.map(b => b.toString(16).padStart(2, '0')).join('')
|
||||||
|
return hashHex
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
loadConfig()
|
loadConfig()
|
||||||
loadAppVersion()
|
loadAppVersion()
|
||||||
@@ -154,6 +179,11 @@ function SettingsPage() {
|
|||||||
const savedExportDefaultExcelCompactColumns = await configService.getExportDefaultExcelCompactColumns()
|
const savedExportDefaultExcelCompactColumns = await configService.getExportDefaultExcelCompactColumns()
|
||||||
const savedExportDefaultConcurrency = await configService.getExportDefaultConcurrency()
|
const savedExportDefaultConcurrency = await configService.getExportDefaultConcurrency()
|
||||||
|
|
||||||
|
const savedAuthEnabled = await configService.getAuthEnabled()
|
||||||
|
const savedAuthUseHello = await configService.getAuthUseHello()
|
||||||
|
setAuthEnabled(savedAuthEnabled)
|
||||||
|
setAuthUseHello(savedAuthUseHello)
|
||||||
|
|
||||||
if (savedPath) setDbPath(savedPath)
|
if (savedPath) setDbPath(savedPath)
|
||||||
if (savedWxid) setWxid(savedWxid)
|
if (savedWxid) setWxid(savedWxid)
|
||||||
if (savedCachePath) setCachePath(savedCachePath)
|
if (savedCachePath) setCachePath(savedCachePath)
|
||||||
@@ -191,7 +221,7 @@ function SettingsPage() {
|
|||||||
|
|
||||||
|
|
||||||
if (savedWhisperModelDir) setWhisperModelDir(savedWhisperModelDir)
|
if (savedWhisperModelDir) setWhisperModelDir(savedWhisperModelDir)
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
console.error('加载配置失败:', e)
|
console.error('加载配置失败:', e)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -217,7 +247,7 @@ function SettingsPage() {
|
|||||||
try {
|
try {
|
||||||
const version = await window.electronAPI.app.getVersion()
|
const version = await window.electronAPI.app.getVersion()
|
||||||
setAppVersion(version)
|
setAppVersion(version)
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
console.error('获取版本号失败:', e)
|
console.error('获取版本号失败:', e)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -256,7 +286,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage('当前已是最新版', true)
|
showMessage('当前已是最新版', true)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`检查更新失败: ${e}`, false)
|
showMessage(`检查更新失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsCheckingUpdate(false)
|
setIsCheckingUpdate(false)
|
||||||
@@ -271,7 +301,7 @@ function SettingsPage() {
|
|||||||
try {
|
try {
|
||||||
showMessage('正在下载更新...', true)
|
showMessage('正在下载更新...', true)
|
||||||
await window.electronAPI.app.downloadAndInstall()
|
await window.electronAPI.app.downloadAndInstall()
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`更新失败: ${e}`, false)
|
showMessage(`更新失败: ${e}`, false)
|
||||||
setIsDownloading(false)
|
setIsDownloading(false)
|
||||||
}
|
}
|
||||||
@@ -366,7 +396,7 @@ function SettingsPage() {
|
|||||||
if (!result.success && result.error) {
|
if (!result.success && result.error) {
|
||||||
showMessage(result.error, false)
|
showMessage(result.error, false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`切换账号后重新连接失败: ${e}`, false)
|
showMessage(`切换账号后重新连接失败: ${e}`, false)
|
||||||
setDbConnected(false)
|
setDbConnected(false)
|
||||||
}
|
}
|
||||||
@@ -403,7 +433,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage(result.error || '未能自动检测到数据库目录', false)
|
showMessage(result.error || '未能自动检测到数据库目录', false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`自动检测失败: ${e}`, false)
|
showMessage(`自动检测失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsDetectingPath(false)
|
setIsDetectingPath(false)
|
||||||
@@ -417,7 +447,7 @@ function SettingsPage() {
|
|||||||
setDbPath(result.filePaths[0])
|
setDbPath(result.filePaths[0])
|
||||||
showMessage('已选择数据库目录', true)
|
showMessage('已选择数据库目录', true)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage('选择目录失败', false)
|
showMessage('选择目录失败', false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -445,7 +475,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
if (!silent) showMessage('未检测到账号目录,请检查路径', false)
|
if (!silent) showMessage('未检测到账号目录,请检查路径', false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
if (!silent) showMessage(`扫描失败: ${e}`, false)
|
if (!silent) showMessage(`扫描失败: ${e}`, false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -461,7 +491,7 @@ function SettingsPage() {
|
|||||||
setCachePath(result.filePaths[0])
|
setCachePath(result.filePaths[0])
|
||||||
showMessage('已选择缓存目录', true)
|
showMessage('已选择缓存目录', true)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage('选择目录失败', false)
|
showMessage('选择目录失败', false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -477,7 +507,7 @@ function SettingsPage() {
|
|||||||
await configService.setWhisperModelDir(dir)
|
await configService.setWhisperModelDir(dir)
|
||||||
showMessage('已选择 Whisper 模型目录', true)
|
showMessage('已选择 Whisper 模型目录', true)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage('选择目录失败', false)
|
showMessage('选择目录失败', false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -501,7 +531,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage(result.error || '模型下载失败', false)
|
showMessage(result.error || '模型下载失败', false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`模型下载失败: ${e}`, false)
|
showMessage(`模型下载失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsWhisperDownloading(false)
|
setIsWhisperDownloading(false)
|
||||||
@@ -533,7 +563,7 @@ function SettingsPage() {
|
|||||||
showMessage(result.error || '自动获取密钥失败', false)
|
showMessage(result.error || '自动获取密钥失败', false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`自动获取密钥失败: ${e}`, false)
|
showMessage(`自动获取密钥失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsFetchingDbKey(false)
|
setIsFetchingDbKey(false)
|
||||||
@@ -566,7 +596,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage(result.error || '自动获取图片密钥失败', false)
|
showMessage(result.error || '自动获取图片密钥失败', false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`自动获取图片密钥失败: ${e}`, false)
|
showMessage(`自动获取图片密钥失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsFetchingImageKey(false)
|
setIsFetchingImageKey(false)
|
||||||
@@ -589,7 +619,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage(result.error || '连接测试失败', false)
|
showMessage(result.error || '连接测试失败', false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`连接测试失败: ${e}`, false)
|
showMessage(`连接测试失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsTesting(false)
|
setIsTesting(false)
|
||||||
@@ -624,8 +654,14 @@ function SettingsPage() {
|
|||||||
await configService.setOnboardingDone(true)
|
await configService.setOnboardingDone(true)
|
||||||
|
|
||||||
// 保存按钮只负责持久化配置,不做连接测试/重连,避免影响聊天页的活动连接
|
// 保存按钮只负责持久化配置,不做连接测试/重连,避免影响聊天页的活动连接
|
||||||
|
|
||||||
|
// 保存安全配置
|
||||||
|
// 注意:这里只处理开关,密码修改是实时生效的(在 renderSecurityTab 里处理)
|
||||||
|
await configService.setAuthEnabled(authEnabled)
|
||||||
|
await configService.setAuthUseHello(authUseHello)
|
||||||
|
|
||||||
showMessage('配置保存成功', true)
|
showMessage('配置保存成功', true)
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`保存配置失败: ${e}`, false)
|
showMessage(`保存配置失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsLoadingState(false)
|
setIsLoadingState(false)
|
||||||
@@ -657,7 +693,7 @@ function SettingsPage() {
|
|||||||
setIsWhisperDownloading(false)
|
setIsWhisperDownloading(false)
|
||||||
setDbConnected(false)
|
setDbConnected(false)
|
||||||
await window.electronAPI.window.openOnboardingWindow()
|
await window.electronAPI.window.openOnboardingWindow()
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`清除配置失败: ${e}`, false)
|
showMessage(`清除配置失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsLoadingState(false)
|
setIsLoadingState(false)
|
||||||
@@ -669,7 +705,7 @@ function SettingsPage() {
|
|||||||
try {
|
try {
|
||||||
const logPath = await window.electronAPI.log.getPath()
|
const logPath = await window.electronAPI.log.getPath()
|
||||||
await window.electronAPI.shell.openPath(logPath)
|
await window.electronAPI.shell.openPath(logPath)
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`打开日志失败: ${e}`, false)
|
showMessage(`打开日志失败: ${e}`, false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -683,7 +719,7 @@ function SettingsPage() {
|
|||||||
}
|
}
|
||||||
await navigator.clipboard.writeText(result.content || '')
|
await navigator.clipboard.writeText(result.content || '')
|
||||||
showMessage('日志已复制到剪贴板', true)
|
showMessage('日志已复制到剪贴板', true)
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`复制日志失败: ${e}`, false)
|
showMessage(`复制日志失败: ${e}`, false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -699,7 +735,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage(`清除分析缓存失败: ${result.error || '未知错误'}`, false)
|
showMessage(`清除分析缓存失败: ${result.error || '未知错误'}`, false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`清除分析缓存失败: ${e}`, false)
|
showMessage(`清除分析缓存失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsClearingAnalyticsCache(false)
|
setIsClearingAnalyticsCache(false)
|
||||||
@@ -716,7 +752,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage(`清除图片缓存失败: ${result.error || '未知错误'}`, false)
|
showMessage(`清除图片缓存失败: ${result.error || '未知错误'}`, false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`清除图片缓存失败: ${e}`, false)
|
showMessage(`清除图片缓存失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsClearingImageCache(false)
|
setIsClearingImageCache(false)
|
||||||
@@ -734,7 +770,7 @@ function SettingsPage() {
|
|||||||
} else {
|
} else {
|
||||||
showMessage(`清除所有缓存失败: ${result.error || '未知错误'}`, false)
|
showMessage(`清除所有缓存失败: ${result.error || '未知错误'}`, false)
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e: any) {
|
||||||
showMessage(`清除所有缓存失败: ${e}`, false)
|
showMessage(`清除所有缓存失败: ${e}`, false)
|
||||||
} finally {
|
} finally {
|
||||||
setIsClearingAllCache(false)
|
setIsClearingAllCache(false)
|
||||||
@@ -797,7 +833,7 @@ function SettingsPage() {
|
|||||||
<div className="form-group">
|
<div className="form-group">
|
||||||
<label>数据库根目录</label>
|
<label>数据库根目录</label>
|
||||||
<span className="form-hint">xwechat_files 目录</span>
|
<span className="form-hint">xwechat_files 目录</span>
|
||||||
<span className="form-hint" style={{ color: '#ff6b6b' }}>⚠️ 目录路径不可包含中文,如有中文请去微信-设置-存储位置点击更改,迁移至全英文目录</span>
|
<span className="form-hint" style={{ color: '#ff6b6b' }}> 目录路径不可包含中文,如有中文请去微信-设置-存储位置点击更改,迁移至全英文目录</span>
|
||||||
<input type="text" placeholder="例如: C:\Users\xxx\Documents\xwechat_files" value={dbPath} onChange={(e) => setDbPath(e.target.value)} />
|
<input type="text" placeholder="例如: C:\Users\xxx\Documents\xwechat_files" value={dbPath} onChange={(e) => setDbPath(e.target.value)} />
|
||||||
<div className="btn-row">
|
<div className="btn-row">
|
||||||
<button className="btn btn-primary" onClick={handleAutoDetectPath} disabled={isDetectingPath}>
|
<button className="btn btn-primary" onClick={handleAutoDetectPath} disabled={isDetectingPath}>
|
||||||
@@ -1210,6 +1246,129 @@ function SettingsPage() {
|
|||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const handleSetupHello = async () => {
|
||||||
|
setIsSettingHello(true)
|
||||||
|
try {
|
||||||
|
const challenge = new Uint8Array(32)
|
||||||
|
window.crypto.getRandomValues(challenge)
|
||||||
|
|
||||||
|
const credential = await navigator.credentials.create({
|
||||||
|
publicKey: {
|
||||||
|
challenge,
|
||||||
|
rp: { name: 'WeFlow', id: window.location.hostname },
|
||||||
|
user: { id: new Uint8Array([1]), name: 'user', displayName: 'User' },
|
||||||
|
pubKeyCredParams: [{ alg: -7, type: 'public-key' }],
|
||||||
|
authenticatorSelection: { userVerification: 'required' },
|
||||||
|
timeout: 60000
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (credential) {
|
||||||
|
setAuthUseHello(true)
|
||||||
|
await configService.setAuthUseHello(true)
|
||||||
|
showMessage('Windows Hello 设置成功', true)
|
||||||
|
}
|
||||||
|
} catch (e: any) {
|
||||||
|
if (e.name !== 'NotAllowedError') {
|
||||||
|
showMessage(`Windows Hello 设置失败: ${e.message}`, false)
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
setIsSettingHello(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleUpdatePassword = async () => {
|
||||||
|
if (!newPassword || newPassword !== confirmPassword) {
|
||||||
|
showMessage('两次密码不一致', false)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 简单的保存逻辑,实际上应该先验证旧密码,但为了简化流程,这里直接允许覆盖
|
||||||
|
// 因为能进入设置页面说明已经解锁了
|
||||||
|
try {
|
||||||
|
const hash = await sha256(newPassword)
|
||||||
|
await configService.setAuthPassword(hash)
|
||||||
|
await configService.setAuthEnabled(true)
|
||||||
|
setAuthEnabled(true)
|
||||||
|
setNewPassword('')
|
||||||
|
setConfirmPassword('')
|
||||||
|
showMessage('密码已更新', true)
|
||||||
|
} catch (e: any) {
|
||||||
|
showMessage('密码更新失败', false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const renderSecurityTab = () => (
|
||||||
|
<div className="tab-content">
|
||||||
|
<div className="form-group">
|
||||||
|
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
|
||||||
|
<div>
|
||||||
|
<label>启用应用锁</label>
|
||||||
|
<span className="form-hint">每次启动应用时需要验证密码</span>
|
||||||
|
</div>
|
||||||
|
<label className="switch">
|
||||||
|
<input type="checkbox" checked={authEnabled} onChange={(e) => setAuthEnabled(e.target.checked)} />
|
||||||
|
<span className="switch-slider" />
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="divider" />
|
||||||
|
|
||||||
|
<div className="form-group">
|
||||||
|
<label>重置密码</label>
|
||||||
|
<span className="form-hint">设置新的启动密码</span>
|
||||||
|
|
||||||
|
<div style={{ marginTop: 10, display: 'flex', flexDirection: 'column', gap: 10 }}>
|
||||||
|
<input
|
||||||
|
type="password"
|
||||||
|
className="field-input"
|
||||||
|
placeholder="新密码"
|
||||||
|
value={newPassword}
|
||||||
|
onChange={e => setNewPassword(e.target.value)}
|
||||||
|
/>
|
||||||
|
<div style={{ display: 'flex', gap: 10 }}>
|
||||||
|
<input
|
||||||
|
type="password"
|
||||||
|
className="field-input"
|
||||||
|
placeholder="确认新密码"
|
||||||
|
value={confirmPassword}
|
||||||
|
onChange={e => setConfirmPassword(e.target.value)}
|
||||||
|
style={{ flex: 1 }}
|
||||||
|
/>
|
||||||
|
<button className="btn btn-primary" onClick={handleUpdatePassword} disabled={!newPassword}>更新</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="divider" />
|
||||||
|
|
||||||
|
<div className="form-group">
|
||||||
|
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
|
||||||
|
<div>
|
||||||
|
<label>Windows Hello</label>
|
||||||
|
<span className="form-hint">使用面容、指纹快速解锁</span>
|
||||||
|
{!helloAvailable && <div className="form-hint warning" style={{ color: '#ff4d4f' }}> 当前设备不支持 Windows Hello</div>}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
{authUseHello ? (
|
||||||
|
<button className="btn btn-secondary btn-sm" onClick={() => setAuthUseHello(false)}>关闭</button>
|
||||||
|
) : (
|
||||||
|
<button
|
||||||
|
className="btn btn-secondary btn-sm"
|
||||||
|
onClick={handleSetupHello}
|
||||||
|
disabled={!helloAvailable || isSettingHello}
|
||||||
|
>
|
||||||
|
{isSettingHello ? '设置中...' : '开启与设置'}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
|
||||||
const renderAboutTab = () => (
|
const renderAboutTab = () => (
|
||||||
<div className="tab-content about-tab">
|
<div className="tab-content about-tab">
|
||||||
<div className="about-card">
|
<div className="about-card">
|
||||||
@@ -1321,6 +1480,7 @@ function SettingsPage() {
|
|||||||
{activeTab === 'whisper' && renderWhisperTab()}
|
{activeTab === 'whisper' && renderWhisperTab()}
|
||||||
{activeTab === 'export' && renderExportTab()}
|
{activeTab === 'export' && renderExportTab()}
|
||||||
{activeTab === 'cache' && renderCacheTab()}
|
{activeTab === 'cache' && renderCacheTab()}
|
||||||
|
{activeTab === 'security' && renderSecurityTab()}
|
||||||
{activeTab === 'about' && renderAboutTab()}
|
{activeTab === 'about' && renderAboutTab()}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -112,7 +112,6 @@
|
|||||||
-webkit-app-region: drag;
|
-webkit-app-region: drag;
|
||||||
|
|
||||||
[data-mode="dark"] & {
|
[data-mode="dark"] & {
|
||||||
background: #18181b;
|
|
||||||
border-right-color: rgba(255, 255, 255, 0.08);
|
border-right-color: rgba(255, 255, 255, 0.08);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -152,7 +151,7 @@
|
|||||||
margin-top: 2px;
|
margin-top: 2px;
|
||||||
|
|
||||||
[data-mode="dark"] .welcome-sidebar & {
|
[data-mode="dark"] .welcome-sidebar & {
|
||||||
color: rgba(255, 255, 255, 0.45);
|
color: rgba(255, 255, 255, 0.6); // 稍微调亮一点
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -188,7 +187,7 @@
|
|||||||
border-radius: 12px;
|
border-radius: 12px;
|
||||||
|
|
||||||
[data-mode="dark"] .welcome-sidebar & {
|
[data-mode="dark"] .welcome-sidebar & {
|
||||||
opacity: 0.7;
|
opacity: 0.75; // 整体调亮一点,原来是0.7
|
||||||
}
|
}
|
||||||
|
|
||||||
&.active,
|
&.active,
|
||||||
@@ -236,8 +235,8 @@
|
|||||||
transition: all 0.3s ease;
|
transition: all 0.3s ease;
|
||||||
|
|
||||||
[data-mode="dark"] .welcome-sidebar & {
|
[data-mode="dark"] .welcome-sidebar & {
|
||||||
border-color: rgba(255, 255, 255, 0.1);
|
border-color: rgba(255, 255, 255, 0.2); // 稍微调亮边框
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.05);
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-item.active & {
|
.nav-item.active & {
|
||||||
@@ -281,7 +280,7 @@
|
|||||||
color: #1a1a1a;
|
color: #1a1a1a;
|
||||||
|
|
||||||
[data-mode="dark"] .welcome-sidebar & {
|
[data-mode="dark"] .welcome-sidebar & {
|
||||||
color: #ffffff;
|
color: rgba(255, 255, 255, 0.9); // 提高非活动标题亮度
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-item.active & {
|
.nav-item.active & {
|
||||||
@@ -299,7 +298,8 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.nav-item.active & {
|
.nav-item.active & {
|
||||||
color: rgba(255, 255, 255, 0.85);
|
color: #ffffff; // 活动描述使用纯白
|
||||||
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -315,7 +315,7 @@
|
|||||||
border-top: 1px dashed var(--border-color);
|
border-top: 1px dashed var(--border-color);
|
||||||
|
|
||||||
[data-mode="dark"] .welcome-sidebar & {
|
[data-mode="dark"] .welcome-sidebar & {
|
||||||
color: rgba(255, 255, 255, 0.5);
|
color: rgba(255, 255, 255, 0.65); // 提高底部文字亮度
|
||||||
border-top-color: rgba(255, 255, 255, 0.1);
|
border-top-color: rgba(255, 255, 255, 0.1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,8 @@ const steps = [
|
|||||||
{ id: 'db', title: '数据库目录', desc: '定位 xwechat_files 目录' },
|
{ id: 'db', title: '数据库目录', desc: '定位 xwechat_files 目录' },
|
||||||
{ id: 'cache', title: '缓存目录', desc: '设置本地缓存存储位置(可选)' },
|
{ id: 'cache', title: '缓存目录', desc: '设置本地缓存存储位置(可选)' },
|
||||||
{ id: 'key', title: '解密密钥', desc: '获取密钥与自动识别账号' },
|
{ id: 'key', title: '解密密钥', desc: '获取密钥与自动识别账号' },
|
||||||
{ id: 'image', title: '图片密钥', desc: '获取 XOR 与 AES 密钥' }
|
{ id: 'image', title: '图片密钥', desc: '获取 XOR 与 AES 密钥' },
|
||||||
|
{ id: 'security', title: '安全防护', desc: '保护你的数据' }
|
||||||
]
|
]
|
||||||
|
|
||||||
interface WelcomePageProps {
|
interface WelcomePageProps {
|
||||||
@@ -46,6 +47,64 @@ function WelcomePage({ standalone = false }: WelcomePageProps) {
|
|||||||
const [imageKeyStatus, setImageKeyStatus] = useState('')
|
const [imageKeyStatus, setImageKeyStatus] = useState('')
|
||||||
const [isManualStartPrompt, setIsManualStartPrompt] = useState(false)
|
const [isManualStartPrompt, setIsManualStartPrompt] = useState(false)
|
||||||
|
|
||||||
|
// 安全相关 state
|
||||||
|
const [enableAuth, setEnableAuth] = useState(false)
|
||||||
|
const [authPassword, setAuthPassword] = useState('')
|
||||||
|
const [authConfirmPassword, setAuthConfirmPassword] = useState('')
|
||||||
|
const [enableHello, setEnableHello] = useState(false)
|
||||||
|
const [helloAvailable, setHelloAvailable] = useState(false)
|
||||||
|
const [isSettingHello, setIsSettingHello] = useState(false)
|
||||||
|
|
||||||
|
// 检查 Hello 可用性
|
||||||
|
useEffect(() => {
|
||||||
|
if (window.PublicKeyCredential) {
|
||||||
|
void PublicKeyCredential.isUserVerifyingPlatformAuthenticatorAvailable().then(setHelloAvailable)
|
||||||
|
}
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
async function sha256(message: string) {
|
||||||
|
const msgBuffer = new TextEncoder().encode(message)
|
||||||
|
const hashBuffer = await crypto.subtle.digest('SHA-256', msgBuffer)
|
||||||
|
const hashArray = Array.from(new Uint8Array(hashBuffer))
|
||||||
|
const hashHex = hashArray.map(b => b.toString(16).padStart(2, '0')).join('')
|
||||||
|
return hashHex
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleSetupHello = async () => {
|
||||||
|
setIsSettingHello(true)
|
||||||
|
try {
|
||||||
|
// 注册凭证 (WebAuthn)
|
||||||
|
const challenge = new Uint8Array(32)
|
||||||
|
window.crypto.getRandomValues(challenge)
|
||||||
|
|
||||||
|
const credential = await navigator.credentials.create({
|
||||||
|
publicKey: {
|
||||||
|
challenge,
|
||||||
|
rp: { name: 'WeFlow', id: window.location.hostname },
|
||||||
|
user: {
|
||||||
|
id: new Uint8Array([1]),
|
||||||
|
name: 'user',
|
||||||
|
displayName: 'User'
|
||||||
|
},
|
||||||
|
pubKeyCredParams: [{ alg: -7, type: 'public-key' }],
|
||||||
|
authenticatorSelection: { userVerification: 'required' },
|
||||||
|
timeout: 60000
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (credential) {
|
||||||
|
setEnableHello(true)
|
||||||
|
// 成功提示?
|
||||||
|
}
|
||||||
|
} catch (e: any) {
|
||||||
|
if (e.name !== 'NotAllowedError') {
|
||||||
|
setError('Windows Hello 设置失败: ' + e.message)
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
setIsSettingHello(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const removeDb = window.electronAPI.key.onDbKeyStatus((payload) => {
|
const removeDb = window.electronAPI.key.onDbKeyStatus((payload) => {
|
||||||
setDbKeyStatus(payload.message)
|
setDbKeyStatus(payload.message)
|
||||||
@@ -227,6 +286,12 @@ function WelcomePage({ standalone = false }: WelcomePageProps) {
|
|||||||
if (currentStep.id === 'cache') return true
|
if (currentStep.id === 'cache') return true
|
||||||
if (currentStep.id === 'key') return decryptKey.length === 64 && Boolean(wxid)
|
if (currentStep.id === 'key') return decryptKey.length === 64 && Boolean(wxid)
|
||||||
if (currentStep.id === 'image') return true
|
if (currentStep.id === 'image') return true
|
||||||
|
if (currentStep.id === 'security') {
|
||||||
|
if (enableAuth) {
|
||||||
|
return authPassword.length > 0 && authPassword === authConfirmPassword
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -277,6 +342,15 @@ function WelcomePage({ standalone = false }: WelcomePageProps) {
|
|||||||
imageXorKey: typeof parsedXorKey === 'number' && !Number.isNaN(parsedXorKey) ? parsedXorKey : 0,
|
imageXorKey: typeof parsedXorKey === 'number' && !Number.isNaN(parsedXorKey) ? parsedXorKey : 0,
|
||||||
imageAesKey
|
imageAesKey
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// 保存安全配置
|
||||||
|
if (enableAuth && authPassword) {
|
||||||
|
const hash = await sha256(authPassword)
|
||||||
|
await configService.setAuthEnabled(true)
|
||||||
|
await configService.setAuthPassword(hash)
|
||||||
|
await configService.setAuthUseHello(enableHello)
|
||||||
|
}
|
||||||
|
|
||||||
await configService.setOnboardingDone(true)
|
await configService.setOnboardingDone(true)
|
||||||
|
|
||||||
setDbConnected(true, dbPath)
|
setDbConnected(true, dbPath)
|
||||||
@@ -450,7 +524,7 @@ function WelcomePage({ standalone = false }: WelcomePageProps) {
|
|||||||
|
|
||||||
<div className="field-hint">请选择微信-设置-存储位置对应的目录</div>
|
<div className="field-hint">请选择微信-设置-存储位置对应的目录</div>
|
||||||
<div className="field-hint warning">
|
<div className="field-hint warning">
|
||||||
⚠️ 目录路径不可包含中文,如有中文请先在微信中迁移至全英文目录
|
目录路径不可包含中文,如有中文请先在微信中迁移至全英文目录
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
@@ -525,6 +599,74 @@ function WelcomePage({ standalone = false }: WelcomePageProps) {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{currentStep.id === 'security' && (
|
||||||
|
<div className="form-group">
|
||||||
|
<div className="security-toggle-header" style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center', marginBottom: 16 }}>
|
||||||
|
<div className="toggle-info">
|
||||||
|
<label className="field-label" style={{ marginBottom: 0 }}>启用应用锁</label>
|
||||||
|
<div className="field-hint">每次启动应用时需要验证密码</div>
|
||||||
|
</div>
|
||||||
|
<label className="switch">
|
||||||
|
<input type="checkbox" checked={enableAuth} onChange={e => setEnableAuth(e.target.checked)} />
|
||||||
|
<span className="switch-slider" />
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{enableAuth && (
|
||||||
|
<div className="security-settings" style={{ marginTop: 20, padding: 16, backgroundColor: 'var(--bg-secondary)', borderRadius: 8 }}>
|
||||||
|
<div className="form-group">
|
||||||
|
<label className="field-label">应用密码</label>
|
||||||
|
<input
|
||||||
|
type="password"
|
||||||
|
className="field-input"
|
||||||
|
placeholder="请输入密码"
|
||||||
|
value={authPassword}
|
||||||
|
onChange={e => setAuthPassword(e.target.value)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="form-group">
|
||||||
|
<label className="field-label">确认密码</label>
|
||||||
|
<input
|
||||||
|
type="password"
|
||||||
|
className="field-input"
|
||||||
|
placeholder="请再次输入密码"
|
||||||
|
value={authConfirmPassword}
|
||||||
|
onChange={e => setAuthConfirmPassword(e.target.value)}
|
||||||
|
/>
|
||||||
|
{authPassword && authConfirmPassword && authPassword !== authConfirmPassword && (
|
||||||
|
<div className="error-text" style={{ color: '#ff4d4f', fontSize: 12, marginTop: 4 }}>两次密码不一致</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="divider" style={{ margin: '20px 0', borderTop: '1px solid var(--border-color)' }}></div>
|
||||||
|
|
||||||
|
<div className="security-toggle-header" style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
|
||||||
|
<div className="toggle-info">
|
||||||
|
<label className="field-label" style={{ marginBottom: 0 }}>Windows Hello</label>
|
||||||
|
<div className="field-hint">使用面容、指纹或 PIN 码快速解锁</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{enableHello ? (
|
||||||
|
<div style={{ color: '#52c41a', display: 'flex', alignItems: 'center', gap: 6 }}>
|
||||||
|
<CheckCircle2 size={16} /> 已开启
|
||||||
|
<button className="btn btn-ghost btn-sm" onClick={() => setEnableHello(false)} style={{ padding: '2px 8px', height: 24, fontSize: 12 }}>关闭</button>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<button
|
||||||
|
className="btn btn-secondary btn-sm"
|
||||||
|
disabled={!helloAvailable || isSettingHello}
|
||||||
|
onClick={handleSetupHello}
|
||||||
|
>
|
||||||
|
{isSettingHello ? '设置中...' : (helloAvailable ? '点击开启' : '不可用')}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{!helloAvailable && <div className="field-hint warning"> 当前设备不支持 Windows Hello 或未设置 PIN 码</div>}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{currentStep.id === 'image' && (
|
{currentStep.id === 'image' && (
|
||||||
<div className="form-group">
|
<div className="form-group">
|
||||||
<div className="grid-2">
|
<div className="grid-2">
|
||||||
@@ -564,8 +706,8 @@ function WelcomePage({ standalone = false }: WelcomePageProps) {
|
|||||||
|
|
||||||
{currentStep.id === 'intro' && (
|
{currentStep.id === 'intro' && (
|
||||||
<div className="intro-footer">
|
<div className="intro-footer">
|
||||||
<p>接下来的几个步骤将引导您连接本地微信数据库。</p>
|
<p>接下来的几个步骤将引导你连接本地微信数据库。</p>
|
||||||
<p>WeFlow 需要访问您的本地数据文件以提供分析与导出功能。</p>
|
<p>WeFlow 需要访问你的本地数据文件以提供分析与导出功能。</p>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
|||||||
@@ -30,7 +30,12 @@ export const CONFIG_KEYS = {
|
|||||||
EXPORT_DEFAULT_VOICE_AS_TEXT: 'exportDefaultVoiceAsText',
|
EXPORT_DEFAULT_VOICE_AS_TEXT: 'exportDefaultVoiceAsText',
|
||||||
EXPORT_DEFAULT_EXCEL_COMPACT_COLUMNS: 'exportDefaultExcelCompactColumns',
|
EXPORT_DEFAULT_EXCEL_COMPACT_COLUMNS: 'exportDefaultExcelCompactColumns',
|
||||||
EXPORT_DEFAULT_TXT_COLUMNS: 'exportDefaultTxtColumns',
|
EXPORT_DEFAULT_TXT_COLUMNS: 'exportDefaultTxtColumns',
|
||||||
EXPORT_DEFAULT_CONCURRENCY: 'exportDefaultConcurrency'
|
EXPORT_DEFAULT_CONCURRENCY: 'exportDefaultConcurrency',
|
||||||
|
|
||||||
|
// 安全
|
||||||
|
AUTH_ENABLED: 'authEnabled',
|
||||||
|
AUTH_PASSWORD: 'authPassword',
|
||||||
|
AUTH_USE_HELLO: 'authUseHello'
|
||||||
} as const
|
} as const
|
||||||
|
|
||||||
export interface WxidConfig {
|
export interface WxidConfig {
|
||||||
@@ -365,3 +370,32 @@ export async function getExportDefaultConcurrency(): Promise<number | null> {
|
|||||||
export async function setExportDefaultConcurrency(concurrency: number): Promise<void> {
|
export async function setExportDefaultConcurrency(concurrency: number): Promise<void> {
|
||||||
await config.set(CONFIG_KEYS.EXPORT_DEFAULT_CONCURRENCY, concurrency)
|
await config.set(CONFIG_KEYS.EXPORT_DEFAULT_CONCURRENCY, concurrency)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// === 安全相关 ===
|
||||||
|
|
||||||
|
export async function getAuthEnabled(): Promise<boolean> {
|
||||||
|
const value = await config.get(CONFIG_KEYS.AUTH_ENABLED)
|
||||||
|
return value === true
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function setAuthEnabled(enabled: boolean): Promise<void> {
|
||||||
|
await config.set(CONFIG_KEYS.AUTH_ENABLED, enabled)
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getAuthPassword(): Promise<string> {
|
||||||
|
const value = await config.get(CONFIG_KEYS.AUTH_PASSWORD)
|
||||||
|
return (value as string) || ''
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function setAuthPassword(passwordHash: string): Promise<void> {
|
||||||
|
await config.set(CONFIG_KEYS.AUTH_PASSWORD, passwordHash)
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getAuthUseHello(): Promise<boolean> {
|
||||||
|
const value = await config.get(CONFIG_KEYS.AUTH_USE_HELLO)
|
||||||
|
return value === true
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function setAuthUseHello(useHello: boolean): Promise<void> {
|
||||||
|
await config.set(CONFIG_KEYS.AUTH_USE_HELLO, useHello)
|
||||||
|
}
|
||||||
|
|||||||
@@ -235,7 +235,8 @@
|
|||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
html, body {
|
html,
|
||||||
|
body {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
font-family: 'HarmonyOS Sans SC', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
font-family: 'HarmonyOS Sans SC', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
@@ -307,3 +308,60 @@ html, body {
|
|||||||
box-shadow: var(--shadow-sm);
|
box-shadow: var(--shadow-sm);
|
||||||
padding: 16px;
|
padding: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 全局 Switch 开关样式
|
||||||
|
.switch {
|
||||||
|
position: relative;
|
||||||
|
display: inline-block;
|
||||||
|
width: 44px;
|
||||||
|
height: 24px;
|
||||||
|
flex-shrink: 0;
|
||||||
|
|
||||||
|
input {
|
||||||
|
opacity: 0;
|
||||||
|
width: 0;
|
||||||
|
height: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.switch-slider {
|
||||||
|
position: absolute;
|
||||||
|
cursor: pointer;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background-color: var(--bg-tertiary);
|
||||||
|
transition: 0.3s;
|
||||||
|
border-radius: 24px;
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
|
||||||
|
&::before {
|
||||||
|
position: absolute;
|
||||||
|
content: "";
|
||||||
|
height: 18px;
|
||||||
|
width: 18px;
|
||||||
|
left: 2px;
|
||||||
|
bottom: 2px;
|
||||||
|
background-color: var(--text-tertiary);
|
||||||
|
transition: 0.3s;
|
||||||
|
border-radius: 50%;
|
||||||
|
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
input:checked+.switch-slider {
|
||||||
|
background-color: var(--primary);
|
||||||
|
border-color: var(--primary);
|
||||||
|
|
||||||
|
&::before {
|
||||||
|
transform: translateX(20px);
|
||||||
|
background-color: #ffffff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 禁用状态
|
||||||
|
input:disabled+.switch-slider {
|
||||||
|
opacity: 0.5;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user