Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement Emoji Picker #17195

Merged
merged 5 commits into from
Sep 13, 2023
Merged

Implement Emoji Picker #17195

merged 5 commits into from
Sep 13, 2023

Conversation

smohamedjavid
Copy link
Member

@smohamedjavid smohamedjavid commented Sep 5, 2023

fixes #17057

Summary

This PR adds Emoji Picker in the app for usage in Message Composer and Wallet Account.

Android iOS
Android.mp4
iOS.mp4

We can not depend on the external libraries as they are not modular enough to style and don't match our design. So, we created our emoji picker from a dataset to match the design and deliver good performance with rendering and searching.

Dataset:

Emoji data is pulled from the emojibase. It's lightweight, compiles from the Unicode consortium, contains the latest Unicode 15 emojis such as🫸🫷🫨🪼🪭 and includes skin tone support.

We have used the compact dataset: https://cdn.jsdelivr.net/npm/emojibase-data@latest/en/compact.json and removed skin tones (skins) from the data as skin tone support is not planned for MVP (Post MVP, it will be easier to add support to skin tone). Additionally, we have removed the order key as the data set is already ordered. The dataset (JSON) size stands around ~206 KB.

Review notes

As the Emoji Picker needs to be displayed on a bottom sheet screen, we can not use Section List (which is ideal for this case) as the scroll won't work in Android due to the Gesture on the bottom sheet screen. We have to use the Flat List from react-native-gesture-library.

Since the dataset is not grouped, we categorised them for rendering in status-im2.contexts.emoji-picker.data ns and partitioned it for easy rendering using FlatList.

The search is a simple filter on the dataset. It supports search using the emoji name, tags, and emoticons.

Testing notes

This PR can go through only the design review as it has not yet been added to the screens beyond the wallet.

⚠️ NOTES:

  • If you see any rectangle with a cross or question inside it, that emoji is not supported in your device.
  • The recent emojis section will be implemented in a separate issue.

Platforms

  • Android
  • iOS

Steps to test

In Preview:

  • Open Status
  • Navigate to Quo2 Preview > avatar > account avatar
  • Tap on the Open emoji picker button button

In Actual App:

  • Open Status
  • Log in to your account
  • Navigate to the new Wallet screen by long pressing on the wallet tab
  • Tap on the + button, then tap on Add account in bottom drawer to create new wallet account
  • Tap the reaction/emoji button next to the wallet account avatar.

status: ready

@smohamedjavid smohamedjavid added the feature feature requests label Sep 5, 2023
@smohamedjavid smohamedjavid self-assigned this Sep 5, 2023
@status-im-auto
Copy link
Member

status-im-auto commented Sep 5, 2023

Jenkins Builds

Click to see older builds (35)
Commit #️⃣ Finished (UTC) Duration Platform Result
1d48846 #1 2023-09-05 16:02:59 ~2 min tests 📄log
✔️ 1d48846 #1 2023-09-05 16:06:10 ~5 min android-e2e 🤖apk 📲
✔️ 1d48846 #1 2023-09-05 16:06:15 ~6 min android 🤖apk 📲
✔️ 1d48846 #2 2023-09-05 16:21:29 ~6 min ios 📱ipa 📲
✔️ 86994c1 #2 2023-09-05 17:43:42 ~5 min android-e2e 🤖apk 📲
✔️ 86994c1 #3 2023-09-05 17:43:56 ~6 min ios 📱ipa 📲
✔️ 86994c1 #2 2023-09-05 17:47:14 ~9 min android 🤖apk 📲
✔️ 86994c1 #2 2023-09-05 17:47:31 ~9 min tests 📄log
✔️ 1e8ed48 #3 2023-09-05 18:59:35 ~5 min android-e2e 🤖apk 📲
✔️ 1e8ed48 #4 2023-09-05 18:59:47 ~5 min ios 📱ipa 📲
✔️ 1e8ed48 #3 2023-09-05 19:02:18 ~8 min android 🤖apk 📲
✔️ 1e8ed48 #3 2023-09-05 19:03:10 ~9 min tests 📄log
✔️ 07e9d9e #4 2023-09-05 20:15:11 ~5 min android 🤖apk 📲
✔️ 07e9d9e #5 2023-09-05 20:15:53 ~6 min ios 📱ipa 📲
✔️ 07e9d9e #4 2023-09-05 20:18:45 ~9 min android-e2e 🤖apk 📲
✔️ 07e9d9e #4 2023-09-05 20:19:05 ~9 min tests 📄log
✔️ 4576d7e #5 2023-09-07 19:32:08 ~6 min android 🤖apk 📲
✔️ 4576d7e #5 2023-09-07 19:32:12 ~6 min android-e2e 🤖apk 📲
✔️ 4576d7e #6 2023-09-07 19:32:43 ~6 min ios 📱ipa 📲
✔️ 7c4bbd2 #6 2023-09-07 19:39:52 ~5 min android-e2e 🤖apk 📲
✔️ 7c4bbd2 #7 2023-09-07 19:40:37 ~6 min ios 📱ipa 📲
✔️ 7c4bbd2 #6 2023-09-07 19:43:29 ~9 min android 🤖apk 📲
✔️ 7c4bbd2 #6 2023-09-07 19:43:51 ~9 min tests 📄log
✔️ 5501208 #8 2023-09-11 13:54:20 ~6 min ios 📱ipa 📲
✔️ 5501208 #7 2023-09-11 13:54:37 ~6 min android 🤖apk 📲
✔️ 5501208 #7 2023-09-11 13:57:56 ~9 min android-e2e 🤖apk 📲
✔️ 5501208 #7 2023-09-11 13:58:20 ~10 min tests 📄log
✔️ 56e2025 #8 2023-09-11 15:51:35 ~5 min android-e2e 🤖apk 📲
✔️ 56e2025 #9 2023-09-11 15:52:15 ~6 min ios 📱ipa 📲
✔️ 56e2025 #8 2023-09-11 15:53:25 ~7 min android 🤖apk 📲
✔️ 56e2025 #8 2023-09-11 15:55:38 ~9 min tests 📄log
✔️ 6af0822 #9 2023-09-12 16:31:43 ~5 min android 🤖apk 📲
✔️ 6af0822 #9 2023-09-12 16:31:45 ~5 min android-e2e 🤖apk 📲
✔️ 6af0822 #10 2023-09-12 16:32:26 ~6 min ios 📱ipa 📲
✔️ 6af0822 #9 2023-09-12 16:34:57 ~8 min tests 📄log
Commit #️⃣ Finished (UTC) Duration Platform Result
✔️ 2acbdb5 #10 2023-09-13 11:48:09 ~5 min android-e2e 🤖apk 📲
✔️ 2acbdb5 #11 2023-09-13 11:52:13 ~9 min ios 📱ipa 📲
✔️ 2acbdb5 #10 2023-09-13 11:52:22 ~9 min android 🤖apk 📲
✔️ 2acbdb5 #10 2023-09-13 11:52:53 ~10 min tests 📄log
✔️ f3e0b8f #11 2023-09-13 17:33:07 ~6 min android-e2e 🤖apk 📲
✔️ f3e0b8f #11 2023-09-13 17:33:14 ~6 min android 🤖apk 📲
✔️ f3e0b8f #12 2023-09-13 17:34:42 ~7 min ios 📱ipa 📲
✔️ f3e0b8f #11 2023-09-13 17:35:53 ~8 min tests 📄log

@smohamedjavid smohamedjavid force-pushed the feature/emoji-picker branch 3 times, most recently from 1e8ed48 to 07e9d9e Compare September 5, 2023 20:09
@smohamedjavid smohamedjavid changed the title [WIP] Emoji Picker Implement Emoji Picker Sep 5, 2023
@smohamedjavid smohamedjavid marked this pull request as ready for review September 5, 2023 21:41
@flexsurfer
Copy link
Member

we can not use Section List (which is ideal for this case)

i believe @OmarBasem has implemented it with gesture lib

@OmarBasem
Copy link
Contributor

OmarBasem commented Sep 6, 2023

we can not use Section List (which is ideal for this case)

i believe @OmarBasem has implemented it with gesture lib

Yes! Thanks @flexsurfer for noticing this

@smohamedjavid we have a custom gesture section list. You can use it as gesture/section-list. API is same as rn/section-list

@smohamedjavid
Copy link
Member Author

@flexsurfer @OmarBasem - Thank you for the review. The Section List implemented in react-native.gesture ns uses the Gesture FlatList to achieve the same. It is ideal when you display the section header all the time. The search results in the Emoji Picker don’t display a section header. We can update the SectionList implementation in react-native.gesture ns. But, it may introduce bugs for screens which use it.

The Gesture FlatList which is used here inspired by that implementation to provide a few additions which help the emoji picker’s behaviour.

@OmarBasem
Copy link
Contributor

It is ideal when you display the section header all the time. The search results in the Emoji Picker don’t display a section header.

@smohamedjavid I am not sure what you mean by this. What is the UI difference between using gesture/section-list and rn/section-list ?

@smohamedjavid
Copy link
Member Author

@smohamedjavid I am not sure what you mean by this. What is the UI difference between using gesture/section-list and rn/section-list ?

I meant to say that the emoji picker design is a mix of having and not having a category/section header.
The header is removed when we display the search results.

The section-list implementation in react-native.gesture ns requires the render-section-header-fn, and for flattening the data, the title is required to mark it as a header.

(defn- flatten-sections
[sections]
(mapcat (fn [{:keys [title data]}]
(into [{:title title :header? true}] data))
sections))
(defn section-list
[{:keys [sections render-section-header-fn render-fn] :as props}]
(let [data (flatten-sections sections)]
[flat-list
(merge props
{:data data
:render-fn (fn [p1 p2 p3 p4]
(if (:header? p1)
[render-section-header-fn p1 p2 p3 p4]
[render-fn p1 p2 p3 p4]))})]))

Apologies for making you infer my response in the wrong way. Should have been more clear earlier 👍

Copy link
Contributor

@ilmotta ilmotta left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Really nice work @smohamedjavid! Especially for putting in the effort to make the component perform well on Android and iOS.

In terms of performance, one open question remained from our pairing which I'm still curious, how can we leverage caching in future PRs? Would React's memo work in our favor? For example, could we cache the entire first emoji "page" in the flat list since that's always the same? Thus striking a decent balance between trading CPU for memory.

{:ref set-scroll-ref
:scroll-enabled @scroll-enabled
:data data
:initial-num-to-render 15
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

One thing I've noticed (on Android) is that the initial-num-to-render 15 is much lower than the number of emojis displayed on screen at any given time. This causes causes switching between categories to flash for longer while flat list asynchronously process the remaining items. By flash I mean the user perceives an intermediary blank state while switching between categories for a longer time.

If we could display half of the items on screen, more or less like Telegram and others, that could help us rip the best of both worlds.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

initial-num-to-render denotes the number of items to render when the list is mounted (initial batch). This should be the minimum number to fill the screen as it preserves these items for scroll to top action.

Due to the list size, the FlatList renders half of the list by default.
Screenshot 2023-09-11 at 4 05 38 PM

I have increased the amount of items to render at each batch, which in turn produces a better fill rate when scrolling through the list. 👍

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the correction :)

Copy link
Contributor

@J-Son89 J-Son89 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nice @smohamedjavid 😎

@smohamedjavid
Copy link
Member Author

In terms of performance, one open question remained from our pairing which I'm still curious, how can we leverage caching in future PRs? Would React's memo work in our favor? For example, could we cache the entire first emoji "page" in the flat list since that's always the same? Thus striking a decent balance between trading CPU for memory.

As far as I know, we can't cache React component render. Once the sheet is closed, the component is unmounted and It's cleared from the memory. I need to do a RnD on these improvements. Also, the React memo is not needed here as the items are static and we have a unique key for each emoji to prevent re-rendering.

However, I have increased the initial batch number to have a good experience with scrolling to the top of the list.

@@ -45,7 +45,9 @@
:type :grey
:background :photo
:icon-only? true
:on-press #(js/alert "pressed")
:on-press #(rf/dispatch [:emoji-picker/open
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🥳

@status-im-auto
Copy link
Member

77% of end-end tests have passed

Total executed tests: 43
Failed tests: 10
Passed tests: 33
IDs of failed tests: 702732,702957,703086,702894,702783,703503,702786,702731,702808,702958 

Failed tests (10)

Click to expand
  • Rerun failed tests

  • Class TestActivityMultipleDevicePRTwo:

    1. test_activity_center_mentions, id: 702957

    Device 2: Looking for chat: 'general'
    Device 2: Click until `ChatMessageInput` by `accessibility id`: `chat-message-input` will be presented

    Test setup failed: medium/test_activity_center.py:326: in prepare_devices
        self.channel_2.chat_message_input.wait_for_visibility_of_element(20)
    ../views/base_element.py:139: in wait_for_visibility_of_element
        raise TimeoutException(
     Device 2: ChatMessageInput by accessibility id:`chat-message-input` is not found on the screen after wait_for_visibility_of_element
    



    Device sessions

    2. test_activity_center_admin_notification_accept_swipe, id: 702958

    Test setup failed: medium/test_activity_center.py:326: in prepare_devices
        self.channel_2.chat_message_input.wait_for_visibility_of_element(20)
    ../views/base_element.py:139: in wait_for_visibility_of_element
        raise TimeoutException(
     Device 2: ChatMessageInput by accessibility id:`chat-message-input` is not found on the screen after wait_for_visibility_of_element
    



    Class TestOneToOneChatMultipleSharedDevicesNewUiTwo:

    1. test_1_1_chat_is_shown_message_sent_delivered_from_offline, id: 702783

    Device 2: Find Text by xpath: //*[starts-with(@text,'test message')]/ancestor::android.view.ViewGroup[@content-desc='chat-item']//*[@content-desc='message-status']/android.widget.TextView
    Device 2: Text is Sent

    critical/chats/test_1_1_public_chats.py:1416: in test_1_1_chat_is_shown_message_sent_delivered_from_offline
        self.errors.verify_no_errors()
    base_test_case.py:191: in verify_no_errors
        pytest.fail('\n '.join([self.errors.pop(0) for _ in range(len(self.errors))]))
     Message status was not changed to Delivered, it's Sent after back up online!
    



    Device sessions

    Class TestOneToOneChatMultipleSharedDevicesNewUi:

    1. test_1_1_chat_pin_messages, id: 702731

    Test is not run, e2e blocker  
    

    [[reason: [NOTRUN] Pin feature is in development]]

    Class TestCommunityOneDeviceMerged:

    1. test_community_discovery, id: 703503

    Test is not run, e2e blocker  
    

    [[reason: [NOTRUN] https://github.com//issues/17175]]

    Class TestCommunityMultipleDeviceMerged:

    1. test_community_mark_all_messages_as_read, id: 703086

    Device 1: Tap on found: Button
    Device 1: Click until Text by accessibility id: community-description-text will be presented

    critical/test_public_chat_browsing.py:1028: in test_community_mark_all_messages_as_read
        self.errors.verify_no_errors()
    base_test_case.py:191: in verify_no_errors
        pytest.fail('\n '.join([self.errors.pop(0) for _ in range(len(self.errors))]))
     New messages counter is not shown in home > Community element
    E    New messages counter is not shown in community channel element
    



    Device sessions

    2. test_community_contact_block_unblock_offline, id: 702894

    Device 1: Click until ChatMessageInput by accessibility id: chat-message-input will be presented
    Device 1: Looking for a message by text: Hurray! unblocked

    critical/test_public_chat_browsing.py:967: in test_community_contact_block_unblock_offline
        self.errors.verify_no_errors()
    base_test_case.py:191: in verify_no_errors
        pytest.fail('\n '.join([self.errors.pop(0) for _ in range(len(self.errors))]))
     Hurray! unblocked was not received in public chat after user unblock!
    



    Device sessions

    Class TestCommunityMultipleDeviceMergedTwo:

    1. test_community_mentions_push_notification, id: 702786

    # STEP: Invited member gets push notification with the mention and tap it
    Device 2: Getting PN by 'user_2'

    critical/test_public_chat_browsing.py:1149: in test_community_mentions_push_notification
        self.errors.verify_no_errors()
    base_test_case.py:191: in verify_no_errors
        pytest.fail('\n '.join([self.errors.pop(0) for _ in range(len(self.errors))]))
     Push notification with the mention was not received by admin
    E    Can not edit a message with a mention
    E    Push notification with the mention was not received by the invited member 
    

    [[Issue with username in PN, issue #6 in 15500]]

    Device sessions

    Class TestGroupChatMultipleDeviceMergedNewUI:

    1. test_group_chat_pin_messages, id: 702732

    Test is not run, e2e blocker  
    

    [[reason: [NOTRUN] Pin feature is in development]]

    2. test_group_chat_offline_pn, id: 702808

    Device 3: Looking for a message by text: message from old member
    Device 3: Looking for a message by text: message from new member

    critical/chats/test_group_chat.py:442: in test_group_chat_offline_pn
        self.errors.verify_no_errors()
    base_test_case.py:191: in verify_no_errors
        pytest.fail('\n '.join([self.errors.pop(0) for _ in range(len(self.errors))]))
     Messages PN was not fetched from offline
    



    Device sessions

    Passed tests (33)

    Click to expand

    Class TestActivityCenterContactRequestMultipleDevicePR:

    1. test_activity_center_contact_request_accept_swipe_mark_all_as_read, id: 702851
    Device sessions

    2. test_activity_center_contact_request_decline, id: 702850
    Device sessions

    Class TestOneToOneChatMultipleSharedDevicesNewUiTwo:

    1. test_1_1_chat_delete_via_long_press_relogin, id: 702784
    Device sessions

    2. test_1_1_chat_mute_chat, id: 703496
    Device sessions

    Class TestOneToOneChatMultipleSharedDevicesNewUi:

    1. test_1_1_chat_emoji_send_reply_and_open_link, id: 702782
    Device sessions

    2. test_1_1_chat_text_message_delete_push_disappear, id: 702733
    Device sessions

    3. test_1_1_chat_push_emoji, id: 702813
    Device sessions

    4. test_1_1_chat_non_latin_messages_stack_update_profile_photo, id: 702745
    Device sessions

    5. test_1_1_chat_edit_message, id: 702855
    Device sessions

    6. test_1_1_chat_send_image_save_and_share, id: 703391
    Device sessions

    7. test_1_1_chat_message_reaction, id: 702730
    Device sessions

    Class TestCommunityOneDeviceMerged:

    1. test_restore_multiaccount_with_waku_backup_remove_switch, id: 703133
    Device sessions

    2. test_community_copy_and_paste_message_in_chat_input, id: 702742
    Device sessions

    3. test_community_undo_delete_message, id: 702869
    Device sessions

    4. test_community_navigate_to_channel_when_relaunch, id: 702846
    Device sessions

    5. test_community_mute_community_and_channel, id: 703382
    Device sessions

    Class TestCommunityMultipleDeviceMerged:

    1. test_community_several_images_send_reply, id: 703194
    Device sessions

    2. test_community_one_image_send_reply, id: 702859
    Device sessions

    3. test_community_emoji_send_copy_paste_reply, id: 702840
    Device sessions

    4. test_community_message_delete, id: 702839
    Device sessions

    5. test_community_message_send_check_timestamps_sender_username, id: 702838
    Device sessions

    6. test_community_links_with_previews_github_youtube_twitter_gif_send_enable, id: 702844
    Device sessions

    7. test_community_message_edit, id: 702843
    Device sessions

    8. test_community_unread_messages_badge, id: 702841
    Device sessions

    Class TestActivityMultipleDevicePR:

    1. test_navigation_jump_to, id: 702936
    Device sessions

    2. test_activity_center_reply_read_unread_delete_filter_swipe, id: 702947
    Device sessions

    Class TestCommunityMultipleDeviceMergedTwo:

    1. test_community_markdown_support, id: 702809
    Device sessions

    2. test_community_hashtag_links_to_community_channels, id: 702948
    Device sessions

    3. test_community_leave, id: 702845
    Device sessions

    Class TestGroupChatMultipleDeviceMergedNewUI:

    1. test_group_chat_mute_chat, id: 703495
    Device sessions

    2. test_group_chat_send_image_save_and_share, id: 703297
    Device sessions

    3. test_group_chat_reactions, id: 703202
    Device sessions

    4. test_group_chat_join_send_text_messages_push, id: 702807
    Device sessions

    Copy link
    Contributor

    @OmarBasem OmarBasem left a comment

    Choose a reason for hiding this comment

    The reason will be displayed to describe this comment to others. Learn more.

    Nice work!

    @smohamedjavid smohamedjavid force-pushed the feature/emoji-picker branch 2 times, most recently from 6af0822 to 2acbdb5 Compare September 13, 2023 11:42
    Copy link

    @Francesca-G Francesca-G left a comment

    Choose a reason for hiding this comment

    The reason will be displayed to describe this comment to others. Learn more.

    Looks good ✨

    Adding follow up for the ice cream icon at the bottom that needs to be updated, other than this it all looks nice :)

    (implementation: top, design: bottom)

    Screenshot 2023-09-13 alle 14 27 12 Screenshot 2023-09-13 alle 14 29 05

    @smohamedjavid
    Copy link
    Member Author

    @Francesca-G - Thanks for the review. Duly appreciated.

    Regarding the ice cream icon. It's named as food in the file.

    and when I tried to locate it on the Iconset design file, I could see the food icon is a popsicle (🍭) instead of an Ice cream in a bowl (🍨). The implementation is based on the Iconset.

    Iconset: https://www.figma.com/file/qLLuMLfpGxK9OfpIavwsmK/Iconset?type=design&node-id=3239%3A987&mode=design&t=eReFKRlS316vlM5v-1

    Kindly advise.

    @Francesca-G
    Copy link

    @smohamedjavid Filipe kindly confirmed we need to use the one in the iconset (popsicle):

    so no ice cream bowl.
    Thank you for spotting the design discrepancy :)

    @smohamedjavid
    Copy link
    Member Author

    No worries, and thanks a lot for the clarification, @Francesca-G.

    I believe we can remove the followup required label for this PR.

    @smohamedjavid smohamedjavid merged commit 0003800 into develop Sep 13, 2023
    @smohamedjavid smohamedjavid deleted the feature/emoji-picker branch September 13, 2023 17:38
    Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
    Labels
    feature feature requests
    Projects
    Archived in project
    Archived in project
    Development

    Successfully merging this pull request may close these issues.

    Wallet: Emoji picker
    8 participants