Skip to content

Commit 71c5d2e

Browse files
nautatvasindresorhus
andauthoredApr 14, 2020
Add built-in support for spellchecking (#94)
Co-authored-by: Sindre Sorhus <sindresorhus@gmail.com>
1 parent 2f1bae0 commit 71c5d2e

File tree

6 files changed

+133
-14
lines changed

6 files changed

+133
-14
lines changed
 

‎fixture-menu.js

+5-1
Original file line numberDiff line numberDiff line change
@@ -33,5 +33,9 @@ contextMenu({
3333
(async () => {
3434
await app.whenReady();
3535

36-
await (new BrowserWindow()).loadFile(path.join(__dirname, 'fixture.html'));
36+
await (new BrowserWindow({
37+
webPreferences: {
38+
spellcheck: true
39+
}
40+
})).loadFile(path.join(__dirname, 'fixture.html'));
3741
})();

‎fixture.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ contextMenu({
4545

4646
await (new BrowserWindow({
4747
webPreferences: {
48-
nodeIntegration: true
48+
spellcheck: true
4949
}
5050
})).loadFile(path.join(__dirname, 'fixture.html'));
5151
})();

‎index.d.ts

+42-8
Original file line numberDiff line numberDiff line change
@@ -12,11 +12,25 @@ import {
1212
declare namespace contextMenu {
1313
interface Labels {
1414
/**
15-
The placeholder `{selection}` will be replaced by the currently selected text.
15+
@default 'Correct Automatically'
16+
*/
17+
readonly correctAutomatically?: string;
18+
19+
/**
20+
@default 'Learn Spelling'
21+
*/
22+
readonly learnSpelling?: string;
1623

24+
/**
25+
The placeholder `{selection}` will be replaced by the currently selected text.
1726
@default 'Look Up “{selection}”'
1827
*/
1928
readonly lookUpSelection?: string;
29+
30+
/**
31+
@default 'Search with Google'
32+
*/
33+
readonly searchWithGoogle?: string;
2034

2135
/**
2236
@default 'Cut'
@@ -80,7 +94,10 @@ declare namespace contextMenu {
8094

8195
interface Actions {
8296
readonly separator: () => MenuItemConstructorOptions;
97+
readonly correctAutomatically: (options: ActionOptions) => MenuItemConstructorOptions;
98+
readonly learnSpelling: (options: ActionOptions) => MenuItemConstructorOptions;
8399
readonly lookUpSelection: (options: ActionOptions) => MenuItemConstructorOptions;
100+
readonly searchWithGoogle: (options: ActionOptions) => MenuItemConstructorOptions;
84101
readonly cut: (options: ActionOptions) => MenuItemConstructorOptions;
85102
readonly copy: (options: ActionOptions) => MenuItemConstructorOptions;
86103
readonly paste: (options: ActionOptions) => MenuItemConstructorOptions;
@@ -129,6 +146,13 @@ declare namespace contextMenu {
129146
*/
130147
readonly showLookUpSelection?: boolean;
131148

149+
/**
150+
Show the `Search with Google` menu item when right-clicking text on macOS.
151+
152+
@default true
153+
*/
154+
readonly showSearchWithGoogle?: boolean;
155+
132156
/**
133157
Show the `Copy Image` menu item when right-clicking on an image.
134158
@@ -219,13 +243,16 @@ declare namespace contextMenu {
219243
The following options are ignored when `menu` is used:
220244
221245
- `showLookUpSelection`
246+
- `showSearchWithGoogle`
222247
- `showCopyImage`
223248
- `showCopyImageAddress`
224249
- `showSaveImageAs`
225250
- `showInspectElement`
226251
- `showServices`
252+
253+
To get spellchecking, “Correct Automatically”, and “Learn Spelling” in the menu, please enable the `spellcheck` preference in browser window: `new BrowserWindow({webPreferences: {spellcheck: true}})`
227254
228-
@default [defaultActions.cut(), defaultActions.copy(), defaultActions.paste(), defaultActions.separator(), defaultActions.saveImage(), defaultActions.saveImageAs(), defaultActions.copyLink(), defaultActions.copyImage(), defaultActions.copyImageAddress(), defaultActions.separator(), defaultActions.copyLink(), defaultActions.separator(), defaultActions.inspect()]
255+
@default [...dictionarySuggestions, defaultActions.separator(), defaultActions.correctAutomatically(), defaultActions.separator(), defaultActions.learnSpelling(), defaultActions.separator(), defaultActions.lookUpSelection(), defaultActions.separator(),defaultActions.searchWithGoogle(), defaultActions.cut(), defaultActions.copy(), defaultActions.paste(), defaultActions.separator(), defaultActions.saveImage(), defaultActions.saveImageAs(), defaultActions.copyLink(), defaultActions.copyImage(), defaultActions.copyImageAddress(), defaultActions.separator(), defaultActions.copyLink(), defaultActions.separator(), defaultActions.inspect()]
229256
*/
230257
readonly menu?: (
231258
defaultActions: Actions,
@@ -246,17 +273,24 @@ import {app, BrowserWindow} from 'electron';
246273
import contextMenu = require('electron-context-menu');
247274
248275
contextMenu({
249-
prepend: (params, browserWindow) => [{
250-
label: 'Rainbow',
251-
// Only show it when right-clicking images
252-
visible: params.mediaType === 'image'
253-
}]
276+
prepend: (defaultActions, params, browserWindow) => [
277+
{
278+
label: 'Rainbow',
279+
// Only show it when right-clicking images
280+
visible: params.mediaType === 'image'
281+
}
282+
]
254283
});
255284
256285
let mainWindow;
257286
(async () => {
258287
await app.whenReady();
259-
mainWindow = new BrowserWindow();
288+
289+
mainWindow = new BrowserWindow({
290+
webPreferences: {
291+
spellcheck: true
292+
}
293+
});
260294
});
261295
```
262296
*/

‎index.js

+62
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,24 @@ const create = (win, options) => {
4141

4242
const defaultActions = {
4343
separator: () => ({type: 'separator'}),
44+
correctAutomatically: decorateMenuItem({
45+
id: 'correctAutomatically',
46+
label: 'Correct Spelling Automatically',
47+
visible: props.isEditable && hasText && props.misspelledWord && props.dictionarySuggestions.length > 0,
48+
click() {
49+
const target = webContents(win);
50+
target.insertText(props.dictionarySuggestions[0]);
51+
}
52+
}),
53+
learnSpelling: decorateMenuItem({
54+
id: 'learnSpelling',
55+
label: 'Learn Spelling',
56+
visible: props.isEditable && hasText && props.misspelledWord,
57+
click() {
58+
const target = webContents(win);
59+
target.session.addWordToSpellCheckerDictionary(props.misspelledWord);
60+
}
61+
}),
4462
lookUpSelection: decorateMenuItem({
4563
id: 'lookUpSelection',
4664
label: 'Look Up “{selection}”',
@@ -51,6 +69,16 @@ const create = (win, options) => {
5169
}
5270
}
5371
}),
72+
searchWithGoogle: decorateMenuItem({
73+
id: 'searchWithGoogle',
74+
label: 'Search with Google',
75+
visible: hasText,
76+
click() {
77+
const url = new URL('https://www.google.com/search');
78+
url.searchParams.set('q', props.selectionText);
79+
electron.shell.openExternal(url.toString());
80+
}
81+
}),
5482
cut: decorateMenuItem({
5583
id: 'cut',
5684
label: 'Cut',
@@ -173,10 +201,44 @@ const create = (win, options) => {
173201

174202
const shouldShowInspectElement = typeof options.showInspectElement === 'boolean' ? options.showInspectElement : isDev;
175203

204+
function word(suggestion) {
205+
return {
206+
id: 'dictionarySuggestions',
207+
label: suggestion,
208+
visible: props.isEditable && hasText && props.misspelledWord,
209+
click(menuItem) {
210+
const target = webContents(win);
211+
target.insertText(menuItem.label);
212+
}
213+
};
214+
}
215+
216+
let dictionarySuggestions = [];
217+
if (hasText && props.misspelledWord && props.dictionarySuggestions.length > 0) {
218+
dictionarySuggestions = props.dictionarySuggestions.map(word);
219+
} else {
220+
dictionarySuggestions.push(
221+
{
222+
id: 'dictionarySuggestions',
223+
label: 'No Guesses Found',
224+
visible: hasText && props.misspelledWord,
225+
enabled: false
226+
}
227+
);
228+
}
229+
176230
let menuTemplate = [
231+
defaultActions.separator(),
232+
...dictionarySuggestions,
233+
defaultActions.separator(),
234+
defaultActions.correctAutomatically(),
235+
defaultActions.separator(),
236+
defaultActions.learnSpelling(),
177237
defaultActions.separator(),
178238
options.showLookUpSelection !== false && defaultActions.lookUpSelection(),
179239
defaultActions.separator(),
240+
options.showSearchWithGoogle !== false && defaultActions.searchWithGoogle(),
241+
defaultActions.separator(),
180242
defaultActions.cut(),
181243
defaultActions.copy(),
182244
defaultActions.paste(),

‎package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@
3737
"devDependencies": {
3838
"@types/node": "^12.0.10",
3939
"ava": "^2.1.0",
40-
"electron": "^7.1.1",
40+
"electron": "^8.0.0",
4141
"tsd": "^0.10.0",
4242
"xo": "^0.25.3"
4343
},

‎readme.md

+22-3
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,12 @@ contextMenu({
4343
let mainWindow;
4444
(async () => {
4545
await app.whenReady();
46-
mainWindow = new BrowserWindow();
46+
47+
mainWindow = new BrowserWindow(
48+
webPreferences: {
49+
spellcheck: true
50+
}
51+
);
4752
})();
4853
```
4954

@@ -90,6 +95,13 @@ Default: `true`
9095

9196
Show the `Look Up {selection}` menu item when right-clicking text on macOS.
9297

98+
#### showSearchWithGoogle
99+
100+
Type: `boolean`\
101+
Default: `true`
102+
103+
Show the `Search with Google` menu item when right-clicking text on macOS.
104+
93105
#### showCopyImage
94106

95107
Type: `boolean`\
@@ -184,6 +196,8 @@ Even though you include an action, it will still only be shown/enabled when appr
184196

185197
`MenuItem` labels may contain the the placeholder `{selection}` which will be replaced by the currently selected text as described in [`options.labels`](#labels).
186198

199+
To get spellchecking, “Correct Automatically”, and “Learn Spelling” in the menu, please enable the `spellcheck` preference in browser window: `new BrowserWindow({webPreferences: {spellcheck: true}})`
200+
187201
The following options are ignored when `menu` is used:
188202

189203
- `showLookUpSelection`
@@ -192,11 +206,16 @@ The following options are ignored when `menu` is used:
192206
- `showSaveImageAs`
193207
- `showInspectElement`
194208
- `showServices`
195-
209+
- `showSearchWithGoogle`
210+
196211
Default actions:
197212

213+
- `spellCheck`
214+
- `correctAutomatically`
215+
- `learnSpelling`
198216
- `separator`
199217
- `lookUpSelection`
218+
- `searchWithGoogle`
200219
- `cut`
201220
- `copy`
202221
- `paste`
@@ -208,7 +227,7 @@ Default actions:
208227
- `inspect`
209228
- `services`
210229

211-
Example:
230+
Example for actions:
212231

213232
```js
214233
{

0 commit comments

Comments
 (0)
Please sign in to comment.