// Copyright 2021 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package markdown import ( "bytes" "strings" ) type Empty struct { Position } func (b *Empty) PrintHTML(buf *bytes.Buffer) {} func (b *Empty) printMarkdown(*bytes.Buffer, mdState) {} type Paragraph struct { Position Text *Text } func (b *Paragraph) PrintHTML(buf *bytes.Buffer) { buf.WriteString("

") b.Text.PrintHTML(buf) buf.WriteString("

\n") } func (b *Paragraph) printMarkdown(buf *bytes.Buffer, s mdState) { // // Ignore prefix when in a list. // if s.bullet == 0 { // buf.WriteString(s.prefix) // } b.Text.printMarkdown(buf, s) } type paraBuilder struct { text []string table *tableBuilder } func (b *paraBuilder) extend(p *parseState, s line) (line, bool) { return s, false } func (b *paraBuilder) build(p buildState) Block { if b.table != nil { return b.table.build(p) } s := strings.Join(b.text, "\n") for s != "" { end, ok := parseLinkRefDef(p, s) if !ok { break } s = s[skipSpace(s, end):] } if s == "" { return &Empty{p.pos()} } // Recompute EndLine because a line of b.text // might have been taken away to start a table. pos := p.pos() pos.EndLine = pos.StartLine + len(b.text) - 1 return &Paragraph{ pos, p.newText(pos, s), } } func newPara(p *parseState, s line) (line, bool) { // Process paragraph continuation text or start new paragraph. b := p.para() indented := p.lineDepth == len(p.stack)-2 // fully indented, not playing "pargraph continuation text" games text := s.trimSpaceString() if b != nil && b.table != nil { if indented && text != "" && text != "|" { // Continue table. b.table.addRow(text) return line{}, true } // Blank or unindented line ends table. // (So does a new block structure, but the caller has checked that already.) // So does a line with just a pipe: // https://github.com/github/cmark-gfm/pull/127 and // https://github.com/github/cmark-gfm/pull/128 // fixed a buffer overread by rejecting | by itself as a table line. // That seems to violate the spec, but we will play along. b = nil } // If we are looking for tables and this is a table start, start a table. if p.Table && b != nil && indented && len(b.text) > 0 && isTableStart(b.text[len(b.text)-1], text) { hdr := b.text[len(b.text)-1] b.text = b.text[:len(b.text)-1] tb := new(paraBuilder) p.addBlock(tb) tb.table = new(tableBuilder) tb.table.start(hdr, text) return line{}, true } if b != nil { for i := p.lineDepth; i < len(p.stack); i++ { p.stack[i].pos.EndLine = p.lineno } } else { // Note: Ends anything without a matching prefix. b = new(paraBuilder) p.addBlock(b) } b.text = append(b.text, text) return line{}, true }