-
Notifications
You must be signed in to change notification settings - Fork 166
/
replace.mjs
85 lines (71 loc) · 2.59 KB
/
replace.mjs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
// Emojies & shortcuts replacement logic.
//
// Note: In theory, it could be faster to parse :smile: in inline chain and
// leave only shortcuts here. But, who care...
//
export default function create_rule (md, emojies, shortcuts, scanRE, replaceRE) {
const arrayReplaceAt = md.utils.arrayReplaceAt
const ucm = md.utils.lib.ucmicro
const has = md.utils.has
const ZPCc = new RegExp([ucm.Z.source, ucm.P.source, ucm.Cc.source].join('|'))
function splitTextToken (text, level, Token) {
let last_pos = 0
const nodes = []
text.replace(replaceRE, function (match, offset, src) {
let emoji_name
// Validate emoji name
if (has(shortcuts, match)) {
// replace shortcut with full name
emoji_name = shortcuts[match]
// Don't allow letters before any shortcut (as in no ":/" in http://)
if (offset > 0 && !ZPCc.test(src[offset - 1])) return
// Don't allow letters after any shortcut
if (offset + match.length < src.length && !ZPCc.test(src[offset + match.length])) {
return
}
} else {
emoji_name = match.slice(1, -1)
}
// Add new tokens to pending list
if (offset > last_pos) {
const token = new Token('text', '', 0)
token.content = text.slice(last_pos, offset)
nodes.push(token)
}
const token = new Token('emoji', '', 0)
token.markup = emoji_name
token.content = emojies[emoji_name]
nodes.push(token)
last_pos = offset + match.length
})
if (last_pos < text.length) {
const token = new Token('text', '', 0)
token.content = text.slice(last_pos)
nodes.push(token)
}
return nodes
}
return function emoji_replace (state) {
let token
const blockTokens = state.tokens
let autolinkLevel = 0
for (let j = 0, l = blockTokens.length; j < l; j++) {
if (blockTokens[j].type !== 'inline') { continue }
let tokens = blockTokens[j].children
// We scan from the end, to keep position when new tags added.
// Use reversed logic in links start/end match
for (let i = tokens.length - 1; i >= 0; i--) {
token = tokens[i]
if (token.type === 'link_open' || token.type === 'link_close') {
if (token.info === 'auto') { autolinkLevel -= token.nesting }
}
if (token.type === 'text' && autolinkLevel === 0 && scanRE.test(token.content)) {
// replace current node
blockTokens[j].children = tokens = arrayReplaceAt(
tokens, i, splitTextToken(token.content, token.level, state.Token)
)
}
}
}
}
};