test02/node_modules/markdown-it-emoji/lib/replace.mjs
罗佳鸿 6aa1ebe342
Some checks are pending
部署文档 / deploy-gh-pages (push) Waiting to run
first commit
2024-08-13 10:11:19 +08:00

86 lines
2.6 KiB
JavaScript

// 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)
)
}
}
}
}
};