2020-06-29 13:05:08 +00:00
|
|
|
const COMMAND_LEVELS = {
|
2020-07-16 10:09:10 +00:00
|
|
|
book: 10,
|
|
|
|
part: 20,
|
|
|
|
chapter: 30,
|
|
|
|
section: 40,
|
|
|
|
subsection: 50,
|
|
|
|
subsubsection: 60,
|
|
|
|
paragraph: 70,
|
|
|
|
subparagraph: 80
|
2020-06-29 13:05:08 +00:00
|
|
|
}
|
|
|
|
|
2020-07-16 10:09:10 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* RegExp matcher parts:
|
|
|
|
*
|
|
|
|
* REGEX_START: begining of line, any number of spaces, double \ (required)
|
|
|
|
* REGEX_COMMAND: any of the listed commands (required)
|
|
|
|
* REGEX_SPACING: spaces and * between groups (optional)
|
|
|
|
* REGEX_SHORT_TITLE: a text between square brackets (optional)
|
|
|
|
* REGEX_TITLE: a text between curly brackets (required)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
const REGEX_START = '^\\s*\\\\'
|
|
|
|
const REGEX_COMMAND = `(${Object.keys(COMMAND_LEVELS).join('|')})`
|
|
|
|
const REGEX_SPACING = '\\s?\\*?\\s?'
|
|
|
|
const REGEX_SHORT_TITLE = '(\\[([^\\]]+)\\])?'
|
|
|
|
const REGEX_TITLE = '{(.*)}'
|
|
|
|
const MATCHER = new RegExp(
|
|
|
|
`${REGEX_START}${REGEX_COMMAND}${REGEX_SPACING}${REGEX_SHORT_TITLE}${REGEX_SPACING}${REGEX_TITLE}`
|
|
|
|
)
|
|
|
|
|
2020-06-29 13:05:08 +00:00
|
|
|
function matchOutline(content) {
|
|
|
|
const lines = content.split('\n')
|
|
|
|
const flatOutline = []
|
|
|
|
lines.forEach((line, lineId) => {
|
2020-07-16 10:09:10 +00:00
|
|
|
const match = line.match(MATCHER)
|
2020-06-29 13:05:08 +00:00
|
|
|
if (!match) return
|
2020-07-16 10:09:10 +00:00
|
|
|
const [, command, , shortTitle, title] = match
|
|
|
|
|
2020-06-29 13:05:08 +00:00
|
|
|
flatOutline.push({
|
|
|
|
line: lineId + 1,
|
2020-07-16 10:09:10 +00:00
|
|
|
title: matchDisplayTitle(shortTitle || title),
|
2020-06-29 13:05:08 +00:00
|
|
|
level: COMMAND_LEVELS[command]
|
|
|
|
})
|
|
|
|
})
|
|
|
|
return flatOutline
|
|
|
|
}
|
|
|
|
|
2020-07-16 10:09:10 +00:00
|
|
|
const DISPLAY_TITLE_REGEX = new RegExp('([^\\\\]*)\\\\([^{]+){([^}]+)}(.*)')
|
|
|
|
/*
|
|
|
|
* Attempt to improve the display of the outline title for titles with commands.
|
|
|
|
* Either skip the command (for labels) or display the command's content instead
|
|
|
|
* of the entire command.
|
|
|
|
*
|
|
|
|
* e.g. "Label \\label{foo} between" => "Label between"
|
|
|
|
* e.g. "TT \\texttt{Bar}" => "TT Bar"
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
function matchDisplayTitle(title) {
|
|
|
|
const closingBracketPosition = title.indexOf('}')
|
|
|
|
if (closingBracketPosition < 0) {
|
|
|
|
// simple title (no commands)
|
|
|
|
return title
|
|
|
|
}
|
|
|
|
|
|
|
|
const titleMatch = title.match(DISPLAY_TITLE_REGEX)
|
|
|
|
if (!titleMatch) {
|
|
|
|
// no contained commands; strip everything after the first closing bracket
|
|
|
|
return title.substring(0, closingBracketPosition)
|
|
|
|
}
|
|
|
|
|
|
|
|
const [, textBefore, command, commandContent, textAfter] = titleMatch
|
|
|
|
if (command === 'label') {
|
|
|
|
// label: don't display them at all
|
|
|
|
title = `${textBefore}${textAfter}`
|
|
|
|
} else {
|
|
|
|
// display the content of the command. Works well for formatting commands
|
|
|
|
title = `${textBefore}${commandContent}${textAfter}`
|
|
|
|
}
|
|
|
|
|
|
|
|
return title
|
|
|
|
}
|
|
|
|
|
2020-06-29 13:05:08 +00:00
|
|
|
function nestOutline(flatOutline) {
|
|
|
|
let parentOutlines = {}
|
|
|
|
let nestedOutlines = []
|
|
|
|
flatOutline.forEach(outline => {
|
2020-07-16 10:09:10 +00:00
|
|
|
const parentOutlineLevels = Object.keys(parentOutlines)
|
|
|
|
|
|
|
|
// find the nearest parent outline
|
|
|
|
const nearestParentLevel = parentOutlineLevels
|
|
|
|
.reverse()
|
|
|
|
.find(level => level < outline.level)
|
|
|
|
const parentOutline = parentOutlines[nearestParentLevel]
|
2020-06-29 13:05:08 +00:00
|
|
|
if (!parentOutline) {
|
|
|
|
// top level
|
|
|
|
nestedOutlines.push(outline)
|
|
|
|
} else if (!parentOutline.children) {
|
|
|
|
// first outline in this node
|
|
|
|
parentOutline.children = [outline]
|
|
|
|
} else {
|
|
|
|
// push outline to node
|
|
|
|
parentOutline.children.push(outline)
|
|
|
|
}
|
2020-07-16 10:09:10 +00:00
|
|
|
|
|
|
|
// store the outline as new parent at its level and forget lower-level
|
|
|
|
// outlines (if any) as they shouldn't get any children anymore
|
2020-06-29 13:05:08 +00:00
|
|
|
parentOutlines[outline.level] = outline
|
2020-07-16 10:09:10 +00:00
|
|
|
parentOutlineLevels
|
|
|
|
.filter(level => level > outline.level)
|
|
|
|
.forEach(level => delete parentOutlines[level])
|
2020-06-29 13:05:08 +00:00
|
|
|
})
|
|
|
|
return nestedOutlines
|
|
|
|
}
|
|
|
|
|
2020-07-28 09:37:46 +00:00
|
|
|
export { matchOutline, nestOutline }
|