Overlay 2.0 (#12)

This commit is contained in:
2026-03-01 02:36:51 -08:00
committed by GitHub
parent 45df3c466b
commit 44c7761c7c
397 changed files with 15139 additions and 7127 deletions

View File

@@ -6,6 +6,17 @@ import {
createPrewarmSubtitleDictionariesMainHandler,
} from './subtitle-tokenization-main-deps';
function createDeferred(): {
promise: Promise<void>;
resolve: () => void;
} {
let resolve!: () => void;
const promise = new Promise<void>((nextResolve) => {
resolve = nextResolve;
});
return { promise, resolve };
}
test('tokenizer deps builder records known-word lookups and maps readers', () => {
const calls: string[] = [];
const deps = createBuildTokenizerDepsMainHandler({
@@ -19,10 +30,12 @@ test('tokenizer deps builder records known-word lookups and maps readers', () =>
isKnownWord: (text) => text === 'known',
recordLookup: (hit) => calls.push(`lookup:${hit}`),
getKnownWordMatchMode: () => 'surface',
getNPlusOneEnabled: () => true,
getMinSentenceWordsForNPlusOne: () => 3,
getJlptLevel: () => 'N2',
getJlptEnabled: () => true,
getFrequencyDictionaryEnabled: () => true,
getFrequencyDictionaryMatchMode: () => 'surface',
getFrequencyRank: () => 5,
getYomitanGroupDebugEnabled: () => false,
getMecabTokenizer: () => null,
@@ -33,7 +46,9 @@ test('tokenizer deps builder records known-word lookups and maps readers', () =>
deps.setYomitanParserWindow(null);
deps.setYomitanParserReadyPromise(null);
deps.setYomitanParserInitPromise(null);
assert.equal(deps.getNPlusOneEnabled?.(), true);
assert.equal(deps.getMinSentenceWordsForNPlusOne?.(), 3);
assert.equal(deps.getFrequencyDictionaryMatchMode?.(), 'surface');
assert.deepEqual(calls, ['lookup:true', 'lookup:false', 'set-window', 'set-ready', 'set-init']);
});
@@ -75,3 +90,122 @@ test('dictionary prewarm runs both dictionary loaders', async () => {
await prewarm();
assert.deepEqual(calls.sort(), ['freq', 'jlpt']);
});
test('dictionary prewarm shows OSD spinner while loading and completion when loaded', async () => {
const osdMessages: string[] = [];
const clearedTimers: unknown[] = [];
let tick: (() => void) | null = null;
const jlptDeferred = createDeferred();
const freqDeferred = createDeferred();
const prewarm = createPrewarmSubtitleDictionariesMainHandler({
ensureJlptDictionaryLookup: async () => jlptDeferred.promise,
ensureFrequencyDictionaryLookup: async () => freqDeferred.promise,
shouldShowOsdNotification: () => true,
showMpvOsd: (message) => {
osdMessages.push(message);
},
setInterval: (callback) => {
tick = callback;
return 1;
},
clearInterval: (timer) => {
clearedTimers.push(timer);
},
});
const prewarmPromise = prewarm({ showLoadingOsd: true });
assert.deepEqual(osdMessages, ['Loading subtitle annotations |']);
if (!tick) {
throw new Error('expected loading spinner tick callback');
}
const tickCallback: () => void = tick;
tickCallback();
assert.deepEqual(osdMessages, [
'Loading subtitle annotations |',
'Loading subtitle annotations /',
]);
jlptDeferred.resolve();
freqDeferred.resolve();
await prewarmPromise;
assert.deepEqual(osdMessages, [
'Loading subtitle annotations |',
'Loading subtitle annotations /',
'Subtitle annotations loaded',
]);
assert.deepEqual(clearedTimers, [1]);
});
test('dictionary prewarm can show OSD while awaiting background-started load', async () => {
const osdMessages: string[] = [];
const jlptDeferred = createDeferred();
const freqDeferred = createDeferred();
const prewarm = createPrewarmSubtitleDictionariesMainHandler({
ensureJlptDictionaryLookup: async () => jlptDeferred.promise,
ensureFrequencyDictionaryLookup: async () => freqDeferred.promise,
shouldShowOsdNotification: () => true,
showMpvOsd: (message) => {
osdMessages.push(message);
},
setInterval: () => 1,
clearInterval: () => undefined,
});
const backgroundWarmupPromise = prewarm();
const tokenizationWarmupPromise = prewarm({ showLoadingOsd: true });
assert.deepEqual(osdMessages, ['Loading subtitle annotations |']);
jlptDeferred.resolve();
freqDeferred.resolve();
await backgroundWarmupPromise;
await tokenizationWarmupPromise;
assert.deepEqual(osdMessages, ['Loading subtitle annotations |', 'Subtitle annotations loaded']);
});
test('dictionary prewarm does not show OSD when notifications are disabled', async () => {
const osdMessages: string[] = [];
const prewarm = createPrewarmSubtitleDictionariesMainHandler({
ensureJlptDictionaryLookup: async () => undefined,
ensureFrequencyDictionaryLookup: async () => undefined,
shouldShowOsdNotification: () => false,
showMpvOsd: (message) => {
osdMessages.push(message);
},
});
await prewarm({ showLoadingOsd: true });
assert.deepEqual(osdMessages, []);
});
test('dictionary prewarm clears loading OSD timer even if notifications are disabled before completion', async () => {
const clearedTimers: unknown[] = [];
const jlptDeferred = createDeferred();
const freqDeferred = createDeferred();
let shouldShowNotification = true;
const prewarm = createPrewarmSubtitleDictionariesMainHandler({
ensureJlptDictionaryLookup: async () => jlptDeferred.promise,
ensureFrequencyDictionaryLookup: async () => freqDeferred.promise,
shouldShowOsdNotification: () => shouldShowNotification,
showMpvOsd: () => undefined,
setInterval: () => 'loading-timer',
clearInterval: (timer) => {
clearedTimers.push(timer);
},
});
const prewarmPromise = prewarm({ showLoadingOsd: true });
shouldShowNotification = false;
jlptDeferred.resolve();
freqDeferred.resolve();
await prewarmPromise;
assert.deepEqual(clearedTimers, ['loading-timer']);
});