mirror of
https://github.com/hcengineering/platform.git
synced 2025-01-08 21:27:45 +03:00
fix: implement better todos parsing (#6497)
Signed-off-by: Alexander Onnikov <Alexander.Onnikov@xored.com>
This commit is contained in:
parent
579caad220
commit
964b8d1ba4
408
packages/text/src/markdown/__tests__/markdown.test.ts
Normal file
408
packages/text/src/markdown/__tests__/markdown.test.ts
Normal file
@ -0,0 +1,408 @@
|
|||||||
|
//
|
||||||
|
// Copyright © 2024 Hardcore Engineering Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License. You may
|
||||||
|
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
//
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
//
|
||||||
|
|
||||||
|
import { ServerKit } from '../../kits/server-kit'
|
||||||
|
import { parseMessageMarkdown } from '..'
|
||||||
|
|
||||||
|
const refUrl: string = 'ref://'
|
||||||
|
const imageUrl: string = 'http://localhost'
|
||||||
|
|
||||||
|
const extensions = [ServerKit]
|
||||||
|
|
||||||
|
const tests: Array<{ name: string, markdown: string, markup: object }> = [
|
||||||
|
{
|
||||||
|
name: 'simple text',
|
||||||
|
markdown: 'Lorem ipsum dolor sit amet.',
|
||||||
|
markup: {
|
||||||
|
type: 'doc',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'Lorem ipsum dolor sit amet.',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'text with heading',
|
||||||
|
markdown: `# Lorem ipsum
|
||||||
|
|
||||||
|
Lorem ipsum dolor sit amet.
|
||||||
|
`,
|
||||||
|
markup: {
|
||||||
|
type: 'doc',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'heading',
|
||||||
|
attrs: { level: 1 },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'Lorem ipsum',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'Lorem ipsum dolor sit amet.',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'bullet list',
|
||||||
|
markdown: `# bullet list
|
||||||
|
- list item 1
|
||||||
|
- list item 2
|
||||||
|
`,
|
||||||
|
markup: {
|
||||||
|
type: 'doc',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'heading',
|
||||||
|
attrs: { level: 1 },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'bullet list',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'bulletList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'listItem',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'list item 1',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'listItem',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'list item 2',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'todos',
|
||||||
|
markdown: `# TODO
|
||||||
|
- [ ] todo 1
|
||||||
|
- [x] todo 2
|
||||||
|
`,
|
||||||
|
markup: {
|
||||||
|
type: 'doc',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'heading',
|
||||||
|
attrs: { level: 1 },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'TODO',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'todoList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'todoItem',
|
||||||
|
attrs: { checked: false },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'todo 1',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'todoItem',
|
||||||
|
attrs: { checked: true },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'todo 2',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'todos followed by list items',
|
||||||
|
markdown: `# todo and list
|
||||||
|
- [ ] todo 1
|
||||||
|
- [x] todo 2
|
||||||
|
- list item 1
|
||||||
|
- list item 2
|
||||||
|
`,
|
||||||
|
markup: {
|
||||||
|
type: 'doc',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'heading',
|
||||||
|
attrs: { level: 1 },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'todo and list',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'todoList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'todoItem',
|
||||||
|
attrs: { checked: false },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'todo 1',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'todoItem',
|
||||||
|
attrs: { checked: true },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'todo 2',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'bulletList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'listItem',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'list item 1',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'listItem',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'list item 2',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'todos followed by list items',
|
||||||
|
markdown: `# mixed lists
|
||||||
|
- [ ] todo 1
|
||||||
|
- list item 1
|
||||||
|
- [x] todo 2
|
||||||
|
- list item 2
|
||||||
|
`,
|
||||||
|
markup: {
|
||||||
|
type: 'doc',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'heading',
|
||||||
|
attrs: { level: 1 },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'mixed lists',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'todoList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'todoItem',
|
||||||
|
attrs: { checked: false },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'todo 1',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'bulletList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'listItem',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'list item 1',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'todoList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'todoItem',
|
||||||
|
attrs: { checked: true },
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'todo 2',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'bulletList',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'listItem',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'paragraph',
|
||||||
|
content: [
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: 'list item 2',
|
||||||
|
marks: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
describe('markdown', () => {
|
||||||
|
tests.forEach(({ name, markdown, markup }) => {
|
||||||
|
it(name, () => {
|
||||||
|
const parsed = parseMessageMarkdown(markdown, refUrl, imageUrl, extensions)
|
||||||
|
expect(parsed).toEqual(markup)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
@ -589,12 +589,6 @@ interface TaskListStateCore extends StateCore {
|
|||||||
const startsWithTodoMarkdown = (token: Token): boolean => /^\[[xX \u00A0]\][ \u00A0]/.test(token.content)
|
const startsWithTodoMarkdown = (token: Token): boolean => /^\[[xX \u00A0]\][ \u00A0]/.test(token.content)
|
||||||
const isCheckedTodoItem = (token: Token): boolean => /^\[[xX]\][ \u00A0]/.test(token.content)
|
const isCheckedTodoItem = (token: Token): boolean => /^\[[xX]\][ \u00A0]/.test(token.content)
|
||||||
|
|
||||||
const isTodoListItemInline = (tokens: Token[], index: number): boolean =>
|
|
||||||
isInlineToken(tokens[index]) &&
|
|
||||||
isParagraphToken(tokens[index - 1]) &&
|
|
||||||
isListItemToken(tokens[index - 2]) &&
|
|
||||||
startsWithTodoMarkdown(tokens[index])
|
|
||||||
|
|
||||||
export class MarkdownParser {
|
export class MarkdownParser {
|
||||||
tokenizer: MarkdownIt
|
tokenizer: MarkdownIt
|
||||||
tokenHandlers: Record<string, (state: MarkdownParseState, tok: Token) => void>
|
tokenHandlers: Record<string, (state: MarkdownParseState, tok: Token) => void>
|
||||||
@ -607,7 +601,7 @@ export class MarkdownParser {
|
|||||||
this.tokenizer = MarkdownIt('default', {
|
this.tokenizer = MarkdownIt('default', {
|
||||||
html: true
|
html: true
|
||||||
})
|
})
|
||||||
this.tokenizer.core.ruler.after('inline', 'task_list', this.taskListRule)
|
this.tokenizer.core.ruler.after('inline', 'task_list', this.listRule)
|
||||||
|
|
||||||
this.tokenHandlers = tokenHandlers(tokensBlock, tokensNode, tokensMark, specialRule, ignoreRule, extensions)
|
this.tokenHandlers = tokenHandlers(tokensBlock, tokensNode, tokensMark, specialRule, ignoreRule, extensions)
|
||||||
}
|
}
|
||||||
@ -625,64 +619,77 @@ export class MarkdownParser {
|
|||||||
return doc
|
return doc
|
||||||
}
|
}
|
||||||
|
|
||||||
taskListRule: RuleCore = (state: TaskListStateCore): boolean => {
|
listRule: RuleCore = (state: TaskListStateCore): boolean => {
|
||||||
const tokens = state.tokens
|
const tokens = state.tokens
|
||||||
|
|
||||||
interface TodoListItemDescriptor {
|
// step #1 - convert list items to todo items
|
||||||
start?: number
|
for (let open = 0; open < tokens.length; open++) {
|
||||||
end?: number
|
if (isTodoListItem(tokens, open)) {
|
||||||
}
|
convertTodoItem(tokens, open)
|
||||||
|
|
||||||
let todoListStartIdx: number | undefined
|
|
||||||
let todoListItems: TodoListItemDescriptor[] = []
|
|
||||||
let todoListItem: TodoListItemDescriptor | undefined
|
|
||||||
let isTodoList = false
|
|
||||||
for (let i = 0; i < tokens.length; i++) {
|
|
||||||
if (tokens[i].type === 'bullet_list_open') {
|
|
||||||
todoListStartIdx = i
|
|
||||||
isTodoList = true
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tokens[i].type === 'list_item_open') {
|
|
||||||
todoListItem = {
|
|
||||||
start: i
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tokens[i].type === 'inline') {
|
// step #2 - convert lists to proper type
|
||||||
if (todoListItem === undefined || !isTodoListItemInline(tokens, i)) {
|
let closeIdx = -1
|
||||||
isTodoList = false
|
let lastItemIdx = -1
|
||||||
}
|
for (let i = tokens.length - 1; i >= 0; i--) {
|
||||||
}
|
|
||||||
|
|
||||||
if (tokens[i].type === 'list_item_close' && todoListItem !== undefined) {
|
|
||||||
todoListItem.end = i
|
|
||||||
if (isTodoList) {
|
|
||||||
todoListItems.push(todoListItem)
|
|
||||||
}
|
|
||||||
todoListItem = undefined
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tokens[i].type === 'bullet_list_close') {
|
if (tokens[i].type === 'bullet_list_close') {
|
||||||
if (isTodoList && todoListStartIdx !== undefined) {
|
closeIdx = i
|
||||||
// Transform tokens
|
lastItemIdx = -1
|
||||||
tokens[todoListStartIdx].type = 'todo_list_open'
|
} else if (tokens[i].type === 'list_item_close' || tokens[i].type === 'todo_item_close') {
|
||||||
tokens[i].type = 'todo_list_close'
|
// when found item close token of different type, split the list
|
||||||
|
if (lastItemIdx === -1) {
|
||||||
for (const item of todoListItems) {
|
lastItemIdx = i
|
||||||
if (item.start !== undefined && item.end !== undefined) {
|
} else if (tokens[i].type !== tokens[lastItemIdx].type) {
|
||||||
tokens[item.start].type = 'todo_item_open'
|
tokens.splice(i + 1, 0, new state.Token('bullet_list_open', 'ul', 1))
|
||||||
tokens[item.end].type = 'todo_item_close'
|
tokens.splice(i + 1, 0, new state.Token('bullet_list_close', 'ul', -1))
|
||||||
|
convertTodoList(tokens, i + 2, closeIdx + 2, lastItemIdx + 2)
|
||||||
const inline = tokens[item.start + 2]
|
closeIdx = i + 1
|
||||||
|
lastItemIdx = i
|
||||||
if (tokens[item.start].attrs == null) {
|
}
|
||||||
tokens[item.start].attrs = []
|
} else if (tokens[i].type === 'bullet_list_open' && tokens[i].level === tokens[closeIdx].level) {
|
||||||
|
// when found list open token of the same level, decide what to do
|
||||||
|
if (lastItemIdx !== -1) {
|
||||||
|
convertTodoList(tokens, i, closeIdx, lastItemIdx)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isCheckedTodoItem(inline)) {
|
// Reset closeIdx and lastItemIdx for the next list
|
||||||
;(tokens[item.start].attrs as any).push(['checked', 'true'])
|
closeIdx = -1
|
||||||
|
lastItemIdx = -1
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function convertTodoList (tokens: Token[], open: number, close: number, item: number): void {
|
||||||
|
if (tokens[open].type !== 'bullet_list_open') {
|
||||||
|
throw new Error('bullet_list_open token expected')
|
||||||
|
}
|
||||||
|
if (tokens[close].type !== 'bullet_list_close') {
|
||||||
|
throw new Error('bullet_list_close token expected')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tokens[item].type === 'todo_item_close') {
|
||||||
|
tokens[open].type = 'todo_list_open'
|
||||||
|
tokens[close].type = 'todo_list_close'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function convertTodoItem (tokens: Token[], open: number): boolean {
|
||||||
|
const close = findListItemCloseToken(tokens, open)
|
||||||
|
if (close !== -1) {
|
||||||
|
tokens[open].type = 'todo_item_open'
|
||||||
|
tokens[close].type = 'todo_item_close'
|
||||||
|
|
||||||
|
const inline = tokens[open + 2]
|
||||||
|
|
||||||
|
if (tokens[open].attrs == null) {
|
||||||
|
tokens[open].attrs = []
|
||||||
|
}
|
||||||
|
|
||||||
|
;(tokens[open].attrs as any).push(['checked', isCheckedTodoItem(inline) ? 'true' : 'false'])
|
||||||
|
|
||||||
if (inline.children !== null) {
|
if (inline.children !== null) {
|
||||||
const newContent = inline.children[0].content.slice(4)
|
const newContent = inline.children[0].content.slice(4)
|
||||||
@ -699,24 +706,45 @@ export class MarkdownParser {
|
|||||||
const metaValues = metaTok.content.slice(5, -4).split(',')
|
const metaValues = metaTok.content.slice(5, -4).split(',')
|
||||||
for (const mv of metaValues) {
|
for (const mv of metaValues) {
|
||||||
if (mv.startsWith('todoid')) {
|
if (mv.startsWith('todoid')) {
|
||||||
;(tokens[item.start].attrs as any).push(['todoid', mv.slice(7)])
|
;(tokens[open].attrs as any).push(['todoid', mv.slice(7)])
|
||||||
}
|
}
|
||||||
if (mv.startsWith('userid')) {
|
if (mv.startsWith('userid')) {
|
||||||
;(tokens[item.start].attrs as any).push(['userid', mv.slice(7)])
|
;(tokens[open].attrs as any).push(['userid', mv.slice(7)])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
todoListStartIdx = undefined
|
|
||||||
todoListItems = []
|
|
||||||
isTodoList = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
function findListItemCloseToken (tokens: Token[], open: number): number {
|
||||||
|
if (tokens[open].type !== 'list_item_open') {
|
||||||
|
throw new Error('list_item_open token expected')
|
||||||
|
}
|
||||||
|
|
||||||
|
const level = tokens[open].level
|
||||||
|
for (let close = open + 1; close < tokens.length; close++) {
|
||||||
|
if (tokens[close].type === 'list_item_close' && tokens[close].level === level) {
|
||||||
|
return close
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
|
// todo token structure
|
||||||
|
// tokens[i].type === list_item_open
|
||||||
|
// tokens[i + 1].type === paragraph
|
||||||
|
// tokens[i + 2].type === inline
|
||||||
|
function isTodoListItem (tokens: Token[], pos: number): boolean {
|
||||||
|
return (
|
||||||
|
isListItemToken(tokens[pos]) &&
|
||||||
|
isParagraphToken(tokens[pos + 1]) &&
|
||||||
|
isInlineToken(tokens[pos + 2]) &&
|
||||||
|
startsWithTodoMarkdown(tokens[pos + 2])
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user