104 lines
3.7 KiB
JavaScript
104 lines
3.7 KiB
JavaScript
'use client';
|
||
|
||
import { useState, useEffect } from 'react';
|
||
import { motion, AnimatePresence } from 'framer-motion';
|
||
|
||
const ANNOUNCEMENT_KEY = 'hasClosedAnnouncement_v20';
|
||
|
||
export default function Announcement() {
|
||
const [isVisible, setIsVisible] = useState(false);
|
||
|
||
useEffect(() => {
|
||
const hasClosed = localStorage.getItem(ANNOUNCEMENT_KEY);
|
||
if (!hasClosed) {
|
||
setIsVisible(true);
|
||
}
|
||
}, []);
|
||
|
||
const handleClose = () => {
|
||
// 清理历史 ANNOUNCEMENT_KEY
|
||
const keysToRemove = [];
|
||
for (let i = 0; i < localStorage.length; i++) {
|
||
const key = localStorage.key(i);
|
||
if (key && key.startsWith('hasClosedAnnouncement_v') && key !== ANNOUNCEMENT_KEY) {
|
||
keysToRemove.push(key);
|
||
}
|
||
}
|
||
keysToRemove.forEach((k) => localStorage.removeItem(k));
|
||
|
||
localStorage.setItem(ANNOUNCEMENT_KEY, 'true');
|
||
setIsVisible(false);
|
||
};
|
||
|
||
return (
|
||
<AnimatePresence>
|
||
{isVisible && (
|
||
<motion.div
|
||
initial={{ opacity: 0 }}
|
||
animate={{ opacity: 1 }}
|
||
exit={{ opacity: 0 }}
|
||
style={{
|
||
position: 'fixed',
|
||
inset: 0,
|
||
zIndex: 100,
|
||
display: 'flex',
|
||
alignItems: 'center',
|
||
justifyContent: 'center',
|
||
background: 'rgba(0, 0, 0, 0.6)',
|
||
backdropFilter: 'blur(4px)',
|
||
padding: '20px',
|
||
}}
|
||
>
|
||
<motion.div
|
||
initial={{ scale: 0.95, opacity: 0, y: 10 }}
|
||
animate={{ scale: 1, opacity: 1, y: 0 }}
|
||
exit={{ scale: 0.95, opacity: 0, y: 10 }}
|
||
className="glass"
|
||
style={{
|
||
width: '100%',
|
||
maxWidth: '400px',
|
||
padding: '24px',
|
||
position: 'relative',
|
||
display: 'flex',
|
||
flexDirection: 'column',
|
||
gap: '16px',
|
||
maxHeight: 'calc(100dvh - 40px)',
|
||
overflow: 'hidden',
|
||
}}
|
||
>
|
||
<div className="title" style={{ display: 'flex', alignItems: 'center', gap: '12px', fontWeight: 700, fontSize: '18px', color: 'var(--accent)' }}>
|
||
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
||
<path d="M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z"></path>
|
||
<line x1="12" y1="9" x2="12" y2="13"></line>
|
||
<line x1="12" y1="17" x2="12.01" y2="17"></line>
|
||
</svg>
|
||
<span>公告</span>
|
||
</div>
|
||
<div style={{ color: 'var(--text)', lineHeight: '1.6', fontSize: '15px', overflowY: 'auto', minHeight: 0, flex: 1, paddingRight: '4px' }}>
|
||
<p>v0.2.9 更新内容:</p>
|
||
<p>1. 排序新增按昨日涨幅排序。</p>
|
||
<p>2. 排序个性化设置支持切换排序形式。</p>
|
||
<p>3. 全局设置新增显示/隐藏大盘指数。</p>
|
||
<p>4. 新增持有天数。</p>
|
||
<p>5. 登录方式支持 Github。</p>
|
||
<br/>
|
||
关联板块实时估值还在测试,会在近期上线。
|
||
<p>如有建议和问题,欢迎进用户支持群反馈。</p>
|
||
</div>
|
||
|
||
<div style={{ display: 'flex', justifyContent: 'flex-end', marginTop: '8px' }}>
|
||
<button
|
||
className="button"
|
||
onClick={handleClose}
|
||
style={{ width: '100%', justifyContent: 'center', display: 'flex', alignItems: 'center' }}
|
||
>
|
||
我知道了
|
||
</button>
|
||
</div>
|
||
</motion.div>
|
||
</motion.div>
|
||
)}
|
||
</AnimatePresence>
|
||
);
|
||
}
|