网页划词高亮工具

提供网页划词高亮功能

  1. // ==UserScript==
  2. // @name 网页划词高亮工具
  3. // @namespace http://tampermonkey.net/
  4. // @version 0.2.1
  5. // @description 提供网页划词高亮功能
  6. // @author sunny43
  7. // @license MIT
  8. // @match *://*/*
  9. // @grant GM_addStyle
  10. // @grant GM_setValue
  11. // @grant GM_getValue
  12. // @grant GM_registerMenuCommand
  13. // ==/UserScript==
  14.  
  15. (function () {
  16. 'use strict';
  17.  
  18. const STYLE_PREFIX = 'sunny43-';
  19.  
  20. // 全局变量
  21. let highlights = [];
  22. let currentPageUrl = window.location.href;
  23. let currentDomain = window.location.hostname;
  24. let settings = GM_getValue('highlight_settings', {
  25. colors: ['#ff909c', '#b89fff', '#74b4ff', '#70d382', '#ffcb7e'],
  26. activeColor: '#ff909c',
  27. minTextLength: 1,
  28. enableFuzzyMatch: true,
  29. maxContextDistance: 50,
  30. sidebarDescription: '高亮工具',
  31. sidebarWidth: 320,
  32. showFloatingButton: true
  33. });
  34. let savedRange = null; // 保存选区范围
  35. let ignoreNextClick = false; // 忽略下一次点击的标志
  36. let menuDisplayTimer = null; // 菜单显示定时器
  37. let menuOperationInProgress = false; // 添加菜单操作锁定
  38. // 启用列表
  39. let enabledList = GM_getValue('enabled_list', {
  40. domains: [],
  41. urls: []
  42. });
  43. // 检查当前页面是否启用高亮功能
  44. let isHighlightEnabled = enabledList.domains.includes(currentDomain) ||
  45. enabledList.urls.includes(currentPageUrl);
  46. let updateSidebarHighlights = null;
  47.  
  48. GM_addStyle(`
  49. /* 高亮菜单样式 */
  50. .${STYLE_PREFIX}highlight-menu {
  51. position: absolute;
  52. background: #333336;
  53. border: none;
  54. border-radius: 24px;
  55. box-shadow: 0 4px 16px rgba(0,0,0,0.3);
  56. padding: 10px 8px;
  57. z-index: 9999;
  58. display: flex;
  59. flex-direction: row;
  60. align-items: center;
  61. font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
  62. color: #fff;
  63. opacity: 0; /* 初始隐藏 */
  64. transition: opacity 0.2s ease-in;
  65. pointer-events: none; /* 隐藏时不响应事件 */
  66. }
  67. .${STYLE_PREFIX}highlight-menu.${STYLE_PREFIX}show {
  68. opacity: 1;
  69. pointer-events: auto; /* 显示时响应事件 */
  70. }
  71.  
  72. /* 菜单箭头样式 */
  73. .${STYLE_PREFIX}highlight-menu::after {
  74. content: '';
  75. position: absolute;
  76. bottom: -6px;
  77. left: var(--arrow-left, 50%);
  78. width: 12px;
  79. height: 6px;
  80. background-color: #333336;
  81. clip-path: polygon(0 0, 100% 0, 50% 100%);
  82. margin-left: -6px;
  83. }
  84. .${STYLE_PREFIX}highlight-menu.${STYLE_PREFIX}arrow-top::after {
  85. top: -6px;
  86. bottom: auto;
  87. clip-path: polygon(0 100%, 100% 100%, 50% 0);
  88. }
  89.  
  90. /* 颜色选择区域 */
  91. .${STYLE_PREFIX}highlight-menu-colors {
  92. display: flex;
  93. flex-direction: row;
  94. align-items: center;
  95. margin: 0 2px;
  96. flex-wrap: nowrap;
  97. flex: 0 0 auto;
  98. }
  99.  
  100. /* 颜色选择按钮 */
  101. .${STYLE_PREFIX}highlight-menu-color {
  102. width: 22px;
  103. height: 22px;
  104. border-radius: 50%;
  105. margin: 0 3px;
  106. cursor: pointer;
  107. position: relative;
  108. display: flex;
  109. align-items: center;
  110. justify-content: center;
  111. transition: transform 0.15s ease;
  112. box-shadow: inset 0 0 0 1px rgba(255,255,255,0.12);
  113. flex-shrink: 0;
  114. }
  115. .${STYLE_PREFIX}highlight-menu-color:hover {
  116. transform: scale(1.12);
  117. }
  118. .${STYLE_PREFIX}highlight-menu-color.${STYLE_PREFIX}active::after {
  119. content: "";
  120. width: 12px;
  121. height: 12px;
  122. background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 24 24' fill='none' stroke='%23333336' stroke-width='4' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='20 6 9 17 4 12'%3E%3C/polyline%3E%3C/svg%3E");
  123. background-repeat: no-repeat;
  124. background-position: center;
  125. background-size: contain;
  126. }
  127.  
  128. /* 菜单按钮通用样式 */
  129. .${STYLE_PREFIX}highlight-menu-action {
  130. height: 22px;
  131. margin: 0 2px;
  132. cursor: pointer;
  133. padding: 0 10px;
  134. border-radius: 12px;
  135. color: #fff;
  136. font-size: 13px;
  137. background: rgba(255,255,255,0.1);
  138. border: none;
  139. transition: all 0.15s ease;
  140. white-space: nowrap;
  141. display: flex;
  142. align-items: center;
  143. justify-content: center;
  144. flex-shrink: 0;
  145. }
  146. .${STYLE_PREFIX}highlight-menu-action:hover {
  147. background: rgba(255,255,255,0.2);
  148. }
  149.  
  150. /* 删除按钮样式 */
  151. .${STYLE_PREFIX}highlight-action-delete {
  152. color: #f0f0f0;
  153. font-weight: 500;
  154. position: relative;
  155. overflow: hidden;
  156. transition: all 0.25s cubic-bezier(0.2, 0.8, 0.2, 1);
  157. margin-left: 3px;
  158. }
  159. .${STYLE_PREFIX}highlight-action-delete:hover {
  160. background: rgba(255,82,82,0.12);
  161. color: #ff6b6b;
  162. transform: translateY(-1px);
  163. box-shadow: 0 2px 8px rgba(255,82,82,0.25);
  164. }
  165. .${STYLE_PREFIX}highlight-action-delete:active {
  166. transform: translateY(0px);
  167. background: rgba(255,82,82,0.2);
  168. }
  169.  
  170. /* 闪烁效果用于高亮跳转 */
  171. @keyframes ${STYLE_PREFIX}highlightFlash {
  172. 0%, 100% { opacity: 1; }
  173. 50% { opacity: 0.3; }
  174. }
  175. .${STYLE_PREFIX}highlight-flash {
  176. animation: ${STYLE_PREFIX}highlightFlash 0.5s ease 4;
  177. box-shadow: 0 0 0 3px rgba(255, 255, 0, 0.7) !important;
  178. position: relative;
  179. z-index: 10;
  180. }
  181.  
  182. /* 浮动按钮样式 */
  183. #${STYLE_PREFIX}floating-button {
  184. position: fixed;
  185. bottom: 20px;
  186. right: 20px;
  187. z-index: 10000;
  188. width: 38px;
  189. height: 38px;
  190. border: none;
  191. border-radius: 6px;
  192. cursor: pointer;
  193. background-color: #262A33;
  194. color: #E8E9EB;
  195. box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);
  196. transition: all 0.2s ease;
  197. display: flex;
  198. align-items: center;
  199. justify-content: center;
  200. }
  201. #${STYLE_PREFIX}floating-button:hover {
  202. background-color: #3BA5D8;
  203. box-shadow: 0 4px 12px rgba(59, 165, 216, 0.25);
  204. }
  205. #${STYLE_PREFIX}floating-button:active {
  206. transform: translateY(0px);
  207. }
  208. /* 侧边栏样式 */
  209. #${STYLE_PREFIX}sidebar {
  210. position: fixed;
  211. top: 0;
  212. right: -280px;
  213. width: 280px;
  214. height: 100%;
  215. background: linear-gradient(135deg, #262A33 0%, #1A1D24 100%);
  216. box-shadow: -1px 0 5px rgba(0, 0, 0, 0.15);
  217. transition: right 0.3s ease;
  218. z-index: 9999;
  219. font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
  220. }
  221. `);
  222.  
  223. // 保存启用列表
  224. function saveEnabledList() {
  225. GM_setValue('enabled_list', enabledList);
  226. // 刷新当前状态
  227. isHighlightEnabled = enabledList.domains.includes(currentDomain) ||
  228. enabledList.urls.includes(currentPageUrl);
  229.  
  230. // 更新浮动按钮显示状态
  231. const floatingButton = document.getElementById(`${STYLE_PREFIX}floating-button`);
  232. if (floatingButton) {
  233. floatingButton.style.display = (settings.showFloatingButton && isHighlightEnabled) ? 'flex' : 'none';
  234. }
  235. }
  236.  
  237. // 启用域名
  238. function enableDomain(domain) {
  239. if (!enabledList.domains.includes(domain)) {
  240. enabledList.domains.push(domain);
  241. saveEnabledList();
  242. }
  243. }
  244.  
  245. // 启用域名
  246. function disableDomain(domain) {
  247. enabledList.domains = enabledList.domains.filter(d => d !== domain);
  248. saveEnabledList();
  249. }
  250.  
  251. // 启用URL
  252. function enableUrl(url) {
  253. if (!enabledList.urls.includes(url)) {
  254. enabledList.urls.push(url);
  255. saveEnabledList();
  256. }
  257. }
  258.  
  259. // 启用URL
  260. function disableUrl(url) {
  261. enabledList.urls = enabledList.urls.filter(u => u !== url);
  262. saveEnabledList();
  263. }
  264.  
  265. // 加载当前页面的高亮
  266. function loadHighlights() {
  267. const allHighlights = GM_getValue('highlights', {});
  268. highlights = allHighlights[currentPageUrl] || [];
  269. return highlights;
  270. }
  271.  
  272. // 保存高亮到存储
  273. function saveHighlights() {
  274. const allHighlights = GM_getValue('highlights', {});
  275. allHighlights[currentPageUrl] = highlights;
  276. GM_setValue('highlights', allHighlights);
  277. }
  278.  
  279. // 保存设置
  280. function saveSettings() {
  281. GM_setValue('highlight_settings', settings);
  282. }
  283.  
  284. // 移除高亮菜单
  285. function removeHighlightMenu() {
  286. if (window.currentMenuCloseHandler) {
  287. document.removeEventListener('click', window.currentMenuCloseHandler);
  288. window.currentMenuCloseHandler = null;
  289. }
  290. const existingMenus = document.querySelectorAll(`.${STYLE_PREFIX}highlight-menu`);
  291. if (existingMenus.length) {
  292. existingMenus.forEach(menu => {
  293. menu.classList.remove(`${STYLE_PREFIX}show`);
  294. setTimeout(() => {
  295. if (menu && menu.parentNode) {
  296. menu.parentNode.removeChild(menu);
  297. }
  298. }, 200);
  299. });
  300. }
  301. clearTimeout(menuDisplayTimer);
  302. ignoreNextClick = false;
  303. menuOperationInProgress = false;
  304. }
  305.  
  306. // 高亮选中文本
  307. function highlightSelection(color) {
  308. if (!isHighlightEnabled) {
  309. return null;
  310. }
  311. const selection = window.getSelection();
  312. if (!selection.rangeCount) return null;
  313. const range = selection.getRangeAt(0);
  314. const selectedText = selection.toString().trim();
  315. if (!selectedText || selectedText.length < settings.minTextLength) {
  316. return null;
  317. }
  318. const highlightId = 'highlight-' + Date.now() + '-' + Math.floor(Math.random() * 10000);
  319. const highlightElement = document.createElement('span');
  320. highlightElement.className = `${STYLE_PREFIX}highlight-marked`;
  321. highlightElement.dataset.highlightId = highlightId;
  322. highlightElement.style.backgroundColor = color;
  323.  
  324. // ★ 先从未修改前的文本中提取上下文
  325. let prefix = '', suffix = '';
  326. if (range.startContainer.nodeType === Node.TEXT_NODE) {
  327. const originalText = range.startContainer.textContent;
  328. const startOffset = range.startOffset;
  329. const endOffset = startOffset + selectedText.length;
  330. prefix = extractValidContext(originalText, startOffset, 20, "backward");
  331. suffix = extractValidContext(originalText, endOffset, 20, "forward");
  332. }
  333.  
  334. try {
  335. // 再进行DOM操作前,提取上下文后才调用 extractContents
  336. const fragment = range.extractContents();
  337. highlightElement.appendChild(fragment);
  338. range.insertNode(highlightElement);
  339.  
  340. const highlight = {
  341. id: highlightId,
  342. text: selectedText,
  343. color: color,
  344. timestamp: Date.now(),
  345. url: currentPageUrl,
  346. prefix: prefix, // 前置上下文
  347. suffix: suffix // 后置上下文
  348. };
  349.  
  350. highlights.push(highlight);
  351. saveHighlights();
  352.  
  353. highlightElement.addEventListener('click', (e) => {
  354. e.preventDefault();
  355. e.stopPropagation();
  356. removeHighlightMenu();
  357. setTimeout(() => {
  358. showHighlightEditMenu(e, highlightId);
  359. }, 10);
  360. });
  361.  
  362. // 检查侧边栏是否打开,如果打开则刷新高亮列表
  363. const sidebar = document.getElementById(`${STYLE_PREFIX}sidebar`);
  364. if (sidebar && sidebar.style.right === '0px' && updateSidebarHighlights) {
  365. updateSidebarHighlights();
  366. }
  367.  
  368. selection.removeAllRanges();
  369. return highlightId;
  370. } catch (e) {
  371. console.warn('高亮失败:', e);
  372. try {
  373. findAndHighlight(selectedText, color, highlightId);
  374.  
  375. // 检查侧边栏是否打开,如果打开则刷新高亮列表
  376. const sidebar = document.getElementById(`${STYLE_PREFIX}sidebar`);
  377. if (sidebar && sidebar.style.right === '0px' && updateSidebarHighlights) {
  378. updateSidebarHighlights();
  379. }
  380.  
  381. return highlightId;
  382. } catch (error) {
  383. console.error('替代高亮方法也失败:', error);
  384. return null;
  385. }
  386. }
  387. }
  388.  
  389. // 根据ID删除高亮
  390. function removeHighlightById(highlightId) {
  391. const highlightElement = document.querySelector(`.${STYLE_PREFIX}highlight-marked[data-highlight-id="${highlightId}"]`);
  392. if (highlightElement) {
  393. const textNode = document.createTextNode(highlightElement.textContent);
  394. highlightElement.parentNode.replaceChild(textNode, highlightElement);
  395. }
  396. highlights = highlights.filter(h => h.id !== highlightId);
  397. saveHighlights();
  398.  
  399. // 检查侧边栏是否打开,如果打开则刷新高亮列表
  400. const sidebar = document.getElementById(`${STYLE_PREFIX}sidebar`);
  401. if (sidebar && sidebar.style.right === '0px' && updateSidebarHighlights) {
  402. updateSidebarHighlights();
  403. }
  404. }
  405.  
  406. // 使用 MutationObserver 监听 DOM 变化,动态恢复高亮
  407. function observeDomChanges() {
  408. let debounceTimer; // 新增变量用于防抖
  409. const observer = new MutationObserver((mutations) => {
  410. clearTimeout(debounceTimer);
  411. debounceTimer = setTimeout(() => {
  412. applyHighlights();
  413. }, 300);
  414. });
  415. observer.observe(document.body, {
  416. childList: true,
  417. subtree: true,
  418. characterData: true
  419. });
  420. }
  421.  
  422. // 更改高亮颜色
  423. function changeHighlightColor(highlightId, newColor) {
  424. const highlightElement = document.querySelector(`.${STYLE_PREFIX}highlight-marked[data-highlight-id="${highlightId}"]`);
  425. if (highlightElement) {
  426. highlightElement.style.backgroundColor = newColor;
  427. }
  428. const index = highlights.findIndex(h => h.id === highlightId);
  429. if (index !== -1) {
  430. highlights[index].color = newColor;
  431. saveHighlights();
  432. }
  433.  
  434. // 检查侧边栏是否打开,如果打开则刷新高亮列表
  435. const sidebar = document.getElementById(`${STYLE_PREFIX}sidebar`);
  436. if (sidebar && sidebar.style.right === '0px' && updateSidebarHighlights) {
  437. updateSidebarHighlights();
  438. }
  439. }
  440.  
  441. // 显示/隐藏侧边栏
  442. function toggleSidebar(forceShow = true) {
  443. const sidebar = document.getElementById(`${STYLE_PREFIX}sidebar`);
  444. const floatingButton = document.getElementById(`${STYLE_PREFIX}floating-button`);
  445. if (!sidebar) return;
  446.  
  447. if (forceShow) {
  448. sidebar.style.right = '0px';
  449. // 显示侧边栏时隐藏浮动按钮
  450. if (floatingButton) {
  451. floatingButton.style.display = 'none';
  452. }
  453. if (updateSidebarHighlights) {
  454. updateSidebarHighlights();
  455. }
  456. } else {
  457. const width = sidebar.style.width || '300px';
  458. const wasVisible = sidebar.style.right === '0px';
  459. sidebar.style.right = wasVisible ? `-${width}` : '0px';
  460.  
  461. // 更新浮动按钮显示状态
  462. if (floatingButton) {
  463. if (wasVisible) {
  464. // 关闭侧边栏时,根据设置和启用状态决定是否显示浮动按钮
  465. floatingButton.style.display = (settings.showFloatingButton && isHighlightEnabled) ? 'flex' : 'none';
  466. } else {
  467. // 打开侧边栏时,隐藏浮动按钮
  468. floatingButton.style.display = 'none';
  469. }
  470. }
  471.  
  472. if (sidebar.style.right === '0px' && updateSidebarHighlights) {
  473. updateSidebarHighlights();
  474. }
  475. }
  476. }
  477.  
  478. // 切换浮动按钮显示/隐藏
  479. function toggleFloatingButton() {
  480. const floatingButton = document.getElementById(`${STYLE_PREFIX}floating-button`);
  481. if (!floatingButton) return;
  482.  
  483. settings.showFloatingButton = !settings.showFloatingButton;
  484. // 即使设置为显示,在启用页面也不显示按钮
  485. floatingButton.style.display = (settings.showFloatingButton && isHighlightEnabled) ? 'flex' : 'none'; saveSettings();
  486. }
  487.  
  488. // 显示高亮编辑菜单
  489. function showHighlightEditMenu(event, highlightId) {
  490. if (!isHighlightEnabled) {
  491. return;
  492. }
  493. removeHighlightMenu();
  494. if (menuOperationInProgress) return;
  495. menuOperationInProgress = true;
  496. event.preventDefault();
  497. event.stopPropagation();
  498. ignoreNextClick = true;
  499. const highlight = highlights.find(h => h.id === highlightId);
  500. if (!highlight) {
  501. menuOperationInProgress = false;
  502. return;
  503. }
  504. const menu = createHighlightMenu(false);
  505. menu.dataset.currentHighlightId = highlightId;
  506. menu.querySelectorAll(`.${STYLE_PREFIX}highlight-menu-color`).forEach(colorBtn => {
  507. colorBtn.classList.remove(`${STYLE_PREFIX}active`);
  508. });
  509. const activeColorButton = menu.querySelector(`.${STYLE_PREFIX}highlight-menu-color[data-color="${highlight.color}"]`);
  510. if (activeColorButton) {
  511. activeColorButton.classList.add(`${STYLE_PREFIX}active`);
  512. }
  513. const menuHeight = 50;
  514. let menuTop = event.clientY + window.scrollY - menuHeight - 10;
  515. let showAbove = true;
  516. if (event.clientY < menuHeight + 10) {
  517. menuTop = event.clientY + window.scrollY + 10;
  518. showAbove = false;
  519. }
  520. menu.style.top = `${menuTop}px`;
  521. const menuWidth = menu.offsetWidth || 200;
  522. let menuLeft;
  523. if (event.clientX - (menuWidth / 2) < 5) {
  524. menuLeft = 5;
  525. } else if (event.clientX + (menuWidth / 2) > window.innerWidth - 5) {
  526. menuLeft = window.innerWidth - menuWidth - 5;
  527. } else {
  528. menuLeft = event.clientX - (menuWidth / 2);
  529. }
  530. menu.style.left = `${menuLeft}px`;
  531. const arrowLeft = event.clientX - menuLeft;
  532. const minArrowLeft = 12;
  533. const maxArrowLeft = menuWidth - 12;
  534. const safeArrowLeft = Math.max(minArrowLeft, Math.min(arrowLeft, maxArrowLeft));
  535. menu.style.setProperty('--arrow-left', `${safeArrowLeft}px`);
  536. if (!showAbove) {
  537. menu.classList.add(`${STYLE_PREFIX}arrow-top`);
  538. } else {
  539. menu.classList.remove(`${STYLE_PREFIX}arrow-top`);
  540. }
  541. requestAnimationFrame(() => {
  542. menu.classList.add(`${STYLE_PREFIX}show`);
  543. // 使用 once:true 来自动清理事件监听
  544. document.addEventListener('click', function closeMenu(e) {
  545. if (ignoreNextClick) {
  546. ignoreNextClick = false;
  547. return;
  548. }
  549. if (!menu.contains(e.target)) {
  550. removeHighlightMenu();
  551. }
  552. }, { once: true });
  553. setTimeout(() => {
  554. ignoreNextClick = false;
  555. menuOperationInProgress = false;
  556. }, 50);
  557. });
  558. }
  559.  
  560. // 查找并高亮文本
  561. function findAndHighlight(searchText, color, highlightId) {
  562. // 遍历所有文本节点查找匹配内容
  563. const treeWalker = document.createTreeWalker(
  564. document.body,
  565. NodeFilter.SHOW_TEXT,
  566. null
  567. );
  568. while (treeWalker.nextNode()) {
  569. const node = treeWalker.currentNode;
  570. const textContent = node.textContent;
  571. if (!textContent || textContent.trim().length === 0) continue;
  572. const idx = textContent.indexOf(searchText);
  573. if (idx !== -1) {
  574. const range = document.createRange();
  575. range.setStart(node, idx);
  576. range.setEnd(node, idx + searchText.length);
  577. const highlightElement = document.createElement('span');
  578. highlightElement.className = `${STYLE_PREFIX}highlight-marked`;
  579. highlightElement.dataset.highlightId = highlightId;
  580. highlightElement.style.backgroundColor = color;
  581. try {
  582. const fragment = range.extractContents();
  583. highlightElement.appendChild(fragment);
  584. range.insertNode(highlightElement);
  585. highlightElement.addEventListener('click', (e) => {
  586. e.preventDefault();
  587. e.stopPropagation();
  588. showHighlightEditMenu(e, highlightId);
  589. });
  590. // 新高亮直接返回 true
  591. return true;
  592. } catch (e) {
  593. console.warn('应用高亮失败:', e);
  594. }
  595. }
  596. }
  597. return false;
  598. }
  599.  
  600. // 应用页面上的所有高亮
  601. function applyHighlights() {
  602. // 按 timestamp 降序排序(从后向前恢复)
  603. highlights.sort((a, b) => (b.timestamp || 0) - (a.timestamp || 0));
  604. highlights.forEach(highlight => {
  605. const restored = advancedRestoreHighlight(highlight);
  606. if (!restored) {
  607. console.warn('多步恢复失败:', highlight.id);
  608. }
  609. });
  610. }
  611.  
  612. // 创建高亮菜单
  613. function createHighlightMenu(isNewHighlight = true) {
  614. removeHighlightMenu();
  615. ignoreNextClick = true;
  616. const menu = document.createElement('div');
  617. menu.className = `${STYLE_PREFIX}highlight-menu`;
  618. menu.innerHTML = `
  619. <div class="${STYLE_PREFIX}highlight-menu-colors">
  620. ${settings.colors.map(color => `
  621. <div class="${STYLE_PREFIX}highlight-menu-color"
  622. style="background-color: ${color};"
  623. data-color="${color}">
  624. </div>
  625. `).join('')}
  626. </div>
  627. `;
  628. // 无论如何先置空操作ID
  629. menu.dataset.currentHighlightId = '';
  630. document.body.appendChild(menu);
  631.  
  632. // 如果是新建高亮,确保所有颜色块没有激活状态
  633. if (isNewHighlight) {
  634. menu.querySelectorAll(`.${STYLE_PREFIX}highlight-menu-color`).forEach(el => {
  635. el.classList.remove(`${STYLE_PREFIX}active`);
  636. });
  637. }
  638.  
  639. menu.querySelectorAll(`.${STYLE_PREFIX}highlight-menu-color`).forEach(el => {
  640. el.addEventListener('click', (e) => {
  641. const color = el.dataset.color;
  642. const isActive = el.classList.contains(`${STYLE_PREFIX}active`);
  643. const currentHighlightId = menu.dataset.currentHighlightId;
  644. if (isActive) {
  645. if (currentHighlightId) {
  646. removeHighlightById(currentHighlightId);
  647. menu.dataset.currentHighlightId = '';
  648. menu.querySelectorAll(`.${STYLE_PREFIX}highlight-menu-color`)
  649. .forEach(colorEl => colorEl.classList.remove(`${STYLE_PREFIX}active`));
  650. const sel = window.getSelection();
  651. sel.removeAllRanges();
  652. if (savedRange) {
  653. sel.addRange(savedRange.cloneRange());
  654. }
  655. } else {
  656. window.getSelection().removeAllRanges();
  657. menu.querySelectorAll(`.${STYLE_PREFIX}highlight-menu-color`)
  658. .forEach(colorEl => colorEl.classList.remove(`${STYLE_PREFIX}active`));
  659. }
  660. removeHighlightMenu();
  661. } else {
  662. settings.activeColor = color;
  663. saveSettings();
  664. if (currentHighlightId) {
  665. changeHighlightColor(currentHighlightId, color);
  666. } else {
  667. const selection = window.getSelection();
  668. if (selection.toString().trim() === '' && savedRange) {
  669. selection.removeAllRanges();
  670. selection.addRange(savedRange.cloneRange());
  671. }
  672. const newHighlightId = highlightSelection(color);
  673. if (newHighlightId) {
  674. menu.dataset.currentHighlightId = newHighlightId;
  675. }
  676. }
  677. menu.querySelectorAll(`.${STYLE_PREFIX}highlight-menu-color`)
  678. .forEach(colorEl => colorEl.classList.toggle(`${STYLE_PREFIX}active`, colorEl.dataset.color === color));
  679. }
  680. e.stopPropagation();
  681. });
  682. });
  683. return menu;
  684. }
  685.  
  686. // 显示高亮菜单
  687. function showHighlightMenu() {
  688. if (!isHighlightEnabled) {
  689. return;
  690. }
  691. if (menuOperationInProgress) return;
  692. menuOperationInProgress = true;
  693. const selection = window.getSelection();
  694. const selectedText = selection.toString().trim();
  695. if (selectedText === '') {
  696. menuOperationInProgress = false;
  697. return;
  698. }
  699. const menu = createHighlightMenu(true);
  700. const range = selection.getRangeAt(0);
  701. const rects = range.getClientRects();
  702. if (rects.length === 0) {
  703. menuOperationInProgress = false;
  704. return;
  705. }
  706. const targetRect = rects[0];
  707. const menuHeight = 50;
  708. let initialTop = window.scrollY + targetRect.top - menuHeight - 8;
  709. let showAbove = true;
  710. if (targetRect.top < menuHeight + 10) {
  711. initialTop = window.scrollY + targetRect.bottom + 8;
  712. showAbove = false;
  713. }
  714. menu.style.top = `${initialTop}px`;
  715. setTimeout(() => {
  716. const menuWidth = menu.offsetWidth;
  717. const textCenterX = targetRect.left + (targetRect.width / 2);
  718. let menuLeft;
  719. if (textCenterX - (menuWidth / 2) < 5) {
  720. menuLeft = 5;
  721. } else if (textCenterX + (menuWidth / 2) > window.innerWidth - 5) {
  722. menuLeft = window.innerWidth - menuWidth - 5;
  723. } else {
  724. menuLeft = textCenterX - (menuWidth / 2);
  725. }
  726. menu.style.left = `${menuLeft}px`;
  727. menu.style.transform = 'none';
  728. const arrowLeft = textCenterX - menuLeft;
  729. const minArrowLeft = 12;
  730. const maxArrowLeft = menuWidth - 12;
  731. const safeArrowLeft = Math.max(minArrowLeft, Math.min(arrowLeft, maxArrowLeft));
  732. menu.style.setProperty('--arrow-left', `${safeArrowLeft}px`);
  733. if (!showAbove) {
  734. menu.classList.add(`${STYLE_PREFIX}arrow-top`);
  735. } else {
  736. menu.classList.remove(`${STYLE_PREFIX}arrow-top`);
  737. }
  738. requestAnimationFrame(() => {
  739. menu.classList.add(`${STYLE_PREFIX}show`);
  740. });
  741. }, 0);
  742. document.addEventListener('click', function closeMenu(e) {
  743. if (ignoreNextClick) {
  744. ignoreNextClick = false;
  745. return;
  746. }
  747. if (!menu.contains(e.target)) {
  748. removeHighlightMenu();
  749. }
  750. }, { once: true });
  751. setTimeout(() => {
  752. ignoreNextClick = false;
  753. menuOperationInProgress = false;
  754. }, 100);
  755. }
  756.  
  757. // 注册(不可用)事件
  758. function registerEvents() {
  759. document.addEventListener('mouseup', function (e) {
  760. if (!isHighlightEnabled) {
  761. return;
  762. }
  763. if (e.target.closest(`.${STYLE_PREFIX}highlight-menu`)) {
  764. return;
  765. }
  766. const selection = window.getSelection();
  767. const selectedText = selection.toString().trim();
  768. if (selectedText.length < (settings.minTextLength || 1)) {
  769. return;
  770. }
  771. if (selection.rangeCount > 0) {
  772. savedRange = selection.getRangeAt(0).cloneRange();
  773. }
  774. removeHighlightMenu();
  775. clearTimeout(menuDisplayTimer);
  776. ignoreNextClick = true;
  777. menuDisplayTimer = setTimeout(() => {
  778. showHighlightMenu();
  779. }, 10);
  780. });
  781. }
  782.  
  783. function fuzzyContextMatch(highlight) {
  784. // 如果未启用模糊匹配,则直接返回 false
  785. if (!settings.enableFuzzyMatch) return false;
  786.  
  787. if (!highlight.text) return false;
  788. const textNodes = document.createTreeWalker(
  789. document.body,
  790. NodeFilter.SHOW_TEXT,
  791. null
  792. );
  793. const pattern = highlight.text.trim();
  794.  
  795. // 预处理前缀和后缀
  796. const storedPrefix = highlight.prefix ? highlight.prefix.trim() : '';
  797. const storedSuffix = highlight.suffix ? highlight.suffix.trim() : '';
  798.  
  799. // 存储所有匹配项及其上下文评分
  800. const matches = [];
  801.  
  802. // 记录匹配的总数,以便特殊处理单一匹配的情况
  803. let matchCount = 0;
  804.  
  805. while (textNodes.nextNode()) {
  806. const node = textNodes.currentNode;
  807. const textContent = node.textContent;
  808. if (!textContent || textContent.trim().length === 0) continue;
  809.  
  810. // 查找所有可能的匹配位置
  811. let startIdx = 0;
  812. while (startIdx < textContent.length) {
  813. const idx = textContent.indexOf(pattern, startIdx);
  814. if (idx === -1) break;
  815.  
  816. matchCount++;
  817.  
  818. // 获取上下文
  819. const actualPrefix = textContent.substring(Math.max(0, idx - 40), idx).trim();
  820. const actualSuffix = textContent.substring(idx + pattern.length,
  821. idx + pattern.length + 40).trim();
  822.  
  823. // 计算上下文匹配分数
  824. let score = 0;
  825.  
  826. // 前缀匹配分数计算 (更宽松版)
  827. if (storedPrefix && actualPrefix) {
  828. if (actualPrefix.includes(storedPrefix)) {
  829. score += 10; // 前缀完全包含加10分
  830. } else {
  831. // 尝试寻找部分匹配
  832. for (let i = 1; i <= Math.min(storedPrefix.length, 12); i++) {
  833. const prefixEnd = storedPrefix.slice(-i);
  834. if (actualPrefix.endsWith(prefixEnd)) {
  835. score += i / 2; // 匹配前缀尾部,加分
  836. break;
  837. }
  838. }
  839.  
  840. // 尝试在前缀中查找关键片段
  841. if (storedPrefix.length > 6) {
  842. for (let i = 0; i < storedPrefix.length - 5; i++) {
  843. const fragment = storedPrefix.substring(i, i + 5);
  844. if (actualPrefix.includes(fragment)) {
  845. score += 2.5; // 找到显著片段加2.5分
  846. break;
  847. }
  848. }
  849. }
  850. }
  851. }
  852.  
  853. // 后缀匹配分数计算 (更宽松版)
  854. if (storedSuffix && actualSuffix) {
  855. if (actualSuffix.includes(storedSuffix)) {
  856. score += 10; // 后缀完全包含加10分
  857. } else {
  858. // 尝试寻找部分匹配
  859. for (let i = 1; i <= Math.min(storedSuffix.length, 12); i++) {
  860. const suffixStart = storedSuffix.slice(0, i);
  861. if (actualSuffix.startsWith(suffixStart)) {
  862. score += i / 2; // 匹配后缀头部,加分
  863. break;
  864. }
  865. }
  866.  
  867. // 尝试在后缀中查找关键片段
  868. if (storedSuffix.length > 6) {
  869. for (let i = 0; i < storedSuffix.length - 5; i++) {
  870. const fragment = storedSuffix.substring(i, i + 5);
  871. if (actualSuffix.includes(fragment)) {
  872. score += 2.5; // 找到显著片段加2.5分
  873. break;
  874. }
  875. }
  876. }
  877. }
  878. }
  879.  
  880. // 如果是单个字符的高亮,给予额外分数,避免完全相同的短文本被错过
  881. if (pattern.length <= 3 && (actualPrefix.includes(storedPrefix) || actualSuffix.includes(storedSuffix))) {
  882. score += 5;
  883. }
  884.  
  885. // 记录匹配项
  886. matches.push({
  887. node,
  888. idx,
  889. score,
  890. actualPrefix,
  891. actualSuffix
  892. });
  893.  
  894. startIdx = idx + 1; // 继续搜索下一个匹配
  895. }
  896. }
  897.  
  898. // 特殊情况:如果页面上只有一个匹配,直接使用它
  899. if (matchCount === 1 && matches.length === 1) {
  900. const match = matches[0];
  901. try {
  902. const range = document.createRange();
  903. range.setStart(match.node, match.idx);
  904. range.setEnd(match.node, match.idx + pattern.length);
  905.  
  906. const highlightElement = document.createElement('span');
  907. highlightElement.className = `${STYLE_PREFIX}highlight-marked`;
  908. highlightElement.dataset.highlightId = highlight.id;
  909. highlightElement.style.backgroundColor = highlight.color;
  910.  
  911. const fragment = range.extractContents();
  912. highlightElement.appendChild(fragment);
  913. range.insertNode(highlightElement);
  914.  
  915. highlightElement.addEventListener('click', (e) => {
  916. e.preventDefault();
  917. e.stopPropagation();
  918. showHighlightEditMenu(e, highlight.id);
  919. });
  920.  
  921. return true;
  922. } catch (e) {
  923. console.warn('唯一模糊匹配高亮失败:', e);
  924. }
  925. }
  926.  
  927. // 如果有多个匹配项,选择分数最高的
  928. if (matches.length > 0) {
  929. // 按分数降序排序
  930. matches.sort((a, b) => b.score - a.score);
  931. const bestMatch = matches[0];
  932.  
  933. // 降低得分阈值到2,使更多匹配可以被接受
  934. const threshold = matchCount > 1 ? 2 : 0.5;
  935.  
  936. if (bestMatch.score >= threshold) {
  937. // 构造高亮
  938. try {
  939. const range = document.createRange();
  940. range.setStart(bestMatch.node, bestMatch.idx);
  941. range.setEnd(bestMatch.node, bestMatch.idx + pattern.length);
  942.  
  943. const highlightElement = document.createElement('span');
  944. highlightElement.className = `${STYLE_PREFIX}highlight-marked`;
  945. highlightElement.dataset.highlightId = highlight.id;
  946. highlightElement.style.backgroundColor = highlight.color;
  947.  
  948. const fragment = range.extractContents();
  949. highlightElement.appendChild(fragment);
  950. range.insertNode(highlightElement);
  951.  
  952. highlightElement.addEventListener('click', (e) => {
  953. e.preventDefault();
  954. e.stopPropagation();
  955. showHighlightEditMenu(e, highlight.id);
  956. });
  957.  
  958. return true;
  959. } catch (e) {
  960. console.warn('模糊匹配插入高亮失败:', e);
  961. }
  962. } else {
  963. console.log('模糊匹配分数过低,尝试进一步降低要求:', {
  964. text: pattern,
  965. bestScore: bestMatch.score
  966. });
  967.  
  968. // 如果分数太低但至少有一个匹配,可以考虑用最后的回退机制
  969. if (matches.length > 0 && pattern.length <= 5) {
  970. // 对于短文本,如果我们有任何匹配并且上下文也有一些匹配,就使用它
  971. const bestMatch = matches[0];
  972. try {
  973. const range = document.createRange();
  974. range.setStart(bestMatch.node, bestMatch.idx);
  975. range.setEnd(bestMatch.node, bestMatch.idx + pattern.length);
  976.  
  977. const highlightElement = document.createElement('span');
  978. highlightElement.className = `${STYLE_PREFIX}highlight-marked`;
  979. highlightElement.dataset.highlightId = highlight.id;
  980. highlightElement.style.backgroundColor = highlight.color;
  981.  
  982. console.log('使用回退机制恢复短文本高亮:', pattern);
  983.  
  984. const fragment = range.extractContents();
  985. highlightElement.appendChild(fragment);
  986. range.insertNode(highlightElement);
  987.  
  988. highlightElement.addEventListener('click', (e) => {
  989. e.preventDefault();
  990. e.stopPropagation();
  991. showHighlightEditMenu(e, highlight.id);
  992. });
  993.  
  994. return true;
  995. } catch (e) {
  996. console.warn('回退机制高亮失败:', e);
  997. }
  998. }
  999. }
  1000. }
  1001.  
  1002. return false;
  1003. }
  1004.  
  1005. function advancedRestoreHighlight(highlight) {
  1006. // 检查是否已经有相同ID的高亮存在
  1007. const existingHighlight = document.querySelector(`.${STYLE_PREFIX}highlight-marked[data-highlight-id="${highlight.id}"]`);
  1008. if (existingHighlight) {
  1009. // 高亮已存在,不需要重复恢复
  1010. return true;
  1011. }
  1012.  
  1013. // 使用文本上下文匹配恢复高亮
  1014. const contextRestored = restoreHighlightUsingContext(highlight);
  1015. if (contextRestored) {
  1016. // 只保存最新一次恢复记录
  1017. highlight.recoveryHistory = {
  1018. timestamp: Date.now(),
  1019. method: 'contextMatch',
  1020. success: true
  1021. };
  1022. saveHighlights(); // 保存更新后的恢复历史
  1023. return true;
  1024. }
  1025.  
  1026. // 尝试模糊匹配
  1027. if (fuzzyContextMatch(highlight)) {
  1028. highlight.recoveryHistory = {
  1029. timestamp: Date.now(),
  1030. method: 'fuzzyContext',
  1031. success: true
  1032. };
  1033. saveHighlights();
  1034. return true;
  1035. }
  1036.  
  1037. // 记录失败状态
  1038. highlight.recoveryHistory = {
  1039. timestamp: Date.now(),
  1040. method: 'fallback',
  1041. success: false
  1042. };
  1043. saveHighlights();
  1044. return false;
  1045. }
  1046.  
  1047. function restoreHighlightUsingContext(highlight) {
  1048. // 遍历页面中所有文本节点
  1049. const treeWalker = document.createTreeWalker(
  1050. document.body,
  1051. NodeFilter.SHOW_TEXT,
  1052. null
  1053. );
  1054.  
  1055. // 如果存在前缀和后缀信息,预处理它们以便于比较
  1056. const storedPrefix = highlight.prefix ? highlight.prefix.trim() : '';
  1057. const storedSuffix = highlight.suffix ? highlight.suffix.trim() : '';
  1058.  
  1059. // 存储所有可能的匹配及其得分
  1060. const matches = [];
  1061.  
  1062. // 记录找到多少个完全相同的文本
  1063. let exactTextMatches = 0;
  1064.  
  1065. while (treeWalker.nextNode()) {
  1066. const node = treeWalker.currentNode;
  1067. const textContent = node.textContent;
  1068. if (!textContent || textContent.trim().length === 0) continue;
  1069.  
  1070. const idx = textContent.indexOf(highlight.text);
  1071. if (idx !== -1) {
  1072. // 记录找到了一个文本匹配
  1073. exactTextMatches++;
  1074.  
  1075. // 获取当前节点中匹配区域前后的上下文
  1076. const actualPrefix = textContent.substring(Math.max(0, idx - 30), idx).trim();
  1077. const actualSuffix = textContent.substring(idx + highlight.text.length,
  1078. idx + highlight.text.length + 30).trim();
  1079.  
  1080. // 计算上下文匹配得分
  1081. let score = 1; // 基础分:找到了文本
  1082.  
  1083. // 前缀匹配得分计算 (改进版)
  1084. if (storedPrefix && actualPrefix) {
  1085. if (actualPrefix.includes(storedPrefix)) {
  1086. score += 10; // 前缀完全包含加10分
  1087. } else if (storedPrefix.length > 3) {
  1088. // 尝试匹配前缀的尾部
  1089. for (let i = Math.min(storedPrefix.length, actualPrefix.length); i >= 3; i--) {
  1090. if (storedPrefix.slice(-i) === actualPrefix.slice(-i)) {
  1091. score += i / 2; // 匹配长度越长分数越高
  1092. break;
  1093. }
  1094. }
  1095.  
  1096. // 另外尝试寻找前缀中的部分匹配
  1097. if (storedPrefix.length >= 8) {
  1098. for (let i = 0; i < storedPrefix.length - 6; i++) {
  1099. const fragment = storedPrefix.substring(i, i + 6);
  1100. if (actualPrefix.includes(fragment)) {
  1101. score += 3; // 找到部分匹配加3分
  1102. break;
  1103. }
  1104. }
  1105. }
  1106. }
  1107. }
  1108.  
  1109. // 后缀匹配得分计算 (改进版)
  1110. if (storedSuffix && actualSuffix) {
  1111. if (actualSuffix.includes(storedSuffix)) {
  1112. score += 10; // 后缀完全包含加10分
  1113. } else if (storedSuffix.length > 3) {
  1114. // 尝试匹配后缀的开头
  1115. for (let i = Math.min(storedSuffix.length, actualSuffix.length); i >= 3; i--) {
  1116. if (storedSuffix.slice(0, i) === actualSuffix.slice(0, i)) {
  1117. score += i / 2; // 匹配长度越长分数越高
  1118. break;
  1119. }
  1120. }
  1121.  
  1122. // 另外尝试寻找后缀中的部分匹配
  1123. if (storedSuffix.length >= 8) {
  1124. for (let i = 0; i < storedSuffix.length - 6; i++) {
  1125. const fragment = storedSuffix.substring(i, i + 6);
  1126. if (actualSuffix.includes(fragment)) {
  1127. score += 3; // 找到部分匹配加3分
  1128. break;
  1129. }
  1130. }
  1131. }
  1132. }
  1133. }
  1134.  
  1135. matches.push({
  1136. node,
  1137. idx,
  1138. score,
  1139. actualPrefix,
  1140. actualSuffix
  1141. });
  1142. }
  1143. }
  1144.  
  1145. // 特殊情况处理:如果页面上只有一个文本匹配,直接使用它
  1146. if (exactTextMatches === 1 && matches.length === 1) {
  1147. const match = matches[0];
  1148. try {
  1149. const range = document.createRange();
  1150. range.setStart(match.node, match.idx);
  1151. range.setEnd(match.node, match.idx + highlight.text.length);
  1152.  
  1153. const highlightElement = document.createElement('span');
  1154. highlightElement.className = `${STYLE_PREFIX}highlight-marked`;
  1155. highlightElement.dataset.highlightId = highlight.id;
  1156. highlightElement.style.backgroundColor = highlight.color;
  1157.  
  1158. const fragment = range.extractContents();
  1159. highlightElement.appendChild(fragment);
  1160. range.insertNode(highlightElement);
  1161.  
  1162. highlightElement.addEventListener('click', (e) => {
  1163. e.preventDefault();
  1164. e.stopPropagation();
  1165. showHighlightEditMenu(e, highlight.id);
  1166. });
  1167.  
  1168. return true;
  1169. } catch (e) {
  1170. console.warn('唯一文本匹配恢复高亮失败:', e);
  1171. }
  1172. }
  1173.  
  1174. // 如果找到多个匹配项,选择得分最高的
  1175. if (matches.length > 0) {
  1176. // 按得分降序排序
  1177. matches.sort((a, b) => b.score - a.score);
  1178. const bestMatch = matches[0];
  1179.  
  1180. // 降低匹配阈值,从5降到3,使得更多的匹配可以被接受
  1181. const minScore = exactTextMatches > 1 ? 3 : 1;
  1182.  
  1183. if (bestMatch.score >= minScore) {
  1184. try {
  1185. const range = document.createRange();
  1186. range.setStart(bestMatch.node, bestMatch.idx);
  1187. range.setEnd(bestMatch.node, bestMatch.idx + highlight.text.length);
  1188.  
  1189. const highlightElement = document.createElement('span');
  1190. highlightElement.className = `${STYLE_PREFIX}highlight-marked`;
  1191. highlightElement.dataset.highlightId = highlight.id;
  1192. highlightElement.style.backgroundColor = highlight.color;
  1193.  
  1194. const fragment = range.extractContents();
  1195. highlightElement.appendChild(fragment);
  1196. range.insertNode(highlightElement);
  1197.  
  1198. highlightElement.addEventListener('click', (e) => {
  1199. e.preventDefault();
  1200. e.stopPropagation();
  1201. showHighlightEditMenu(e, highlight.id);
  1202. });
  1203.  
  1204. return true;
  1205. } catch (e) {
  1206. console.warn('上下文匹配恢复高亮失败:', e);
  1207. }
  1208. } else {
  1209. console.log('匹配分数过低,尝试模糊匹配:', {
  1210. text: highlight.text,
  1211. bestScore: bestMatch.score,
  1212. matchCount: matches.length
  1213. });
  1214. }
  1215. }
  1216.  
  1217. return false;
  1218. }
  1219.  
  1220. function extractValidContext(text, start, count, direction) {
  1221. // direction: "backward" 从 start 往前提取, "forward" 从 start 往后提取
  1222. let result = "";
  1223. let processedChars = 0;
  1224.  
  1225. // 对于短文本或单个字符,我们提取更多上下文
  1226. const adjustedCount = count * (text.length <= 3 ? 2 : 1);
  1227.  
  1228. if (direction === "backward") {
  1229. for (let i = start - 1; i >= 0 && processedChars < adjustedCount * 2; i--) {
  1230. const ch = text.charAt(i);
  1231. // 只计算有效字符(中文、英文、数字)
  1232. if (/[\u4e00-\u9fffA-Za-z0-9]/.test(ch)) {
  1233. result = ch + result;
  1234. processedChars++;
  1235. if (processedChars >= adjustedCount) break;
  1236. } else {
  1237. // 空格和标点也记录,但不计入有效字符数
  1238. result = ch + result;
  1239. }
  1240. }
  1241. } else { // forward
  1242. for (let i = start; i < text.length && processedChars < adjustedCount * 2; i++) {
  1243. const ch = text.charAt(i);
  1244. // 只计算有效字符(中文、英文、数字)
  1245. if (/[\u4e00-\u9fffA-Za-z0-9]/.test(ch)) {
  1246. result += ch;
  1247. processedChars++;
  1248. if (processedChars >= adjustedCount) break;
  1249. } else {
  1250. // 空格和标点也记录,但不计入有效字符数
  1251. result += ch;
  1252. }
  1253. }
  1254. }
  1255. return result;
  1256. }
  1257.  
  1258. // 添加浮动按钮和侧边栏功能
  1259. function createFloatingButtonAndSidebar() {
  1260. const tooltipStyle = document.createElement('style');
  1261. tooltipStyle.textContent = `
  1262. .${STYLE_PREFIX}tooltip {
  1263. position: absolute;
  1264. background: #1A1D24;
  1265. color: #E8E9EB;
  1266. padding: 4px 10px;
  1267. border-radius: 4px;
  1268. font-size: 12px;
  1269. pointer-events: none;
  1270. white-space: nowrap;
  1271. z-index: 10000;
  1272. opacity: 0;
  1273. transition: opacity 0.2s;
  1274. bottom: 130%;
  1275. left: 50%;
  1276. transform: translateX(-50%);
  1277. box-shadow: 0 2px 8px rgba(0, 0, 0, 0.25);
  1278. border: 1px solid rgba(59, 165, 216, 0.2);
  1279. }
  1280. .${STYLE_PREFIX}tooltip::after {
  1281. content: "";
  1282. position: absolute;
  1283. top: 100%;
  1284. left: 50%;
  1285. margin-left: -5px;
  1286. border-width: 5px;
  1287. border-style: solid;
  1288. border-color: #1A1D24 transparent transparent transparent;
  1289. }
  1290. .${STYLE_PREFIX}tooltip-container {
  1291. position: relative;
  1292. }
  1293. `;
  1294. document.head.appendChild(tooltipStyle);
  1295. // 创建浮动按钮
  1296. const floatingButton = document.createElement('button');
  1297. floatingButton.id = `${STYLE_PREFIX}floating-button`;
  1298. // 使用 SVG 图标,代表"汉堡菜单"
  1299. floatingButton.innerHTML = `
  1300. <svg viewBox="0 0 100 80" width="16" height="16" fill="#ccc" xmlns="http://www.w3.org/2000/svg">
  1301. <rect width="100" height="10"></rect>
  1302. <rect y="30" width="100" height="10"></rect>
  1303. <rect y="60" width="100" height="10"></rect>
  1304. </svg>
  1305. `;
  1306. // 根据设置和启用状态决定是否显示
  1307. floatingButton.style.display = (settings.showFloatingButton && isHighlightEnabled) ? 'flex' : 'none';
  1308. document.body.appendChild(floatingButton);
  1309. // 创建侧边栏(初始隐藏)
  1310. const sidebar = document.createElement('div');
  1311. sidebar.id = `${STYLE_PREFIX}sidebar`;
  1312. Object.assign(sidebar.style, {
  1313. position: 'fixed',
  1314. top: '0',
  1315. right: '-280px',
  1316. width: '280px',
  1317. height: '100%',
  1318. boxShadow: '-1px 0 8px rgba(0, 0, 0, 0.2)',
  1319. transition: 'none',
  1320. zIndex: '9999',
  1321. overflow: 'hidden',
  1322. display: 'flex',
  1323. flexDirection: 'column',
  1324. color: '#E8E9EB',
  1325. fontFamily: '-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif',
  1326. background: 'linear-gradient(135deg, #262A33 0%, #1A1D24 100%)',
  1327. borderLeft: '1px solid rgba(255, 255, 255, 0.03)',
  1328. });
  1329.  
  1330. // 构建侧边栏内部结构
  1331. sidebar.innerHTML = `
  1332. <div class="${STYLE_PREFIX}sidebar-header">
  1333. <div class="${STYLE_PREFIX}sidebar-title" title="双击修改标题">
  1334. ${settings.sidebarDescription || '网页划词高亮工具'}
  1335. </div>
  1336. <div class="${STYLE_PREFIX}sidebar-controls">
  1337. <button class="${STYLE_PREFIX}sidebar-close" title="关闭侧边栏">
  1338. <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
  1339. <path d="M18 6L6 18M6 6l12 12"></path>
  1340. </svg>
  1341. </button>
  1342. </div>
  1343. </div>
  1344. <div class="${STYLE_PREFIX}sidebar-tabs">
  1345. <button class="${STYLE_PREFIX}sidebar-tab ${STYLE_PREFIX}active" data-tab="highlights">高亮列表</button>
  1346. <button class="${STYLE_PREFIX}sidebar-tab" data-tab="disabled">启用管理</button>
  1347. </div>
  1348. <div class="${STYLE_PREFIX}sidebar-content">
  1349. <div class="${STYLE_PREFIX}tab-panel ${STYLE_PREFIX}active" data-panel="highlights">
  1350. <div class="${STYLE_PREFIX}highlights-list"></div>
  1351. </div>
  1352. <div class="${STYLE_PREFIX}tab-panel" data-panel="disabled">
  1353. <div class="${STYLE_PREFIX}disabled-container"></div>
  1354. </div>
  1355. </div>
  1356. `;
  1357.  
  1358. document.body.appendChild(sidebar);
  1359. setTimeout(() => {
  1360. sidebar.style.transition = 'right 0.3s ease';
  1361. }, 10);
  1362.  
  1363. // 设置侧边栏内部元素样式
  1364. const header = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-header`);
  1365. Object.assign(header.style, {
  1366. display: 'flex',
  1367. justifyContent: 'space-between',
  1368. alignItems: 'center',
  1369. boxSizing: 'border-box',
  1370. borderBottom: '1px solid rgba(255, 255, 255, 0.05)',
  1371. height: '42px',
  1372. background: 'rgba(26, 29, 36, 0.8)',
  1373. padding: '0 16px',
  1374. });
  1375.  
  1376. const title = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-title`);
  1377. Object.assign(title.style, {
  1378. fontSize: '13px',
  1379. fontWeight: '600',
  1380. flex: '1',
  1381. overflow: 'hidden',
  1382. textOverflow: 'ellipsis',
  1383. whiteSpace: 'nowrap',
  1384. cursor: 'default',
  1385. letterSpacing: '0.3px',
  1386. opacity: '0.9',
  1387. });
  1388.  
  1389. const controls = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-controls`);
  1390. Object.assign(controls.style, {
  1391. display: 'flex',
  1392. gap: '8px'
  1393. });
  1394.  
  1395. // 设置按钮样式
  1396. sidebar.querySelectorAll(`.${STYLE_PREFIX}sidebar-controls button`).forEach(btn => {
  1397. Object.assign(btn.style, {
  1398. background: 'none',
  1399. border: 'none',
  1400. cursor: 'pointer',
  1401. padding: '3px', // 从5px减小到3px
  1402. display: 'flex',
  1403. alignItems: 'center',
  1404. justifyContent: 'center',
  1405. color: '#ccc',
  1406. borderRadius: '3px',
  1407. transition: 'background-color 0.2s'
  1408. });
  1409.  
  1410. // 添加按钮悬停效果
  1411. btn.addEventListener('mouseenter', () => {
  1412. btn.style.backgroundColor = 'rgba(255, 255, 255, 0.1)';
  1413. });
  1414.  
  1415. btn.addEventListener('mouseleave', () => {
  1416. btn.style.backgroundColor = 'transparent';
  1417. });
  1418. });
  1419.  
  1420. // 设置标签页样式
  1421. const tabs = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-tabs`);
  1422. Object.assign(tabs.style, {
  1423. display: 'flex',
  1424. borderBottom: '1px solid rgba(255, 255, 255, 0.05)',
  1425. padding: '0',
  1426. justifyContent: 'center',
  1427. backgroundColor: 'rgba(26, 29, 36, 0.6)',
  1428. height: '36px'
  1429. });
  1430.  
  1431. sidebar.querySelectorAll(`.${STYLE_PREFIX}sidebar-tab`).forEach(tab => {
  1432. Object.assign(tab.style, {
  1433. height: '100%',
  1434. cursor: 'pointer',
  1435. fontWeight: '500',
  1436. background: 'none',
  1437. border: 'none',
  1438. color: '#ADADB8',
  1439. borderBottom: '2px solid transparent',
  1440. margin: '0',
  1441. transition: 'all 0.2s ease',
  1442. flex: '1',
  1443. display: 'flex',
  1444. alignItems: 'center',
  1445. justifyContent: 'center',
  1446. letterSpacing: '0.3px',
  1447. fontSize: '13px',
  1448. padding: '0 12px',
  1449. opacity: '0.75',
  1450. });
  1451. });
  1452.  
  1453. // 激活的标签页样式
  1454. const activeTab = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-tab.${STYLE_PREFIX}active`);
  1455. if (activeTab) {
  1456. Object.assign(activeTab.style, {
  1457. color: '#3BA5D8',
  1458. borderBottom: '2px solid #3BA5D8', // 改为湖蓝色
  1459. backgroundColor: 'rgba(59, 165, 216, 0.05)',
  1460. opacity: '1',
  1461. });
  1462. }
  1463.  
  1464. // 内容区域样式
  1465. const contentArea = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-content`);
  1466. Object.assign(contentArea.style, {
  1467. flex: '1',
  1468. overflow: 'hidden',
  1469. position: 'relative'
  1470. });
  1471.  
  1472. // 设置面板样式
  1473. sidebar.querySelectorAll(`.${STYLE_PREFIX}tab-panel`).forEach(panel => {
  1474. Object.assign(panel.style, {
  1475. height: '100%',
  1476. width: '100%',
  1477. position: 'absolute',
  1478. top: '0',
  1479. left: '0',
  1480. padding: '14px 14px 14px 14px', // 修改为:上右下左,底部padding设为0
  1481. boxSizing: 'border-box',
  1482. overflow: 'auto',
  1483. display: 'none'
  1484. });
  1485. });
  1486.  
  1487. // 显示当前活动面板
  1488. const activePanel = sidebar.querySelector(`.${STYLE_PREFIX}tab-panel.${STYLE_PREFIX}active`);
  1489. if (activePanel) {
  1490. activePanel.style.display = 'block';
  1491. }
  1492.  
  1493. // 添加侧边栏拖拽调整区域(位于侧边栏的最左侧)
  1494. const resizer = document.createElement('div');
  1495. Object.assign(resizer.style, {
  1496. position: 'absolute',
  1497. left: '0',
  1498. top: '0',
  1499. width: '5px',
  1500. height: '100%',
  1501. cursor: 'ew-resize',
  1502. backgroundColor: 'transparent'
  1503. });
  1504. sidebar.appendChild(resizer);
  1505.  
  1506. // 拖拽事件逻辑
  1507. resizer.addEventListener('mousedown', initResize);
  1508.  
  1509. function initResize(e) {
  1510. e.preventDefault();
  1511. window.addEventListener('mousemove', resizeSidebar);
  1512. window.addEventListener('mouseup', stopResize);
  1513. }
  1514.  
  1515. function resizeSidebar(e) {
  1516. // 计算出新的宽度:侧边栏右对齐,宽度 = 窗口宽度 - 鼠标水平位置
  1517. const newWidth = window.innerWidth - e.clientX;
  1518. // 限制最小宽度为 150px,最大宽度为窗口 80%
  1519. if (newWidth >= 150 && newWidth <= window.innerWidth * 0.8) {
  1520. sidebar.style.width = newWidth + 'px';
  1521. // 更新设置中的宽度
  1522. settings.sidebarWidth = newWidth;
  1523. saveSettings();
  1524. }
  1525. }
  1526.  
  1527. function stopResize(e) {
  1528. window.removeEventListener('mousemove', resizeSidebar);
  1529. window.removeEventListener('mouseup', stopResize);
  1530. }
  1531.  
  1532. // 标签页切换事件
  1533. sidebar.querySelectorAll(`.${STYLE_PREFIX}sidebar-tab`).forEach(tab => {
  1534. tab.addEventListener('click', () => {
  1535. // 移除所有标签页和面板的活动状态
  1536. sidebar.querySelectorAll(`.${STYLE_PREFIX}sidebar-tab`).forEach(t => {
  1537. t.classList.remove(`${STYLE_PREFIX}active`);
  1538. t.style.color = '#ccc';
  1539. t.style.borderBottom = '2px solid transparent';
  1540. t.style.backgroundColor = 'transparent';
  1541. });
  1542.  
  1543. sidebar.querySelectorAll(`.${STYLE_PREFIX}tab-panel`).forEach(p => {
  1544. p.classList.remove(`${STYLE_PREFIX}active`);
  1545. p.style.display = 'none';
  1546. });
  1547.  
  1548. // 激活当前标签和面板
  1549. tab.classList.add(`${STYLE_PREFIX}active`);
  1550. tab.style.color = '#fff';
  1551. // 这里需要修改,将红色改为湖蓝色
  1552. tab.style.borderBottom = '2px solid #3BA5D8'; // 修改为湖蓝色,与初始样式一致
  1553. tab.style.backgroundColor = 'rgba(59, 165, 216, 0.05)'; // 添加微妙的背景色
  1554.  
  1555. const panelId = tab.getAttribute('data-tab');
  1556. const panel = sidebar.querySelector(`.${STYLE_PREFIX}tab-panel[data-panel="${panelId}"]`);
  1557. if (panel) {
  1558. panel.classList.add(`${STYLE_PREFIX}active`);
  1559. panel.style.display = 'block';
  1560. }
  1561. });
  1562.  
  1563. // 悬停效果也应修改为一致的颜色
  1564. tab.addEventListener('mouseenter', () => {
  1565. if (!tab.classList.contains(`${STYLE_PREFIX}active`)) {
  1566. tab.style.backgroundColor = 'rgba(59, 165, 216, 0.05)'; // 改为湖蓝色背景
  1567. tab.style.borderBottom = '2px solid rgba(59, 165, 216, 0.3)'; // 淡化的湖蓝色边框
  1568. }
  1569. });
  1570. });
  1571.  
  1572. // 标题双击编辑功能
  1573. const titleElement = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-title`);
  1574. titleElement.addEventListener('dblclick', () => {
  1575. const currentTitle = titleElement.textContent.trim() || '高亮工具';
  1576. const input = document.createElement('input');
  1577. input.type = 'text';
  1578. input.value = currentTitle;
  1579.  
  1580. // 美化输入框样式
  1581. Object.assign(input.style, {
  1582. width: '100%',
  1583. fontSize: '14px',
  1584. fontWeight: '500',
  1585. padding: '2px 6px',
  1586. boxSizing: 'border-box',
  1587. border: '1px solid rgba(116, 180, 255, 0.5)',
  1588. borderRadius: '3px',
  1589. outline: 'none',
  1590. background: 'rgba(0, 0, 0, 0.2)',
  1591. color: '#fff',
  1592. transition: 'all 0.15s ease',
  1593. boxShadow: '0 0 0 1px rgba(116, 180, 255, 0.3)'
  1594. });
  1595.  
  1596. // 替换标题内容为输入框
  1597. titleElement.innerHTML = '';
  1598. titleElement.appendChild(input);
  1599. input.focus();
  1600. input.select();
  1601.  
  1602. // 添加输入框聚焦样式
  1603. input.addEventListener('focus', () => {
  1604. input.style.background = 'rgba(20, 20, 20, 0.4)';
  1605. input.style.boxShadow = '0 0 0 2px rgba(116, 180, 255, 0.4)';
  1606. });
  1607.  
  1608. // 确认修改:输入框失焦或按下 Enter 键时更新标题
  1609. const confirmChange = () => {
  1610. const newTitle = input.value.trim() || '高亮工具';
  1611. settings.sidebarDescription = newTitle;
  1612. titleElement.textContent = newTitle;
  1613. saveSettings();
  1614. };
  1615.  
  1616. input.addEventListener('blur', confirmChange);
  1617. input.addEventListener('keydown', (event) => {
  1618. if (event.key === 'Enter') {
  1619. input.blur();
  1620. } else if (event.key === 'Escape') {
  1621. // 按ESC键取消编辑
  1622. titleElement.textContent = currentTitle;
  1623. input.blur();
  1624. }
  1625. });
  1626. });
  1627.  
  1628. // 关闭按钮事件
  1629. const closeButton = sidebar.querySelector(`.${STYLE_PREFIX}sidebar-close`);
  1630. closeButton.addEventListener('click', () => {
  1631. sidebar.style.right = `-${parseInt(sidebar.style.width)}px`;
  1632.  
  1633. // 侧边栏关闭时,如果设置允许显示浮动按钮且当前页面未启用,则恢复显示浮动按钮
  1634. if (settings.showFloatingButton && isHighlightEnabled) {
  1635. floatingButton.style.display = 'flex';
  1636. }
  1637. });
  1638.  
  1639. // 浮动按钮点击后切换侧边栏的显示和隐藏
  1640. floatingButton.addEventListener('click', () => {
  1641. if (sidebar.style.right === '0px') {
  1642. sidebar.style.right = `-${parseInt(sidebar.style.width)}px`;
  1643. // 如果设置允许显示浮动按钮且当前页面已启用,则显示浮动按钮
  1644. if (settings.showFloatingButton && isHighlightEnabled) { // 正确的变量
  1645. floatingButton.style.display = 'flex';
  1646. }
  1647. } else {
  1648. sidebar.style.right = '0px';
  1649. // 当侧边栏显示时,隐藏浮动按钮
  1650. floatingButton.style.display = 'none';
  1651. // 刷新高亮列表
  1652. if (updateSidebarHighlights) {
  1653. updateSidebarHighlights();
  1654. }
  1655. }
  1656. });
  1657.  
  1658. // 初始设置宽度
  1659. if (settings.sidebarWidth) {
  1660. sidebar.style.width = `${settings.sidebarWidth}px`;
  1661. sidebar.style.right = `-${settings.sidebarWidth}px`; // 确保初始位置与实际宽度匹配
  1662. } else {
  1663. sidebar.style.right = '-300px'; // 默认宽度的对应位置
  1664. }
  1665.  
  1666. // 渲染高亮列表面板
  1667. function renderHighlightsList() {
  1668. const highlightsListContainer = sidebar.querySelector(`.${STYLE_PREFIX}highlights-list`);
  1669. if (!highlightsListContainer) return;
  1670.  
  1671. // 清空容器
  1672. highlightsListContainer.innerHTML = '';
  1673. Object.assign(highlightsListContainer.style, {
  1674. height: 'calc(100vh - 120px)', // 调整合适高度,减去顶栏和标签栏高度
  1675. overflow: 'hidden',
  1676. display: 'flex', // 添加flex布局
  1677. flexDirection: 'column', // 确保子元素垂直排列
  1678. paddingBottom: '0', // 确保底部无padding
  1679. width: '100%', // 确保宽度不超过父容器
  1680. position: 'relative', // 为绝对定位的子元素提供定位上下文
  1681. boxSizing: 'border-box', // 确保内边距不会增加元素实际宽度
  1682. });
  1683.  
  1684. // 创建高亮列表
  1685. const listContainer = document.createElement('div');
  1686. listContainer.className = `${STYLE_PREFIX}highlights-items`;
  1687. Object.assign(listContainer.style, {
  1688. display: 'flex',
  1689. flexDirection: 'column',
  1690. gap: '8px',
  1691. height: 'calc(100% - 12px)',
  1692. overflow: 'auto',
  1693. paddingRight: '8px',
  1694. paddingBottom: '4px'
  1695. });
  1696.  
  1697. // 自定义滚动条样式
  1698. const styleEl = document.createElement('style');
  1699. styleEl.textContent = `
  1700. .${STYLE_PREFIX}highlights-items::-webkit-scrollbar {
  1701. width: 5px; /* 更细的滚动条 */
  1702. }
  1703. .${STYLE_PREFIX}highlights-items::-webkit-scrollbar-track {
  1704. background: rgba(255, 255, 255, 0.03); /* 更微妙的轨道 */
  1705. border-radius: 3px;
  1706. }
  1707. .${STYLE_PREFIX}highlights-items::-webkit-scrollbar-thumb {
  1708. background: rgba(255, 255, 255, 0.15); /* 更微妙的滑块 */
  1709. border-radius: 3px;
  1710. }
  1711. .${STYLE_PREFIX}highlights-items::-webkit-scrollbar-thumb:hover {
  1712. background: rgba(255, 255, 255, 0.25);
  1713. }
  1714. `;
  1715. document.head.appendChild(styleEl);
  1716.  
  1717. // 排序高亮,按时间倒序
  1718. const sortedHighlights = [...highlights].sort((a, b) => (b.timestamp || 0) - (a.timestamp || 0));
  1719.  
  1720. if (sortedHighlights.length === 0) {
  1721. // 显示空状态
  1722. const emptyState = document.createElement('div');
  1723. Object.assign(emptyState.style, {
  1724. display: 'flex',
  1725. flexDirection: 'column',
  1726. alignItems: 'center',
  1727. justifyContent: 'center',
  1728. padding: '60px 20px',
  1729. textAlign: 'center',
  1730. color: '#999',
  1731. fontSize: '13px'
  1732. });
  1733.  
  1734. // 使用SVG图标作为空状态图标
  1735. emptyState.innerHTML = `
  1736. <svg width="60" height="60" viewBox="0 0 24 24" fill="none" stroke="rgba(255,255,255,0.15)"
  1737. stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round">
  1738. <path d="M17 3a2.828 2.828 0 1 1 4 4L7.5 20.5 2 22l1.5-5.5L17 3z"></path>
  1739. </svg>
  1740. <p style="margin-top:16px; font-size:13px !important;">暂无高亮内容<br>选中文本并点击颜色进行高亮</p>
  1741. `;
  1742. listContainer.appendChild(emptyState);
  1743. } else {
  1744. // 渲染所有高亮项目
  1745. sortedHighlights.forEach((highlight, index) => {
  1746. const highlightItem = createHighlightItem(highlight, index);
  1747. listContainer.appendChild(highlightItem);
  1748. });
  1749. }
  1750.  
  1751. highlightsListContainer.appendChild(listContainer);
  1752.  
  1753. // 创建底部固定按钮栏
  1754. const bottomActionBar = document.createElement('div');
  1755. bottomActionBar.className = `${STYLE_PREFIX}highlights-bottom-actions`;
  1756. Object.assign(bottomActionBar.style, {
  1757. display: 'flex',
  1758. position: 'absolute',
  1759. bottom: '0',
  1760. left: '0',
  1761. right: '0',
  1762. padding: '0',
  1763. zIndex: '1',
  1764. gap: '10px',
  1765. boxSizing: 'border-box',
  1766. background: 'transparent',
  1767. });
  1768.  
  1769. // 创建刷新按钮
  1770. const refreshBtn = document.createElement('button');
  1771. refreshBtn.textContent = '刷新列表';
  1772. Object.assign(refreshBtn.style, {
  1773. flex: '1',
  1774. background: 'rgba(59, 165, 216, 0.12)',
  1775. border: '1px solid rgba(59, 165, 216, 0.08)',
  1776. borderRadius: '4px',
  1777. padding: '8px 12px',
  1778. color: '#3BA5D8',
  1779. fontSize: '13px',
  1780. fontWeight: '500',
  1781. cursor: 'pointer',
  1782. transition: 'all 0.2s ease'
  1783. });
  1784.  
  1785. // 添加悬停效果
  1786. refreshBtn.addEventListener('mouseenter', () => {
  1787. refreshBtn.style.background = 'rgba(59, 165, 216, 0.2)';
  1788. });
  1789. refreshBtn.addEventListener('mouseleave', () => {
  1790. refreshBtn.style.background = 'rgba(59, 165, 216, 0.12)';
  1791. });
  1792.  
  1793. refreshBtn.addEventListener('click', () => {
  1794. // 刷新高亮列表
  1795. loadHighlights();
  1796. applyHighlights();
  1797. renderHighlightsList();
  1798. });
  1799.  
  1800. // 创建清除按钮
  1801. const clearBtn = document.createElement('button');
  1802. clearBtn.textContent = '清除全部';
  1803. Object.assign(clearBtn.style, {
  1804. flex: '1',
  1805. background: 'rgba(255, 82, 82, 0.12)',
  1806. border: '1px solid rgba(255, 82, 82, 0.08)',
  1807. color: '#ff6b6b',
  1808. borderRadius: '4px',
  1809. padding: '8px 12px',
  1810. fontSize: '13px',
  1811. fontWeight: '500',
  1812. cursor: 'pointer',
  1813. transition: 'all 0.2s ease'
  1814. });
  1815.  
  1816. // 添加悬停效果
  1817. clearBtn.addEventListener('mouseenter', () => {
  1818. clearBtn.style.background = 'rgba(255, 82, 82, 0.2)';
  1819. });
  1820. clearBtn.addEventListener('mouseleave', () => {
  1821. clearBtn.style.background = 'rgba(255, 82, 82, 0.12)';
  1822. });
  1823.  
  1824. clearBtn.addEventListener('click', () => {
  1825. if (highlights.length === 0) return;
  1826.  
  1827. // 确认删除
  1828. if (confirm('确定要删除所有高亮吗?此操作不可撤销。')) {
  1829. // 移除DOM中的高亮元素
  1830. document.querySelectorAll(`.${STYLE_PREFIX}highlight-marked`).forEach(el => {
  1831. const textNode = document.createTextNode(el.textContent);
  1832. el.parentNode.replaceChild(textNode, el);
  1833. });
  1834.  
  1835. // 清空高亮数组
  1836. highlights = [];
  1837. saveHighlights();
  1838. renderHighlightsList();
  1839. }
  1840. });
  1841.  
  1842. bottomActionBar.appendChild(refreshBtn);
  1843. bottomActionBar.appendChild(clearBtn);
  1844. highlightsListContainer.appendChild(bottomActionBar);
  1845. }
  1846.  
  1847. updateSidebarHighlights = renderHighlightsList;
  1848.  
  1849. // 创建单个高亮项目
  1850. function createHighlightItem(highlight, index) {
  1851. const item = document.createElement('div');
  1852. item.className = `${STYLE_PREFIX}highlight-item`;
  1853. item.dataset.highlightId = highlight.id;
  1854.  
  1855. Object.assign(item.style, {
  1856. backgroundColor: 'rgba(42, 46, 54, 0.4)',
  1857. borderRadius: '4px',
  1858. padding: '12px 14px',
  1859. position: 'relative',
  1860. transition: 'all 0.15s ease',
  1861. border: '1px solid rgba(255, 255, 255, 0.03)',
  1862. margin: '0 0 8px 0',
  1863. });
  1864.  
  1865. // 颜色指示器
  1866. const colorIndicator = document.createElement('div');
  1867. Object.assign(colorIndicator.style, {
  1868. position: 'absolute',
  1869. top: '0',
  1870. left: '0',
  1871. width: '3px',
  1872. height: '100%',
  1873. backgroundColor: highlight.color,
  1874. borderTopLeftRadius: '12px',
  1875. borderBottomLeftRadius: '12px'
  1876. });
  1877.  
  1878. // 高亮内容
  1879. const content = document.createElement('div');
  1880. Object.assign(content.style, {
  1881. paddingLeft: '4px',
  1882. color: '#E8E9EB',
  1883. fontSize: '14px',
  1884. lineHeight: '1.5',
  1885. marginBottom: '8px',
  1886. wordBreak: 'break-word',
  1887. display: '-webkit-box',
  1888. WebkitBoxOrient: 'vertical',
  1889. WebkitLineClamp: '2',
  1890. overflow: 'hidden',
  1891. textOverflow: 'ellipsis'
  1892. });
  1893.  
  1894. // 处理高亮文本,避免XSS
  1895. const textNode = document.createTextNode(highlight.text);
  1896. content.appendChild(textNode);
  1897.  
  1898. // 底部信息栏
  1899. const infoBar = document.createElement('div');
  1900. Object.assign(infoBar.style, {
  1901. display: 'flex',
  1902. justifyContent: 'space-between',
  1903. alignItems: 'center',
  1904. fontSize: '12px',
  1905. marginTop: '6px',
  1906. paddingLeft: '3px' // 与内容区域左边距统一
  1907. });
  1908.  
  1909. // 时间信息
  1910. const timeInfo = document.createElement('div');
  1911. Object.assign(timeInfo.style, {
  1912. color: '#999',
  1913. fontSize: '12px'
  1914. });
  1915.  
  1916. // 格式化时间
  1917. const date = new Date(highlight.timestamp);
  1918. const formattedDate = `${date.getMonth() + 1}月${date.getDate()}日 ${date.getHours().toString().padStart(2, '0')}:${date.getMinutes().toString().padStart(2, '0')}`;
  1919. timeInfo.textContent = formattedDate;
  1920.  
  1921. // 操作按钮容器
  1922. const actionButtons = document.createElement('div');
  1923. Object.assign(actionButtons.style, {
  1924. display: 'flex',
  1925. gap: '10px'
  1926. });
  1927.  
  1928. // 跳转按钮
  1929. const jumpButton = document.createElement('div');
  1930. Object.assign(jumpButton.style, {
  1931. position: 'relative',
  1932. background: 'none',
  1933. border: 'none',
  1934. padding: '3px',
  1935. cursor: 'pointer',
  1936. color: '#74b4ff',
  1937. display: 'flex',
  1938. alignItems: 'center',
  1939. fontSize: '12px',
  1940. transition: 'color 0.15s ease'
  1941. });
  1942. jumpButton.className = `${STYLE_PREFIX}tooltip-container`;
  1943. jumpButton.innerHTML = `
  1944. <div class="${STYLE_PREFIX}tooltip">跳转到此高亮</div>
  1945. <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor"
  1946. stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
  1947. <path d="M15 3h6v6M14 10l6.16-6.16M9 21H3v-6M10 14l-6.16 6.16"></path>
  1948. </svg>
  1949. `;
  1950.  
  1951. jumpButton.addEventListener('mouseenter', () => {
  1952. jumpButton.style.color = '#a0cfff';
  1953. jumpButton.querySelector(`.${STYLE_PREFIX}tooltip`).style.opacity = '1';
  1954. });
  1955.  
  1956. jumpButton.addEventListener('mouseleave', () => {
  1957. jumpButton.style.color = '#74b4ff';
  1958. jumpButton.querySelector(`.${STYLE_PREFIX}tooltip`).style.opacity = '0';
  1959. });
  1960.  
  1961. // 删除按钮
  1962. const deleteButton = document.createElement('div');
  1963. Object.assign(deleteButton.style, {
  1964. position: 'relative',
  1965. background: 'none',
  1966. border: 'none',
  1967. padding: '3px',
  1968. cursor: 'pointer',
  1969. color: 'rgba(190, 60, 60, 0.8)',
  1970. display: 'flex',
  1971. alignItems: 'center',
  1972. fontSize: '12px',
  1973. transition: 'color 0.15s ease'
  1974. });
  1975. deleteButton.className = `${STYLE_PREFIX}tooltip-container`;
  1976. deleteButton.innerHTML = `
  1977. <div class="${STYLE_PREFIX}tooltip">删除此高亮</div>
  1978. <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor"
  1979. stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
  1980. <path d="M3 6h18M19 6v14a2 2 0 01-2 2H7a2 2 0 01-2-2V6m3 0V4a2 2 0 012-2h4a2 2 0 012 2v2"></path>
  1981. </svg>
  1982. `;
  1983.  
  1984. deleteButton.addEventListener('mouseenter', () => {
  1985. deleteButton.style.color = 'rgba(255, 80, 80, 0.95)';
  1986. deleteButton.querySelector(`.${STYLE_PREFIX}tooltip`).style.opacity = '1';
  1987. });
  1988.  
  1989. deleteButton.addEventListener('mouseleave', () => {
  1990. deleteButton.style.color = 'rgba(190, 60, 60, 0.8)';
  1991. deleteButton.querySelector(`.${STYLE_PREFIX}tooltip`).style.opacity = '0';
  1992. });
  1993.  
  1994. actionButtons.appendChild(jumpButton);
  1995. actionButtons.appendChild(deleteButton);
  1996.  
  1997. infoBar.appendChild(timeInfo);
  1998. infoBar.appendChild(actionButtons);
  1999.  
  2000. // 添加项目悬停效果
  2001. item.addEventListener('mouseenter', () => {
  2002. item.style.backgroundColor = 'rgba(59, 165, 216, 0.1)';
  2003. item.style.borderColor = 'rgba(59, 165, 216, 0.15)';
  2004. });
  2005. item.addEventListener('mouseleave', () => {
  2006. item.style.backgroundColor = 'rgba(42, 46, 54, 0.4)';
  2007. item.style.borderColor = 'rgba(255, 255, 255, 0.03)';
  2008. });
  2009.  
  2010.  
  2011. item.appendChild(colorIndicator);
  2012. item.appendChild(content);
  2013. item.appendChild(infoBar);
  2014.  
  2015. return item;
  2016. }
  2017.  
  2018. // 滚动到指定高亮
  2019. function scrollToHighlight(highlightId) {
  2020. const highlightElement = document.querySelector(`.${STYLE_PREFIX}highlight-marked[data-highlight-id="${highlightId}"]`);
  2021. if (highlightElement) {
  2022. // 平滑滚动到元素
  2023. highlightElement.scrollIntoView({ behavior: 'smooth', block: 'center' });
  2024.  
  2025. // 添加闪烁效果,并设置临时样式使其更加明显
  2026. highlightElement.classList.add(`${STYLE_PREFIX}highlight-flash`);
  2027.  
  2028. // 保存原有的样式以便恢复
  2029. const originalTransition = highlightElement.style.transition;
  2030.  
  2031. // 添加过渡效果使视觉反馈更平滑
  2032. highlightElement.style.transition = 'all 0.3s ease';
  2033.  
  2034. setTimeout(() => {
  2035. highlightElement.classList.remove(`${STYLE_PREFIX}highlight-flash`);
  2036. highlightElement.style.transition = originalTransition;
  2037. }, 2500);
  2038. }
  2039. }
  2040.  
  2041. // 初始渲染高亮列表
  2042. renderHighlightsList();
  2043.  
  2044. // 渲染启用管理面板内容
  2045. function renderEnabledPanel() {
  2046. const container = sidebar.querySelector(`.${STYLE_PREFIX}disabled-container`);
  2047. if (!container) return;
  2048.  
  2049. // 清空容器
  2050. container.innerHTML = '';
  2051.  
  2052. // 给容器本身设置明确的颜色
  2053. Object.assign(container.style, {
  2054. color: '#f0f0f0' // 确保所有文本默认为浅色
  2055. });
  2056.  
  2057. // 添加当前页面管理区域
  2058. const currentPageSection = document.createElement('div');
  2059. currentPageSection.className = `${STYLE_PREFIX}disabled-section`;
  2060. Object.assign(currentPageSection.style, {
  2061. marginBottom: '20px',
  2062. color: '#f0f0f0' // 明确设置文本颜色
  2063. });
  2064.  
  2065. const currentPageTitle = document.createElement('div');
  2066. currentPageTitle.className = `${STYLE_PREFIX}disabled-title`;
  2067. currentPageTitle.innerHTML = `<span>当前页面</span>`;
  2068. Object.assign(currentPageTitle.style, {
  2069. fontSize: '13px',
  2070. fontWeight: '600',
  2071. color: '#eee', // 确保标题颜色为浅色
  2072. marginBottom: '10px',
  2073. display: 'flex',
  2074. alignItems: 'center',
  2075. gap: '8px'
  2076. });
  2077.  
  2078. // 当前页面状态
  2079. const currentStatus = document.createElement('div');
  2080. currentStatus.className = `${STYLE_PREFIX}current-status`;
  2081. currentStatus.innerHTML = renderCurrentPageStatus();
  2082.  
  2083. currentPageSection.appendChild(currentPageTitle);
  2084. currentPageSection.appendChild(currentStatus);
  2085. container.appendChild(currentPageSection);
  2086.  
  2087. // 启用域名列表区域
  2088. const domainsSection = document.createElement('div');
  2089. domainsSection.className = `${STYLE_PREFIX}disabled-section`;
  2090. Object.assign(domainsSection.style, {
  2091. marginBottom: '20px'
  2092. });
  2093.  
  2094. const domainsTitle = document.createElement('div');
  2095. domainsTitle.className = `${STYLE_PREFIX}disabled-title`;
  2096. domainsTitle.innerHTML = `<span>启用域名列表</span>`;
  2097. Object.assign(domainsTitle.style, {
  2098. fontSize: '13px',
  2099. fontWeight: '600',
  2100. color: '#eee',
  2101. marginBottom: '10px',
  2102. display: 'flex',
  2103. alignItems: 'center',
  2104. gap: '8px'
  2105. });
  2106.  
  2107. const domainsList = document.createElement('div');
  2108. domainsList.className = `${STYLE_PREFIX}domains-list`;
  2109. domainsList.innerHTML = renderEnabledDomains();
  2110.  
  2111. // 添加域名表单
  2112. const addDomainForm = document.createElement('div');
  2113. addDomainForm.className = `${STYLE_PREFIX}add-disabled-form`;
  2114. Object.assign(addDomainForm.style, {
  2115. display: 'flex',
  2116. marginTop: '12px',
  2117. gap: '0'
  2118. });
  2119.  
  2120. const domainInput = document.createElement('input');
  2121. domainInput.className = `${STYLE_PREFIX}add-disabled-input`;
  2122. domainInput.id = 'add-domain-input';
  2123. domainInput.placeholder = '输入域名...';
  2124. Object.assign(domainInput.style, {
  2125. flex: '1',
  2126. backgroundColor: 'rgba(255, 255, 255, 0.07)',
  2127. border: '1px solid rgba(255, 255, 255, 0.1)',
  2128. borderRadius: '4px 0 0 4px',
  2129. padding: '8px 12px',
  2130. fontSize: '13px',
  2131. color: '#fff',
  2132. outline: 'none'
  2133. });
  2134.  
  2135. const addDomainBtn = document.createElement('button');
  2136. addDomainBtn.className = `${STYLE_PREFIX}add-disabled-button`;
  2137. addDomainBtn.id = 'add-domain-btn';
  2138. addDomainBtn.textContent = '添加';
  2139. Object.assign(addDomainBtn.style, {
  2140. backgroundColor: 'rgba(190, 60, 60, 0.8)',
  2141. color: '#f5e0e0',
  2142. border: 'none',
  2143. borderRadius: '0 4px 4px 0',
  2144. padding: '8px 16px',
  2145. fontSize: '13px',
  2146. fontWeight: '500',
  2147. cursor: 'pointer',
  2148. transition: 'all 0.2s ease'
  2149. });
  2150.  
  2151. addDomainForm.appendChild(domainInput);
  2152. addDomainForm.appendChild(addDomainBtn);
  2153.  
  2154. domainsSection.appendChild(domainsTitle);
  2155. domainsSection.appendChild(domainsList);
  2156. domainsSection.appendChild(addDomainForm);
  2157. container.appendChild(domainsSection);
  2158.  
  2159. // 启用URL列表区域
  2160. const urlsSection = document.createElement('div');
  2161. urlsSection.className = `${STYLE_PREFIX}disabled-section`;
  2162.  
  2163. const urlsTitle = document.createElement('div');
  2164. urlsTitle.className = `${STYLE_PREFIX}disabled-title`;
  2165. urlsTitle.innerHTML = `<span>启用网址列表</span>`;
  2166. Object.assign(urlsTitle.style, {
  2167. fontSize: '13px',
  2168. fontWeight: '600',
  2169. color: '#eee',
  2170. marginBottom: '10px',
  2171. display: 'flex',
  2172. alignItems: 'center',
  2173. gap: '8px'
  2174. });
  2175.  
  2176. const urlsList = document.createElement('div');
  2177. urlsList.className = `${STYLE_PREFIX}urls-list`;
  2178. urlsList.innerHTML = renderEnabledUrls();
  2179.  
  2180. // 添加URL表单
  2181. const addUrlForm = document.createElement('div');
  2182. addUrlForm.className = `${STYLE_PREFIX}add-disabled-form`;
  2183. Object.assign(addUrlForm.style, {
  2184. display: 'flex',
  2185. marginTop: '12px',
  2186. gap: '0'
  2187. });
  2188.  
  2189. const urlInput = document.createElement('input');
  2190. urlInput.className = `${STYLE_PREFIX}add-disabled-input`;
  2191. urlInput.id = 'add-url-input';
  2192. urlInput.placeholder = '输入网址...';
  2193. Object.assign(urlInput.style, {
  2194. flex: '1',
  2195. backgroundColor: 'rgba(255, 255, 255, 0.07)',
  2196. border: '1px solid rgba(255, 255, 255, 0.1)',
  2197. borderRadius: '4px 0 0 4px',
  2198. padding: '8px 12px',
  2199. fontSize: '13px',
  2200. color: '#fff',
  2201. outline: 'none'
  2202. });
  2203.  
  2204. const addUrlBtn = document.createElement('button');
  2205. addUrlBtn.className = `${STYLE_PREFIX}add-disabled-button`;
  2206. addUrlBtn.id = 'add-url-btn';
  2207. addUrlBtn.textContent = '添加';
  2208. Object.assign(addUrlBtn.style, {
  2209. backgroundColor: 'rgba(190, 60, 60, 0.8)',
  2210. color: '#f5e0e0',
  2211. border: 'none',
  2212. borderRadius: '0 4px 4px 0',
  2213. padding: '8px 16px',
  2214. fontSize: '13px',
  2215. fontWeight: '500',
  2216. cursor: 'pointer',
  2217. transition: 'all 0.2s ease'
  2218. });
  2219.  
  2220. addUrlForm.appendChild(urlInput);
  2221. addUrlForm.appendChild(addUrlBtn);
  2222.  
  2223. urlsSection.appendChild(urlsTitle);
  2224. urlsSection.appendChild(urlsList);
  2225. urlsSection.appendChild(addUrlForm);
  2226. container.appendChild(urlsSection);
  2227.  
  2228. // 绑定事件
  2229. bindDisabledPanelEvents();
  2230. }
  2231.  
  2232. // 渲染当前页面状态
  2233. function renderCurrentPageStatus() {
  2234. const isDomainEnabled = enabledList.domains.includes(currentDomain);
  2235. const isUrlEnabled = enabledList.urls.includes(currentPageUrl);
  2236.  
  2237. if (isDomainEnabled || isUrlEnabled) {
  2238. return `
  2239. <div class="${STYLE_PREFIX}disabled-item">
  2240. <div class="${STYLE_PREFIX}disabled-info">
  2241. <span>${isDomainEnabled ? `此域名 (${currentDomain}) 已启用高亮` : '此网址已启用高亮'}</span>
  2242. </div>
  2243. <span class="${STYLE_PREFIX}disabled-action" data-type="${isDomainEnabled ? 'domain' : 'url'}" data-value="${isDomainEnabled ? currentDomain : currentPageUrl}">
  2244. 禁用
  2245. </span>
  2246. </div>
  2247. `;
  2248. } else {
  2249. return `
  2250. <div class="${STYLE_PREFIX}current-page-actions">
  2251. <button class="${STYLE_PREFIX}disable-btn" id="enable-domain-btn">
  2252. 启用此域名
  2253. </button>
  2254. <button class="${STYLE_PREFIX}disable-btn" id="enable-url-btn">
  2255. 启用此网址
  2256. </button>
  2257. </div>
  2258. `;
  2259. }
  2260. }
  2261.  
  2262. // 渲染启用域名列表
  2263. function renderEnabledDomains() {
  2264. if (enabledList.domains.length === 0) {
  2265. return `<div class="${STYLE_PREFIX}empty-list">没有启用的域名</div>`;
  2266. }
  2267.  
  2268. return enabledList.domains.map(domain => `
  2269. <div class="${STYLE_PREFIX}disabled-item">
  2270. <div class="${STYLE_PREFIX}disabled-info">
  2271. <span>${domain}</span>
  2272. </div>
  2273. <span class="${STYLE_PREFIX}disabled-action" data-type="domain" data-value="${domain}">
  2274. 删除
  2275. </span>
  2276. </div>
  2277. `).join('');
  2278. }
  2279.  
  2280. // 渲染启用URL列表
  2281. function renderEnabledUrls() {
  2282. if (enabledList.urls.length === 0) {
  2283. return `<div class="${STYLE_PREFIX}empty-list">没有启用的网址</div>`;
  2284. }
  2285.  
  2286. return enabledList.urls.map(url => {
  2287. // 为了美观,截断过长的URL
  2288. const displayUrl = url.length > 40 ? url.substring(0, 37) + '...' : url;
  2289.  
  2290. return `
  2291. <div class="${STYLE_PREFIX}disabled-item" title="${url}">
  2292. <div class="${STYLE_PREFIX}disabled-info">
  2293. <span>${displayUrl}</span>
  2294. </div>
  2295. <span class="${STYLE_PREFIX}disabled-action" data-type="url" data-value="${url}">
  2296. 删除
  2297. </span>
  2298. </div>
  2299. `;
  2300. }).join('');
  2301. }
  2302.  
  2303. // 绑定启用管理面板事件
  2304. function bindDisabledPanelEvents() {
  2305. // 启用当前域名按钮
  2306. const enableDomainBtn = document.getElementById('enable-domain-btn');
  2307. if (enableDomainBtn) {
  2308. enableDomainBtn.addEventListener('click', () => {
  2309. if (confirm('确定要启用域名 "' + currentDomain + '" 的高亮功能吗?')) {
  2310. enableDomain(currentDomain);
  2311. renderEnabledPanel();
  2312. }
  2313. });
  2314. }
  2315.  
  2316. // 启用当前网址按钮
  2317. const enableUrlBtn = document.getElementById('enable-url-btn');
  2318. if (enableUrlBtn) {
  2319. enableUrlBtn.addEventListener('click', () => {
  2320. if (confirm('确定要启用当前网址的高亮功能吗?')) {
  2321. enableUrl(currentPageUrl);
  2322. renderEnabledPanel();
  2323. }
  2324. });
  2325. }
  2326.  
  2327. // 添加样式
  2328. const styleSheet = document.createElement('style');
  2329. styleSheet.textContent = `
  2330. #${STYLE_PREFIX}sidebar,
  2331. #${STYLE_PREFIX}sidebar *,
  2332. .${STYLE_PREFIX}disabled-section,
  2333. .${STYLE_PREFIX}disabled-title,
  2334. .${STYLE_PREFIX}disabled-title span,
  2335. .${STYLE_PREFIX}disabled-item,
  2336. .${STYLE_PREFIX}disabled-info,
  2337. .${STYLE_PREFIX}disabled-info span,
  2338. .${STYLE_PREFIX}empty-list,
  2339. .${STYLE_PREFIX}sidebar-tab,
  2340. .${STYLE_PREFIX}highlight-item {
  2341. color: #E8E9EB !important;
  2342. }
  2343. .${STYLE_PREFIX}disabled-item {
  2344. display: flex;
  2345. justify-content: space-between;
  2346. align-items: center;
  2347. padding: 8px 12px;
  2348. background-color: rgba(42, 46, 54, 0.4);
  2349. border-radius: 4px;
  2350. margin-bottom: 6px;
  2351. transition: all 0.2s ease;
  2352. border: 1px solid rgba(255, 255, 255, 0.03);
  2353. }
  2354. .${STYLE_PREFIX}disabled-item:hover {
  2355. background-color: rgba(59, 165, 216, 0.1);
  2356. border-color: rgba(59, 165, 216, 0.15);
  2357. }
  2358. .${STYLE_PREFIX}disabled-info {
  2359. display: flex;
  2360. align-items: center;
  2361. gap: 8px;
  2362. font-size: 13px;
  2363. color: #E8E9EB;
  2364. flex: 1;
  2365. overflow: hidden;
  2366. text-overflow: ellipsis;
  2367. white-space: nowrap;
  2368. }
  2369. .${STYLE_PREFIX}disabled-action {
  2370. color: #3BA5D8 !important;
  2371. font-size: 12px;
  2372. cursor: pointer;
  2373. padding: 2px 6px;
  2374. border-radius: 3px;
  2375. transition: all 0.2s;
  2376. opacity: 0.85;
  2377. }
  2378. .${STYLE_PREFIX}disabled-action:hover {
  2379. background-color: rgba(59, 165, 216, 0.15);
  2380. opacity: 1;
  2381. }
  2382. .${STYLE_PREFIX}empty-list {
  2383. padding: 10px;
  2384. color: #ADADB8 !important;
  2385. font-style: italic;
  2386. font-size: 13px;
  2387. text-align: center;
  2388. background-color: rgba(42, 46, 54, 0.2);
  2389. border-radius: 4px;
  2390. }
  2391. .${STYLE_PREFIX}current-page-actions {
  2392. display: flex;
  2393. gap: 10px;
  2394. }
  2395. .${STYLE_PREFIX}disable-btn {
  2396. flex: 1;
  2397. background: rgba(59, 165, 216, 0.12);
  2398. border: none;
  2399. border-radius: 4px;
  2400. padding: 8px 12px;
  2401. color: #3BA5D8;
  2402. font-size: 13px;
  2403. font-weight: 500;
  2404. cursor: pointer;
  2405. transition: all 0.2s ease;
  2406. }
  2407. .${STYLE_PREFIX}disable-btn:hover {
  2408. background-color: rgba(59, 165, 216, 0.2);
  2409. }
  2410. .${STYLE_PREFIX}add-disabled-input:focus {
  2411. border-color: #3BA5D8;
  2412. background-color: rgba(255, 255, 255, 0.1);
  2413. }
  2414. .${STYLE_PREFIX}add-disabled-button:hover {
  2415. background-color: #3BA5D8;
  2416. }
  2417. `;
  2418. document.head.appendChild(styleSheet);
  2419.  
  2420. // 删除按钮事件
  2421. document.querySelectorAll(`.${STYLE_PREFIX}disabled-action`).forEach(btn => {
  2422. btn.addEventListener('click', (e) => {
  2423. const type = e.target.dataset.type;
  2424. const value = e.target.dataset.value;
  2425.  
  2426. if (e.target.textContent.trim() === '删除') {
  2427. if (type === 'domain') {
  2428. enabledList.domains = enabledList.domains.filter(d => d !== value);
  2429. } else if (type === 'url') {
  2430. enabledList.urls = enabledList.urls.filter(u => u !== value);
  2431. }
  2432. saveEnabledList();
  2433. renderEnabledPanel();
  2434. } else if (e.target.textContent.trim() === '启用') {
  2435. if (type === 'domain') {
  2436. enableDomain(value);
  2437. } else if (type === 'url') {
  2438. enableUrl(value);
  2439. }
  2440. renderEnabledPanel();
  2441. } else if (e.target.textContent.trim() === '禁用') {
  2442. // 添加对禁用按钮的处理
  2443. if (type === 'domain') {
  2444. disableDomain(value);
  2445. } else if (type === 'url') {
  2446. disableUrl(value);
  2447. }
  2448. saveEnabledList();
  2449. renderEnabledPanel();
  2450. }
  2451. });
  2452. });
  2453.  
  2454. // 添加域名按钮
  2455. const addDomainBtn = document.getElementById('add-domain-btn');
  2456. if (addDomainBtn) {
  2457. Object.assign(addDomainBtn.style, {
  2458. backgroundColor: 'rgba(59, 165, 216, 0.8)',
  2459. color: '#f0f8ff',
  2460. border: 'none',
  2461. borderRadius: '0 4px 4px 0',
  2462. padding: '8px 16px',
  2463. fontSize: '13px',
  2464. fontWeight: '500',
  2465. cursor: 'pointer',
  2466. transition: 'all 0.2s ease'
  2467. });
  2468. addDomainBtn.addEventListener('click', () => {
  2469. const input = document.getElementById('add-domain-input');
  2470. const domain = input.value.trim();
  2471.  
  2472. if (domain) {
  2473. if (!enabledList.domains.includes(domain)) {
  2474. enabledList.domains.push(domain);
  2475. saveEnabledList();
  2476. input.value = '';
  2477. renderEnabledPanel();
  2478. } else {
  2479. alert('该域名已在启用列表中');
  2480. }
  2481. }
  2482. });
  2483. }
  2484.  
  2485. // 添加URL按钮
  2486. const addUrlBtn = document.getElementById('add-url-btn');
  2487. if (addUrlBtn) {
  2488. Object.assign(addUrlBtn.style, {
  2489. backgroundColor: 'rgba(59, 165, 216, 0.8)',
  2490. color: '#f0f8ff',
  2491. });
  2492. addUrlBtn.addEventListener('click', () => {
  2493. const input = document.getElementById('add-url-input');
  2494. const url = input.value.trim();
  2495.  
  2496. if (url) {
  2497. if (!enabledList.urls.includes(url)) {
  2498. enabledList.urls.push(url);
  2499. saveEnabledList();
  2500. input.value = '';
  2501. renderEnabledPanel();
  2502. } else {
  2503. alert('该网址已在启用列表中');
  2504. }
  2505. }
  2506. });
  2507. }
  2508.  
  2509. // 输入框回车事件
  2510. const domainInput = document.getElementById('add-domain-input');
  2511. if (domainInput) {
  2512. domainInput.addEventListener('keydown', (e) => {
  2513. if (e.key === 'Enter') {
  2514. document.getElementById('add-domain-btn').click();
  2515. }
  2516. });
  2517. }
  2518.  
  2519. const urlInput = document.getElementById('add-url-input');
  2520. if (urlInput) {
  2521. urlInput.addEventListener('keydown', (e) => {
  2522. if (e.key === 'Enter') {
  2523. document.getElementById('add-url-btn').click();
  2524. }
  2525. });
  2526. }
  2527. }
  2528.  
  2529. // 初始渲染启用管理面板
  2530. renderEnabledPanel();
  2531. }
  2532.  
  2533. function init() {
  2534. loadHighlights();
  2535. registerEvents();
  2536. if (document.readyState === 'complete') {
  2537. setTimeout(() => {
  2538. applyHighlights();
  2539. observeDomChanges();
  2540. }, 500);
  2541. } else {
  2542. window.addEventListener('load', () => {
  2543. setTimeout(() => {
  2544. applyHighlights();
  2545. observeDomChanges();
  2546. }, 500);
  2547. });
  2548. }
  2549. // 注册(不可用)油猴菜单命令
  2550. GM_registerMenuCommand('打开侧边栏', () => {
  2551. toggleSidebar(true);
  2552. });
  2553. GM_registerMenuCommand('切换浮动按钮显示/隐藏', toggleFloatingButton);
  2554. GM_registerMenuCommand('启用当前域名高亮', () => {
  2555. if (enabledList.domains.includes(currentDomain)) {
  2556. alert(`当前域名(${currentDomain})已启用高亮功能`);
  2557. } else {
  2558. if (confirm(`确定要启用当前域名(${currentDomain})的高亮功能吗?`)) {
  2559. enableDomain(currentDomain);
  2560. }
  2561. }
  2562. });
  2563. GM_registerMenuCommand('启用当前网址高亮', () => {
  2564. if (enabledList.urls.includes(currentPageUrl)) {
  2565. alert(`当前网址已启用高亮功能`);
  2566. } else {
  2567. if (confirm(`确定要启用当前网址的高亮功能吗?`)) {
  2568. enableUrl(currentPageUrl);
  2569. }
  2570. }
  2571. });
  2572. }
  2573.  
  2574. init();
  2575. createFloatingButtonAndSidebar();
  2576. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址