mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-29 19:02:26 -05:00 
			
		
		
		
	* [chore] Refactor HTML templates and CSS * eslint * ignore "Local" * rss tests * fiddle with OG just a tiny bit * dick around with polls a bit more so SR stops saying "clickable" * remove break * oh lord * don't lazy load avatar * fix ogmeta tests * clean up some cruft * catch remaining calls to c.HTML * fix error rendering + stack overflow in tag * allow templating attributes * fix indent * set aria-hidden on status complementary content, since it's already present in the label anyway * tidy up templating calls a little * try to make styling a bit more consistent + readable * fix up some remaining CSS issues * fix up reports
		
			
				
	
	
		
			117 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			117 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // GoToSocial
 | |
| // Copyright (C) GoToSocial Authors admin@gotosocial.org
 | |
| // SPDX-License-Identifier: AGPL-3.0-or-later
 | |
| //
 | |
| // This program is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU Affero General Public License as published by
 | |
| // the Free Software Foundation, either version 3 of the License, or
 | |
| // (at your option) any later version.
 | |
| //
 | |
| // This program is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| // GNU Affero General Public License for more details.
 | |
| //
 | |
| // You should have received a copy of the GNU Affero General Public License
 | |
| // along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| package text
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"html"
 | |
| 	"html/template"
 | |
| 
 | |
| 	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
 | |
| 	"github.com/superseriousbusiness/gotosocial/internal/regexes"
 | |
| )
 | |
| 
 | |
| // EmojifyWeb replaces emoji shortcodes like `:example:` in the given HTML
 | |
| // fragment with `<img>` tags suitable for rendering on the web frontend.
 | |
| func EmojifyWeb(emojis []apimodel.Emoji, html template.HTML) template.HTML {
 | |
| 	out := emojify(
 | |
| 		emojis,
 | |
| 		string(html),
 | |
| 		func(url, code string, buf *bytes.Buffer) {
 | |
| 			buf.WriteString(`<img src="`)
 | |
| 			buf.WriteString(url)
 | |
| 			buf.WriteString(`" title=":`)
 | |
| 			buf.WriteString(code)
 | |
| 			buf.WriteString(`:" alt=":`)
 | |
| 			buf.WriteString(code)
 | |
| 			buf.WriteString(`:" class="emoji" `)
 | |
| 			// Lazy load emojis when
 | |
| 			// they scroll into view.
 | |
| 			buf.WriteString(`loading="lazy" `)
 | |
| 			// Limit size to avoid showing
 | |
| 			// huge emojis when unstyled.
 | |
| 			buf.WriteString(`width="25" height="25"/>`)
 | |
| 		},
 | |
| 	)
 | |
| 
 | |
| 	// If input was safe,
 | |
| 	// we can trust output.
 | |
| 	return template.HTML(out) // #nosec G203
 | |
| }
 | |
| 
 | |
| // EmojifyRSS replaces emoji shortcodes like `:example:` in the given text
 | |
| // fragment with `<img>` tags suitable for rendering as RSS content.
 | |
| func EmojifyRSS(emojis []apimodel.Emoji, text string) string {
 | |
| 	return emojify(
 | |
| 		emojis,
 | |
| 		text,
 | |
| 		func(url, code string, buf *bytes.Buffer) {
 | |
| 			buf.WriteString(`<img src="`)
 | |
| 			buf.WriteString(url)
 | |
| 			buf.WriteString(`" title=":`)
 | |
| 			buf.WriteString(code)
 | |
| 			buf.WriteString(`:" alt=":`)
 | |
| 			buf.WriteString(code)
 | |
| 			buf.WriteString(`:" `)
 | |
| 			// Limit size to avoid showing
 | |
| 			// huge emojis in RSS readers.
 | |
| 			buf.WriteString(`width="25" height="25"/>`)
 | |
| 		},
 | |
| 	)
 | |
| }
 | |
| 
 | |
| // Demojify replaces emoji shortcodes like `:example:` in the given text
 | |
| // fragment with empty strings, essentially stripping them from the text.
 | |
| // This is useful for text used in OG Meta headers.
 | |
| func Demojify(text string) string {
 | |
| 	return regexes.EmojiFinder.ReplaceAllString(text, "")
 | |
| }
 | |
| 
 | |
| func emojify(
 | |
| 	emojis []apimodel.Emoji,
 | |
| 	input string,
 | |
| 	write func(url, code string, buf *bytes.Buffer),
 | |
| ) string {
 | |
| 	// Build map of shortcodes. Normalize each
 | |
| 	// shortcode by readding closing colons.
 | |
| 	emojisMap := make(map[string]apimodel.Emoji, len(emojis))
 | |
| 	for _, emoji := range emojis {
 | |
| 		shortcode := ":" + emoji.Shortcode + ":"
 | |
| 		emojisMap[shortcode] = emoji
 | |
| 	}
 | |
| 
 | |
| 	return regexes.ReplaceAllStringFunc(
 | |
| 		regexes.EmojiFinder,
 | |
| 		input,
 | |
| 		func(shortcode string, buf *bytes.Buffer) string {
 | |
| 			// Look for emoji with this shortcode.
 | |
| 			emoji, ok := emojisMap[shortcode]
 | |
| 			if !ok {
 | |
| 				return shortcode
 | |
| 			}
 | |
| 
 | |
| 			// Escape raw emoji content.
 | |
| 			url := html.EscapeString(emoji.URL)
 | |
| 			code := html.EscapeString(emoji.Shortcode)
 | |
| 
 | |
| 			// Write emoji repr to buffer.
 | |
| 			write(url, code, buf)
 | |
| 			return buf.String()
 | |
| 		},
 | |
| 	)
 | |
| }
 |