2021-05-13 17:18:51 +00:00
|
|
|
// Copyright 2020-2021 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2020-03-20 21:00:11 +00:00
|
|
|
/* eslint-disable strict */
|
|
|
|
|
2020-11-03 00:47:46 +00:00
|
|
|
const { Menu, clipboard, nativeImage } = require('electron');
|
2020-03-20 21:00:11 +00:00
|
|
|
const osLocale = require('os-locale');
|
2020-04-15 20:44:46 +00:00
|
|
|
const { uniq } = require('lodash');
|
2020-11-03 00:47:46 +00:00
|
|
|
const url = require('url');
|
2021-05-13 17:18:51 +00:00
|
|
|
const { maybeParseUrl } = require('../ts/util/url');
|
2020-04-15 20:44:46 +00:00
|
|
|
|
|
|
|
function getLanguages(userLocale, availableLocales) {
|
|
|
|
const baseLocale = userLocale.split('-')[0];
|
|
|
|
// Attempt to find the exact locale
|
|
|
|
const candidateLocales = uniq([userLocale, baseLocale]).filter(l =>
|
|
|
|
availableLocales.includes(l)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (candidateLocales.length > 0) {
|
|
|
|
return candidateLocales;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If no languages were found then just return all locales that start with the
|
|
|
|
// base
|
|
|
|
return uniq(availableLocales.filter(l => l.startsWith(baseLocale)));
|
|
|
|
}
|
2020-03-20 21:00:11 +00:00
|
|
|
|
|
|
|
exports.setup = (browserWindow, messages) => {
|
|
|
|
const { session } = browserWindow.webContents;
|
|
|
|
const userLocale = osLocale.sync().replace(/_/g, '-');
|
2020-04-15 20:44:46 +00:00
|
|
|
const availableLocales = session.availableSpellCheckerLanguages;
|
|
|
|
const languages = getLanguages(userLocale, availableLocales);
|
2020-03-20 21:00:11 +00:00
|
|
|
console.log(`spellcheck: user locale: ${userLocale}`);
|
2020-04-15 20:44:46 +00:00
|
|
|
console.log(
|
|
|
|
'spellcheck: available spellchecker languages: ',
|
|
|
|
availableLocales
|
|
|
|
);
|
2020-03-20 21:00:11 +00:00
|
|
|
console.log('spellcheck: setting languages to: ', languages);
|
|
|
|
session.setSpellCheckerLanguages(languages);
|
|
|
|
|
|
|
|
browserWindow.webContents.on('context-menu', (_event, params) => {
|
|
|
|
const { editFlags } = params;
|
|
|
|
const isMisspelled = Boolean(params.misspelledWord);
|
|
|
|
const isLink = Boolean(params.linkURL);
|
2020-11-03 00:47:46 +00:00
|
|
|
const isImage =
|
|
|
|
params.mediaType === 'image' && params.hasImageContents && params.srcURL;
|
|
|
|
const showMenu =
|
|
|
|
params.isEditable || editFlags.canCopy || isLink || isImage;
|
2020-03-20 21:00:11 +00:00
|
|
|
|
|
|
|
// Popup editor menu
|
|
|
|
if (showMenu) {
|
|
|
|
const template = [];
|
|
|
|
|
|
|
|
if (isMisspelled) {
|
|
|
|
if (params.dictionarySuggestions.length > 0) {
|
|
|
|
template.push(
|
|
|
|
...params.dictionarySuggestions.map(label => ({
|
|
|
|
label,
|
|
|
|
click: () => {
|
|
|
|
browserWindow.webContents.replaceMisspelling(label);
|
|
|
|
},
|
|
|
|
}))
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
template.push({
|
|
|
|
label: messages.contextMenuNoSuggestions.message,
|
|
|
|
enabled: false,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
template.push({ type: 'separator' });
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.isEditable) {
|
|
|
|
if (editFlags.canUndo) {
|
|
|
|
template.push({ label: messages.editMenuUndo.message, 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: messages.editMenuRedo.message, role: 'redo' });
|
|
|
|
}
|
|
|
|
if (editFlags.canUndo || editFlags.canRedo) {
|
|
|
|
template.push({ type: 'separator' });
|
|
|
|
}
|
|
|
|
if (editFlags.canCut) {
|
|
|
|
template.push({ label: messages.editMenuCut.message, role: 'cut' });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:47:46 +00:00
|
|
|
if (editFlags.canCopy || isLink || isImage) {
|
|
|
|
let click;
|
|
|
|
let label;
|
|
|
|
|
|
|
|
if (isLink) {
|
|
|
|
click = () => {
|
|
|
|
clipboard.writeText(params.linkURL);
|
|
|
|
};
|
|
|
|
label = messages.contextMenuCopyLink.message;
|
|
|
|
} else if (isImage) {
|
|
|
|
click = () => {
|
2021-05-13 17:18:51 +00:00
|
|
|
const parsedSrcUrl = maybeParseUrl(params.srcURL);
|
|
|
|
if (!parsedSrcUrl || parsedSrcUrl.protocol !== 'file:') {
|
2020-11-03 00:47:46 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const image = nativeImage.createFromPath(
|
|
|
|
url.fileURLToPath(params.srcURL)
|
|
|
|
);
|
|
|
|
clipboard.writeImage(image);
|
|
|
|
};
|
|
|
|
label = messages.contextMenuCopyImage.message;
|
|
|
|
} else {
|
|
|
|
label = messages.editMenuCopy.message;
|
|
|
|
}
|
|
|
|
|
2020-03-20 21:00:11 +00:00
|
|
|
template.push({
|
2020-11-03 00:47:46 +00:00
|
|
|
label,
|
|
|
|
role: isLink || isImage ? undefined : 'copy',
|
|
|
|
click,
|
2020-03-20 21:00:11 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:47:46 +00:00
|
|
|
if (editFlags.canPaste && !isImage) {
|
2020-03-20 21:00:11 +00:00
|
|
|
template.push({ label: messages.editMenuPaste.message, role: 'paste' });
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:47:46 +00:00
|
|
|
if (editFlags.canPaste && !isImage) {
|
2020-03-20 21:00:11 +00:00
|
|
|
template.push({
|
|
|
|
label: messages.editMenuPasteAndMatchStyle.message,
|
|
|
|
role: 'pasteAndMatchStyle',
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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: messages.editMenuSelectAll.message,
|
|
|
|
role: 'selectall',
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const menu = Menu.buildFromTemplate(template);
|
|
|
|
menu.popup(browserWindow);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
2020-04-15 20:44:46 +00:00
|
|
|
|
|
|
|
exports.getLanguages = getLanguages;
|