@liveblocks/react-ui 1.12.0-initial1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +65 -0
- package/dist/components/Comment.js +473 -0
- package/dist/components/Comment.js.map +1 -0
- package/dist/components/Comment.mjs +447 -0
- package/dist/components/Comment.mjs.map +1 -0
- package/dist/components/Composer.js +299 -0
- package/dist/components/Composer.js.map +1 -0
- package/dist/components/Composer.mjs +297 -0
- package/dist/components/Composer.mjs.map +1 -0
- package/dist/components/InboxNotification.js +374 -0
- package/dist/components/InboxNotification.js.map +1 -0
- package/dist/components/InboxNotification.mjs +372 -0
- package/dist/components/InboxNotification.mjs.map +1 -0
- package/dist/components/InboxNotificationList.js +19 -0
- package/dist/components/InboxNotificationList.js.map +1 -0
- package/dist/components/InboxNotificationList.mjs +17 -0
- package/dist/components/InboxNotificationList.mjs.map +1 -0
- package/dist/components/Thread.js +188 -0
- package/dist/components/Thread.js.map +1 -0
- package/dist/components/Thread.mjs +167 -0
- package/dist/components/Thread.mjs.map +1 -0
- package/dist/components/internal/Attribution.js +26 -0
- package/dist/components/internal/Attribution.js.map +1 -0
- package/dist/components/internal/Attribution.mjs +24 -0
- package/dist/components/internal/Attribution.mjs.map +1 -0
- package/dist/components/internal/Avatar.js +40 -0
- package/dist/components/internal/Avatar.js.map +1 -0
- package/dist/components/internal/Avatar.mjs +38 -0
- package/dist/components/internal/Avatar.mjs.map +1 -0
- package/dist/components/internal/Button.js +24 -0
- package/dist/components/internal/Button.js.map +1 -0
- package/dist/components/internal/Button.mjs +22 -0
- package/dist/components/internal/Button.mjs.map +1 -0
- package/dist/components/internal/Dropdown.js +72 -0
- package/dist/components/internal/Dropdown.js.map +1 -0
- package/dist/components/internal/Dropdown.mjs +47 -0
- package/dist/components/internal/Dropdown.mjs.map +1 -0
- package/dist/components/internal/Emoji.js +18 -0
- package/dist/components/internal/Emoji.js.map +1 -0
- package/dist/components/internal/Emoji.mjs +16 -0
- package/dist/components/internal/Emoji.mjs.map +1 -0
- package/dist/components/internal/EmojiPicker.js +186 -0
- package/dist/components/internal/EmojiPicker.js.map +1 -0
- package/dist/components/internal/EmojiPicker.mjs +162 -0
- package/dist/components/internal/EmojiPicker.mjs.map +1 -0
- package/dist/components/internal/Icon.js +28 -0
- package/dist/components/internal/Icon.js.map +1 -0
- package/dist/components/internal/Icon.mjs +24 -0
- package/dist/components/internal/Icon.mjs.map +1 -0
- package/dist/components/internal/InboxNotificationThread.js +116 -0
- package/dist/components/internal/InboxNotificationThread.js.map +1 -0
- package/dist/components/internal/InboxNotificationThread.mjs +112 -0
- package/dist/components/internal/InboxNotificationThread.mjs.map +1 -0
- package/dist/components/internal/List.js +34 -0
- package/dist/components/internal/List.js.map +1 -0
- package/dist/components/internal/List.mjs +32 -0
- package/dist/components/internal/List.mjs.map +1 -0
- package/dist/components/internal/Room.js +22 -0
- package/dist/components/internal/Room.js.map +1 -0
- package/dist/components/internal/Room.mjs +20 -0
- package/dist/components/internal/Room.mjs.map +1 -0
- package/dist/components/internal/Tooltip.js +91 -0
- package/dist/components/internal/Tooltip.js.map +1 -0
- package/dist/components/internal/Tooltip.mjs +65 -0
- package/dist/components/internal/Tooltip.mjs.map +1 -0
- package/dist/components/internal/User.js +41 -0
- package/dist/components/internal/User.js.map +1 -0
- package/dist/components/internal/User.mjs +39 -0
- package/dist/components/internal/User.mjs.map +1 -0
- package/dist/components.js +64 -0
- package/dist/components.js.map +1 -0
- package/dist/components.mjs +41 -0
- package/dist/components.mjs.map +1 -0
- package/dist/config.js +33 -0
- package/dist/config.js.map +1 -0
- package/dist/config.mjs +30 -0
- package/dist/config.mjs.map +1 -0
- package/dist/constants.js +10 -0
- package/dist/constants.js.map +1 -0
- package/dist/constants.mjs +6 -0
- package/dist/constants.mjs.map +1 -0
- package/dist/icons/ArrowDown.js +15 -0
- package/dist/icons/ArrowDown.js.map +1 -0
- package/dist/icons/ArrowDown.mjs +13 -0
- package/dist/icons/ArrowDown.mjs.map +1 -0
- package/dist/icons/Check.js +15 -0
- package/dist/icons/Check.js.map +1 -0
- package/dist/icons/Check.mjs +13 -0
- package/dist/icons/Check.mjs.map +1 -0
- package/dist/icons/Cross.js +17 -0
- package/dist/icons/Cross.js.map +1 -0
- package/dist/icons/Cross.mjs +15 -0
- package/dist/icons/Cross.mjs.map +1 -0
- package/dist/icons/Delete.js +15 -0
- package/dist/icons/Delete.js.map +1 -0
- package/dist/icons/Delete.mjs +13 -0
- package/dist/icons/Delete.mjs.map +1 -0
- package/dist/icons/Edit.js +15 -0
- package/dist/icons/Edit.js.map +1 -0
- package/dist/icons/Edit.mjs +13 -0
- package/dist/icons/Edit.mjs.map +1 -0
- package/dist/icons/Ellipsis.js +26 -0
- package/dist/icons/Ellipsis.js.map +1 -0
- package/dist/icons/Ellipsis.mjs +24 -0
- package/dist/icons/Ellipsis.mjs.map +1 -0
- package/dist/icons/Emoji.js +27 -0
- package/dist/icons/Emoji.js.map +1 -0
- package/dist/icons/Emoji.mjs +25 -0
- package/dist/icons/Emoji.mjs.map +1 -0
- package/dist/icons/EmojiAdd.js +29 -0
- package/dist/icons/EmojiAdd.js.map +1 -0
- package/dist/icons/EmojiAdd.mjs +27 -0
- package/dist/icons/EmojiAdd.mjs.map +1 -0
- package/dist/icons/Mention.js +17 -0
- package/dist/icons/Mention.js.map +1 -0
- package/dist/icons/Mention.mjs +15 -0
- package/dist/icons/Mention.mjs.map +1 -0
- package/dist/icons/Missing.js +19 -0
- package/dist/icons/Missing.js.map +1 -0
- package/dist/icons/Missing.mjs +17 -0
- package/dist/icons/Missing.mjs.map +1 -0
- package/dist/icons/Resolve.js +19 -0
- package/dist/icons/Resolve.js.map +1 -0
- package/dist/icons/Resolve.mjs +17 -0
- package/dist/icons/Resolve.mjs.map +1 -0
- package/dist/icons/Resolved.js +21 -0
- package/dist/icons/Resolved.js.map +1 -0
- package/dist/icons/Resolved.mjs +19 -0
- package/dist/icons/Resolved.mjs.map +1 -0
- package/dist/icons/Search.js +15 -0
- package/dist/icons/Search.js.map +1 -0
- package/dist/icons/Search.mjs +13 -0
- package/dist/icons/Search.mjs.map +1 -0
- package/dist/icons/Send.js +15 -0
- package/dist/icons/Send.js.map +1 -0
- package/dist/icons/Send.mjs +13 -0
- package/dist/icons/Send.mjs.map +1 -0
- package/dist/icons/Spinner.js +22 -0
- package/dist/icons/Spinner.js.map +1 -0
- package/dist/icons/Spinner.mjs +20 -0
- package/dist/icons/Spinner.mjs.map +1 -0
- package/dist/index.d.mts +476 -0
- package/dist/index.d.ts +476 -0
- package/dist/index.js +20 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +11 -0
- package/dist/index.mjs.map +1 -0
- package/dist/overrides.js +100 -0
- package/dist/overrides.js.map +1 -0
- package/dist/overrides.mjs +77 -0
- package/dist/overrides.mjs.map +1 -0
- package/dist/primitives/Comment/index.js +113 -0
- package/dist/primitives/Comment/index.js.map +1 -0
- package/dist/primitives/Comment/index.mjs +109 -0
- package/dist/primitives/Comment/index.mjs.map +1 -0
- package/dist/primitives/Comment/utils.js +25 -0
- package/dist/primitives/Comment/utils.js.map +1 -0
- package/dist/primitives/Comment/utils.mjs +20 -0
- package/dist/primitives/Comment/utils.mjs.map +1 -0
- package/dist/primitives/Composer/contexts.js +34 -0
- package/dist/primitives/Composer/contexts.js.map +1 -0
- package/dist/primitives/Composer/contexts.mjs +27 -0
- package/dist/primitives/Composer/contexts.mjs.map +1 -0
- package/dist/primitives/Composer/index.js +771 -0
- package/dist/primitives/Composer/index.js.map +1 -0
- package/dist/primitives/Composer/index.mjs +762 -0
- package/dist/primitives/Composer/index.mjs.map +1 -0
- package/dist/primitives/Composer/utils.js +102 -0
- package/dist/primitives/Composer/utils.js.map +1 -0
- package/dist/primitives/Composer/utils.mjs +93 -0
- package/dist/primitives/Composer/utils.mjs.map +1 -0
- package/dist/primitives/EmojiPicker/contexts.js +19 -0
- package/dist/primitives/EmojiPicker/contexts.js.map +1 -0
- package/dist/primitives/EmojiPicker/contexts.mjs +16 -0
- package/dist/primitives/EmojiPicker/contexts.mjs.map +1 -0
- package/dist/primitives/EmojiPicker/index.js +434 -0
- package/dist/primitives/EmojiPicker/index.js.map +1 -0
- package/dist/primitives/EmojiPicker/index.mjs +430 -0
- package/dist/primitives/EmojiPicker/index.mjs.map +1 -0
- package/dist/primitives/EmojiPicker/utils.js +329 -0
- package/dist/primitives/EmojiPicker/utils.js.map +1 -0
- package/dist/primitives/EmojiPicker/utils.mjs +325 -0
- package/dist/primitives/EmojiPicker/utils.mjs.map +1 -0
- package/dist/primitives/Timestamp.js +121 -0
- package/dist/primitives/Timestamp.js.map +1 -0
- package/dist/primitives/Timestamp.mjs +119 -0
- package/dist/primitives/Timestamp.mjs.map +1 -0
- package/dist/primitives/index.d.mts +502 -0
- package/dist/primitives/index.d.ts +502 -0
- package/dist/primitives/index.js +16 -0
- package/dist/primitives/index.js.map +1 -0
- package/dist/primitives/index.mjs +9 -0
- package/dist/primitives/index.mjs.map +1 -0
- package/dist/primitives/internal/Emoji.js +30 -0
- package/dist/primitives/internal/Emoji.js.map +1 -0
- package/dist/primitives/internal/Emoji.mjs +28 -0
- package/dist/primitives/internal/Emoji.mjs.map +1 -0
- package/dist/shared.js +21 -0
- package/dist/shared.js.map +1 -0
- package/dist/shared.mjs +19 -0
- package/dist/shared.mjs.map +1 -0
- package/dist/slate/plugins/auto-formatting.js +83 -0
- package/dist/slate/plugins/auto-formatting.js.map +1 -0
- package/dist/slate/plugins/auto-formatting.mjs +81 -0
- package/dist/slate/plugins/auto-formatting.mjs.map +1 -0
- package/dist/slate/plugins/auto-links.js +175 -0
- package/dist/slate/plugins/auto-links.js.map +1 -0
- package/dist/slate/plugins/auto-links.mjs +172 -0
- package/dist/slate/plugins/auto-links.mjs.map +1 -0
- package/dist/slate/plugins/empty-clear-formatting.js +18 -0
- package/dist/slate/plugins/empty-clear-formatting.js.map +1 -0
- package/dist/slate/plugins/empty-clear-formatting.mjs +16 -0
- package/dist/slate/plugins/empty-clear-formatting.mjs.map +1 -0
- package/dist/slate/plugins/mentions.js +112 -0
- package/dist/slate/plugins/mentions.js.map +1 -0
- package/dist/slate/plugins/mentions.mjs +105 -0
- package/dist/slate/plugins/mentions.mjs.map +1 -0
- package/dist/slate/utils/get-character.js +50 -0
- package/dist/slate/utils/get-character.js.map +1 -0
- package/dist/slate/utils/get-character.mjs +47 -0
- package/dist/slate/utils/get-character.mjs.map +1 -0
- package/dist/slate/utils/get-dom-range.js +17 -0
- package/dist/slate/utils/get-dom-range.js.map +1 -0
- package/dist/slate/utils/get-dom-range.mjs +15 -0
- package/dist/slate/utils/get-dom-range.mjs.map +1 -0
- package/dist/slate/utils/get-match-range.js +53 -0
- package/dist/slate/utils/get-match-range.js.map +1 -0
- package/dist/slate/utils/get-match-range.mjs +51 -0
- package/dist/slate/utils/get-match-range.mjs.map +1 -0
- package/dist/slate/utils/is-empty-string.js +8 -0
- package/dist/slate/utils/is-empty-string.js.map +1 -0
- package/dist/slate/utils/is-empty-string.mjs +6 -0
- package/dist/slate/utils/is-empty-string.mjs.map +1 -0
- package/dist/slate/utils/is-empty.js +10 -0
- package/dist/slate/utils/is-empty.js.map +1 -0
- package/dist/slate/utils/is-empty.mjs +8 -0
- package/dist/slate/utils/is-empty.mjs.map +1 -0
- package/dist/slate/utils/is-selection-collapsed.js +10 -0
- package/dist/slate/utils/is-selection-collapsed.js.map +1 -0
- package/dist/slate/utils/is-selection-collapsed.mjs +8 -0
- package/dist/slate/utils/is-selection-collapsed.mjs.map +1 -0
- package/dist/slate/utils/marks.js +43 -0
- package/dist/slate/utils/marks.js.map +1 -0
- package/dist/slate/utils/marks.mjs +38 -0
- package/dist/slate/utils/marks.mjs.map +1 -0
- package/dist/utils/Persist.js +85 -0
- package/dist/utils/Persist.js.map +1 -0
- package/dist/utils/Persist.mjs +81 -0
- package/dist/utils/Persist.mjs.map +1 -0
- package/dist/utils/Portal.js +27 -0
- package/dist/utils/Portal.js.map +1 -0
- package/dist/utils/Portal.mjs +25 -0
- package/dist/utils/Portal.mjs.map +1 -0
- package/dist/utils/capitalize.js +8 -0
- package/dist/utils/capitalize.js.map +1 -0
- package/dist/utils/capitalize.mjs +6 -0
- package/dist/utils/capitalize.mjs.map +1 -0
- package/dist/utils/chunk.js +12 -0
- package/dist/utils/chunk.js.map +1 -0
- package/dist/utils/chunk.mjs +10 -0
- package/dist/utils/chunk.mjs.map +1 -0
- package/dist/utils/clamp.js +8 -0
- package/dist/utils/clamp.js.map +1 -0
- package/dist/utils/clamp.mjs +6 -0
- package/dist/utils/clamp.mjs.map +1 -0
- package/dist/utils/class-names.js +8 -0
- package/dist/utils/class-names.js.map +1 -0
- package/dist/utils/class-names.mjs +6 -0
- package/dist/utils/class-names.mjs.map +1 -0
- package/dist/utils/exists.js +8 -0
- package/dist/utils/exists.js.map +1 -0
- package/dist/utils/exists.mjs +6 -0
- package/dist/utils/exists.mjs.map +1 -0
- package/dist/utils/find-last-index.js +15 -0
- package/dist/utils/find-last-index.js.map +1 -0
- package/dist/utils/find-last-index.mjs +13 -0
- package/dist/utils/find-last-index.mjs.map +1 -0
- package/dist/utils/flush-sync.js +12 -0
- package/dist/utils/flush-sync.js.map +1 -0
- package/dist/utils/flush-sync.mjs +10 -0
- package/dist/utils/flush-sync.mjs.map +1 -0
- package/dist/utils/get-initials.js +13 -0
- package/dist/utils/get-initials.js.map +1 -0
- package/dist/utils/get-initials.mjs +11 -0
- package/dist/utils/get-initials.mjs.map +1 -0
- package/dist/utils/intl.js +24 -0
- package/dist/utils/intl.js.map +1 -0
- package/dist/utils/intl.mjs +20 -0
- package/dist/utils/intl.mjs.map +1 -0
- package/dist/utils/is-apple.js +9 -0
- package/dist/utils/is-apple.js.map +1 -0
- package/dist/utils/is-apple.mjs +7 -0
- package/dist/utils/is-apple.mjs.map +1 -0
- package/dist/utils/is-key.js +26 -0
- package/dist/utils/is-key.js.map +1 -0
- package/dist/utils/is-key.mjs +24 -0
- package/dist/utils/is-key.mjs.map +1 -0
- package/dist/utils/memoize.js +19 -0
- package/dist/utils/memoize.js.map +1 -0
- package/dist/utils/memoize.mjs +17 -0
- package/dist/utils/memoize.mjs.map +1 -0
- package/dist/utils/pluralize.js +8 -0
- package/dist/utils/pluralize.js.map +1 -0
- package/dist/utils/pluralize.mjs +6 -0
- package/dist/utils/pluralize.mjs.map +1 -0
- package/dist/utils/request-idle-callback.js +15 -0
- package/dist/utils/request-idle-callback.js.map +1 -0
- package/dist/utils/request-idle-callback.mjs +12 -0
- package/dist/utils/request-idle-callback.mjs.map +1 -0
- package/dist/utils/request-submit.js +20 -0
- package/dist/utils/request-submit.js.map +1 -0
- package/dist/utils/request-submit.mjs +18 -0
- package/dist/utils/request-submit.mjs.map +1 -0
- package/dist/utils/url.js +22 -0
- package/dist/utils/url.js.map +1 -0
- package/dist/utils/url.mjs +20 -0
- package/dist/utils/url.mjs.map +1 -0
- package/dist/utils/use-controllable-state.js +34 -0
- package/dist/utils/use-controllable-state.js.map +1 -0
- package/dist/utils/use-controllable-state.mjs +32 -0
- package/dist/utils/use-controllable-state.mjs.map +1 -0
- package/dist/utils/use-id.js +29 -0
- package/dist/utils/use-id.js.map +1 -0
- package/dist/utils/use-id.mjs +27 -0
- package/dist/utils/use-id.mjs.map +1 -0
- package/dist/utils/use-index.js +32 -0
- package/dist/utils/use-index.js.map +1 -0
- package/dist/utils/use-index.mjs +30 -0
- package/dist/utils/use-index.mjs.map +1 -0
- package/dist/utils/use-initial.js +10 -0
- package/dist/utils/use-initial.js.map +1 -0
- package/dist/utils/use-initial.mjs +8 -0
- package/dist/utils/use-initial.mjs.map +1 -0
- package/dist/utils/use-interval.js +24 -0
- package/dist/utils/use-interval.js.map +1 -0
- package/dist/utils/use-interval.mjs +22 -0
- package/dist/utils/use-interval.mjs.map +1 -0
- package/dist/utils/use-latest.js +14 -0
- package/dist/utils/use-latest.js.map +1 -0
- package/dist/utils/use-latest.mjs +12 -0
- package/dist/utils/use-latest.mjs.map +1 -0
- package/dist/utils/use-layout-effect.js +8 -0
- package/dist/utils/use-layout-effect.js.map +1 -0
- package/dist/utils/use-layout-effect.mjs +6 -0
- package/dist/utils/use-layout-effect.mjs.map +1 -0
- package/dist/utils/use-refs.js +24 -0
- package/dist/utils/use-refs.js.map +1 -0
- package/dist/utils/use-refs.mjs +22 -0
- package/dist/utils/use-refs.mjs.map +1 -0
- package/dist/utils/use-rerender.js +14 -0
- package/dist/utils/use-rerender.js.map +1 -0
- package/dist/utils/use-rerender.mjs +12 -0
- package/dist/utils/use-rerender.mjs.map +1 -0
- package/dist/utils/use-transition.js +12 -0
- package/dist/utils/use-transition.js.map +1 -0
- package/dist/utils/use-transition.mjs +10 -0
- package/dist/utils/use-transition.mjs.map +1 -0
- package/dist/utils/use-visible.js +48 -0
- package/dist/utils/use-visible.js.map +1 -0
- package/dist/utils/use-visible.mjs +46 -0
- package/dist/utils/use-visible.mjs.map +1 -0
- package/dist/utils/visually-hidden.js +17 -0
- package/dist/utils/visually-hidden.js.map +1 -0
- package/dist/utils/visually-hidden.mjs +15 -0
- package/dist/utils/visually-hidden.mjs.map +1 -0
- package/dist/utils/wrap.js +9 -0
- package/dist/utils/wrap.js.map +1 -0
- package/dist/utils/wrap.mjs +7 -0
- package/dist/utils/wrap.mjs.map +1 -0
- package/dist/version.js +10 -0
- package/dist/version.js.map +1 -0
- package/dist/version.mjs +6 -0
- package/dist/version.mjs.map +1 -0
- package/package.json +142 -0
- package/primitives/README.md +5 -0
- package/primitives/package.json +4 -0
- package/src/styles/constants.css +17 -0
- package/src/styles/dark/attributes.css +8 -0
- package/src/styles/dark/index.css +29 -0
- package/src/styles/dark/media-query.css +7 -0
- package/src/styles/index.css +1437 -0
- package/src/styles/utils.css +36 -0
- package/styles/dark/attributes.css +1 -0
- package/styles/dark/attributes.css.d.ts +1 -0
- package/styles/dark/attributes.css.map +1 -0
- package/styles/dark/media-query.css +1 -0
- package/styles/dark/media-query.css.d.ts +1 -0
- package/styles/dark/media-query.css.map +1 -0
- package/styles.css +1 -0
- package/styles.css.d.ts +1 -0
- package/styles.css.map +1 -0
|
@@ -0,0 +1,434 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
var reactSlot = require('@radix-ui/react-slot');
|
|
5
|
+
var React = require('react');
|
|
6
|
+
var reactVirtuoso = require('react-virtuoso');
|
|
7
|
+
var isKey = require('../../utils/is-key.js');
|
|
8
|
+
var requestIdleCallback = require('../../utils/request-idle-callback.js');
|
|
9
|
+
var useTransition = require('../../utils/use-transition.js');
|
|
10
|
+
var visuallyHidden = require('../../utils/visually-hidden.js');
|
|
11
|
+
var Emoji = require('../internal/Emoji.js');
|
|
12
|
+
var contexts = require('./contexts.js');
|
|
13
|
+
var utils = require('./utils.js');
|
|
14
|
+
|
|
15
|
+
const DEFAULT_COLUMNS = 10;
|
|
16
|
+
const DEFAULT_LOCALE = "en";
|
|
17
|
+
const LOADING_MINIMUM_TIMEOUT = 100;
|
|
18
|
+
const EMOJIPICKER_ROOT_NAME = "EmojiPickerRoot";
|
|
19
|
+
const EMOJIPICKER_CONTENT_NAME = "EmojiPickerContent";
|
|
20
|
+
const EMOJIPICKER_SEARCH_NAME = "EmojiPickerSearch";
|
|
21
|
+
function EmojiPickerRoot({
|
|
22
|
+
columns = DEFAULT_COLUMNS,
|
|
23
|
+
locale = DEFAULT_LOCALE,
|
|
24
|
+
onEmojiSelect,
|
|
25
|
+
children
|
|
26
|
+
}) {
|
|
27
|
+
const emojiData = React.useRef();
|
|
28
|
+
const search = React.useRef("");
|
|
29
|
+
const [, startEmojisTransition] = useTransition.useTransition();
|
|
30
|
+
const [data, setData] = React.useState();
|
|
31
|
+
const [error, setError] = React.useState();
|
|
32
|
+
const [selectedColumnIndex, setSelectedColumnIndex] = React.useState(0);
|
|
33
|
+
const [selectedRowIndex, setSelectedRowIndex] = React.useState(0);
|
|
34
|
+
const [interaction, setInteraction] = React.useState("none");
|
|
35
|
+
const selectCurrentEmoji = React.useCallback(() => {
|
|
36
|
+
if (onEmojiSelect) {
|
|
37
|
+
const emoji = data?.rows[selectedRowIndex]?.[selectedColumnIndex];
|
|
38
|
+
if (emoji) {
|
|
39
|
+
onEmojiSelect(emoji.emoji);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}, [data?.rows, onEmojiSelect, selectedColumnIndex, selectedRowIndex]);
|
|
43
|
+
const resetSelection = React.useCallback(() => {
|
|
44
|
+
setSelectedColumnIndex(0);
|
|
45
|
+
setSelectedRowIndex(0);
|
|
46
|
+
}, []);
|
|
47
|
+
const setPointerSelection = React.useCallback(
|
|
48
|
+
(columnIndex, rowIndex) => {
|
|
49
|
+
setInteraction("pointer");
|
|
50
|
+
setSelectedColumnIndex(columnIndex);
|
|
51
|
+
setSelectedRowIndex(rowIndex);
|
|
52
|
+
},
|
|
53
|
+
[]
|
|
54
|
+
);
|
|
55
|
+
const moveSelection = React.useCallback(
|
|
56
|
+
(direction, event) => {
|
|
57
|
+
if (!data) {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
event.preventDefault();
|
|
61
|
+
if (interaction === "none") {
|
|
62
|
+
setInteraction("keyboard");
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
setInteraction("keyboard");
|
|
66
|
+
switch (direction) {
|
|
67
|
+
case "left": {
|
|
68
|
+
if (selectedColumnIndex === 0) {
|
|
69
|
+
const previousRowIndex = selectedRowIndex - 1;
|
|
70
|
+
const previousRow = data.rows[previousRowIndex];
|
|
71
|
+
if (previousRow) {
|
|
72
|
+
setSelectedRowIndex(previousRowIndex);
|
|
73
|
+
setSelectedColumnIndex(previousRow.length - 1);
|
|
74
|
+
}
|
|
75
|
+
} else {
|
|
76
|
+
setSelectedColumnIndex(selectedColumnIndex - 1);
|
|
77
|
+
}
|
|
78
|
+
break;
|
|
79
|
+
}
|
|
80
|
+
case "right": {
|
|
81
|
+
const currentRow = data.rows[selectedRowIndex];
|
|
82
|
+
if (selectedColumnIndex === currentRow.length - 1) {
|
|
83
|
+
const nextRowIndex = selectedRowIndex + 1;
|
|
84
|
+
const nextRow = data.rows[nextRowIndex];
|
|
85
|
+
if (nextRow) {
|
|
86
|
+
setSelectedRowIndex(nextRowIndex);
|
|
87
|
+
setSelectedColumnIndex(0);
|
|
88
|
+
}
|
|
89
|
+
} else {
|
|
90
|
+
setSelectedColumnIndex(selectedColumnIndex + 1);
|
|
91
|
+
}
|
|
92
|
+
break;
|
|
93
|
+
}
|
|
94
|
+
case "up": {
|
|
95
|
+
const previousRow = data.rows[selectedRowIndex - 1];
|
|
96
|
+
if (previousRow) {
|
|
97
|
+
setSelectedRowIndex(selectedRowIndex - 1);
|
|
98
|
+
if (!previousRow[selectedColumnIndex]) {
|
|
99
|
+
setSelectedColumnIndex(previousRow.length - 1);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
break;
|
|
103
|
+
}
|
|
104
|
+
case "down": {
|
|
105
|
+
const nextRow = data.rows[selectedRowIndex + 1];
|
|
106
|
+
if (nextRow) {
|
|
107
|
+
setSelectedRowIndex(selectedRowIndex + 1);
|
|
108
|
+
if (!nextRow[selectedColumnIndex]) {
|
|
109
|
+
setSelectedColumnIndex(nextRow.length - 1);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
break;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
},
|
|
116
|
+
[data, interaction, selectedColumnIndex, selectedRowIndex]
|
|
117
|
+
);
|
|
118
|
+
const updateEmojis = React.useCallback(() => {
|
|
119
|
+
if (!emojiData.current) {
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
startEmojisTransition(() => {
|
|
123
|
+
setData(() => {
|
|
124
|
+
if (!emojiData.current) {
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
const filteredEmojis = utils.filterEmojis(
|
|
128
|
+
emojiData.current.emojis,
|
|
129
|
+
search.current
|
|
130
|
+
);
|
|
131
|
+
return utils.generateEmojiPickerData(
|
|
132
|
+
filteredEmojis,
|
|
133
|
+
emojiData.current.categories,
|
|
134
|
+
columns
|
|
135
|
+
);
|
|
136
|
+
});
|
|
137
|
+
resetSelection();
|
|
138
|
+
});
|
|
139
|
+
}, [columns, resetSelection]);
|
|
140
|
+
const handleSearch = React.useCallback(
|
|
141
|
+
(value) => {
|
|
142
|
+
search.current = value;
|
|
143
|
+
updateEmojis();
|
|
144
|
+
},
|
|
145
|
+
[updateEmojis]
|
|
146
|
+
);
|
|
147
|
+
const initializeEmojiData = React.useCallback(
|
|
148
|
+
async (locale2) => {
|
|
149
|
+
try {
|
|
150
|
+
emojiData.current = await utils.getEmojiData(locale2);
|
|
151
|
+
updateEmojis();
|
|
152
|
+
} catch (error2) {
|
|
153
|
+
setError(error2);
|
|
154
|
+
}
|
|
155
|
+
},
|
|
156
|
+
[updateEmojis]
|
|
157
|
+
);
|
|
158
|
+
React.useEffect(() => {
|
|
159
|
+
let idleCallbackId;
|
|
160
|
+
const timeoutId = setTimeout(() => {
|
|
161
|
+
idleCallbackId = requestIdleCallback.requestIdleCallback(() => {
|
|
162
|
+
initializeEmojiData(locale);
|
|
163
|
+
});
|
|
164
|
+
}, LOADING_MINIMUM_TIMEOUT);
|
|
165
|
+
return () => {
|
|
166
|
+
clearTimeout(timeoutId);
|
|
167
|
+
requestIdleCallback.cancelIdleCallback(idleCallbackId);
|
|
168
|
+
};
|
|
169
|
+
}, [locale]);
|
|
170
|
+
React.useEffect(() => {
|
|
171
|
+
if (interaction === "none") {
|
|
172
|
+
resetSelection();
|
|
173
|
+
}
|
|
174
|
+
}, [interaction]);
|
|
175
|
+
return /* @__PURE__ */ React.createElement(contexts.EmojiPickerContext.Provider, {
|
|
176
|
+
value: {
|
|
177
|
+
data,
|
|
178
|
+
error,
|
|
179
|
+
isLoading: !data && !error,
|
|
180
|
+
columns,
|
|
181
|
+
onSearch: handleSearch,
|
|
182
|
+
onEmojiSelect,
|
|
183
|
+
selectCurrentEmoji,
|
|
184
|
+
selectedRowIndex,
|
|
185
|
+
selectedColumnIndex,
|
|
186
|
+
moveSelection,
|
|
187
|
+
setPointerSelection,
|
|
188
|
+
interaction,
|
|
189
|
+
setInteraction
|
|
190
|
+
}
|
|
191
|
+
}, children);
|
|
192
|
+
}
|
|
193
|
+
const EmojiPickerSearch = React.forwardRef(
|
|
194
|
+
({ asChild, value, defaultValue, onChange, ...props }, forwardedRef) => {
|
|
195
|
+
const Component = asChild ? reactSlot.Slot : "input";
|
|
196
|
+
const {
|
|
197
|
+
onSearch,
|
|
198
|
+
selectCurrentEmoji,
|
|
199
|
+
moveSelection,
|
|
200
|
+
interaction,
|
|
201
|
+
setInteraction
|
|
202
|
+
} = contexts.useEmojiPicker();
|
|
203
|
+
const handleChange = React.useCallback(
|
|
204
|
+
(event) => {
|
|
205
|
+
onChange?.(event);
|
|
206
|
+
if (event.isDefaultPrevented()) {
|
|
207
|
+
return;
|
|
208
|
+
}
|
|
209
|
+
const value2 = event.target.value;
|
|
210
|
+
setInteraction(value2 ? "keyboard" : "none");
|
|
211
|
+
onSearch(value2);
|
|
212
|
+
},
|
|
213
|
+
[onChange, onSearch, setInteraction]
|
|
214
|
+
);
|
|
215
|
+
const handleKeyDown = React.useCallback(
|
|
216
|
+
(event) => {
|
|
217
|
+
if (event.isDefaultPrevented()) {
|
|
218
|
+
return;
|
|
219
|
+
}
|
|
220
|
+
if (isKey.isKey(event, "ArrowLeft")) {
|
|
221
|
+
moveSelection("left", event);
|
|
222
|
+
} else if (isKey.isKey(event, "ArrowRight")) {
|
|
223
|
+
moveSelection("right", event);
|
|
224
|
+
} else if (isKey.isKey(event, "ArrowUp")) {
|
|
225
|
+
moveSelection("up", event);
|
|
226
|
+
} else if (isKey.isKey(event, "ArrowDown")) {
|
|
227
|
+
moveSelection("down", event);
|
|
228
|
+
} else if (isKey.isKey(event, "Enter")) {
|
|
229
|
+
if (interaction !== "none") {
|
|
230
|
+
event.preventDefault();
|
|
231
|
+
selectCurrentEmoji();
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
},
|
|
235
|
+
[interaction, moveSelection, selectCurrentEmoji]
|
|
236
|
+
);
|
|
237
|
+
React.useEffect(() => {
|
|
238
|
+
onSearch(
|
|
239
|
+
value ? String(value) : defaultValue ? String(defaultValue) : ""
|
|
240
|
+
);
|
|
241
|
+
}, []);
|
|
242
|
+
return /* @__PURE__ */ React.createElement(Component, {
|
|
243
|
+
type: "search",
|
|
244
|
+
value,
|
|
245
|
+
defaultValue,
|
|
246
|
+
onChange: handleChange,
|
|
247
|
+
onKeyDown: handleKeyDown,
|
|
248
|
+
...props,
|
|
249
|
+
ref: forwardedRef
|
|
250
|
+
});
|
|
251
|
+
}
|
|
252
|
+
);
|
|
253
|
+
const defaultContentComponents = {
|
|
254
|
+
CategoryHeader: ({ category, ...props }) => /* @__PURE__ */ React.createElement("div", {
|
|
255
|
+
...props
|
|
256
|
+
}, category),
|
|
257
|
+
Row: ({ children, attributes, ...props }) => /* @__PURE__ */ React.createElement("div", {
|
|
258
|
+
...props
|
|
259
|
+
}, children),
|
|
260
|
+
Emoji: ({ emoji, ...props }) => /* @__PURE__ */ React.createElement("button", {
|
|
261
|
+
...props
|
|
262
|
+
}, /* @__PURE__ */ React.createElement(Emoji.Emoji, {
|
|
263
|
+
emoji
|
|
264
|
+
})),
|
|
265
|
+
Loading: (props) => /* @__PURE__ */ React.createElement("div", {
|
|
266
|
+
...props
|
|
267
|
+
}),
|
|
268
|
+
Empty: (props) => /* @__PURE__ */ React.createElement("div", {
|
|
269
|
+
...props
|
|
270
|
+
}),
|
|
271
|
+
Grid: (props) => /* @__PURE__ */ React.createElement("div", {
|
|
272
|
+
...props
|
|
273
|
+
}),
|
|
274
|
+
Error: ({ error, ...props }) => /* @__PURE__ */ React.createElement("div", {
|
|
275
|
+
...props
|
|
276
|
+
})
|
|
277
|
+
};
|
|
278
|
+
const placeholderRowAttributes = {
|
|
279
|
+
rowIndex: -1,
|
|
280
|
+
categoryRowIndex: -1,
|
|
281
|
+
categoryRowsCount: 0
|
|
282
|
+
};
|
|
283
|
+
const VirtuosoScroller = React.forwardRef(
|
|
284
|
+
({ children, ...props }, forwardedRef) => {
|
|
285
|
+
return /* @__PURE__ */ React.createElement("div", {
|
|
286
|
+
...props,
|
|
287
|
+
tabIndex: -1,
|
|
288
|
+
"data-test-id": void 0,
|
|
289
|
+
ref: forwardedRef
|
|
290
|
+
}, children);
|
|
291
|
+
}
|
|
292
|
+
);
|
|
293
|
+
const VirtuosoTopList = React.forwardRef(
|
|
294
|
+
({ children, ...props }, forwardedRef) => {
|
|
295
|
+
return /* @__PURE__ */ React.createElement("div", {
|
|
296
|
+
...props,
|
|
297
|
+
"data-test-id": void 0,
|
|
298
|
+
ref: forwardedRef
|
|
299
|
+
}, children);
|
|
300
|
+
}
|
|
301
|
+
);
|
|
302
|
+
const EmojiPickerContent = React.forwardRef(
|
|
303
|
+
({ components, asChild, ...props }, forwardedRef) => {
|
|
304
|
+
const Component = asChild ? reactSlot.Slot : "div";
|
|
305
|
+
const virtuosoRef = React.useRef(null);
|
|
306
|
+
const {
|
|
307
|
+
data,
|
|
308
|
+
error,
|
|
309
|
+
isLoading,
|
|
310
|
+
columns,
|
|
311
|
+
onEmojiSelect,
|
|
312
|
+
selectedColumnIndex,
|
|
313
|
+
selectedRowIndex,
|
|
314
|
+
setPointerSelection,
|
|
315
|
+
interaction,
|
|
316
|
+
setInteraction
|
|
317
|
+
} = contexts.useEmojiPicker();
|
|
318
|
+
const selectedEmoji = React.useMemo(
|
|
319
|
+
() => data?.rows[selectedRowIndex]?.[selectedColumnIndex],
|
|
320
|
+
[data?.rows, selectedColumnIndex, selectedRowIndex]
|
|
321
|
+
);
|
|
322
|
+
const { Loading, Empty, Error, CategoryHeader, Grid, Row, Emoji } = React.useMemo(
|
|
323
|
+
() => ({ ...defaultContentComponents, ...components }),
|
|
324
|
+
[components]
|
|
325
|
+
);
|
|
326
|
+
const VirtuosoList = React.useMemo(
|
|
327
|
+
() => React.forwardRef(
|
|
328
|
+
({ children, ...props2 }, forwardedRef2) => {
|
|
329
|
+
return /* @__PURE__ */ React.createElement("div", {
|
|
330
|
+
role: "grid",
|
|
331
|
+
"aria-colcount": columns,
|
|
332
|
+
...props2,
|
|
333
|
+
"data-test-id": void 0,
|
|
334
|
+
ref: forwardedRef2
|
|
335
|
+
}, children);
|
|
336
|
+
}
|
|
337
|
+
),
|
|
338
|
+
[columns]
|
|
339
|
+
);
|
|
340
|
+
const placeholderColumns = React.useMemo(
|
|
341
|
+
() => Array(columns).fill("\u{1F32B}\uFE0F"),
|
|
342
|
+
[columns]
|
|
343
|
+
);
|
|
344
|
+
const preventDefault = React.useCallback((event) => {
|
|
345
|
+
event.preventDefault();
|
|
346
|
+
}, []);
|
|
347
|
+
const handleEmojiPointerLeave = React.useCallback(() => {
|
|
348
|
+
if (interaction === "pointer") {
|
|
349
|
+
setInteraction("none");
|
|
350
|
+
}
|
|
351
|
+
}, [interaction, setInteraction]);
|
|
352
|
+
React.useEffect(() => {
|
|
353
|
+
if (interaction === "keyboard") {
|
|
354
|
+
virtuosoRef.current?.scrollIntoView({
|
|
355
|
+
index: selectedRowIndex,
|
|
356
|
+
behavior: "auto"
|
|
357
|
+
});
|
|
358
|
+
}
|
|
359
|
+
}, [interaction, selectedRowIndex]);
|
|
360
|
+
return /* @__PURE__ */ React.createElement(Component, {
|
|
361
|
+
...props,
|
|
362
|
+
ref: forwardedRef
|
|
363
|
+
}, /* @__PURE__ */ React.createElement("div", {
|
|
364
|
+
style: {
|
|
365
|
+
visibility: "hidden",
|
|
366
|
+
height: 0
|
|
367
|
+
}
|
|
368
|
+
}, /* @__PURE__ */ React.createElement(Row, {
|
|
369
|
+
attributes: placeholderRowAttributes
|
|
370
|
+
}, placeholderColumns.map((placeholder, index) => /* @__PURE__ */ React.createElement(Emoji, {
|
|
371
|
+
emoji: placeholder,
|
|
372
|
+
key: index
|
|
373
|
+
})))), isLoading ? /* @__PURE__ */ React.createElement(Loading, null) : error ? /* @__PURE__ */ React.createElement(Error, {
|
|
374
|
+
error
|
|
375
|
+
}) : data.count === 0 ? /* @__PURE__ */ React.createElement(Empty, null) : /* @__PURE__ */ React.createElement(Grid, null, /* @__PURE__ */ React.createElement(reactVirtuoso.GroupedVirtuoso, {
|
|
376
|
+
ref: virtuosoRef,
|
|
377
|
+
components: {
|
|
378
|
+
Scroller: VirtuosoScroller,
|
|
379
|
+
List: VirtuosoList,
|
|
380
|
+
TopItemList: VirtuosoTopList
|
|
381
|
+
},
|
|
382
|
+
groupCounts: data.categoriesRowCounts,
|
|
383
|
+
groupContent: (groupIndex) => {
|
|
384
|
+
return /* @__PURE__ */ React.createElement(CategoryHeader, {
|
|
385
|
+
category: data.categories[groupIndex]
|
|
386
|
+
});
|
|
387
|
+
},
|
|
388
|
+
itemContent: (rowIndex, groupIndex) => {
|
|
389
|
+
const categoryRowIndex = data.categoriesRowIndices[groupIndex].indexOf(rowIndex);
|
|
390
|
+
const categoryRowsCount = data.categoriesRowCounts[groupIndex];
|
|
391
|
+
return /* @__PURE__ */ React.createElement(Row, {
|
|
392
|
+
attributes: {
|
|
393
|
+
rowIndex,
|
|
394
|
+
categoryRowIndex,
|
|
395
|
+
categoryRowsCount
|
|
396
|
+
}
|
|
397
|
+
}, data.rows[rowIndex].map((emoji, columnIndex) => {
|
|
398
|
+
const isSelected = interaction !== "none" && selectedColumnIndex === columnIndex && selectedRowIndex === rowIndex;
|
|
399
|
+
return /* @__PURE__ */ React.createElement(Emoji, {
|
|
400
|
+
key: emoji.emoji,
|
|
401
|
+
role: "gridcell",
|
|
402
|
+
"aria-colindex": columnIndex,
|
|
403
|
+
"aria-selected": isSelected || void 0,
|
|
404
|
+
"data-selected": isSelected || void 0,
|
|
405
|
+
onMouseDown: preventDefault,
|
|
406
|
+
tabIndex: -1,
|
|
407
|
+
onPointerEnter: () => {
|
|
408
|
+
setPointerSelection(columnIndex, rowIndex);
|
|
409
|
+
},
|
|
410
|
+
onPointerLeave: handleEmojiPointerLeave,
|
|
411
|
+
onClick: (event) => {
|
|
412
|
+
onEmojiSelect?.(emoji.emoji);
|
|
413
|
+
event.stopPropagation();
|
|
414
|
+
},
|
|
415
|
+
emoji: emoji.emoji
|
|
416
|
+
});
|
|
417
|
+
}));
|
|
418
|
+
}
|
|
419
|
+
})), selectedEmoji && interaction !== "none" && /* @__PURE__ */ React.createElement("div", {
|
|
420
|
+
"aria-live": "polite",
|
|
421
|
+
style: visuallyHidden.visuallyHidden
|
|
422
|
+
}, selectedEmoji.name));
|
|
423
|
+
}
|
|
424
|
+
);
|
|
425
|
+
if (process.env.NODE_ENV !== "production") {
|
|
426
|
+
EmojiPickerRoot.displayName = EMOJIPICKER_ROOT_NAME;
|
|
427
|
+
EmojiPickerContent.displayName = EMOJIPICKER_CONTENT_NAME;
|
|
428
|
+
EmojiPickerSearch.displayName = EMOJIPICKER_SEARCH_NAME;
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
exports.Content = EmojiPickerContent;
|
|
432
|
+
exports.Root = EmojiPickerRoot;
|
|
433
|
+
exports.Search = EmojiPickerSearch;
|
|
434
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sources":["../../../src/primitives/EmojiPicker/index.tsx"],"sourcesContent":["\"use client\";\n\nimport { Slot } from \"@radix-ui/react-slot\";\nimport type { ChangeEvent, KeyboardEvent, SyntheticEvent } from \"react\";\nimport React, {\n forwardRef,\n useCallback,\n useEffect,\n useMemo,\n useRef,\n useState,\n} from \"react\";\nimport type {\n GroupedVirtuosoHandle,\n ListProps as VirtuosoListProps,\n ScrollerProps,\n TopItemListProps,\n} from \"react-virtuoso\";\nimport { GroupedVirtuoso } from \"react-virtuoso\";\n\nimport { isKey } from \"../../utils/is-key\";\nimport {\n cancelIdleCallback,\n requestIdleCallback,\n} from \"../../utils/request-idle-callback\";\nimport { useTransition } from \"../../utils/use-transition\";\nimport { visuallyHidden } from \"../../utils/visually-hidden\";\nimport { Emoji as EmojiPrimitive } from \"../internal/Emoji\";\nimport { EmojiPickerContext, useEmojiPicker } from \"./contexts\";\nimport type {\n EmojiData,\n EmojiPickerContentComponents,\n EmojiPickerContentEmojiRowAttributes,\n EmojiPickerContentProps,\n EmojiPickerData,\n EmojiPickerRootProps,\n EmojiPickerSearchProps,\n EmojiPickerSelectionDirection,\n} from \"./types\";\nimport { filterEmojis, generateEmojiPickerData, getEmojiData } from \"./utils\";\n\nconst DEFAULT_COLUMNS = 10;\nconst DEFAULT_LOCALE = \"en\";\nconst LOADING_MINIMUM_TIMEOUT = 100;\n\nconst EMOJIPICKER_ROOT_NAME = \"EmojiPickerRoot\";\nconst EMOJIPICKER_CONTENT_NAME = \"EmojiPickerContent\";\nconst EMOJIPICKER_SEARCH_NAME = \"EmojiPickerSearch\";\n\n/**\n * @private\n * The EmojiPicker primitive is undocumented for now and subject to change,\n * use at your own risk. If you have any feedback on it, please let us know!\n * See how we use it in the default components to learn how to use it:\n * https://github.com/liveblocks/liveblocks/blob/main/packages/liveblocks-react-ui/src/components/internal/EmojiPicker.tsx.\n *\n * Surrounds the emoji picker, it handles emoji data and coordinates\n * `EmojiPicker.Search` and `EmojiPicker.Content`.\n *\n * @example\n * <EmojiPicker.Root>\n * <EmojiPicker.Search />\n * <EmojiPicker.Content />\n * </EmojiPicker.Root>\n */\nfunction EmojiPickerRoot({\n columns = DEFAULT_COLUMNS,\n locale = DEFAULT_LOCALE,\n onEmojiSelect,\n children,\n}: EmojiPickerRootProps) {\n const emojiData = useRef<EmojiData>();\n const search = useRef(\"\");\n const [, startEmojisTransition] = useTransition();\n const [data, setData] = useState<EmojiPickerData>();\n const [error, setError] = useState<Error>();\n const [selectedColumnIndex, setSelectedColumnIndex] = useState(0);\n const [selectedRowIndex, setSelectedRowIndex] = useState(0);\n const [interaction, setInteraction] = useState<\n \"keyboard\" | \"pointer\" | \"none\"\n >(\"none\");\n\n const selectCurrentEmoji = useCallback(() => {\n if (onEmojiSelect) {\n const emoji = data?.rows[selectedRowIndex]?.[selectedColumnIndex];\n\n if (emoji) {\n onEmojiSelect(emoji.emoji);\n }\n }\n }, [data?.rows, onEmojiSelect, selectedColumnIndex, selectedRowIndex]);\n\n const resetSelection = useCallback(() => {\n setSelectedColumnIndex(0);\n setSelectedRowIndex(0);\n }, []);\n\n const setPointerSelection = useCallback(\n (columnIndex: number, rowIndex: number) => {\n setInteraction(\"pointer\");\n setSelectedColumnIndex(columnIndex);\n setSelectedRowIndex(rowIndex);\n },\n []\n );\n\n const moveSelection = useCallback(\n (\n direction: EmojiPickerSelectionDirection,\n event: KeyboardEvent<HTMLInputElement>\n ) => {\n if (!data) {\n return;\n }\n\n event.preventDefault();\n\n if (interaction === \"none\") {\n setInteraction(\"keyboard\");\n return;\n }\n\n setInteraction(\"keyboard\");\n\n switch (direction) {\n // If first column, move to last column of previous row (if available)\n // Otherwise, move to previous column\n case \"left\": {\n if (selectedColumnIndex === 0) {\n const previousRowIndex = selectedRowIndex - 1;\n const previousRow = data.rows[previousRowIndex];\n\n if (previousRow) {\n setSelectedRowIndex(previousRowIndex);\n setSelectedColumnIndex(previousRow.length - 1);\n }\n } else {\n setSelectedColumnIndex(selectedColumnIndex - 1);\n }\n\n break;\n }\n\n // If last column, move to first column of next row (if available)\n // Otherwise, move to next column\n case \"right\": {\n const currentRow = data.rows[selectedRowIndex];\n\n if (selectedColumnIndex === currentRow.length - 1) {\n const nextRowIndex = selectedRowIndex + 1;\n const nextRow = data.rows[nextRowIndex];\n\n if (nextRow) {\n setSelectedRowIndex(nextRowIndex);\n setSelectedColumnIndex(0);\n }\n } else {\n setSelectedColumnIndex(selectedColumnIndex + 1);\n }\n\n break;\n }\n\n // Move to same column of previous row\n // If same column is not available, move to last column of previous row\n case \"up\": {\n const previousRow = data.rows[selectedRowIndex - 1];\n\n if (previousRow) {\n setSelectedRowIndex(selectedRowIndex - 1);\n\n if (!previousRow[selectedColumnIndex]) {\n setSelectedColumnIndex(previousRow.length - 1);\n }\n }\n\n break;\n }\n\n // Move to same column of next row\n // If same column is not available, move to last column of next row\n case \"down\": {\n const nextRow = data.rows[selectedRowIndex + 1];\n\n if (nextRow) {\n setSelectedRowIndex(selectedRowIndex + 1);\n\n if (!nextRow[selectedColumnIndex]) {\n setSelectedColumnIndex(nextRow.length - 1);\n }\n }\n\n break;\n }\n }\n },\n [data, interaction, selectedColumnIndex, selectedRowIndex]\n );\n\n const updateEmojis = useCallback(() => {\n if (!emojiData.current) {\n return;\n }\n\n startEmojisTransition(() => {\n setData(() => {\n if (!emojiData.current) {\n return;\n }\n\n const filteredEmojis = filterEmojis(\n emojiData.current.emojis,\n search.current\n );\n\n return generateEmojiPickerData(\n filteredEmojis,\n emojiData.current.categories,\n columns\n );\n });\n resetSelection();\n });\n }, [columns, resetSelection]);\n\n const handleSearch = useCallback(\n (value: string) => {\n search.current = value;\n updateEmojis();\n },\n [updateEmojis]\n );\n\n const initializeEmojiData = useCallback(\n async (locale: string) => {\n try {\n emojiData.current = await getEmojiData(locale);\n updateEmojis();\n } catch (error) {\n setError(error as Error);\n }\n },\n [updateEmojis]\n );\n\n useEffect(() => {\n let idleCallbackId: number;\n const timeoutId = setTimeout(() => {\n idleCallbackId = requestIdleCallback(() => {\n initializeEmojiData(locale);\n });\n }, LOADING_MINIMUM_TIMEOUT);\n\n return () => {\n clearTimeout(timeoutId);\n cancelIdleCallback(idleCallbackId);\n };\n }, [locale]); // eslint-disable-line react-hooks/exhaustive-deps\n\n useEffect(() => {\n if (interaction === \"none\") {\n resetSelection();\n }\n }, [interaction]); // eslint-disable-line react-hooks/exhaustive-deps\n\n return (\n <EmojiPickerContext.Provider\n value={{\n data: data as EmojiPickerData,\n error: error as undefined,\n isLoading: (!data && !error) as false,\n columns,\n onSearch: handleSearch,\n onEmojiSelect,\n selectCurrentEmoji,\n selectedRowIndex,\n selectedColumnIndex,\n moveSelection,\n setPointerSelection,\n interaction,\n setInteraction,\n }}\n >\n {children}\n </EmojiPickerContext.Provider>\n );\n}\n\n/**\n * @private\n * The EmojiPicker primitive is undocumented for now and subject to change,\n * use at your own risk. If you have any feedback on it, please let us know!\n * See how we use it in the default components to learn how to use it:\n * https://github.com/liveblocks/liveblocks/blob/main/packages/liveblocks-react-ui/src/components/internal/EmojiPicker.tsx.\n *\n * The search input of the emoji picker. It also affects the focus and selection\n * within `EmojiPicker.Content`.\n *\n * @example\n * <EmojiPicker.Search />\n */\nconst EmojiPickerSearch = forwardRef<HTMLInputElement, EmojiPickerSearchProps>(\n ({ asChild, value, defaultValue, onChange, ...props }, forwardedRef) => {\n const Component = asChild ? Slot : \"input\";\n const {\n onSearch,\n selectCurrentEmoji,\n moveSelection,\n interaction,\n setInteraction,\n } = useEmojiPicker();\n\n const handleChange = useCallback(\n (event: ChangeEvent<HTMLInputElement>) => {\n onChange?.(event);\n\n if (event.isDefaultPrevented()) {\n return;\n }\n\n const value = event.target.value;\n setInteraction(value ? \"keyboard\" : \"none\");\n onSearch(value);\n },\n [onChange, onSearch, setInteraction]\n );\n\n const handleKeyDown = useCallback(\n (event: KeyboardEvent<HTMLInputElement>) => {\n if (event.isDefaultPrevented()) {\n return;\n }\n\n if (isKey(event, \"ArrowLeft\")) {\n moveSelection(\"left\", event);\n } else if (isKey(event, \"ArrowRight\")) {\n moveSelection(\"right\", event);\n } else if (isKey(event, \"ArrowUp\")) {\n moveSelection(\"up\", event);\n } else if (isKey(event, \"ArrowDown\")) {\n moveSelection(\"down\", event);\n } else if (isKey(event, \"Enter\")) {\n if (interaction !== \"none\") {\n event.preventDefault();\n selectCurrentEmoji();\n }\n }\n },\n [interaction, moveSelection, selectCurrentEmoji]\n );\n\n useEffect(() => {\n onSearch(\n value ? String(value) : defaultValue ? String(defaultValue) : \"\"\n );\n }, []); // eslint-disable-line react-hooks/exhaustive-deps\n\n return (\n <Component\n type=\"search\"\n value={value}\n defaultValue={defaultValue}\n onChange={handleChange}\n onKeyDown={handleKeyDown}\n {...props}\n ref={forwardedRef}\n />\n );\n }\n);\n\nconst defaultContentComponents: EmojiPickerContentComponents = {\n CategoryHeader: ({ category, ...props }) => <div {...props}>{category}</div>,\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Row: ({ children, attributes, ...props }) => <div {...props}>{children}</div>,\n Emoji: ({ emoji, ...props }) => (\n <button {...props}>\n <EmojiPrimitive emoji={emoji} />\n </button>\n ),\n Loading: (props) => <div {...props} />,\n Empty: (props) => <div {...props} />,\n Grid: (props) => <div {...props} />,\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Error: ({ error, ...props }) => <div {...props} />,\n};\n\nconst placeholderRowAttributes: EmojiPickerContentEmojiRowAttributes = {\n rowIndex: -1,\n categoryRowIndex: -1,\n categoryRowsCount: 0,\n};\n\nconst VirtuosoScroller = forwardRef<HTMLDivElement, ScrollerProps>(\n ({ children, ...props }, forwardedRef) => {\n return (\n <div {...props} tabIndex={-1} data-test-id={undefined} ref={forwardedRef}>\n {children}\n </div>\n );\n }\n);\n\nconst VirtuosoTopList = forwardRef<HTMLDivElement, TopItemListProps>(\n ({ children, ...props }, forwardedRef) => {\n return (\n <div {...props} data-test-id={undefined} ref={forwardedRef}>\n {children}\n </div>\n );\n }\n);\n\n/**\n * @private\n * The EmojiPicker primitive is undocumented for now and subject to change,\n * use at your own risk. If you have any feedback on it, please let us know!\n * See how we use it in the default components to learn how to use it:\n * https://github.com/liveblocks/liveblocks/blob/main/packages/liveblocks-react-ui/src/components/internal/EmojiPicker.tsx.\n *\n * The main content of the emoji picker, either displaying the emoji grid or various\n * alternative states (loading, empty, and error).\n *\n * @example\n * <EmojiPicker.Content\n * components={{\n * Loading: EmojiPickerLoading,\n * Empty: EmojiPickerEmpty,\n * Error: EmojiPickerError,\n * CategoryHeader: EmojiPickerCategoryHeader,\n * Grid: EmojiPickerGrid,\n * Row: EmojiPickerRow,\n * Emoji: EmojiPickerEmoji,\n * }}\n * />\n */\nconst EmojiPickerContent = forwardRef<HTMLDivElement, EmojiPickerContentProps>(\n ({ components, asChild, ...props }, forwardedRef) => {\n const Component = asChild ? Slot : \"div\";\n const virtuosoRef = useRef<GroupedVirtuosoHandle>(null);\n const {\n data,\n error,\n isLoading,\n columns,\n onEmojiSelect,\n selectedColumnIndex,\n selectedRowIndex,\n setPointerSelection,\n interaction,\n setInteraction,\n } = useEmojiPicker();\n const selectedEmoji = useMemo(\n () => data?.rows[selectedRowIndex]?.[selectedColumnIndex],\n [data?.rows, selectedColumnIndex, selectedRowIndex]\n );\n const { Loading, Empty, Error, CategoryHeader, Grid, Row, Emoji } = useMemo(\n () => ({ ...defaultContentComponents, ...components }),\n [components]\n );\n const VirtuosoList = useMemo(\n () =>\n forwardRef<HTMLDivElement, VirtuosoListProps>(\n ({ children, ...props }, forwardedRef) => {\n return (\n <div\n role=\"grid\"\n aria-colcount={columns}\n {...props}\n data-test-id={undefined}\n ref={forwardedRef}\n >\n {children}\n </div>\n );\n }\n ),\n [columns]\n );\n const placeholderColumns = useMemo(\n () => Array<string>(columns).fill(\"🌫️\"),\n [columns]\n );\n\n const preventDefault = useCallback((event: SyntheticEvent) => {\n event.preventDefault();\n }, []);\n\n const handleEmojiPointerLeave = useCallback(() => {\n if (interaction === \"pointer\") {\n setInteraction(\"none\");\n }\n }, [interaction, setInteraction]);\n\n useEffect(() => {\n if (interaction === \"keyboard\") {\n virtuosoRef.current?.scrollIntoView({\n index: selectedRowIndex,\n behavior: \"auto\",\n });\n }\n }, [interaction, selectedRowIndex]);\n\n return (\n <Component {...props} ref={forwardedRef}>\n {/* Virtualized rows are absolutely positioned so they won't make\n the container automatically pick up their width. To achieve\n an automatic width, we add a relative (but hidden) full row. */}\n <div\n style={{\n visibility: \"hidden\",\n height: 0,\n }}\n >\n <Row attributes={placeholderRowAttributes}>\n {placeholderColumns.map((placeholder, index) => (\n <Emoji emoji={placeholder} key={index} />\n ))}\n </Row>\n </div>\n {isLoading ? (\n <Loading />\n ) : error ? (\n <Error error={error} />\n ) : data.count === 0 ? (\n <Empty />\n ) : (\n <Grid>\n <GroupedVirtuoso\n ref={virtuosoRef}\n components={{\n Scroller: VirtuosoScroller,\n List: VirtuosoList,\n TopItemList: VirtuosoTopList,\n }}\n groupCounts={data.categoriesRowCounts}\n groupContent={(groupIndex) => {\n return (\n <CategoryHeader category={data.categories[groupIndex]} />\n );\n }}\n itemContent={(rowIndex, groupIndex) => {\n const categoryRowIndex =\n data.categoriesRowIndices[groupIndex].indexOf(rowIndex);\n const categoryRowsCount = data.categoriesRowCounts[groupIndex];\n\n return (\n <Row\n attributes={{\n rowIndex,\n categoryRowIndex,\n categoryRowsCount,\n }}\n >\n {data.rows[rowIndex].map((emoji, columnIndex) => {\n const isSelected =\n interaction !== \"none\" &&\n selectedColumnIndex === columnIndex &&\n selectedRowIndex === rowIndex;\n\n return (\n <Emoji\n key={emoji.emoji}\n role=\"gridcell\"\n aria-colindex={columnIndex}\n aria-selected={isSelected || undefined}\n data-selected={isSelected || undefined}\n onMouseDown={preventDefault}\n tabIndex={-1}\n onPointerEnter={() => {\n setPointerSelection(columnIndex, rowIndex);\n }}\n onPointerLeave={handleEmojiPointerLeave}\n onClick={(event) => {\n onEmojiSelect?.(emoji.emoji);\n event.stopPropagation();\n }}\n emoji={emoji.emoji}\n />\n );\n })}\n </Row>\n );\n }}\n />\n </Grid>\n )}\n {selectedEmoji && interaction !== \"none\" && (\n <div aria-live=\"polite\" style={visuallyHidden}>\n {selectedEmoji.name}\n </div>\n )}\n </Component>\n );\n }\n);\n\nif (process.env.NODE_ENV !== \"production\") {\n EmojiPickerRoot.displayName = EMOJIPICKER_ROOT_NAME;\n EmojiPickerContent.displayName = EMOJIPICKER_CONTENT_NAME;\n EmojiPickerSearch.displayName = EMOJIPICKER_SEARCH_NAME;\n}\n\n// NOTE: Every export from this file will be available publicly as EmojiPicker.*\nexport {\n EmojiPickerContent as Content,\n EmojiPickerRoot as Root,\n EmojiPickerSearch as Search,\n};\n"],"names":[],"mappings":";;;;;;;;;;;;;;AAyCA;AACA;AACA;AAEA;AACA;AACA;AAkBA;AAAyB;AACb;AACD;AACT;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAIA;AACE;AACE;AAEA;AACE;AAAyB;AAC3B;AACF;AAGF;AACE;AACA;AAAqB;AAGvB;AAA4B;AAExB;AACA;AACA;AAA4B;AAC9B;AACC;AAGH;AAAsB;AAKlB;AACE;AAAA;AAGF;AAEA;AACE;AACA;AAAA;AAGF;AAEA;AAAmB;AAIf;AACE;AACA;AAEA;AACE;AACA;AAA6C;AAC/C;AAEA;AAA8C;AAGhD;AAAA;AACF;AAKE;AAEA;AACE;AACA;AAEA;AACE;AACA;AAAwB;AAC1B;AAEA;AAA8C;AAGhD;AAAA;AACF;AAKE;AAEA;AACE;AAEA;AACE;AAA6C;AAC/C;AAGF;AAAA;AACF;AAKE;AAEA;AACE;AAEA;AACE;AAAyC;AAC3C;AAGF;AAAA;AACF;AACF;AACF;AACyD;AAG3D;AACE;AACE;AAAA;AAGF;AACE;AACE;AACE;AAAA;AAGF;AAAuB;AACH;AACX;AAGT;AAAO;AACL;AACkB;AAClB;AACF;AAEF;AAAe;AAChB;AAGH;AAAqB;AAEjB;AACA;AAAa;AACf;AACa;AAGf;AAA4B;AAExB;AACE;AACA;AAAa;AAEb;AAAuB;AACzB;AACF;AACa;AAGf;AACE;AACA;AACE;AACE;AAA0B;AAC3B;AAGH;AACE;AACA;AAAiC;AACnC;AAGF;AACE;AACE;AAAe;AACjB;AAGF;AACG;AACQ;AACL;AACA;AACsB;AACtB;AACU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACF;AAKN;AAeA;AAA0B;AAEtB;AACA;AAAM;AACJ;AACA;AACA;AACA;AACA;AAGF;AAAqB;AAEjB;AAEA;AACE;AAAA;AAGF;AACA;AACA;AAAc;AAChB;AACmC;AAGrC;AAAsB;AAElB;AACE;AAAA;AAGF;AACE;AAA2B;AAE3B;AAA4B;AAE5B;AAAyB;AAEzB;AAA2B;AAE3B;AACE;AACA;AAAmB;AACrB;AACF;AACF;AAC+C;AAGjD;AACE;AAAA;AACgE;AAChE;AAGF;AACG;AACM;AACL;AACA;AACU;AACC;AACP;AACC;AACP;AAGN;AAEA;AAA+D;AAChB;AAAQ;AAAiB;AAExB;AAAQ;AAAiB;AAEpE;AAAW;AACT;AAAe;AAClB;AAEmB;AAAQ;AAAO;AACjB;AAAQ;AAAO;AAChB;AAAQ;AAAO;AAEA;AAAQ;AAC3C;AAEA;AAAuE;AAC3D;AACQ;AAEpB;AAEA;AAAyB;AAErB;AACG;AAAQ;AAAiB;AAAkB;AAAgB;AAE5D;AAGN;AAEA;AAAwB;AAEpB;AACG;AAAQ;AAAqB;AAAgB;AAE9C;AAGN;AAyBA;AAA2B;AAEvB;AACA;AACA;AAAM;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEF;AAAsB;AACiB;AACa;AAEpD;AAAoE;AACd;AACzC;AAEb;AAAqB;AAEjB;AAEI;AACG;AACM;AACU;AACX;AACU;AACT;AAGP;AAEJ;AACF;AACM;AAEV;AAA2B;AACc;AAC/B;AAGV;AACE;AAAqB;AAGvB;AACE;AACE;AAAqB;AACvB;AAGF;AACE;AACE;AAAoC;AAC3B;AACG;AACX;AACH;AAGF;AACG;AAAc;AAAY;AAIxB;AACQ;AACO;AACJ;AACV;AAEC;AAAgB;AAEZ;AAAa;AAAkB;AAOnC;AAAM;AAKJ;AACM;AACO;AACA;AACJ;AACO;AACf;AACkB;AAEhB;AACG;AAAyC;AAAa;AAE3D;AAEE;AAEA;AAEA;AACG;AACa;AACV;AACA;AACA;AACF;AAGE;AAKA;AACG;AACY;AACN;AACU;AACc;AACA;AAChB;AACH;AAER;AAAyC;AAC3C;AACgB;AAEd;AACA;AAAsB;AACxB;AACa;AACf;AAGN;AAEJ;AAKH;AAAc;AAAgB;AAInC;AAGN;AAEA;AACE;AACA;AACA;AACF;;;;"}
|