|
|
|
// Process links like https://example.org/
|
|
|
|
|
|
|
|
// RFC3986: scheme = ALPHA *( ALPHA / DIGIT / "+" / "-" / "." )
|
|
|
|
const SCHEME_RE = /(?:^|[^a-z0-9.+-])([a-z][a-z0-9.+-]*)$/i
|
|
|
|
|
|
|
|
|
|
|
|
export default function linkify (state, silent) {
|
|
|
|
if (!state.md.options.linkify) return false
|
|
|
|
if (state.linkLevel > 0) return false
|
|
|
|
|
|
|
|
const pos = state.pos
|
|
|
|
const max = state.posMax
|
|
|
|
|
|
|
|
if (pos + 3 > max) return false
|
|
|
|
if (state.src.charCodeAt(pos) !== 0x3A/* : */) return false
|
|
|
|
if (state.src.charCodeAt(pos + 1) !== 0x2F/* / */) return false
|
|
|
|
if (state.src.charCodeAt(pos + 2) !== 0x2F/* / */) return false
|
|
|
|
|
|
|
|
const match = state.pending.match(SCHEME_RE)
|
|
|
|
if (!match) return false
|
|
|
|
|
|
|
|
const proto = match[1]
|
|
|
|
|
|
|
|
const link = state.md.linkify.matchAtStart(state.src.slice(pos - proto.length))
|
|
|
|
if (!link) return false
|
|
|
|
|
|
|
|
let url = link.url
|
|
|
|
|
|
|
|
// invalid link, but still detected by linkify somehow;
|
|
|
|
// need to check to prevent infinite loop below
|
|
|
|
if (url.length <= proto.length) return false
|
|
|
|
|
|
|
|
// disallow '*' at the end of the link (conflicts with emphasis)
|
|
|
|
url = url.replace(/\*+$/, '')
|
|
|
|
|
|
|
|
const fullUrl = state.md.normalizeLink(url)
|
|
|
|
if (!state.md.validateLink(fullUrl)) return false
|
|
|
|
|
|
|
|
if (!silent) {
|
|
|
|
state.pending = state.pending.slice(0, -proto.length)
|
|
|
|
|
|
|
|
const token_o = state.push('link_open', 'a', 1)
|
|
|
|
token_o.attrs = [['href', fullUrl]]
|
|
|
|
token_o.markup = 'linkify'
|
|
|
|
token_o.info = 'auto'
|
|
|
|
|
|
|
|
const token_t = state.push('text', '', 0)
|
|
|
|
token_t.content = state.md.normalizeLinkText(url)
|
|
|
|
|
|
|
|
const token_c = state.push('link_close', 'a', -1)
|
|
|
|
token_c.markup = 'linkify'
|
|
|
|
token_c.info = 'auto'
|
|
|
|
}
|
|
|
|
|
|
|
|
state.pos += url.length - proto.length
|
|
|
|
return true
|
|
|
|
}
|