Skip to content

Commit 00c5fcb

Browse files
6543AbdulrhmnGhanem
authored andcommitted
Add custom emoji support (go-gitea#16004)
1 parent a292a06 commit 00c5fcb

File tree

11 files changed

+57
-23
lines changed

11 files changed

+57
-23
lines changed

custom/conf/app.example.ini

+8-3
Original file line numberDiff line numberDiff line change
@@ -1029,11 +1029,16 @@ PATH =
10291029
;; All available themes. Allow users select personalized themes regardless of the value of `DEFAULT_THEME`.
10301030
;THEMES = gitea,arc-green
10311031
;;
1032-
;;All available reactions users can choose on issues/prs and comments.
1033-
;;Values can be emoji alias (:smile:) or a unicode emoji.
1034-
;;For custom reactions, add a tightly cropped square image to public/emoji/img/reaction_name.png
1032+
;; All available reactions users can choose on issues/prs and comments.
1033+
;; Values can be emoji alias (:smile:) or a unicode emoji.
1034+
;; For custom reactions, add a tightly cropped square image to public/emoji/img/reaction_name.png
10351035
;REACTIONS = +1, -1, laugh, hooray, confused, heart, rocket, eyes
10361036
;;
1037+
;; Additional Emojis not defined in the utf8 standard
1038+
;; By default we support gitea (:gitea:), to add more copy them to public/emoji/img/emoji_name.png and add it to this config.
1039+
;; Dont mistake it for Reactions.
1040+
;CUSTOM_EMOJIS = gitea
1041+
;;
10371042
;; Whether the full name of the users should be shown where possible. If the full name isn't set, the username will be used.
10381043
;DEFAULT_SHOW_FULL_NAME = false
10391044
;;

docs/content/doc/advanced/config-cheat-sheet.en-us.md

+3
Original file line numberDiff line numberDiff line change
@@ -181,6 +181,9 @@ The following configuration set `Content-Type: application/vnd.android.package-a
181181
- `REACTIONS`: All available reactions users can choose on issues/prs and comments
182182
Values can be emoji alias (:smile:) or a unicode emoji.
183183
For custom reactions, add a tightly cropped square image to public/emoji/img/reaction_name.png
184+
- `CUSTOM_EMOJIS`: **gitea**: Additional Emojis not defined in the utf8 standard.
185+
By default we support gitea (:gitea:), to add more copy them to public/emoji/img/emoji_name.png and
186+
add it to this config.
184187
- `DEFAULT_SHOW_FULL_NAME`: **false**: Whether the full name of the users should be shown where possible. If the full name isn't set, the username will be used.
185188
- `SEARCH_REPO_DESCRIPTION`: **true**: Whether to search within description at repository search on explore page.
186189
- `USE_SERVICE_WORKER`: **true**: Whether to enable a Service Worker to cache frontend assets.

modules/markup/html.go

+8-15
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@ package markup
66

77
import (
88
"bytes"
9-
"fmt"
109
"io"
1110
"io/ioutil"
1211
"net/url"
@@ -66,7 +65,7 @@ var (
6665
blackfridayExtRegex = regexp.MustCompile(`[^:]*:user-content-`)
6766

6867
// EmojiShortCodeRegex find emoji by alias like :smile:
69-
EmojiShortCodeRegex = regexp.MustCompile(`\:[\w\+\-]+\:{1}`)
68+
EmojiShortCodeRegex = regexp.MustCompile(`:[\w\+\-]+:`)
7069
)
7170

7271
// CSS class for action keywords (e.g. "closes: #1")
@@ -460,28 +459,23 @@ func createEmoji(content, class, name string) *html.Node {
460459
return span
461460
}
462461

463-
func createCustomEmoji(alias, class string) *html.Node {
464-
462+
func createCustomEmoji(alias string) *html.Node {
465463
span := &html.Node{
466464
Type: html.ElementNode,
467465
Data: atom.Span.String(),
468466
Attr: []html.Attribute{},
469467
}
470-
if class != "" {
471-
span.Attr = append(span.Attr, html.Attribute{Key: "class", Val: class})
472-
span.Attr = append(span.Attr, html.Attribute{Key: "aria-label", Val: alias})
473-
}
468+
span.Attr = append(span.Attr, html.Attribute{Key: "class", Val: "emoji"})
469+
span.Attr = append(span.Attr, html.Attribute{Key: "aria-label", Val: alias})
474470

475471
img := &html.Node{
476472
Type: html.ElementNode,
477473
DataAtom: atom.Img,
478474
Data: "img",
479475
Attr: []html.Attribute{},
480476
}
481-
if class != "" {
482-
img.Attr = append(img.Attr, html.Attribute{Key: "alt", Val: fmt.Sprintf(`:%s:`, alias)})
483-
img.Attr = append(img.Attr, html.Attribute{Key: "src", Val: fmt.Sprintf(`%s/assets/img/emoji/%s.png`, setting.StaticURLPrefix, alias)})
484-
}
477+
img.Attr = append(img.Attr, html.Attribute{Key: "alt", Val: ":" + alias + ":"})
478+
img.Attr = append(img.Attr, html.Attribute{Key: "src", Val: setting.StaticURLPrefix + "/assets/img/emoji/" + alias + ".png"})
485479

486480
span.AppendChild(img)
487481
return span
@@ -948,9 +942,8 @@ func emojiShortCodeProcessor(ctx *RenderContext, node *html.Node) {
948942
converted := emoji.FromAlias(alias)
949943
if converted == nil {
950944
// check if this is a custom reaction
951-
s := strings.Join(setting.UI.Reactions, " ") + "gitea"
952-
if strings.Contains(s, alias) {
953-
replaceContent(node, m[0], m[1], createCustomEmoji(alias, "emoji"))
945+
if _, exist := setting.UI.CustomEmojisMap[alias]; exist {
946+
replaceContent(node, m[0], m[1], createCustomEmoji(alias))
954947
node = node.NextSibling.NextSibling
955948
start = 0
956949
continue

modules/markup/html_test.go

+12-1
Original file line numberDiff line numberDiff line change
@@ -284,7 +284,18 @@ func TestRender_emoji(t *testing.T) {
284284
test(
285285
":gitea:",
286286
`<p><span class="emoji" aria-label="gitea"><img alt=":gitea:" src="`+setting.StaticURLPrefix+`/assets/img/emoji/gitea.png"/></span></p>`)
287-
287+
test(
288+
":custom-emoji:",
289+
`<p>:custom-emoji:</p>`)
290+
setting.UI.CustomEmojisMap["custom-emoji"] = ":custom-emoji:"
291+
test(
292+
":custom-emoji:",
293+
`<p><span class="emoji" aria-label="custom-emoji"><img alt=":custom-emoji:" src="`+setting.StaticURLPrefix+`/assets/img/emoji/custom-emoji.png"/></span></p>`)
294+
test(
295+
"这是字符:1::+1: some🐊 \U0001f44d:custom-emoji: :gitea:",
296+
`<p>这是字符:1:<span class="emoji" aria-label="thumbs up">👍</span> some<span class="emoji" aria-label="crocodile">🐊</span> `+
297+
`<span class="emoji" aria-label="thumbs up">👍</span><span class="emoji" aria-label="custom-emoji"><img alt=":custom-emoji:" src="`+setting.StaticURLPrefix+`/assets/img/emoji/custom-emoji.png"/></span> `+
298+
`<span class="emoji" aria-label="gitea"><img alt=":gitea:" src="`+setting.StaticURLPrefix+`/assets/img/emoji/gitea.png"/></span></p>`)
288299
test(
289300
"Some text with 😄 in the middle",
290301
`<p>Some text with <span class="emoji" aria-label="grinning face with smiling eyes">😄</span> in the middle</p>`)

modules/setting/setting.go

+9-1
Original file line numberDiff line numberDiff line change
@@ -208,7 +208,9 @@ var (
208208
DefaultTheme string
209209
Themes []string
210210
Reactions []string
211-
ReactionsMap map[string]bool
211+
ReactionsMap map[string]bool `ini:"-"`
212+
CustomEmojis []string
213+
CustomEmojisMap map[string]string `ini:"-"`
212214
SearchRepoDescription bool
213215
UseServiceWorker bool
214216

@@ -256,6 +258,8 @@ var (
256258
DefaultTheme: `gitea`,
257259
Themes: []string{`gitea`, `arc-green`},
258260
Reactions: []string{`+1`, `-1`, `laugh`, `hooray`, `confused`, `heart`, `rocket`, `eyes`},
261+
CustomEmojis: []string{`gitea`},
262+
CustomEmojisMap: map[string]string{"gitea": ":gitea:"},
259263
Notification: struct {
260264
MinTimeout time.Duration
261265
TimeoutStep time.Duration
@@ -983,6 +987,10 @@ func NewContext() {
983987
for _, reaction := range UI.Reactions {
984988
UI.ReactionsMap[reaction] = true
985989
}
990+
UI.CustomEmojisMap = make(map[string]string)
991+
for _, emoji := range UI.CustomEmojis {
992+
UI.CustomEmojisMap[emoji] = ":" + emoji + ":"
993+
}
986994
}
987995

988996
func parseAuthorizedPrincipalsAllow(values []string) ([]string, bool) {

modules/structs/settings.go

+1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ type GeneralRepoSettings struct {
1818
type GeneralUISettings struct {
1919
DefaultTheme string `json:"default_theme"`
2020
AllowedReactions []string `json:"allowed_reactions"`
21+
CustomEmojis []string `json:"custom_emojis"`
2122
}
2223

2324
// GeneralAPISettings contains global api settings exposed by it

modules/templates/helper.go

+3
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,9 @@ func NewFuncMap() []template.FuncMap {
9090
"AllowedReactions": func() []string {
9191
return setting.UI.Reactions
9292
},
93+
"CustomEmojis": func() map[string]string {
94+
return setting.UI.CustomEmojisMap
95+
},
9396
"Safe": Safe,
9497
"SafeJS": SafeJS,
9598
"JSEscape": JSEscape,

routers/api/v1/settings/settings.go

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ func GetGeneralUISettings(ctx *context.APIContext) {
2525
ctx.JSON(http.StatusOK, api.GeneralUISettings{
2626
DefaultTheme: setting.UI.DefaultTheme,
2727
AllowedReactions: setting.UI.Reactions,
28+
CustomEmojis: setting.UI.CustomEmojis,
2829
})
2930
}
3031

templates/base/head.tmpl

+1
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@
3030
AppVer: '{{AppVer}}',
3131
AppSubUrl: '{{AppSubUrl}}',
3232
AssetUrlPrefix: '{{AssetUrlPrefix}}',
33+
CustomEmojis: {{CustomEmojis}},
3334
UseServiceWorker: {{UseServiceWorker}},
3435
csrf: '{{.CsrfToken}}',
3536
HighlightJS: {{if .RequireHighlightJS}}true{{else}}false{{end}},

templates/swagger/v1_json.tmpl

+7
Original file line numberDiff line numberDiff line change
@@ -14481,6 +14481,13 @@
1448114481
},
1448214482
"x-go-name": "AllowedReactions"
1448314483
},
14484+
"custom_emojis": {
14485+
"type": "array",
14486+
"items": {
14487+
"type": "string"
14488+
},
14489+
"x-go-name": "CustomEmojis"
14490+
},
1448414491
"default_theme": {
1448514492
"type": "string",
1448614493
"x-go-name": "DefaultTheme"

web_src/js/features/emoji.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,9 @@
11
import emojis from '../../../assets/emoji.json';
22

33
const {AssetUrlPrefix} = window.config;
4+
const {CustomEmojis} = window.config;
45

5-
const tempMap = {gitea: ':gitea:'};
6+
const tempMap = {...CustomEmojis};
67
for (const {emoji, aliases} of emojis) {
78
for (const alias of aliases || []) {
89
tempMap[alias] = emoji;
@@ -23,8 +24,8 @@ for (const key of emojiKeys) {
2324
// retrieve HTML for given emoji name
2425
export function emojiHTML(name) {
2526
let inner;
26-
if (name === 'gitea') {
27-
inner = `<img alt=":${name}:" src="${AssetUrlPrefix}/img/emoji/gitea.png">`;
27+
if (Object.prototype.hasOwnProperty.call(CustomEmojis, name)) {
28+
inner = `<img alt=":${name}:" src="${AssetUrlPrefix}/img/emoji/${name}.png">`;
2829
} else {
2930
inner = emojiString(name);
3031
}

0 commit comments

Comments
 (0)