mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-26 20:11:02 +00:00 
			
		
		
		
	Backport #25184 by @jtran Fixes #25160. `data-source-position` of checkboxes in a task list was incorrect whenever there was YAML front matter. This would result in issue content or PR descriptions getting corrupted with random `x` or space characters when a user checked or unchecked a task. Co-authored-by: Jonathan Tran <jon@allspice.io>
		
			
				
	
	
		
			217 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			217 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2020 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package markdown
 | |
| 
 | |
| import (
 | |
| 	"strconv"
 | |
| 
 | |
| 	"github.com/yuin/goldmark/ast"
 | |
| )
 | |
| 
 | |
| // Details is a block that contains Summary and details
 | |
| type Details struct {
 | |
| 	ast.BaseBlock
 | |
| }
 | |
| 
 | |
| // Dump implements Node.Dump .
 | |
| func (n *Details) Dump(source []byte, level int) {
 | |
| 	ast.DumpHelper(n, source, level, nil, nil)
 | |
| }
 | |
| 
 | |
| // KindDetails is the NodeKind for Details
 | |
| var KindDetails = ast.NewNodeKind("Details")
 | |
| 
 | |
| // Kind implements Node.Kind.
 | |
| func (n *Details) Kind() ast.NodeKind {
 | |
| 	return KindDetails
 | |
| }
 | |
| 
 | |
| // NewDetails returns a new Paragraph node.
 | |
| func NewDetails() *Details {
 | |
| 	return &Details{
 | |
| 		BaseBlock: ast.BaseBlock{},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // IsDetails returns true if the given node implements the Details interface,
 | |
| // otherwise false.
 | |
| func IsDetails(node ast.Node) bool {
 | |
| 	_, ok := node.(*Details)
 | |
| 	return ok
 | |
| }
 | |
| 
 | |
| // Summary is a block that contains the summary of details block
 | |
| type Summary struct {
 | |
| 	ast.BaseBlock
 | |
| }
 | |
| 
 | |
| // Dump implements Node.Dump .
 | |
| func (n *Summary) Dump(source []byte, level int) {
 | |
| 	ast.DumpHelper(n, source, level, nil, nil)
 | |
| }
 | |
| 
 | |
| // KindSummary is the NodeKind for Summary
 | |
| var KindSummary = ast.NewNodeKind("Summary")
 | |
| 
 | |
| // Kind implements Node.Kind.
 | |
| func (n *Summary) Kind() ast.NodeKind {
 | |
| 	return KindSummary
 | |
| }
 | |
| 
 | |
| // NewSummary returns a new Summary node.
 | |
| func NewSummary() *Summary {
 | |
| 	return &Summary{
 | |
| 		BaseBlock: ast.BaseBlock{},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // IsSummary returns true if the given node implements the Summary interface,
 | |
| // otherwise false.
 | |
| func IsSummary(node ast.Node) bool {
 | |
| 	_, ok := node.(*Summary)
 | |
| 	return ok
 | |
| }
 | |
| 
 | |
| // TaskCheckBoxListItem is a block that represents a list item of a markdown block with a checkbox
 | |
| type TaskCheckBoxListItem struct {
 | |
| 	*ast.ListItem
 | |
| 	IsChecked      bool
 | |
| 	SourcePosition int
 | |
| }
 | |
| 
 | |
| // KindTaskCheckBoxListItem is the NodeKind for TaskCheckBoxListItem
 | |
| var KindTaskCheckBoxListItem = ast.NewNodeKind("TaskCheckBoxListItem")
 | |
| 
 | |
| // Dump implements Node.Dump .
 | |
| func (n *TaskCheckBoxListItem) Dump(source []byte, level int) {
 | |
| 	m := map[string]string{}
 | |
| 	m["IsChecked"] = strconv.FormatBool(n.IsChecked)
 | |
| 	m["SourcePosition"] = strconv.FormatInt(int64(n.SourcePosition), 10)
 | |
| 	ast.DumpHelper(n, source, level, m, nil)
 | |
| }
 | |
| 
 | |
| // Kind implements Node.Kind.
 | |
| func (n *TaskCheckBoxListItem) Kind() ast.NodeKind {
 | |
| 	return KindTaskCheckBoxListItem
 | |
| }
 | |
| 
 | |
| // NewTaskCheckBoxListItem returns a new TaskCheckBoxListItem node.
 | |
| func NewTaskCheckBoxListItem(listItem *ast.ListItem) *TaskCheckBoxListItem {
 | |
| 	return &TaskCheckBoxListItem{
 | |
| 		ListItem: listItem,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // IsTaskCheckBoxListItem returns true if the given node implements the TaskCheckBoxListItem interface,
 | |
| // otherwise false.
 | |
| func IsTaskCheckBoxListItem(node ast.Node) bool {
 | |
| 	_, ok := node.(*TaskCheckBoxListItem)
 | |
| 	return ok
 | |
| }
 | |
| 
 | |
| // Icon is an inline for a fomantic icon
 | |
| type Icon struct {
 | |
| 	ast.BaseInline
 | |
| 	Name []byte
 | |
| }
 | |
| 
 | |
| // Dump implements Node.Dump .
 | |
| func (n *Icon) Dump(source []byte, level int) {
 | |
| 	m := map[string]string{}
 | |
| 	m["Name"] = string(n.Name)
 | |
| 	ast.DumpHelper(n, source, level, m, nil)
 | |
| }
 | |
| 
 | |
| // KindIcon is the NodeKind for Icon
 | |
| var KindIcon = ast.NewNodeKind("Icon")
 | |
| 
 | |
| // Kind implements Node.Kind.
 | |
| func (n *Icon) Kind() ast.NodeKind {
 | |
| 	return KindIcon
 | |
| }
 | |
| 
 | |
| // NewIcon returns a new Paragraph node.
 | |
| func NewIcon(name string) *Icon {
 | |
| 	return &Icon{
 | |
| 		BaseInline: ast.BaseInline{},
 | |
| 		Name:       []byte(name),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // IsIcon returns true if the given node implements the Icon interface,
 | |
| // otherwise false.
 | |
| func IsIcon(node ast.Node) bool {
 | |
| 	_, ok := node.(*Icon)
 | |
| 	return ok
 | |
| }
 | |
| 
 | |
| // ColorPreview is an inline for a color preview
 | |
| type ColorPreview struct {
 | |
| 	ast.BaseInline
 | |
| 	Color []byte
 | |
| }
 | |
| 
 | |
| // Dump implements Node.Dump.
 | |
| func (n *ColorPreview) Dump(source []byte, level int) {
 | |
| 	m := map[string]string{}
 | |
| 	m["Color"] = string(n.Color)
 | |
| 	ast.DumpHelper(n, source, level, m, nil)
 | |
| }
 | |
| 
 | |
| // KindColorPreview is the NodeKind for ColorPreview
 | |
| var KindColorPreview = ast.NewNodeKind("ColorPreview")
 | |
| 
 | |
| // Kind implements Node.Kind.
 | |
| func (n *ColorPreview) Kind() ast.NodeKind {
 | |
| 	return KindColorPreview
 | |
| }
 | |
| 
 | |
| // NewColorPreview returns a new Span node.
 | |
| func NewColorPreview(color []byte) *ColorPreview {
 | |
| 	return &ColorPreview{
 | |
| 		BaseInline: ast.BaseInline{},
 | |
| 		Color:      color,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // IsColorPreview returns true if the given node implements the ColorPreview interface,
 | |
| // otherwise false.
 | |
| func IsColorPreview(node ast.Node) bool {
 | |
| 	_, ok := node.(*ColorPreview)
 | |
| 	return ok
 | |
| }
 | |
| 
 | |
| const (
 | |
| 	AttentionNote    string = "Note"
 | |
| 	AttentionWarning string = "Warning"
 | |
| )
 | |
| 
 | |
| // Attention is an inline for a color preview
 | |
| type Attention struct {
 | |
| 	ast.BaseInline
 | |
| 	AttentionType string
 | |
| }
 | |
| 
 | |
| // Dump implements Node.Dump.
 | |
| func (n *Attention) Dump(source []byte, level int) {
 | |
| 	m := map[string]string{}
 | |
| 	m["AttentionType"] = n.AttentionType
 | |
| 	ast.DumpHelper(n, source, level, m, nil)
 | |
| }
 | |
| 
 | |
| // KindAttention is the NodeKind for Attention
 | |
| var KindAttention = ast.NewNodeKind("Attention")
 | |
| 
 | |
| // Kind implements Node.Kind.
 | |
| func (n *Attention) Kind() ast.NodeKind {
 | |
| 	return KindAttention
 | |
| }
 | |
| 
 | |
| // NewAttention returns a new Attention node.
 | |
| func NewAttention(attentionType string) *Attention {
 | |
| 	return &Attention{
 | |
| 		BaseInline:    ast.BaseInline{},
 | |
| 		AttentionType: attentionType,
 | |
| 	}
 | |
| }
 |