mirror of
https://github.com/obsqrbtz/goose-highlighter.git
synced 2026-04-08 20:19:06 +03:00
139 lines
4.4 KiB
TypeScript
139 lines
4.4 KiB
TypeScript
import { HighlightList, MessageData } from '../types.js';
|
|
import { StorageService } from '../services/StorageService.js';
|
|
import { MessageService } from '../services/MessageService.js';
|
|
import { HighlightEngine } from './HighlightEngine.js';
|
|
|
|
export class ContentScript {
|
|
private lists: HighlightList[] = [];
|
|
private isGlobalHighlightEnabled = true;
|
|
private exceptionsList: string[] = [];
|
|
private isCurrentSiteException = false;
|
|
private matchCase = false;
|
|
private matchWhole = false;
|
|
private highlightEngine: HighlightEngine;
|
|
private isProcessing = false;
|
|
|
|
constructor() {
|
|
this.highlightEngine = new HighlightEngine(() => this.processHighlights());
|
|
this.initialize();
|
|
}
|
|
|
|
private async initialize(): Promise<void> {
|
|
await this.loadSettings();
|
|
this.setupMessageListener();
|
|
this.processHighlights();
|
|
}
|
|
|
|
private async loadSettings(): Promise<void> {
|
|
const data = await StorageService.get([
|
|
'lists',
|
|
'globalHighlightEnabled',
|
|
'matchCaseEnabled',
|
|
'matchWholeEnabled',
|
|
'exceptionsList'
|
|
]);
|
|
|
|
this.lists = data.lists || [];
|
|
this.isGlobalHighlightEnabled = data.globalHighlightEnabled ?? true;
|
|
this.matchCase = data.matchCaseEnabled ?? false;
|
|
this.matchWhole = data.matchWholeEnabled ?? false;
|
|
this.exceptionsList = data.exceptionsList || [];
|
|
this.isCurrentSiteException = this.checkCurrentSiteException();
|
|
}
|
|
|
|
private checkCurrentSiteException(): boolean {
|
|
const currentHostname = window.location.hostname;
|
|
return this.exceptionsList.includes(currentHostname);
|
|
}
|
|
|
|
private setupMessageListener(): void {
|
|
MessageService.onMessage((message: MessageData, sender: any, sendResponse: (response?: any) => void) => {
|
|
switch (message.type) {
|
|
case 'WORD_LIST_UPDATED':
|
|
this.handleWordListUpdate();
|
|
return false;
|
|
case 'GLOBAL_TOGGLE_UPDATED':
|
|
this.handleGlobalToggleUpdate(message.enabled!);
|
|
return false;
|
|
case 'MATCH_OPTIONS_UPDATED':
|
|
this.handleMatchOptionsUpdate(message.matchCase!, message.matchWhole!);
|
|
return false;
|
|
case 'EXCEPTIONS_LIST_UPDATED':
|
|
this.handleExceptionsUpdate();
|
|
return false;
|
|
case 'GET_PAGE_HIGHLIGHTS':
|
|
this.handleGetPageHighlights(sendResponse);
|
|
return true;
|
|
case 'SCROLL_TO_HIGHLIGHT':
|
|
this.handleScrollToHighlight(message.word!, message.index!);
|
|
return false;
|
|
}
|
|
return false;
|
|
});
|
|
}
|
|
|
|
|
|
private async handleWordListUpdate(): Promise<void> {
|
|
const data = await StorageService.get(['lists']);
|
|
this.lists = data.lists || [];
|
|
this.processHighlights();
|
|
}
|
|
|
|
private handleGlobalToggleUpdate(enabled: boolean): void {
|
|
this.isGlobalHighlightEnabled = enabled;
|
|
this.processHighlights();
|
|
}
|
|
|
|
private handleMatchOptionsUpdate(matchCase: boolean, matchWhole: boolean): void {
|
|
this.matchCase = matchCase;
|
|
this.matchWhole = matchWhole;
|
|
this.processHighlights();
|
|
}
|
|
|
|
private async handleExceptionsUpdate(): Promise<void> {
|
|
const data = await StorageService.get(['exceptionsList']);
|
|
this.exceptionsList = data.exceptionsList || [];
|
|
this.isCurrentSiteException = this.checkCurrentSiteException();
|
|
this.processHighlights();
|
|
}
|
|
|
|
private processHighlights(): void {
|
|
if (this.isProcessing) return;
|
|
this.isProcessing = true;
|
|
|
|
try {
|
|
if (!this.isGlobalHighlightEnabled || this.isCurrentSiteException) {
|
|
this.highlightEngine.clearHighlights();
|
|
this.highlightEngine.stopObserving();
|
|
return;
|
|
}
|
|
|
|
this.highlightEngine.highlight(this.lists, this.matchCase, this.matchWhole);
|
|
} finally {
|
|
this.isProcessing = false;
|
|
}
|
|
}
|
|
|
|
private handleGetPageHighlights(sendResponse: (response: any) => void): void {
|
|
const activeWords: ActiveWord[] = [];
|
|
|
|
for (const list of this.lists) {
|
|
if (!list.active) continue;
|
|
for (const word of list.words) {
|
|
if (!word.active) continue;
|
|
activeWords.push({
|
|
text: word.wordStr,
|
|
background: word.background || list.background,
|
|
foreground: word.foreground || list.foreground
|
|
});
|
|
}
|
|
}
|
|
|
|
const highlights = this.highlightEngine.getPageHighlights(activeWords);
|
|
sendResponse({ highlights });
|
|
}
|
|
|
|
private handleScrollToHighlight(word: string, index: number): void {
|
|
this.highlightEngine.scrollToHighlight(word, index);
|
|
}
|
|
} |