signal-desktop/ts/quill/signal-clipboard/index.ts
Sidney Keese 7af2284c6b
Composition area: Only paste HTML that originated in Signal
Co-authored-by: Chris Svenningsen <chris@carbonfive.com>
2020-11-11 17:01:45 -08:00

106 lines
2.4 KiB
TypeScript

// Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import Quill from 'quill';
import Delta from 'quill-delta';
import { getTextFromOps } from '../util';
const getSelectionHTML = () => {
const selection = window.getSelection();
if (selection === null) {
return '';
}
const range = selection.getRangeAt(0);
const contents = range.cloneContents();
const div = document.createElement('div');
div.appendChild(contents);
return div.innerHTML;
};
export class SignalClipboard {
quill: Quill;
constructor(quill: Quill) {
this.quill = quill;
this.quill.root.addEventListener('copy', e => this.onCaptureCopy(e, false));
this.quill.root.addEventListener('cut', e => this.onCaptureCopy(e, true));
this.quill.root.addEventListener('paste', e => this.onCapturePaste(e));
}
onCaptureCopy(event: ClipboardEvent, isCut = false): void {
event.preventDefault();
if (event.clipboardData === null) {
return;
}
const range = this.quill.getSelection();
if (range === null) {
return;
}
const contents = this.quill.getContents(range.index, range.length);
if (contents === null) {
return;
}
const { ops } = contents;
if (ops === undefined) {
return;
}
const text = getTextFromOps(ops);
const html = getSelectionHTML();
event.clipboardData.setData('text/plain', text);
event.clipboardData.setData('text/signal', html);
if (isCut) {
this.quill.deleteText(range.index, range.length, 'user');
}
}
onCapturePaste(event: ClipboardEvent): void {
if (event.clipboardData === null) {
return;
}
this.quill.focus();
const clipboard = this.quill.getModule('clipboard');
const selection = this.quill.getSelection();
if (selection === null) {
return;
}
const text = event.clipboardData.getData('text/plain');
const html = event.clipboardData.getData('text/signal');
const { scrollTop } = this.quill.scrollingContainer;
this.quill.selection.update('silent');
if (selection) {
setTimeout(() => {
const delta = new Delta()
.retain(selection.index)
.concat(clipboard.convert(html || text));
this.quill.updateContents(delta, 'user');
this.quill.setSelection(delta.length(), 0, 'silent');
this.quill.scrollingContainer.scrollTop = scrollTop;
}, 1);
}
event.preventDefault();
}
}