/**
* mmnote v1.0.0 - my markdown note.
* Copyright (c) 2025, mmnote.com. (MIT Licensed)
* https://mmnote.com
*/
// 保存笔记的路径前缀
const SAVE_PATH = '_tmp';
// 有效的笔记名称格式(只允许字母、数字、下划线和连字符)
const VALID_NOTE_PATTERN = /^[a-zA-Z0-9_-]+$/;
/**
* 使用 notePath 生成加密密钥
* @param {string} notePath - 笔记路径
* @returns {Promise<CryptoKey>} 加密密钥
*/
async function generateEncryptionKey(notePath) {
const encoder = new TextEncoder();
const keyData = encoder.encode(notePath);
const hashBuffer = await crypto.subtle.digest('SHA-256', keyData);
return await crypto.subtle.importKey(
'raw',
hashBuffer,
{ name: 'AES-GCM' },
false,
['encrypt', 'decrypt']
);
}
/**
* 加密文本
* @param {string} text - 要加密的文本
* @param {CryptoKey} key - 加密密钥
* @returns {Promise<string>} 加密后的base64字符串
*/
async function encryptText(text, key) {
const encoder = new TextEncoder();
const data = encoder.encode(text);
const iv = crypto.getRandomValues(new Uint8Array(12));
const encryptedData = await crypto.subtle.encrypt(
{ name: 'AES-GCM', iv },
key,
data
);
const encryptedArray = new Uint8Array(encryptedData);
const resultArray = new Uint8Array(iv.length + encryptedArray.length);
resultArray.set(iv);
resultArray.set(encryptedArray, iv.length);
return btoa(String.fromCharCode(...resultArray));
}
/**
* 解密文本
* @param {string} encryptedText - 加密的base64字符串
* @param {CryptoKey} key - 解密密钥
* @returns {Promise<string>} 解密后的文本
*/
async function decryptText(encryptedText, key) {
try {
const encryptedArray = new Uint8Array(atob(encryptedText).split('').map(c => c.charCodeAt(0)));
const iv = encryptedArray.slice(0, 12);
const data = encryptedArray.slice(12);
const decryptedData = await crypto.subtle.decrypt(
{ name: 'AES-GCM', iv },
key,
data
);
const decoder = new TextDecoder();
return decoder.decode(decryptedData);
} catch (error) {
console.error('解密失败:', error);
return ''; // 解密失败返回空字符串
}
}
/**
* 处理所有传入的请求
* @param {Request} request - 传入的请求对象
* @returns {Response} 响应对象
*/
async function handleRequest(request) {
const url = new URL(request.url);
const pathParts = url.pathname.split('/').filter(Boolean);
const noteName = pathParts[0];
const action = pathParts[1];
// 处理分享相关的请求
if (noteName === 'share') {
if (request.method === 'POST' && action) {
try {
const shareData = await request.json();
await NOTES_KV.put('share_' + action, JSON.stringify(shareData));
return new Response(null, { status: 200 });
} catch (error) {
return new Response('保存分享数据失败', { status: 500 });
}
} else if (request.method === 'GET' && action) {
return await handleShareRequest(action);
}
}
// 如果笔记名无效或不存在,生成随机名称并重定向
if (!noteName || noteName.length > 64 || !VALID_NOTE_PATTERN.test(noteName)) {
const randomNoteName = generateRandomNoteName();
return Response.redirect(`${url.origin}/${randomNoteName}`, 302);
}
const notePath = `${SAVE_PATH}/${noteName}`;
// 处理密码相关的请求
if (action) {
// 使用笔记名和固定盐值生成密码存储键
const passwordKey = await generatePasswordKey(noteName);
switch (action) {
case 'password':
if (request.method === 'POST') {
const { password } = await request.json();
// 为每个密码生成唯一盐值
const salt = crypto.getRandomValues(new Uint8Array(16));
const saltHex = Array.from(salt).map(b => b.toString(16).padStart(2, '0')).join('');
// 使用盐值和密码生成最终哈希
const finalHash = await hashPasswordWithSalt(password, salt);
// 存储盐值和哈希
await NOTES_KV.put(passwordKey, JSON.stringify({
hash: finalHash,
salt: saltHex
}));
return new Response(null, { status: 200 });
} else if (request.method === 'DELETE') {
const { password } = await request.json();
const storedData = await NOTES_KV.get(passwordKey);
if (storedData) {
const { hash, salt } = JSON.parse(storedData);
const saltArray = new Uint8Array(salt.match(/.{2}/g).map(byte => parseInt(byte, 16)));
const checkHash = await hashPasswordWithSalt(password, saltArray);
if (checkHash === hash) {
await NOTES_KV.delete(passwordKey);
return new Response(null, { status: 200 });
}
}
return new Response('Invalid password', { status: 401 });
}
break;
case 'password-check':
const hasPassword = await NOTES_KV.get(passwordKey);
return new Response(null, { status: hasPassword ? 200 : 404 });
case 'password-verify':
const { password } = await request.json();
const storedData = await NOTES_KV.get(passwordKey);
if (storedData) {
const { hash, salt } = JSON.parse(storedData);
const saltArray = new Uint8Array(salt.match(/.{2}/g).map(byte => parseInt(byte, 16)));
const checkHash = await hashPasswordWithSalt(password, saltArray);
if (checkHash === hash) {
return new Response(null, { status: 200 });
}
}
return new Response('Invalid password', { status: 401 });
}
}
const raw = url.searchParams.has('raw');
// 根据请求方法分发处理
switch (request.method) {
case 'POST':
return await handlePostRequest(request, notePath);
case 'GET':
return raw || isCommandLineRequest(request)
? await handleRawRequest(notePath)
: await handleGetRequest(notePath, noteName);
default:
return new Response('Method Not Allowed', { status: 405 });
}
}
/**
* 使用笔记名和固定盐值生成密码存储键
* @param {string} noteName - 笔记名称
* @returns {Promise<string>} 密码存储键
*/
async function generatePasswordKey(noteName) {
const encoder = new TextEncoder();
const data = encoder.encode(noteName + '_pwd_protected');
const hashBuffer = await crypto.subtle.digest('SHA-256', data);
const hashArray = Array.from(new Uint8Array(hashBuffer));
return '_secure_' + hashArray.map(b => b.toString(16).padStart(2, '0')).join('');
}
/**
* 使用盐值对密码进行哈希
* @param {string} password - 原始密码
* @param {Uint8Array} salt - 盐值
* @returns {Promise<string>} 哈希后的密码
*/
async function hashPasswordWithSalt(password, salt) {
const encoder = new TextEncoder();
const passwordData = encoder.encode(password);
// 将密码和盐值连接
const dataToHash = new Uint8Array(passwordData.length + salt.length);
dataToHash.set(passwordData);
dataToHash.set(salt, passwordData.length);
// 进行哈希
const hashBuffer = await crypto.subtle.digest('SHA-256', dataToHash);
const hashArray = Array.from(new Uint8Array(hashBuffer));
return hashArray.map(b => b.toString(16).padStart(2, '0')).join('');
}
/**
* 处理POST请求 - 保存或删除笔记内容
* @param {Request} request - POST请求对象
* @param {string} notePath - 笔记保存路径
* @returns {Response} 响应对象
*/
async function handlePostRequest(request, notePath) {
const formData = await request.formData();
const text = formData.get('text') || await request.text();
// 如果内容为空,删除笔记
if (text.trim().length === 0) {
await handleDeleteRequest(notePath);
return new Response('Note will be deleted', { status: 200 });
} else {
await saveNoteContent(notePath, text);
return new Response(null, { status: 204 });
}
}
/**
* 处理GET请求 - 返回笔记的HTML页面
* @param {string} notePath - 笔记路径
* @param {string} noteName - 笔记名称
* @returns {Response} HTML响应
*/
async function handleGetRequest(notePath, noteName) {
const noteContent = await getNoteContent(notePath);
const html = generateHTML(noteName, noteContent);
return new Response(html, { headers: { 'Content-Type': 'text/html' } });
}
/**
* 处理原始内容请求 - 返回纯文本格式的笔记内容
* @param {string} notePath - 笔记路径
* @returns {Response} 文本响应
*/
async function handleRawRequest(notePath) {
const noteContent = await getNoteContent(notePath);
return noteContent
? new Response(noteContent, { headers: { 'Content-Type': 'text/plain' } })
: new Response('404 Not Found', { status: 404 });
}
/**
* 处理删除请求
* @param {string} notePath - 要删除的笔记路径
* @returns {Response} 响应对象
*/
async function handleDeleteRequest(notePath) {
await deleteNoteContent(notePath);
return new Response(null, { status: 204 });
}
/**
* 生成随机笔记名称
* @returns {string} 5位随机字符串
*/
function generateRandomNoteName() {
const chars = '0123456789abcdefghijklmnopqrstuvwxyz';
return Array.from({ length: 5 }, () => chars[Math.floor(Math.random() * chars.length)]).join('');
}
/**
* 检查是否为命令行请求
* @param {Request} request - 请求对象
* @returns {boolean} 是否为命令行请求
*/
function isCommandLineRequest(request) {
const userAgent = request.headers.get('User-Agent') || '';
return userAgent.startsWith('curl') || userAgent.startsWith('Wget');
}
function generateHTML(noteName, noteContent) {
return `<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>${noteName} - 在线笔记</title>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/styles/github.min.css">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/styles/github-dark.min.css" media="(prefers-color-scheme: dark)">
<!-- 添加KaTeX依赖 -->
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/katex@0.16.9/dist/katex.min.css">
<script src="https://cdn.jsdelivr.net/npm/katex@0.16.9/dist/katex.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/katex@0.16.9/dist/contrib/auto-render.min.js"></script>
<!-- 添加Mermaid依赖 -->
<script src="https://cdn.jsdelivr.net/npm/mermaid@10.6.1/dist/mermaid.min.js"></script>
<style>
:root {
--primary-color: #4e92d1;
--secondary-color: #6c757d;
--bg-color: #ffffff;
--text-color: #333333;
--border-color: #e0e0e0;
--container-bg: #ffffff;
--editor-bg: #f8f8f8;
--shadow-color: rgba(0, 0, 0, 0.1);
--hover-color: #f0f0f0;
--link-color: #0366d6;
--link-hover-color: #0969da;
--link-visited-color: #6f42c1;
}
[data-theme="dark"] {
--primary-color: #a2c2f5;
--secondary-color: #9ca3af;
--bg-color: #1a1a1a;
--text-color: #f1f1f1;
--border-color: #404040;
--container-bg: #2a2a2a;
--editor-bg: #333333;
--shadow-color: rgba(0, 0, 0, 0.3);
--hover-color: #3a3a3a;
--link-color: #58a6ff;
--link-hover-color: #79b8ff;
--link-visited-color: #bc8cff;
}
* {
box-sizing: border-box;
margin: 0;
padding: 0;
}
body {
margin: 0;
background: var(--bg-color);
color: var(--text-color);
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
font-size: 16px;
line-height: 1.6;
transition: all 0.3s ease;
}
.container {
width: 100%;
max-width: 100%;
margin: 0 auto;
padding: 10px;
min-height: 100vh;
height: 100vh;
display: flex;
flex-direction: column;
background-color: var(--container-bg);
}
.container.toolbar-hidden {
padding: 10px 10px 10px 10px;
}
.container.toolbar-hidden .editor-container {
height: calc(100vh - 20px);
}
@media (min-width: 1200px) {
.container {
max-width: 95%;
box-shadow: 0 0 20px var(--shadow-color);
}
}
.toolbar {
display: flex;
justify-content: space-between;
align-items: center;
padding: 12px 15px;
background: var(--editor-bg);
border-radius: 12px;
margin-bottom: 10px;
gap: 10px;
overflow-x: auto;
-webkit-overflow-scrolling: touch;
scrollbar-width: none;
flex-shrink: 0;
transition: all 0.3s ease;
cursor: grab;
}
/* 添加工具栏滚动条样式 */
.toolbar::-webkit-scrollbar {
height: 6px;
width: 6px;
}
.toolbar::-webkit-scrollbar-track {
background: transparent;
}
.toolbar::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 3px;
}
.toolbar::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
/* 拖动时的光标样式 */
.toolbar.dragging {
cursor: grabbing;
}
.toolbar.hidden {
display: none;
}
.toolbar-left {
display: flex;
align-items: center;
gap: 15px;
flex-shrink: 0; /* 防止工具栏压缩 */
}
.toolbar-right {
display: flex;
align-items: center;
gap: 15px;
flex-shrink: 0; /* 防止工具栏压缩 */
}
.toolbar-divider {
width: 1px;
height: 20px;
background-color: var(--border-color);
margin: 0 2px;
}
.switch-label {
display: flex;
align-items: center;
gap: 8px;
cursor: pointer;
color: var(--text-color);
font-size: 0.9rem;
}
.switch-label input[type="checkbox"] {
width: 16px;
height: 16px;
}
.mode-toggle {
background: none;
border: none;
font-size: 1.2rem;
cursor: pointer;
padding: 8px;
border-radius: 50%;
transition: all 0.3s ease;
color: var(--text-color);
display: flex;
align-items: center;
justify-content: center;
}
.mode-toggle:hover {
background: var(--hover-color);
}
.mode-toggle .sun-icon,
.mode-toggle .moon-icon {
display: none;
}
[data-theme="dark"] .moon-icon {
display: block;
}
[data-theme="light"] .sun-icon {
display: block;
}
.icon-container {
width: 20px;
height: 20px;
display: flex;
align-items: center;
justify-content: center;
}
.note-name {
color: var(--primary-color);
font-weight: 500;
cursor: pointer;
padding: 5px 10px;
border-radius: 6px;
transition: all 0.3s ease;
display: flex;
align-items: center;
gap: 5px;
transform-origin: left center;
}
.note-name:hover {
background: var(--hover-color);
transform: scale(1.1);
}
#save-status {
color: var(--secondary-color);
font-size: 0.9rem;
margin-left: 10px;
}
.editor-container {
display: grid;
grid-template-columns: 1fr;
gap: 15px;
flex: 1;
min-height: 0;
margin: 0;
position: relative;
transition: all 0.3s ease;
}
/* 编辑器包装器 */
.editor-wrapper {
position: relative;
width: 100%;
height: 100%;
display: flex;
background: var(--editor-bg);
border: 1px solid var(--border-color);
border-radius: 8px;
overflow: hidden;
flex-direction: column;
}
/* 设置统一的行高和字符宽度 */
.editor-wrapper textarea,
.line-numbers span {
line-height: 1.5;
min-height: 1.5em;
font-size: 14px;
font-family: 'Consolas', 'Monaco', monospace;
box-sizing: border-box;
letter-spacing: 0;
}
.line-numbers {
padding: 15px 2px 40px 2px;
background: var(--editor-bg);
border-right: 1px solid var(--border-color);
color: var(--secondary-color);
user-select: none;
overflow: hidden;
min-width: 28px;
width: auto;
display: flex;
flex-direction: column;
align-items: flex-end;
}
.line-numbers span {
display: block;
padding: 0 4px;
min-width: 24px;
text-align: right;
white-space: nowrap;
}
.line-numbers.hidden {
display: none;
}
/* 编辑器主体区域 */
.editor-main {
display: flex;
flex: 1;
overflow: hidden;
position: relative;
}
/* 状态栏样式统一 */
.status-bar,
.preview-status-bar {
position: relative;
height: 25px;
background: var(--editor-bg);
border-top: 1px solid var(--border-color);
display: flex;
align-items: center;
padding: 0 10px;
font-size: 12px;
color: var(--secondary-color);
justify-content: space-between;
overflow-x: auto;
overflow-y: hidden;
-webkit-overflow-scrolling: touch;
scrollbar-width: none;
white-space: nowrap;
flex-shrink: 0;
}
/* 状态栏滚动条样式 */
.status-bar::-webkit-scrollbar,
.preview-status-bar::-webkit-scrollbar {
height: 4px;
width: 4px;
}
.status-bar::-webkit-scrollbar-track,
.preview-status-bar::-webkit-scrollbar-track {
background: transparent;
}
.status-bar::-webkit-scrollbar-thumb,
.preview-status-bar::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 2px;
}
.status-bar::-webkit-scrollbar-thumb:hover,
.preview-status-bar::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
.status-left,
.preview-status-left {
display: flex;
align-items: center;
flex-wrap: nowrap;
min-width: min-content;
gap: 8px;
padding-right: 8px;
}
.status-right,
.preview-status-right {
display: flex;
align-items: center;
margin-left: auto;
padding-left: 15px;
flex-wrap: nowrap;
min-width: min-content;
gap: 8px;
}
/* 确保状态栏内容不会被截断 */
.status-bar > div,
.preview-status-bar > div {
flex-shrink: 0;
}
.status-item,
.preview-status-item {
display: flex;
align-items: center;
gap: 5px;
font-family: 'Consolas', 'Monaco', monospace;
white-space: nowrap;
color: var(--secondary-color);
}
.status-item label,
.preview-status-item label {
display: flex;
align-items: center;
gap: 4px;
cursor: pointer;
font-family: inherit;
font-size: inherit;
color: var(--secondary-color);
}
.status-item input[type="checkbox"],
.preview-status-item input[type="checkbox"] {
width: 14px;
height: 14px;
cursor: pointer;
margin: 0;
}
.status-item span,
.preview-status-item span {
color: var(--secondary-color);
}
.editor-container textarea {
flex: 1;
height: 100%;
padding: 15px 10px;
border: none;
border-radius: 0;
background: var(--editor-bg);
color: var(--text-color);
resize: none;
tab-size: 4;
-moz-tab-size: 4;
white-space: pre-wrap;
word-wrap: break-word;
overflow-wrap: break-word;
overflow-x: hidden;
width: 100%;
transition: all 0.3s ease;
}
.editor-container textarea:focus {
outline: none;
}
.editor-container.preview-mode {
grid-template-columns: 1fr 1fr;
}
.preview-container {
display: none;
height: 100%;
padding: 0; /* 移除padding */
border-radius: 8px;
border: 1px solid var(--border-color);
background: var(--editor-bg);
color: var(--text-color);
overflow: hidden; /* 改为hidden,防止整体滚动 */
position: relative;
-webkit-user-select: none; /* Safari */
-ms-user-select: none; /* IE 10+ */
user-select: none; /* Standard syntax */
}
/* 添加预览内容容器样式 */
.preview-content {
height: calc(100% - 25px); /* 减去状态栏高度 */
padding: 10px 12px;
overflow-y: auto;
overflow-x: hidden;
-webkit-user-select: text; /* Safari */
-ms-user-select: text; /* IE 10+ */
user-select: text; /* Standard syntax */
}
/* 预览区状态栏样式 */
.preview-status-bar {
position: absolute;
bottom: 0;
left: 0;
right: 0;
height: 25px;
background: var(--editor-bg);
border-top: 1px solid var(--border-color);
display: flex;
align-items: center;
padding: 0 10px;
font-size: 12px;
color: var(--secondary-color);
z-index: 1;
justify-content: space-between;
min-width: 100%;
white-space: nowrap;
}
.preview-status-left {
display: flex;
align-items: center;
gap: 8px;
flex-shrink: 0;
}
.preview-status-right {
display: flex;
align-items: center;
gap: 15px;
flex-shrink: 0;
margin-left: auto;
}
.preview-status-item {
display: flex;
align-items: center;
gap: 5px;
font-family: 'Consolas', 'Monaco', monospace;
white-space: nowrap;
flex-shrink: 0;
}
.preview-status-item label {
display: flex;
align-items: center;
gap: 4px;
cursor: pointer;
font-family: inherit;
font-size: inherit;
color: var(--secondary-color);
}
.preview-status-item input[type="checkbox"] {
width: 14px;
height: 14px;
cursor: pointer;
margin: 0;
}
.preview-status-item span {
color: var(--secondary-color);
}
.fullscreen-toggle {
background: none;
border: none;
color: var(--text-color);
cursor: pointer;
padding: 2px 8px;
border-radius: 4px;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.2s ease;
font-size: 12px;
}
.fullscreen-toggle:hover {
background: var(--hover-color);
}
/* Markdown 内容样式 */
.preview-container > *:first-child {
margin-top: 0;
}
.preview-container > *:last-child {
margin-bottom: 0;
}
.preview-container h1,
.preview-container h2,
.preview-container h3,
.preview-container h4,
.preview-container h5,
.preview-container h6 {
margin-top: 1.8em;
margin-bottom: 0.8em;
line-height: 1.2;
color: var(--text-color);
}
.preview-container h1:first-child,
.preview-container h2:first-child,
.preview-container h3:first-child,
.preview-container h4:first-child,
.preview-container h5:first-child,
.preview-container h6:first-child {
margin-top: 0;
}
.preview-container p {
margin: 1.2em 0;
line-height: 1.8;
}
.preview-container a {
display: inline;
color: var(--link-color);
text-decoration: none;
border-bottom: 1px solid transparent;
transition: all 0.2s ease;
padding: 2px 4px;
margin: 0 -4px;
border-radius: 4px;
text-align: left;
}
/* 处理单独一行的链接 */
.preview-container p > a:only-child {
display: inline-block;
text-align: left;
width: auto;
}
/* 确保链接在段落中的对齐方式 */
.preview-container p {
text-align: left;
}
.preview-container a:hover {
background: var(--hover-color);
color: var(--link-hover-color);
border-bottom-color: var(--link-hover-color);
}
.preview-container a:visited {
color: var(--link-visited-color);
}
.preview-container a:visited:hover {
background: var(--hover-color);
border-bottom-color: var(--link-visited-color);
}
.preview-container a[href^="http"]::after {
content: "↗";
display: inline;
margin-left: 2px;
font-size: 0.9em;
opacity: 0.6;
}
.preview-container ul,
.preview-container ol {
margin: 1em 0;
padding-left: 1.5em;
}
.preview-container li {
margin: 0.5em 0;
}
.preview-container blockquote {
margin: 1.2em 0;
padding: 1em 1.2em;
border-left: 4px solid var(--border-color);
background: var(--bg-color);
border-radius: 0 4px 4px 0;
display: flow-root;
width: fit-content;
max-width: 100%;
box-shadow: 0 2px 4px var(--shadow-color);
transition: all 0.2s ease;
}
.preview-container blockquote > *:first-child {
margin-top: 0;
}
.preview-container blockquote > *:last-child {
margin-bottom: 0;
}
.preview-container blockquote p {
margin: 0.8em 0;
line-height: 1.6;
}
.preview-container blockquote + blockquote {
margin-top: -0.5em;
}
/* 嵌套引用的样式 */
.preview-container blockquote blockquote {
margin: 0.8em 0;
border-left-color: var(--secondary-color);
background: var(--editor-bg);
box-shadow: none;
}
/* 移动端适配 */
@media (max-width: 768px) {
.preview-container blockquote {
padding: 0.8em 1em;
margin: 1em 0;
width: 100%;
}
}
/* 代码块基础样式 */
.preview-container pre {
background: var(--editor-bg);
padding: 1.2em 1em;
border-radius: 8px;
overflow: auto;
position: relative;
margin: 1.5em 0;
font-family: 'Consolas', 'Monaco', 'Courier New', monospace;
line-height: 1.5;
font-size: 0.95em;
border: 1px solid var(--border-color);
scrollbar-width: thin;
scrollbar-color: var(--secondary-color) transparent;
}
/* 行内代码样式 */
.preview-container code {
background: var(--editor-bg);
padding: 0.2em 0.4em;
margin: 0 0.2em;
border-radius: 4px;
font-size: 0.9em;
font-family: 'Consolas', 'Monaco', 'Courier New', monospace;
border: 1px solid var(--border-color);
}
/* 代码块中的代码样式 */
.preview-container pre code {
background: none;
padding: 0;
margin: 0;
font-size: 0.95em;
white-space: pre;
word-break: normal;
word-wrap: normal;
line-height: inherit;
tab-size: 2;
hyphens: none;
border: none;
}
/* 代码块滚动条样式 */
.preview-container pre::-webkit-scrollbar {
width: 6px;
height: 6px;
}
.preview-container pre::-webkit-scrollbar-track {
background: transparent;
}
.preview-container pre::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 3px;
border: 2px solid var(--editor-bg);
}
.preview-container pre::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
/* 代码块语言标签 */
.preview-container pre::before {
content: attr(data-language);
position: absolute;
top: 0.5em;
right: 0.5em;
font-size: 0.85em;
color: var(--secondary-color);
padding: 0.2em 0.5em;
border-radius: 3px;
background: var(--container-bg);
opacity: 0.8;
transition: opacity 0.2s ease;
}
.preview-container pre:hover::before {
opacity: 1;
}
/* 代码高亮主题 - 浅色模式 */
.hljs {
color: #383a42;
background: var(--editor-bg);
}
.hljs-comment,
.hljs-quote {
color: #a0a1a7;
font-style: italic;
}
.hljs-doctag,
.hljs-keyword,
.hljs-formula {
color: #a626a4;
}
.hljs-section,
.hljs-name,
.hljs-selector-tag,
.hljs-deletion,
.hljs-subst {
color: #e45649;
}
.hljs-literal {
color: #0184bb;
}
.hljs-string,
.hljs-regexp,
.hljs-addition,
.hljs-attribute,
.hljs-meta .hljs-string {
color: #50a14f;
}
.hljs-attr,
.hljs-variable,
.hljs-template-variable,
.hljs-type,
.hljs-selector-class,
.hljs-selector-attr,
.hljs-selector-pseudo,
.hljs-number {
color: #986801;
}
.hljs-symbol,
.hljs-bullet,
.hljs-link,
.hljs-meta,
.hljs-selector-id,
.hljs-title {
color: #4078f2;
}
.hljs-built_in,
.hljs-title.class_,
.hljs-class .hljs-title {
color: #c18401;
}
.hljs-emphasis {
font-style: italic;
}
.hljs-strong {
font-weight: bold;
}
/* 代码高亮主题 - 深色模式 */
[data-theme="dark"] .hljs {
color: #abb2bf;
background: var(--editor-bg);
}
[data-theme="dark"] .hljs-comment,
[data-theme="dark"] .hljs-quote {
color: #7f848e;
font-style: italic;
}
[data-theme="dark"] .hljs-doctag,
[data-theme="dark"] .hljs-keyword,
[data-theme="dark"] .hljs-formula {
color: #c678dd;
}
[data-theme="dark"] .hljs-section,
[data-theme="dark"] .hljs-name,
[data-theme="dark"] .hljs-selector-tag,
[data-theme="dark"] .hljs-deletion,
[data-theme="dark"] .hljs-subst {
color: #e06c75;
}
[data-theme="dark"] .hljs-literal {
color: #56b6c2;
}
[data-theme="dark"] .hljs-string,
[data-theme="dark"] .hljs-regexp,
[data-theme="dark"] .hljs-addition,
[data-theme="dark"] .hljs-attribute,
[data-theme="dark"] .hljs-meta .hljs-string {
color: #98c379;
}
[data-theme="dark"] .hljs-attr,
[data-theme="dark"] .hljs-variable,
[data-theme="dark"] .hljs-template-variable,
[data-theme="dark"] .hljs-type,
[data-theme="dark"] .hljs-selector-class,
[data-theme="dark"] .hljs-selector-attr,
[data-theme="dark"] .hljs-selector-pseudo,
[data-theme="dark"] .hljs-number {
color: #d19a66;
}
[data-theme="dark"] .hljs-symbol,
[data-theme="dark"] .hljs-bullet,
[data-theme="dark"] .hljs-link,
[data-theme="dark"] .hljs-meta,
[data-theme="dark"] .hljs-selector-id,
[data-theme="dark"] .hljs-title {
color: #61afef;
}
[data-theme="dark"] .hljs-built_in,
[data-theme="dark"] .hljs-title.class_,
[data-theme="dark"] .hljs-class .hljs-title {
color: #e6c07b;
}
[data-theme="dark"] .hljs-emphasis {
font-style: italic;
}
[data-theme="dark"] .hljs-strong {
font-weight: bold;
}
/* 代码复制按钮 */
.copy-button {
position: absolute;
top: 0.5em;
right: 0.5em;
padding: 0.2em 0.5em;
font-size: 0.85em;
color: var(--text-color);
background: var(--container-bg);
border: 1px solid var(--border-color);
border-radius: 3px;
cursor: pointer;
opacity: 0;
transition: all 0.2s ease;
display: flex;
align-items: center;
gap: 4px;
}
.preview-container pre:hover .copy-button {
opacity: 0.8;
}
.copy-button:hover {
opacity: 1 !important;
background: var(--hover-color);
}
.copy-button.copied {
color: #4caf50;
border-color: #4caf50;
opacity: 1;
}
/* 移动端适配 */
@media (max-width: 768px) {
.preview-container pre {
padding: 1em 0.8em;
font-size: 0.9em;
}
.preview-container code {
font-size: 0.85em;
}
.preview-container pre::before {
opacity: 1;
}
}
/* 代码块中的链接样式 */
.preview-container pre a,
.preview-container code a {
border-bottom: none;
}
.preview-container pre a::after,
.preview-container code a::after {
display: none;
}
/* 图片链接样式 */
.preview-container a:has(img) {
border-bottom: none;
}
.preview-container a:has(img)::after {
display: none;
}
.preview-container hr {
margin: 2em 0;
border: none;
border-top: 1px solid var(--border-color);
}
.preview-container table {
width: 100%;
border-collapse: collapse;
margin: 1.5em 0;
overflow-x: auto;
display: block;
}
.preview-container th,
.preview-container td {
border: 1px solid var(--border-color);
padding: 8px 12px;
text-align: left;
}
.preview-container th {
background-color: var(--hover-color);
font-weight: 600;
}
.preview-container tr:nth-child(even) {
background-color: var(--editor-bg);
}
.preview-container tr:hover {
background-color: var(--hover-color);
}
.preview-container img {
max-width: 100%;
margin: 1em 0;
border-radius: 4px;
}
/* 滚动条样式 */
.editor-container textarea,
.preview-container,
.preview-content {
scrollbar-width: thin;
scrollbar-color: var(--secondary-color) transparent;
}
.editor-container textarea::-webkit-scrollbar,
.preview-container::-webkit-scrollbar,
.preview-content::-webkit-scrollbar {
width: 8px;
height: 8px;
}
.editor-container textarea::-webkit-scrollbar-track,
.preview-container::-webkit-scrollbar-track,
.preview-content::-webkit-scrollbar-track {
background: transparent;
}
.editor-container textarea::-webkit-scrollbar-thumb,
.preview-container::-webkit-scrollbar-thumb,
.preview-content::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 4px;
border: 2px solid var(--editor-bg);
}
.editor-container textarea::-webkit-scrollbar-thumb:hover,
.preview-container::-webkit-scrollbar-thumb:hover,
.preview-content::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
/* 移动端滚动条优化 */
@media (max-width: 768px) {
.editor-container textarea::-webkit-scrollbar,
.preview-container::-webkit-scrollbar,
.preview-content::-webkit-scrollbar {
width: 6px;
height: 6px;
}
.editor-container textarea::-webkit-scrollbar-thumb,
.preview-container::-webkit-scrollbar-thumb,
.preview-content::-webkit-scrollbar-thumb {
border-width: 1.5px;
}
}
/* 移动端响应式布局优化 */
@media (max-width: 768px) {
.container {
padding: 5px;
height: 100vh;
max-height: -webkit-fill-available;
display: flex;
flex-direction: column;
width: 100%;
}
.container.toolbar-hidden {
padding: 5px;
}
.editor-container {
flex: 1;
min-height: 0;
gap: 5px;
margin-bottom: env(safe-area-inset-bottom, 15px);
width: 100%;
display: flex;
flex-direction: column;
}
.editor-container.preview-mode {
display: flex;
flex-direction: column;
height: calc(100vh - 60px - env(safe-area-inset-bottom, 15px)); /* 减去工具栏和底部安全区域 */
}
.editor-wrapper,
.preview-container {
flex: 1;
min-height: 0;
overflow: auto;
-webkit-overflow-scrolling: touch;
width: 100%;
}
.editor-container.preview-mode .editor-wrapper,
.editor-container.preview-mode .preview-container {
flex: 1;
height: 0; /* 让flex:1生效 */
min-height: 0;
max-height: none;
width: 100%;
}
/* 处理键盘弹出时的布局 */
@supports (-webkit-touch-callout: none) {
.editor-container.preview-mode {
height: calc(100vh - 60px - env(safe-area-inset-bottom, 15px) - env(keyboard-inset-height, 0px));
}
}
/* 确保内容可滚动 */
.editor-container.preview-mode .editor-wrapper textarea,
.editor-container.preview-mode .preview-container {
height: 100%;
overflow-y: auto;
}
}
/* 处理超小屏幕设备 */
@media (max-width: 320px) {
.container {
padding: 3px 3px 12px 3px;
}
.container.toolbar-hidden {
padding: 8px;
}
.editor-wrapper textarea,
.line-numbers span {
font-size: 15px;
}
.preview-container {
font-size: 15px;
padding: 10px 10px 20px 10px;
}
}
/* 处理横屏模式 */
@media (max-height: 480px) and (orientation: landscape) {
.container {
padding: 5px;
width: 100%;
}
.container.toolbar-hidden {
padding: 5px;
}
.toolbar {
padding: 6px 10px;
margin-bottom: 8px;
width: 100%;
}
.editor-container {
height: calc(100vh - 80px);
width: 100%;
}
.editor-container.preview-mode {
grid-template-columns: 1fr 1fr; /* 横屏时恢复左右布局 */
gap: 8px;
width: 100%;
}
.editor-wrapper,
.preview-container {
width: 100%;
height: 100%;
box-sizing: border-box;
}
.editor-wrapper textarea,
.preview-container {
width: 100%;
height: 100%;
box-sizing: border-box;
}
.editor-container.preview-mode .editor-wrapper,
.editor-container.preview-mode .preview-container {
height: 100%;
width: 100%;
}
/* 横屏模式下状态栏优化 */
.status-bar,
.preview-status-bar {
height: 24px;
font-size: 11px;
padding: 0 6px;
width: 100%;
}
.status-bar .status-left,
.status-bar .status-right,
.preview-status-bar .preview-status-left,
.preview-status-bar .preview-status-right {
gap: 8px;
}
.status-bar .status-right,
.preview-status-bar .preview-status-right {
padding-left: 8px;
}
.status-item,
.preview-status-item {
margin-right: 8px;
}
.status-item label,
.preview-status-item label {
gap: 2px;
}
.status-item input[type="checkbox"],
.preview-status-item input[type="checkbox"] {
width: 12px;
height: 12px;
}
/* 调整状态栏右侧开关的间距 */
.status-right .status-item {
margin-right: 4px;
}
.status-right .switch-label,
.preview-status-right .switch-label {
font-size: 11px;
padding: 2px 4px;
}
}
/* 适配折叠屏设备 */
@media (max-width: 350px) and (min-height: 600px) {
.container {
padding: 4px;
}
.toolbar {
flex-direction: column;
align-items: stretch;
}
.toolbar-left,
.toolbar-right {
justify-content: center;
}
.editor-container {
height: calc(100vh - 140px);
}
}
/* 适配深色模式和高对比度显示 */
@media (prefers-contrast: high) {
:root {
--border-color: #666666;
--shadow-color: rgba(0, 0, 0, 0.3);
}
.editor-container textarea,
.preview-container {
border-width: 2px;
}
}
/* 适配强制颜色模式 */
@media (forced-colors: active) {
:root {
--border-color: CanvasText;
--text-color: CanvasText;
--bg-color: Canvas;
--primary-color: LinkText;
--secondary-color: GrayText;
--container-bg: Canvas;
--editor-bg: Canvas;
--hover-color: Highlight;
--link-color: LinkText;
--link-hover-color: LinkText;
--link-visited-color: VisitedText;
}
.editor-container textarea,
.preview-container,
.toolbar,
.status-bar,
.preview-status-bar {
border: 1px solid CanvasText;
}
.markdown-toolbar button,
.toolbar-select,
.fullscreen-toggle {
border: 1px solid CanvasText;
background: Canvas;
color: CanvasText;
}
.markdown-toolbar button:hover,
.toolbar-select:hover,
.fullscreen-toggle:hover {
background: Highlight;
color: HighlightText;
}
}
/* 减少动画以适应省电模式 */
@media (prefers-reduced-motion: reduce) {
* {
transition: none !important;
}
}
/* 工具栏样式 */
.markdown-toolbar {
display: flex;
gap: 4px;
align-items: center;
}
.markdown-toolbar button {
padding: 4px 8px;
background: var(--editor-bg);
border: 1px solid var(--border-color);
border-radius: 4px;
color: var(--text-color);
cursor: pointer;
font-size: 14px;
display: flex;
align-items: center;
justify-content: center;
min-width: 28px;
height: 28px;
transition: all 0.2s ease;
}
.markdown-toolbar button:hover {
background: var(--hover-color);
}
.toolbar-select {
padding: 4px 8px;
border: 1px solid var(--border-color);
border-radius: 4px;
background: var(--editor-bg);
color: var(--text-color);
font-size: 14px;
cursor: pointer;
}
.toolbar-select:hover {
background: var(--hover-color);
}
/* Emoji 选择器样式 */
.emoji-picker {
position: fixed; /* 改为fixed定位,避免滚动问题 */
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
background: var(--container-bg);
border: 1px solid var(--border-color);
border-radius: 8px;
box-shadow: 0 2px 8px var(--shadow-color);
padding: 15px;
display: none;
z-index: 1000;
max-height: 80vh;
width: 90%;
max-width: 400px;
overflow-y: auto;
-webkit-overflow-scrolling: touch;
}
.emoji-picker-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(40px, 1fr));
gap: 8px;
}
.emoji-picker button {
width: 40px;
height: 40px;
padding: 8px;
border: none;
background: none;
cursor: pointer;
border-radius: 8px;
transition: all 0.2s ease;
font-size: 20px;
display: flex;
align-items: center;
justify-content: center;
}
.emoji-picker button:hover {
background: var(--hover-color);
}
.emoji-picker-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.5);
z-index: 999;
display: none;
}
/* 全屏模式样式 */
.preview-container.fullscreen {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
width: 100vw;
height: 100vh;
z-index: 9999;
border-radius: 0;
border: none;
}
/* 真实全屏模式的样式 */
.preview-container:fullscreen {
background-color: var(--container-bg);
width: 100vw;
height: 100vh;
padding: 0;
margin: 0;
border: none;
border-radius: 0;
}
.preview-container:fullscreen .preview-content {
height: calc(100vh - 25px);
padding: 20px;
max-width: 1200px;
margin: 0 auto;
}
.preview-container:fullscreen .preview-status-bar {
position: fixed;
bottom: 0;
left: 0;
right: 0;
background: var(--editor-bg);
border-top: 1px solid var(--border-color);
}
/* 移动端全屏适配 */
@media (max-width: 768px) {
.preview-container:fullscreen .preview-content {
padding: 15px;
height: calc(100vh - 25px - env(safe-area-inset-bottom, 0px));
}
.preview-container:fullscreen .preview-status-bar {
padding-bottom: env(safe-area-inset-bottom, 0px);
}
}
.preview-container.fullscreen .preview-content {
height: calc(100vh - 25px);
padding: 20px;
}
/* 全屏切换按钮样式 */
.fullscreen-toggle {
background: none;
border: none;
color: var(--text-color);
cursor: pointer;
padding: 2px 8px;
border-radius: 4px;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.2s ease;
font-size: 12px;
}
.fullscreen-toggle:hover {
background: var(--hover-color);
}
.fullscreen-toggle .fullscreen-icon {
display: inline-flex;
align-items: center;
justify-content: center;
width: 16px;
height: 16px;
}
/* 预览区双击时禁止选中文本 */
.preview-container.fullscreen .preview-content {
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
/* 预览区恢复正常文本选择 */
.preview-container:not(.fullscreen) .preview-content {
-webkit-user-select: text;
-moz-user-select: text;
-ms-user-select: text;
user-select: text;
}
/* 全屏模式下的状态栏样式 */
.preview-container.fullscreen .preview-status-bar {
position: fixed;
bottom: env(safe-area-inset-bottom, 0);
left: 0;
right: 0;
background: var(--editor-bg);
border-top: 1px solid var(--border-color);
z-index: 10000;
}
/* 响应式布局 - 移动端优化 */
@media (max-width: 768px) {
.toolbar {
padding: 8px;
margin-bottom: 12px;
white-space: nowrap;
gap: 8px;
}
.toolbar-left,
.toolbar-right {
gap: 8px;
}
.markdown-toolbar button {
min-width: 32px;
height: 32px;
padding: 4px;
}
.toolbar-select {
padding: 4px;
font-size: 13px;
}
.note-name {
font-size: 0.9rem;
padding: 4px 8px;
}
.emoji-picker {
padding: 10px;
}
.emoji-picker button {
width: 36px;
height: 36px;
font-size: 18px;
}
}
/* 移动端状态栏适配 */
@media (max-width: 768px) {
.preview-status-bar {
padding: 0 8px;
overflow: hidden;
}
.preview-status-right {
gap: 8px;
}
.preview-status-item {
margin-left: 8px;
font-size: 11px;
}
}
/* 处理超小屏幕设备状态栏 */
@media (max-width: 320px) {
.preview-status-bar {
padding: 0 5px;
}
.preview-status-item {
margin-left: 5px;
font-size: 10px;
}
}
/* 全屏模式下的移动端优化 */
@media (max-width: 768px) {
.preview-container.fullscreen {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
width: 100vw;
height: 100vh;
height: -webkit-fill-available;
z-index: 9999;
border-radius: 0;
padding-bottom: env(safe-area-inset-bottom, 0);
}
.preview-container.fullscreen .preview-content {
height: calc(100vh - 25px - env(safe-area-inset-bottom, 0));
padding: 15px;
}
}
/* 处理横屏模式 */
@media (max-height: 480px) and (orientation: landscape) {
.preview-container.fullscreen .preview-status-bar {
height: 24px;
font-size: 11px;
padding: 0 6px;
}
.preview-container.fullscreen .preview-content {
height: calc(100vh - 24px);
padding: 10px;
}
}
/* 在现有样式的末尾添加密码相关样式 */
.password-dialog {
display: none;
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background: var(--container-bg);
border: 1px solid var(--border-color);
border-radius: 12px;
padding: 24px;
box-shadow: 0 8px 24px var(--shadow-color);
z-index: 1000;
width: 90%;
max-width: 360px;
transition: all 0.3s ease;
}
.password-dialog h3 {
margin: 0 0 20px 0;
color: var(--text-color);
font-size: 1.2em;
font-weight: 500;
display: flex;
align-items: center;
gap: 8px;
}
.password-dialog h3::before {
content: '🔒';
font-size: 1.1em;
}
.password-dialog input[type="password"] {
width: 100%;
padding: 10px 14px;
margin-bottom: 12px;
border: 1px solid var(--border-color);
border-radius: 8px;
background: var(--editor-bg);
color: var(--text-color);
font-size: 15px;
transition: all 0.2s ease;
}
.password-dialog input[type="password"]:hover {
border-color: var(--secondary-color);
}
.password-dialog input[type="password"]:focus {
outline: none;
border-color: var(--primary-color);
box-shadow: 0 0 0 2px var(--primary-color-alpha);
}
.password-dialog-message {
margin-bottom: 20px;
font-size: 0.95em;
color: #e74c3c;
min-height: 20px;
transition: all 0.3s ease;
opacity: 0;
display: flex;
align-items: center;
gap: 6px;
}
.password-dialog-message::before {
content: '⚠️';
font-size: 1.1em;
}
.password-dialog-message.success {
color: #2ecc71;
}
.password-dialog-message.success::before {
content: '✅';
}
.password-dialog-message.show {
opacity: 1;
}
.password-dialog-buttons {
display: flex;
justify-content: flex-end;
gap: 12px;
}
.password-dialog button {
padding: 8px 16px;
border: 1px solid var(--border-color);
border-radius: 8px;
background: var(--editor-bg);
color: var(--text-color);
cursor: pointer;
font-size: 14px;
font-weight: 500;
transition: all 0.2s ease;
display: flex;
align-items: center;
gap: 6px;
}
.password-dialog button:hover {
background: var(--hover-color);
border-color: var(--secondary-color);
}
.password-dialog button.primary {
background: var(--primary-color);
color: white;
border-color: var(--primary-color);
}
.password-dialog button.primary:hover {
opacity: 0.9;
transform: translateY(-1px);
}
.password-dialog button:active {
transform: translateY(1px);
}
.password-dialog-overlay {
display: none;
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.5);
backdrop-filter: blur(4px);
z-index: 999;
transition: all 0.3s ease;
}
.password-protected .editor-container,
.password-protected .toolbar {
filter: blur(8px);
pointer-events: none;
user-select: none;
}
.password-status {
display: flex;
align-items: center;
gap: 8px;
color: var(--text-color);
font-size: 0.9rem;
padding: 4px 8px;
border-radius: 6px;
cursor: pointer;
transition: all 0.2s ease;
}
.password-status:hover {
background: var(--hover-color);
}
.password-status-icon {
font-size: 1.2rem;
transition: transform 0.3s ease;
}
.password-status:hover .password-status-icon {
transform: scale(1.1);
}
@media (max-width: 768px) {
.password-dialog {
width: 90%;
padding: 20px;
/* 保持在屏幕中间 */
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
margin: 0;
}
.password-dialog h3 {
font-size: 1.1em;
}
.password-dialog input[type="password"] {
font-size: 14px;
padding: 8px 12px;
}
.password-dialog button {
padding: 7px 14px;
}
}
@media (max-width: 480px) {
.password-dialog {
width: 100%;
max-width: none;
border-radius: 12px 12px 0 0;
bottom: 0;
top: auto;
transform: translateX(-50%);
padding-bottom: calc(20px + env(safe-area-inset-bottom));
}
}
/* 工具栏右侧按钮样式 */
.toolbar-right {
display: flex;
align-items: center;
gap: 8px;
}
.toolbar-button {
display: flex;
align-items: center;
gap: 8px;
color: var(--text-color);
font-size: 0.9rem;
padding: 4px 8px;
border-radius: 6px;
cursor: pointer;
transition: all 0.2s ease;
border: none;
background: none;
}
.toolbar-button:hover {
background: var(--hover-color);
}
.toolbar-button .icon {
font-size: 1.2rem;
transition: transform 0.3s ease;
}
.toolbar-button:hover .icon {
transform: scale(1.1);
}
.toolbar-button .label {
display: none;
}
@media (min-width: 768px) {
.toolbar-button {
padding: 6px 12px;
}
.toolbar-button .label {
display: inline;
}
}
/* 主题切换按钮特定样式 */
.theme-toggle .sun-icon,
.theme-toggle .moon-icon {
display: none;
}
[data-theme="dark"] .moon-icon {
display: block;
}
[data-theme="light"] .sun-icon {
display: block;
}
/* 密码状态按钮特定样式 */
.password-status {
display: flex;
align-items: center;
gap: 8px;
color: var(--text-color);
font-size: 0.9rem;
padding: 4px 8px;
border-radius: 6px;
cursor: pointer;
transition: all 0.2s ease;
}
.password-status:hover {
background: var(--hover-color);
}
.password-status .icon {
font-size: 1.2rem;
transition: transform 0.3s ease;
}
.password-status:hover .icon {
transform: scale(1.1);
}
/* 添加 Toast 提示框样式 */
.toast-container {
position: fixed;
top: 20px;
right: 20px;
z-index: 10000;
display: flex;
flex-direction: column;
gap: 10px;
pointer-events: none;
/* 保持在右上角 */
top: 20px;
right: 20px;
width: auto;
max-width: calc(100% - 40px);
transform: none;
}
.toast {
background: var(--container-bg);
color: var(--text-color);
padding: 12px 24px;
border-radius: 8px;
box-shadow: 0 4px 12px var(--shadow-color);
font-size: 14px;
display: flex;
align-items: center;
gap: 8px;
opacity: 0;
transform: translateX(100%);
transition: all 0.3s ease;
border: 1px solid var(--border-color);
pointer-events: all;
max-width: 300px;
word-break: break-word;
}
.toast.show {
opacity: 1;
transform: translateX(0);
}
.toast.success {
border-left: 4px solid #4caf50;
}
.toast.error {
border-left: 4px solid #f44336;
}
.toast.info {
border-left: 4px solid #2196f3;
}
.toast.warning {
border-left: 4px solid #ff9800;
}
.toast-icon {
font-size: 18px;
flex-shrink: 0;
}
.toast-message {
flex: 1;
margin-right: 8px;
}
.toast-close {
cursor: pointer;
opacity: 0.7;
transition: opacity 0.2s ease;
padding: 4px;
margin: -4px;
border-radius: 4px;
}
.toast-close:hover {
opacity: 1;
background: var(--hover-color);
}
@media (max-width: 768px) {
.toast-container {
/* 保持在右上角 */
top: 20px;
right: 20px;
width: auto;
max-width: calc(100% - 40px);
transform: none;
}
.toast {
width: 100%;
transform: translateY(100%);
}
.toast.show {
transform: translateY(0);
}
}
/* 添加复制全部按钮样式 */
.copy-all-button {
background: none;
border: none;
color: var(--text-color);
cursor: pointer;
padding: 0px 4px;
border-radius: 4px;
font-size: 12px;
display: flex;
align-items: center;
gap: 4px;
transition: all 0.2s ease;
}
.copy-all-button:hover {
background: var(--hover-color);
}
/* 添加复制全部的动效样式 */
.copy-all-item {
cursor: pointer;
transition: all 0.2s ease;
padding: 0px 4px;
border-radius: 4px;
display: flex;
align-items: center;
gap: 4px;
}
.copy-all-item:hover {
background: var(--hover-color);
}
.copy-all-item .copy-icon {
transition: transform 0.2s ease;
}
.copy-all-item:hover .copy-icon {
transform: scale(1.1);
}
.copy-all-item.copied {
animation: copied-animation 0.5s ease;
}
@keyframes copied-animation {
0% { transform: scale(1); }
50% { transform: scale(1.1); }
100% { transform: scale(1); }
}
/* 表格样式 */
.content table {
width: 100%;
border-collapse: collapse;
margin: 1.5em 0;
overflow-x: auto;
display: block;
}
.content th,
.content td {
border: 1px solid var(--border-color);
padding: 8px 12px;
text-align: left;
}
.content th {
background-color: var(--hover-color);
font-weight: 600;
}
.content tr:nth-child(even) {
background-color: var(--editor-bg);
}
.content tr:hover {
background-color: var(--hover-color);
}
/* 移动端表格适配 */
@media (max-width: 768px) {
.content table {
font-size: 14px;
}
.content th,
.content td {
padding: 6px 8px;
}
}
/* 表格滚动条样式 */
.content table::-webkit-scrollbar {
height: 8px;
width: 8px;
}
.content table::-webkit-scrollbar-track {
background: transparent;
}
.content table::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 4px;
border: 2px solid var(--editor-bg);
}
.content table::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
/* 表格滚动条样式 */
.content table::-webkit-scrollbar {
height: 8px;
width: 8px;
}
.content table::-webkit-scrollbar-track {
background: transparent;
}
.content table::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 4px;
border: 2px solid var(--editor-bg);
}
.content table::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
/* 图片样式 */
.content img {
max-width: 100%;
height: auto;
margin: 1em 0;
border-radius: 8px;
display: block;
box-shadow: 0 2px 8px var(--shadow-color);
transition: all 0.3s ease;
opacity: 0;
animation: fadeIn 0.5s ease forwards;
}
/* 图片容器 */
.content p:has(img) {
text-align: center;
margin: 2em 0;
}
/* 图片悬停效果 */
.content img:hover {
transform: scale(1.01);
box-shadow: 0 4px 12px var(--shadow-color);
}
/* 图片标题样式 */
.content img + em {
display: block;
text-align: center;
color: var(--secondary-color);
font-size: 0.9em;
margin-top: 0.5em;
}
/* 移动端图片适配 */
@media (max-width: 768px) {
.content img {
border-radius: 6px;
margin: 0.8em 0;
}
.content p:has(img) {
margin: 1.5em 0;
}
/* 禁用移动端图片缩放动画 */
.content img:hover {
transform: none;
box-shadow: 0 2px 8px var(--shadow-color);
}
}
/* 图片加载动画 */
@keyframes fadeIn {
from {
opacity: 0;
transform: translateY(10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
/* 图片加载失败样式 */
.content img:not([src]),
.content img[src=""],
.content img[src*=""] {
position: relative;
min-height: 100px;
background: var(--editor-bg);
border: 1px dashed var(--border-color);
display: flex;
align-items: center;
justify-content: center;
}
.content img:not([src])::after,
.content img[src=""]::after,
.content img[src*=""]::after {
content: "图片加载失败";
position: absolute;
color: var(--secondary-color);
font-size: 0.9em;
}
/* 大图查看模式 */
.content img.enlarged {
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
max-width: 90vw;
max-height: 90vh;
object-fit: contain;
z-index: 1000;
cursor: zoom-out;
margin: 0;
padding: 0;
background: var(--bg-color);
box-shadow: 0 0 20px var(--shadow-color);
}
/* 大图查看遮罩层 */
.image-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.8);
z-index: 999;
display: none;
opacity: 0;
transition: opacity 0.3s ease;
}
.image-overlay.active {
display: block;
opacity: 1;
}
/* 代码块复制按钮样式 */
.content pre {
position: relative;
}
.copy-button {
position: absolute;
top: 0.5em;
right: 0.5em;
padding: 0.2em 0.5em;
font-size: 0.85em;
color: var(--text-color);
background: var(--container-bg);
border: 1px solid var(--border-color);
border-radius: 3px;
cursor: pointer;
opacity: 0;
transition: all 0.2s ease;
display: flex;
align-items: center;
gap: 4px;
z-index: 2;
}
.content pre:hover .copy-button {
opacity: 0.8;
}
.copy-button:hover {
opacity: 1 !important;
background: var(--hover-color);
transform: translateY(-1px);
}
.copy-button.copied {
color: #4caf50;
border-color: #4caf50;
opacity: 1;
}
.copy-button.error {
color: #f44336;
border-color: #f44336;
opacity: 1;
}
/* 代码块语言标签 */
.content pre::before {
content: attr(data-language);
position: absolute;
top: 0;
right: 0;
padding: 0.2em 0.5em;
font-size: 0.85em;
background: var(--container-bg);
border-bottom-left-radius: 4px;
color: var(--secondary-color);
opacity: 0.8;
transition: opacity 0.2s ease;
}
.content pre:hover::before {
opacity: 0;
}
@media (max-width: 768px) {
.copy-button {
padding: 0.15em 0.4em;
font-size: 0.8em;
}
.content pre::before {
font-size: 0.8em;
padding: 0.15em 0.4em;
}
}
.info-bar {
display: flex;
justify-content: space-between;
align-items: center;
padding: 15px;
background: var(--editor-bg);
border-radius: 12px;
margin-bottom: 20px;
font-size: 14px;
border: 1px solid var(--border-color);
overflow-x: auto;
-webkit-overflow-scrolling: touch;
scrollbar-width: none;
gap: 15px;
min-width: 0;
}
.info-bar::-webkit-scrollbar {
height: 6px;
width: 6px;
}
.info-bar::-webkit-scrollbar-track {
background: transparent;
}
.info-bar::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 3px;
}
.info-bar::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
.info-left {
display: flex;
gap: 15px;
flex-shrink: 0;
margin-right: auto;
}
.info-right {
display: flex;
align-items: center;
gap: 15px;
flex-shrink: 0;
margin-left: auto;
}
.info-item {
display: flex;
align-items: center;
gap: 8px;
padding: 4px 10px;
border-radius: 6px;
transition: all 0.2s ease;
white-space: nowrap;
flex-shrink: 0;
}
.info-item:hover {
background: var(--hover-color);
}
.theme-toggle {
display: flex;
align-items: center;
gap: 8px;
padding: 4px 10px;
border-radius: 6px;
border: none;
color: var(--text-color);
cursor: pointer;
font-size: 14px;
transition: all 0.2s ease;
background: transparent;
white-space: nowrap;
flex-shrink: 0;
}
.theme-toggle:hover {
background: var(--hover-color);
}
.theme-toggle:active {
transform: scale(0.95);
}
.theme-toggle .sun-icon,
.theme-toggle .moon-icon {
display: none;
font-size: 1.2rem;
}
.theme-toggle .label {
font-size: 14px;
}
[data-theme="dark"] .moon-icon {
display: block;
}
[data-theme="light"] .sun-icon {
display: block;
}
@media (max-width: 768px) {
.theme-toggle {
padding: 4px 8px;
}
.theme-toggle .label {
font-size: 13px;
}
}
@media (max-width: 768px) {
.info-bar {
padding: 10px;
gap: 10px;
}
.info-left,
.info-right {
gap: 10px;
}
.info-item {
padding: 4px 8px;
font-size: 13px;
}
}
/* 添加KaTeX相关样式 */
.katex-display {
overflow-x: auto;
overflow-y: hidden;
padding: 1em 0;
margin: 1em 0;
}
.katex-display::-webkit-scrollbar {
height: 6px;
}
.katex-display::-webkit-scrollbar-track {
background: transparent;
}
.katex-display::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 3px;
}
.katex-display::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
/* Mermaid图表样式 */
.mermaid {
margin: 1.5em 0;
text-align: center;
background: var(--editor-bg);
padding: 1em;
border-radius: 8px;
border: 1px solid var(--border-color);
overflow-x: auto;
}
.mermaid svg {
max-width: 100%;
height: auto;
}
/* 深色模式下的Mermaid样式 */
[data-theme="dark"] .mermaid {
--mermaid-bg: var(--editor-bg);
--mermaid-fg: var(--text-color);
--mermaid-edge: var(--text-color);
--mermaid-label: var(--text-color);
--mermaid-cluster: var(--border-color);
}
[data-theme="dark"] .mermaid .node rect,
[data-theme="dark"] .mermaid .node circle,
[data-theme="dark"] .mermaid .node ellipse,
[data-theme="dark"] .mermaid .node polygon,
[data-theme="dark"] .mermaid .node path {
fill: var(--mermaid-bg);
stroke: var(--mermaid-fg);
}
[data-theme="dark"] .mermaid .edgePath .path {
stroke: var(--mermaid-edge) !important;
}
[data-theme="dark"] .mermaid .edgeLabel {
color: var(--mermaid-label);
background-color: var(--mermaid-bg);
}
[data-theme="dark"] .mermaid .cluster rect {
fill: var(--mermaid-bg) !important;
stroke: var(--mermaid-cluster) !important;
}
[data-theme="dark"] .mermaid .label {
color: var(--mermaid-label);
}
[data-theme="dark"] .mermaid .node .label {
color: var(--mermaid-label);
}
[data-theme="dark"] .mermaid marker {
fill: var(--mermaid-edge);
}
</style>
<script src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/emoji-toolkit@7.0.0/lib/js/joypixels.min.js"></script>
<script src="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/highlight.min.js"></script>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/emoji-toolkit@7.0.0/extras/css/joypixels.min.css">
</head>
<body>
<div class="container">
<div class="toolbar">
<div class="toolbar-left">
<div class="note-name" onclick="copyNoteLink()" title="点击复制链接">
📋 ${noteName}
</div>
<div class="toolbar-divider"></div>
<!-- 字体和字号选择器 -->
<select id="font-family" class="toolbar-select" onchange="applyFont()">
<option value="default">默认字体</option>
<option value="serif">宋体</option>
<option value="yahei">微软雅黑</option>
<option value="kaiti">楷体</option>
<option value="heiti">黑体</option>
<option value="fangsong">仿宋</option>
<option value="songti">新宋体</option>
<option value="monospace">等宽字体</option>
<option value="arial">Arial</option>
<option value="times">Times New Roman</option>
<option value="helvetica">Helvetica</option>
</select>
<select id="font-size" class="toolbar-select" onchange="applyFontSize()">
<option value="12">12px</option>
<option value="13">13px</option>
<option value="14">14px</option>
<option value="15">15px</option>
<option value="16" selected>16px</option>
<option value="17">17px</option>
<option value="18">18px</option>
<option value="20">20px</option>
<option value="22">22px</option>
<option value="24">24px</option>
<option value="26">26px</option>
<option value="28">28px</option>
<option value="32">32px</option>
</select>
<div class="toolbar-divider"></div>
<!-- Markdown 工具栏 -->
<div class="markdown-toolbar">
<button onclick="applyMarkdown('bold')" title="粗体 Ctrl+B">B</button>
<button onclick="applyMarkdown('italic')" title="斜体 Ctrl+I">I</button>
<button onclick="applyMarkdown('heading')" title="标题 Ctrl+H">H</button>
<button onclick="applyMarkdown('strikethrough')" title="删除线 Ctrl+D">S</button>
<button onclick="applyMarkdown('list')" title="无序列表 Ctrl+U">•</button>
<button onclick="applyMarkdown('ordered-list')" title="有序列表 Ctrl+O">1.</button>
<button onclick="applyMarkdown('task')" title="任务列表 Ctrl+T">☐</button>
<button onclick="applyMarkdown('quote')" title="引用 Ctrl+Q">""</button>
<button onclick="applyMarkdown('code')" title="代码 Ctrl+K">{}</button>
<button onclick="applyMarkdown('table')" title="表格">⚏</button>
<button onclick="applyMarkdown('divider')" title="分割线">—</button>
<button onclick="applyMarkdown('link')" title="链接 Ctrl+L">🔗</button>
<button onclick="applyMarkdown('image')" title="图片 Ctrl+P">🖼</button>
<button onclick="showEmojiPicker()" title="表情">😊</button>
<div class="toolbar-divider"></div>
<button onclick="applyMarkdown('latex-inline')" title="行内公式">∑</button>
<button onclick="applyMarkdown('latex-block')" title="公式块">∫</button>
<button onclick="applyMarkdown('mermaid')" title="流程图">📊</button>
</div>
<span id="save-status"></span>
</div>
<div class="toolbar-right">
<div class="password-status toolbar-button" onclick="showPasswordDialog()" title="密码保护设置">
<span class="icon" id="password-status-icon">🔓</span>
<span class="label">密码保护</span>
</div>
<button onclick="toggleDarkMode()" class="toolbar-button theme-toggle" title="切换主题">
<div class="icon">
<span class="sun-icon">☀️</span>
<span class="moon-icon">🌙</span>
</div>
<span class="label">主题</span>
</button>
<div class="share-button toolbar-button" onclick="shareNote()" title="分享笔记">
<span class="icon">📤</span>
<span class="label">分享</span>
</div>
</div>
</div>
<div class="editor-container">
<div class="editor-wrapper">
<div class="editor-main">
<div class="line-numbers"></div>
<textarea id="content" placeholder="开始输入笔记内容..." onscroll="handleEditorScroll()">${noteContent}</textarea>
</div>
<div class="status-bar">
<div class="status-left">
<div class="status-item">
<label title="显示/隐藏行号">
<input type="checkbox" id="line-numbers-toggle" checked onchange="toggleLineNumbers()">
行号
</label>
</div>
<div class="status-item">
<label title="显示/隐藏工具栏">
<input type="checkbox" id="toolbar-toggle" checked onchange="toggleToolbar()">
工具栏
</label>
</div>
<div class="status-item">
<label class="switch-label">
<input type="checkbox" id="preview-toggle" onchange="togglePreview()">
预览
</label>
</div>
</div>
<div class="status-right">
<div class="status-item copy-all-item" onclick="copyAllContent()" title="复制全部内容">
<span class="copy-icon">📋</span>
<span>复制全部</span>
</div>
<div class="status-item">
<span>字数:</span>
<span id="char-count">0</span>
</div>
<div class="status-item">
<span>单词:</span>
<span id="word-count">0</span>
</div>
<div class="status-item">
<span>行:</span>
<span id="line-count">1</span>
</div>
<div class="status-item">
<span>列:</span>
<span id="column-count">1</span>
</div>
</div>
</div>
</div>
<div id="preview" class="preview-container">
<div class="preview-content" onscroll="handlePreviewScroll()" ondblclick="handlePreviewDoubleTap(event)"></div>
<div class="preview-status-bar">
<div class="preview-status-left">
<button onclick="togglePreviewFullscreen()" class="fullscreen-toggle" title="切换全屏">
<span class="fullscreen-icon" id="fullscreen-icon">⛶</span>
</button>
<div class="preview-status-item">
<label class="switch-label">
<input type="checkbox" id="sync-scroll-toggle" onchange="toggleSyncScroll()">
同步滚动
</label>
</div>
</div>
<div class="preview-status-right">
<div class="preview-status-item">
<span>字数:</span>
<span id="preview-char-count">0</span>
</div>
<div class="preview-status-item">
<span>单词:</span>
<span id="preview-word-count">0</span>
</div>
<div class="preview-status-item">
<span>段落:</span>
<span id="preview-paragraph-count">0</span>
</div>
</div>
</div>
</div>
</div>
</div>
<div id="emoji-picker" class="emoji-picker"></div>
<!-- 添加密码对话框 -->
<div class="password-dialog-overlay" id="password-overlay"></div>
<div class="password-dialog" id="password-dialog">
<h3 id="password-dialog-title">设置密码保护</h3>
<input type="password" id="password-input" placeholder="请输入密码" autocomplete="new-password">
<div class="password-dialog-message" id="password-message"></div>
<div class="password-dialog-buttons">
<button onclick="closePasswordDialog()">
<span>取消</span>
</button>
<button class="primary" onclick="handlePasswordAction()" id="password-action-btn">
<span>确定</span>
</button>
</div>
</div>
<!-- 添加 Toast 容器 -->
<div class="toast-container" id="toast-container"></div>
<div class="image-overlay" id="imageOverlay" onclick="closeEnlargedImage()"></div>
<script>
const content = document.getElementById('content');
const preview = document.getElementById('preview');
const previewToggle = document.getElementById('preview-toggle');
const syncScrollToggle = document.getElementById('sync-scroll-toggle');
const saveStatus = document.getElementById('save-status');
let saveTimeout;
let isEditorScrolling = false;
let isPreviewScrolling = false;
let isSyncScrollEnabled = true;
let isEmptyNote = false; // Flag to track if the note is empty and space was added
document.addEventListener('DOMContentLoaded', () => {
// 配置 marked
marked.setOptions({
gfm: true,
breaks: true,
tables: true,
headerIds: true,
mangle: false,
sanitize: false,
smartLists: true,
smartypants: true,
xhtml: false,
langPrefix: 'language-',
pedantic: false,
highlight: function(code, lang) {
if (lang && hljs.getLanguage(lang)) {
try {
return hljs.highlight(code, { language: lang }).value;
} catch (err) {}
}
try {
return hljs.highlightAuto(code).value;
} catch (err) {}
return code;
}
});
// 自定义 emoji 渲染
const renderer = new marked.Renderer();
const originalText = renderer.text.bind(renderer);
renderer.text = (text) => {
return joypixels.shortnameToImage(originalText(text));
};
marked.setOptions({ renderer });
// 初始化主题
const savedTheme = localStorage.getItem('theme') || 'light';
document.documentElement.setAttribute('data-theme', savedTheme);
// 初始化预览状态
const showPreview = localStorage.getItem('preview') === 'true';
previewToggle.checked = showPreview;
// 初始化同步滚动状态
const savedSyncScroll = localStorage.getItem('sync-scroll');
isSyncScrollEnabled = savedSyncScroll !== 'false';
syncScrollToggle.checked = isSyncScrollEnabled;
const editorContainer = document.querySelector('.editor-container');
editorContainer.classList.toggle('preview-mode', showPreview);
preview.style.display = showPreview ? 'block' : 'none';
if (showPreview) {
updatePreview(content.value);
}
content.addEventListener('input', () => {
updatePreview(content.value);
// If the content becomes empty, insert a space to trigger the save
if (content.value.trim().length === 0) {
if (!isEmptyNote) {
content.value = ' '; // Automatically add a space if content is empty
saveStatus.textContent = 'Note will be deleted'; // Notify the user that the note will be deleted
isEmptyNote = true; // Flag that space was added
}
} else {
// If content is no longer empty, remove the space and update status
if (isEmptyNote) {
content.value = content.value.trim(); // Remove the space if user starts typing again
isEmptyNote = false; // Reset the flag
}
saveStatus.textContent = 'Saving...'; // Show saving status while content is being edited
}
debounceSaveContent(content.value);
});
window.addEventListener('resize', () => {
if (previewToggle.checked) {
syncScrollPositions('editor');
}
});
// 初始化行号显示状态
const showLineNumbers = localStorage.getItem('show-line-numbers') !== 'false';
document.getElementById('line-numbers-toggle').checked = showLineNumbers;
const lineNumbers = document.querySelector('.line-numbers');
lineNumbers.classList.toggle('hidden', !showLineNumbers);
// 初始化行号和文本统计
updateLineNumbers();
updateTextStats();
// 初始化工具栏显示状态
const showToolbar = localStorage.getItem('show-toolbar') !== 'false';
document.getElementById('toolbar-toggle').checked = showToolbar;
const toolbar = document.querySelector('.toolbar');
const container = document.querySelector('.container');
toolbar.classList.toggle('hidden', !showToolbar);
container.classList.toggle('toolbar-hidden', !showToolbar);
// 恢复全屏状态
const savedFullscreen = localStorage.getItem('preview-fullscreen') === 'true';
if (savedFullscreen) {
togglePreviewFullscreen();
}
// 添加 Firefox 双击事件监听
const previewContent = document.querySelector('.preview-content');
if (previewContent) {
previewContent.addEventListener('mousedown', (e) => {
if (e.detail === 2) { // 检测双击
e.preventDefault(); // 阻止默认的文本选择
}
});
}
// 添加工具栏鼠标滚动支持
let isMouseDown = false;
let startX;
let scrollLeft;
toolbar.addEventListener('mousedown', (e) => {
isMouseDown = true;
toolbar.classList.add('dragging');
startX = e.pageX - toolbar.offsetLeft;
scrollLeft = toolbar.scrollLeft;
});
toolbar.addEventListener('mouseleave', () => {
isMouseDown = false;
toolbar.classList.remove('dragging');
});
toolbar.addEventListener('mouseup', () => {
isMouseDown = false;
toolbar.classList.remove('dragging');
});
toolbar.addEventListener('mousemove', (e) => {
if (!isMouseDown) return;
e.preventDefault();
const x = e.pageX - toolbar.offsetLeft;
const walk = (x - startX) * 2;
toolbar.scrollLeft = scrollLeft - walk;
});
// 支持鼠标滚轮横向滚动
toolbar.addEventListener('wheel', (e) => {
e.preventDefault();
toolbar.scrollLeft += e.deltaY;
});
});
function togglePreview() {
const showPreview = previewToggle.checked;
const editorContainer = document.querySelector('.editor-container');
editorContainer.classList.toggle('preview-mode', showPreview);
preview.style.display = showPreview ? 'block' : 'none';
localStorage.setItem('preview', showPreview);
if (showPreview) {
updatePreview(content.value);
setTimeout(() => syncScrollPositions('editor'), 100);
}
}
function toggleSyncScroll() {
isSyncScrollEnabled = syncScrollToggle.checked;
localStorage.setItem('sync-scroll', isSyncScrollEnabled);
}
function updatePreview(text) {
if (previewToggle.checked) {
const previewContent = preview.querySelector('.preview-content');
const scrollPos = previewContent.scrollTop;
// 渲染Markdown内容
previewContent.innerHTML = marked.parse(text);
// 渲染LaTeX公式
renderMathInElement(previewContent, {
delimiters: [
{left: '$$', right: '$$', display: true},
{left: '$', right: '$', display: false},
// 移除普通方括号作为数学公式的标记
// {left: '\\[', right: '\\]', display: true},
// 替换为更明确的数学公式标记
{left: 'math\\[', right: '\\]', display: true},
{left: '\\(', right: '\\)', display: false},
{left: '\\begin{align}', right: '\\end{align}', display: true}, // 添加对 align 环境的支持
],
throwOnError: false,
output: 'html'
});
// 渲染Mermaid图表
mermaid.initialize({
startOnLoad: false,
theme: document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'default',
securityLevel: 'loose',
fontFamily: 'var(--font-family)',
});
const mermaidDiagrams = previewContent.querySelectorAll('pre code.language-mermaid');
mermaidDiagrams.forEach(async (diagram, index) => {
try {
const pre = diagram.parentElement;
const mermaidDiv = document.createElement('div');
mermaidDiv.className = 'mermaid';
mermaidDiv.id = 'mermaid-' + Date.now() + '-' + index; // 添加唯一ID
mermaidDiv.textContent = diagram.textContent;
pre.parentNode.replaceChild(mermaidDiv, pre);
} catch (error) {
console.error('Mermaid渲染错误:', error);
}
});
// 等待所有图表渲染完成
if (mermaidDiagrams.length > 0) {
mermaid.run();
}
// 更新预览区统计信息
updatePreviewStats(text);
// 为所有代码块添加复制按钮和语言标签
const codeBlocks = previewContent.querySelectorAll('pre code');
codeBlocks.forEach(code => {
const pre = code.parentElement;
// 跳过Mermaid图表
if (code.classList.contains('language-mermaid')) return;
// 获取语言类名
const langClass = Array.from(code.classList).find(cl => cl.startsWith('language-'));
const language = langClass ? langClass.replace('language-', '') : '代码';
// 设置语言标签
pre.setAttribute('data-language', language);
// 添加复制按钮
if (!pre.querySelector('.copy-button')) {
const button = document.createElement('button');
button.className = 'copy-button';
button.innerHTML = '📋 复制';
button.onclick = (e) => {
e.preventDefault();
copyToClipboard(code.innerText, button);
};
pre.appendChild(button);
}
});
// 重新应用代码高亮
hljs.highlightAll();
previewContent.scrollTop = scrollPos;
}
}
async function copyToClipboard(text, button) {
try {
await navigator.clipboard.writeText(text);
const originalText = button.innerHTML;
button.innerHTML = '✅ 已复制';
button.classList.add('copied');
setTimeout(() => {
button.innerHTML = originalText;
button.classList.remove('copied');
}, 2000);
} catch (err) {
console.error('复制失败:', err);
button.innerHTML = '❌ 复制失败';
setTimeout(() => {
button.innerHTML = '📋 复制';
}, 2000);
}
}
function handleEditorScroll() {
if (!isPreviewScrolling && previewToggle.checked && isSyncScrollEnabled) {
isEditorScrolling = true;
syncScrollPositions('editor');
setTimeout(() => { isEditorScrolling = false; }, 50);
}
}
function handlePreviewScroll() {
if (!isEditorScrolling && previewToggle.checked && isSyncScrollEnabled) {
isPreviewScrolling = true;
const previewContent = preview.querySelector('.preview-content');
const editorHeight = content.scrollHeight - content.clientHeight;
const previewHeight = previewContent.scrollHeight - previewContent.clientHeight;
if (previewHeight > 0) {
const scrollPercentage = previewContent.scrollTop / previewHeight;
content.scrollTop = scrollPercentage * editorHeight;
}
setTimeout(() => { isPreviewScrolling = false; }, 50);
}
}
function syncScrollPositions(source) {
if (!previewToggle.checked || !isSyncScrollEnabled) return;
const editor = content;
const previewContent = preview.querySelector('.preview-content');
const editorHeight = editor.scrollHeight - editor.clientHeight;
const previewHeight = previewContent.scrollHeight - previewContent.clientHeight;
if (source === 'editor' && editorHeight > 0) {
const scrollPercentage = editor.scrollTop / editorHeight;
previewContent.scrollTop = scrollPercentage * previewHeight;
} else if (source === 'preview' && previewHeight > 0) {
const scrollPercentage = previewContent.scrollTop / previewHeight;
editor.scrollTop = scrollPercentage * editorHeight;
}
}
function debounceSaveContent(text) {
clearTimeout(saveTimeout);
saveTimeout = setTimeout(() => saveContent(text), 1000);
}
async function saveContent(text) {
try {
const response = await fetch(window.location.pathname, {
method: 'POST',
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
body: 'text=' + encodeURIComponent(text),
});
saveStatus.textContent = response.status === 204 ? '已保存' : '笔记将被删除';
} catch (error) {
saveStatus.textContent = '保存失败';
}
setTimeout(() => {
saveStatus.textContent = '';
}, 2000);
}
// 添加 Toast 提示框功能
function showToast(message, type = 'info', duration = 3000) {
const container = document.getElementById('toast-container');
const toast = document.createElement('div');
toast.className = 'toast ' + type;
// 设置图标
const icons = {
success: '✅',
error: '❌',
info: 'ℹ️',
warning: '⚠️'
};
toast.innerHTML =
'<span class="toast-icon">' + (icons[type] || icons.info) + '</span>' +
'<span class="toast-message">' + message + '</span>' +
'<span class="toast-close" onclick="this.parentElement.remove()">✕</span>';
container.appendChild(toast);
// 触发重排以启动动画
void toast.offsetWidth;
toast.classList.add('show');
// 自动关闭
setTimeout(() => {
toast.classList.remove('show');
setTimeout(() => toast.remove(), 300);
}, duration);
}
// 替换原有的 alert 调用
function copyNoteLink() {
const link = window.location.href;
navigator.clipboard.writeText(link).then(() => {
showToast('笔记链接已复制到剪贴板!', 'success');
}).catch(() => {
showToast('复制失败,请手动复制链接。', 'error');
});
}
function toggleDarkMode() {
const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? 'light' : 'dark';
document.documentElement.setAttribute('data-theme', theme);
localStorage.setItem('theme', theme);
// 更新Mermaid主题
mermaid.initialize({
startOnLoad: false,
theme: theme === 'dark' ? 'dark' : 'default',
securityLevel: 'loose',
fontFamily: 'var(--font-family)',
});
// 重新渲染预览内容以更新Mermaid图表
if (previewToggle.checked) {
updatePreview(content.value);
}
}
// Markdown 编辑功能
function applyMarkdown(type) {
const textarea = document.getElementById('content');
const start = textarea.selectionStart;
const end = textarea.selectionEnd;
const text = textarea.value;
let result;
switch(type) {
case 'bold':
result = insertAround(text, start, end, '**');
break;
case 'italic':
result = insertAround(text, start, end, '_');
break;
case 'heading':
result = insertAtLineStart(text, start, '# ');
break;
case 'strikethrough':
result = insertAround(text, start, end, '~~');
break;
case 'list':
result = insertAtLineStart(text, start, '- ');
break;
case 'ordered-list':
result = insertAtLineStart(text, start, '1. ');
break;
case 'task':
result = insertAtLineStart(text, start, '- [ ] ');
break;
case 'quote':
result = insertAtLineStart(text, start, '> ');
break;
case 'code':
result = insertAround(text, start, end, '\`\`\`\\n', '\\n\`\`\`');
break;
case 'table':
result = insertTable(text, start);
break;
case 'divider':
result = insertDivider(text, start);
break;
case 'link':
result = insertLink(text, start, end);
break;
case 'image':
result = insertImage(text, start, end);
break;
case 'latex-inline':
result = insertAround(text, start, end, '$');
break;
case 'latex-block':
result = insertAround(text, start, end, '$$\\n', '\\n$$');
break;
case 'mermaid':
result = insertMermaid(text, start);
break;
}
if (result) {
textarea.value = result.text;
textarea.selectionStart = result.selectionStart;
textarea.selectionEnd = result.selectionEnd;
textarea.focus();
updatePreview(textarea.value);
debounceSaveContent(textarea.value);
}
}
// 辅助函数:在选中文本周围插入标记
function insertAround(text, start, end, mark, endMark = mark) {
const selection = text.substring(start, end);
const before = text.substring(0, start);
const after = text.substring(end);
const newText = before + mark + selection + endMark + after;
return {
text: newText,
selectionStart: start + mark.length,
selectionEnd: end + mark.length
};
}
// 辅助函数:在行首插入标记
function insertAtLineStart(text, start, mark) {
const lines = text.split('\\n');
let currentPos = 0;
let targetLine = 0;
// 找到光标所在行
for (let i = 0; i < lines.length; i++) {
if (currentPos + lines[i].length >= start) {
targetLine = i;
break;
}
currentPos += lines[i].length + 1;
}
// 在目标行前添加标记
lines[targetLine] = mark + lines[targetLine];
return {
text: lines.join('\\n'),
selectionStart: currentPos + mark.length,
selectionEnd: currentPos + mark.length
};
}
// 插入表格
function insertTable(text, start) {
const tableTemplate = '\\n| 标题1 | 标题2 | 标题3 |\\n|--------|--------|--------|\\n| 内容1 | 内容2 | 内容3 |\\n';
const newText = text.substring(0, start) + tableTemplate + text.substring(start);
return {
text: newText,
selectionStart: start + tableTemplate.length,
selectionEnd: start + tableTemplate.length
};
}
// 添加插入Mermaid图表的函数
function insertMermaid(text, start) {
const mermaidTemplate = '\\n\`\`\`mermaid\\ngraph TD\\n A[开始] --> B[步骤1]\\n B --> C[步骤2]\\n C --> D[结束]\\n\`\`\`\\n';
const newText = text.substring(0, start) + mermaidTemplate + text.substring(start);
return {
text: newText,
selectionStart: start + mermaidTemplate.length,
selectionEnd: start + mermaidTemplate.length
};
}
// 插入链接
function insertLink(text, start, end) {
const selection = text.substring(start, end).trim();
const link = selection || '链接文字';
const template = '[' + link + '](https://)';
const newText = text.substring(0, start) + template + text.substring(end);
return {
text: newText,
selectionStart: start + link.length + 3,
selectionEnd: start + template.length - 1
};
}
// 插入图片
function insertImage(text, start, end) {
const template = '';
return {
text: text.substring(0, start) + template + text.substring(end),
selectionStart: start + 2,
selectionEnd: start + 6
};
}
// 插入分割线
function insertDivider(text, start) {
const divider = '\\n---\\n';
const newText = text.substring(0, start) + divider + text.substring(start);
return {
text: newText,
selectionStart: start + divider.length,
selectionEnd: start + divider.length
};
}
// 字体设置
function applyFont() {
const select = document.getElementById('font-family');
const textarea = document.getElementById('content');
const fontMap = {
'default': '',
'serif': 'SimSun, serif',
'yahei': '"Microsoft YaHei", "微软雅黑", sans-serif',
'kaiti': 'KaiTi, "楷体", serif',
'heiti': 'SimHei, "黑体", sans-serif',
'fangsong': 'FangSong, "仿宋", serif',
'songti': 'NSimSun, "新宋体", serif',
'monospace': 'Monaco, Consolas, monospace',
'arial': 'Arial, sans-serif',
'times': '"Times New Roman", Times, serif',
'helvetica': 'Helvetica, Arial, sans-serif'
};
textarea.style.fontFamily = fontMap[select.value] || '';
}
// 字号设置
function applyFontSize() {
const select = document.getElementById('font-size');
const textarea = document.getElementById('content');
textarea.style.fontSize = select.value + 'px';
}
// Emoji 选择器
function showEmojiPicker() {
const picker = document.getElementById('emoji-picker');
const overlay = document.querySelector('.emoji-picker-overlay') || createOverlay();
if (picker.style.display === 'block') {
picker.style.display = 'none';
overlay.style.display = 'none';
return;
}
// 如果是第一次显示,初始化表情列表
if (!picker.children.length) {
const gridContainer = document.createElement('div');
gridContainer.className = 'emoji-picker-grid';
const emojis = ['😀', '😃', '😄', '😁', '😅', '😂', '🤣', '😊',
'😇', '🙂', '🙃', '😉', '😌', '😍', '🥰', '😘',
'😗', '😙', '😚', '😋', '😛', '😝', '😜', '🤪',
'🤨', '🧐', '🤓', '😎', '🤩', '🥳', '😏', '😒',
'❤️', '🙌', '👍', '🎉', '✨', '🔥', '💡', '⭐',
'💪', '🎯', '✅', '❌', '💬', '👀', '🎨', '🎮',
'🎵', '🎬', '📚', '💻', '🔍', '⚡', '🌈', '🍀'];
emojis.forEach(emoji => {
const button = document.createElement('button');
button.textContent = emoji;
button.onclick = () => {
insertEmoji(emoji);
picker.style.display = 'none';
overlay.style.display = 'none';
};
gridContainer.appendChild(button);
});
picker.appendChild(gridContainer);
}
overlay.style.display = 'block';
picker.style.display = 'block';
}
// 创建遮罩层
function createOverlay() {
const overlay = document.createElement('div');
overlay.className = 'emoji-picker-overlay';
document.body.appendChild(overlay);
overlay.addEventListener('click', () => {
const picker = document.getElementById('emoji-picker');
picker.style.display = 'none';
overlay.style.display = 'none';
});
return overlay;
}
// 插入表情
function insertEmoji(emoji) {
const textarea = document.getElementById('content');
const start = textarea.selectionStart;
const end = textarea.selectionEnd;
const text = textarea.value;
textarea.value = text.substring(0, start) + emoji + text.substring(end);
textarea.selectionStart = textarea.selectionEnd = start + emoji.length;
textarea.focus();
updatePreview(textarea.value);
debounceSaveContent(textarea.value);
}
// 添加快捷键支持
document.addEventListener('keydown', function(e) {
if (!e.ctrlKey) return;
const shortcuts = {
'b': 'bold',
'i': 'italic',
'h': 'heading',
'd': 'strikethrough',
'u': 'list',
'o': 'ordered-list',
't': 'task',
'q': 'quote',
'k': 'code',
'l': 'link',
'p': 'image'
};
if (shortcuts[e.key.toLowerCase()]) {
e.preventDefault();
applyMarkdown(shortcuts[e.key.toLowerCase()]);
}
});
// 更新行号和调整布局
function updateLineNumbers() {
const textarea = document.getElementById('content');
const lineNumbers = document.querySelector('.line-numbers');
const fullText = textarea.value;
const totalLines = fullText.endsWith('\\n') ?
fullText.split('\\n').length :
fullText.split('\\n').length;
// 清空现有行号
lineNumbers.innerHTML = '';
// 为每一行创建行号 span 元素
for (let i = 0; i < totalLines; i++) {
const span = document.createElement('span');
span.textContent = i + 1;
lineNumbers.appendChild(span);
}
// 确保至少有一行
if (totalLines === 0) {
const span = document.createElement('span');
span.textContent = '1';
lineNumbers.appendChild(span);
}
// 调整行号区域宽度
const maxLineNumber = totalLines || 1;
const minWidth = 28; // 最小宽度
const digitWidth = 8; // 每个数字的估计宽度
const newWidth = Math.max(minWidth, String(maxLineNumber).length * digitWidth + 8);
lineNumbers.style.minWidth = newWidth + 'px';
}
// 监听窗口大小变化
window.addEventListener('resize', () => {
updateLineNumbers();
});
// 更新文本统计
function updateTextStats() {
const textarea = document.getElementById('content');
const text = textarea.value;
const position = textarea.selectionStart;
// 计算字符数(不包括空格和换行)
const charCount = text.replace(/\s/g, '').length;
// 计算单词数
const wordCount = text.trim().split(/\s+/).filter(word => word.length > 0).length;
// 计算当前行和列
const textBeforeCursor = text.substring(0, position);
const lines = textBeforeCursor.split('\\n');
const currentLine = lines.length;
const currentLineContent = lines[lines.length - 1];
const currentColumn = currentLineContent ? currentLineContent.length + 1 : 1;
// 更新显示(使用更紧凑的格式)
document.getElementById('char-count').textContent = String(charCount);
document.getElementById('word-count').textContent = String(wordCount);
document.getElementById('line-count').textContent = String(currentLine);
document.getElementById('column-count').textContent = String(currentColumn);
}
// 同步滚动行号
function syncLineNumbersScroll() {
const textarea = document.getElementById('content');
const lineNumbers = document.querySelector('.line-numbers');
// 计算最大可滚动高度
const maxScroll = textarea.scrollHeight - textarea.clientHeight;
const currentScroll = textarea.scrollTop;
// 确保不会滚动过头
if (currentScroll <= maxScroll) {
lineNumbers.scrollTop = currentScroll;
}
}
// 监听输入事件
content.addEventListener('input', () => {
updatePreview(content.value);
updateLineNumbers();
updateTextStats();
debounceSaveContent(content.value);
// 输入时也同步滚动
syncLineNumbersScroll();
});
// 监听光标位置变化
content.addEventListener('keyup', updateTextStats);
content.addEventListener('click', updateTextStats);
content.addEventListener('scroll', syncLineNumbersScroll);
// 切换行号显示/隐藏
function toggleLineNumbers() {
const lineNumbers = document.querySelector('.line-numbers');
const isVisible = document.getElementById('line-numbers-toggle').checked;
lineNumbers.classList.toggle('hidden', !isVisible);
localStorage.setItem('show-line-numbers', isVisible);
}
// 切换工具栏显示/隐藏
function toggleToolbar() {
const isVisible = document.getElementById('toolbar-toggle').checked;
const toolbar = document.querySelector('.toolbar');
const container = document.querySelector('.container');
toolbar.classList.toggle('hidden', !isVisible);
container.classList.toggle('toolbar-hidden', !isVisible);
localStorage.setItem('show-toolbar', isVisible);
// 触发一次窗口大小变化事件,以更新编辑器布局
window.dispatchEvent(new Event('resize'));
}
// 添加预览区统计功能
function updatePreviewStats(text) {
// 计算字符数(不包括空格和换行)
const charCount = text.replace(/\\s/g, '').length;
// 计算单词数
const wordCount = text.trim().split(/\\s+/).filter(word => word.length > 0).length;
// 计算段落数(通过空行分隔)
const paragraphCount = text.split(/\\n\\s*\\n/).filter(para => para.trim().length > 0).length;
// 更新显示
document.getElementById('preview-char-count').textContent = String(charCount);
document.getElementById('preview-word-count').textContent = String(wordCount);
document.getElementById('preview-paragraph-count').textContent = String(paragraphCount);
}
// 添加全屏切换功能
function togglePreviewFullscreen() {
const preview = document.getElementById('preview');
const fullscreenIcon = document.getElementById('fullscreen-icon');
const isFullscreen = preview.classList.contains('fullscreen');
if (isFullscreen) {
// 如果已经是容器全屏,切换到真实全屏
if (document.fullscreenElement) {
document.exitFullscreen();
} else {
preview.requestFullscreen();
}
} else {
// 首次点击进入容器全屏
preview.classList.add('fullscreen');
fullscreenIcon.textContent = '⛶';
}
// 触发resize事件以更新布局
window.dispatchEvent(new Event('resize'));
// 保存全屏状态到本地存储
localStorage.setItem('preview-fullscreen', preview.classList.contains('fullscreen'));
}
// ESC键退出全屏
function handleFullscreenEsc(e) {
if (e.key === 'Escape') {
const preview = document.getElementById('preview');
if (preview.classList.contains('fullscreen')) {
preview.classList.remove('fullscreen');
const fullscreenIcon = document.getElementById('fullscreen-icon');
fullscreenIcon.textContent = '⛶';
localStorage.setItem('preview-fullscreen', false);
}
}
}
// 监听浏览器全屏变化事件
document.addEventListener('fullscreenchange', () => {
const preview = document.getElementById('preview');
const fullscreenIcon = document.getElementById('fullscreen-icon');
if (!document.fullscreenElement && preview.classList.contains('fullscreen')) {
// 从真实全屏退出时,也退出容器全屏
preview.classList.remove('fullscreen');
fullscreenIcon.textContent = '⛶';
localStorage.setItem('preview-fullscreen', false);
}
});
// 双击进入/退出全屏(移动端支持)
function handlePreviewDoubleTap(e) {
const preview = document.getElementById('preview');
// 确保不是在状态栏上双击
if (!e.target.closest('.preview-status-bar')) {
e.preventDefault(); // 阻止默认行为
// 清除任何可能的文本选择
window.getSelection().removeAllRanges();
togglePreviewFullscreen();
}
}
// 添加密码相关功能
let isPasswordProtected = false;
let isPasswordVerified = false;
let currentPasswordAction = '';
async function checkPasswordProtection() {
try {
const response = await fetch(window.location.pathname + '/password-check', {
method: 'GET',
});
if (response.status === 200) {
isPasswordProtected = true;
document.body.classList.add('password-protected'); // 添加密码保护状态类
showPasswordVerification();
}
updatePasswordStatus();
} catch (error) {
console.error('检查密码保护状态失败:', error);
}
}
function showPasswordMessage(message, isSuccess = false) {
const messageEl = document.getElementById('password-message');
messageEl.textContent = message;
messageEl.classList.toggle('success', isSuccess);
messageEl.classList.add('show');
// 1秒后自动隐藏成功消息
if (isSuccess) {
setTimeout(() => {
messageEl.classList.remove('show');
}, 1000);
}
}
function clearPasswordMessage() {
const messageEl = document.getElementById('password-message');
messageEl.classList.remove('show', 'success');
messageEl.textContent = '';
}
function showPasswordDialog() {
const dialog = document.getElementById('password-dialog');
const overlay = document.getElementById('password-overlay');
const title = document.getElementById('password-dialog-title');
const actionBtn = document.getElementById('password-action-btn');
const passwordInput = document.getElementById('password-input');
clearPasswordMessage();
if (isPasswordProtected && isPasswordVerified) {
title.textContent = '移除密码保护';
actionBtn.textContent = '移除';
currentPasswordAction = 'remove';
} else if (!isPasswordProtected) {
title.textContent = '设置密码保护';
actionBtn.textContent = '设置';
currentPasswordAction = 'set';
}
dialog.style.display = 'block';
overlay.style.display = 'block';
passwordInput.focus();
// 添加回车键事件监听
passwordInput.onkeydown = (e) => {
if (e.key === 'Enter') {
e.preventDefault();
handlePasswordAction();
}
};
}
function showPasswordVerification() {
const dialog = document.getElementById('password-dialog');
const overlay = document.getElementById('password-overlay');
const title = document.getElementById('password-dialog-title');
const actionBtn = document.getElementById('password-action-btn');
const passwordInput = document.getElementById('password-input');
clearPasswordMessage();
title.textContent = '请输入密码';
actionBtn.textContent = '验证';
currentPasswordAction = 'verify';
dialog.style.display = 'block';
overlay.style.display = 'block';
passwordInput.focus();
// 添加回车键事件监听
passwordInput.onkeydown = (e) => {
if (e.key === 'Enter') {
e.preventDefault();
handlePasswordAction();
}
};
}
function closePasswordDialog() {
const dialog = document.getElementById('password-dialog');
const overlay = document.getElementById('password-overlay');
const passwordInput = document.getElementById('password-input');
dialog.style.display = 'none';
overlay.style.display = 'none';
passwordInput.value = '';
passwordInput.onkeydown = null; // 移除回车键事件监听
clearPasswordMessage();
}
async function handlePasswordAction() {
const password = document.getElementById('password-input').value;
if (!password) {
showPasswordMessage('请输入密码');
showToast('请输入密码', 'warning');
return;
}
try {
let response;
switch (currentPasswordAction) {
case 'set':
response = await fetch(window.location.pathname + '/password', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ password: await hashPassword(password) })
});
if (response.status === 200) {
isPasswordProtected = true;
isPasswordVerified = true;
document.body.classList.add('password-protected');
updatePasswordStatus();
showPasswordMessage('密码保护已设置', true);
showToast('密码保护已设置', 'success');
setTimeout(closePasswordDialog, 1500);
}
break;
case 'verify':
response = await fetch(window.location.pathname + '/password-verify', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ password: await hashPassword(password) })
});
if (response.status === 200) {
isPasswordVerified = true;
document.body.classList.remove('password-protected');
updatePasswordStatus();
showPasswordMessage('密码验证成功', true);
showToast('密码验证成功', 'success');
setTimeout(closePasswordDialog, 1500);
} else {
showPasswordMessage('密码错误');
showToast('密码错误', 'error');
return;
}
break;
case 'remove':
response = await fetch(window.location.pathname + '/password', {
method: 'DELETE',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ password: await hashPassword(password) })
});
if (response.status === 200) {
isPasswordProtected = false;
isPasswordVerified = false;
document.body.classList.remove('password-protected');
updatePasswordStatus();
showPasswordMessage('密码保护已移除', true);
showToast('密码保护已移除', 'success');
setTimeout(closePasswordDialog, 1500);
} else {
showPasswordMessage('密码错误');
showToast('密码错误', 'error');
return;
}
break;
}
} catch (error) {
console.error('密码操作失败:', error);
showPasswordMessage('操作失败,请重试');
showToast('操作失败,请重试', 'error');
}
}
async function hashPassword(password) {
const encoder = new TextEncoder();
const data = encoder.encode(password);
const hashBuffer = await crypto.subtle.digest('SHA-256', data);
const hashArray = Array.from(new Uint8Array(hashBuffer));
return hashArray.map(b => b.toString(16).padStart(2, '0')).join('');
}
function updatePasswordStatus() {
const icon = document.getElementById('password-status-icon');
const label = icon.nextElementSibling;
if (isPasswordProtected) {
icon.textContent = '🔒';
icon.title = '已启用密码保护';
label.textContent = '已加密';
} else {
icon.textContent = '🔓';
icon.title = '未启用密码保护';
label.textContent = '密码保护';
}
}
// 在页面加载时检查密码保护状态
document.addEventListener('DOMContentLoaded', async () => {
await checkPasswordProtection();
// ... existing DOMContentLoaded code ...
});
// 修改复制全部内容功能,添加动效
async function copyAllContent() {
const copyAllItem = document.querySelector('.copy-all-item');
const textarea = document.getElementById('content');
const text = textarea.value;
try {
await navigator.clipboard.writeText(text);
copyAllItem.classList.add('copied');
setTimeout(() => copyAllItem.classList.remove('copied'), 500);
showToast('已复制全部内容到剪贴板', 'success');
} catch (err) {
console.error('复制失败:', err);
showToast('复制失败,请重试', 'error');
}
}
async function shareNote() {
const preview = document.querySelector('.preview-content');
if (!preview) {
showToast('请先开启预览模式', 'warning');
return;
}
try {
// 生成分享ID
const shareId = Array.from({length: 8}, () => '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'[Math.floor(Math.random() * 62)]).join('');
// 准备分享数据
const shareData = {
content: preview.innerHTML,
createTime: new Date().toISOString(),
lastEditTime: new Date().toISOString(),
visitCount: 0
};
// 保存分享数据
const response = await fetch('/share/' + shareId, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(shareData)
});
if (response.ok) {
// 生成分享链接
const shareUrl = window.location.origin + '/share/' + shareId;
// 复制链接到剪贴板
await navigator.clipboard.writeText(shareUrl);
showToast('分享链接已复制到剪贴板', 'success');
} else {
showToast('分享失败,请重试', 'error');
}
} catch (error) {
console.error('分享失败:', error);
showToast('分享失败,请重试', 'error');
}
}
// 图片点击放大
document.addEventListener('DOMContentLoaded', () => {
const content = document.querySelector('.content');
const overlay = document.getElementById('imageOverlay');
content.addEventListener('click', (e) => {
if (e.target.tagName === 'IMG' && !e.target.classList.contains('enlarged')) {
e.target.classList.add('enlarged');
overlay.classList.add('active');
document.body.style.overflow = 'hidden';
}
});
});
// 关闭放大的图片
function closeEnlargedImage() {
const enlargedImage = document.querySelector('.enlarged');
const overlay = document.getElementById('imageOverlay');
if (enlargedImage) {
enlargedImage.classList.remove('enlarged');
overlay.classList.remove('active');
document.body.style.overflow = '';
}
}
// ESC键关闭放大图片
document.addEventListener('keydown', (e) => {
if (e.key === 'Escape') {
closeEnlargedImage();
}
});
// 图片加载错误处理
document.addEventListener('DOMContentLoaded', () => {
const images = document.querySelectorAll('.content img');
images.forEach(img => {
img.onerror = () => {
img.src = '';
};
});
});
</script>
</body>
</html>`;
}
/**
* 从KV存储获取笔记内容
* @param {string} notePath - 笔记路径
* @returns {Promise<string>} 笔记内容
*/
async function getNoteContent(notePath) {
const encryptedText = await NOTES_KV.get(notePath);
if (!encryptedText) return '';
const key = await generateEncryptionKey(notePath);
return await decryptText(encryptedText, key);
}
/**
* 保存笔记内容到KV存储
* @param {string} notePath - 笔记路径
* @param {string} text - 笔记内容
*/
async function saveNoteContent(notePath, text) {
const key = await generateEncryptionKey(notePath);
const encryptedText = await encryptText(text, key);
await NOTES_KV.put(notePath, encryptedText);
}
/**
* 从KV存储删除笔记
* @param {string} notePath - 要删除的笔记路径
*/
async function deleteNoteContent(notePath) {
await NOTES_KV.delete(notePath);
}
/**
* 处理分享页面请求
* @param {string} shareId - 分享ID
* @returns {Response} 响应对象
*/
async function handleShareRequest(shareId) {
// 获取分享数据
const shareKey = 'share_' + shareId;
const shareData = await NOTES_KV.get(shareKey);
if (!shareData) {
return new Response('分享内容不存在或已过期', { status: 404 });
}
try {
const data = JSON.parse(shareData);
// 更新访问次数
data.visitCount++;
await NOTES_KV.put(shareKey, JSON.stringify(data));
// 生成分享页面
const html = generateShareHTML(shareId, data);
return new Response(html, {
headers: { 'Content-Type': 'text/html;charset=utf-8' }
});
} catch (error) {
return new Response('加载分享内容失败', { status: 500 });
}
}
/**
* 生成分享页面HTML
* @param {string} shareId - 分享ID
* @param {Object} data - 分享数据
* @returns {string} HTML内容
*/
function generateShareHTML(shareId, data) {
const shareTime = new Date(data.createTime).toLocaleString('zh-CN');
return `<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>分享的笔记</title>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/styles/github.min.css">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/styles/github-dark.min.css" media="(prefers-color-scheme: dark)">
<!-- 添加KaTeX依赖 -->
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/katex@0.16.9/dist/katex.min.css">
<script src="https://cdn.jsdelivr.net/npm/katex@0.16.9/dist/katex.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/katex@0.16.9/dist/contrib/auto-render.min.js"></script>
<!-- 添加Mermaid依赖 -->
<script src="https://cdn.jsdelivr.net/npm/mermaid@10.6.1/dist/mermaid.min.js"></script>
<style>
:root {
--primary-color: #4e92d1;
--secondary-color: #6c757d;
--bg-color: #ffffff;
--text-color: #333333;
--border-color: #e0e0e0;
--container-bg: #ffffff;
--editor-bg: #f8f8f8;
--shadow-color: rgba(0, 0, 0, 0.1);
--hover-color: #f0f0f0;
--link-color: #0366d6;
--link-hover-color: #0969da;
--link-visited-color: #6f42c1;
}
[data-theme="dark"] {
--primary-color: #a2c2f5;
--secondary-color: #9ca3af;
--bg-color: #1a1a1a;
--text-color: #f1f1f1;
--border-color: #404040;
--container-bg: #2a2a2a;
--editor-bg: #333333;
--shadow-color: rgba(0, 0, 0, 0.3);
--hover-color: #3a3a3a;
--link-color: #58a6ff;
--link-hover-color: #79b8ff;
--link-visited-color: #bc8cff;
}
* {
box-sizing: border-box;
margin: 0;
padding: 0;
}
body {
margin: 0;
background: var(--bg-color);
color: var(--text-color);
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
font-size: 16px;
line-height: 1.6;
transition: all 0.3s ease;
}
.container {
width: 100%;
max-width: 100%;
margin: 0 auto;
padding: 20px;
min-height: 100vh;
background-color: var(--container-bg);
}
@media (min-width: 1200px) {
.container {
max-width: 95%;
box-shadow: 0 0 20px var(--shadow-color);
}
}
.info-bar {
display: flex;
justify-content: space-between;
align-items: center;
padding: 15px;
background: var(--editor-bg);
border-radius: 12px;
margin-bottom: 20px;
font-size: 14px;
border: 1px solid var(--border-color);
overflow-x: auto;
-webkit-overflow-scrolling: touch;
scrollbar-width: none;
gap: 15px;
min-width: 0;
}
.info-bar::-webkit-scrollbar {
height: 6px;
width: 6px;
}
.info-bar::-webkit-scrollbar-track {
background: transparent;
}
.info-bar::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 3px;
}
.info-bar::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
.info-left {
display: flex;
gap: 15px;
flex-shrink: 0;
margin-right: auto;
}
.info-right {
display: flex;
align-items: center;
gap: 15px;
flex-shrink: 0;
margin-left: auto;
}
.info-item {
display: flex;
align-items: center;
gap: 8px;
padding: 4px 10px;
border-radius: 6px;
transition: all 0.2s ease;
white-space: nowrap;
flex-shrink: 0;
}
.info-item:hover {
background: var(--hover-color);
}
.theme-toggle {
display: flex;
align-items: center;
gap: 8px;
padding: 4px 10px;
border-radius: 6px;
border: none;
color: var(--text-color);
cursor: pointer;
font-size: 14px;
transition: all 0.2s ease;
background: transparent;
white-space: nowrap;
flex-shrink: 0;
}
.theme-toggle:hover {
background: var(--hover-color);
}
.theme-toggle:active {
transform: scale(0.95);
}
.theme-toggle .sun-icon,
.theme-toggle .moon-icon {
display: none;
font-size: 1.2rem;
}
.theme-toggle .label {
font-size: 14px;
}
[data-theme="dark"] .moon-icon {
display: block;
}
[data-theme="light"] .sun-icon {
display: block;
}
@media (max-width: 768px) {
.theme-toggle {
padding: 4px 8px;
}
.theme-toggle .label {
font-size: 13px;
}
}
@media (max-width: 768px) {
.info-bar {
padding: 10px;
gap: 10px;
}
.info-left,
.info-right {
gap: 10px;
}
.info-item {
padding: 4px 8px;
font-size: 13px;
}
}
.content {
padding: 20px;
border: 1px solid var(--border-color);
border-radius: 12px;
background: var(--editor-bg);
overflow-x: auto;
}
/* Markdown 内容样式 */
.content > *:first-child {
margin-top: 0;
}
.content > *:last-child {
margin-bottom: 0;
}
.content h1,
.content h2,
.content h3,
.content h4,
.content h5,
.content h6 {
margin-top: 1.8em;
margin-bottom: 0.8em;
line-height: 1.2;
color: var(--text-color);
}
.content h1:first-child,
.content h2:first-child,
.content h3:first-child,
.content h4:first-child,
.content h5:first-child,
.content h6:first-child {
margin-top: 0;
}
.content p {
text-align: left !important;
margin: 1.2em 0;
line-height: 1.8;
}
.content p:has(> a:only-child) {
text-align: left !important;
}
.content a {
display: inline;
color: var(--link-color);
text-decoration: none;
border-bottom: 1px solid transparent;
transition: all 0.2s ease;
padding: 2px 4px;
margin: 0 -4px;
border-radius: 4px;
text-align: left !important;
}
/* 处理单独一行的链接 */
.content p > a:only-child {
display: inline-block;
text-align: left !important;
width: fit-content;
margin-left: 0;
}
/* 处理图片链接的特殊情况 */
.content p:has(img) {
text-align: center;
margin: 2em 0;
}
.content a:hover {
background: var(--hover-color);
color: var(--link-hover-color);
border-bottom-color: var(--link-hover-color);
}
.content a:visited {
color: var(--link-visited-color);
}
.content a:visited:hover {
background: var(--hover-color);
border-bottom-color: var(--link-visited-color);
}
.content a[href^="http"]::after {
content: "↗";
display: inline;
margin-left: 2px;
font-size: 0.9em;
opacity: 0.6;
}
.content ul,
.content ol {
margin: 1em 0;
padding-left: 1.5em;
}
.content li {
margin: 0.5em 0;
}
.content blockquote {
margin: 1.2em 0;
padding: 1em 1.2em;
border-left: 4px solid var(--border-color);
background: var(--bg-color);
border-radius: 0 4px 4px 0;
display: flow-root;
width: fit-content;
max-width: 100%;
box-shadow: 0 2px 4px var(--shadow-color);
transition: all 0.2s ease;
}
.content blockquote > *:first-child {
margin-top: 0;
}
.content blockquote > *:last-child {
margin-bottom: 0;
}
.content blockquote p {
margin: 0.8em 0;
line-height: 1.6;
}
.content blockquote + blockquote {
margin-top: -0.5em;
}
.content blockquote blockquote {
margin: 0.8em 0;
border-left-color: var(--secondary-color);
background: var(--editor-bg);
box-shadow: none;
}
.content pre {
background: var(--editor-bg);
padding: 1.2em 1em;
border-radius: 8px;
overflow: auto;
position: relative;
margin: 1.5em 0;
font-family: 'Consolas', 'Monaco', 'Courier New', monospace;
line-height: 1.5;
font-size: 0.95em;
border: 1px solid var(--border-color);
scrollbar-width: thin;
scrollbar-color: var(--secondary-color) transparent;
}
.content code {
background: var(--editor-bg);
padding: 0.2em 0.4em;
margin: 0 0.2em;
border-radius: 4px;
font-size: 0.9em;
font-family: 'Consolas', 'Monaco', 'Courier New', monospace;
border: 1px solid var(--border-color);
}
.content pre code {
background: none;
padding: 0;
margin: 0;
font-size: 0.95em;
white-space: pre;
word-break: normal;
word-wrap: normal;
line-height: inherit;
tab-size: 2;
hyphens: none;
border: none;
}
.content pre::-webkit-scrollbar {
width: 6px;
height: 6px;
}
.content pre::-webkit-scrollbar-track {
background: transparent;
}
.content pre::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 3px;
border: 2px solid var(--editor-bg);
}
.content pre::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
.content pre::before {
content: attr(data-language);
position: absolute;
top: 0.5em;
right: 0.5em;
font-size: 0.85em;
color: var(--secondary-color);
padding: 0.2em 0.5em;
border-radius: 3px;
background: var(--container-bg);
opacity: 0.8;
transition: opacity 0.2s ease;
}
.content pre:hover::before {
opacity: 1;
}
.copy-button {
position: absolute;
top: 0.5em;
right: 0.5em;
padding: 0.2em 0.5em;
font-size: 0.85em;
color: var(--text-color);
background: var(--container-bg);
border: 1px solid var(--border-color);
border-radius: 3px;
cursor: pointer;
opacity: 0;
transition: all 0.2s ease;
display: flex;
align-items: center;
gap: 4px;
}
.content pre:hover .copy-button {
opacity: 0.8;
}
.copy-button:hover {
opacity: 1 !important;
background: var(--hover-color);
}
.copy-button.copied {
color: #4caf50;
border-color: #4caf50;
opacity: 1;
}
/* Toast 提示框样式 */
.toast-container {
position: fixed;
top: 20px;
right: 20px;
z-index: 10000;
display: flex;
flex-direction: column;
gap: 10px;
pointer-events: none;
}
.toast {
background: var(--container-bg);
color: var(--text-color);
padding: 12px 24px;
border-radius: 8px;
box-shadow: 0 4px 12px var(--shadow-color);
font-size: 14px;
display: flex;
align-items: center;
gap: 8px;
opacity: 0;
transform: translateX(100%);
transition: all 0.3s ease;
border: 1px solid var(--border-color);
pointer-events: all;
max-width: 300px;
word-break: break-word;
}
.toast.show {
opacity: 1;
transform: translateX(0);
}
.toast.success {
border-left: 4px solid #4caf50;
}
.toast.error {
border-left: 4px solid #f44336;
}
.toast-icon {
font-size: 18px;
flex-shrink: 0;
}
.toast-message {
flex: 1;
margin-right: 8px;
}
.toast-close {
cursor: pointer;
opacity: 0.7;
transition: opacity 0.2s ease;
padding: 4px;
margin: -4px;
border-radius: 4px;
}
.toast-close:hover {
opacity: 1;
background: var(--hover-color);
}
/* 表格样式 */
.content table {
width: 100%;
border-collapse: collapse;
margin: 1.5em 0;
overflow-x: auto;
display: block;
}
.content th,
.content td {
border: 1px solid var(--border-color);
padding: 8px 12px;
text-align: left;
}
.content th {
background-color: var(--hover-color);
font-weight: 600;
}
.content tr:nth-child(even) {
background-color: var(--editor-bg);
}
.content tr:hover {
background-color: var(--hover-color);
}
/* 移动端表格适配 */
@media (max-width: 768px) {
.content table {
font-size: 14px;
}
.content th,
.content td {
padding: 6px 8px;
}
}
/* 表格滚动条样式 */
.content table::-webkit-scrollbar {
height: 8px;
width: 8px;
}
.content table::-webkit-scrollbar-track {
background: transparent;
}
.content table::-webkit-scrollbar-thumb {
background-color: var(--secondary-color);
border-radius: 4px;
border: 2px solid var(--editor-bg);
}
.content table::-webkit-scrollbar-thumb:hover {
background-color: var(--text-color);
}
/* 图片样式 */
.content img {
max-width: 100%;
height: auto;
margin: 1em 0;
border-radius: 8px;
display: block;
box-shadow: 0 2px 8px var(--shadow-color);
transition: all 0.3s ease;
opacity: 0;
animation: fadeIn 0.5s ease forwards;
}
/* 图片容器 */
.content p:has(img) {
text-align: center;
margin: 2em 0;
}
/* 图片悬停效果 */
.content img:hover {
transform: scale(1.01);
box-shadow: 0 4px 12px var(--shadow-color);
}
/* 图片标题样式 */
.content img + em {
display: block;
text-align: center;
color: var(--secondary-color);
font-size: 0.9em;
margin-top: 0.5em;
}
/* 移动端图片适配 */
@media (max-width: 768px) {
.content img {
border-radius: 6px;
margin: 0.8em 0;
}
.content p:has(img) {
margin: 1.5em 0;
}
/* 禁用移动端图片缩放动画 */
.content img:hover {
transform: none;
box-shadow: 0 2px 8px var(--shadow-color);
}
}
/* 图片加载动画 */
@keyframes fadeIn {
from {
opacity: 0;
transform: translateY(10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
/* 图片加载失败样式 */
.content img:not([src]),
.content img[src=""],
.content img[src*=""] {
position: relative;
min-height: 100px;
background: var(--editor-bg);
border: 1px dashed var(--border-color);
display: flex;
align-items: center;
justify-content: center;
}
.content img:not([src])::after,
.content img[src=""]::after,
.content img[src*=""]::after {
content: "图片加载失败";
position: absolute;
color: var(--secondary-color);
font-size: 0.9em;
}
/* 大图查看模式 */
.content img.enlarged {
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
max-width: 90vw;
max-height: 90vh;
object-fit: contain;
z-index: 1000;
cursor: zoom-out;
margin: 0;
padding: 0;
background: var(--bg-color);
box-shadow: 0 0 20px var(--shadow-color);
}
/* 大图查看遮罩层 */
.image-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.8);
z-index: 999;
display: none;
opacity: 0;
transition: opacity 0.3s ease;
}
.image-overlay.active {
display: block;
opacity: 1;
}
</style>
</head>
<body>
<div class="container">
<div class="info-bar">
<div class="info-left">
<div class="info-item">
<span>📝 笔记</span>
<span>${shareId}</span>
</div>
<div class="info-item">
<span>🕒 分享于</span>
<span>${shareTime}</span>
</div>
<div class="info-item">
<span>👀 访问</span>
<span>${data.visitCount}</span>
</div>
</div>
<div class="info-right">
<button onclick="toggleDarkMode()" class="theme-toggle" title="切换主题">
<span class="sun-icon">☀️</span>
<span class="moon-icon">🌙</span>
<span class="label">主题</span>
</button>
</div>
</div>
<div class="content">
${data.content}
</div>
</div>
<div class="image-overlay" id="imageOverlay" onclick="closeEnlargedImage()"></div>
<div class="toast-container" id="toast-container"></div>
<script src="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/highlight.min.js"></script>
<script>
// 初始化主题和代码块
document.addEventListener('DOMContentLoaded', () => {
const savedTheme = localStorage.getItem('theme') || 'light';
document.documentElement.setAttribute('data-theme', savedTheme);
// 为所有代码块添加复制按钮和语言标签
const codeBlocks = document.querySelectorAll('pre code');
codeBlocks.forEach(code => {
const pre = code.parentElement;
// 获取语言类名
const langClass = Array.from(code.classList).find(cl => cl.startsWith('language-'));
const language = langClass ? langClass.replace('language-', '') : '代码';
pre.setAttribute('data-language', language);
// 添加复制按钮
const button = document.createElement('button');
button.className = 'copy-button';
button.innerHTML = '<span>📋</span><span>复制</span>';
button.onclick = async (e) => {
e.preventDefault();
e.stopPropagation();
await copyCode(code, button);
};
pre.appendChild(button);
});
// 应用代码高亮
hljs.highlightAll();
// 图片点击放大
const content = document.querySelector('.content');
const overlay = document.getElementById('imageOverlay');
content.addEventListener('click', (e) => {
if (e.target.tagName === 'IMG' && !e.target.classList.contains('enlarged')) {
e.target.classList.add('enlarged');
overlay.classList.add('active');
document.body.style.overflow = 'hidden';
}
});
// 图片加载错误处理
const images = document.querySelectorAll('.content img');
images.forEach(img => {
img.onerror = () => {
img.src = '';
};
});
});
// 复制代码到剪贴板
async function copyCode(codeElement, button) {
const originalText = button.innerHTML;
const code = codeElement.innerText;
try {
await navigator.clipboard.writeText(code);
button.innerHTML = '<span>✅</span><span>已复制</span>';
button.classList.add('copied');
showToast('代码已复制到剪贴板', 'success');
} catch (err) {
console.error('复制失败:', err);
button.innerHTML = '<span>❌</span><span>复制失败</span>';
button.classList.add('error');
showToast('复制失败,请重试', 'error');
}
setTimeout(() => {
button.innerHTML = originalText;
button.classList.remove('copied', 'error');
}, 2000);
}
// 切换暗色模式
function toggleDarkMode() {
const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? 'light' : 'dark';
document.documentElement.setAttribute('data-theme', theme);
localStorage.setItem('theme', theme);
// 更新Mermaid主题
mermaid.initialize({
startOnLoad: false,
theme: theme === 'dark' ? 'dark' : 'default',
securityLevel: 'loose',
fontFamily: 'var(--font-family)',
});
// 重新渲染预览内容以更新Mermaid图表
if (previewToggle.checked) {
updatePreview(content.value);
}
}
// 关闭放大的图片
function closeEnlargedImage() {
const enlargedImage = document.querySelector('.enlarged');
const overlay = document.getElementById('imageOverlay');
if (enlargedImage) {
enlargedImage.classList.remove('enlarged');
overlay.classList.remove('active');
document.body.style.overflow = '';
}
}
// ESC键关闭放大图片
document.addEventListener('keydown', (e) => {
if (e.key === 'Escape') {
closeEnlargedImage();
}
});
// Toast 提示框
function showToast(message, type = 'info', duration = 3000) {
const container = document.getElementById('toast-container');
const toast = document.createElement('div');
toast.className = 'toast ' + type;
const icons = {
success: '✅',
error: '❌',
info: 'ℹ️',
warning: '⚠️'
};
toast.innerHTML =
'<span class="toast-icon">' + (icons[type] || icons.info) + '</span>' +
'<span class="toast-message">' + message + '</span>' +
'<span class="toast-close" onclick="this.parentElement.remove()">✕</span>';
container.appendChild(toast);
void toast.offsetWidth;
toast.classList.add('show');
setTimeout(() => {
toast.classList.remove('show');
setTimeout(() => toast.remove(), 300);
}, duration);
}
</script>
</body>
</html>`;
}
// 监听所有fetch请求
addEventListener('fetch', event => {
event.respondWith(handleRequest(event.request));
});