You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
87 lines
3.1 KiB
TypeScript
87 lines
3.1 KiB
TypeScript
import { type BrowserWindow, Menu } from 'electron';
|
|
import { sync as osLocaleSync } from 'os-locale';
|
|
import type { SetupI18nReturnType } from '../types/localizer';
|
|
|
|
export const setup = (browserWindow: BrowserWindow, i18n: SetupI18nReturnType) => {
|
|
const { session } = browserWindow.webContents;
|
|
// NOTE: we do not rely on the locale parsed by node here because we want
|
|
// to support a broader list of spell checks than what the app is localised for.
|
|
// For instance: en_AU is not a supported language on crowdin, but we still want the user to
|
|
// - if they have the dictionary installed for it - we should be able to spell check "esky", "arvo" or "bogan"
|
|
const userLocale = process.env.LANGUAGE
|
|
? process.env.LANGUAGE
|
|
: osLocaleSync().replace(/_/g, '-');
|
|
const userLocales = [userLocale, userLocale.split('-')[0], userLocale.split('_')[0]];
|
|
|
|
const available = session.availableSpellCheckerLanguages;
|
|
const languages = userLocales.filter(l => available.includes(l));
|
|
console.log(`spellcheck: userLocales: ${userLocales}`);
|
|
console.log(`spellcheck: user locale: ${userLocale}`);
|
|
console.log('spellcheck: available spellchecker languages: ', available);
|
|
console.log('spellcheck: setting languages to: ', languages);
|
|
session.setSpellCheckerLanguages(languages);
|
|
|
|
browserWindow.webContents.on('context-menu', (_event: any, params: any) => {
|
|
const { editFlags } = params;
|
|
const isMisspelled = Boolean(params.misspelledWord);
|
|
const showMenu = params.isEditable || editFlags.canCopy;
|
|
|
|
// Popup editor menu
|
|
if (showMenu) {
|
|
const template = [];
|
|
|
|
if (isMisspelled) {
|
|
if (params.dictionarySuggestions.length > 0) {
|
|
template.push(
|
|
...params.dictionarySuggestions.map((label: any) => ({
|
|
label,
|
|
click: () => {
|
|
browserWindow.webContents.replaceMisspelling(label);
|
|
},
|
|
}))
|
|
);
|
|
} else {
|
|
template.push({
|
|
label: i18n('noSuggestions'),
|
|
enabled: false,
|
|
});
|
|
}
|
|
template.push({ type: 'separator' });
|
|
}
|
|
|
|
if (params.isEditable) {
|
|
if (editFlags.canUndo) {
|
|
template.push({ label: i18n('undo'), role: 'undo' });
|
|
}
|
|
// This is only ever `true` if undo was triggered via the context menu
|
|
// (not ctrl/cmd+z)
|
|
if (editFlags.canRedo) {
|
|
template.push({ label: i18n('redo'), role: 'redo' });
|
|
}
|
|
if (editFlags.canUndo || editFlags.canRedo) {
|
|
template.push({ type: 'separator' });
|
|
}
|
|
if (editFlags.canCut) {
|
|
template.push({ label: i18n('cut'), role: 'cut' });
|
|
}
|
|
}
|
|
|
|
if (editFlags.canPaste) {
|
|
template.push({ label: i18n('paste'), role: 'paste' });
|
|
}
|
|
|
|
// Only enable select all in editors because select all in non-editors
|
|
// results in all the UI being selected
|
|
if (editFlags.canSelectAll && params.isEditable) {
|
|
template.push({
|
|
label: i18n('selectAll'),
|
|
role: 'selectall',
|
|
});
|
|
}
|
|
|
|
const menu = Menu.buildFromTemplate(template);
|
|
menu.popup({ window: browserWindow });
|
|
}
|
|
});
|
|
};
|