diff options
author | Hsieh Chin Fan <pham@topo.tw> | 2024-09-11 18:21:32 +0800 |
---|---|---|
committer | Hsieh Chin Fan <pham@topo.tw> | 2024-09-11 18:21:33 +0800 |
commit | 0e6110ad9be2e444880b13e9f670f636ccd09381 (patch) | |
tree | fa4c4a102eb3950d801ed20ee35138c07e435b9f /src | |
parent | 7e62f6353fa485bd184e9910f933791a414d3398 (diff) |
feat: Use CSS selector on DocLinks
* Now module export method createDocLinks()
* DocLink now decides target by CSS selector in title with prefix "=>",
or just by doc fragment
Diffstat (limited to 'src')
-rw-r--r-- | src/dumbymap.mjs | 72 | ||||
-rw-r--r-- | src/editor.mjs | 5 |
2 files changed, 49 insertions, 28 deletions
diff --git a/src/dumbymap.mjs b/src/dumbymap.mjs index cd3b8ff..a896d0d 100644 --- a/src/dumbymap.mjs +++ b/src/dumbymap.mjs | |||
@@ -7,7 +7,8 @@ import LeaderLine from 'leader-line' | |||
7 | import PlainDraggable from 'plain-draggable' | 7 | import PlainDraggable from 'plain-draggable' |
8 | import { render, parseConfigsFromYaml } from 'mapclay' | 8 | import { render, parseConfigsFromYaml } from 'mapclay' |
9 | 9 | ||
10 | function onRemove(element, callback) { | 10 | // Utils {{{ |
11 | const onRemove = (element, callback) => { | ||
11 | const parent = element.parentNode; | 12 | const parent = element.parentNode; |
12 | if (!parent) throw new Error("The node must already be attached"); | 13 | if (!parent) throw new Error("The node must already be attached"); |
13 | 14 | ||
@@ -23,12 +24,47 @@ function onRemove(element, callback) { | |||
23 | }); | 24 | }); |
24 | obs.observe(parent, { childList: true, }); | 25 | obs.observe(parent, { childList: true, }); |
25 | } | 26 | } |
27 | // }}} | ||
28 | // FUNCTION: Get DocLinks from normal special anchor element {{{ | ||
26 | 29 | ||
27 | // Render: Markdown -> HTML {{{ | 30 | const docLinkSelector = 'a[href^="#"][title^="=>"]' |
28 | export const markdown2HTML = (container, mdContent) => { | 31 | export const createDocLinks = (container) => Array.from(container.querySelectorAll(docLinkSelector)) |
32 | .map(link => { | ||
33 | link.classList.add('with-leader-line', 'doclink') | ||
34 | link.lines = [] | ||
35 | |||
36 | link.onmouseover = () => { | ||
37 | const label = decodeURIComponent(link.href.split('#')[1]) | ||
38 | const selector = link.title.split('=>')[1] ?? '#' + label | ||
39 | const target = document.querySelector(selector) | ||
40 | if (!target?.checkVisibility()) return | ||
41 | |||
42 | const line = new LeaderLine({ | ||
43 | start: link, | ||
44 | end: target, | ||
45 | middleLabel: LeaderLine.pathLabel({ | ||
46 | text: label, | ||
47 | fontWeight: 'bold', | ||
48 | }), | ||
49 | hide: true, | ||
50 | path: "magnet" | ||
51 | }) | ||
52 | link.lines.push(line) | ||
53 | line.show('draw', { duration: 300, }) | ||
54 | } | ||
55 | link.onmouseout = () => { | ||
56 | link.lines.forEach(line => line.remove()) | ||
57 | link.lines.length = 0 | ||
58 | } | ||
29 | 59 | ||
60 | return link | ||
61 | }) | ||
62 | // }}} | ||
63 | export const markdown2HTML = (container, mdContent) => { | ||
64 | // Render: Markdown -> HTML {{{ | ||
30 | Array.from(container.children).map(e => e.remove()) | 65 | Array.from(container.children).map(e => e.remove()) |
31 | 66 | ||
67 | |||
32 | container.innerHTML = '<div class="SemanticHtml"></div>' | 68 | container.innerHTML = '<div class="SemanticHtml"></div>' |
33 | const htmlHolder = container.querySelector('.SemanticHtml') | 69 | const htmlHolder = container.querySelector('.SemanticHtml') |
34 | 70 | ||
@@ -50,7 +86,11 @@ export const markdown2HTML = (container, mdContent) => { | |||
50 | 86 | ||
51 | // Add close tag for block with more than 2 empty lines | 87 | // Add close tag for block with more than 2 empty lines |
52 | md.block.ruler.before('table', 'draggable_block', (state, startLine) => { | 88 | md.block.ruler.before('table', 'draggable_block', (state, startLine) => { |
53 | if (state.src[state.bMarks[startLine - 1]] === '\n' && state.src[state.bMarks[startLine - 2]] === '\n') { | 89 | if ( |
90 | state.src[state.bMarks[startLine - 1]] === '\n' && | ||
91 | state.src[state.bMarks[startLine - 2]] === '\n' && | ||
92 | state.tokens.at(-1).type !== 'list_item_open' // Quick hack for not adding tag after "::marker" for <li> | ||
93 | ) { | ||
54 | state.push('draggable_block_close', '', -1); | 94 | state.push('draggable_block_close', '', -1); |
55 | state.push('draggable_block_open', '', 1); | 95 | state.push('draggable_block_open', '', 1); |
56 | } | 96 | } |
@@ -68,34 +108,10 @@ export const markdown2HTML = (container, mdContent) => { | |||
68 | 108 | ||
69 | 109 | ||
70 | // TODO Improve it! | 110 | // TODO Improve it! |
71 | const docLinks = Array.from(container.querySelectorAll('a[href^="#"][title^="doc"]')) | ||
72 | docLinks.forEach(link => { | ||
73 | link.classList.add('with-leader-line', 'doclink') | ||
74 | link.lines = [] | ||
75 | |||
76 | link.onmouseover = () => { | ||
77 | const target = document.querySelector(link.getAttribute('href')) | ||
78 | if (!target?.checkVisibility()) return | ||
79 | |||
80 | const line = new LeaderLine({ | ||
81 | start: link, | ||
82 | end: target, | ||
83 | hide: true, | ||
84 | path: "magnet" | ||
85 | }) | ||
86 | link.lines.push(line) | ||
87 | line.show('draw', { duration: 300, }) | ||
88 | } | ||
89 | link.onmouseout = () => { | ||
90 | link.lines.forEach(line => line.remove()) | ||
91 | link.lines.length = 0 | ||
92 | } | ||
93 | }) | ||
94 | 111 | ||
95 | return container | 112 | return container |
96 | //}}} | 113 | //}}} |
97 | } | 114 | } |
98 | |||
99 | // FIXME Don't use hard-coded CSS selector | 115 | // FIXME Don't use hard-coded CSS selector |
100 | export const generateMaps = async (container) => { | 116 | export const generateMaps = async (container) => { |
101 | // LeaderLine {{{ | 117 | // LeaderLine {{{ |
diff --git a/src/editor.mjs b/src/editor.mjs index 1544589..107743b 100644 --- a/src/editor.mjs +++ b/src/editor.mjs | |||
@@ -1,5 +1,6 @@ | |||
1 | import { markdown2HTML, generateMaps } from './dumbymap' | 1 | import { markdown2HTML, generateMaps } from './dumbymap' |
2 | import { defaultAliasesForRenderer, parseConfigsFromYaml } from 'mapclay' | 2 | import { defaultAliasesForRenderer, parseConfigsFromYaml } from 'mapclay' |
3 | import { createDocLinks } from './dumbymap.mjs' | ||
3 | 4 | ||
4 | // Set up Editor {{{ | 5 | // Set up Editor {{{ |
5 | 6 | ||
@@ -12,6 +13,7 @@ const toggleMaps = (container) => { | |||
12 | document.activeElement.blur(); | 13 | document.activeElement.blur(); |
13 | } else { | 14 | } else { |
14 | markdown2HTML(HtmlContainer, editor.value()) | 15 | markdown2HTML(HtmlContainer, editor.value()) |
16 | createDocLinks(container) | ||
15 | container.setAttribute('data-layout', 'none') | 17 | container.setAttribute('data-layout', 'none') |
16 | } | 18 | } |
17 | } | 19 | } |
@@ -70,9 +72,12 @@ const editor = new EasyMDE({ | |||
70 | 72 | ||
71 | const cm = editor.codemirror | 73 | const cm = editor.codemirror |
72 | markdown2HTML(HtmlContainer, editor.value()) | 74 | markdown2HTML(HtmlContainer, editor.value()) |
75 | createDocLinks(HtmlContainer) | ||
73 | 76 | ||
77 | // Re-render HTML by editor content | ||
74 | cm.on("change", () => { | 78 | cm.on("change", () => { |
75 | markdown2HTML(HtmlContainer, editor.value()) | 79 | markdown2HTML(HtmlContainer, editor.value()) |
80 | createDocLinks(HtmlContainer) | ||
76 | }) | 81 | }) |
77 | // }}} | 82 | // }}} |
78 | 83 | ||