mirror of
https://github.com/laurent22/joplin.git
synced 2024-12-24 10:27:10 +02:00
Electron: Added more toolbar icons and upgraded Font Awesome
This commit is contained in:
parent
c748281d86
commit
5d9c2c0904
4
ElectronClient/app/css/font-awesome.min.css
vendored
4
ElectronClient/app/css/font-awesome.min.css
vendored
File diff suppressed because one or more lines are too long
5
ElectronClient/app/css/fontawesome-all.min.css
vendored
Normal file
5
ElectronClient/app/css/fontawesome-all.min.css
vendored
Normal file
File diff suppressed because one or more lines are too long
Binary file not shown.
@ -26,6 +26,7 @@ const mimeUtils = require('lib/mime-utils.js').mime;
|
||||
const ArrayUtils = require('lib/ArrayUtils');
|
||||
const urlUtils = require('lib/urlUtils');
|
||||
const dialogs = require('./dialogs');
|
||||
const markdownUtils = require('lib/markdownUtils');
|
||||
|
||||
require('brace/mode/markdown');
|
||||
// https://ace.c9.io/build/kitchen-sink.html
|
||||
@ -54,6 +55,7 @@ class NoteTextComponent extends React.Component {
|
||||
// to automatically set the title.
|
||||
newAndNoTitleChangeNoteId: null,
|
||||
bodyHtml: '',
|
||||
lastKeys: [],
|
||||
};
|
||||
|
||||
this.lastLoadedNoteId_ = null;
|
||||
@ -102,6 +104,13 @@ class NoteTextComponent extends React.Component {
|
||||
}
|
||||
}
|
||||
|
||||
this.onEditorKeyDown_ = (event) => {
|
||||
const lastKeys = this.state.lastKeys.slice();
|
||||
lastKeys.push(event.key);
|
||||
while (lastKeys.length > 2) lastKeys.splice(0, 1);
|
||||
this.setState({ lastKeys: lastKeys });
|
||||
}
|
||||
|
||||
this.onDrop_ = async (event) => {
|
||||
const files = event.dataTransfer.files;
|
||||
if (!files || !files.length) return;
|
||||
@ -117,30 +126,21 @@ class NoteTextComponent extends React.Component {
|
||||
await this.commandAttachFile(filesToAttach);
|
||||
}
|
||||
|
||||
this.aceEditor_selectionChange = (selection) => {
|
||||
const ranges = selection.getAllRanges();
|
||||
|
||||
const updateSelectionRange = () => {
|
||||
const ranges = this.rawEditor().getSelection().getAllRanges();
|
||||
if (!ranges || !ranges.length || !this.state.note) {
|
||||
this.setState({
|
||||
//selection: null,
|
||||
selectionRange: null,
|
||||
});
|
||||
return;
|
||||
this.setState({ selectionRange: null });
|
||||
} else {
|
||||
this.setState({ selectionRange: ranges[0] });
|
||||
}
|
||||
}
|
||||
|
||||
this.setState({ selectionRange: ranges[0] });
|
||||
this.aceEditor_selectionChange = (selection) => {
|
||||
updateSelectionRange();
|
||||
}
|
||||
|
||||
// let newSelection = {
|
||||
// start: this.cursorPositionToTextOffsets(range.start, this.state.note.body),
|
||||
// end: this.cursorPositionToTextOffsets(range.end, this.state.note.body),
|
||||
// };
|
||||
|
||||
// if (newSelection.start === newSelection.end) newSelection = null;
|
||||
|
||||
// this.setState({
|
||||
// selection: newSelection,
|
||||
// selectionRange: range,
|
||||
// });
|
||||
this.aceEditor_focus = (event) => {
|
||||
updateSelectionRange();
|
||||
}
|
||||
}
|
||||
|
||||
@ -345,6 +345,7 @@ class NoteTextComponent extends React.Component {
|
||||
lastSavedNote: Object.assign({}, note),
|
||||
webviewReady: webviewReady,
|
||||
folder: parentFolder,
|
||||
lastKeys: [],
|
||||
};
|
||||
|
||||
if (!note) {
|
||||
@ -544,6 +545,7 @@ class NoteTextComponent extends React.Component {
|
||||
if (this.editor_) {
|
||||
this.editor_.editor.renderer.off('afterRender', this.onAfterEditorRender_);
|
||||
document.querySelector('#note-editor').removeEventListener('paste', this.onEditorPaste_, true);
|
||||
document.querySelector('#note-editor').removeEventListener('keydown', this.onEditorKeyDown_);
|
||||
}
|
||||
|
||||
this.editor_ = element;
|
||||
@ -571,6 +573,24 @@ class NoteTextComponent extends React.Component {
|
||||
}
|
||||
|
||||
document.querySelector('#note-editor').addEventListener('paste', this.onEditorPaste_, true);
|
||||
document.querySelector('#note-editor').addEventListener('keydown', this.onEditorKeyDown_);
|
||||
|
||||
// Disable Markdown auto-completion (eg. auto-adding a dash after a line with a dash.
|
||||
// https://github.com/ajaxorg/ace/issues/2754
|
||||
const that = this; // The "this" within the function below refers to something else
|
||||
this.editor_.editor.getSession().getMode().getNextLineIndent = function(state, line) {
|
||||
const ls = that.state.lastKeys;
|
||||
if (ls.length >= 2 && ls[ls.length - 1] === 'Enter' && ls[ls.length - 2] === 'Enter') return this.$getIndent(line);
|
||||
|
||||
if (line.indexOf('- [ ] ') === 0 || line.indexOf('- [x] ') === 0 || line.indexOf('- [X] ') === 0) return '- [ ] ';
|
||||
if (line.indexOf('- ') === 0) return '- ';
|
||||
if (line.indexOf('* ') === 0) return '* ';
|
||||
|
||||
const bulletNumber = markdownUtils.olLineNumber(line);
|
||||
if (bulletNumber) return (bulletNumber + 1) + '. ';
|
||||
|
||||
return this.$getIndent(line);
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
@ -752,28 +772,31 @@ class NoteTextComponent extends React.Component {
|
||||
}, 10);
|
||||
}
|
||||
|
||||
selectionRangePreviousLine() {
|
||||
if (!this.state.selectionRange) return '';
|
||||
lineAtRow(row) {
|
||||
if (!this.state.note) return '';
|
||||
|
||||
const range = this.state.selectionRange
|
||||
const body = this.state.note.body
|
||||
|
||||
if (!range) return '';
|
||||
const row = range.start.row;
|
||||
if (row <= 0) return '';
|
||||
const lines = body.split('\n');
|
||||
if (lines.length <= row - 1) return '';
|
||||
return lines[row - 1];
|
||||
if (row < 0 || row >= lines.length) return '';
|
||||
return lines[row];
|
||||
}
|
||||
|
||||
wrapSelectionWithStrings(string1, string2 = '') {
|
||||
selectionRangePreviousLine() {
|
||||
if (!this.state.selectionRange) return '';
|
||||
const row = this.state.selectionRange.start.row;
|
||||
return this.lineAtRow(row - 1);
|
||||
}
|
||||
|
||||
selectionRangeCurrentLine() {
|
||||
if (!this.state.selectionRange) return '';
|
||||
const row = this.state.selectionRange.start.row;
|
||||
return this.lineAtRow(row);
|
||||
}
|
||||
|
||||
wrapSelectionWithStrings(string1, string2 = '', defaultText = '') {
|
||||
if (!this.rawEditor() || !this.state.note) return;
|
||||
|
||||
const selection = this.state.selectionRange ? this.rangeToTextOffsets(this.state.selectionRange, this.state.note.body) : null;
|
||||
|
||||
console.info(this.state.selectionRange);
|
||||
|
||||
let newBody = this.state.note.body;
|
||||
|
||||
if (selection && selection.start !== selection.end) {
|
||||
@ -800,12 +823,25 @@ class NoteTextComponent extends React.Component {
|
||||
const cursorPos = this.cursorPosition();
|
||||
const s1 = this.state.note.body.substr(0, cursorPos);
|
||||
const s2 = this.state.note.body.substr(cursorPos);
|
||||
newBody = s1 + string1 + string2 + s2;
|
||||
newBody = s1 + string1 + defaultText + string2 + s2;
|
||||
|
||||
const r = this.state.selectionRange;
|
||||
const newRange = !r ? null : {
|
||||
start: { row: r.start.row, column: r.start.column + string1.length },
|
||||
end: { row: r.end.row, column: r.start.column + string1.length + defaultText.length },
|
||||
};
|
||||
|
||||
this.updateEditorWithDelay((editor) => {
|
||||
if (defaultText && newRange) {
|
||||
const range = this.state.selectionRange;
|
||||
range.setStart(newRange.start.row, newRange.start.column);
|
||||
range.setEnd(newRange.end.row, newRange.end.column);
|
||||
editor.getSession().getSelection().setSelectionRange(range, false);
|
||||
} else {
|
||||
for (let i = 0; i < string1.length; i++) {
|
||||
editor.getSession().getSelection().moveCursorRight();
|
||||
}
|
||||
}
|
||||
editor.focus();
|
||||
}, 10);
|
||||
}
|
||||
@ -815,7 +851,6 @@ class NoteTextComponent extends React.Component {
|
||||
}
|
||||
|
||||
commandTextBold() {
|
||||
// TODO: Insert default text and select it
|
||||
this.wrapSelectionWithStrings('**', '**', _('strong text'));
|
||||
}
|
||||
|
||||
@ -823,23 +858,39 @@ class NoteTextComponent extends React.Component {
|
||||
this.wrapSelectionWithStrings('*', '*', _('emphasized text'));
|
||||
}
|
||||
|
||||
commandTextCheckbox() {
|
||||
const previousLine = this.selectionRangePreviousLine()
|
||||
let newLine = '\n';
|
||||
// Normally we want to add a newline before a checkbox to separate it from the previous text
|
||||
// however if the previous line is already checkbox we don't add a new line, so that the new
|
||||
// one is part of the same group.
|
||||
if (previousLine.indexOf('- [') === 0 && this.state.selectionRange && this.state.selectionRange.start.column === 0) newLine = '';
|
||||
this.wrapSelectionWithStrings(newLine + '- [ ] ');
|
||||
}
|
||||
|
||||
commandTextCode() {
|
||||
this.wrapSelectionWithStrings('`', '`');
|
||||
}
|
||||
|
||||
// commandTextListUl() {
|
||||
// this.wrapSelectionWithStrings('\n- ');
|
||||
// }
|
||||
addListItem(item) {
|
||||
const currentLine = this.selectionRangeCurrentLine();
|
||||
let newLine = '\n'
|
||||
if (!currentLine) newLine = '';
|
||||
this.wrapSelectionWithStrings(newLine + item);
|
||||
}
|
||||
|
||||
commandTextCheckbox() {
|
||||
this.addListItem('- [ ] ');
|
||||
}
|
||||
|
||||
commandTextListUl() {
|
||||
this.addListItem('- ');
|
||||
}
|
||||
|
||||
commandTextListOl() {
|
||||
let bulletNumber = markdownUtils.olLineNumber(this.selectionRangeCurrentLine());
|
||||
if (!bulletNumber) bulletNumber = markdownUtils.olLineNumber(this.selectionRangePreviousLine());
|
||||
if (!bulletNumber) bulletNumber = 0;
|
||||
this.addListItem((bulletNumber + 1) + '. ');
|
||||
}
|
||||
|
||||
commandTextHeading() {
|
||||
this.addListItem('## ');
|
||||
}
|
||||
|
||||
commandTextHorizontalRule() {
|
||||
this.addListItem('* * *');
|
||||
}
|
||||
|
||||
async commandTextLink() {
|
||||
const url = await dialogs.prompt(_('Insert Hyperlink'));
|
||||
@ -892,15 +943,13 @@ class NoteTextComponent extends React.Component {
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Hyperlink'),
|
||||
iconName: 'fa-link',
|
||||
onClick: () => { return this.commandTextLink(); },
|
||||
type: 'separator',
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Checkbox'),
|
||||
iconName: 'fa-check-square',
|
||||
onClick: () => { return this.commandTextCheckbox(); },
|
||||
tooltip: _('Hyperlink'),
|
||||
iconName: 'fa-link',
|
||||
onClick: () => { return this.commandTextLink(); },
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
@ -915,6 +964,44 @@ class NoteTextComponent extends React.Component {
|
||||
onClick: () => { return this.commandAttachFile(); },
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
type: 'separator',
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Numbered List'),
|
||||
iconName: 'fa-list-ol',
|
||||
onClick: () => { return this.commandTextListOl(); },
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Bulleted List'),
|
||||
iconName: 'fa-list-ul',
|
||||
onClick: () => { return this.commandTextListUl(); },
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Checkbox'),
|
||||
iconName: 'fa-check-square',
|
||||
onClick: () => { return this.commandTextCheckbox(); },
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Heading'),
|
||||
iconName: 'fa-heading',
|
||||
onClick: () => { return this.commandTextHeading(); },
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Horizontal Rule'),
|
||||
iconName: 'fa-ellipsis-h',
|
||||
onClick: () => { return this.commandTextHorizontalRule(); },
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
type: 'separator',
|
||||
});
|
||||
|
||||
toolbarItems.push({
|
||||
tooltip: _('Tags'),
|
||||
iconName: 'fa-tags',
|
||||
@ -923,7 +1010,7 @@ class NoteTextComponent extends React.Component {
|
||||
|
||||
if (note.is_todo) {
|
||||
const item = {
|
||||
iconName: 'fa-clock-o',
|
||||
iconName: 'fa-bell',
|
||||
enabled: !note.todo_completed,
|
||||
onClick: () => { return this.commandSetAlarm(); },
|
||||
}
|
||||
@ -1119,6 +1206,7 @@ class NoteTextComponent extends React.Component {
|
||||
onChange={(body) => { this.aceEditor_change(body) }}
|
||||
showPrintMargin={false}
|
||||
onSelectionChange={this.aceEditor_selectionChange}
|
||||
onFocus={this.aceEditor_focus}
|
||||
|
||||
// Disable warning: "Automatically scrolling cursor into view after
|
||||
// selection change this will be disabled in the next version set
|
||||
|
@ -4,6 +4,7 @@ const { reg } = require('lib/registry.js');
|
||||
const { themeStyle } = require('../theme.js');
|
||||
const { _ } = require('lib/locale.js');
|
||||
const ToolbarButton = require('./ToolbarButton.min.js');
|
||||
const ToolbarSpace = require('./ToolbarSpace.min.js');
|
||||
|
||||
class ToolbarComponent extends React.Component {
|
||||
|
||||
@ -25,17 +26,17 @@ class ToolbarComponent extends React.Component {
|
||||
key += o.title ? o.title : '';
|
||||
const itemType = !('type' in o) ? 'button' : o.type;
|
||||
|
||||
if (!key) key = o.type + '_' + i;
|
||||
|
||||
const props = Object.assign({
|
||||
key: key,
|
||||
theme: this.props.theme,
|
||||
}, o);
|
||||
|
||||
if (itemType === 'button') {
|
||||
itemComps.push(<ToolbarButton
|
||||
{...props}
|
||||
/>);
|
||||
} else if (itemType === 'text') {
|
||||
|
||||
itemComps.push(<ToolbarButton {...props} />);
|
||||
} else if (itemType === 'separator') {
|
||||
itemComps.push(<ToolbarSpace {...props} />);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -5,24 +5,9 @@ const { themeStyle } = require('../theme.js');
|
||||
class ToolbarButton extends React.Component {
|
||||
|
||||
render() {
|
||||
//const style = this.props.style;
|
||||
const theme = themeStyle(this.props.theme);
|
||||
|
||||
const style = {
|
||||
height: theme.toolbarHeight,
|
||||
minWidth: theme.toolbarHeight,
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
paddingLeft: theme.headerButtonHPadding,
|
||||
paddingRight: theme.headerButtonHPadding,
|
||||
color: theme.color,
|
||||
textDecoration: 'none',
|
||||
fontFamily: theme.fontFamily,
|
||||
fontSize: theme.fontSize,
|
||||
boxSizing: 'border-box',
|
||||
cursor: 'default',
|
||||
justifyContent: 'center',
|
||||
};
|
||||
const style = Object.assign({}, theme.toolbarStyle);
|
||||
|
||||
const title = this.props.title ? this.props.title : '';
|
||||
const tooltip = this.props.tooltip ? this.props.tooltip : title;
|
||||
|
22
ElectronClient/app/gui/ToolbarSpace.jsx
Normal file
22
ElectronClient/app/gui/ToolbarSpace.jsx
Normal file
@ -0,0 +1,22 @@
|
||||
const React = require('react');
|
||||
const { connect } = require('react-redux');
|
||||
const { themeStyle } = require('../theme.js');
|
||||
|
||||
class ToolbarSpace extends React.Component {
|
||||
|
||||
render() {
|
||||
const theme = themeStyle(this.props.theme);
|
||||
const style = Object.assign({}, theme.toolbarStyle);
|
||||
style.minWidth = style.height / 2;
|
||||
|
||||
return (
|
||||
<span
|
||||
style={style}
|
||||
>
|
||||
</span>
|
||||
);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
module.exports = ToolbarSpace;
|
@ -4,7 +4,7 @@
|
||||
<meta charset="UTF-8">
|
||||
<title>Joplin</title>
|
||||
<link rel="stylesheet" href="style.css">
|
||||
<link rel="stylesheet" href="css/font-awesome.min.css">
|
||||
<link rel="stylesheet" href="css/fontawesome-all.min.css">
|
||||
<link rel="stylesheet" href="node_modules/react-datetime/css/react-datetime.css">
|
||||
<link rel="stylesheet" href="node_modules/smalltalk/css/smalltalk.css">
|
||||
<style>
|
||||
|
@ -82,6 +82,22 @@ globalStyle.h1Style.fontSize *= 1.5;
|
||||
globalStyle.h2Style = Object.assign({}, globalStyle.textStyle);
|
||||
globalStyle.h2Style.fontSize *= 1.3;
|
||||
|
||||
globalStyle.toolbarStyle = {
|
||||
height: globalStyle.toolbarHeight,
|
||||
minWidth: globalStyle.toolbarHeight,
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
paddingLeft: globalStyle.headerButtonHPadding,
|
||||
paddingRight: globalStyle.headerButtonHPadding,
|
||||
color: globalStyle.color,
|
||||
textDecoration: 'none',
|
||||
fontFamily: globalStyle.fontFamily,
|
||||
fontSize: globalStyle.fontSize,
|
||||
boxSizing: 'border-box',
|
||||
cursor: 'default',
|
||||
justifyContent: 'center',
|
||||
};
|
||||
|
||||
let themeCache_ = {};
|
||||
|
||||
function themeStyle(theme) {
|
||||
|
BIN
ElectronClient/app/webfonts/fa-solid-900.woff2
Normal file
BIN
ElectronClient/app/webfonts/fa-solid-900.woff2
Normal file
Binary file not shown.
@ -32,6 +32,11 @@ const markdownUtils = {
|
||||
return output;
|
||||
},
|
||||
|
||||
olLineNumber(line) {
|
||||
const match = line.match(/^(\d+)\.(\s.*|)$/);
|
||||
return match ? Number(match[1]) : 0;
|
||||
},
|
||||
|
||||
};
|
||||
|
||||
module.exports = markdownUtils;
|
Loading…
Reference in New Issue
Block a user