1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-26 18:58:21 +02:00

Electron: Fixes #663: Fixed copy, cut and paste context menu on text editor

This commit is contained in:
Laurent Cozic 2018-09-04 18:20:41 +01:00
parent eaf3eef2d3
commit cdd70230af
2 changed files with 93 additions and 11 deletions

View File

@ -62,7 +62,9 @@ class ElectronAppWrapper {
require('electron-context-menu')({ require('electron-context-menu')({
shouldShowMenu: (event, params) => { shouldShowMenu: (event, params) => {
return params.isEditable; // params.inputFieldType === 'none' when right-clicking the text editor. This is a bit of a hack to detect it because in this
// case we don't want to use the built-in context menu but a custom one.
return params.isEditable && params.inputFieldType !== 'none';
}, },
}); });

View File

@ -87,13 +87,14 @@ class NoteTextComponent extends React.Component {
this.onNoteTypeToggle_ = (event) => { if (event.noteId === this.props.noteId) this.reloadNote(this.props); } this.onNoteTypeToggle_ = (event) => { if (event.noteId === this.props.noteId) this.reloadNote(this.props); }
this.onTodoToggle_ = (event) => { if (event.noteId === this.props.noteId) this.reloadNote(this.props); } this.onTodoToggle_ = (event) => { if (event.noteId === this.props.noteId) this.reloadNote(this.props); }
this.onEditorPaste_ = async (event) => { this.onEditorPaste_ = async (event = null) => {
const formats = clipboard.availableFormats(); const formats = clipboard.availableFormats();
for (let i = 0; i < formats.length; i++) { for (let i = 0; i < formats.length; i++) {
const format = formats[i].toLowerCase(); const format = formats[i].toLowerCase();
const formatType = format.split('/')[0] const formatType = format.split('/')[0]
if (formatType === 'image') { if (formatType === 'image') {
event.preventDefault(); if (event) event.preventDefault();
const image = clipboard.readImage(); const image = clipboard.readImage();
@ -114,6 +115,32 @@ class NoteTextComponent extends React.Component {
this.setState({ lastKeys: lastKeys }); this.setState({ lastKeys: lastKeys });
} }
this.onEditorContextMenu_ = (event) => {
const menu = new Menu();
const selectedText = this.selectedText();
const clipboardText = clipboard.readText();
menu.append(new MenuItem({label: _('Cut'), enabled: !!selectedText, click: async () => {
this.editorCutText();
}}));
menu.append(new MenuItem({label: _('Copy'), enabled: !!selectedText, click: async () => {
this.editorCopyText();
}}));
menu.append(new MenuItem({label: _('Paste'), enabled: true, click: async () => {
if (clipboardText) {
this.editorPasteText();
} else {
// To handle pasting images
this.onEditorPaste_();
}
}}));
menu.popup(bridge().window());
}
this.onDrop_ = async (event) => { this.onDrop_ = async (event) => {
const files = event.dataTransfer.files; const files = event.dataTransfer.files;
if (!files || !files.length) return; if (!files || !files.length) return;
@ -466,8 +493,6 @@ class NoteTextComponent extends React.Component {
const menu = new Menu() const menu = new Menu()
console.info(itemType);
if (itemType === "image" || itemType === "resource") { if (itemType === "image" || itemType === "resource") {
const resource = await Resource.load(arg0.resourceId); const resource = await Resource.load(arg0.resourceId);
const resourcePath = Resource.fullPath(resource); const resourcePath = Resource.fullPath(resource);
@ -594,6 +619,7 @@ class NoteTextComponent extends React.Component {
this.editor_.editor.renderer.off('afterRender', this.onAfterEditorRender_); this.editor_.editor.renderer.off('afterRender', this.onAfterEditorRender_);
document.querySelector('#note-editor').removeEventListener('paste', this.onEditorPaste_, true); document.querySelector('#note-editor').removeEventListener('paste', this.onEditorPaste_, true);
document.querySelector('#note-editor').removeEventListener('keydown', this.onEditorKeyDown_); document.querySelector('#note-editor').removeEventListener('keydown', this.onEditorKeyDown_);
document.querySelector('#note-editor').removeEventListener('contextmenu', this.onEditorContextMenu_);
} }
this.editor_ = element; this.editor_ = element;
@ -622,6 +648,7 @@ class NoteTextComponent extends React.Component {
document.querySelector('#note-editor').addEventListener('paste', this.onEditorPaste_, true); document.querySelector('#note-editor').addEventListener('paste', this.onEditorPaste_, true);
document.querySelector('#note-editor').addEventListener('keydown', this.onEditorKeyDown_); document.querySelector('#note-editor').addEventListener('keydown', this.onEditorKeyDown_);
document.querySelector('#note-editor').addEventListener('contextmenu', this.onEditorContextMenu_);
const lineLeftSpaces = function(line) { const lineLeftSpaces = function(line) {
let output = ''; let output = '';
@ -894,6 +921,49 @@ class NoteTextComponent extends React.Component {
return lines[row]; return lines[row];
} }
selectedText() {
if (!this.state.note || !this.state.note.body) return '';
const selection = this.textOffsetSelection();
if (!selection || selection.start === selection.end) return '';
return this.state.note.body.substr(selection.start, selection.end - selection.start);
}
editorCopyText() {
clipboard.writeText(this.selectedText());
}
editorCutText() {
const selectedText = this.selectedText();
if (!selectedText) return;
clipboard.writeText(selectedText);
const s = this.textOffsetSelection();
if (!s || s.start === s.end) return '';
const s1 = this.state.note.body.substr(0, s.start);
const s2 = this.state.note.body.substr(s.end);
shared.noteComponent_change(this, 'body', s1 + s2);
this.updateEditorWithDelay((editor) => {
const range = this.selectionRange_;
range.setStart(range.start.row, range.start.column);
range.setEnd(range.start.row, range.start.column);
editor.getSession().getSelection().setSelectionRange(range, false);
editor.focus();
}, 10);
}
editorPasteText() {
const s = this.textOffsetSelection();
const s1 = this.state.note.body.substr(0, s.start);
const s2 = this.state.note.body.substr(s.end);
this.wrapSelectionWithStrings("", "", '', clipboard.readText());
}
selectionRangePreviousLine() { selectionRangePreviousLine() {
if (!this.selectionRange_) return ''; if (!this.selectionRange_) return '';
const row = this.selectionRange_.start.row; const row = this.selectionRange_.start.row;
@ -906,16 +976,20 @@ class NoteTextComponent extends React.Component {
return this.lineAtRow(row); return this.lineAtRow(row);
} }
wrapSelectionWithStrings(string1, string2 = '', defaultText = '') { textOffsetSelection() {
return this.selectionRange_ ? this.rangeToTextOffsets(this.selectionRange_, this.state.note.body) : null;
}
wrapSelectionWithStrings(string1, string2 = '', defaultText = '', replacementText = '') {
if (!this.rawEditor() || !this.state.note) return; if (!this.rawEditor() || !this.state.note) return;
const selection = this.selectionRange_ ? this.rangeToTextOffsets(this.selectionRange_, this.state.note.body) : null; const selection = this.textOffsetSelection();
let newBody = this.state.note.body; let newBody = this.state.note.body;
if (selection && selection.start !== selection.end) { if (selection && selection.start !== selection.end) {
const s1 = this.state.note.body.substr(0, selection.start); const s1 = this.state.note.body.substr(0, selection.start);
const s2 = this.state.note.body.substr(selection.start, selection.end - selection.start); const s2 = replacementText ? replacementText : this.state.note.body.substr(selection.start, selection.end - selection.start);
const s3 = this.state.note.body.substr(selection.end); const s3 = this.state.note.body.substr(selection.end);
newBody = s1 + string1 + s2 + string2 + s3; newBody = s1 + string1 + s2 + string2 + s3;
@ -926,6 +1000,11 @@ class NoteTextComponent extends React.Component {
end: { row: r.end.row, column: r.end.column + string1.length}, end: { row: r.end.row, column: r.end.column + string1.length},
}; };
if (replacementText) {
const diff = replacementText.length - (selection.end - selection.start);
newRange.end.column += diff;
}
this.updateEditorWithDelay((editor) => { this.updateEditorWithDelay((editor) => {
const range = this.selectionRange_; const range = this.selectionRange_;
range.setStart(newRange.start.row, newRange.start.column); range.setStart(newRange.start.row, newRange.start.column);
@ -934,19 +1013,20 @@ class NoteTextComponent extends React.Component {
editor.focus(); editor.focus();
}); });
} else { } else {
let middleText = replacementText ? replacementText : defaultText;
const textOffset = this.currentTextOffset(); const textOffset = this.currentTextOffset();
const s1 = this.state.note.body.substr(0, textOffset); const s1 = this.state.note.body.substr(0, textOffset);
const s2 = this.state.note.body.substr(textOffset); const s2 = this.state.note.body.substr(textOffset);
newBody = s1 + string1 + defaultText + string2 + s2; newBody = s1 + string1 + middleText + string2 + s2;
const p = this.textOffsetToCursorPosition(textOffset + string1.length, newBody); const p = this.textOffsetToCursorPosition(textOffset + string1.length, newBody);
const newRange = { const newRange = {
start: { row: p.row, column: p.column }, start: { row: p.row, column: p.column },
end: { row: p.row, column: p.column + defaultText.length }, end: { row: p.row, column: p.column + middleText.length },
}; };
this.updateEditorWithDelay((editor) => { this.updateEditorWithDelay((editor) => {
if (defaultText && newRange) { if (middleText && newRange) {
const range = this.selectionRange_; const range = this.selectionRange_;
range.setStart(newRange.start.row, newRange.start.column); range.setStart(newRange.start.row, newRange.start.column);
range.setEnd(newRange.end.row, newRange.end.column); range.setEnd(newRange.end.row, newRange.end.column);