mirror of
https://github.com/diamondburned/cchat-gtk.git
synced 2025-01-25 03:16:49 +00:00
136 lines
2.9 KiB
Go
136 lines
2.9 KiB
Go
package attrmap
|
|
|
|
import (
|
|
"fmt"
|
|
"html"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/diamondburned/cchat/text"
|
|
)
|
|
|
|
type AppendMap struct {
|
|
appended map[int]string // for opening tags
|
|
prepended map[int]string // for closing tags
|
|
indices []int
|
|
}
|
|
|
|
func NewAppendedMap() AppendMap {
|
|
return AppendMap{
|
|
appended: map[int]string{},
|
|
prepended: map[int]string{},
|
|
indices: []int{},
|
|
}
|
|
}
|
|
|
|
func (a *AppendMap) appendIndex(ind int) {
|
|
// Backwards search which should make things faster.
|
|
for i := len(a.indices) - 1; i >= 0; i-- {
|
|
if a.indices[i] == ind {
|
|
return
|
|
}
|
|
}
|
|
|
|
a.indices = append(a.indices, ind)
|
|
}
|
|
|
|
func (a *AppendMap) Anchor(start, end int, href string) {
|
|
a.AnchorNU(start, end, href)
|
|
}
|
|
|
|
// AnchorNU makes a new <a> tag without underlines and colors.
|
|
func (a *AppendMap) AnchorNU(start, end int, href string) {
|
|
a.Openf(start, `<a href="%s">`, html.EscapeString(href))
|
|
a.Close(end, "</a>")
|
|
// a.Anchor(start, end, href)
|
|
a.Span(start, end, `underline="none"`)
|
|
}
|
|
|
|
func (a *AppendMap) Span(start, end int, attrs ...string) {
|
|
a.Openf(start, "<span %s>", strings.Join(attrs, " "))
|
|
a.Close(end, "</span>")
|
|
}
|
|
|
|
// Pad inserts 2 spaces into start and end. It ensures that not more than 1
|
|
// space is inserted.
|
|
func (a *AppendMap) Pad(start, end int) {
|
|
// Ensure that the starting point does not already have a space.
|
|
if !posHaveSpace(a.appended, start) {
|
|
a.Open(start, " ")
|
|
}
|
|
if !posHaveSpace(a.prepended, end) {
|
|
a.Close(end, " ")
|
|
}
|
|
}
|
|
|
|
func posHaveSpace(tags map[int]string, pos int) bool {
|
|
tg, ok := tags[pos]
|
|
if !ok || len(tg) == 0 {
|
|
return false
|
|
}
|
|
|
|
// Check trailing spaces.
|
|
if tg[0] == ' ' {
|
|
return true
|
|
}
|
|
if tg[len(tg)-1] == ' ' {
|
|
return true
|
|
}
|
|
|
|
// Check spaces mid-tag. This works because strings are always escaped.
|
|
return strings.Contains(tg, "> <")
|
|
}
|
|
|
|
func (a *AppendMap) Pair(start, end int, open, close string) {
|
|
a.Open(start, open)
|
|
a.Close(end, close)
|
|
}
|
|
|
|
func (a *AppendMap) Openf(ind int, f string, argv ...interface{}) {
|
|
a.Open(ind, fmt.Sprintf(f, argv...))
|
|
}
|
|
|
|
func (a *AppendMap) Open(ind int, attr string) {
|
|
if str, ok := a.appended[ind]; ok {
|
|
a.appended[ind] = str + attr // append
|
|
return
|
|
}
|
|
|
|
a.appended[ind] = attr
|
|
a.appendIndex(ind)
|
|
}
|
|
|
|
func (a *AppendMap) Close(ind int, attr string) {
|
|
if str, ok := a.prepended[ind]; ok {
|
|
a.prepended[ind] = attr + str // prepend
|
|
return
|
|
}
|
|
|
|
a.prepended[ind] = attr
|
|
a.appendIndex(ind)
|
|
}
|
|
|
|
func (a AppendMap) Get(ind int) (tags string) {
|
|
if t, ok := a.appended[ind]; ok {
|
|
tags += t
|
|
}
|
|
if t, ok := a.prepended[ind]; ok {
|
|
tags += t
|
|
}
|
|
return
|
|
}
|
|
|
|
func (a *AppendMap) Finalize(strlen int) []int {
|
|
// make sure there's always a closing tag at the end so the entire string
|
|
// gets flushed.
|
|
a.Close(strlen, "")
|
|
sort.Ints(a.indices)
|
|
return a.indices
|
|
}
|
|
|
|
// CoverAll returns true if the given start and end covers the entire text
|
|
// segment.
|
|
func CoverAll(txt text.Rich, start, end int) bool {
|
|
return start == 0 && end == len(txt.Content)
|
|
}
|