Mailspring/internal_packages/composer-spellcheck/lib/spellcheck-composer-extension.es6
mbilker a1b9775b6d lint(*): correct eslint errors, warnings still exist
There are warnings in `spellcheck-composer-extension.es6` for the `while (true)`
loops.
2016-03-01 10:58:29 -05:00

194 lines
7.2 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import {DOMUtils, ComposerExtension, NylasSpellchecker} from 'nylas-exports';
import {remote} from 'electron';
const MenuItem = remote.require('menu-item');
const SpellcheckCache = {};
export default class SpellcheckComposerExtension extends ComposerExtension {
static isMisspelled(word) {
if (SpellcheckCache[word] === undefined) {
SpellcheckCache[word] = NylasSpellchecker.isMisspelled(word);
}
return SpellcheckCache[word];
}
static onContentChanged({editor}) {
SpellcheckComposerExtension.update(editor);
}
static onShowContextMenu = ({editor, menu})=> {
const selection = editor.currentSelection();
const range = DOMUtils.Mutating.getRangeAtAndSelectWord(selection, 0);
const word = range.toString();
if (SpellcheckComposerExtension.isMisspelled(word)) {
const corrections = NylasSpellchecker.getCorrectionsForMisspelling(word);
if (corrections.length > 0) {
corrections.forEach((correction)=> {
menu.append(new MenuItem({
label: correction,
click: SpellcheckComposerExtension.applyCorrection.bind(SpellcheckComposerExtension, editor, range, selection, correction),
}));
});
} else {
menu.append(new MenuItem({ label: 'No Guesses Found', enabled: false}));
}
menu.append(new MenuItem({ type: 'separator' }));
menu.append(new MenuItem({
label: 'Learn Spelling',
click: SpellcheckComposerExtension.learnSpelling.bind(SpellcheckComposerExtension, editor, word),
}));
menu.append(new MenuItem({ type: 'separator' }));
}
}
static applyCorrection = (editor, range, selection, correction)=> {
DOMUtils.Mutating.applyTextInRange(range, selection, correction);
SpellcheckComposerExtension.update(editor);
}
static learnSpelling = (editor, word)=> {
NylasSpellchecker.add(word);
delete SpellcheckCache[word];
SpellcheckComposerExtension.update(editor);
}
static update = (editor) => {
SpellcheckComposerExtension._unwrapWords(editor);
SpellcheckComposerExtension._wrapMisspelledWords(editor);
}
// Creates a shallow copy of a selection object where anchorNode / focusNode
// can be changed, and provides it to the callback provided. After the callback
// runs, it applies the new selection if `snapshot.modified` has been set.
// Note: This is different from ExposedSelection because the nodes are not cloned.
// In the callback functions, we need to check whether the anchor/focus nodes
// are INSIDE the nodes we're adjusting.
static _whileApplyingSelectionChanges = (cb)=> {
const selection = document.getSelection();
const selectionSnapshot = {
anchorNode: selection.anchorNode,
anchorOffset: selection.anchorOffset,
focusNode: selection.focusNode,
focusOffset: selection.focusOffset,
modified: false,
};
cb(selectionSnapshot);
if (selectionSnapshot.modified) {
selection.setBaseAndExtent(selectionSnapshot.anchorNode, selectionSnapshot.anchorOffset, selectionSnapshot.focusNode, selectionSnapshot.focusOffset);
}
}
// Removes all of the <spelling> nodes found in the provided `editor`.
// It normalizes the DOM after removing spelling nodes to ensure that words
// are not split between text nodes. (ie: doesn, 't => doesn't)
static _unwrapWords = (editor)=> {
SpellcheckComposerExtension._whileApplyingSelectionChanges((selectionSnapshot)=> {
const spellingNodes = editor.rootNode.querySelectorAll('spelling');
for (let ii = 0; ii < spellingNodes.length; ii++) {
const node = spellingNodes[ii];
if (selectionSnapshot.anchorNode === node) {
selectionSnapshot.anchorNode = node.firstChild;
}
if (selectionSnapshot.focusNode === node) {
selectionSnapshot.focusNode = node.firstChild;
}
selectionSnapshot.modified = true;
while (node.firstChild) {
node.parentNode.insertBefore(node.firstChild, node);
}
node.parentNode.removeChild(node);
}
});
editor.rootNode.normalize();
}
// Traverses all of the text nodes within the provided `editor`. If it finds a
// text node with a misspelled word, it splits it, wraps the misspelled word
// with a <spelling> node and updates the selection to account for the change.
static _wrapMisspelledWords = (editor) => {
SpellcheckComposerExtension._whileApplyingSelectionChanges((selectionSnapshot)=> {
const treeWalker = document.createTreeWalker(editor.rootNode, NodeFilter.SHOW_TEXT);
const nodeList = [];
while (treeWalker.nextNode()) {
nodeList.push(treeWalker.currentNode);
}
// Note: As a performance optimization, we stop spellchecking after encountering
// 30 misspelled words. This keeps the runtime of this method bounded!
let nodeMisspellingsFound = 0;
while (true) {
const node = nodeList.shift();
if ((node === undefined) || (nodeMisspellingsFound > 30)) {
break;
}
const nodeContent = node.textContent;
const nodeWordRegexp = /(\w[\w'-]*\w|\w)/g; // https://regex101.com/r/bG5yC4/1
while (true) {
const match = nodeWordRegexp.exec(nodeContent);
if ((match === null) || (nodeMisspellingsFound > 30)) {
break;
}
if (SpellcheckComposerExtension.isMisspelled(match[0])) {
// The insertion point is currently at the end of this misspelled word.
// Do not mark it until the user types a space or leaves.
if ((selectionSnapshot.focusNode === node) && (selectionSnapshot.focusOffset === match.index + match[0].length)) {
continue;
}
const matchNode = (match.index === 0) ? node : node.splitText(match.index);
const afterMatchNode = matchNode.splitText(match[0].length);
const spellingSpan = document.createElement('spelling');
spellingSpan.classList.add('misspelled');
spellingSpan.innerText = match[0];
matchNode.parentNode.replaceChild(spellingSpan, matchNode);
for (const prop of ['anchor', 'focus']) {
if (selectionSnapshot[`${prop}Node`] === node) {
if (selectionSnapshot[`${prop}Offset`] > match.index + match[0].length) {
selectionSnapshot[`${prop}Node`] = afterMatchNode;
selectionSnapshot[`${prop}Offset`] -= match.index + match[0].length;
selectionSnapshot.modified = true;
} else if (selectionSnapshot[`${prop}Offset`] > match.index) {
selectionSnapshot[`${prop}Node`] = spellingSpan.childNodes[0];
selectionSnapshot[`${prop}Offset`] -= match.index;
selectionSnapshot.modified = true;
}
}
}
nodeMisspellingsFound += 1;
nodeList.unshift(afterMatchNode);
break;
}
}
}
});
}
static finalizeSessionBeforeSending = ({session}) => {
const body = session.draft().body;
const clean = body.replace(/<\/?spelling[^>]*>/g, '');
if (body !== clean) {
return session.changes.add({body: clean});
}
return Promise.resolve();
}
}
SpellcheckComposerExtension.SpellcheckCache = SpellcheckCache;