93 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			93 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { bundledThemesInfo } from 'shiki';
 | |
| import { getHighlighterCore, loadWasm } from 'shiki/core';
 | |
| import darkPlus from 'shiki/themes/dark-plus.mjs';
 | |
| import { unique } from './array.js';
 | |
| import { deepClone } from './clone.js';
 | |
| import { deepMerge } from './merge.js';
 | |
| import type { Highlighter, LanguageRegistration, ThemeRegistration, ThemeRegistrationRaw } from 'shiki';
 | |
| import { ColdDeviceStorage } from '@/store.js';
 | |
| import lightTheme from '@/themes/_light.json5';
 | |
| import darkTheme from '@/themes/_dark.json5';
 | |
| 
 | |
| let _highlighter: Highlighter | null = null;
 | |
| 
 | |
| export async function getTheme(mode: 'light' | 'dark', getName: true): Promise<string>;
 | |
| export async function getTheme(mode: 'light' | 'dark', getName?: false): Promise<ThemeRegistration | ThemeRegistrationRaw>;
 | |
| export async function getTheme(mode: 'light' | 'dark', getName = false): Promise<ThemeRegistration | ThemeRegistrationRaw | string | null> {
 | |
| 	const theme = deepClone(ColdDeviceStorage.get(mode === 'light' ? 'lightTheme' : 'darkTheme'));
 | |
| 
 | |
| 	if (theme.base) {
 | |
| 		const base = [lightTheme, darkTheme].find(x => x.id === theme.base);
 | |
| 		if (base?.codeHighlighter) theme.codeHighlighter = { ...base.codeHighlighter, ...theme.codeHighlighter };
 | |
| 	}
 | |
| 
 | |
| 	if (theme.codeHighlighter) {
 | |
| 		let _res: ThemeRegistration = {};
 | |
| 		if (theme.codeHighlighter.base === '_none_') {
 | |
| 			_res = deepClone(theme.codeHighlighter.overrides);
 | |
| 		} else {
 | |
| 			const base = await bundledThemesInfo.find(t => t.id === theme.codeHighlighter!.base)?.import() ?? darkPlus;
 | |
| 			_res = deepMerge(theme.codeHighlighter.overrides ?? {}, 'default' in base ? base.default : base);
 | |
| 		}
 | |
| 		if (_res.name == null) {
 | |
| 			_res.name = theme.id;
 | |
| 		}
 | |
| 		_res.type = mode;
 | |
| 
 | |
| 		if (getName) {
 | |
| 			return _res.name;
 | |
| 		}
 | |
| 		return _res;
 | |
| 	}
 | |
| 
 | |
| 	if (getName) {
 | |
| 		return 'dark-plus';
 | |
| 	}
 | |
| 	return darkPlus;
 | |
| }
 | |
| 
 | |
| export async function getHighlighter(): Promise<Highlighter> {
 | |
| 	if (!_highlighter) {
 | |
| 		return await initHighlighter();
 | |
| 	}
 | |
| 	return _highlighter;
 | |
| }
 | |
| 
 | |
| export async function initHighlighter() {
 | |
| 	const aiScriptGrammar = await import('aiscript-vscode/aiscript/syntaxes/aiscript.tmLanguage.json');
 | |
| 
 | |
| 	await loadWasm(import('shiki/onig.wasm?init'));
 | |
| 
 | |
| 	// テーマの重複を消す
 | |
| 	const themes = unique([
 | |
| 		darkPlus,
 | |
| 		...(await Promise.all([getTheme('light'), getTheme('dark')])),
 | |
| 	]);
 | |
| 
 | |
| 	const highlighter = await getHighlighterCore({
 | |
| 		themes,
 | |
| 		langs: [
 | |
| 			import('shiki/langs/javascript.mjs'),
 | |
| 			aiScriptGrammar.default as unknown as LanguageRegistration,
 | |
| 		],
 | |
| 	});
 | |
| 
 | |
| 	ColdDeviceStorage.watch('lightTheme', async () => {
 | |
| 		const newTheme = await getTheme('light');
 | |
| 		if (newTheme.name && !highlighter.getLoadedThemes().includes(newTheme.name)) {
 | |
| 			highlighter.loadTheme(newTheme);
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	ColdDeviceStorage.watch('darkTheme', async () => {
 | |
| 		const newTheme = await getTheme('dark');
 | |
| 		if (newTheme.name && !highlighter.getLoadedThemes().includes(newTheme.name)) {
 | |
| 			highlighter.loadTheme(newTheme);
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	_highlighter = highlighter;
 | |
| 
 | |
| 	return highlighter;
 | |
| }
 | 
