123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212 |
- import { isSpace, normalizeReference } from '../common/utils.mjs'
- export default function reference (state, startLine, _endLine, silent) {
- let pos = state.bMarks[startLine] + state.tShift[startLine]
- let max = state.eMarks[startLine]
- let nextLine = startLine + 1
- // if it's indented more than 3 spaces, it should be a code block
- if (state.sCount[startLine] - state.blkIndent >= 4) { return false }
- if (state.src.charCodeAt(pos) !== 0x5B/* [ */) { return false }
- function getNextLine (nextLine) {
- const endLine = state.lineMax
- if (nextLine >= endLine || state.isEmpty(nextLine)) {
- // empty line or end of input
- return null
- }
- let isContinuation = false
- // this would be a code block normally, but after paragraph
- // it's considered a lazy continuation regardless of what's there
- if (state.sCount[nextLine] - state.blkIndent > 3) { isContinuation = true }
- // quirk for blockquotes, this line should already be checked by that rule
- if (state.sCount[nextLine] < 0) { isContinuation = true }
- if (!isContinuation) {
- const terminatorRules = state.md.block.ruler.getRules('reference')
- const oldParentType = state.parentType
- state.parentType = 'reference'
- // Some tags can terminate paragraph without empty line.
- let terminate = false
- for (let i = 0, l = terminatorRules.length; i < l; i++) {
- if (terminatorRules[i](state, nextLine, endLine, true)) {
- terminate = true
- break
- }
- }
- state.parentType = oldParentType
- if (terminate) {
- // terminated by another block
- return null
- }
- }
- const pos = state.bMarks[nextLine] + state.tShift[nextLine]
- const max = state.eMarks[nextLine]
- // max + 1 explicitly includes the newline
- return state.src.slice(pos, max + 1)
- }
- let str = state.src.slice(pos, max + 1)
- max = str.length
- let labelEnd = -1
- for (pos = 1; pos < max; pos++) {
- const ch = str.charCodeAt(pos)
- if (ch === 0x5B /* [ */) {
- return false
- } else if (ch === 0x5D /* ] */) {
- labelEnd = pos
- break
- } else if (ch === 0x0A /* \n */) {
- const lineContent = getNextLine(nextLine)
- if (lineContent !== null) {
- str += lineContent
- max = str.length
- nextLine++
- }
- } else if (ch === 0x5C /* \ */) {
- pos++
- if (pos < max && str.charCodeAt(pos) === 0x0A) {
- const lineContent = getNextLine(nextLine)
- if (lineContent !== null) {
- str += lineContent
- max = str.length
- nextLine++
- }
- }
- }
- }
- if (labelEnd < 0 || str.charCodeAt(labelEnd + 1) !== 0x3A/* : */) { return false }
- // [label]: destination 'title'
- // ^^^ skip optional whitespace here
- for (pos = labelEnd + 2; pos < max; pos++) {
- const ch = str.charCodeAt(pos)
- if (ch === 0x0A) {
- const lineContent = getNextLine(nextLine)
- if (lineContent !== null) {
- str += lineContent
- max = str.length
- nextLine++
- }
- } else if (isSpace(ch)) {
- /* eslint no-empty:0 */
- } else {
- break
- }
- }
- // [label]: destination 'title'
- // ^^^^^^^^^^^ parse this
- const destRes = state.md.helpers.parseLinkDestination(str, pos, max)
- if (!destRes.ok) { return false }
- const href = state.md.normalizeLink(destRes.str)
- if (!state.md.validateLink(href)) { return false }
- pos = destRes.pos
- // save cursor state, we could require to rollback later
- const destEndPos = pos
- const destEndLineNo = nextLine
- // [label]: destination 'title'
- // ^^^ skipping those spaces
- const start = pos
- for (; pos < max; pos++) {
- const ch = str.charCodeAt(pos)
- if (ch === 0x0A) {
- const lineContent = getNextLine(nextLine)
- if (lineContent !== null) {
- str += lineContent
- max = str.length
- nextLine++
- }
- } else if (isSpace(ch)) {
- /* eslint no-empty:0 */
- } else {
- break
- }
- }
- // [label]: destination 'title'
- // ^^^^^^^ parse this
- let titleRes = state.md.helpers.parseLinkTitle(str, pos, max)
- while (titleRes.can_continue) {
- const lineContent = getNextLine(nextLine)
- if (lineContent === null) break
- str += lineContent
- pos = max
- max = str.length
- nextLine++
- titleRes = state.md.helpers.parseLinkTitle(str, pos, max, titleRes)
- }
- let title
- if (pos < max && start !== pos && titleRes.ok) {
- title = titleRes.str
- pos = titleRes.pos
- } else {
- title = ''
- pos = destEndPos
- nextLine = destEndLineNo
- }
- // skip trailing spaces until the rest of the line
- while (pos < max) {
- const ch = str.charCodeAt(pos)
- if (!isSpace(ch)) { break }
- pos++
- }
- if (pos < max && str.charCodeAt(pos) !== 0x0A) {
- if (title) {
- // garbage at the end of the line after title,
- // but it could still be a valid reference if we roll back
- title = ''
- pos = destEndPos
- nextLine = destEndLineNo
- while (pos < max) {
- const ch = str.charCodeAt(pos)
- if (!isSpace(ch)) { break }
- pos++
- }
- }
- }
- if (pos < max && str.charCodeAt(pos) !== 0x0A) {
- // garbage at the end of the line
- return false
- }
- const label = normalizeReference(str.slice(1, labelEnd))
- if (!label) {
- // CommonMark 0.20 disallows empty labels
- return false
- }
- // Reference can not terminate anything. This check is for safety only.
- /* istanbul ignore if */
- if (silent) { return true }
- if (typeof state.env.references === 'undefined') {
- state.env.references = {}
- }
- if (typeof state.env.references[label] === 'undefined') {
- state.env.references[label] = { title, href }
- }
- state.line = nextLine
- return true
- }
|