From a1a7b67e633ad9afd9409db5d5b3ee05e6d270e8 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 14 Aug 2021 21:10:24 -0400
Subject: [PATCH 01/30] Group custom emojis by pack in emoji picker

---
 src/components/emoji_picker/emoji_picker.js   | 29 ++++++++++++++-----
 src/components/emoji_picker/emoji_picker.scss | 13 +++++++++
 src/components/emoji_picker/emoji_picker.vue  | 10 +++++++
 3 files changed, 45 insertions(+), 7 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 6b589079..ce7738ee 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -182,17 +182,32 @@ const EmojiPicker = {
     customEmojiBuffer () {
       return this.filteredEmoji.slice(0, this.customEmojiBufferSlice)
     },
+    groupedCustomEmojis () {
+      const packOf = emoji => (emoji.tags.filter(k => k.startsWith('pack:'))[0] || '').slice(5)
+      return this.customEmojiBuffer.reduce((res, emoji) => {
+        const pack = packOf(emoji)
+        if (!res[pack]) {
+          res[pack] = {
+            id: `custom-${pack}`,
+            text: pack,
+            /// FIXME
+            // icon: 'smile-beam',
+            image: emoji.imageUrl,
+            emojis: []
+          }
+        }
+        res[pack].emojis.push(emoji)
+        return res
+      }, {})
+    },
     emojis () {
       const standardEmojis = this.$store.state.instance.emoji || []
-      const customEmojis = this.customEmojiBuffer
+      // const customEmojis = this.customEmojiBuffer
 
       return [
-        {
-          id: 'custom',
-          text: this.$t('emoji.custom'),
-          icon: 'smile-beam',
-          emojis: customEmojis
-        },
+        ...Object
+          .keys(this.groupedCustomEmojis)
+          .map(k => this.groupedCustomEmojis[k]),
         {
           id: 'standard',
           text: this.$t('emoji.unicode'),
diff --git a/src/components/emoji_picker/emoji_picker.scss b/src/components/emoji_picker/emoji_picker.scss
index ec711758..46cfa1c8 100644
--- a/src/components/emoji_picker/emoji_picker.scss
+++ b/src/components/emoji_picker/emoji_picker.scss
@@ -18,6 +18,19 @@
   --lightText: var(--popoverLightText, $fallback--lightText);
   --icon: var(--popoverIcon, $fallback--icon);
 
+  &-header-image {
+    display: inline-flex;
+    justify-content: center;
+    align-items: center;
+    width: 30px;
+    height: 24px;
+    img {
+      max-width: 100%;
+      max-height: 100%;
+      object-fit: contain;
+    }
+  }
+
   .keep-open,
   .too-many-emoji {
     padding: 7px;
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 3262a3d9..65aa5bb6 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -13,7 +13,17 @@
           :title="group.text"
           @click.prevent="highlight(group.id)"
         >
+          <span
+            v-if="group.image"
+            class="emoji-picker-header-image"
+          >
+            <img
+              :alt="group.text"
+              :src="group.image"
+            >
+          </span>
           <FAIcon
+            v-else
             :icon="group.icon"
             fixed-width
           />

From 44bedcd725378eb32784999ec92f136ec83c72e2 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 14 Aug 2021 21:23:45 -0400
Subject: [PATCH 02/30] Fix load more emoji action

---
 src/components/emoji_picker/emoji_picker.js |  5 ++++-
 src/modules/instance.js                     | 12 +++++++++++-
 2 files changed, 15 insertions(+), 2 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index ce7738ee..ef9c3745 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -97,7 +97,7 @@ const EmojiPicker = {
       }
     },
     triggerLoadMore (target) {
-      const ref = this.$refs['group-end-custom']
+      const ref = this.$refs[`group-end-${this.lastNonUnicodeGroupId}`][0]
       if (!ref) return
       const bottom = ref.offsetTop + ref.offsetHeight
 
@@ -216,6 +216,9 @@ const EmojiPicker = {
         }
       ]
     },
+    lastNonUnicodeGroupId () {
+      return this.emojis[this.emojis.length - 2].id
+    },
     emojisView () {
       return this.emojis.filter(value => value.emojis.length > 0)
     },
diff --git a/src/modules/instance.js b/src/modules/instance.js
index 220463ca..c6d124b9 100644
--- a/src/modules/instance.js
+++ b/src/modules/instance.js
@@ -163,6 +163,16 @@ const instance = {
         if (res.ok) {
           const result = await res.json()
           const values = Array.isArray(result) ? Object.assign({}, ...result) : result
+          const caseInsensitiveStrCmp = (a, b) => {
+            const la = a.toLowerCase()
+            const lb = b.toLowerCase()
+            return la > lb ? 1 : (la < lb ? -1 : 0)
+          }
+          const byPackThenByName = (a, b) => {
+            const packOf = emoji => (emoji.tags.filter(k => k.startsWith('pack:'))[0] || '').slice(5)
+            return caseInsensitiveStrCmp(packOf(a), packOf(b)) || caseInsensitiveStrCmp(a.displayText, b.displayText)
+          }
+
           const emoji = Object.entries(values).map(([key, value]) => {
             const imageUrl = value.image_url
             return {
@@ -173,7 +183,7 @@ const instance = {
             }
             // Technically could use tags but those are kinda useless right now,
             // should have been "pack" field, that would be more useful
-          }).sort((a, b) => a.displayText.toLowerCase() > b.displayText.toLowerCase() ? 1 : -1)
+          }).sort(byPackThenByName)
           commit('setInstanceOption', { name: 'customEmoji', value: emoji })
         } else {
           throw (res)

From aea760ce1cb5b2ec42cb2133f446dca07ed3516e Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 14 Aug 2021 21:50:58 -0400
Subject: [PATCH 03/30] Display all emoji groups on emoji picker header

---
 src/components/emoji_picker/emoji_picker.js   | 28 +++++++++++++++++--
 src/components/emoji_picker/emoji_picker.scss |  6 +++-
 src/components/emoji_picker/emoji_picker.vue  |  4 +--
 3 files changed, 33 insertions(+), 5 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index ef9c3745..4b322f7d 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -37,6 +37,8 @@ const filterByKeyword = (list, keyword = '') => {
   return orderedEmojiList.flat()
 }
 
+const packOf = emoji => (emoji.tags.filter(k => k.startsWith('pack:'))[0] || '').slice(5)
+
 const EmojiPicker = {
   props: {
     enableStickerPicker: {
@@ -173,9 +175,12 @@ const EmojiPicker = {
       }
       return 0
     },
+    allEmojis () {
+      return this.$store.state.instance.customEmoji || []
+    },
     filteredEmoji () {
       return filterByKeyword(
-        this.$store.state.instance.customEmoji || [],
+        this.allEmojis,
         this.keyword
       )
     },
@@ -183,7 +188,6 @@ const EmojiPicker = {
       return this.filteredEmoji.slice(0, this.customEmojiBufferSlice)
     },
     groupedCustomEmojis () {
-      const packOf = emoji => (emoji.tags.filter(k => k.startsWith('pack:'))[0] || '').slice(5)
       return this.customEmojiBuffer.reduce((res, emoji) => {
         const pack = packOf(emoji)
         if (!res[pack]) {
@@ -200,6 +204,26 @@ const EmojiPicker = {
         return res
       }, {})
     },
+    allEmojiGroups () {
+      return this.allEmojis
+        .reduce((res, emoji) => {
+          const packName = packOf(emoji)
+          const packId = `custom-${packName}`
+          if (res.filter(k => k.id === packId).length === 0) {
+            res.push({
+              id: packId,
+              text: packName,
+              image: emoji.imageUrl
+            })
+          }
+          return res
+        }, [])
+        .concat({
+          id: 'standard',
+          text: this.$t('emoji.unicode'),
+          icon: 'box-open'
+        })
+    },
     emojis () {
       const standardEmojis = this.$store.state.instance.emoji || []
       // const customEmojis = this.customEmojiBuffer
diff --git a/src/components/emoji_picker/emoji_picker.scss b/src/components/emoji_picker/emoji_picker.scss
index 46cfa1c8..9d7a9bae 100644
--- a/src/components/emoji_picker/emoji_picker.scss
+++ b/src/components/emoji_picker/emoji_picker.scss
@@ -51,6 +51,7 @@
     display: flex;
     height: 32px;
     padding: 10px 7px 5px;
+    overflow-x: auto;
   }
 
   .content {
@@ -62,6 +63,9 @@
 
   .emoji-tabs {
     flex-grow: 1;
+    display: flex;
+    flex-direction: row;
+    flex-wrap: nowrap;
   }
 
   .emoji-groups {
@@ -69,6 +73,7 @@
   }
 
   .additional-tabs {
+    display: block;
     border-left: 1px solid;
     border-left-color: $fallback--icon;
     border-left-color: var(--icon, $fallback--icon);
@@ -78,7 +83,6 @@
 
   .additional-tabs,
   .emoji-tabs {
-    display: block;
     min-width: 0;
     flex-basis: auto;
     flex-shrink: 1;
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 65aa5bb6..5bcee2bc 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -3,12 +3,12 @@
     <div class="heading">
       <span class="emoji-tabs">
         <span
-          v-for="group in emojis"
+          v-for="group in allEmojiGroups"
           :key="group.id"
           class="emoji-tabs-item"
           :class="{
             active: activeGroupView === group.id,
-            disabled: group.emojis.length === 0
+            disabled: false
           }"
           :title="group.text"
           @click.prevent="highlight(group.id)"

From 34ae3d45dcbe76cf6833fb368e8f259e457ecaf4 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 14 Aug 2021 23:37:00 -0400
Subject: [PATCH 04/30] Load visible emoji groups when scrolling

---
 src/components/emoji_picker/emoji_picker.js  | 101 ++++++++++++++-----
 src/components/emoji_picker/emoji_picker.vue |   4 +-
 2 files changed, 75 insertions(+), 30 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 4b322f7d..4c53d38c 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -56,7 +56,8 @@ const EmojiPicker = {
       keepOpen: false,
       customEmojiBufferSlice: LOAD_EMOJI_BY,
       customEmojiTimeout: null,
-      customEmojiLoadAllConfirmed: false
+      customEmojiLoadAllConfirmed: false,
+      groupLoadedCount: {}
     }
   },
   components: {
@@ -78,7 +79,9 @@ const EmojiPicker = {
       const target = (e && e.target) || this.$refs['emoji-groups']
       this.updateScrolledClass(target)
       this.scrolledGroup(target)
-      this.triggerLoadMore(target)
+      this.$nextTick(() => {
+        this.triggerLoadMore(target)
+      })
     },
     highlight (key) {
       const ref = this.$refs['group-' + key]
@@ -87,6 +90,7 @@ const EmojiPicker = {
       this.activeGroup = key
       this.$nextTick(() => {
         this.$refs['emoji-groups'].scrollTop = top + 1
+        this.loadEmoji(key)
       })
     },
     updateScrolledClass (target) {
@@ -99,28 +103,40 @@ const EmojiPicker = {
       }
     },
     triggerLoadMore (target) {
-      const ref = this.$refs[`group-end-${this.lastNonUnicodeGroupId}`][0]
-      if (!ref) return
-      const bottom = ref.offsetTop + ref.offsetHeight
+      Object.keys(this.allCustomGroups)
+        .map(groupId => {
+          const ref = this.$refs[`group-end-${groupId}`][0]
+          if (!ref) return undefined
 
-      const scrollerBottom = target.scrollTop + target.clientHeight
-      const scrollerTop = target.scrollTop
-      const scrollerMax = target.scrollHeight
+          const bottom = ref.offsetTop + ref.offsetHeight
 
-      // Loads more emoji when they come into view
-      const approachingBottom = bottom - scrollerBottom < LOAD_EMOJI_MARGIN
-      // Always load when at the very top in case there's no scroll space yet
-      const atTop = scrollerTop < 5
-      // Don't load when looking at unicode category or at the very bottom
-      const bottomAboveViewport = bottom < scrollerTop || scrollerBottom === scrollerMax
-      if (!bottomAboveViewport && (approachingBottom || atTop)) {
-        this.loadEmoji()
-      }
+          const group = this.$refs[`group-${groupId}`][0]
+          const top = group.offsetTop
+
+          const scrollerBottom = target.scrollTop + target.clientHeight
+          const scrollerTop = target.scrollTop
+          const scrollerMax = target.scrollHeight
+
+          // Loads more emoji when they come into view
+          const approachingBottom = bottom - scrollerBottom < LOAD_EMOJI_MARGIN
+          // Always load when at the very top in case there's no scroll space yet
+          const atTop = scrollerTop < top + target.clientHeight / 2 && top < scrollerBottom
+          // Don't load when looking at unicode category or at the very bottom
+          const bottomAboveViewport = bottom < scrollerTop || scrollerBottom === scrollerMax
+          if (!bottomAboveViewport && (approachingBottom || atTop)) {
+            return groupId
+          }
+          return undefined
+        })
+        .filter(k => k)
+        .map(k => {
+          this.loadEmoji(k)
+        })
     },
     scrolledGroup (target) {
       const top = target.scrollTop + 5
       this.$nextTick(() => {
-        this.emojisView.forEach(group => {
+        this.allEmojiGroups.forEach(group => {
           const ref = this.$refs['group-' + group.id]
           if (ref.offsetTop <= top) {
             this.activeGroup = group.id
@@ -128,14 +144,21 @@ const EmojiPicker = {
         })
       })
     },
-    loadEmoji () {
-      const allLoaded = this.customEmojiBuffer.length === this.filteredEmoji.length
+    loadEmoji (loadGroup) {
+      if (!this.allCustomGroups[loadGroup]) {
+        return
+      }
+
+      const allLoaded = this.loadedCount[loadGroup] >= this.allCustomGroups[loadGroup].emojis.length
 
       if (allLoaded) {
         return
       }
 
-      this.customEmojiBufferSlice += LOAD_EMOJI_BY
+      this.groupLoadedCount = {
+        ...this.groupLoadedCount,
+        [loadGroup]: this.loadedCount[loadGroup] + LOAD_EMOJI_BY
+      }
     },
     startEmojiLoad (forceUpdate = false) {
       if (!forceUpdate) {
@@ -156,6 +179,9 @@ const EmojiPicker = {
     },
     setShowStickers (value) {
       this.showingStickers = value
+    },
+    limitedEmojis (list, groupId) {
+      return list.slice(0, this.loadedCount[groupId])
     }
   },
   watch: {
@@ -204,24 +230,36 @@ const EmojiPicker = {
         return res
       }, {})
     },
-    allEmojiGroups () {
-      return this.allEmojis
+    allCustomGroups () {
+      return this.filteredEmoji
         .reduce((res, emoji) => {
           const packName = packOf(emoji)
           const packId = `custom-${packName}`
-          if (res.filter(k => k.id === packId).length === 0) {
-            res.push({
+          if (!res[packId]) {
+            res[packId] = ({
               id: packId,
               text: packName,
-              image: emoji.imageUrl
+              image: emoji.imageUrl,
+              emojis: []
             })
           }
+          res[packId].emojis.push(emoji)
           return res
-        }, [])
+        }, {})
+    },
+    sensibleInitialAmountForAGroup () {
+      const groupCount = Object.keys(this.allCustomGroups).length
+      return Math.max(Math.floor(LOAD_EMOJI_BY / Math.max(groupCount, 1)), 1)
+    },
+    allEmojiGroups () {
+      const standardEmojis = this.$store.state.instance.emoji || []
+      return Object.entries(this.allCustomGroups)
+        .map(([_, v]) => v)
         .concat({
           id: 'standard',
           text: this.$t('emoji.unicode'),
-          icon: 'box-open'
+          icon: 'box-open',
+          emojis: filterByKeyword(standardEmojis, this.keyword)
         })
     },
     emojis () {
@@ -240,6 +278,13 @@ const EmojiPicker = {
         }
       ]
     },
+    loadedCount () {
+      return Object.keys(this.allCustomGroups)
+        .reduce((res, groupId) => {
+          res[groupId] = this.groupLoadedCount[groupId] || this.sensibleInitialAmountForAGroup
+          return res
+        }, {})
+    },
     lastNonUnicodeGroupId () {
       return this.emojis[this.emojis.length - 2].id
     },
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 5bcee2bc..1630269d 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -66,7 +66,7 @@
           @scroll="onScroll"
         >
           <div
-            v-for="group in emojisView"
+            v-for="group in allEmojiGroups"
             :key="group.id"
             class="emoji-group"
           >
@@ -77,7 +77,7 @@
               {{ group.text }}
             </h6>
             <span
-              v-for="emoji in group.emojis"
+              v-for="emoji in limitedEmojis(group.emojis, group.id)"
               :key="group.id + emoji.displayText"
               :title="emoji.displayText"
               class="emoji-item"

From 34bcfc5e0b32aeb129b4d2daa15bfd2202d3742f Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sun, 15 Aug 2021 00:03:31 -0400
Subject: [PATCH 05/30] Load emoji properly on first showing

---
 src/components/emoji_picker/emoji_picker.js | 10 +++++++++-
 1 file changed, 9 insertions(+), 1 deletion(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 4c53d38c..487b904c 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -57,7 +57,8 @@ const EmojiPicker = {
       customEmojiBufferSlice: LOAD_EMOJI_BY,
       customEmojiTimeout: null,
       customEmojiLoadAllConfirmed: false,
-      groupLoadedCount: {}
+      groupLoadedCount: {},
+      firstLoaded: false
     }
   },
   components: {
@@ -166,6 +167,13 @@ const EmojiPicker = {
       }
       this.$nextTick(() => {
         this.$refs['emoji-groups'].scrollTop = 0
+        this.$nextTick(() => {
+          if (this.firstLoaded) {
+            return
+          }
+          this.triggerLoadMore(this.$refs['emoji-groups'])
+          this.firstLoaded = true
+        })
       })
       const bufferSize = this.customEmojiBuffer.length
       const bufferPrefilledAll = bufferSize === this.filteredEmoji.length

From 5de792a78ec9e86ae79e3fb0be2078915f7a9282 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sun, 15 Aug 2021 00:43:35 -0400
Subject: [PATCH 06/30] Optimise emoji picker loading process

---
 src/components/emoji_picker/emoji_picker.js  | 83 ++++----------------
 src/components/emoji_picker/emoji_picker.vue |  4 +-
 src/modules/instance.js                      | 18 +++++
 3 files changed, 34 insertions(+), 71 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 487b904c..3d1ca135 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -105,6 +105,7 @@ const EmojiPicker = {
     },
     triggerLoadMore (target) {
       Object.keys(this.allCustomGroups)
+        .filter(id => this.filteredEmojiGroups.filter(group => group.id === id).length > 0)
         .map(groupId => {
           const ref = this.$refs[`group-end-${groupId}`][0]
           if (!ref) return undefined
@@ -122,9 +123,10 @@ const EmojiPicker = {
           const approachingBottom = bottom - scrollerBottom < LOAD_EMOJI_MARGIN
           // Always load when at the very top in case there's no scroll space yet
           const atTop = scrollerTop < top + target.clientHeight / 2 && top < scrollerBottom
+          const unscrollable = top - bottom < target.clientHeight
           // Don't load when looking at unicode category or at the very bottom
           const bottomAboveViewport = bottom < scrollerTop || scrollerBottom === scrollerMax
-          if (!bottomAboveViewport && (approachingBottom || atTop)) {
+          if (!bottomAboveViewport && (approachingBottom || atTop || unscrollable)) {
             return groupId
           }
           return undefined
@@ -175,12 +177,6 @@ const EmojiPicker = {
           this.firstLoaded = true
         })
       })
-      const bufferSize = this.customEmojiBuffer.length
-      const bufferPrefilledAll = bufferSize === this.filteredEmoji.length
-      if (bufferPrefilledAll && !forceUpdate) {
-        return
-      }
-      this.customEmojiBufferSlice = LOAD_EMOJI_BY
     },
     toggleStickers () {
       this.showingStickers = !this.showingStickers
@@ -190,6 +186,9 @@ const EmojiPicker = {
     },
     limitedEmojis (list, groupId) {
       return list.slice(0, this.loadedCount[groupId])
+    },
+    filterByKeyword (list, keyword) {
+      return filterByKeyword(list, keyword)
     }
   },
   watch: {
@@ -209,51 +208,8 @@ const EmojiPicker = {
       }
       return 0
     },
-    allEmojis () {
-      return this.$store.state.instance.customEmoji || []
-    },
-    filteredEmoji () {
-      return filterByKeyword(
-        this.allEmojis,
-        this.keyword
-      )
-    },
-    customEmojiBuffer () {
-      return this.filteredEmoji.slice(0, this.customEmojiBufferSlice)
-    },
-    groupedCustomEmojis () {
-      return this.customEmojiBuffer.reduce((res, emoji) => {
-        const pack = packOf(emoji)
-        if (!res[pack]) {
-          res[pack] = {
-            id: `custom-${pack}`,
-            text: pack,
-            /// FIXME
-            // icon: 'smile-beam',
-            image: emoji.imageUrl,
-            emojis: []
-          }
-        }
-        res[pack].emojis.push(emoji)
-        return res
-      }, {})
-    },
     allCustomGroups () {
-      return this.filteredEmoji
-        .reduce((res, emoji) => {
-          const packName = packOf(emoji)
-          const packId = `custom-${packName}`
-          if (!res[packId]) {
-            res[packId] = ({
-              id: packId,
-              text: packName,
-              image: emoji.imageUrl,
-              emojis: []
-            })
-          }
-          res[packId].emojis.push(emoji)
-          return res
-        }, {})
+      return this.$store.getters.groupedCustomEmojis
     },
     sensibleInitialAmountForAGroup () {
       const groupCount = Object.keys(this.allCustomGroups).length
@@ -270,21 +226,13 @@ const EmojiPicker = {
           emojis: filterByKeyword(standardEmojis, this.keyword)
         })
     },
-    emojis () {
-      const standardEmojis = this.$store.state.instance.emoji || []
-      // const customEmojis = this.customEmojiBuffer
-
-      return [
-        ...Object
-          .keys(this.groupedCustomEmojis)
-          .map(k => this.groupedCustomEmojis[k]),
-        {
-          id: 'standard',
-          text: this.$t('emoji.unicode'),
-          icon: 'box-open',
-          emojis: filterByKeyword(standardEmojis, this.keyword)
-        }
-      ]
+    filteredEmojiGroups () {
+      return this.allEmojiGroups
+        .map(group => ({
+          ...group,
+          emojis: filterByKeyword(group.emojis, this.keyword)
+        }))
+        .filter(group => group.emojis.length > 0)
     },
     loadedCount () {
       return Object.keys(this.allCustomGroups)
@@ -296,9 +244,6 @@ const EmojiPicker = {
     lastNonUnicodeGroupId () {
       return this.emojis[this.emojis.length - 2].id
     },
-    emojisView () {
-      return this.emojis.filter(value => value.emojis.length > 0)
-    },
     stickerPickerEnabled () {
       return (this.$store.state.instance.stickers || []).length !== 0
     }
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 1630269d..8bbde389 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -3,7 +3,7 @@
     <div class="heading">
       <span class="emoji-tabs">
         <span
-          v-for="group in allEmojiGroups"
+          v-for="group in filteredEmojiGroups"
           :key="group.id"
           class="emoji-tabs-item"
           :class="{
@@ -66,7 +66,7 @@
           @scroll="onScroll"
         >
           <div
-            v-for="group in allEmojiGroups"
+            v-for="group in filteredEmojiGroups"
             :key="group.id"
             class="emoji-group"
           >
diff --git a/src/modules/instance.js b/src/modules/instance.js
index c6d124b9..44925202 100644
--- a/src/modules/instance.js
+++ b/src/modules/instance.js
@@ -114,6 +114,24 @@ const instance = {
         .map(key => [key, state[key]])
         .reduce((acc, [key, value]) => ({ ...acc, [key]: value }), {})
     },
+    groupedCustomEmojis (state) {
+      return state.customEmoji
+        .reduce((res, emoji) => {
+          emoji.tags.forEach(packName => {
+            const packId = `custom-${packName}`
+            if (!res[packId]) {
+              res[packId] = ({
+                id: packId,
+                text: packName,
+                image: emoji.imageUrl,
+                emojis: []
+              })
+            }
+            res[packId].emojis.push(emoji)
+          })
+          return res
+        }, {})
+    },
     instanceDomain (state) {
       return new URL(state.server).hostname
     }

From 61975244bcae8637bd06bc0c3bc5ec34a7b6563d Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sun, 15 Aug 2021 00:53:57 -0400
Subject: [PATCH 07/30] Lint

---
 src/components/emoji_picker/emoji_picker.js | 2 --
 1 file changed, 2 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 3d1ca135..87e6495c 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -37,8 +37,6 @@ const filterByKeyword = (list, keyword = '') => {
   return orderedEmojiList.flat()
 }
 
-const packOf = emoji => (emoji.tags.filter(k => k.startsWith('pack:'))[0] || '').slice(5)
-
 const EmojiPicker = {
   props: {
     enableStickerPicker: {

From 1c8384203c078ae2581c09814a9e324b8f127357 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Thu, 7 Oct 2021 23:23:58 -0400
Subject: [PATCH 08/30] Use lozad for lazy image loading

Ref: grouped-emoji-picker
---
 package.json                                 |  1 +
 src/components/emoji_picker/emoji_picker.js  |  6 +++++-
 src/components/emoji_picker/emoji_picker.vue |  3 ++-
 src/directives/lazy_image_container.js       | 13 +++++++++++++
 yarn.lock                                    |  5 +++++
 5 files changed, 26 insertions(+), 2 deletions(-)
 create mode 100644 src/directives/lazy_image_container.js

diff --git a/package.json b/package.json
index e9a246b3..db702df6 100644
--- a/package.json
+++ b/package.json
@@ -32,6 +32,7 @@
     "diff": "3.5.0",
     "escape-html": "1.0.3",
     "localforage": "1.10.0",
+    "lozad": "^1.16.0",
     "parse-link-header": "1.0.1",
     "phoenix": "1.6.2",
     "punycode.js": "2.1.0",
diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 87e6495c..7b92db2e 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -1,5 +1,6 @@
 import { defineAsyncComponent } from 'vue'
 import Checkbox from '../checkbox/checkbox.vue'
+import LazyImageContainer from '../../directives/lazy_image_container'
 import { library } from '@fortawesome/fontawesome-svg-core'
 import {
   faBoxOpen,
@@ -63,6 +64,9 @@ const EmojiPicker = {
     StickerPicker: defineAsyncComponent(() => import('../sticker_picker/sticker_picker.vue')),
     Checkbox
   },
+  directives: {
+    LazyImageContainer
+  },
   methods: {
     onStickerUploaded (e) {
       this.$emit('sticker-uploaded', e)
@@ -183,7 +187,7 @@ const EmojiPicker = {
       this.showingStickers = value
     },
     limitedEmojis (list, groupId) {
-      return list.slice(0, this.loadedCount[groupId])
+      return list // list.slice(0, this.loadedCount[groupId])
     },
     filterByKeyword (list, keyword) {
       return filterByKeyword(list, keyword)
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 8bbde389..b2477339 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -61,6 +61,7 @@
         </div>
         <div
           ref="emoji-groups"
+          v-lazy-image-container
           class="emoji-groups"
           :class="groupsScrolledClass"
           @scroll="onScroll"
@@ -86,7 +87,7 @@
               <span v-if="!emoji.imageUrl">{{ emoji.replacement }}</span>
               <img
                 v-else
-                :src="emoji.imageUrl"
+                :data-src="emoji.imageUrl"
               >
             </span>
             <span :ref="'group-end-' + group.id" />
diff --git a/src/directives/lazy_image_container.js b/src/directives/lazy_image_container.js
new file mode 100644
index 00000000..44adc828
--- /dev/null
+++ b/src/directives/lazy_image_container.js
@@ -0,0 +1,13 @@
+
+import lozad from 'lozad'
+
+const LazyImageContainer = {
+  inserted (el) {
+    const images = el.querySelectorAll('img')
+    console.log(images.length)
+    el.$observer = lozad(images)
+    el.$observer.observe()
+  }
+}
+
+export default LazyImageContainer
diff --git a/yarn.lock b/yarn.lock
index 042d87ac..cf98fef3 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -6383,6 +6383,11 @@ lower-case@^1.1.1:
   version "1.1.4"
   resolved "https://registry.yarnpkg.com/lower-case/-/lower-case-1.1.4.tgz#9a2cabd1b9e8e0ae993a4bf7d5875c39c42e8eac"
 
+lozad@^1.16.0:
+  version "1.16.0"
+  resolved "https://registry.yarnpkg.com/lozad/-/lozad-1.16.0.tgz#86ce732c64c69926ccdebb81c8c90bb3735948b4"
+  integrity sha512-JBr9WjvEFeKoyim3svo/gsQPTkgG/mOHJmDctZ/+U9H3ymUuvEkqpn8bdQMFsvTMcyRJrdJkLv0bXqGm0sP72w==
+
 lru-cache@^4.0.1:
   version "4.1.5"
   resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-4.1.5.tgz#8bbe50ea85bed59bc9e33dcab8235ee9bcf443cd"

From 618282ef4878433a9a0741ccd48ed5c45ad49a0f Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 01:02:16 -0400
Subject: [PATCH 09/30] Clean up legacy code in emoji picker

Ref: grouped-emoji-picker
---
 src/components/emoji_input/emoji_input.js    |   1 -
 src/components/emoji_picker/emoji_picker.js  | 130 ++++---------------
 src/components/emoji_picker/emoji_picker.vue |   3 +-
 src/directives/lazy_image_container.js       |  13 --
 4 files changed, 28 insertions(+), 119 deletions(-)
 delete mode 100644 src/directives/lazy_image_container.js

diff --git a/src/components/emoji_input/emoji_input.js b/src/components/emoji_input/emoji_input.js
index 391cc5b5..be9a80f3 100644
--- a/src/components/emoji_input/emoji_input.js
+++ b/src/components/emoji_input/emoji_input.js
@@ -205,7 +205,6 @@ const EmojiInput = {
     },
     triggerShowPicker () {
       this.showPicker = true
-      this.$refs.picker.startEmojiLoad()
       this.$nextTick(() => {
         this.scrollIntoView()
         this.focusPickerInput()
diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 7b92db2e..5a7c3e67 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -1,6 +1,6 @@
 import { defineAsyncComponent } from 'vue'
 import Checkbox from '../checkbox/checkbox.vue'
-import LazyImageContainer from '../../directives/lazy_image_container'
+import lozad from 'lozad'
 import { library } from '@fortawesome/fontawesome-svg-core'
 import {
   faBoxOpen,
@@ -53,7 +53,6 @@ const EmojiPicker = {
       showingStickers: false,
       groupsScrolledClass: 'scrolled-top',
       keepOpen: false,
-      customEmojiBufferSlice: LOAD_EMOJI_BY,
       customEmojiTimeout: null,
       customEmojiLoadAllConfirmed: false,
       groupLoadedCount: {},
@@ -64,9 +63,6 @@ const EmojiPicker = {
     StickerPicker: defineAsyncComponent(() => import('../sticker_picker/sticker_picker.vue')),
     Checkbox
   },
-  directives: {
-    LazyImageContainer
-  },
   methods: {
     onStickerUploaded (e) {
       this.$emit('sticker-uploaded', e)
@@ -81,10 +77,6 @@ const EmojiPicker = {
     onScroll (e) {
       const target = (e && e.target) || this.$refs['emoji-groups']
       this.updateScrolledClass(target)
-      this.scrolledGroup(target)
-      this.$nextTick(() => {
-        this.triggerLoadMore(target)
-      })
     },
     highlight (key) {
       const ref = this.$refs['group-' + key]
@@ -93,7 +85,6 @@ const EmojiPicker = {
       this.activeGroup = key
       this.$nextTick(() => {
         this.$refs['emoji-groups'].scrollTop = top + 1
-        this.loadEmoji(key)
       })
     },
     updateScrolledClass (target) {
@@ -105,101 +96,48 @@ const EmojiPicker = {
         this.groupsScrolledClass = 'scrolled-middle'
       }
     },
-    triggerLoadMore (target) {
-      Object.keys(this.allCustomGroups)
-        .filter(id => this.filteredEmojiGroups.filter(group => group.id === id).length > 0)
-        .map(groupId => {
-          const ref = this.$refs[`group-end-${groupId}`][0]
-          if (!ref) return undefined
-
-          const bottom = ref.offsetTop + ref.offsetHeight
-
-          const group = this.$refs[`group-${groupId}`][0]
-          const top = group.offsetTop
-
-          const scrollerBottom = target.scrollTop + target.clientHeight
-          const scrollerTop = target.scrollTop
-          const scrollerMax = target.scrollHeight
-
-          // Loads more emoji when they come into view
-          const approachingBottom = bottom - scrollerBottom < LOAD_EMOJI_MARGIN
-          // Always load when at the very top in case there's no scroll space yet
-          const atTop = scrollerTop < top + target.clientHeight / 2 && top < scrollerBottom
-          const unscrollable = top - bottom < target.clientHeight
-          // Don't load when looking at unicode category or at the very bottom
-          const bottomAboveViewport = bottom < scrollerTop || scrollerBottom === scrollerMax
-          if (!bottomAboveViewport && (approachingBottom || atTop || unscrollable)) {
-            return groupId
-          }
-          return undefined
-        })
-        .filter(k => k)
-        .map(k => {
-          this.loadEmoji(k)
-        })
-    },
-    scrolledGroup (target) {
-      const top = target.scrollTop + 5
-      this.$nextTick(() => {
-        this.allEmojiGroups.forEach(group => {
-          const ref = this.$refs['group-' + group.id]
-          if (ref.offsetTop <= top) {
-            this.activeGroup = group.id
-          }
-        })
-      })
-    },
-    loadEmoji (loadGroup) {
-      if (!this.allCustomGroups[loadGroup]) {
-        return
-      }
-
-      const allLoaded = this.loadedCount[loadGroup] >= this.allCustomGroups[loadGroup].emojis.length
-
-      if (allLoaded) {
-        return
-      }
-
-      this.groupLoadedCount = {
-        ...this.groupLoadedCount,
-        [loadGroup]: this.loadedCount[loadGroup] + LOAD_EMOJI_BY
-      }
-    },
-    startEmojiLoad (forceUpdate = false) {
-      if (!forceUpdate) {
-        this.keyword = ''
-      }
-      this.$nextTick(() => {
-        this.$refs['emoji-groups'].scrollTop = 0
-        this.$nextTick(() => {
-          if (this.firstLoaded) {
-            return
-          }
-          this.triggerLoadMore(this.$refs['emoji-groups'])
-          this.firstLoaded = true
-        })
-      })
-    },
     toggleStickers () {
       this.showingStickers = !this.showingStickers
     },
     setShowStickers (value) {
       this.showingStickers = value
     },
-    limitedEmojis (list, groupId) {
-      return list // list.slice(0, this.loadedCount[groupId])
-    },
     filterByKeyword (list, keyword) {
       return filterByKeyword(list, keyword)
+    },
+    initializeLazyLoad () {
+      this.destroyLazyLoad()
+      this.$lozad = lozad('img', {})
+      this.$lozad.observe()
+    },
+    destroyLazyLoad () {
+      if (this.$lozad) {
+        if (this.$lozad.observer) {
+          this.$lozad.observer.disconnect()
+        }
+        if (this.$lozad.mutationObserver) {
+          this.$lozad.mutationObserver.disconnect()
+        }
+      }
     }
   },
   watch: {
     keyword () {
       this.customEmojiLoadAllConfirmed = false
       this.onScroll()
-      this.startEmojiLoad(true)
+      // Wait for the dom to change
+      this.$nextTick(() => this.initializeLazyLoad())
+    },
+    allCustomGroups () {
+      this.$nextTick(() => this.initializeLazyLoad())
     }
   },
+  mounted () {
+    this.initializeLazyLoad()
+  },
+  destroyed () {
+    this.destroyLazyLoad()
+  },
   computed: {
     activeGroupView () {
       return this.showingStickers ? '' : this.activeGroup
@@ -213,10 +151,6 @@ const EmojiPicker = {
     allCustomGroups () {
       return this.$store.getters.groupedCustomEmojis
     },
-    sensibleInitialAmountForAGroup () {
-      const groupCount = Object.keys(this.allCustomGroups).length
-      return Math.max(Math.floor(LOAD_EMOJI_BY / Math.max(groupCount, 1)), 1)
-    },
     allEmojiGroups () {
       const standardEmojis = this.$store.state.instance.emoji || []
       return Object.entries(this.allCustomGroups)
@@ -236,16 +170,6 @@ const EmojiPicker = {
         }))
         .filter(group => group.emojis.length > 0)
     },
-    loadedCount () {
-      return Object.keys(this.allCustomGroups)
-        .reduce((res, groupId) => {
-          res[groupId] = this.groupLoadedCount[groupId] || this.sensibleInitialAmountForAGroup
-          return res
-        }, {})
-    },
-    lastNonUnicodeGroupId () {
-      return this.emojis[this.emojis.length - 2].id
-    },
     stickerPickerEnabled () {
       return (this.$store.state.instance.stickers || []).length !== 0
     }
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index b2477339..ea6f1e4f 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -61,7 +61,6 @@
         </div>
         <div
           ref="emoji-groups"
-          v-lazy-image-container
           class="emoji-groups"
           :class="groupsScrolledClass"
           @scroll="onScroll"
@@ -78,7 +77,7 @@
               {{ group.text }}
             </h6>
             <span
-              v-for="emoji in limitedEmojis(group.emojis, group.id)"
+              v-for="emoji in group.emojis"
               :key="group.id + emoji.displayText"
               :title="emoji.displayText"
               class="emoji-item"
diff --git a/src/directives/lazy_image_container.js b/src/directives/lazy_image_container.js
deleted file mode 100644
index 44adc828..00000000
--- a/src/directives/lazy_image_container.js
+++ /dev/null
@@ -1,13 +0,0 @@
-
-import lozad from 'lozad'
-
-const LazyImageContainer = {
-  inserted (el) {
-    const images = el.querySelectorAll('img')
-    console.log(images.length)
-    el.$observer = lozad(images)
-    el.$observer.observe()
-  }
-}
-
-export default LazyImageContainer

From 38190e50e4124dac19643478d975bf9cbc68ddaa Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 01:11:32 -0400
Subject: [PATCH 10/30] Fix scrol->highlight behaviour

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.js | 18 ++++++++++++++++++
 1 file changed, 18 insertions(+)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 5a7c3e67..4d221aeb 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -77,6 +77,18 @@ const EmojiPicker = {
     onScroll (e) {
       const target = (e && e.target) || this.$refs['emoji-groups']
       this.updateScrolledClass(target)
+      this.scrolledGroup(target)
+    },
+    scrolledGroup (target) {
+      const top = target.scrollTop + 5
+      this.$nextTick(() => {
+        this.allEmojiGroups.forEach(group => {
+          const ref = this.$refs['group-' + group.id]
+          if (ref[0].offsetTop <= top) {
+            this.activeGroup = group.id
+          }
+        })
+      })
     },
     highlight (key) {
       const ref = this.$refs['group-' + key]
@@ -133,6 +145,9 @@ const EmojiPicker = {
     }
   },
   mounted () {
+    if (this.defaultGroup) {
+      this.highlight(this.defaultGroup)
+    }
     this.initializeLazyLoad()
   },
   destroyed () {
@@ -151,6 +166,9 @@ const EmojiPicker = {
     allCustomGroups () {
       return this.$store.getters.groupedCustomEmojis
     },
+    defaultGroup () {
+      return Object.keys(this.allCustomGroups)[0]
+    },
     allEmojiGroups () {
       const standardEmojis = this.$store.state.instance.emoji || []
       return Object.entries(this.allCustomGroups)

From 62e14d215d7f566b5bdfd6e85e9ca49eee069f37 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 01:20:35 -0400
Subject: [PATCH 11/30] Clean up unused variables

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.js | 13 +------------
 1 file changed, 1 insertion(+), 12 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 4d221aeb..bdc1ac12 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -14,13 +14,6 @@ library.add(
   faSmileBeam
 )
 
-// At widest, approximately 20 emoji are visible in a row,
-// loading 3 rows, could be overkill for narrow picker
-const LOAD_EMOJI_BY = 60
-
-// When to start loading new batch emoji, in pixels
-const LOAD_EMOJI_MARGIN = 64
-
 const filterByKeyword = (list, keyword = '') => {
   if (keyword === '') return list
 
@@ -53,10 +46,7 @@ const EmojiPicker = {
       showingStickers: false,
       groupsScrolledClass: 'scrolled-top',
       keepOpen: false,
-      customEmojiTimeout: null,
-      customEmojiLoadAllConfirmed: false,
-      groupLoadedCount: {},
-      firstLoaded: false
+      customEmojiTimeout: null
     }
   },
   components: {
@@ -135,7 +125,6 @@ const EmojiPicker = {
   },
   watch: {
     keyword () {
-      this.customEmojiLoadAllConfirmed = false
       this.onScroll()
       // Wait for the dom to change
       this.$nextTick(() => this.initializeLazyLoad())

From ce59d6962dd6333f269311c25073b587df728993 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 13:06:03 -0400
Subject: [PATCH 12/30] Remove useless class `disabled` in emoji picker

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.vue | 3 +--
 1 file changed, 1 insertion(+), 2 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index ea6f1e4f..70296fa3 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -7,8 +7,7 @@
           :key="group.id"
           class="emoji-tabs-item"
           :class="{
-            active: activeGroupView === group.id,
-            disabled: false
+            active: activeGroupView === group.id
           }"
           :title="group.text"
           @click.prevent="highlight(group.id)"

From f82519e15e2659ef4473436d09171e9b1f57241e Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 13:17:47 -0400
Subject: [PATCH 13/30] Use StillImage for emoji group header

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.js  | 4 +++-
 src/components/emoji_picker/emoji_picker.vue | 4 ++--
 2 files changed, 5 insertions(+), 3 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index bdc1ac12..b57c8afe 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -1,5 +1,6 @@
 import { defineAsyncComponent } from 'vue'
 import Checkbox from '../checkbox/checkbox.vue'
+import StillImage from '../still-image/still-image.vue'
 import lozad from 'lozad'
 import { library } from '@fortawesome/fontawesome-svg-core'
 import {
@@ -51,7 +52,8 @@ const EmojiPicker = {
   },
   components: {
     StickerPicker: defineAsyncComponent(() => import('../sticker_picker/sticker_picker.vue')),
-    Checkbox
+    Checkbox,
+    StillImage
   },
   methods: {
     onStickerUploaded (e) {
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 70296fa3..452a8de1 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -16,10 +16,10 @@
             v-if="group.image"
             class="emoji-picker-header-image"
           >
-            <img
+            <still-image
               :alt="group.text"
               :src="group.image"
-            >
+            />
           </span>
           <FAIcon
             v-else

From 3058f47394485ca1aef3cdc872627916e0cbc2c8 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 14:10:17 -0400
Subject: [PATCH 14/30] Fix vertical scrollbar of emoji picker header

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.scss | 29 ++++++++++++++-----
 1 file changed, 22 insertions(+), 7 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.scss b/src/components/emoji_picker/emoji_picker.scss
index 9d7a9bae..a988f9eb 100644
--- a/src/components/emoji_picker/emoji_picker.scss
+++ b/src/components/emoji_picker/emoji_picker.scss
@@ -1,5 +1,10 @@
 @import '../../_variables.scss';
 
+$emoji-picker-header-height: 36px;
+$emoji-picker-header-picture-width: 32px;
+$emoji-picker-header-picture-height: 32px;
+$emoji-picker-emoji-size: 32px;
+
 .emoji-picker {
   display: flex;
   flex-direction: column;
@@ -22,9 +27,11 @@
     display: inline-flex;
     justify-content: center;
     align-items: center;
-    width: 30px;
-    height: 24px;
-    img {
+    width: $emoji-picker-header-picture-width;
+    max-width: $emoji-picker-header-picture-width;
+    height: $emoji-picker-header-picture-height;
+    max-height: $emoji-picker-header-picture-height;
+    .still-image {
       max-width: 100%;
       max-height: 100%;
       object-fit: contain;
@@ -49,7 +56,7 @@
 
   .heading {
     display: flex;
-    height: 32px;
+    //height: $emoji-picker-header-height;
     padding: 10px 7px 5px;
     overflow-x: auto;
   }
@@ -86,11 +93,19 @@
     min-width: 0;
     flex-basis: auto;
     flex-shrink: 1;
+    display: flex;
+    align-content: center;
 
     &-item {
       padding: 0 7px;
       cursor: pointer;
       font-size: 24px;
+      width: $emoji-picker-header-picture-width;
+      max-width: $emoji-picker-header-picture-width;
+      height: $emoji-picker-header-picture-height;
+      max-height: $emoji-picker-header-picture-height;
+      display: flex;
+      align-items: center;
 
       &.disabled {
         opacity: 0.5;
@@ -178,11 +193,11 @@
     }
 
     &-item {
-      width: 32px;
-      height: 32px;
+      width: $emoji-picker-emoji-size;
+      height: $emoji-picker-emoji-size;
       box-sizing: border-box;
       display: flex;
-      font-size: 32px;
+      font-size: $emoji-picker-emoji-size;
       align-items: center;
       justify-content: center;
       margin: 4px;

From 68614af0ce9285c1f95824bb397da4361abea971 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 14:46:00 -0400
Subject: [PATCH 15/30] Fix sticker picker heading tab

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.scss | 9 ++++-----
 1 file changed, 4 insertions(+), 5 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.scss b/src/components/emoji_picker/emoji_picker.scss
index a988f9eb..b4606396 100644
--- a/src/components/emoji_picker/emoji_picker.scss
+++ b/src/components/emoji_picker/emoji_picker.scss
@@ -56,9 +56,7 @@ $emoji-picker-emoji-size: 32px;
 
   .heading {
     display: flex;
-    //height: $emoji-picker-header-height;
     padding: 10px 7px 5px;
-    overflow-x: auto;
   }
 
   .content {
@@ -73,6 +71,7 @@ $emoji-picker-emoji-size: 32px;
     display: flex;
     flex-direction: row;
     flex-wrap: nowrap;
+    overflow-x: auto;
   }
 
   .emoji-groups {
@@ -80,7 +79,8 @@ $emoji-picker-emoji-size: 32px;
   }
 
   .additional-tabs {
-    display: block;
+    display: flex;
+    flex: 1;
     border-left: 1px solid;
     border-left-color: $fallback--icon;
     border-left-color: var(--icon, $fallback--icon);
@@ -90,9 +90,8 @@ $emoji-picker-emoji-size: 32px;
 
   .additional-tabs,
   .emoji-tabs {
-    min-width: 0;
     flex-basis: auto;
-    flex-shrink: 1;
+    // flex-shrink: 1;
     display: flex;
     align-content: center;
 

From 0b74749d8e09b33e209072af49b3d5a2ef72bb9d Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 15:09:24 -0400
Subject: [PATCH 16/30] Lazy-load emoji picker in post form

When clicking the reply button, we used to load the whole emoji picker.
This causes a considerable delay even if the user is not going to use
the emoji picker. Now the content of the emoji picker is loaded only
after the user has explicitly opened the emoji picker.

Ref: grouped-emoji-picker
---
 src/components/emoji_input/emoji_input.vue   |  1 +
 src/components/emoji_picker/emoji_picker.js  | 24 ++++++++++++++++----
 src/components/emoji_picker/emoji_picker.vue |  9 ++++++--
 3 files changed, 27 insertions(+), 7 deletions(-)

diff --git a/src/components/emoji_input/emoji_input.vue b/src/components/emoji_input/emoji_input.vue
index aa2950ce..3d9a629b 100644
--- a/src/components/emoji_input/emoji_input.vue
+++ b/src/components/emoji_input/emoji_input.vue
@@ -19,6 +19,7 @@
         v-if="enableEmojiPicker"
         ref="picker"
         :class="{ hide: !showPicker }"
+        :showing="showPicker"
         :enable-sticker-picker="enableStickerPicker"
         class="emoji-picker-panel"
         @emoji="insert"
diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index b57c8afe..578f7053 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -38,6 +38,10 @@ const EmojiPicker = {
       required: false,
       type: Boolean,
       default: false
+    },
+    showing: {
+      required: true,
+      type: Boolean
     }
   },
   data () {
@@ -47,7 +51,9 @@ const EmojiPicker = {
       showingStickers: false,
       groupsScrolledClass: 'scrolled-top',
       keepOpen: false,
-      customEmojiTimeout: null
+      customEmojiTimeout: null,
+      // Lazy-load only after the first time `showing` becomes true.
+      contentLoaded: false
     }
   },
   components: {
@@ -114,6 +120,9 @@ const EmojiPicker = {
       this.$lozad = lozad('img', {})
       this.$lozad.observe()
     },
+    waitForDomAndInitializeLazyLoad() {
+      this.$nextTick(() => this.initializeLazyLoad())
+    },
     destroyLazyLoad () {
       if (this.$lozad) {
         if (this.$lozad.observer) {
@@ -128,18 +137,23 @@ const EmojiPicker = {
   watch: {
     keyword () {
       this.onScroll()
-      // Wait for the dom to change
-      this.$nextTick(() => this.initializeLazyLoad())
+      this.waitForDomAndInitializeLazyLoad()
     },
     allCustomGroups () {
-      this.$nextTick(() => this.initializeLazyLoad())
+      this.waitForDomAndInitializeLazyLoad()
+    },
+    showing (val) {
+      if (val) {
+        this.contentLoaded = true
+        this.waitForDomAndInitializeLazyLoad()
+      }
     }
   },
   mounted () {
     if (this.defaultGroup) {
       this.highlight(this.defaultGroup)
     }
-    this.initializeLazyLoad()
+    this.waitForDomAndInitializeLazyLoad()
   },
   destroyed () {
     this.destroyLazyLoad()
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 452a8de1..db4347e8 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -1,5 +1,7 @@
 <template>
-  <div class="emoji-picker panel panel-default panel-body">
+  <div
+    class="emoji-picker panel panel-default panel-body"
+  >
     <div class="heading">
       <span class="emoji-tabs">
         <span
@@ -45,7 +47,10 @@
         </span>
       </span>
     </div>
-    <div class="content">
+    <div
+      v-if="contentLoaded"
+      class="content"
+    >
       <div
         class="emoji-content"
         :class="{hidden: showingStickers}"

From 55ecabf3a3f5b93a375f0026198519ee2423f3ab Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 15:25:13 -0400
Subject: [PATCH 17/30] Lint

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.js | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 578f7053..29feec5b 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -120,7 +120,7 @@ const EmojiPicker = {
       this.$lozad = lozad('img', {})
       this.$lozad.observe()
     },
-    waitForDomAndInitializeLazyLoad() {
+    waitForDomAndInitializeLazyLoad () {
       this.$nextTick(() => this.initializeLazyLoad())
     },
     destroyLazyLoad () {

From 2186bd83a6d2c03aa321c3954aa8fb1a9574ffea Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 15:30:55 -0400
Subject: [PATCH 18/30] Group emojis only by pack and remove pack: prefix

Ref: grouped-emoji-picker
---
 src/modules/instance.js | 8 +++++++-
 1 file changed, 7 insertions(+), 1 deletion(-)

diff --git a/src/modules/instance.js b/src/modules/instance.js
index 44925202..15825783 100644
--- a/src/modules/instance.js
+++ b/src/modules/instance.js
@@ -115,9 +115,15 @@ const instance = {
         .reduce((acc, [key, value]) => ({ ...acc, [key]: value }), {})
     },
     groupedCustomEmojis (state) {
+      const packsOf = emoji => {
+        return emoji.tags
+          .filter(k => k.startsWith('pack:'))
+          .map(k => k.slice(5)) // remove 'pack:' prefix
+      }
+
       return state.customEmoji
         .reduce((res, emoji) => {
-          emoji.tags.forEach(packName => {
+          packsOf(emoji).forEach(packName => {
             const packId = `custom-${packName}`
             if (!res[packId]) {
               res[packId] = ({

From 2c0ca850d879c37cf364fe90f62fc160185ea1dd Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 8 Oct 2021 15:47:39 -0400
Subject: [PATCH 19/30] Fix error on emoji picker first load

Ref: grouped-emoji-picker
---
 src/components/emoji_picker/emoji_picker.js | 20 +++++++++++++++-----
 1 file changed, 15 insertions(+), 5 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 29feec5b..471626d0 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -132,6 +132,18 @@ const EmojiPicker = {
           this.$lozad.mutationObserver.disconnect()
         }
       }
+    },
+    onShowing () {
+      const oldContentLoaded = this.contentLoaded
+      this.contentLoaded = true
+      this.waitForDomAndInitializeLazyLoad()
+      if (!oldContentLoaded) {
+        this.$nextTick(() => {
+          if (this.defaultGroup) {
+            this.highlight(this.defaultGroup)
+          }
+        })
+      }
     }
   },
   watch: {
@@ -144,16 +156,14 @@ const EmojiPicker = {
     },
     showing (val) {
       if (val) {
-        this.contentLoaded = true
-        this.waitForDomAndInitializeLazyLoad()
+        this.onShowing()
       }
     }
   },
   mounted () {
-    if (this.defaultGroup) {
-      this.highlight(this.defaultGroup)
+    if (this.showing) {
+      this.onShowing()
     }
-    this.waitForDomAndInitializeLazyLoad()
   },
   destroyed () {
     this.destroyLazyLoad()

From 08ad3ba54bf2bdd7904fe0fc821a8cca67a67ab5 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 01:35:16 -0500
Subject: [PATCH 20/30] Use StillImage to render emojis in emoji picker

---
 src/components/emoji_picker/emoji_picker.js  | 15 +++++++++++++--
 src/components/emoji_picker/emoji_picker.vue |  5 +++--
 src/components/still-image/still-image.js    | 19 +++++++++++++++++--
 src/components/still-image/still-image.vue   |  5 +++--
 4 files changed, 36 insertions(+), 8 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 471626d0..c0e90434 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -117,8 +117,19 @@ const EmojiPicker = {
     },
     initializeLazyLoad () {
       this.destroyLazyLoad()
-      this.$lozad = lozad('img', {})
-      this.$lozad.observe()
+      this.$nextTick(() => {
+        this.$lozad = lozad('.still-image.emoji-picker-emoji', {
+          load: el => {
+            const vn = el.__vue__
+            if (!vn) {
+              return
+            }
+
+            vn.loadLazy()
+          }
+        })
+        this.$lozad.observe()
+      })
     },
     waitForDomAndInitializeLazyLoad () {
       this.$nextTick(() => this.initializeLazyLoad())
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index db4347e8..b2dae51e 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -88,10 +88,11 @@
               @click.stop.prevent="onEmoji(emoji)"
             >
               <span v-if="!emoji.imageUrl">{{ emoji.replacement }}</span>
-              <img
+              <still-image
                 v-else
+                class="emoji-picker-emoji"
                 :data-src="emoji.imageUrl"
-              >
+              />
             </span>
             <span :ref="'group-end-' + group.id" />
           </div>
diff --git a/src/components/still-image/still-image.js b/src/components/still-image/still-image.js
index d7abbcb5..1806d33b 100644
--- a/src/components/still-image/still-image.js
+++ b/src/components/still-image/still-image.js
@@ -7,16 +7,23 @@ const StillImage = {
     'imageLoadHandler',
     'alt',
     'height',
-    'width'
+    'width',
+    'dataSrc'
   ],
   data () {
     return {
+      // for lazy loading, see loadLazy()
+      realSrc: this.src,
       stopGifs: this.$store.getters.mergedConfig.stopGifs
     }
   },
   computed: {
     animated () {
-      return this.stopGifs && (this.mimetype === 'image/gif' || this.src.endsWith('.gif'))
+      if (!this.realSrc) {
+        return false
+      }
+
+      return this.stopGifs && (this.mimetype === 'image/gif' || this.realSrc.endsWith('.gif'))
     },
     style () {
       const appendPx = (str) => /\d$/.test(str) ? str + 'px' : str
@@ -27,7 +34,15 @@ const StillImage = {
     }
   },
   methods: {
+    loadLazy () {
+      if (this.dataSrc) {
+        this.realSrc = this.dataSrc
+      }
+    },
     onLoad () {
+      if (!this.realSrc) {
+        return
+      }
       const image = this.$refs.src
       if (!image) return
       this.imageLoadHandler && this.imageLoadHandler(image)
diff --git a/src/components/still-image/still-image.vue b/src/components/still-image/still-image.vue
index e939b532..eb553c14 100644
--- a/src/components/still-image/still-image.vue
+++ b/src/components/still-image/still-image.vue
@@ -11,10 +11,11 @@
     <!-- NOTE: key is required to force to re-render img tag when src is changed -->
     <img
       ref="src"
-      :key="src"
+      :key="realSrc"
       :alt="alt"
       :title="alt"
-      :src="src"
+      :data-src="dataSrc"
+      :src="realSrc"
       :referrerpolicy="referrerpolicy"
       @load="onLoad"
       @error="onError"

From 4f989cec26cd440ad1998c473f6807456a7d43b1 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 01:37:19 -0500
Subject: [PATCH 21/30] Clean up emoji picker css

---
 src/components/emoji_picker/emoji_picker.scss | 1 -
 1 file changed, 1 deletion(-)

diff --git a/src/components/emoji_picker/emoji_picker.scss b/src/components/emoji_picker/emoji_picker.scss
index b4606396..a7c89303 100644
--- a/src/components/emoji_picker/emoji_picker.scss
+++ b/src/components/emoji_picker/emoji_picker.scss
@@ -91,7 +91,6 @@ $emoji-picker-emoji-size: 32px;
   .additional-tabs,
   .emoji-tabs {
     flex-basis: auto;
-    // flex-shrink: 1;
     display: flex;
     align-content: center;
 

From 2c8c900e09245f8282a748cee7d28191e8dff422 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 02:17:59 -0500
Subject: [PATCH 22/30] Scroll active tab header into view in emoji picker

---
 src/components/emoji_picker/emoji_picker.js  | 18 ++++++++++++++++++
 src/components/emoji_picker/emoji_picker.vue |  6 +++++-
 2 files changed, 23 insertions(+), 1 deletion(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index c0e90434..612e0c19 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -86,8 +86,26 @@ const EmojiPicker = {
             this.activeGroup = group.id
           }
         })
+        this.scrollHeader()
       })
     },
+    scrollHeader () {
+      // Scroll the active tab's header into view
+      const headerRef = this.$refs['group-header-' + this.activeGroup][0]
+      const left = headerRef.offsetLeft
+      const right = left + headerRef.offsetWidth
+      const headerCont = this.$refs.header
+      const currentScroll = headerCont.scrollLeft
+      const currentScrollRight = currentScroll + headerCont.clientWidth
+      const setScroll = s => { headerCont.scrollLeft = s }
+
+      const margin = 7 // .emoji-tabs-item: padding
+      if (left - margin < currentScroll) {
+        setScroll(left - margin)
+      } else if (right + margin > currentScrollRight) {
+        setScroll(right + margin - headerCont.clientWidth)
+      }
+    },
     highlight (key) {
       const ref = this.$refs['group-' + key]
       const top = ref.offsetTop
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index b2dae51e..b202df91 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -3,9 +3,13 @@
     class="emoji-picker panel panel-default panel-body"
   >
     <div class="heading">
-      <span class="emoji-tabs">
+      <span
+        ref="header"
+        class="emoji-tabs"
+      >
         <span
           v-for="group in filteredEmojiGroups"
+          :ref="'group-header-' + group.id"
           :key="group.id"
           class="emoji-tabs-item"
           :class="{

From b72afe428b7b6c4495ce5cde8419cd0beb2a2770 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 16:13:02 -0500
Subject: [PATCH 23/30] Generate grouped unicode emojis from unicode-emoji-json

---
 .gitignore            |    1 +
 build/build.js        |    3 +
 build/dev-server.js   |    3 +
 build/update-emoji.js |   27 +
 package.json          |    1 +
 static/emoji.json     | 1431 -----------------------------------------
 yarn.lock             |    5 +
 7 files changed, 40 insertions(+), 1431 deletions(-)
 create mode 100644 build/update-emoji.js
 delete mode 100644 static/emoji.json

diff --git a/.gitignore b/.gitignore
index 479d57c4..4df5ec83 100644
--- a/.gitignore
+++ b/.gitignore
@@ -7,3 +7,4 @@ test/e2e/reports
 selenium-debug.log
 .idea/
 config/local.json
+static/emoji.json
diff --git a/build/build.js b/build/build.js
index b3c9aad4..35969eb6 100644
--- a/build/build.js
+++ b/build/build.js
@@ -18,6 +18,9 @@ console.log(
 var spinner = ora('building for production...')
 spinner.start()
 
+var updateEmoji = require('./update-emoji').updateEmoji
+updateEmoji()
+
 var assetsPath = path.join(config.build.assetsRoot, config.build.assetsSubDirectory)
 rm('-rf', assetsPath)
 mkdir('-p', assetsPath)
diff --git a/build/dev-server.js b/build/dev-server.js
index c06192bd..008e9e54 100644
--- a/build/dev-server.js
+++ b/build/dev-server.js
@@ -10,6 +10,9 @@ var webpackConfig = process.env.NODE_ENV === 'testing'
   ? require('./webpack.prod.conf')
   : require('./webpack.dev.conf')
 
+var updateEmoji = require('./update-emoji').updateEmoji
+updateEmoji()
+
 // default port where dev server listens for incoming traffic
 var port = process.env.PORT || config.dev.port
 // Define HTTP proxies to your custom API backend
diff --git a/build/update-emoji.js b/build/update-emoji.js
new file mode 100644
index 00000000..5336a95b
--- /dev/null
+++ b/build/update-emoji.js
@@ -0,0 +1,27 @@
+
+module.exports = {
+  updateEmoji () {
+    const emojis = require('unicode-emoji-json/data-by-group')
+    const fs = require('fs')
+
+    Object.keys(emojis)
+      .map(k => {
+        emojis[k].map(e => {
+          delete e.unicode_version
+          delete e.emoji_version
+          delete e.skin_tone_support_unicode_version
+        })
+      })
+
+    const res = {}
+    Object.keys(emojis)
+      .map(k => {
+        const groupId = k.replace('&', 'and').replace(/ /g, '-').toLowerCase()
+        res[groupId] = emojis[k]
+      })
+
+    console.log('Updating emojis...')
+    fs.writeFileSync('static/emoji.json', JSON.stringify(res))
+    console.log('Done.')
+  }
+}
diff --git a/package.json b/package.json
index db702df6..c5d8b678 100644
--- a/package.json
+++ b/package.json
@@ -115,6 +115,7 @@
     "stylelint": "13.6.1",
     "stylelint-config-standard": "20.0.0",
     "stylelint-rscss": "0.4.0",
+    "unicode-emoji-json": "^0.3.0",
     "url-loader": "1.1.2",
     "vue-loader": "^16.0.0",
     "vue-style-loader": "4.1.2",
diff --git a/static/emoji.json b/static/emoji.json
deleted file mode 100644
index 12b91b3f..00000000
--- a/static/emoji.json
+++ /dev/null
@@ -1,1431 +0,0 @@
-{
-  "100": "๐Ÿ’ฏ",
-  "1234": "๐Ÿ”ข",
-  "1st_place_medal": "๐Ÿฅ‡",
-  "2nd_place_medal": "๐Ÿฅˆ",
-  "3rd_place_medal": "๐Ÿฅ‰",
-  "8ball": "๐ŸŽฑ",
-  "a_button_blood_type": "๐Ÿ…ฐ",
-  "ab": "๐Ÿ†Ž",
-  "abacus": "๐Ÿงฎ",
-  "abc": "๐Ÿ”ค",
-  "abcd": "๐Ÿ”ก",
-  "accept": "๐Ÿ‰‘",
-  "adhesive_bandage": "๐Ÿฉน",
-  "admission_tickets": "๐ŸŽŸ",
-  "adult": "๐Ÿง‘",
-  "aerial_tramway": "๐Ÿšก",
-  "airplane": "โœˆ",
-  "airplane_arriving": "๐Ÿ›ฌ",
-  "airplane_departure": "๐Ÿ›ซ",
-  "alarm_clock": "โฐ",
-  "alembic": "โš—๏ธ",
-  "alien": "๐Ÿ‘ฝ",
-  "ambulance": "๐Ÿš‘",
-  "amphora": "๐Ÿบ",
-  "anchor": "โš“",
-  "angel": "๐Ÿ‘ผ",
-  "anger": "๐Ÿ’ข",
-  "anger_right": "๐Ÿ—ฏ",
-  "angry": "๐Ÿ˜ ",
-  "anguished": "๐Ÿ˜ง",
-  "ant": "๐Ÿœ",
-  "apple": "๐ŸŽ",
-  "aquarius": "โ™’",
-  "aries": "โ™ˆ",
-  "arrow_backward": "โ—€๏ธ",
-  "arrow_double_down": "โฌ",
-  "arrow_double_up": "โซ",
-  "arrow_down": "โฌ‡๏ธ",
-  "arrow_down_small": "๐Ÿ”ฝ",
-  "arrow_forward": "โ–ถ๏ธ",
-  "arrow_heading_down": "โคต๏ธ",
-  "arrow_heading_up": "โคด๏ธ",
-  "arrow_left": "โฌ…๏ธ",
-  "arrow_lower_left": "โ†™๏ธ",
-  "arrow_lower_right": "โ†˜๏ธ",
-  "arrow_right": "โžก",
-  "arrow_right_hook": "โ†ช๏ธ",
-  "arrow_up": "โฌ†๏ธ",
-  "arrow_up_down": "โ†•",
-  "arrow_up_small": "๐Ÿ”ผ",
-  "arrow_upper_left": "โ†–",
-  "arrow_upper_right": "โ†—๏ธ",
-  "arrows_clockwise": "๐Ÿ”ƒ",
-  "arrows_counterclockwise": "๐Ÿ”„",
-  "art": "๐ŸŽจ",
-  "articulated_lorry": "๐Ÿš›",
-  "artist_palette": "๐ŸŽจ",
-  "asterisk": "*โƒฃ",
-  "astonished": "๐Ÿ˜ฒ",
-  "athletic_shoe": "๐Ÿ‘Ÿ",
-  "atm": "๐Ÿง",
-  "atom": "โš›",
-  "atom_symbol": "โš›๏ธ",
-  "auto_rickshaw": "๐Ÿ›บ",
-  "automobile": "๐Ÿš—",
-  "avocado": "๐Ÿฅ‘",
-  "axe": "๐Ÿช“",
-  "b_button_blood_type": "๐Ÿ…ฑ",
-  "baby": "๐Ÿ‘ถ",
-  "baby_bottle": "๐Ÿผ",
-  "baby_chick": "๐Ÿค",
-  "baby_symbol": "๐Ÿšผ",
-  "back": "๐Ÿ”™",
-  "bacon": "๐Ÿฅ“",
-  "badger": "๐Ÿฆก",
-  "badminton": "๐Ÿธ",
-  "bagel": "๐Ÿฅฏ",
-  "baggage_claim": "๐Ÿ›„",
-  "baguette_bread": "๐Ÿฅ–",
-  "balance_scale": "โš–๏ธ",
-  "bald": "๐Ÿฆฒ",
-  "ballet_shoes": "๐Ÿฉฐ",
-  "balloon": "๐ŸŽˆ",
-  "ballot_box": "๐Ÿ—ณ",
-  "ballot_box_with_check": "โ˜‘๏ธ",
-  "bamboo": "๐ŸŽ",
-  "banana": "๐ŸŒ",
-  "bangbang": "โ€ผ๏ธ",
-  "banjo": "๐Ÿช•",
-  "bank": "๐Ÿฆ",
-  "bar_chart": "๐Ÿ“Š",
-  "barber": "๐Ÿ’ˆ",
-  "baseball": "โšพ",
-  "basket": "๐Ÿงบ",
-  "basketball": "๐Ÿ€",
-  "basketballer": "โ›น",
-  "bat": "๐Ÿฆ‡",
-  "bath": "๐Ÿ›€",
-  "bathtub": "๐Ÿ›",
-  "battery": "๐Ÿ”‹",
-  "beach_umbrella": "โ›ฑ",
-  "beach_with_umbrella": "๐Ÿ–",
-  "bear": "๐Ÿป",
-  "beard": "๐Ÿง”",
-  "bearded_person": "๐Ÿง”",
-  "bed": "๐Ÿ›",
-  "bee": "๐Ÿ",
-  "beer": "๐Ÿบ",
-  "beers": "๐Ÿป",
-  "beetle": "๐Ÿž",
-  "beginner": "๐Ÿ”ฐ",
-  "bell": "๐Ÿ””",
-  "bellhop_bell": "๐Ÿ›Ž",
-  "bento": "๐Ÿฑ",
-  "beverage_box": "๐Ÿงƒ",
-  "bicyclist": "๐Ÿšด",
-  "bike": "๐Ÿšฒ",
-  "bikini": "๐Ÿ‘™",
-  "billed_cap": "๐Ÿงข",
-  "biohazard": "โ˜ฃ๏ธ",
-  "bird": "๐Ÿฆ",
-  "birthday": "๐ŸŽ‚",
-  "black_circle": "โšซ",
-  "black_heart": "๐Ÿ–ค",
-  "black_joker": "๐Ÿƒ",
-  "black_large_square": "โฌ›",
-  "black_medium_small_square": "โ—พ",
-  "black_medium_square": "โ—ผ",
-  "black_nib": "โœ’๏ธ",
-  "black_small_square": "โ–ช",
-  "black_square_button": "๐Ÿ”ฒ",
-  "blond_haired_person": "๐Ÿ‘ฑ",
-  "blossom": "๐ŸŒผ",
-  "blowfish": "๐Ÿก",
-  "blue_book": "๐Ÿ“˜",
-  "blue_car": "๐Ÿš™",
-  "blue_circle": "๐Ÿ”ต",
-  "blue_heart": "๐Ÿ’™",
-  "blue_square": "๐ŸŸฆ",
-  "blush": "๐Ÿ˜Š",
-  "boar": "๐Ÿ—",
-  "bomb": "๐Ÿ’ฃ",
-  "bone": "๐Ÿฆด",
-  "book": "๐Ÿ“–",
-  "bookmark": "๐Ÿ”–",
-  "bookmark_tabs": "๐Ÿ“‘",
-  "books": "๐Ÿ“š",
-  "boom": "๐Ÿ’ฅ",
-  "boot": "๐Ÿ‘ข",
-  "bouquet": "๐Ÿ’",
-  "bow": "๐Ÿ™‡",
-  "bow_and_arrow": "๐Ÿน",
-  "bowl_with_spoon": "๐Ÿฅฃ",
-  "bowling": "๐ŸŽณ",
-  "boxing_glove": "๐ŸฅŠ",
-  "boy": "๐Ÿ‘ฆ",
-  "brain": "๐Ÿง ",
-  "bread": "๐Ÿž",
-  "breast_feeding": "๐Ÿคฑ",
-  "breastfeeding": "๐Ÿคฑ",
-  "brick": "๐Ÿงฑ",
-  "bride_with_veil": "๐Ÿ‘ฐ",
-  "bridge_at_night": "๐ŸŒ‰",
-  "briefcase": "๐Ÿ’ผ",
-  "briefs": "๐Ÿฉฒ",
-  "broccoli": "๐Ÿฅฆ",
-  "broken_heart": "๐Ÿ’”",
-  "broom": "๐Ÿงน",
-  "brown_circle": "๐ŸŸค",
-  "brown_heart": "๐ŸคŽ",
-  "bug": "๐Ÿ›",
-  "building_construction": "๐Ÿ—",
-  "bulb": "๐Ÿ’ก",
-  "bullettrain_front": "๐Ÿš…",
-  "bullettrain_side": "๐Ÿš„",
-  "burrito": "๐ŸŒฏ",
-  "bus": "๐ŸšŒ",
-  "busstop": "๐Ÿš",
-  "bust_in_silhouette": "๐Ÿ‘ค",
-  "busts_in_silhouette": "๐Ÿ‘ฅ",
-  "butter": "๐Ÿงˆ",
-  "butterfly": "๐Ÿฆ‹",
-  "cactus": "๐ŸŒต",
-  "cake": "๐Ÿฐ",
-  "calendar": "๐Ÿ“†",
-  "call_me": "๐Ÿค™",
-  "call_me_hand": "๐Ÿค™",
-  "calling": "๐Ÿ“ฒ",
-  "camel": "๐Ÿซ",
-  "camera": "๐Ÿ“ท",
-  "camera_with_flash": "๐Ÿ“ธ",
-  "camping": "๐Ÿ•",
-  "cancer": "โ™‹",
-  "candle": "๐Ÿ•ฏ",
-  "candy": "๐Ÿฌ",
-  "canned_food": "๐Ÿฅซ",
-  "canoe": "๐Ÿ›ถ",
-  "capital_abcd": "๐Ÿ” ",
-  "capricorn": "โ™‘",
-  "card_file_box": "๐Ÿ—ƒ",
-  "card_index": "๐Ÿ“‡",
-  "card_index_dividers": "๐Ÿ—‚",
-  "carousel_horse": "๐ŸŽ ",
-  "carrot": "๐Ÿฅ•",
-  "cat": "๐Ÿฑ",
-  "cat2": "๐Ÿˆ",
-  "cd": "๐Ÿ’ฟ",
-  "chains": "โ›“๏ธ",
-  "chair": "๐Ÿช‘",
-  "champagne": "๐Ÿพ",
-  "champagne_glass": "๐Ÿฅ‚",
-  "chart": "๐Ÿ’น",
-  "chart_with_downwards_trend": "๐Ÿ“‰",
-  "chart_with_upwards_trend": "๐Ÿ“ˆ",
-  "check_box_with_check": "โ˜‘",
-  "check_mark": "โœ”",
-  "checkered_flag": "๐Ÿ",
-  "cheese": "๐Ÿง€",
-  "cheese_wedge": "๐Ÿง€",
-  "cherries": "๐Ÿ’",
-  "cherry_blossom": "๐ŸŒธ",
-  "chess_pawn": "โ™Ÿ",
-  "chestnut": "๐ŸŒฐ",
-  "chicken": "๐Ÿ”",
-  "child": "๐Ÿง’",
-  "children_crossing": "๐Ÿšธ",
-  "chipmunk": "๐Ÿฟ",
-  "chocolate_bar": "๐Ÿซ",
-  "chopsticks": "๐Ÿฅข",
-  "christmas_tree": "๐ŸŽ„",
-  "church": "โ›ช",
-  "cinema": "๐ŸŽฆ",
-  "circled_m": "โ“‚",
-  "circus_tent": "๐ŸŽช",
-  "city_dusk": "๐ŸŒ†",
-  "city_sunset": "๐ŸŒ‡",
-  "cityscape": "๐Ÿ™",
-  "cityscape_at_dusk": "๐ŸŒ†",
-  "cl": "๐Ÿ†‘",
-  "clap": "๐Ÿ‘",
-  "clapper": "๐ŸŽฌ",
-  "classical_building": "๐Ÿ›",
-  "clinking_glasses": "๐Ÿฅ‚",
-  "clipboard": "๐Ÿ“‹",
-  "clock1": "๐Ÿ•",
-  "clock10": "๐Ÿ•™",
-  "clock1030": "๐Ÿ•ฅ",
-  "clock11": "๐Ÿ•š",
-  "clock1130": "๐Ÿ•ฆ",
-  "clock12": "๐Ÿ•›",
-  "clock1230": "๐Ÿ•ง",
-  "clock130": "๐Ÿ•œ",
-  "clock2": "๐Ÿ•‘",
-  "clock230": "๐Ÿ•",
-  "clock3": "๐Ÿ•’",
-  "clock330": "๐Ÿ•ž",
-  "clock4": "๐Ÿ•“",
-  "clock430": "๐Ÿ•Ÿ",
-  "clock5": "๐Ÿ•”",
-  "clock530": "๐Ÿ• ",
-  "clock6": "๐Ÿ••",
-  "clock630": "๐Ÿ•ก",
-  "clock7": "๐Ÿ•–",
-  "clock730": "๐Ÿ•ข",
-  "clock8": "๐Ÿ•—",
-  "clock830": "๐Ÿ•ฃ",
-  "clock9": "๐Ÿ•˜",
-  "clock930": "๐Ÿ•ค",
-  "closed_book": "๐Ÿ“•",
-  "closed_lock_with_key": "๐Ÿ”",
-  "closed_umbrella": "๐ŸŒ‚",
-  "cloud": "โ˜๏ธ",
-  "cloud_with_lightning": "๐ŸŒฉ",
-  "cloud_with_lightning_and_rain": "โ›ˆ๏ธ",
-  "cloud_with_rain": "๐ŸŒง",
-  "cloud_with_snow": "๐ŸŒจ",
-  "clown": "๐Ÿคก",
-  "clown_face": "๐Ÿคก",
-  "club_suit": "โ™ฃ๏ธ",
-  "clubs": "โ™ฃ",
-  "coat": "๐Ÿงฅ",
-  "cocktail": "๐Ÿธ",
-  "coconut": "๐Ÿฅฅ",
-  "coffee": "โ˜•",
-  "coffin": "โšฐ๏ธ",
-  "cold_face": "๐Ÿฅถ",
-  "cold_sweat": "๐Ÿ˜ฐ",
-  "comet": "โ˜„๏ธ",
-  "compass": "๐Ÿงญ",
-  "compression": "๐Ÿ—œ",
-  "computer": "๐Ÿ’ป",
-  "computer_mouse": "๐Ÿ–ฑ",
-  "confetti_ball": "๐ŸŽŠ",
-  "confounded": "๐Ÿ˜–",
-  "confused": "๐Ÿ˜•",
-  "congratulations": "ใŠ—",
-  "construction": "๐Ÿšง",
-  "construction_worker": "๐Ÿ‘ท",
-  "control_knobs": "๐ŸŽ›",
-  "convenience_store": "๐Ÿช",
-  "cookie": "๐Ÿช",
-  "cooking": "๐Ÿณ",
-  "cool": "๐Ÿ†’",
-  "cop": "๐Ÿ‘ฎ",
-  "copyright": "ยฉ",
-  "corn": "๐ŸŒฝ",
-  "couch_and_lamp": "๐Ÿ›‹",
-  "couple": "๐Ÿ‘ซ",
-  "couple_with_heart": "๐Ÿ’‘",
-  "couplekiss": "๐Ÿ’",
-  "cow": "๐Ÿฎ",
-  "cow2": "๐Ÿ„",
-  "cowboy": "๐Ÿค ",
-  "cowboy_hat_face": "๐Ÿค ",
-  "crab": "๐Ÿฆ€",
-  "crayon": "๐Ÿ–",
-  "crazy_face": "๐Ÿคช",
-  "credit_card": "๐Ÿ’ณ",
-  "crescent_moon": "๐ŸŒ™",
-  "cricket": "๐Ÿฆ—",
-  "cricket_game": "๐Ÿ",
-  "crocodile": "๐ŸŠ",
-  "croissant": "๐Ÿฅ",
-  "cross": "โœ๏ธ",
-  "crossed_fingers": "๐Ÿคž",
-  "crossed_flags": "๐ŸŽŒ",
-  "crossed_swords": "โš”๏ธ",
-  "crown": "๐Ÿ‘‘",
-  "cry": "๐Ÿ˜ข",
-  "crying_cat_face": "๐Ÿ˜ฟ",
-  "crystal_ball": "๐Ÿ”ฎ",
-  "cucumber": "๐Ÿฅ’",
-  "cup_with_straw": "๐Ÿฅค",
-  "cupcake": "๐Ÿง",
-  "cupid": "๐Ÿ’˜",
-  "curling_stone": "๐ŸฅŒ",
-  "curly_hair": "๐Ÿฆฑ",
-  "curly_loop": "โžฐ",
-  "currency_exchange": "๐Ÿ’ฑ",
-  "curry": "๐Ÿ›",
-  "custard": "๐Ÿฎ",
-  "customs": "๐Ÿ›ƒ",
-  "cut_of_meat": "๐Ÿฅฉ",
-  "cyclone": "๐ŸŒ€",
-  "dagger": "๐Ÿ—ก",
-  "dancer": "๐Ÿ’ƒ",
-  "dancers": "๐Ÿ‘ฏ",
-  "dango": "๐Ÿก",
-  "dark_skin_tone": "๐Ÿฟ",
-  "dark_sunglasses": "๐Ÿ•ถ",
-  "dart": "๐ŸŽฏ",
-  "dash": "๐Ÿ’จ",
-  "date": "๐Ÿ“…",
-  "deaf_person": "๐Ÿง",
-  "deciduous_tree": "๐ŸŒณ",
-  "deer": "๐ŸฆŒ",
-  "department_store": "๐Ÿฌ",
-  "derelict_house": "๐Ÿš",
-  "desert": "๐Ÿœ",
-  "desert_island": "๐Ÿ",
-  "desktop_computer": "๐Ÿ–ฅ",
-  "detective": "๐Ÿ•ต",
-  "diamond_shape_with_a_dot_inside": "๐Ÿ’ ",
-  "diamond_suit": "โ™ฆ๏ธ",
-  "diamonds": "โ™ฆ",
-  "disappointed": "๐Ÿ˜ž",
-  "disappointed_relieved": "๐Ÿ˜ฅ",
-  "diving_mask": "๐Ÿคฟ",
-  "diya_lamp": "๐Ÿช”",
-  "dizzy": "๐Ÿ’ซ",
-  "dizzy_face": "๐Ÿ˜ต",
-  "dna": "๐Ÿงฌ",
-  "do_not_litter": "๐Ÿšฏ",
-  "dog": "๐Ÿถ",
-  "dog2": "๐Ÿ•",
-  "dollar": "๐Ÿ’ต",
-  "dolls": "๐ŸŽŽ",
-  "dolphin": "๐Ÿฌ",
-  "door": "๐Ÿšช",
-  "double_exclamation_mark": "โ€ผ",
-  "doughnut": "๐Ÿฉ",
-  "dove": "๐Ÿ•Š",
-  "down_arrow": "โฌ‡",
-  "downleft_arrow": "โ†™",
-  "downright_arrow": "โ†˜",
-  "dragon": "๐Ÿ‰",
-  "dragon_face": "๐Ÿฒ",
-  "dress": "๐Ÿ‘—",
-  "dromedary_camel": "๐Ÿช",
-  "drooling_face": "๐Ÿคค",
-  "drop_of_blood": "๐Ÿฉธ",
-  "droplet": "๐Ÿ’ง",
-  "drum": "๐Ÿฅ",
-  "duck": "๐Ÿฆ†",
-  "dumpling": "๐ŸฅŸ",
-  "dvd": "๐Ÿ“€",
-  "e-mail": "๐Ÿ“ง",
-  "eagle": "๐Ÿฆ…",
-  "ear": "๐Ÿ‘‚",
-  "ear_of_rice": "๐ŸŒพ",
-  "ear_with_hearing_aid": "๐Ÿฆป",
-  "earth_africa": "๐ŸŒ",
-  "earth_americas": "๐ŸŒŽ",
-  "earth_asia": "๐ŸŒ",
-  "egg": "๐Ÿฅš",
-  "eggplant": "๐Ÿ†",
-  "eight": "8โƒฃ",
-  "eight_pointed_black_star": "โœด๏ธ",
-  "eight_spoked_asterisk": "โœณ๏ธ",
-  "eightpointed_star": "โœด",
-  "eightspoked_asterisk": "โœณ",
-  "eject_button": "โ",
-  "electric_plug": "๐Ÿ”Œ",
-  "elephant": "๐Ÿ˜",
-  "elf": "๐Ÿง",
-  "end": "๐Ÿ”š",
-  "envelope": "โœ‰",
-  "envelope_with_arrow": "๐Ÿ“ฉ",
-  "euro": "๐Ÿ’ถ",
-  "european_castle": "๐Ÿฐ",
-  "european_post_office": "๐Ÿค",
-  "evergreen_tree": "๐ŸŒฒ",
-  "exclamation": "โ—",
-  "exclamation_question_mark": "โ‰",
-  "exploding_head": "๐Ÿคฏ",
-  "expressionless": "๐Ÿ˜‘",
-  "eye": "๐Ÿ‘",
-  "eyeglasses": "๐Ÿ‘“",
-  "eyes": "๐Ÿ‘€",
-  "face_vomiting": "๐Ÿคฎ",
-  "face_with_hand_over_mouth": "๐Ÿคญ",
-  "face_with_headbandage": "๐Ÿค•",
-  "face_with_monocle": "๐Ÿง",
-  "face_with_raised_eyebrow": "๐Ÿคจ",
-  "face_with_symbols_on_mouth": "๐Ÿคฌ",
-  "face_with_symbols_over_mouth": "๐Ÿคฌ",
-  "face_with_thermometer": "๐Ÿค’",
-  "factory": "๐Ÿญ",
-  "fairy": "๐Ÿงš",
-  "falafel": "๐Ÿง†",
-  "fallen_leaf": "๐Ÿ‚",
-  "family": "๐Ÿ‘ช",
-  "fast_forward": "โฉ",
-  "fax": "๐Ÿ“ ",
-  "fearful": "๐Ÿ˜จ",
-  "feet": "๐Ÿพ",
-  "female_sign": "โ™€",
-  "ferris_wheel": "๐ŸŽก",
-  "ferry": "โ›ด๏ธ",
-  "field_hockey": "๐Ÿ‘",
-  "file_cabinet": "๐Ÿ—„",
-  "file_folder": "๐Ÿ“",
-  "film_frames": "๐ŸŽž",
-  "film_projector": "๐Ÿ“ฝ",
-  "fingers_crossed": "๐Ÿคž",
-  "fire": "๐Ÿ”ฅ",
-  "fire_engine": "๐Ÿš’",
-  "fire_extinguisher": "๐Ÿงฏ",
-  "firecracker": "๐Ÿงจ",
-  "fireworks": "๐ŸŽ†",
-  "first_place": "๐Ÿฅ‡",
-  "first_quarter_moon": "๐ŸŒ“",
-  "first_quarter_moon_with_face": "๐ŸŒ›",
-  "fish": "๐ŸŸ",
-  "fish_cake": "๐Ÿฅ",
-  "fishing_pole_and_fish": "๐ŸŽฃ",
-  "fist": "โœŠ",
-  "five": "5โƒฃ",
-  "flag_black": "๐Ÿด",
-  "flag_white": "๐Ÿณ",
-  "flags": "๐ŸŽ",
-  "flamingo": "๐Ÿฆฉ",
-  "flashlight": "๐Ÿ”ฆ",
-  "flat_shoe": "๐Ÿฅฟ",
-  "fleur-de-lis": "โšœ",
-  "fleurde-lis": "โšœ๏ธ",
-  "floppy_disk": "๐Ÿ’พ",
-  "flower_playing_cards": "๐ŸŽด",
-  "flushed": "๐Ÿ˜ณ",
-  "flying_disc": "๐Ÿฅ",
-  "flying_saucer": "๐Ÿ›ธ",
-  "fog": "๐ŸŒซ",
-  "foggy": "๐ŸŒ",
-  "foot": "๐Ÿฆถ",
-  "football": "๐Ÿˆ",
-  "footprints": "๐Ÿ‘ฃ",
-  "fork_and_knife": "๐Ÿด",
-  "fork_and_knife_with_plate": "๐Ÿฝ",
-  "fortune_cookie": "๐Ÿฅ ",
-  "fountain": "โ›ฒ",
-  "fountain_pen": "๐Ÿ–‹",
-  "four": "4โƒฃ",
-  "four_leaf_clover": "๐Ÿ€",
-  "fox": "๐ŸฆŠ",
-  "framed_picture": "๐Ÿ–ผ",
-  "free": "๐Ÿ†“",
-  "french_bread": "๐Ÿฅ–",
-  "fried_shrimp": "๐Ÿค",
-  "fries": "๐ŸŸ",
-  "frog": "๐Ÿธ",
-  "frowning": "๐Ÿ˜ฆ",
-  "frowning_face": "โ˜น๏ธ",
-  "fuelpump": "โ›ฝ",
-  "full_moon": "๐ŸŒ•",
-  "full_moon_with_face": "๐ŸŒ",
-  "funeral_urn": "โšฑ๏ธ",
-  "game_die": "๐ŸŽฒ",
-  "garlic": "๐Ÿง„",
-  "gear": "โš™๏ธ",
-  "gem": "๐Ÿ’Ž",
-  "gemini": "โ™Š",
-  "genie": "๐Ÿงž",
-  "ghost": "๐Ÿ‘ป",
-  "gift": "๐ŸŽ",
-  "gift_heart": "๐Ÿ’",
-  "giraffe": "๐Ÿฆ’",
-  "girl": "๐Ÿ‘ง",
-  "glass_of_milk": "๐Ÿฅ›",
-  "globe_with_meridians": "๐ŸŒ",
-  "gloves": "๐Ÿงค",
-  "goal": "๐Ÿฅ…",
-  "goal_net": "๐Ÿฅ…",
-  "goat": "๐Ÿ",
-  "goggles": "๐Ÿฅฝ",
-  "golf": "โ›ณ",
-  "golfer": "๐ŸŒ",
-  "gorilla": "๐Ÿฆ",
-  "grapes": "๐Ÿ‡",
-  "green_apple": "๐Ÿ",
-  "green_book": "๐Ÿ“—",
-  "green_circle": "๐ŸŸข",
-  "green_heart": "๐Ÿ’š",
-  "green_salad": "๐Ÿฅ—",
-  "green_square": "๐ŸŸฉ",
-  "grey_exclamation": "โ•",
-  "grey_question": "โ”",
-  "grimacing": "๐Ÿ˜ฌ",
-  "grin": "๐Ÿ˜",
-  "grinning": "๐Ÿ˜€",
-  "guard": "๐Ÿ’‚",
-  "guardsman": "๐Ÿ’‚",
-  "guide_dog": "๐Ÿฆฎ",
-  "guitar": "๐ŸŽธ",
-  "gun": "๐Ÿ”ซ",
-  "haircut": "๐Ÿ’‡",
-  "hamburger": "๐Ÿ”",
-  "hammer": "๐Ÿ”จ",
-  "hammer_and_pick": "โš’๏ธ",
-  "hammer_and_wrench": "๐Ÿ› ",
-  "hamster": "๐Ÿน",
-  "hand_with_fingers_splayed": "๐Ÿ–",
-  "handbag": "๐Ÿ‘œ",
-  "handshake": "๐Ÿค",
-  "hash": "#โƒฃ",
-  "hatched_chick": "๐Ÿฅ",
-  "hatching_chick": "๐Ÿฃ",
-  "head_bandage": "๐Ÿค•",
-  "headphones": "๐ŸŽง",
-  "hear_no_evil": "๐Ÿ™‰",
-  "heart": "โค๏ธ",
-  "heart_decoration": "๐Ÿ’Ÿ",
-  "heart_exclamation": "โฃ",
-  "heart_eyes": "๐Ÿ˜",
-  "heart_eyes_cat": "๐Ÿ˜ป",
-  "heart_suit": "โ™ฅ๏ธ",
-  "heartbeat": "๐Ÿ’“",
-  "heartpulse": "๐Ÿ’—",
-  "hearts": "โ™ฅ",
-  "heavy_check_mark": "โœ”๏ธ",
-  "heavy_division_sign": "โž—",
-  "heavy_dollar_sign": "๐Ÿ’ฒ",
-  "heavy_minus_sign": "โž–",
-  "heavy_multiplication_x": "โœ–๏ธ",
-  "heavy_plus_sign": "โž•",
-  "hedgehog": "๐Ÿฆ”",
-  "helicopter": "๐Ÿš",
-  "herb": "๐ŸŒฟ",
-  "hibiscus": "๐ŸŒบ",
-  "high_brightness": "๐Ÿ”†",
-  "high_heel": "๐Ÿ‘ ",
-  "hiking_boot": "๐Ÿฅพ",
-  "hindu_temple": "๐Ÿ›•",
-  "hippopotamus": "๐Ÿฆ›",
-  "hockey": "๐Ÿ’",
-  "hole": "๐Ÿ•ณ",
-  "honey_pot": "๐Ÿฏ",
-  "horse": "๐Ÿด",
-  "horse_racing": "๐Ÿ‡",
-  "hospital": "๐Ÿฅ",
-  "hot_face": "๐Ÿฅต",
-  "hot_pepper": "๐ŸŒถ",
-  "hot_springs": "โ™จ",
-  "hotdog": "๐ŸŒญ",
-  "hotel": "๐Ÿจ",
-  "hotsprings": "โ™จ๏ธ",
-  "hourglass": "โŒ›",
-  "hourglass_flowing_sand": "โณ",
-  "house": "๐Ÿ ",
-  "house_with_garden": "๐Ÿก",
-  "houses": "๐Ÿ˜",
-  "hugging": "๐Ÿค—",
-  "hundred_points": "๐Ÿ’ฏ",
-  "hushed": "๐Ÿ˜ฏ",
-  "ice": "๐ŸงŠ",
-  "ice_cream": "๐Ÿจ",
-  "ice_hockey": "๐Ÿ’",
-  "ice_skate": "โ›ธ๏ธ",
-  "icecream": "๐Ÿฆ",
-  "id": "๐Ÿ†”",
-  "ideograph_advantage": "๐Ÿ‰",
-  "imp": "๐Ÿ‘ฟ",
-  "inbox_tray": "๐Ÿ“ฅ",
-  "incoming_envelope": "๐Ÿ“จ",
-  "index_pointing_up": "โ˜",
-  "infinity": "โ™พ",
-  "information": "โ„น๏ธ",
-  "information_desk_person": "๐Ÿ’",
-  "information_source": "โ„น",
-  "innocent": "๐Ÿ˜‡",
-  "input_numbers": "๐Ÿ”ข",
-  "interrobang": "โ‰๏ธ",
-  "iphone": "๐Ÿ“ฑ",
-  "izakaya_lantern": "๐Ÿฎ",
-  "jack_o_lantern": "๐ŸŽƒ",
-  "japan": "๐Ÿ—พ",
-  "japanese_castle": "๐Ÿฏ",
-  "japanese_congratulations_button": "ใŠ—๏ธ",
-  "japanese_free_of_charge_button": "๐Ÿˆš",
-  "japanese_goblin": "๐Ÿ‘บ",
-  "japanese_ogre": "๐Ÿ‘น",
-  "japanese_reserved_button": "๐Ÿˆฏ",
-  "japanese_secret_button": "ใŠ™๏ธ",
-  "japanese_service_charge_button": "๐Ÿˆ‚",
-  "jeans": "๐Ÿ‘–",
-  "joy": "๐Ÿ˜‚",
-  "joy_cat": "๐Ÿ˜น",
-  "joystick": "๐Ÿ•น",
-  "kaaba": "๐Ÿ•‹",
-  "kangaroo": "๐Ÿฆ˜",
-  "key": "๐Ÿ”‘",
-  "keyboard": "โŒจ๏ธ",
-  "keycap_ten": "๐Ÿ”Ÿ",
-  "kick_scooter": "๐Ÿ›ด",
-  "kimono": "๐Ÿ‘˜",
-  "kiss": "๐Ÿ’‹",
-  "kissing": "๐Ÿ˜—",
-  "kissing_cat": "๐Ÿ˜ฝ",
-  "kissing_closed_eyes": "๐Ÿ˜š",
-  "kissing_heart": "๐Ÿ˜˜",
-  "kissing_smiling_eyes": "๐Ÿ˜™",
-  "kitchen_knife": "๐Ÿ”ช",
-  "kite": "๐Ÿช",
-  "kiwi": "๐Ÿฅ",
-  "kiwi_fruit": "๐Ÿฅ",
-  "knife": "๐Ÿ”ช",
-  "koala": "๐Ÿจ",
-  "koko": "๐Ÿˆ",
-  "lab_coat": "๐Ÿฅผ",
-  "label": "๐Ÿท",
-  "lacrosse": "๐Ÿฅ",
-  "large_blue_diamond": "๐Ÿ”ท",
-  "large_orange_diamond": "๐Ÿ”ถ",
-  "last_quarter_moon": "๐ŸŒ—",
-  "last_quarter_moon_with_face": "๐ŸŒœ",
-  "last_track_button": "โฎ๏ธ",
-  "latin_cross": "โœ",
-  "laughing": "๐Ÿ˜†",
-  "leafy_green": "๐Ÿฅฌ",
-  "leaves": "๐Ÿƒ",
-  "ledger": "๐Ÿ“’",
-  "left_arrow": "โฌ…",
-  "left_arrow_curving_right": "โ†ช",
-  "left_facing_fist": "๐Ÿค›",
-  "left_luggage": "๐Ÿ›…",
-  "left_right_arrow": "โ†”",
-  "leftfacing_fist": "๐Ÿค›",
-  "leftright_arrow": "โ†”๏ธ",
-  "leftwards_arrow_with_hook": "โ†ฉ๏ธ",
-  "leg": "๐Ÿฆต",
-  "lemon": "๐Ÿ‹",
-  "leo": "โ™Œ",
-  "leopard": "๐Ÿ†",
-  "level_slider": "๐ŸŽš",
-  "libra": "โ™Ž",
-  "light_rail": "๐Ÿšˆ",
-  "light_skin_tone": "๐Ÿป",
-  "link": "๐Ÿ”—",
-  "linked_paperclips": "๐Ÿ–‡",
-  "lion_face": "๐Ÿฆ",
-  "lips": "๐Ÿ‘„",
-  "lipstick": "๐Ÿ’„",
-  "lizard": "๐ŸฆŽ",
-  "llama": "๐Ÿฆ™",
-  "lobster": "๐Ÿฆž",
-  "lock": "๐Ÿ”’",
-  "lock_with_ink_pen": "๐Ÿ”",
-  "lollipop": "๐Ÿญ",
-  "loop": "โžฟ",
-  "lotion_bottle": "๐Ÿงด",
-  "loud_sound": "๐Ÿ”Š",
-  "loudspeaker": "๐Ÿ“ข",
-  "love_hotel": "๐Ÿฉ",
-  "love_letter": "๐Ÿ’Œ",
-  "love_you_gesture": "๐ŸคŸ",
-  "loveyou_gesture": "๐ŸคŸ",
-  "low_brightness": "๐Ÿ”…",
-  "luggage": "๐Ÿงณ",
-  "lying_face": "๐Ÿคฅ",
-  "m": "โ“‚๏ธ",
-  "mag": "๐Ÿ”",
-  "mag_right": "๐Ÿ”Ž",
-  "mage": "๐Ÿง™",
-  "magnet": "๐Ÿงฒ",
-  "mahjong": "๐Ÿ€„",
-  "mailbox": "๐Ÿ“ซ",
-  "mailbox_closed": "๐Ÿ“ช",
-  "mailbox_with_mail": "๐Ÿ“ฌ",
-  "mailbox_with_no_mail": "๐Ÿ“ญ",
-  "male_sign": "โ™‚",
-  "man": "๐Ÿ‘จ",
-  "man_dancing": "๐Ÿ•บ",
-  "man_in_suit": "๐Ÿ•ด",
-  "man_in_tuxedo": "๐Ÿคต",
-  "man_with_chinese_cap": "๐Ÿ‘ฒ",
-  "man_with_gua_pi_mao": "๐Ÿ‘ฒ",
-  "man_with_turban": "๐Ÿ‘ณ",
-  "mango": "๐Ÿฅญ",
-  "mans_shoe": "๐Ÿ‘ž",
-  "mantelpiece_clock": "๐Ÿ•ฐ",
-  "manual_wheelchair": "๐Ÿฆฝ",
-  "maple_leaf": "๐Ÿ",
-  "martial_arts_uniform": "๐Ÿฅ‹",
-  "mask": "๐Ÿ˜ท",
-  "massage": "๐Ÿ’†",
-  "mate": "๐Ÿง‰",
-  "meat_on_bone": "๐Ÿ–",
-  "mechanical_arm": "๐Ÿฆพ",
-  "mechanical_leg": "๐Ÿฆฟ",
-  "medal": "๐Ÿ…",
-  "medical_symbol": "โš•",
-  "medium_skin_tone": "๐Ÿฝ",
-  "mediumdark_skin_tone": "๐Ÿพ",
-  "mediumlight_skin_tone": "๐Ÿผ",
-  "mega": "๐Ÿ“ฃ",
-  "melon": "๐Ÿˆ",
-  "memo": "๐Ÿ“",
-  "menorah": "๐Ÿ•Ž",
-  "mens": "๐Ÿšน",
-  "merperson": "๐Ÿงœ",
-  "metal": "๐Ÿค˜",
-  "metro": "๐Ÿš‡",
-  "microbe": "๐Ÿฆ ",
-  "microphone": "๐ŸŽค",
-  "microscope": "๐Ÿ”ฌ",
-  "middle_finger": "๐Ÿ–•",
-  "military_medal": "๐ŸŽ–",
-  "milk": "๐Ÿฅ›",
-  "milky_way": "๐ŸŒŒ",
-  "minibus": "๐Ÿš",
-  "minidisc": "๐Ÿ’ฝ",
-  "mobile_phone_off": "๐Ÿ“ด",
-  "money_mouth": "๐Ÿค‘",
-  "money_with_wings": "๐Ÿ’ธ",
-  "moneybag": "๐Ÿ’ฐ",
-  "moneymouth_face": "๐Ÿค‘",
-  "monkey": "๐Ÿ’",
-  "monkey_face": "๐Ÿต",
-  "monorail": "๐Ÿš",
-  "moon_cake": "๐Ÿฅฎ",
-  "mortar_board": "๐ŸŽ“",
-  "mosque": "๐Ÿ•Œ",
-  "mosquito": "๐ŸฆŸ",
-  "motor_boat": "๐Ÿ›ฅ",
-  "motor_scooter": "๐Ÿ›ต",
-  "motorcycle": "๐Ÿ",
-  "motorized_wheelchair": "๐Ÿฆผ",
-  "motorway": "๐Ÿ›ฃ",
-  "mount_fuji": "๐Ÿ—ป",
-  "mountain": "โ›ฐ๏ธ",
-  "mountain_bicyclist": "๐Ÿšต",
-  "mountain_cableway": "๐Ÿš ",
-  "mountain_railway": "๐Ÿšž",
-  "mouse": "๐Ÿญ",
-  "mouse2": "๐Ÿ",
-  "movie_camera": "๐ŸŽฅ",
-  "moyai": "๐Ÿ—ฟ",
-  "mrs_claus": "๐Ÿคถ",
-  "multiplication_sign": "โœ–",
-  "muscle": "๐Ÿ’ช",
-  "mushroom": "๐Ÿ„",
-  "musical_keyboard": "๐ŸŽน",
-  "musical_note": "๐ŸŽต",
-  "musical_score": "๐ŸŽผ",
-  "mute": "๐Ÿ”‡",
-  "nail_care": "๐Ÿ’…",
-  "name_badge": "๐Ÿ“›",
-  "national_park": "๐Ÿž",
-  "nauseated_face": "๐Ÿคข",
-  "nazar_amulet": "๐Ÿงฟ",
-  "necktie": "๐Ÿ‘”",
-  "negative_squared_cross_mark": "โŽ",
-  "nerd": "๐Ÿค“",
-  "neutral_face": "๐Ÿ˜",
-  "new": "๐Ÿ†•",
-  "new_moon": "๐ŸŒ‘",
-  "new_moon_with_face": "๐ŸŒš",
-  "newspaper": "๐Ÿ“ฐ",
-  "next_track_button": "โญ๏ธ",
-  "ng": "๐Ÿ†–",
-  "night_with_stars": "๐ŸŒƒ",
-  "nine": "9โƒฃ",
-  "no_bell": "๐Ÿ”•",
-  "no_bicycles": "๐Ÿšณ",
-  "no_entry": "โ›”",
-  "no_entry_sign": "๐Ÿšซ",
-  "no_good": "๐Ÿ™…",
-  "no_mobile_phones": "๐Ÿ“ต",
-  "no_mouth": "๐Ÿ˜ถ",
-  "no_pedestrians": "๐Ÿšท",
-  "no_smoking": "๐Ÿšญ",
-  "non-potable_water": "๐Ÿšฑ",
-  "nose": "๐Ÿ‘ƒ",
-  "notebook": "๐Ÿ““",
-  "notebook_with_decorative_cover": "๐Ÿ“”",
-  "notes": "๐ŸŽถ",
-  "nut_and_bolt": "๐Ÿ”ฉ",
-  "o": "โญ•",
-  "o_button_blood_type": "๐Ÿ…พ",
-  "ocean": "๐ŸŒŠ",
-  "octagonal_sign": "๐Ÿ›‘",
-  "octopus": "๐Ÿ™",
-  "oden": "๐Ÿข",
-  "office": "๐Ÿข",
-  "oil_drum": "๐Ÿ›ข",
-  "ok": "๐Ÿ†—",
-  "ok_hand": "๐Ÿ‘Œ",
-  "ok_woman": "๐Ÿ™†",
-  "old_key": "๐Ÿ—",
-  "older_adult": "๐Ÿง“",
-  "older_man": "๐Ÿ‘ด",
-  "older_person": "๐Ÿง“",
-  "older_woman": "๐Ÿ‘ต",
-  "om_symbol": "๐Ÿ•‰",
-  "on": "๐Ÿ”›",
-  "oncoming_automobile": "๐Ÿš˜",
-  "oncoming_bus": "๐Ÿš",
-  "oncoming_fist": "๐Ÿ‘Š",
-  "oncoming_police_car": "๐Ÿš”",
-  "oncoming_taxi": "๐Ÿš–",
-  "one": "1โƒฃ",
-  "onepiece_swimsuit": "๐Ÿฉฑ",
-  "onion": "๐Ÿง…",
-  "open_file_folder": "๐Ÿ“‚",
-  "open_hands": "๐Ÿ‘",
-  "open_mouth": "๐Ÿ˜ฎ",
-  "ophiuchus": "โ›Ž",
-  "orange_book": "๐Ÿ“™",
-  "orange_circle": "๐ŸŸ ",
-  "orange_heart": "๐Ÿงก",
-  "orange_square": "๐ŸŸง",
-  "orangutan": "๐Ÿฆง",
-  "orthodox_cross": "โ˜ฆ๏ธ",
-  "otter": "๐Ÿฆฆ",
-  "outbox_tray": "๐Ÿ“ค",
-  "owl": "๐Ÿฆ‰",
-  "ox": "๐Ÿ‚",
-  "oyster": "๐Ÿฆช",
-  "p_button": "๐Ÿ…ฟ",
-  "package": "๐Ÿ“ฆ",
-  "page_facing_up": "๐Ÿ“„",
-  "page_with_curl": "๐Ÿ“ƒ",
-  "pager": "๐Ÿ“Ÿ",
-  "paintbrush": "๐Ÿ–Œ",
-  "palm_tree": "๐ŸŒด",
-  "palms_up_together": "๐Ÿคฒ",
-  "pancakes": "๐Ÿฅž",
-  "panda_face": "๐Ÿผ",
-  "paperclip": "๐Ÿ“Ž",
-  "parachute": "๐Ÿช‚",
-  "parrot": "๐Ÿฆœ",
-  "part_alternation_mark": "ใ€ฝ",
-  "partly_sunny": "โ›…",
-  "partying_face": "๐Ÿฅณ",
-  "passenger_ship": "๐Ÿ›ณ",
-  "passport_control": "๐Ÿ›‚",
-  "pause_button": "โธ๏ธ",
-  "peace": "โ˜ฎ",
-  "peace_symbol": "โ˜ฎ๏ธ",
-  "peach": "๐Ÿ‘",
-  "peacock": "๐Ÿฆš",
-  "peanuts": "๐Ÿฅœ",
-  "pear": "๐Ÿ",
-  "pen": "๐Ÿ–Š",
-  "pencil": "๐Ÿ“",
-  "pencil2": "โœ",
-  "penguin": "๐Ÿง",
-  "pensive": "๐Ÿ˜”",
-  "people_with_bunny_ears_partying": "๐Ÿ‘ฏ",
-  "people_wrestling": "๐Ÿคผ",
-  "performing_arts": "๐ŸŽญ",
-  "persevere": "๐Ÿ˜ฃ",
-  "person": "๐Ÿง‘",
-  "person_biking": "๐Ÿšด",
-  "person_bouncing_ball": "โ›น๏ธ",
-  "person_bowing": "๐Ÿ™‡",
-  "person_cartwheeling": "๐Ÿคธ",
-  "person_climbing": "๐Ÿง—",
-  "person_doing_cartwheel": "๐Ÿคธ",
-  "person_facepalming": "๐Ÿคฆ",
-  "person_fencing": "๐Ÿคบ",
-  "person_frowning": "๐Ÿ™",
-  "person_gesturing_no": "๐Ÿ™…",
-  "person_gesturing_ok": "๐Ÿ™†",
-  "person_getting_haircut": "๐Ÿ’‡",
-  "person_getting_massage": "๐Ÿ’†",
-  "person_in_lotus_position": "๐Ÿง˜",
-  "person_in_steamy_room": "๐Ÿง–",
-  "person_juggling": "๐Ÿคน",
-  "person_kneeling": "๐ŸงŽ",
-  "person_mountain_biking": "๐Ÿšต",
-  "person_playing_handball": "๐Ÿคพ",
-  "person_playing_water_polo": "๐Ÿคฝ",
-  "person_pouting": "๐Ÿ™Ž",
-  "person_raising_hand": "๐Ÿ™‹",
-  "person_rowing_boat": "๐Ÿšฃ",
-  "person_running": "๐Ÿƒ",
-  "person_shrugging": "๐Ÿคท",
-  "person_standing": "๐Ÿง",
-  "person_surfing": "๐Ÿ„",
-  "person_swimming": "๐ŸŠ",
-  "person_tipping_hand": "๐Ÿ’",
-  "person_walking": "๐Ÿšถ",
-  "person_wearing_turban": "๐Ÿ‘ณ",
-  "person_with_blond_hair": "๐Ÿ‘ฑ",
-  "person_with_pouting_face": "๐Ÿ™Ž",
-  "petri_dish": "๐Ÿงซ",
-  "pick": "โ›๏ธ",
-  "pie": "๐Ÿฅง",
-  "pig": "๐Ÿท",
-  "pig2": "๐Ÿ–",
-  "pig_nose": "๐Ÿฝ",
-  "pill": "๐Ÿ’Š",
-  "pinching_hand": "๐Ÿค",
-  "pineapple": "๐Ÿ",
-  "ping_pong": "๐Ÿ“",
-  "pisces": "โ™“",
-  "pizza": "๐Ÿ•",
-  "place_of_worship": "๐Ÿ›",
-  "play_button": "โ–ถ",
-  "play_or_pause_button": "โฏ๏ธ",
-  "play_pause": "โฏ",
-  "pleading_face": "๐Ÿฅบ",
-  "point_down": "๐Ÿ‘‡",
-  "point_left": "๐Ÿ‘ˆ",
-  "point_right": "๐Ÿ‘‰",
-  "point_up": "โ˜๏ธ",
-  "point_up_2": "๐Ÿ‘†",
-  "police_car": "๐Ÿš“",
-  "police_officer": "๐Ÿ‘ฎ",
-  "poodle": "๐Ÿฉ",
-  "poop": "๐Ÿ’ฉ",
-  "popcorn": "๐Ÿฟ",
-  "post_office": "๐Ÿฃ",
-  "postal_horn": "๐Ÿ“ฏ",
-  "postbox": "๐Ÿ“ฎ",
-  "potable_water": "๐Ÿšฐ",
-  "potato": "๐Ÿฅ”",
-  "pouch": "๐Ÿ‘",
-  "poultry_leg": "๐Ÿ—",
-  "pound": "๐Ÿ’ท",
-  "pouting_cat": "๐Ÿ˜พ",
-  "pray": "๐Ÿ™",
-  "prayer_beads": "๐Ÿ“ฟ",
-  "pregnant_woman": "๐Ÿคฐ",
-  "pretzel": "๐Ÿฅจ",
-  "prince": "๐Ÿคด",
-  "princess": "๐Ÿ‘ธ",
-  "printer": "๐Ÿ–จ",
-  "probing_cane": "๐Ÿฆฏ",
-  "punch": "๐Ÿ‘Š",
-  "purple_circle": "๐ŸŸฃ",
-  "purple_heart": "๐Ÿ’œ",
-  "purse": "๐Ÿ‘›",
-  "pushpin": "๐Ÿ“Œ",
-  "put_litter_in_its_place": "๐Ÿšฎ",
-  "puzzle_piece": "๐Ÿงฉ",
-  "question": "โ“",
-  "rabbit": "๐Ÿฐ",
-  "rabbit2": "๐Ÿ‡",
-  "raccoon": "๐Ÿฆ",
-  "racehorse": "๐ŸŽ",
-  "racing_car": "๐ŸŽ",
-  "radio": "๐Ÿ“ป",
-  "radio_button": "๐Ÿ”˜",
-  "radioactive": "โ˜ข๏ธ",
-  "rage": "๐Ÿ˜ก",
-  "railway_car": "๐Ÿšƒ",
-  "railway_track": "๐Ÿ›ค",
-  "rainbow": "๐ŸŒˆ",
-  "raised_back_of_hand": "๐Ÿคš",
-  "raised_hand": "โœ‹",
-  "raised_hands": "๐Ÿ™Œ",
-  "raising_hand": "๐Ÿ™‹",
-  "ram": "๐Ÿ",
-  "ramen": "๐Ÿœ",
-  "rat": "๐Ÿ€",
-  "razor": "๐Ÿช’",
-  "receipt": "๐Ÿงพ",
-  "record_button": "โบ๏ธ",
-  "recycle": "โ™ป",
-  "recycling_symbol": "โ™ป๏ธ",
-  "red_car": "๐Ÿš—",
-  "red_circle": "๐Ÿ”ด",
-  "red_envelope": "๐Ÿงง",
-  "red_hair": "๐Ÿฆฐ",
-  "red_heart": "โค",
-  "red_square": "๐ŸŸฅ",
-  "regional_indicator_a": "๐Ÿ‡ฆ",
-  "regional_indicator_b": "๐Ÿ‡ง",
-  "regional_indicator_c": "๐Ÿ‡จ",
-  "regional_indicator_d": "๐Ÿ‡ฉ",
-  "regional_indicator_e": "๐Ÿ‡ช",
-  "regional_indicator_f": "๐Ÿ‡ซ",
-  "regional_indicator_g": "๐Ÿ‡ฌ",
-  "regional_indicator_h": "๐Ÿ‡ญ",
-  "regional_indicator_i": "๐Ÿ‡ฎ",
-  "regional_indicator_j": "๐Ÿ‡ฏ",
-  "regional_indicator_k": "๐Ÿ‡ฐ",
-  "regional_indicator_l": "๐Ÿ‡ฑ",
-  "regional_indicator_m": "๐Ÿ‡ฒ",
-  "regional_indicator_n": "๐Ÿ‡ณ",
-  "regional_indicator_o": "๐Ÿ‡ด",
-  "regional_indicator_p": "๐Ÿ‡ต",
-  "regional_indicator_q": "๐Ÿ‡ถ",
-  "regional_indicator_r": "๐Ÿ‡ท",
-  "regional_indicator_s": "๐Ÿ‡ธ",
-  "regional_indicator_t": "๐Ÿ‡น",
-  "regional_indicator_u": "๐Ÿ‡บ",
-  "regional_indicator_v": "๐Ÿ‡ป",
-  "regional_indicator_w": "๐Ÿ‡ผ",
-  "regional_indicator_x": "๐Ÿ‡ฝ",
-  "regional_indicator_y": "๐Ÿ‡พ",
-  "regional_indicator_z": "๐Ÿ‡ฟ",
-  "registered": "ยฎ",
-  "relieved": "๐Ÿ˜Œ",
-  "reminder_ribbon": "๐ŸŽ—",
-  "repeat": "๐Ÿ”",
-  "repeat_one": "๐Ÿ”‚",
-  "rescue_workerโ€™s_helmet": "โ›‘๏ธ",
-  "restroom": "๐Ÿšป",
-  "reverse_button": "โ—€",
-  "revolving_hearts": "๐Ÿ’ž",
-  "rewind": "โช",
-  "rhino": "๐Ÿฆ",
-  "rhinoceros": "๐Ÿฆ",
-  "ribbon": "๐ŸŽ€",
-  "rice": "๐Ÿš",
-  "rice_ball": "๐Ÿ™",
-  "rice_cracker": "๐Ÿ˜",
-  "rice_scene": "๐ŸŽ‘",
-  "right_arrow": "โžก๏ธ",
-  "right_arrow_curving_down": "โคต",
-  "right_arrow_curving_left": "โ†ฉ",
-  "right_arrow_curving_up": "โคด",
-  "right_facing_fist": "๐Ÿคœ",
-  "rightfacing_fist": "๐Ÿคœ",
-  "ring": "๐Ÿ’",
-  "ringed_planet": "๐Ÿช",
-  "robot": "๐Ÿค–",
-  "rocket": "๐Ÿš€",
-  "rofl": "๐Ÿคฃ",
-  "roll_of_paper": "๐Ÿงป",
-  "rolledup_newspaper": "๐Ÿ—ž",
-  "roller_coaster": "๐ŸŽข",
-  "rolling_eyes": "๐Ÿ™„",
-  "rolling_on_the_floor_laughing": "๐Ÿคฃ",
-  "rooster": "๐Ÿ“",
-  "rose": "๐ŸŒน",
-  "rosette": "๐Ÿต",
-  "rotating_light": "๐Ÿšจ",
-  "round_pushpin": "๐Ÿ“",
-  "rowboat": "๐Ÿšฃ",
-  "rugby_football": "๐Ÿ‰",
-  "runner": "๐Ÿƒ",
-  "running_shirt_with_sash": "๐ŸŽฝ",
-  "safety_pin": "๐Ÿงท",
-  "safety_vest": "๐Ÿฆบ",
-  "sagittarius": "โ™",
-  "sailboat": "โ›ต",
-  "sake": "๐Ÿถ",
-  "salad": "๐Ÿฅ—",
-  "salt": "๐Ÿง‚",
-  "sandal": "๐Ÿ‘ก",
-  "sandwich": "๐Ÿฅช",
-  "santa": "๐ŸŽ…",
-  "sari": "๐Ÿฅป",
-  "satellite": "๐Ÿ“ก",
-  "sauropod": "๐Ÿฆ•",
-  "saxophone": "๐ŸŽท",
-  "scales": "โš–",
-  "scarf": "๐Ÿงฃ",
-  "school": "๐Ÿซ",
-  "school_satchel": "๐ŸŽ’",
-  "scissors": "โœ‚",
-  "scooter": "๐Ÿ›ด",
-  "scorpion": "๐Ÿฆ‚",
-  "scorpius": "โ™",
-  "scream": "๐Ÿ˜ฑ",
-  "scream_cat": "๐Ÿ™€",
-  "scroll": "๐Ÿ“œ",
-  "seat": "๐Ÿ’บ",
-  "second_place": "๐Ÿฅˆ",
-  "secret": "ใŠ™",
-  "see_no_evil": "๐Ÿ™ˆ",
-  "seedling": "๐ŸŒฑ",
-  "selfie": "๐Ÿคณ",
-  "seven": "7โƒฃ",
-  "shallow_pan_of_food": "๐Ÿฅ˜",
-  "shamrock": "โ˜˜๏ธ",
-  "shark": "๐Ÿฆˆ",
-  "shaved_ice": "๐Ÿง",
-  "sheep": "๐Ÿ‘",
-  "shell": "๐Ÿš",
-  "shield": "๐Ÿ›ก",
-  "shinto_shrine": "โ›ฉ๏ธ",
-  "ship": "๐Ÿšข",
-  "shirt": "๐Ÿ‘•",
-  "shopping_bags": "๐Ÿ›",
-  "shopping_cart": "๐Ÿ›’",
-  "shorts": "๐Ÿฉณ",
-  "shower": "๐Ÿšฟ",
-  "shrimp": "๐Ÿฆ",
-  "shushing_face": "๐Ÿคซ",
-  "sign_of_the_horns": "๐Ÿค˜",
-  "signal_strength": "๐Ÿ“ถ",
-  "six": "6โƒฃ",
-  "six_pointed_star": "๐Ÿ”ฏ",
-  "skateboard": "๐Ÿ›น",
-  "ski": "๐ŸŽฟ",
-  "skier": "โ›ท๏ธ",
-  "skull": "๐Ÿ’€",
-  "skull_and_crossbones": "โ˜ ๏ธ",
-  "skull_crossbones": "โ˜ ",
-  "skunk": "๐Ÿฆจ",
-  "sled": "๐Ÿ›ท",
-  "sleeping": "๐Ÿ˜ด",
-  "sleeping_accommodation": "๐Ÿ›Œ",
-  "sleepy": "๐Ÿ˜ช",
-  "slight_frown": "๐Ÿ™",
-  "slight_smile": "๐Ÿ™‚",
-  "slightly_frowning_face": "๐Ÿ™",
-  "slot_machine": "๐ŸŽฐ",
-  "sloth": "๐Ÿฆฅ",
-  "small_airplane": "๐Ÿ›ฉ",
-  "small_blue_diamond": "๐Ÿ”น",
-  "small_orange_diamond": "๐Ÿ”ธ",
-  "small_red_triangle": "๐Ÿ”บ",
-  "small_red_triangle_down": "๐Ÿ”ป",
-  "smile": "๐Ÿ˜„",
-  "smile_cat": "๐Ÿ˜ธ",
-  "smiley": "๐Ÿ˜ƒ",
-  "smiley_cat": "๐Ÿ˜บ",
-  "smiling": "โ˜บ๏ธ",
-  "smiling_face": "โ˜บ",
-  "smiling_face_with_hearts": "๐Ÿฅฐ",
-  "smiling_imp": "๐Ÿ˜ˆ",
-  "smirk": "๐Ÿ˜",
-  "smirk_cat": "๐Ÿ˜ผ",
-  "smoking": "๐Ÿšฌ",
-  "snail": "๐ŸŒ",
-  "snake": "๐Ÿ",
-  "sneezing_face": "๐Ÿคง",
-  "snowboarder": "๐Ÿ‚",
-  "snowcapped_mountain": "๐Ÿ”",
-  "snowflake": "โ„",
-  "snowman": "โ›„",
-  "soap": "๐Ÿงผ",
-  "sob": "๐Ÿ˜ญ",
-  "soccer": "โšฝ",
-  "socks": "๐Ÿงฆ",
-  "softball": "๐ŸฅŽ",
-  "soon": "๐Ÿ”œ",
-  "sos": "๐Ÿ†˜",
-  "sound": "๐Ÿ”‰",
-  "space_invader": "๐Ÿ‘พ",
-  "spade_suit": "โ™ ๏ธ",
-  "spades": "โ™ ",
-  "spaghetti": "๐Ÿ",
-  "sparkle": "โ‡",
-  "sparkler": "๐ŸŽ‡",
-  "sparkles": "โœจ",
-  "sparkling_heart": "๐Ÿ’–",
-  "speak_no_evil": "๐Ÿ™Š",
-  "speaker": "๐Ÿ”ˆ",
-  "speaking_head": "๐Ÿ—ฃ",
-  "speech_balloon": "๐Ÿ’ฌ",
-  "speech_left": "๐Ÿ—จ",
-  "speedboat": "๐Ÿšค",
-  "spider": "๐Ÿ•ท",
-  "spider_web": "๐Ÿ•ธ",
-  "spiral_calendar": "๐Ÿ—“",
-  "spiral_notepad": "๐Ÿ—’",
-  "sponge": "๐Ÿงฝ",
-  "spoon": "๐Ÿฅ„",
-  "squid": "๐Ÿฆ‘",
-  "stadium": "๐ŸŸ",
-  "star": "โญ",
-  "star2": "๐ŸŒŸ",
-  "star_and_crescent": "โ˜ช๏ธ",
-  "star_of_david": "โœก",
-  "star_struck": "๐Ÿคฉ",
-  "stars": "๐ŸŒ ",
-  "starstruck": "๐Ÿคฉ",
-  "station": "๐Ÿš‰",
-  "statue_of_liberty": "๐Ÿ—ฝ",
-  "steam_locomotive": "๐Ÿš‚",
-  "stethoscope": "๐Ÿฉบ",
-  "stew": "๐Ÿฒ",
-  "stop_button": "โน๏ธ",
-  "stopwatch": "โฑ๏ธ",
-  "straight_ruler": "๐Ÿ“",
-  "strawberry": "๐Ÿ“",
-  "stuck_out_tongue": "๐Ÿ˜›",
-  "stuck_out_tongue_closed_eyes": "๐Ÿ˜",
-  "stuck_out_tongue_winking_eye": "๐Ÿ˜œ",
-  "studio_microphone": "๐ŸŽ™",
-  "stuffed_flatbread": "๐Ÿฅ™",
-  "sun": "โ˜€",
-  "sun_behind_large_cloud": "๐ŸŒฅ",
-  "sun_behind_rain_cloud": "๐ŸŒฆ",
-  "sun_behind_small_cloud": "๐ŸŒค",
-  "sun_with_face": "๐ŸŒž",
-  "sunflower": "๐ŸŒป",
-  "sunglasses": "๐Ÿ˜Ž",
-  "sunny": "โ˜€๏ธ",
-  "sunrise": "๐ŸŒ…",
-  "sunrise_over_mountains": "๐ŸŒ„",
-  "superhero": "๐Ÿฆธ",
-  "supervillain": "๐Ÿฆน",
-  "surfer": "๐Ÿ„",
-  "sushi": "๐Ÿฃ",
-  "suspension_railway": "๐ŸšŸ",
-  "swan": "๐Ÿฆข",
-  "sweat": "๐Ÿ˜“",
-  "sweat_drops": "๐Ÿ’ฆ",
-  "sweat_smile": "๐Ÿ˜…",
-  "sweet_potato": "๐Ÿ ",
-  "swimmer": "๐ŸŠ",
-  "symbols": "๐Ÿ”ฃ",
-  "synagogue": "๐Ÿ•",
-  "syringe": "๐Ÿ’‰",
-  "t_rex": "๐Ÿฆ–",
-  "taco": "๐ŸŒฎ",
-  "tada": "๐ŸŽ‰",
-  "takeout_box": "๐Ÿฅก",
-  "tanabata_tree": "๐ŸŽ‹",
-  "tangerine": "๐ŸŠ",
-  "taurus": "โ™‰",
-  "taxi": "๐Ÿš•",
-  "tea": "๐Ÿต",
-  "teddy_bear": "๐Ÿงธ",
-  "telephone": "โ˜Ž",
-  "telephone_receiver": "๐Ÿ“ž",
-  "telescope": "๐Ÿ”ญ",
-  "tennis": "๐ŸŽพ",
-  "tent": "โ›บ",
-  "test_tube": "๐Ÿงช",
-  "thermometer": "๐ŸŒก",
-  "thermometer_face": "๐Ÿค’",
-  "thinking": "๐Ÿค”",
-  "third_place": "๐Ÿฅ‰",
-  "thought_balloon": "๐Ÿ’ญ",
-  "thread": "๐Ÿงต",
-  "three": "3โƒฃ",
-  "thumbsdown": "๐Ÿ‘Ž",
-  "thumbsup": "๐Ÿ‘",
-  "ticket": "๐ŸŽซ",
-  "tiger": "๐Ÿฏ",
-  "tiger2": "๐Ÿ…",
-  "timer_clock": "โฒ๏ธ",
-  "tired_face": "๐Ÿ˜ซ",
-  "tm": "โ„ข",
-  "toilet": "๐Ÿšฝ",
-  "tokyo_tower": "๐Ÿ—ผ",
-  "tomato": "๐Ÿ…",
-  "tone1": "๐Ÿป",
-  "tone2": "๐Ÿผ",
-  "tone3": "๐Ÿฝ",
-  "tone4": "๐Ÿพ",
-  "tone5": "๐Ÿฟ",
-  "tongue": "๐Ÿ‘…",
-  "toolbox": "๐Ÿงฐ",
-  "tooth": "๐Ÿฆท",
-  "top": "๐Ÿ”",
-  "tophat": "๐ŸŽฉ",
-  "tornado": "๐ŸŒช",
-  "track_next": "โญ",
-  "track_previous": "โฎ",
-  "trackball": "๐Ÿ–ฒ",
-  "tractor": "๐Ÿšœ",
-  "trade_mark": "โ„ข๏ธ",
-  "traffic_light": "๐Ÿšฅ",
-  "train": "๐Ÿš‹",
-  "train2": "๐Ÿš†",
-  "tram": "๐ŸšŠ",
-  "trex": "๐Ÿฆ–",
-  "triangular_flag_on_post": "๐Ÿšฉ",
-  "triangular_ruler": "๐Ÿ“",
-  "trident": "๐Ÿ”ฑ",
-  "triumph": "๐Ÿ˜ค",
-  "trolleybus": "๐ŸšŽ",
-  "trophy": "๐Ÿ†",
-  "tropical_drink": "๐Ÿน",
-  "tropical_fish": "๐Ÿ ",
-  "truck": "๐Ÿšš",
-  "trumpet": "๐ŸŽบ",
-  "tulip": "๐ŸŒท",
-  "tumbler_glass": "๐Ÿฅƒ",
-  "turkey": "๐Ÿฆƒ",
-  "turtle": "๐Ÿข",
-  "tv": "๐Ÿ“บ",
-  "twisted_rightwards_arrows": "๐Ÿ”€",
-  "two": "2โƒฃ",
-  "two_hearts": "๐Ÿ’•",
-  "two_men_holding_hands": "๐Ÿ‘ฌ",
-  "two_women_holding_hands": "๐Ÿ‘ญ",
-  "u5272": "๐Ÿˆน",
-  "u5408": "๐Ÿˆด",
-  "u55b6": "๐Ÿˆบ",
-  "u6307": "๐Ÿˆฏ",
-  "u6708": "๐Ÿˆท",
-  "u6709": "๐Ÿˆถ",
-  "u6e80": "๐Ÿˆต",
-  "u7121": "๐Ÿˆš",
-  "u7533": "๐Ÿˆธ",
-  "u7981": "๐Ÿˆฒ",
-  "u7a7a": "๐Ÿˆณ",
-  "umbrella": "โ˜”",
-  "umbrella_on_ground": "โ›ฑ๏ธ",
-  "unamused": "๐Ÿ˜’",
-  "underage": "๐Ÿ”ž",
-  "unicorn": "๐Ÿฆ„",
-  "unlock": "๐Ÿ”“",
-  "up": "๐Ÿ†™",
-  "up_arrow": "โฌ†",
-  "updown_arrow": "โ†•๏ธ",
-  "upleft_arrow": "โ†–๏ธ",
-  "upright_arrow": "โ†—",
-  "upside_down": "๐Ÿ™ƒ",
-  "v": "โœŒ๏ธ",
-  "vampire": "๐Ÿง›",
-  "vertical_traffic_light": "๐Ÿšฆ",
-  "vhs": "๐Ÿ“ผ",
-  "vibration_mode": "๐Ÿ“ณ",
-  "victory_hand": "โœŒ",
-  "video_camera": "๐Ÿ“น",
-  "video_game": "๐ŸŽฎ",
-  "violin": "๐ŸŽป",
-  "virgo": "โ™",
-  "volcano": "๐ŸŒ‹",
-  "volleyball": "๐Ÿ",
-  "vs": "๐Ÿ†š",
-  "vulcan": "๐Ÿ––",
-  "vulcan_salute": "๐Ÿ––",
-  "waffle": "๐Ÿง‡",
-  "walking": "๐Ÿšถ",
-  "waning_crescent_moon": "๐ŸŒ˜",
-  "waning_gibbous_moon": "๐ŸŒ–",
-  "warning": "โš ",
-  "wastebasket": "๐Ÿ—‘",
-  "watch": "โŒš",
-  "water_buffalo": "๐Ÿƒ",
-  "watermelon": "๐Ÿ‰",
-  "wave": "๐Ÿ‘‹",
-  "wavy_dash": "ใ€ฐ๏ธ",
-  "waxing_crescent_moon": "๐ŸŒ’",
-  "waxing_gibbous_moon": "๐ŸŒ”",
-  "wc": "๐Ÿšพ",
-  "weary": "๐Ÿ˜ฉ",
-  "wedding": "๐Ÿ’’",
-  "weightlifter": "๐Ÿ‹",
-  "whale": "๐Ÿณ",
-  "whale2": "๐Ÿ‹",
-  "wheel_of_dharma": "โ˜ธ๏ธ",
-  "wheelchair": "โ™ฟ",
-  "white_check_mark": "โœ…",
-  "white_circle": "โšช",
-  "white_flower": "๐Ÿ’ฎ",
-  "white_hair": "๐Ÿฆณ",
-  "white_heart": "๐Ÿค",
-  "white_large_square": "โฌœ",
-  "white_medium_small_square": "โ—ฝ",
-  "white_medium_square": "โ—ป๏ธ",
-  "white_small_square": "โ–ซ๏ธ",
-  "white_square_button": "๐Ÿ”ณ",
-  "wilted_flower": "๐Ÿฅ€",
-  "wilted_rose": "๐Ÿฅ€",
-  "wind_blowing_face": "๐ŸŒฌ",
-  "wind_chime": "๐ŸŽ",
-  "wine_glass": "๐Ÿท",
-  "wink": "๐Ÿ˜‰",
-  "wolf": "๐Ÿบ",
-  "woman": "๐Ÿ‘ฉ",
-  "woman_with_headscarf": "๐Ÿง•",
-  "womans_clothes": "๐Ÿ‘š",
-  "womans_hat": "๐Ÿ‘’",
-  "womens": "๐Ÿšบ",
-  "woozy_face": "๐Ÿฅด",
-  "world_map": "๐Ÿ—บ",
-  "worried": "๐Ÿ˜Ÿ",
-  "wrench": "๐Ÿ”ง",
-  "writing_hand": "โœ๏ธ",
-  "x": "โŒ",
-  "yarn": "๐Ÿงถ",
-  "yawning_face": "๐Ÿฅฑ",
-  "yellow_circle": "๐ŸŸก",
-  "yellow_heart": "๐Ÿ’›",
-  "yellow_square": "๐ŸŸจ",
-  "yen": "๐Ÿ’ด",
-  "yin_yang": "โ˜ฏ๏ธ",
-  "yoyo": "๐Ÿช€",
-  "yum": "๐Ÿ˜‹",
-  "zany_face": "๐Ÿคช",
-  "zap": "โšก",
-  "zebra": "๐Ÿฆ“",
-  "zero": "0โƒฃ",
-  "zipper_mouth": "๐Ÿค",
-  "zombie": "๐ŸงŸ",
-  "zzz": "๐Ÿ’ค"
-}
\ No newline at end of file
diff --git a/yarn.lock b/yarn.lock
index cf98fef3..f4d4358c 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -9733,6 +9733,11 @@ unicode-canonical-property-names-ecmascript@^2.0.0:
   resolved "https://registry.yarnpkg.com/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz#301acdc525631670d39f6146e0e77ff6bbdebddc"
   integrity sha512-yY5PpDlfVIU5+y/BSCxAJRBIS1Zc2dDG3Ujq+sR0U+JjUevW2JhocOF+soROYDSaAezOzOKuyyixhD6mBknSmQ==
 
+unicode-emoji-json@^0.3.0:
+  version "0.3.0"
+  resolved "https://registry.yarnpkg.com/unicode-emoji-json/-/unicode-emoji-json-0.3.0.tgz#965e097ef8a367081c5036f81873015a95a5c1ad"
+  integrity sha512-yImheILedqhZtVEEenRELu9AnX347ZTA3bVMWAU5WMUv7pdU2hcfpwo2mKN8Rns9uHLmOZP90/4B4SPS5aF/iw==
+
 unicode-match-property-ecmascript@^2.0.0:
   version "2.0.0"
   resolved "https://registry.yarnpkg.com/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-2.0.0.tgz#54fd16e0ecb167cf04cf1f756bdcc92eba7976c3"

From 1953320d80dcbdf895798e6319ec69e1075bc4a8 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 16:55:00 -0500
Subject: [PATCH 24/30] Make emoji picker use grouped unicode emojis

---
 src/components/emoji_input/suggestor.js       |  2 +-
 src/components/emoji_picker/emoji_picker.js   | 16 ++++----
 .../post_status_form/post_status_form.js      |  6 +--
 src/components/react_button/react_button.js   |  4 +-
 .../settings_modal/tabs/profile_tab.js        |  4 +-
 src/modules/instance.js                       | 38 +++++++++++++++----
 6 files changed, 48 insertions(+), 22 deletions(-)

diff --git a/src/components/emoji_input/suggestor.js b/src/components/emoji_input/suggestor.js
index e8efbd1e..befe82be 100644
--- a/src/components/emoji_input/suggestor.js
+++ b/src/components/emoji_input/suggestor.js
@@ -2,7 +2,7 @@
  * suggest - generates a suggestor function to be used by emoji-input
  * data: object providing source information for specific types of suggestions:
  * data.emoji - optional, an array of all emoji available i.e.
- *   (state.instance.emoji + state.instance.customEmoji)
+ *   (getters.standardEmojiList + state.instance.customEmoji)
  * data.users - optional, an array of all known users
  * updateUsersList - optional, a function to search and append to users
  *
diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 612e0c19..5d759ec3 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -213,16 +213,18 @@ const EmojiPicker = {
     defaultGroup () {
       return Object.keys(this.allCustomGroups)[0]
     },
+    unicodeEmojiGroups () {
+      return this.$store.getters.standardEmojiGroupList.map(group => ({
+        id: `standard-${group.id}`,
+        text: this.$t(`emoji.unicode_groups.${group.id}`),
+        icon: 'box-open',
+        emojis: group.emojis
+      }))
+    },
     allEmojiGroups () {
-      const standardEmojis = this.$store.state.instance.emoji || []
       return Object.entries(this.allCustomGroups)
         .map(([_, v]) => v)
-        .concat({
-          id: 'standard',
-          text: this.$t('emoji.unicode'),
-          icon: 'box-open',
-          emojis: filterByKeyword(standardEmojis, this.keyword)
-        })
+        .concat(this.unicodeEmojiGroups)
     },
     filteredEmojiGroups () {
       return this.allEmojiGroups
diff --git a/src/components/post_status_form/post_status_form.js b/src/components/post_status_form/post_status_form.js
index 84a9e29e..a4e57cb7 100644
--- a/src/components/post_status_form/post_status_form.js
+++ b/src/components/post_status_form/post_status_form.js
@@ -164,7 +164,7 @@ const PostStatusForm = {
     emojiUserSuggestor () {
       return suggestor({
         emoji: [
-          ...this.$store.state.instance.emoji,
+          ...this.$store.getters.standardEmojiList,
           ...this.$store.state.instance.customEmoji
         ],
         store: this.$store
@@ -173,13 +173,13 @@ const PostStatusForm = {
     emojiSuggestor () {
       return suggestor({
         emoji: [
-          ...this.$store.state.instance.emoji,
+          ...this.$store.getters.standardEmojiList,
           ...this.$store.state.instance.customEmoji
         ]
       })
     },
     emoji () {
-      return this.$store.state.instance.emoji || []
+      return this.$store.getters.standardEmojiList || []
     },
     customEmoji () {
       return this.$store.state.instance.customEmoji || []
diff --git a/src/components/react_button/react_button.js b/src/components/react_button/react_button.js
index ce82c90d..c2d9bfb5 100644
--- a/src/components/react_button/react_button.js
+++ b/src/components/react_button/react_button.js
@@ -45,7 +45,7 @@ const ReactButton = {
       if (this.filterWord !== '') {
         const filterWordLowercase = this.filterWord.toLowerCase()
         let orderedEmojiList = []
-        for (const emoji of this.$store.state.instance.emoji) {
+        for (const emoji of this.$store.getters.standardEmojiList) {
           if (emoji.replacement === this.filterWord) return [emoji]
 
           const indexOfFilterWord = emoji.displayText.toLowerCase().indexOf(filterWordLowercase)
@@ -58,7 +58,7 @@ const ReactButton = {
         }
         return orderedEmojiList.flat()
       }
-      return this.$store.state.instance.emoji || []
+      return this.$store.getters.standardEmojiList || []
     },
     mergedConfig () {
       return this.$store.getters.mergedConfig
diff --git a/src/components/settings_modal/tabs/profile_tab.js b/src/components/settings_modal/tabs/profile_tab.js
index bee8a7bb..4b68b46d 100644
--- a/src/components/settings_modal/tabs/profile_tab.js
+++ b/src/components/settings_modal/tabs/profile_tab.js
@@ -60,7 +60,7 @@ const ProfileTab = {
     emojiUserSuggestor () {
       return suggestor({
         emoji: [
-          ...this.$store.state.instance.emoji,
+          ...this.$store.getters.standardEmojiList,
           ...this.$store.state.instance.customEmoji
         ],
         store: this.$store
@@ -68,7 +68,7 @@ const ProfileTab = {
     },
     emojiSuggestor () {
       return suggestor({ emoji: [
-        ...this.$store.state.instance.emoji,
+        ...this.$store.getters.standardEmojiList,
         ...this.$store.state.instance.customEmoji
       ] })
     },
diff --git a/src/modules/instance.js b/src/modules/instance.js
index 15825783..64d04464 100644
--- a/src/modules/instance.js
+++ b/src/modules/instance.js
@@ -3,6 +3,18 @@ import { CURRENT_VERSION } from '../services/theme_data/theme_data.service.js'
 import apiService from '../services/api/api.service.js'
 import { instanceDefaultProperties } from './config.js'
 
+const SORTED_EMOJI_GROUP_IDS = [
+  'smileys-and-emotion',
+  'people-and-body',
+  'animals-and-nature',
+  'food-and-drink',
+  'travel-and-places',
+  'activities',
+  'objects',
+  'symbols',
+  'flags'
+]
+
 const defaultState = {
   // Stuff from apiConfig
   name: 'Pleroma FE',
@@ -63,7 +75,7 @@ const defaultState = {
   // Nasty stuff
   customEmoji: [],
   customEmojiFetched: false,
-  emoji: [],
+  emoji: {},
   emojiFetched: false,
   pleromaBackend: true,
   postFormats: [],
@@ -138,6 +150,17 @@ const instance = {
           return res
         }, {})
     },
+    standardEmojiList (state) {
+      return SORTED_EMOJI_GROUP_IDS
+        .map(groupId => state.emoji[groupId] || [])
+        .reduce((a, b) => a.concat(b), [])
+    },
+    standardEmojiGroupList (state) {
+      return SORTED_EMOJI_GROUP_IDS.map(groupId => ({
+        id: groupId,
+        emojis: state.emoji[groupId] || []
+      }))
+    },
     instanceDomain (state) {
       return new URL(state.server).hostname
     }
@@ -164,13 +187,14 @@ const instance = {
         const res = await window.fetch('/static/emoji.json')
         if (res.ok) {
           const values = await res.json()
-          const emoji = Object.keys(values).map((key) => {
-            return {
-              displayText: key,
+          const emoji = Object.keys(values).reduce((res, groupId) => {
+            res[groupId] = values[groupId].map(e => ({
+              displayText: e.name,
               imageUrl: false,
-              replacement: values[key]
-            }
-          }).sort((a, b) => a.name > b.name ? 1 : -1)
+              replacement: e.emoji
+            }))
+            return res
+          }, {})
           commit('setInstanceOption', { name: 'emoji', value: emoji })
         } else {
           throw (res)

From 8e9c6f9528db193c5e6f33ba2cfc308989493338 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 17:14:23 -0500
Subject: [PATCH 25/30] Add icons for unicode emoji groups

---
 src/components/emoji_picker/emoji_picker.js | 36 +++++++++++++++++++--
 1 file changed, 33 insertions(+), 3 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 5d759ec3..58713fb7 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -6,15 +6,45 @@ import { library } from '@fortawesome/fontawesome-svg-core'
 import {
   faBoxOpen,
   faStickyNote,
-  faSmileBeam
+  faSmileBeam,
+  faSmile,
+  faUser,
+  faPaw,
+  faIceCream,
+  faBus,
+  faBasketballBall,
+  faLightbulb,
+  faCode,
+  faFlag
 } from '@fortawesome/free-solid-svg-icons'
 
 library.add(
   faBoxOpen,
   faStickyNote,
-  faSmileBeam
+  faSmileBeam,
+  faSmile,
+  faUser,
+  faPaw,
+  faIceCream,
+  faBus,
+  faBasketballBall,
+  faLightbulb,
+  faCode,
+  faFlag
 )
 
+const UNICODE_EMOJI_GROUP_ICON = {
+  'smileys-and-emotion': 'smile',
+  'people-and-body': 'user',
+  'animals-and-nature': 'paw',
+  'food-and-drink': 'ice-cream',
+  'travel-and-places': 'bus',
+  'activities': 'basketball-ball',
+  'objects': 'lightbulb',
+  'symbols': 'code',
+  'flags': 'flag',
+}
+
 const filterByKeyword = (list, keyword = '') => {
   if (keyword === '') return list
 
@@ -217,7 +247,7 @@ const EmojiPicker = {
       return this.$store.getters.standardEmojiGroupList.map(group => ({
         id: `standard-${group.id}`,
         text: this.$t(`emoji.unicode_groups.${group.id}`),
-        icon: 'box-open',
+        icon: UNICODE_EMOJI_GROUP_ICON[group.id],
         emojis: group.emojis
       }))
     },

From 210a67f444f27c602a9e51bc52fc39d31ad1d911 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 17:16:48 -0500
Subject: [PATCH 26/30] Add English translation for unicode emoji group names

---
 src/i18n/en.json | 11 +++++++++++
 1 file changed, 11 insertions(+)

diff --git a/src/i18n/en.json b/src/i18n/en.json
index f8336e5c..9b09daf6 100644
--- a/src/i18n/en.json
+++ b/src/i18n/en.json
@@ -186,6 +186,17 @@
     "add_emoji": "Insert emoji",
     "custom": "Custom emoji",
     "unicode": "Unicode emoji",
+    "unicode_groups": {
+      "activities": "Activities",
+      "animals-and-nature": "Animals & Nature",
+      "flags": "Flags",
+      "food-and-drink": "Food & Drink",
+      "objects": "Objects",
+      "people-and-body": "People & Body",
+      "smileys-and-emotion": "Smileys & Emotion",
+      "symbols": "Symbols",
+      "travel-and-places": "Travel & Places"
+    },
     "load_all_hint": "Loaded first {saneAmount} emoji, loading all emoji may cause performance issues.",
     "load_all": "Loading all {emojiAmount} emoji"
   },

From 3ed30dd2ba0fcfbd6439190c1c841abb0becb62e Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Sat, 8 Jan 2022 17:17:32 -0500
Subject: [PATCH 27/30] Lint

---
 src/components/emoji_picker/emoji_picker.js | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 58713fb7..5ca0699d 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -42,7 +42,7 @@ const UNICODE_EMOJI_GROUP_ICON = {
   'activities': 'basketball-ball',
   'objects': 'lightbulb',
   'symbols': 'code',
-  'flags': 'flag',
+  'flags': 'flag'
 }
 
 const filterByKeyword = (list, keyword = '') => {

From 906349e3b4847779cb744bfeecffcdc3814a9649 Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Mon, 17 Jan 2022 23:41:11 -0500
Subject: [PATCH 28/30] Make StillImage react to src changes

---
 src/components/still-image/still-image.js | 8 ++++++++
 1 file changed, 8 insertions(+)

diff --git a/src/components/still-image/still-image.js b/src/components/still-image/still-image.js
index 1806d33b..200ef147 100644
--- a/src/components/still-image/still-image.js
+++ b/src/components/still-image/still-image.js
@@ -57,6 +57,14 @@ const StillImage = {
     onError () {
       this.imageLoadError && this.imageLoadError()
     }
+  },
+  watch: {
+    src () {
+      this.realSrc = this.src
+    },
+    dataSrc () {
+      this.$el.removeAttribute('data-loaded')
+    }
   }
 }
 

From d6a14a48e0d7c553c8227b94f118fdcbce3460ff Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Wed, 6 Apr 2022 21:29:50 -0400
Subject: [PATCH 29/30] Make emoji picker work with vue3

---
 src/components/emoji_picker/emoji_picker.js  | 21 ++++++++++++++------
 src/components/emoji_picker/emoji_picker.vue |  8 +++++---
 2 files changed, 20 insertions(+), 9 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.js b/src/components/emoji_picker/emoji_picker.js
index 5ca0699d..daf1a4c2 100644
--- a/src/components/emoji_picker/emoji_picker.js
+++ b/src/components/emoji_picker/emoji_picker.js
@@ -83,7 +83,9 @@ const EmojiPicker = {
       keepOpen: false,
       customEmojiTimeout: null,
       // Lazy-load only after the first time `showing` becomes true.
-      contentLoaded: false
+      contentLoaded: false,
+      groupRefs: {},
+      emojiRefs: {}
     }
   },
   components: {
@@ -92,6 +94,12 @@ const EmojiPicker = {
     StillImage
   },
   methods: {
+    setGroupRef (name) {
+      return el => { this.groupRefs[name] = el }
+    },
+    setEmojiRef (name) {
+      return el => { this.emojiRefs[name] = el }
+    },
     onStickerUploaded (e) {
       this.$emit('sticker-uploaded', e)
     },
@@ -111,8 +119,8 @@ const EmojiPicker = {
       const top = target.scrollTop + 5
       this.$nextTick(() => {
         this.allEmojiGroups.forEach(group => {
-          const ref = this.$refs['group-' + group.id]
-          if (ref[0].offsetTop <= top) {
+          const ref = this.groupRefs['group-' + group.id]
+          if (ref && ref.offsetTop <= top) {
             this.activeGroup = group.id
           }
         })
@@ -121,7 +129,7 @@ const EmojiPicker = {
     },
     scrollHeader () {
       // Scroll the active tab's header into view
-      const headerRef = this.$refs['group-header-' + this.activeGroup][0]
+      const headerRef = this.groupRefs['group-header-' + this.activeGroup]
       const left = headerRef.offsetLeft
       const right = left + headerRef.offsetWidth
       const headerCont = this.$refs.header
@@ -137,7 +145,7 @@ const EmojiPicker = {
       }
     },
     highlight (key) {
-      const ref = this.$refs['group-' + key]
+      const ref = this.groupRefs['group-' + key]
       const top = ref.offsetTop
       this.setShowStickers(false)
       this.activeGroup = key
@@ -168,7 +176,8 @@ const EmojiPicker = {
       this.$nextTick(() => {
         this.$lozad = lozad('.still-image.emoji-picker-emoji', {
           load: el => {
-            const vn = el.__vue__
+            const name = el.getAttribute('data-emoji-name')
+            const vn = this.emojiRefs[name]
             if (!vn) {
               return
             }
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index b202df91..6084fd66 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -9,7 +9,7 @@
       >
         <span
           v-for="group in filteredEmojiGroups"
-          :ref="'group-header-' + group.id"
+          :ref="setGroupRef('group-header-' + group.id)"
           :key="group.id"
           class="emoji-tabs-item"
           :class="{
@@ -79,7 +79,7 @@
             class="emoji-group"
           >
             <h6
-              :ref="'group-' + group.id"
+              :ref="setGroupRef('group-' + group.id)"
               class="emoji-group-title"
             >
               {{ group.text }}
@@ -95,10 +95,12 @@
               <still-image
                 v-else
                 class="emoji-picker-emoji"
+                :ref="setEmojiRef(group.id + emoji.displayText)"
                 :data-src="emoji.imageUrl"
+                :data-emoji-name="group.id + emoji.displayText"
               />
             </span>
-            <span :ref="'group-end-' + group.id" />
+            <span :ref="setGroupRef('group-end-' + group.id)" />
           </div>
         </div>
         <div class="keep-open">

From 0336cd037046030a8817c4eaaf38cef6b242e2ee Mon Sep 17 00:00:00 2001
From: Tusooa Zhu <tusooa@kazv.moe>
Date: Fri, 29 Apr 2022 22:40:06 -0400
Subject: [PATCH 30/30] Limit the width of unsupported multichar emojis

---
 src/components/emoji_picker/emoji_picker.scss | 8 ++++++--
 src/components/emoji_picker/emoji_picker.vue  | 7 +++++--
 2 files changed, 11 insertions(+), 4 deletions(-)

diff --git a/src/components/emoji_picker/emoji_picker.scss b/src/components/emoji_picker/emoji_picker.scss
index a7c89303..92a0cbcb 100644
--- a/src/components/emoji_picker/emoji_picker.scss
+++ b/src/components/emoji_picker/emoji_picker.scss
@@ -195,18 +195,22 @@ $emoji-picker-emoji-size: 32px;
       height: $emoji-picker-emoji-size;
       box-sizing: border-box;
       display: flex;
-      font-size: $emoji-picker-emoji-size;
+      line-height: $emoji-picker-emoji-size;
       align-items: center;
       justify-content: center;
       margin: 4px;
 
       cursor: pointer;
 
-      img {
+      .emoji-picker-emoji.-custom {
         object-fit: contain;
         max-width: 100%;
         max-height: 100%;
       }
+      .emoji-picker-emoji.-unicode {
+        font-size: 24px;
+        overflow: hidden;
+      }
     }
 
   }
diff --git a/src/components/emoji_picker/emoji_picker.vue b/src/components/emoji_picker/emoji_picker.vue
index 6084fd66..d61e5d8b 100644
--- a/src/components/emoji_picker/emoji_picker.vue
+++ b/src/components/emoji_picker/emoji_picker.vue
@@ -91,10 +91,13 @@
               class="emoji-item"
               @click.stop.prevent="onEmoji(emoji)"
             >
-              <span v-if="!emoji.imageUrl">{{ emoji.replacement }}</span>
+              <span
+                v-if="!emoji.imageUrl"
+                class="emoji-picker-emoji -unicode"
+              >{{ emoji.replacement }}</span>
               <still-image
                 v-else
-                class="emoji-picker-emoji"
+                class="emoji-picker-emoji -custom"
                 :ref="setEmojiRef(group.id + emoji.displayText)"
                 :data-src="emoji.imageUrl"
                 :data-emoji-name="group.id + emoji.displayText"