|
- package parser
-
- import (
- "bytes"
- "github.com/yuin/goldmark/ast"
- "github.com/yuin/goldmark/text"
- "github.com/yuin/goldmark/util"
- "regexp"
- )
-
- type rawHTMLParser struct {
- }
-
- var defaultRawHTMLParser = &rawHTMLParser{}
-
- // NewRawHTMLParser return a new InlineParser that can parse
- // inline htmls
- func NewRawHTMLParser() InlineParser {
- return defaultRawHTMLParser
- }
-
- func (s *rawHTMLParser) Trigger() []byte {
- return []byte{'<'}
- }
-
- func (s *rawHTMLParser) Parse(parent ast.Node, block text.Reader, pc Context) ast.Node {
- line, _ := block.PeekLine()
- if len(line) > 1 && util.IsAlphaNumeric(line[1]) {
- return s.parseMultiLineRegexp(openTagRegexp, block, pc)
- }
- if len(line) > 2 && line[1] == '/' && util.IsAlphaNumeric(line[2]) {
- return s.parseMultiLineRegexp(closeTagRegexp, block, pc)
- }
- if bytes.HasPrefix(line, []byte("<!--")) {
- return s.parseMultiLineRegexp(commentRegexp, block, pc)
- }
- if bytes.HasPrefix(line, []byte("<?")) {
- return s.parseSingleLineRegexp(processingInstructionRegexp, block, pc)
- }
- if len(line) > 2 && line[1] == '!' && line[2] >= 'A' && line[2] <= 'Z' {
- return s.parseSingleLineRegexp(declRegexp, block, pc)
- }
- if bytes.HasPrefix(line, []byte("<![CDATA[")) {
- return s.parseMultiLineRegexp(cdataRegexp, block, pc)
- }
- return nil
- }
-
- var tagnamePattern = `([A-Za-z][A-Za-z0-9-]*)`
- var attributePattern = `(?:\s+[a-zA-Z_:][a-zA-Z0-9:._-]*(?:\s*=\s*(?:[^\"'=<>` + "`" + `\x00-\x20]+|'[^']*'|"[^"]*"))?)`
- var openTagRegexp = regexp.MustCompile("^<" + tagnamePattern + attributePattern + `*\s*/?>`)
- var closeTagRegexp = regexp.MustCompile("^</" + tagnamePattern + `\s*>`)
- var commentRegexp = regexp.MustCompile(`^<!---->|<!--(?:-?[^>-])(?:-?[^-])*-->`)
- var processingInstructionRegexp = regexp.MustCompile(`^(?:<\?).*?(?:\?>)`)
- var declRegexp = regexp.MustCompile(`^<![A-Z]+\s+[^>]*>`)
- var cdataRegexp = regexp.MustCompile(`<!\[CDATA\[[\s\S]*?\]\]>`)
-
- func (s *rawHTMLParser) parseSingleLineRegexp(reg *regexp.Regexp, block text.Reader, pc Context) ast.Node {
- line, segment := block.PeekLine()
- match := reg.FindSubmatchIndex(line)
- if match == nil {
- return nil
- }
- node := ast.NewRawHTML()
- node.Segments.Append(segment.WithStop(segment.Start + match[1]))
- block.Advance(match[1])
- return node
- }
-
- var dummyMatch = [][]byte{}
-
- func (s *rawHTMLParser) parseMultiLineRegexp(reg *regexp.Regexp, block text.Reader, pc Context) ast.Node {
- sline, ssegment := block.Position()
- if block.Match(reg) {
- node := ast.NewRawHTML()
- eline, esegment := block.Position()
- block.SetPosition(sline, ssegment)
- for {
- line, segment := block.PeekLine()
- if line == nil {
- break
- }
- l, _ := block.Position()
- start := segment.Start
- if l == sline {
- start = ssegment.Start
- }
- end := segment.Stop
- if l == eline {
- end = esegment.Start
- }
-
- node.Segments.Append(text.NewSegment(start, end))
- if l == eline {
- block.Advance(end - start)
- break
- } else {
- block.AdvanceLine()
- }
- }
- return node
- }
- return nil
- }
-
- func (s *rawHTMLParser) CloseBlock(parent ast.Node, pc Context) {
- // nothing to do
- }
|