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.Openf(start, ``, html.EscapeString(href)) a.Close(end, "") } // AnchorNU makes a new tag without underlines and colors. func (a *AppendMap) AnchorNU(start, end int, href string) { a.Anchor(start, end, href) a.Span(start, end, `underline="none"`) } func (a *AppendMap) Span(start, end int, attrs ...string) { a.Openf(start, "", strings.Join(attrs, " ")) a.Close(end, "") } // 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) }