@dxos/ui-editor 0.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +8 -0
- package/README.md +21 -0
- package/package.json +121 -0
- package/src/defaults.ts +34 -0
- package/src/extensions/annotations.ts +55 -0
- package/src/extensions/autocomplete/autocomplete.ts +151 -0
- package/src/extensions/autocomplete/index.ts +8 -0
- package/src/extensions/autocomplete/match.ts +46 -0
- package/src/extensions/autocomplete/placeholder.ts +117 -0
- package/src/extensions/autocomplete/typeahead.ts +87 -0
- package/src/extensions/automerge/automerge.test.tsx +76 -0
- package/src/extensions/automerge/automerge.ts +105 -0
- package/src/extensions/automerge/cursor.ts +28 -0
- package/src/extensions/automerge/defs.ts +31 -0
- package/src/extensions/automerge/index.ts +5 -0
- package/src/extensions/automerge/sync.ts +79 -0
- package/src/extensions/automerge/update-automerge.ts +50 -0
- package/src/extensions/automerge/update-codemirror.ts +115 -0
- package/src/extensions/autoscroll.ts +165 -0
- package/src/extensions/awareness/awareness-provider.ts +127 -0
- package/src/extensions/awareness/awareness.ts +315 -0
- package/src/extensions/awareness/index.ts +6 -0
- package/src/extensions/blast.ts +363 -0
- package/src/extensions/blocks.ts +131 -0
- package/src/extensions/bookmarks.ts +77 -0
- package/src/extensions/comments.ts +579 -0
- package/src/extensions/debug.ts +15 -0
- package/src/extensions/dnd.ts +39 -0
- package/src/extensions/factories.ts +284 -0
- package/src/extensions/focus.ts +36 -0
- package/src/extensions/folding.ts +63 -0
- package/src/extensions/hashtag.ts +68 -0
- package/src/extensions/index.ts +34 -0
- package/src/extensions/json.ts +57 -0
- package/src/extensions/listener.ts +32 -0
- package/src/extensions/markdown/action.ts +117 -0
- package/src/extensions/markdown/bundle.ts +105 -0
- package/src/extensions/markdown/changes.test.ts +26 -0
- package/src/extensions/markdown/changes.ts +149 -0
- package/src/extensions/markdown/debug.ts +44 -0
- package/src/extensions/markdown/decorate.ts +622 -0
- package/src/extensions/markdown/formatting.test.ts +498 -0
- package/src/extensions/markdown/formatting.ts +1265 -0
- package/src/extensions/markdown/highlight.ts +183 -0
- package/src/extensions/markdown/image.ts +118 -0
- package/src/extensions/markdown/index.ts +13 -0
- package/src/extensions/markdown/link.ts +50 -0
- package/src/extensions/markdown/parser.test.ts +75 -0
- package/src/extensions/markdown/styles.ts +135 -0
- package/src/extensions/markdown/table.ts +150 -0
- package/src/extensions/mention.ts +41 -0
- package/src/extensions/modal.ts +24 -0
- package/src/extensions/modes.ts +41 -0
- package/src/extensions/outliner/commands.ts +270 -0
- package/src/extensions/outliner/editor.test.ts +33 -0
- package/src/extensions/outliner/editor.ts +184 -0
- package/src/extensions/outliner/index.ts +7 -0
- package/src/extensions/outliner/menu.ts +128 -0
- package/src/extensions/outliner/outliner.test.ts +100 -0
- package/src/extensions/outliner/outliner.ts +167 -0
- package/src/extensions/outliner/selection.ts +50 -0
- package/src/extensions/outliner/tree.test.ts +168 -0
- package/src/extensions/outliner/tree.ts +317 -0
- package/src/extensions/preview/index.ts +5 -0
- package/src/extensions/preview/preview.ts +193 -0
- package/src/extensions/replacer.test.ts +75 -0
- package/src/extensions/replacer.ts +93 -0
- package/src/extensions/scrolling.ts +189 -0
- package/src/extensions/selection.ts +100 -0
- package/src/extensions/state.ts +7 -0
- package/src/extensions/submit.ts +62 -0
- package/src/extensions/tags/extended-markdown.test.ts +263 -0
- package/src/extensions/tags/extended-markdown.ts +78 -0
- package/src/extensions/tags/index.ts +7 -0
- package/src/extensions/tags/streamer.ts +243 -0
- package/src/extensions/tags/xml-tags.ts +507 -0
- package/src/extensions/tags/xml-util.test.ts +48 -0
- package/src/extensions/tags/xml-util.ts +93 -0
- package/src/extensions/typewriter.ts +68 -0
- package/src/index.ts +14 -0
- package/src/styles/index.ts +7 -0
- package/src/styles/markdown.ts +26 -0
- package/src/styles/theme.ts +293 -0
- package/src/styles/tokens.ts +17 -0
- package/src/types/index.ts +5 -0
- package/src/types/types.ts +32 -0
- package/src/util/cursor.ts +56 -0
- package/src/util/debug.ts +56 -0
- package/src/util/decorations.ts +21 -0
- package/src/util/dom.ts +36 -0
- package/src/util/facet.ts +13 -0
- package/src/util/index.ts +10 -0
- package/src/util/util.ts +29 -0
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { markdown, markdownLanguage } from '@codemirror/lang-markdown';
|
|
6
|
+
import { EditorState } from '@codemirror/state';
|
|
7
|
+
import { beforeEach, describe, test } from 'vitest';
|
|
8
|
+
|
|
9
|
+
import { type Range } from '../../types';
|
|
10
|
+
import { join } from '../../util';
|
|
11
|
+
|
|
12
|
+
import { type Item, listItemToString, outlinerTree, treeFacet } from './tree';
|
|
13
|
+
|
|
14
|
+
const lines = [
|
|
15
|
+
'- [ ] 1',
|
|
16
|
+
'- [ ] 2',
|
|
17
|
+
' - [ ] 2.1',
|
|
18
|
+
' - [ ] 2.2',
|
|
19
|
+
' - 2.2.1',
|
|
20
|
+
' - 2.2.2',
|
|
21
|
+
' - 2.2.3',
|
|
22
|
+
' - [ ] 2.3',
|
|
23
|
+
'- [ ] 3',
|
|
24
|
+
];
|
|
25
|
+
|
|
26
|
+
const getPos = (line: number) => {
|
|
27
|
+
return lines.slice(0, line).reduce((acc, line) => acc + line.length + 1, 0);
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
const extensions = [markdown({ base: markdownLanguage }), outlinerTree()];
|
|
31
|
+
|
|
32
|
+
describe('tree (boundary conditions)', () => {
|
|
33
|
+
test('empty', ({ expect }) => {
|
|
34
|
+
const state = EditorState.create({ doc: join(''), extensions });
|
|
35
|
+
const tree = state.facet(treeFacet);
|
|
36
|
+
expect(tree).to.exist;
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
test('content range', ({ expect }) => {
|
|
40
|
+
const state = EditorState.create({ doc: '- [ ] A', extensions });
|
|
41
|
+
const tree = state.facet(treeFacet);
|
|
42
|
+
console.log(JSON.stringify(tree, null, 2));
|
|
43
|
+
expect(tree.toJSON()).to.deep.eq({
|
|
44
|
+
type: 'root',
|
|
45
|
+
index: -1,
|
|
46
|
+
level: -1,
|
|
47
|
+
lineRange: { from: 0, to: -1 },
|
|
48
|
+
contentRange: { from: 0, to: -1 },
|
|
49
|
+
children: [
|
|
50
|
+
{
|
|
51
|
+
type: 'task',
|
|
52
|
+
index: 0,
|
|
53
|
+
level: 0,
|
|
54
|
+
lineRange: { from: 0, to: 7 },
|
|
55
|
+
contentRange: { from: 6, to: 7 },
|
|
56
|
+
children: [],
|
|
57
|
+
},
|
|
58
|
+
],
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
const item = tree.find(0);
|
|
62
|
+
expect(item?.contentRange).to.include({ from: 6, to: state.doc.length });
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
test('empty continuation', ({ expect }) => {
|
|
66
|
+
const state = EditorState.create({ doc: join('- [ ] A', ' '), extensions });
|
|
67
|
+
const tree = state.facet(treeFacet);
|
|
68
|
+
tree.traverse((item, level) => {
|
|
69
|
+
console.log(listItemToString(item, level));
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
describe('tree (advanced)', () => {
|
|
75
|
+
let state: EditorState;
|
|
76
|
+
|
|
77
|
+
beforeEach(() => {
|
|
78
|
+
state = EditorState.create({ doc: join(...lines), extensions });
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
test('traverse', ({ expect }) => {
|
|
82
|
+
const tree = state.facet(treeFacet);
|
|
83
|
+
let count = 0;
|
|
84
|
+
tree.traverse((item, level) => {
|
|
85
|
+
console.log(listItemToString(item, level));
|
|
86
|
+
count++;
|
|
87
|
+
});
|
|
88
|
+
expect(count).to.eq(9);
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
test('continguous', ({ expect }) => {
|
|
92
|
+
const tree = state.facet(treeFacet);
|
|
93
|
+
const ranges: Range[] = [];
|
|
94
|
+
tree.traverse((item) => {
|
|
95
|
+
console.log(listItemToString(item));
|
|
96
|
+
ranges.push(item.lineRange);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
// Check no gaps between ranges.
|
|
100
|
+
expect(ranges[0].from).toBe(0);
|
|
101
|
+
expect(ranges[ranges.length - 1].to).toBe(state.doc.length);
|
|
102
|
+
for (let i = 0; i < ranges.length - 1; i++) {
|
|
103
|
+
const current = ranges[i];
|
|
104
|
+
const next = ranges[i + 1];
|
|
105
|
+
expect(current.to + 1).to.eq(next.from);
|
|
106
|
+
}
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
test('find', ({ expect }) => {
|
|
110
|
+
const tree = state.facet(treeFacet);
|
|
111
|
+
expect(tree.find(0)).to.include({ type: 'task' });
|
|
112
|
+
expect(tree.find(state.doc.length)).to.include({ type: 'task' });
|
|
113
|
+
|
|
114
|
+
expect(tree.find(getPos(1))).to.include({ type: 'task' });
|
|
115
|
+
expect(tree.find(getPos(1))).to.eq(tree.find(getPos(1) + 4));
|
|
116
|
+
expect(tree.find(getPos(5))).to.include({ type: 'bullet' });
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
test('siblings', ({ expect }) => {
|
|
120
|
+
const tree = state.facet(treeFacet);
|
|
121
|
+
const items: Item[] = [];
|
|
122
|
+
tree.traverse((item) => {
|
|
123
|
+
items.push(item);
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
expect(items[0].nextSibling).toBe(items[1]);
|
|
127
|
+
expect(items[1].prevSibling).toBe(items[0]);
|
|
128
|
+
|
|
129
|
+
expect(items[1].nextSibling).toBe(items[8]);
|
|
130
|
+
expect(items[8].prevSibling).toBe(items[1]);
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
test('next/previous', ({ expect }) => {
|
|
134
|
+
const tree = state.facet(treeFacet);
|
|
135
|
+
const items: Item[] = [];
|
|
136
|
+
tree.traverse((item) => {
|
|
137
|
+
items.push(item);
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
expect(tree.prev(items[0])).not.to.exist;
|
|
141
|
+
expect(tree.next(items[items.length - 1])).not.to.exist;
|
|
142
|
+
|
|
143
|
+
for (let i = 0; i < items.length - 1; i++) {
|
|
144
|
+
const current = items[i];
|
|
145
|
+
const next = items[i + 1];
|
|
146
|
+
expect(tree.next(current)?.index).toEqual(next.index);
|
|
147
|
+
expect(tree.prev(next)?.index).toEqual(current.index);
|
|
148
|
+
}
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
test('lastDescendant', ({ expect }) => {
|
|
152
|
+
const tree = state.facet(treeFacet);
|
|
153
|
+
{
|
|
154
|
+
const item = tree.find(getPos(0))!;
|
|
155
|
+
expect(tree.lastDescendant(item).index).to.eq(item.index);
|
|
156
|
+
}
|
|
157
|
+
{
|
|
158
|
+
const item = tree.find(getPos(1))!;
|
|
159
|
+
const last = tree.find(getPos(7))!;
|
|
160
|
+
expect(tree.lastDescendant(item).index).to.eq(last.index);
|
|
161
|
+
}
|
|
162
|
+
{
|
|
163
|
+
const item = tree.find(getPos(3))!;
|
|
164
|
+
const last = tree.find(getPos(6))!;
|
|
165
|
+
expect(tree.lastDescendant(item).index).to.eq(last.index);
|
|
166
|
+
}
|
|
167
|
+
});
|
|
168
|
+
});
|
|
@@ -0,0 +1,317 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { syntaxTree } from '@codemirror/language';
|
|
6
|
+
import { type EditorState, type Extension, StateField, type Transaction } from '@codemirror/state';
|
|
7
|
+
import { Facet } from '@codemirror/state';
|
|
8
|
+
import { type SyntaxNode } from '@lezer/common';
|
|
9
|
+
|
|
10
|
+
import { invariant } from '@dxos/invariant';
|
|
11
|
+
|
|
12
|
+
import { type Range } from '../../types';
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Represents a single item in the tree.
|
|
16
|
+
*/
|
|
17
|
+
export interface Item {
|
|
18
|
+
type: 'root' | 'bullet' | 'task' | 'unknown';
|
|
19
|
+
index: number;
|
|
20
|
+
level: number;
|
|
21
|
+
node: SyntaxNode;
|
|
22
|
+
parent?: Item;
|
|
23
|
+
nextSibling?: Item;
|
|
24
|
+
prevSibling?: Item;
|
|
25
|
+
children: Item[];
|
|
26
|
+
/**
|
|
27
|
+
* Actual range.
|
|
28
|
+
* Starts at the start of the line containing the item and ends at the end of the line before the
|
|
29
|
+
* first child or next sibling.
|
|
30
|
+
*/
|
|
31
|
+
lineRange: Range;
|
|
32
|
+
/**
|
|
33
|
+
* Range of the editable content.
|
|
34
|
+
* This doesn't include the list or task marker or indentation.
|
|
35
|
+
*/
|
|
36
|
+
contentRange: Range;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export const itemToJSON = ({ type, index, level, lineRange, contentRange, children }: Item): any => {
|
|
40
|
+
return { type, index, level, lineRange, contentRange, children: children.map(itemToJSON) };
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Tree assumes the entire document is a single contiguous well-formed hierarchy of markdown LiteItem nodes.
|
|
45
|
+
*/
|
|
46
|
+
export class Tree implements Item {
|
|
47
|
+
type: Item['type'] = 'root';
|
|
48
|
+
index = -1;
|
|
49
|
+
level = -1;
|
|
50
|
+
node: Item['node'];
|
|
51
|
+
lineRange: Item['lineRange'];
|
|
52
|
+
contentRange: Item['contentRange'];
|
|
53
|
+
children: Item['children'] = [];
|
|
54
|
+
|
|
55
|
+
constructor(node: SyntaxNode) {
|
|
56
|
+
this.node = node;
|
|
57
|
+
this.lineRange = { from: node.from, to: node.to };
|
|
58
|
+
this.contentRange = this.lineRange;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
toJSON() {
|
|
62
|
+
return itemToJSON(this);
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
get root(): Item {
|
|
66
|
+
return this;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
traverse<T = any>(cb: (item: Item, level: number) => T | void): T | undefined;
|
|
70
|
+
traverse<T = any>(item: Item, cb: (item: Item, level: number) => T | void): T | undefined;
|
|
71
|
+
traverse<T = any>(
|
|
72
|
+
itemOrCb: Item | ((item: Item, level: number) => T | undefined | void),
|
|
73
|
+
maybeCb?: (item: Item, level: number) => T | undefined | void,
|
|
74
|
+
): T | undefined {
|
|
75
|
+
if (typeof itemOrCb === 'function') {
|
|
76
|
+
return traverse<T>(this, itemOrCb);
|
|
77
|
+
} else {
|
|
78
|
+
return traverse<T>(itemOrCb, maybeCb!);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Return the closest item.
|
|
84
|
+
*/
|
|
85
|
+
find(pos: number): Item | undefined {
|
|
86
|
+
return this.traverse<Item>((item) => (item.lineRange.from <= pos && item.lineRange.to >= pos ? item : undefined));
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Return the first child, next sibling, or parent's next sibling.
|
|
91
|
+
*/
|
|
92
|
+
next(item: Item, enter = true): Item | undefined {
|
|
93
|
+
if (enter && item.children.length > 0) {
|
|
94
|
+
return item.children[0];
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
if (item.nextSibling) {
|
|
98
|
+
return item.nextSibling;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
if (item.parent) {
|
|
102
|
+
return this.next(item.parent, false);
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
return undefined;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/**
|
|
109
|
+
* Return the previous sibling, or parent.
|
|
110
|
+
*/
|
|
111
|
+
prev(item: Item): Item | undefined {
|
|
112
|
+
if (item.prevSibling) {
|
|
113
|
+
return this.lastDescendant(item.prevSibling);
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
return item.parent?.type === 'root' ? undefined : item.parent;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* Return the last descendant of the item, or the item itself if it has no children.
|
|
121
|
+
*/
|
|
122
|
+
lastDescendant(item: Item): Item {
|
|
123
|
+
return item.children.length > 0 ? this.lastDescendant(item.children.at(-1)!) : item;
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
export const getRange = (tree: Tree, item: Item): [number, number] => {
|
|
128
|
+
const lastDescendant = tree.lastDescendant(item);
|
|
129
|
+
return [item.lineRange.from, lastDescendant.lineRange.to];
|
|
130
|
+
};
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* Traverse the tree, calling the callback for each item.
|
|
134
|
+
* If the callback returns a value, the traversal is stopped and the value is returned.
|
|
135
|
+
*/
|
|
136
|
+
export const traverse = <T = any>(root: Item, cb: (item: Item, level: number) => T | void): T | undefined => {
|
|
137
|
+
const t = (item: Item, level: number): T | undefined => {
|
|
138
|
+
if (item.type !== 'root') {
|
|
139
|
+
const value = cb(item, level);
|
|
140
|
+
if (value != null) {
|
|
141
|
+
return value;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
for (const child of item.children) {
|
|
146
|
+
const value = t(child, level + 1);
|
|
147
|
+
if (value != null) {
|
|
148
|
+
return value;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
return undefined;
|
|
153
|
+
};
|
|
154
|
+
|
|
155
|
+
return t(root, root.type === 'root' ? -1 : 0);
|
|
156
|
+
};
|
|
157
|
+
|
|
158
|
+
export const getListItemContent = (state: EditorState, item: Item): string => {
|
|
159
|
+
return state.doc.sliceString(item.contentRange.from, item.contentRange.to);
|
|
160
|
+
};
|
|
161
|
+
|
|
162
|
+
export const listItemToString = (item: Item, level = 0) => {
|
|
163
|
+
const indent = ' '.repeat(level);
|
|
164
|
+
const data = {
|
|
165
|
+
i: item.index,
|
|
166
|
+
n: item.nextSibling?.index ?? '∅',
|
|
167
|
+
p: item.prevSibling?.index ?? '∅',
|
|
168
|
+
level: item.level,
|
|
169
|
+
node: format([item.node.from, item.node.to]),
|
|
170
|
+
line: format([item.lineRange.from, item.lineRange.to]),
|
|
171
|
+
content: format([item.contentRange.from, item.contentRange.to]),
|
|
172
|
+
};
|
|
173
|
+
|
|
174
|
+
return `${indent}${item.type[0].toUpperCase()}(${Object.entries(data)
|
|
175
|
+
.map(([k, v]) => `${k}=${v}`)
|
|
176
|
+
.join(', ')})`;
|
|
177
|
+
};
|
|
178
|
+
|
|
179
|
+
const format = (value: any) =>
|
|
180
|
+
JSON.stringify(value, (key: string, value: any) => {
|
|
181
|
+
if (typeof value === 'number') {
|
|
182
|
+
return value.toString().padStart(3, ' ');
|
|
183
|
+
}
|
|
184
|
+
return value;
|
|
185
|
+
}).replaceAll('"', '');
|
|
186
|
+
|
|
187
|
+
export const treeFacet = Facet.define<Tree, Tree>({
|
|
188
|
+
combine: (values) => values[0],
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
export type TreeOptions = {};
|
|
192
|
+
|
|
193
|
+
/**
|
|
194
|
+
* Creates a shadow tree of `ListItem` nodes whenever the document changes.
|
|
195
|
+
* This adds overhead relative to the markdown AST, but allows for efficient traversal of the list items.
|
|
196
|
+
* NOTE: Requires markdown parser to be enabled.
|
|
197
|
+
*/
|
|
198
|
+
export const outlinerTree = (_options: TreeOptions = {}): Extension => {
|
|
199
|
+
const buildTree = (state: EditorState): Tree => {
|
|
200
|
+
let tree: Tree | undefined;
|
|
201
|
+
let parent: Item | undefined;
|
|
202
|
+
let current: Item | undefined;
|
|
203
|
+
let prev: Item | undefined;
|
|
204
|
+
let level = -1;
|
|
205
|
+
let index = -1;
|
|
206
|
+
|
|
207
|
+
// Array to track previous siblings at each level.
|
|
208
|
+
const prevSiblings: (Item | undefined)[] = [];
|
|
209
|
+
|
|
210
|
+
syntaxTree(state).iterate({
|
|
211
|
+
enter: (node) => {
|
|
212
|
+
switch (node.name) {
|
|
213
|
+
case 'Document': {
|
|
214
|
+
tree = new Tree(node.node);
|
|
215
|
+
current = tree;
|
|
216
|
+
break;
|
|
217
|
+
}
|
|
218
|
+
case 'BulletList': {
|
|
219
|
+
invariant(current);
|
|
220
|
+
parent = current;
|
|
221
|
+
if (current) {
|
|
222
|
+
current.lineRange.to = current.node.from;
|
|
223
|
+
}
|
|
224
|
+
prevSiblings[++level] = undefined;
|
|
225
|
+
break;
|
|
226
|
+
}
|
|
227
|
+
case 'ListItem': {
|
|
228
|
+
invariant(parent);
|
|
229
|
+
|
|
230
|
+
// Include all content up to the next sibling or the end of the document.
|
|
231
|
+
const nextSibling = node.node.nextSibling ?? node.node.parent?.nextSibling;
|
|
232
|
+
const docRange: Range = {
|
|
233
|
+
from: state.doc.lineAt(node.from).from,
|
|
234
|
+
to: nextSibling ? nextSibling.from - 1 : state.doc.length,
|
|
235
|
+
};
|
|
236
|
+
|
|
237
|
+
current = {
|
|
238
|
+
type: 'unknown',
|
|
239
|
+
index: ++index,
|
|
240
|
+
level,
|
|
241
|
+
node: node.node,
|
|
242
|
+
lineRange: docRange,
|
|
243
|
+
contentRange: { ...docRange },
|
|
244
|
+
parent,
|
|
245
|
+
prevSibling: prevSiblings[level],
|
|
246
|
+
children: [],
|
|
247
|
+
};
|
|
248
|
+
|
|
249
|
+
// Update sibling refs.
|
|
250
|
+
if (current.prevSibling) {
|
|
251
|
+
current.prevSibling.nextSibling = current;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
// Update previous siblings array at current level.
|
|
255
|
+
prevSiblings[level] = current;
|
|
256
|
+
|
|
257
|
+
// Update previous item (not sibling).
|
|
258
|
+
if (prev) {
|
|
259
|
+
prev.lineRange.to = prev.contentRange.to = current.lineRange.from - 1;
|
|
260
|
+
}
|
|
261
|
+
prev = current;
|
|
262
|
+
|
|
263
|
+
// Update parent.
|
|
264
|
+
parent.children.push(current);
|
|
265
|
+
if (parent.lineRange.to === parent.node.from) {
|
|
266
|
+
parent.lineRange.to = parent.contentRange.to = current.lineRange.from - 1;
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
break;
|
|
270
|
+
}
|
|
271
|
+
case 'ListMark': {
|
|
272
|
+
invariant(current);
|
|
273
|
+
current.type = 'bullet';
|
|
274
|
+
current.contentRange.from = node.from + '- '.length;
|
|
275
|
+
break;
|
|
276
|
+
}
|
|
277
|
+
case 'Task': {
|
|
278
|
+
invariant(current);
|
|
279
|
+
current.type = 'task';
|
|
280
|
+
break;
|
|
281
|
+
}
|
|
282
|
+
case 'TaskMarker': {
|
|
283
|
+
invariant(current);
|
|
284
|
+
current.contentRange.from = node.from + '[ ] '.length;
|
|
285
|
+
break;
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
},
|
|
289
|
+
leave: (node) => {
|
|
290
|
+
if (node.name === 'BulletList') {
|
|
291
|
+
invariant(parent);
|
|
292
|
+
prevSiblings[level--] = undefined;
|
|
293
|
+
parent = parent.parent;
|
|
294
|
+
}
|
|
295
|
+
},
|
|
296
|
+
});
|
|
297
|
+
|
|
298
|
+
invariant(tree);
|
|
299
|
+
return tree;
|
|
300
|
+
};
|
|
301
|
+
|
|
302
|
+
return [
|
|
303
|
+
StateField.define<Tree | undefined>({
|
|
304
|
+
create: (state) => {
|
|
305
|
+
return buildTree(state);
|
|
306
|
+
},
|
|
307
|
+
update: (value: Tree | undefined, tr: Transaction) => {
|
|
308
|
+
if (!tr.docChanged) {
|
|
309
|
+
return value;
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
return buildTree(tr.state);
|
|
313
|
+
},
|
|
314
|
+
provide: (field) => treeFacet.from(field),
|
|
315
|
+
}),
|
|
316
|
+
];
|
|
317
|
+
};
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2023 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { syntaxTree } from '@codemirror/language';
|
|
6
|
+
import { type EditorState, type Extension, RangeSetBuilder, StateField } from '@codemirror/state';
|
|
7
|
+
import { Decoration, type DecorationSet, EditorView, WidgetType } from '@codemirror/view';
|
|
8
|
+
import { type SyntaxNode } from '@lezer/common';
|
|
9
|
+
|
|
10
|
+
export type PreviewBlock = {
|
|
11
|
+
link: PreviewLinkRef;
|
|
12
|
+
el: HTMLElement;
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
export type PreviewLinkRef = {
|
|
16
|
+
suggest?: boolean;
|
|
17
|
+
block?: boolean;
|
|
18
|
+
label: string;
|
|
19
|
+
ref: string;
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
export type PreviewLinkTarget = {
|
|
23
|
+
label: string;
|
|
24
|
+
text?: string;
|
|
25
|
+
object?: any;
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
export type PreviewOptions = {
|
|
29
|
+
addBlockContainer?: (block: PreviewBlock) => void;
|
|
30
|
+
removeBlockContainer?: (block: PreviewBlock) => void;
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Create preview decorations.
|
|
35
|
+
*/
|
|
36
|
+
export const preview = (options: PreviewOptions = {}): Extension => {
|
|
37
|
+
return [
|
|
38
|
+
// NOTE: Atomic block decorations must be created from a state field, now a widget, otherwise it results in the following error:
|
|
39
|
+
// "Block decorations may not be specified via plugins".
|
|
40
|
+
StateField.define<DecorationSet>({
|
|
41
|
+
create: (state) => buildDecorations(state, options),
|
|
42
|
+
update: (decorations, tr) => {
|
|
43
|
+
if (tr.docChanged) {
|
|
44
|
+
return buildDecorations(tr.state, options);
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
return decorations.map(tr.changes);
|
|
48
|
+
},
|
|
49
|
+
provide: (field) => [
|
|
50
|
+
EditorView.decorations.from(field),
|
|
51
|
+
EditorView.atomicRanges.of((view) => view.state.field(field)),
|
|
52
|
+
],
|
|
53
|
+
}),
|
|
54
|
+
];
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Echo references are represented as markdown reference links.
|
|
59
|
+
* https://www.markdownguide.org/basic-syntax/#reference-style-links
|
|
60
|
+
*/
|
|
61
|
+
const buildDecorations = (state: EditorState, options: PreviewOptions): DecorationSet => {
|
|
62
|
+
const builder = new RangeSetBuilder<Decoration>();
|
|
63
|
+
|
|
64
|
+
syntaxTree(state).iterate({
|
|
65
|
+
enter: (node) => {
|
|
66
|
+
switch (node.name) {
|
|
67
|
+
//
|
|
68
|
+
// Inline widget.
|
|
69
|
+
// [Label](dxn:echo:123)
|
|
70
|
+
//
|
|
71
|
+
case 'Link': {
|
|
72
|
+
const link = getLinkRef(state, node.node);
|
|
73
|
+
if (link) {
|
|
74
|
+
builder.add(
|
|
75
|
+
node.from,
|
|
76
|
+
node.to,
|
|
77
|
+
Decoration.replace({
|
|
78
|
+
widget: new PreviewInlineWidget(options, link),
|
|
79
|
+
side: 1,
|
|
80
|
+
}),
|
|
81
|
+
);
|
|
82
|
+
}
|
|
83
|
+
return false;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
//
|
|
87
|
+
// Block widget (transclusion).
|
|
88
|
+
// 
|
|
89
|
+
//
|
|
90
|
+
case 'Image': {
|
|
91
|
+
if (options.addBlockContainer && options.removeBlockContainer) {
|
|
92
|
+
const link = getLinkRef(state, node.node);
|
|
93
|
+
if (link) {
|
|
94
|
+
builder.add(
|
|
95
|
+
node.from,
|
|
96
|
+
node.to,
|
|
97
|
+
Decoration.replace({
|
|
98
|
+
block: true,
|
|
99
|
+
widget: new PreviewBlockWidget(options, link),
|
|
100
|
+
}),
|
|
101
|
+
);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
return false;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
},
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
return builder.finish();
|
|
111
|
+
};
|
|
112
|
+
|
|
113
|
+
/**
|
|
114
|
+
* Link references.
|
|
115
|
+
* [Label](dxn:echo:123) Inline reference
|
|
116
|
+
*  Block reference
|
|
117
|
+
*/
|
|
118
|
+
export const getLinkRef = (state: EditorState, node: SyntaxNode): PreviewLinkRef | undefined => {
|
|
119
|
+
const mark = node.getChildren('LinkMark');
|
|
120
|
+
const urlNode = node.getChild('URL');
|
|
121
|
+
if (mark && urlNode) {
|
|
122
|
+
const url = state.sliceDoc(urlNode.from, urlNode.to);
|
|
123
|
+
if (url.startsWith('dxn:')) {
|
|
124
|
+
const label = state.sliceDoc(mark[0].to, mark[1].from);
|
|
125
|
+
return {
|
|
126
|
+
block: state.sliceDoc(mark[0].from, mark[0].from + 1) === '!',
|
|
127
|
+
label,
|
|
128
|
+
ref: url,
|
|
129
|
+
};
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
};
|
|
133
|
+
|
|
134
|
+
/**
|
|
135
|
+
* Inline widget.
|
|
136
|
+
* [Label](dxn:echo:123)
|
|
137
|
+
*/
|
|
138
|
+
class PreviewInlineWidget extends WidgetType {
|
|
139
|
+
constructor(
|
|
140
|
+
readonly _options: PreviewOptions,
|
|
141
|
+
readonly _link: PreviewLinkRef,
|
|
142
|
+
) {
|
|
143
|
+
super();
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
// override ignoreEvent() {
|
|
147
|
+
// return false;
|
|
148
|
+
// }
|
|
149
|
+
|
|
150
|
+
override eq(other: this) {
|
|
151
|
+
return this._link.ref === other._link.ref && this._link.label === other._link.label;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
override toDOM(_view: EditorView) {
|
|
155
|
+
const root = document.createElement('dx-anchor');
|
|
156
|
+
root.classList.add('dx-tag--anchor');
|
|
157
|
+
root.textContent = this._link.label;
|
|
158
|
+
root.setAttribute('refId', this._link.ref);
|
|
159
|
+
return root;
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Block widget (e.g., for surfaces).
|
|
165
|
+
* ![Label][dxn:echo:123]
|
|
166
|
+
*/
|
|
167
|
+
class PreviewBlockWidget extends WidgetType {
|
|
168
|
+
constructor(
|
|
169
|
+
readonly _options: PreviewOptions,
|
|
170
|
+
readonly _link: PreviewLinkRef,
|
|
171
|
+
) {
|
|
172
|
+
super();
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
// override ignoreEvent() {
|
|
176
|
+
// return true;
|
|
177
|
+
// }
|
|
178
|
+
|
|
179
|
+
override eq(other: this) {
|
|
180
|
+
return this._link.ref === other._link.ref;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
override toDOM(_view: EditorView) {
|
|
184
|
+
const root = document.createElement('div');
|
|
185
|
+
root.classList.add('cm-preview-block', 'density-fine');
|
|
186
|
+
this._options.addBlockContainer?.({ link: this._link, el: root });
|
|
187
|
+
return root;
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
override destroy(root: HTMLDivElement) {
|
|
191
|
+
this._options.removeBlockContainer?.({ link: this._link, el: root });
|
|
192
|
+
}
|
|
193
|
+
}
|