mirror of
https://github.com/obsqrbtz/goose-highlighter.git
synced 2026-04-08 20:19:06 +03:00
fix: do not re-highlight when already processing highlights
This commit is contained in:
@@ -68,7 +68,7 @@ export class ContentScript {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private setupScrollHandler(): void {
|
private setupScrollHandler(): void {
|
||||||
const debouncedProcess = DOMUtils.debounce(() => this.processHighlights(), 300);
|
const debouncedProcess = DOMUtils.debounce(() => this.processHighlights(), CONSTANTS.DEBOUNCE_DELAY);
|
||||||
window.addEventListener('scroll', debouncedProcess);
|
window.addEventListener('scroll', debouncedProcess);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,25 +5,31 @@ export class HighlightEngine {
|
|||||||
private styleSheet: CSSStyleSheet | null = null;
|
private styleSheet: CSSStyleSheet | null = null;
|
||||||
private wordStyleMap = new Map<string, string>();
|
private wordStyleMap = new Map<string, string>();
|
||||||
private observer: MutationObserver;
|
private observer: MutationObserver;
|
||||||
|
private isHighlighting = false;
|
||||||
|
|
||||||
constructor(private onUpdate: () => void) {
|
constructor(private onUpdate: () => void) {
|
||||||
this.observer = new MutationObserver(DOMUtils.debounce((mutations: MutationRecord[]) => {
|
this.observer = new MutationObserver(DOMUtils.debounce((mutations: MutationRecord[]) => {
|
||||||
const hasRelevantChanges = mutations.some((mutation: MutationRecord) => {
|
if (this.isHighlighting) return;
|
||||||
|
|
||||||
|
const hasContentChanges = mutations.some((mutation: MutationRecord) => {
|
||||||
|
if (mutation.type !== 'childList') return false;
|
||||||
|
|
||||||
if (mutation.target instanceof Element && mutation.target.hasAttribute('data-gh')) {
|
if (mutation.target instanceof Element && mutation.target.hasAttribute('data-gh')) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
const addedNodes = Array.from(mutation.addedNodes);
|
|
||||||
const removedNodes = Array.from(mutation.removedNodes);
|
const allNodes = [...Array.from(mutation.addedNodes), ...Array.from(mutation.removedNodes)];
|
||||||
const isOurChange = [...addedNodes, ...removedNodes].some(node =>
|
return allNodes.some(node => {
|
||||||
node instanceof Element && (node.hasAttribute('data-gh') || node.querySelector('[data-gh]'))
|
if (node.nodeType === Node.TEXT_NODE) return true;
|
||||||
);
|
if (node instanceof Element && !node.hasAttribute('data-gh')) return true;
|
||||||
return !isOurChange;
|
return false;
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
if (hasRelevantChanges) {
|
if (hasContentChanges) {
|
||||||
onUpdate();
|
this.onUpdate();
|
||||||
}
|
}
|
||||||
}, 300));
|
}, CONSTANTS.DEBOUNCE_DELAY));
|
||||||
}
|
}
|
||||||
|
|
||||||
private initializeStyleSheet(): void {
|
private initializeStyleSheet(): void {
|
||||||
@@ -37,24 +43,24 @@ export class HighlightEngine {
|
|||||||
|
|
||||||
private updateWordStyles(activeWords: ActiveWord[]): void {
|
private updateWordStyles(activeWords: ActiveWord[]): void {
|
||||||
this.initializeStyleSheet();
|
this.initializeStyleSheet();
|
||||||
|
|
||||||
while (this.styleSheet!.cssRules.length > 0) {
|
while (this.styleSheet!.cssRules.length > 0) {
|
||||||
this.styleSheet!.deleteRule(0);
|
this.styleSheet!.deleteRule(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.wordStyleMap.clear();
|
this.wordStyleMap.clear();
|
||||||
const uniqueStyles = new Map<string, string>();
|
const uniqueStyles = new Map<string, string>();
|
||||||
|
|
||||||
for (const word of activeWords) {
|
for (const word of activeWords) {
|
||||||
const styleKey = `${word.background}-${word.foreground}`;
|
const styleKey = `${word.background}-${word.foreground}`;
|
||||||
if (!uniqueStyles.has(styleKey)) {
|
if (!uniqueStyles.has(styleKey)) {
|
||||||
const className = `highlighted-word-${uniqueStyles.size}`;
|
const className = `highlighted-word-${uniqueStyles.size}`;
|
||||||
uniqueStyles.set(styleKey, className);
|
uniqueStyles.set(styleKey, className);
|
||||||
|
|
||||||
const rule = `.${className} { background: ${word.background}; color: ${word.foreground}; padding: 0 2px; }`;
|
const rule = `.${className} { background: ${word.background}; color: ${word.foreground}; padding: 0 2px; }`;
|
||||||
this.styleSheet!.insertRule(rule, this.styleSheet!.cssRules.length);
|
this.styleSheet!.insertRule(rule, this.styleSheet!.cssRules.length);
|
||||||
}
|
}
|
||||||
|
|
||||||
const lookup = word.text;
|
const lookup = word.text;
|
||||||
this.wordStyleMap.set(lookup, uniqueStyles.get(styleKey)!);
|
this.wordStyleMap.set(lookup, uniqueStyles.get(styleKey)!);
|
||||||
}
|
}
|
||||||
@@ -68,8 +74,8 @@ export class HighlightEngine {
|
|||||||
private getTextNodes(): Text[] {
|
private getTextNodes(): Text[] {
|
||||||
const textNodes: Text[] = [];
|
const textNodes: Text[] = [];
|
||||||
const walker = document.createTreeWalker(
|
const walker = document.createTreeWalker(
|
||||||
document.body,
|
document.body,
|
||||||
NodeFilter.SHOW_TEXT,
|
NodeFilter.SHOW_TEXT,
|
||||||
{
|
{
|
||||||
acceptNode: (node: Text) => {
|
acceptNode: (node: Text) => {
|
||||||
if (node.parentNode && (node.parentNode as Element).hasAttribute('data-gh')) {
|
if (node.parentNode && (node.parentNode as Element).hasAttribute('data-gh')) {
|
||||||
@@ -109,18 +115,22 @@ export class HighlightEngine {
|
|||||||
}
|
}
|
||||||
|
|
||||||
highlight(lists: HighlightList[], matchCase: boolean, matchWhole: boolean): void {
|
highlight(lists: HighlightList[], matchCase: boolean, matchWhole: boolean): void {
|
||||||
|
if (this.isHighlighting) return;
|
||||||
|
this.isHighlighting = true;
|
||||||
|
|
||||||
this.observer.disconnect();
|
this.observer.disconnect();
|
||||||
|
|
||||||
this.clearHighlightsInternal();
|
this.clearHighlightsInternal();
|
||||||
|
|
||||||
const activeWords = this.extractActiveWords(lists);
|
const activeWords = this.extractActiveWords(lists);
|
||||||
if (activeWords.length === 0) {
|
if (activeWords.length === 0) {
|
||||||
this.startObserving();
|
this.startObserving();
|
||||||
|
this.isHighlighting = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.updateWordStyles(activeWords);
|
this.updateWordStyles(activeWords);
|
||||||
|
|
||||||
const wordMap = new Map<string, ActiveWord>();
|
const wordMap = new Map<string, ActiveWord>();
|
||||||
for (const word of activeWords) {
|
for (const word of activeWords) {
|
||||||
const key = matchCase ? word.text : word.text.toLowerCase();
|
const key = matchCase ? word.text : word.text.toLowerCase();
|
||||||
@@ -129,11 +139,11 @@ export class HighlightEngine {
|
|||||||
|
|
||||||
const flags = matchCase ? 'gu' : 'giu';
|
const flags = matchCase ? 'gu' : 'giu';
|
||||||
let wordsPattern = Array.from(wordMap.keys()).map(DOMUtils.escapeRegex).join('|');
|
let wordsPattern = Array.from(wordMap.keys()).map(DOMUtils.escapeRegex).join('|');
|
||||||
|
|
||||||
if (matchWhole) {
|
if (matchWhole) {
|
||||||
wordsPattern = `(?:(?<!\\p{L})|^)(${wordsPattern})(?:(?!\\p{L})|$)`;
|
wordsPattern = `(?:(?<!\\p{L})|^)(${wordsPattern})(?:(?!\\p{L})|$)`;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const pattern = new RegExp(`(${wordsPattern})`, flags);
|
const pattern = new RegExp(`(${wordsPattern})`, flags);
|
||||||
const textNodes = this.getTextNodes();
|
const textNodes = this.getTextNodes();
|
||||||
@@ -155,6 +165,7 @@ export class HighlightEngine {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.startObserving();
|
this.startObserving();
|
||||||
|
this.isHighlighting = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private clearHighlightsInternal(): void {
|
private clearHighlightsInternal(): void {
|
||||||
@@ -166,7 +177,7 @@ export class HighlightEngine {
|
|||||||
parent.normalize();
|
parent.normalize();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (this.styleSheet && this.styleSheet.cssRules.length > 0) {
|
if (this.styleSheet && this.styleSheet.cssRules.length > 0) {
|
||||||
while (this.styleSheet.cssRules.length > 0) {
|
while (this.styleSheet.cssRules.length > 0) {
|
||||||
this.styleSheet.deleteRule(0);
|
this.styleSheet.deleteRule(0);
|
||||||
@@ -182,8 +193,6 @@ export class HighlightEngine {
|
|||||||
this.observer.observe(document.body, {
|
this.observer.observe(document.body, {
|
||||||
childList: true,
|
childList: true,
|
||||||
subtree: true,
|
subtree: true,
|
||||||
characterData: true,
|
|
||||||
// Don't observe attribute changes to avoid triggering on our own style changes
|
|
||||||
attributes: false
|
attributes: false
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user