You've already forked joplin
mirror of
https://github.com/laurent22/joplin.git
synced 2025-11-29 22:48:10 +02:00
Desktop, Mobile: Move several features from Extra Markdown Editor Settings into the main app (#12747)
This commit is contained in:
@@ -0,0 +1,153 @@
|
||||
import { Decoration, EditorView, WidgetType } from '@codemirror/view';
|
||||
import { SyntaxNodeRef } from '@lezer/common';
|
||||
import makeReplaceExtension from './utils/makeInlineReplaceExtension';
|
||||
|
||||
const checkboxClassName = 'cm-ext-checkbox-toggle';
|
||||
|
||||
const toggleCheckbox = (view: EditorView, linePos: number) => {
|
||||
if (linePos >= view.state.doc.length) {
|
||||
// Position out of range
|
||||
return false;
|
||||
}
|
||||
|
||||
const line = view.state.doc.lineAt(linePos);
|
||||
const checkboxMarkup = line.text.match(/\[(x|\s)\]/);
|
||||
if (!checkboxMarkup) {
|
||||
// Couldn't find the checkbox
|
||||
return false;
|
||||
}
|
||||
|
||||
const isChecked = checkboxMarkup[0] === '[x]';
|
||||
const checkboxPos = checkboxMarkup.index! + line.from;
|
||||
|
||||
view.dispatch({
|
||||
changes: [{ from: checkboxPos, to: checkboxPos + 3, insert: isChecked ? '[ ]' : '[x]' }],
|
||||
});
|
||||
return true;
|
||||
};
|
||||
|
||||
class CheckboxWidget extends WidgetType {
|
||||
public constructor(private checked: boolean, private depth: number, private label: string) {
|
||||
super();
|
||||
}
|
||||
|
||||
public eq(other: CheckboxWidget) {
|
||||
return other.checked === this.checked && other.depth === this.depth && other.label === this.label;
|
||||
}
|
||||
|
||||
private applyContainerClasses(container: HTMLElement) {
|
||||
container.classList.add(checkboxClassName);
|
||||
|
||||
for (const className of [...container.classList]) {
|
||||
if (className.startsWith('-depth-')) {
|
||||
container.classList.remove(className);
|
||||
}
|
||||
}
|
||||
|
||||
container.classList.add(`-depth-${this.depth}`);
|
||||
}
|
||||
|
||||
public toDOM(view: EditorView) {
|
||||
const container = document.createElement('span');
|
||||
|
||||
const checkbox = document.createElement('input');
|
||||
checkbox.type = 'checkbox';
|
||||
checkbox.checked = this.checked;
|
||||
checkbox.ariaLabel = this.label;
|
||||
checkbox.title = this.label;
|
||||
container.appendChild(checkbox);
|
||||
|
||||
checkbox.oninput = () => {
|
||||
toggleCheckbox(view, view.posAtDOM(container));
|
||||
};
|
||||
|
||||
this.applyContainerClasses(container);
|
||||
return container;
|
||||
}
|
||||
|
||||
public updateDOM(dom: HTMLElement): boolean {
|
||||
this.applyContainerClasses(dom);
|
||||
|
||||
const input = dom.querySelector('input');
|
||||
if (input) {
|
||||
input.checked = this.checked;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public ignoreEvent() {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
const completedTaskClassName = 'cm-md-completed-item';
|
||||
const completedListItemDecoration = Decoration.line({ class: completedTaskClassName, isFullLine: true });
|
||||
|
||||
const replaceCheckboxes = [
|
||||
EditorView.theme({
|
||||
[`& .${checkboxClassName}`]: {
|
||||
'& > input': {
|
||||
width: '1.1em',
|
||||
height: '1.1em',
|
||||
margin: '4px',
|
||||
verticalAlign: 'middle',
|
||||
},
|
||||
'&:not(.-depth-1) > input': {
|
||||
marginInlineStart: 0,
|
||||
},
|
||||
},
|
||||
[`& .${completedTaskClassName}`]: {
|
||||
opacity: 0.69,
|
||||
},
|
||||
}),
|
||||
EditorView.domEventHandlers({
|
||||
mousedown: (event) => {
|
||||
const target = event.target as Element;
|
||||
if (target.nodeName === 'INPUT' && target.parentElement?.classList?.contains(checkboxClassName)) {
|
||||
// Let the checkbox handle the event
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
},
|
||||
}),
|
||||
makeReplaceExtension({
|
||||
createDecoration: (node, state, parentTags) => {
|
||||
const markerIsChecked = (marker: SyntaxNodeRef) => {
|
||||
const content = state.doc.sliceString(marker.from, marker.to);
|
||||
return content.toLowerCase().indexOf('x') !== -1;
|
||||
};
|
||||
|
||||
if (node.name === 'TaskMarker') {
|
||||
const containerLine = state.doc.lineAt(node.from);
|
||||
const labelText = state.doc.sliceString(node.to, containerLine.to);
|
||||
|
||||
return new CheckboxWidget(markerIsChecked(node), parentTags.get('ListItem') ?? 0, labelText);
|
||||
} else if (node.name === 'Task') {
|
||||
const marker = node.node.getChild('TaskMarker');
|
||||
if (marker && markerIsChecked(marker)) {
|
||||
return completedListItemDecoration;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
},
|
||||
getDecorationRange: (node, state) => {
|
||||
if (node.name === 'TaskMarker') {
|
||||
const container = node.node.parent?.parent;
|
||||
const listMarker = container?.getChild('ListMark');
|
||||
if (!listMarker) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return [listMarker.from, node.to];
|
||||
} else if (node.name === 'Task') {
|
||||
const taskLine = state.doc.lineAt(node.from);
|
||||
return [taskLine.from];
|
||||
}
|
||||
|
||||
return null;
|
||||
},
|
||||
}),
|
||||
];
|
||||
|
||||
export default replaceCheckboxes;
|
||||
Reference in New Issue
Block a user