@sendbird/uikit-react-native 2.1.0 → 2.2.0

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.
Files changed (388) hide show
  1. package/lib/commonjs/components/ChatFlatList.js +2 -1
  2. package/lib/commonjs/components/ChatFlatList.js.map +1 -1
  3. package/lib/commonjs/components/MessageRenderer/FileMessage/BaseFileMessage.js +11 -6
  4. package/lib/commonjs/components/MessageRenderer/FileMessage/BaseFileMessage.js.map +1 -1
  5. package/lib/commonjs/components/MessageRenderer/FileMessage/ImageFileMessage.js +12 -2
  6. package/lib/commonjs/components/MessageRenderer/FileMessage/ImageFileMessage.js.map +1 -1
  7. package/lib/commonjs/components/MessageRenderer/FileMessage/VideoFileMessage.js +25 -10
  8. package/lib/commonjs/components/MessageRenderer/FileMessage/VideoFileMessage.js.map +1 -1
  9. package/lib/commonjs/components/MessageRenderer/UserMessage/BaseUserMessage.js +86 -10
  10. package/lib/commonjs/components/MessageRenderer/UserMessage/BaseUserMessage.js.map +1 -1
  11. package/lib/commonjs/components/MessageRenderer/UserMessage/OpenGraphUserMessage.js +99 -14
  12. package/lib/commonjs/components/MessageRenderer/UserMessage/OpenGraphUserMessage.js.map +1 -1
  13. package/lib/commonjs/components/MessageRenderer/UserMessage/index.js.map +1 -1
  14. package/lib/commonjs/components/MessageRenderer/index.js +22 -3
  15. package/lib/commonjs/components/MessageRenderer/index.js.map +1 -1
  16. package/lib/commonjs/components/ReactionAddons/BottomSheetReactionAddon.js +150 -0
  17. package/lib/commonjs/components/ReactionAddons/BottomSheetReactionAddon.js.map +1 -0
  18. package/lib/commonjs/components/ReactionAddons/MessageReactionAddon.js +135 -0
  19. package/lib/commonjs/components/ReactionAddons/MessageReactionAddon.js.map +1 -0
  20. package/lib/commonjs/components/ReactionAddons/ReactionRoundedButton.js +87 -0
  21. package/lib/commonjs/components/ReactionAddons/ReactionRoundedButton.js.map +1 -0
  22. package/lib/commonjs/components/ReactionAddons/index.js +19 -0
  23. package/lib/commonjs/components/ReactionAddons/index.js.map +1 -0
  24. package/lib/commonjs/components/ReactionBottomSheets/ReactionListBottomSheet.js +150 -0
  25. package/lib/commonjs/components/ReactionBottomSheets/ReactionListBottomSheet.js.map +1 -0
  26. package/lib/commonjs/components/ReactionBottomSheets/ReactionUserListBottomSheet.js +276 -0
  27. package/lib/commonjs/components/ReactionBottomSheets/ReactionUserListBottomSheet.js.map +1 -0
  28. package/lib/commonjs/components/ReactionBottomSheets/index.js +19 -0
  29. package/lib/commonjs/components/ReactionBottomSheets/index.js.map +1 -0
  30. package/lib/commonjs/constants.js +3 -1
  31. package/lib/commonjs/constants.js.map +1 -1
  32. package/lib/commonjs/containers/GroupChannelPreviewContainer.js +4 -1
  33. package/lib/commonjs/containers/GroupChannelPreviewContainer.js.map +1 -1
  34. package/lib/commonjs/containers/SendbirdUIKitContainer.js +118 -78
  35. package/lib/commonjs/containers/SendbirdUIKitContainer.js.map +1 -1
  36. package/lib/commonjs/contexts/ReactionCtx.js +107 -0
  37. package/lib/commonjs/contexts/ReactionCtx.js.map +1 -0
  38. package/lib/commonjs/contexts/SendbirdChatCtx.js +18 -6
  39. package/lib/commonjs/contexts/SendbirdChatCtx.js.map +1 -1
  40. package/lib/commonjs/contexts/UserProfileCtx.js +9 -0
  41. package/lib/commonjs/contexts/UserProfileCtx.js.map +1 -1
  42. package/lib/commonjs/domain/groupChannel/component/GroupChannelInput/EditInput.js +33 -37
  43. package/lib/commonjs/domain/groupChannel/component/GroupChannelInput/EditInput.js.map +1 -1
  44. package/lib/commonjs/domain/groupChannel/component/GroupChannelInput/SendInput.js +28 -11
  45. package/lib/commonjs/domain/groupChannel/component/GroupChannelInput/SendInput.js.map +1 -1
  46. package/lib/commonjs/domain/groupChannel/component/GroupChannelInput/index.js +120 -31
  47. package/lib/commonjs/domain/groupChannel/component/GroupChannelInput/index.js.map +1 -1
  48. package/lib/commonjs/domain/groupChannel/component/GroupChannelMessageList.js +31 -12
  49. package/lib/commonjs/domain/groupChannel/component/GroupChannelMessageList.js.map +1 -1
  50. package/lib/commonjs/domain/groupChannel/component/GroupChannelSuggestedMentionList.js +210 -0
  51. package/lib/commonjs/domain/groupChannel/component/GroupChannelSuggestedMentionList.js.map +1 -0
  52. package/lib/commonjs/domain/groupChannel/index.js +8 -0
  53. package/lib/commonjs/domain/groupChannel/index.js.map +1 -1
  54. package/lib/commonjs/domain/groupChannel/module/createGroupChannelModule.js +4 -0
  55. package/lib/commonjs/domain/groupChannel/module/createGroupChannelModule.js.map +1 -1
  56. package/lib/commonjs/domain/groupChannel/module/moduleContext.js +4 -4
  57. package/lib/commonjs/domain/groupChannel/module/moduleContext.js.map +1 -1
  58. package/lib/commonjs/domain/groupChannel/types.js.map +1 -1
  59. package/lib/commonjs/domain/groupChannelBannedUsers/component/GroupChannelBannedUsersList.js +3 -2
  60. package/lib/commonjs/domain/groupChannelBannedUsers/component/GroupChannelBannedUsersList.js.map +1 -1
  61. package/lib/commonjs/domain/groupChannelList/component/GroupChannelListList.js +2 -1
  62. package/lib/commonjs/domain/groupChannelList/component/GroupChannelListList.js.map +1 -1
  63. package/lib/commonjs/domain/groupChannelMutedMembers/component/GroupChannelMutedMembersList.js +2 -1
  64. package/lib/commonjs/domain/groupChannelMutedMembers/component/GroupChannelMutedMembersList.js.map +1 -1
  65. package/lib/commonjs/domain/groupChannelNotifications/component/GroupChannelNotificationsHeader.js +39 -0
  66. package/lib/commonjs/domain/groupChannelNotifications/component/GroupChannelNotificationsHeader.js.map +1 -0
  67. package/lib/commonjs/domain/groupChannelNotifications/component/GroupChannelNotificationsView.js +140 -0
  68. package/lib/commonjs/domain/groupChannelNotifications/component/GroupChannelNotificationsView.js.map +1 -0
  69. package/lib/commonjs/domain/groupChannelNotifications/index.js +46 -0
  70. package/lib/commonjs/domain/groupChannelNotifications/index.js.map +1 -0
  71. package/lib/commonjs/domain/groupChannelNotifications/module/createGroupChannelNotificationsModule.js +33 -0
  72. package/lib/commonjs/domain/groupChannelNotifications/module/createGroupChannelNotificationsModule.js.map +1 -0
  73. package/lib/commonjs/domain/groupChannelNotifications/module/moduleContext.js +45 -0
  74. package/lib/commonjs/domain/groupChannelNotifications/module/moduleContext.js.map +1 -0
  75. package/lib/commonjs/domain/groupChannelNotifications/types.js +6 -0
  76. package/lib/commonjs/domain/groupChannelNotifications/types.js.map +1 -0
  77. package/lib/commonjs/domain/groupChannelOperators/component/GroupChannelOperatorsList.js +2 -1
  78. package/lib/commonjs/domain/groupChannelOperators/component/GroupChannelOperatorsList.js.map +1 -1
  79. package/lib/commonjs/domain/groupChannelSettings/component/GroupChannelSettingsMenu.js +49 -6
  80. package/lib/commonjs/domain/groupChannelSettings/component/GroupChannelSettingsMenu.js.map +1 -1
  81. package/lib/commonjs/domain/groupChannelSettings/module/moduleContext.js +9 -14
  82. package/lib/commonjs/domain/groupChannelSettings/module/moduleContext.js.map +1 -1
  83. package/lib/commonjs/domain/groupChannelSettings/types.js.map +1 -1
  84. package/lib/commonjs/domain/groupChannelUserList/types.js.map +1 -1
  85. package/lib/commonjs/domain/userList/component/UserListList.js +4 -1
  86. package/lib/commonjs/domain/userList/component/UserListList.js.map +1 -1
  87. package/lib/commonjs/domain/userList/module/createUserListModule.js.map +1 -1
  88. package/lib/commonjs/fragments/createGroupChannelCreateFragment.js.map +1 -1
  89. package/lib/commonjs/fragments/createGroupChannelFragment.js +11 -4
  90. package/lib/commonjs/fragments/createGroupChannelFragment.js.map +1 -1
  91. package/lib/commonjs/fragments/createGroupChannelInviteFragment.js.map +1 -1
  92. package/lib/commonjs/fragments/createGroupChannelNotificationsFragment.js +34 -0
  93. package/lib/commonjs/fragments/createGroupChannelNotificationsFragment.js.map +1 -0
  94. package/lib/commonjs/fragments/createGroupChannelSettingsFragment.js +3 -1
  95. package/lib/commonjs/fragments/createGroupChannelSettingsFragment.js.map +1 -1
  96. package/lib/commonjs/hooks/useConnection.js +3 -0
  97. package/lib/commonjs/hooks/useConnection.js.map +1 -1
  98. package/lib/commonjs/hooks/useContext.js +11 -1
  99. package/lib/commonjs/hooks/useContext.js.map +1 -1
  100. package/lib/commonjs/hooks/useKeyboardStatus.js +93 -0
  101. package/lib/commonjs/hooks/useKeyboardStatus.js.map +1 -0
  102. package/lib/commonjs/hooks/useMentionSuggestion.js +110 -0
  103. package/lib/commonjs/hooks/useMentionSuggestion.js.map +1 -0
  104. package/lib/commonjs/hooks/useMentionTextInput.js +139 -0
  105. package/lib/commonjs/hooks/useMentionTextInput.js.map +1 -0
  106. package/lib/commonjs/index.js +95 -40
  107. package/lib/commonjs/index.js.map +1 -1
  108. package/lib/commonjs/libs/EmojiManager.js +100 -0
  109. package/lib/commonjs/libs/EmojiManager.js.map +1 -0
  110. package/lib/commonjs/libs/MentionConfig.js +47 -0
  111. package/lib/commonjs/libs/MentionConfig.js.map +1 -0
  112. package/lib/commonjs/libs/MentionManager.js +235 -0
  113. package/lib/commonjs/libs/MentionManager.js.map +1 -0
  114. package/lib/commonjs/localization/StringSet.type.js +17 -5
  115. package/lib/commonjs/localization/StringSet.type.js.map +1 -1
  116. package/lib/commonjs/types.js.map +1 -1
  117. package/lib/commonjs/utils/common.js +19 -0
  118. package/lib/commonjs/utils/common.js.map +1 -0
  119. package/lib/commonjs/version.js +1 -1
  120. package/lib/commonjs/version.js.map +1 -1
  121. package/lib/module/components/ChatFlatList.js +3 -2
  122. package/lib/module/components/ChatFlatList.js.map +1 -1
  123. package/lib/module/components/MessageRenderer/FileMessage/BaseFileMessage.js +11 -6
  124. package/lib/module/components/MessageRenderer/FileMessage/BaseFileMessage.js.map +1 -1
  125. package/lib/module/components/MessageRenderer/FileMessage/ImageFileMessage.js +12 -2
  126. package/lib/module/components/MessageRenderer/FileMessage/ImageFileMessage.js.map +1 -1
  127. package/lib/module/components/MessageRenderer/FileMessage/VideoFileMessage.js +25 -10
  128. package/lib/module/components/MessageRenderer/FileMessage/VideoFileMessage.js.map +1 -1
  129. package/lib/module/components/MessageRenderer/UserMessage/BaseUserMessage.js +86 -12
  130. package/lib/module/components/MessageRenderer/UserMessage/BaseUserMessage.js.map +1 -1
  131. package/lib/module/components/MessageRenderer/UserMessage/OpenGraphUserMessage.js +101 -17
  132. package/lib/module/components/MessageRenderer/UserMessage/OpenGraphUserMessage.js.map +1 -1
  133. package/lib/module/components/MessageRenderer/UserMessage/index.js.map +1 -1
  134. package/lib/module/components/MessageRenderer/index.js +21 -4
  135. package/lib/module/components/MessageRenderer/index.js.map +1 -1
  136. package/lib/module/components/ReactionAddons/BottomSheetReactionAddon.js +132 -0
  137. package/lib/module/components/ReactionAddons/BottomSheetReactionAddon.js.map +1 -0
  138. package/lib/module/components/ReactionAddons/MessageReactionAddon.js +118 -0
  139. package/lib/module/components/ReactionAddons/MessageReactionAddon.js.map +1 -0
  140. package/lib/module/components/ReactionAddons/ReactionRoundedButton.js +74 -0
  141. package/lib/module/components/ReactionAddons/ReactionRoundedButton.js.map +1 -0
  142. package/lib/module/components/ReactionAddons/index.js +7 -0
  143. package/lib/module/components/ReactionAddons/index.js.map +1 -0
  144. package/lib/module/components/ReactionBottomSheets/ReactionListBottomSheet.js +135 -0
  145. package/lib/module/components/ReactionBottomSheets/ReactionListBottomSheet.js.map +1 -0
  146. package/lib/module/components/ReactionBottomSheets/ReactionUserListBottomSheet.js +259 -0
  147. package/lib/module/components/ReactionBottomSheets/ReactionUserListBottomSheet.js.map +1 -0
  148. package/lib/module/components/ReactionBottomSheets/index.js +7 -0
  149. package/lib/module/components/ReactionBottomSheets/index.js.map +1 -0
  150. package/lib/module/constants.js +1 -0
  151. package/lib/module/constants.js.map +1 -1
  152. package/lib/module/containers/GroupChannelPreviewContainer.js +4 -1
  153. package/lib/module/containers/GroupChannelPreviewContainer.js.map +1 -1
  154. package/lib/module/containers/SendbirdUIKitContainer.js +116 -80
  155. package/lib/module/containers/SendbirdUIKitContainer.js.map +1 -1
  156. package/lib/module/contexts/ReactionCtx.js +85 -0
  157. package/lib/module/contexts/ReactionCtx.js.map +1 -0
  158. package/lib/module/contexts/SendbirdChatCtx.js +18 -6
  159. package/lib/module/contexts/SendbirdChatCtx.js.map +1 -1
  160. package/lib/module/contexts/UserProfileCtx.js +7 -0
  161. package/lib/module/contexts/UserProfileCtx.js.map +1 -1
  162. package/lib/module/domain/groupChannel/component/GroupChannelInput/EditInput.js +34 -39
  163. package/lib/module/domain/groupChannel/component/GroupChannelInput/EditInput.js.map +1 -1
  164. package/lib/module/domain/groupChannel/component/GroupChannelInput/SendInput.js +24 -13
  165. package/lib/module/domain/groupChannel/component/GroupChannelInput/SendInput.js.map +1 -1
  166. package/lib/module/domain/groupChannel/component/GroupChannelInput/index.js +117 -33
  167. package/lib/module/domain/groupChannel/component/GroupChannelInput/index.js.map +1 -1
  168. package/lib/module/domain/groupChannel/component/GroupChannelMessageList.js +32 -14
  169. package/lib/module/domain/groupChannel/component/GroupChannelMessageList.js.map +1 -1
  170. package/lib/module/domain/groupChannel/component/GroupChannelSuggestedMentionList.js +188 -0
  171. package/lib/module/domain/groupChannel/component/GroupChannelSuggestedMentionList.js.map +1 -0
  172. package/lib/module/domain/groupChannel/index.js +1 -0
  173. package/lib/module/domain/groupChannel/index.js.map +1 -1
  174. package/lib/module/domain/groupChannel/module/createGroupChannelModule.js +3 -0
  175. package/lib/module/domain/groupChannel/module/createGroupChannelModule.js.map +1 -1
  176. package/lib/module/domain/groupChannel/module/moduleContext.js +4 -4
  177. package/lib/module/domain/groupChannel/module/moduleContext.js.map +1 -1
  178. package/lib/module/domain/groupChannel/types.js.map +1 -1
  179. package/lib/module/domain/groupChannelBannedUsers/component/GroupChannelBannedUsersList.js +4 -3
  180. package/lib/module/domain/groupChannelBannedUsers/component/GroupChannelBannedUsersList.js.map +1 -1
  181. package/lib/module/domain/groupChannelList/component/GroupChannelListList.js +3 -2
  182. package/lib/module/domain/groupChannelList/component/GroupChannelListList.js.map +1 -1
  183. package/lib/module/domain/groupChannelMutedMembers/component/GroupChannelMutedMembersList.js +3 -2
  184. package/lib/module/domain/groupChannelMutedMembers/component/GroupChannelMutedMembersList.js.map +1 -1
  185. package/lib/module/domain/groupChannelNotifications/component/GroupChannelNotificationsHeader.js +25 -0
  186. package/lib/module/domain/groupChannelNotifications/component/GroupChannelNotificationsHeader.js.map +1 -0
  187. package/lib/module/domain/groupChannelNotifications/component/GroupChannelNotificationsView.js +122 -0
  188. package/lib/module/domain/groupChannelNotifications/component/GroupChannelNotificationsView.js.map +1 -0
  189. package/lib/module/domain/groupChannelNotifications/index.js +5 -0
  190. package/lib/module/domain/groupChannelNotifications/index.js.map +1 -0
  191. package/lib/module/domain/groupChannelNotifications/module/createGroupChannelNotificationsModule.js +21 -0
  192. package/lib/module/domain/groupChannelNotifications/module/createGroupChannelNotificationsModule.js.map +1 -0
  193. package/lib/module/domain/groupChannelNotifications/module/moduleContext.js +25 -0
  194. package/lib/module/domain/groupChannelNotifications/module/moduleContext.js.map +1 -0
  195. package/lib/module/domain/groupChannelNotifications/types.js +2 -0
  196. package/lib/module/domain/groupChannelNotifications/types.js.map +1 -0
  197. package/lib/module/domain/groupChannelOperators/component/GroupChannelOperatorsList.js +3 -2
  198. package/lib/module/domain/groupChannelOperators/component/GroupChannelOperatorsList.js.map +1 -1
  199. package/lib/module/domain/groupChannelSettings/component/GroupChannelSettingsMenu.js +46 -6
  200. package/lib/module/domain/groupChannelSettings/component/GroupChannelSettingsMenu.js.map +1 -1
  201. package/lib/module/domain/groupChannelSettings/module/moduleContext.js +10 -15
  202. package/lib/module/domain/groupChannelSettings/module/moduleContext.js.map +1 -1
  203. package/lib/module/domain/groupChannelSettings/types.js.map +1 -1
  204. package/lib/module/domain/groupChannelUserList/types.js.map +1 -1
  205. package/lib/module/domain/userList/component/UserListList.js +3 -1
  206. package/lib/module/domain/userList/component/UserListList.js.map +1 -1
  207. package/lib/module/domain/userList/module/createUserListModule.js.map +1 -1
  208. package/lib/module/fragments/createGroupChannelCreateFragment.js.map +1 -1
  209. package/lib/module/fragments/createGroupChannelFragment.js +11 -4
  210. package/lib/module/fragments/createGroupChannelFragment.js.map +1 -1
  211. package/lib/module/fragments/createGroupChannelInviteFragment.js.map +1 -1
  212. package/lib/module/fragments/createGroupChannelNotificationsFragment.js +22 -0
  213. package/lib/module/fragments/createGroupChannelNotificationsFragment.js.map +1 -0
  214. package/lib/module/fragments/createGroupChannelSettingsFragment.js +3 -1
  215. package/lib/module/fragments/createGroupChannelSettingsFragment.js.map +1 -1
  216. package/lib/module/hooks/useConnection.js +3 -0
  217. package/lib/module/hooks/useConnection.js.map +1 -1
  218. package/lib/module/hooks/useContext.js +6 -0
  219. package/lib/module/hooks/useContext.js.map +1 -1
  220. package/lib/module/hooks/useKeyboardStatus.js +75 -0
  221. package/lib/module/hooks/useKeyboardStatus.js.map +1 -0
  222. package/lib/module/hooks/useMentionSuggestion.js +100 -0
  223. package/lib/module/hooks/useMentionSuggestion.js.map +1 -0
  224. package/lib/module/hooks/useMentionTextInput.js +128 -0
  225. package/lib/module/hooks/useMentionTextInput.js.map +1 -0
  226. package/lib/module/index.js +8 -3
  227. package/lib/module/index.js.map +1 -1
  228. package/lib/module/libs/EmojiManager.js +90 -0
  229. package/lib/module/libs/EmojiManager.js.map +1 -0
  230. package/lib/module/libs/MentionConfig.js +39 -0
  231. package/lib/module/libs/MentionConfig.js.map +1 -0
  232. package/lib/module/libs/MentionManager.js +223 -0
  233. package/lib/module/libs/MentionManager.js.map +1 -0
  234. package/lib/module/localization/StringSet.type.js +18 -6
  235. package/lib/module/localization/StringSet.type.js.map +1 -1
  236. package/lib/module/types.js.map +1 -1
  237. package/lib/module/utils/common.js +7 -0
  238. package/lib/module/utils/common.js.map +1 -0
  239. package/lib/module/version.js +1 -1
  240. package/lib/module/version.js.map +1 -1
  241. package/lib/typescript/__template__/types.d.ts +3 -3
  242. package/lib/typescript/src/components/ChannelCover.d.ts +1 -1
  243. package/lib/typescript/src/components/ChatFlatList.d.ts +1 -1
  244. package/lib/typescript/src/components/FileViewer.d.ts +1 -1
  245. package/lib/typescript/src/components/MessageRenderer/AdminMessage/index.d.ts +1 -1
  246. package/lib/typescript/src/components/MessageRenderer/FileMessage/BaseFileMessage.d.ts +2 -2
  247. package/lib/typescript/src/components/MessageRenderer/FileMessage/ImageFileMessage.d.ts +1 -1
  248. package/lib/typescript/src/components/MessageRenderer/FileMessage/VideoFileMessage.d.ts +1 -1
  249. package/lib/typescript/src/components/MessageRenderer/FileMessage/index.d.ts +11 -2
  250. package/lib/typescript/src/components/MessageRenderer/MessageDateSeparator.d.ts +1 -1
  251. package/lib/typescript/src/components/MessageRenderer/MessageIncomingAvatar.d.ts +1 -1
  252. package/lib/typescript/src/components/MessageRenderer/MessageIncomingSenderName.d.ts +1 -1
  253. package/lib/typescript/src/components/MessageRenderer/MessageOutgoingStatus.d.ts +1 -1
  254. package/lib/typescript/src/components/MessageRenderer/MessageTime.d.ts +1 -1
  255. package/lib/typescript/src/components/MessageRenderer/UnknownMessage/index.d.ts +1 -1
  256. package/lib/typescript/src/components/MessageRenderer/UserMessage/BaseUserMessage.d.ts +1 -1
  257. package/lib/typescript/src/components/MessageRenderer/UserMessage/OpenGraphUserMessage.d.ts +2 -2
  258. package/lib/typescript/src/components/MessageRenderer/UserMessage/index.d.ts +4 -1
  259. package/lib/typescript/src/components/MessageRenderer/index.d.ts +4 -3
  260. package/lib/typescript/src/components/NewMessagesButton.d.ts +1 -1
  261. package/lib/typescript/src/components/ReactionAddons/BottomSheetReactionAddon.d.ts +8 -0
  262. package/lib/typescript/src/components/ReactionAddons/MessageReactionAddon.d.ts +6 -0
  263. package/lib/typescript/src/components/ReactionAddons/ReactionRoundedButton.d.ts +14 -0
  264. package/lib/typescript/src/components/ReactionAddons/index.d.ts +11 -0
  265. package/lib/typescript/src/components/ReactionBottomSheets/ReactionListBottomSheet.d.ts +3 -0
  266. package/lib/typescript/src/components/ReactionBottomSheets/ReactionUserListBottomSheet.d.ts +3 -0
  267. package/lib/typescript/src/components/ReactionBottomSheets/index.d.ts +20 -0
  268. package/lib/typescript/src/components/ScrollToBottomButton.d.ts +1 -1
  269. package/lib/typescript/src/components/StatusComposition.d.ts +1 -1
  270. package/lib/typescript/src/components/TypedPlaceholder.d.ts +1 -1
  271. package/lib/typescript/src/components/UserActionBar.d.ts +1 -1
  272. package/lib/typescript/src/components/UserSelectableBar.d.ts +1 -1
  273. package/lib/typescript/src/constants.d.ts +1 -0
  274. package/lib/typescript/src/containers/GroupChannelPreviewContainer.d.ts +1 -1
  275. package/lib/typescript/src/containers/SendbirdUIKitContainer.d.ts +12 -3
  276. package/lib/typescript/src/contexts/LocalizationCtx.d.ts +2 -2
  277. package/lib/typescript/src/contexts/PlatformServiceCtx.d.ts +1 -1
  278. package/lib/typescript/src/contexts/ReactionCtx.d.ts +18 -0
  279. package/lib/typescript/src/contexts/SendbirdChatCtx.d.ts +10 -2
  280. package/lib/typescript/src/contexts/UserProfileCtx.d.ts +4 -4
  281. package/lib/typescript/src/domain/groupChannel/component/GroupChannelInput/EditInput.d.ts +35 -7
  282. package/lib/typescript/src/domain/groupChannel/component/GroupChannelInput/SendInput.d.ts +32 -5
  283. package/lib/typescript/src/domain/groupChannel/component/GroupChannelInput/index.d.ts +19 -2
  284. package/lib/typescript/src/domain/groupChannel/component/GroupChannelSuggestedMentionList.d.ts +3 -0
  285. package/lib/typescript/src/domain/groupChannel/index.d.ts +1 -0
  286. package/lib/typescript/src/domain/groupChannel/module/createGroupChannelModule.d.ts +1 -1
  287. package/lib/typescript/src/domain/groupChannel/types.d.ts +24 -7
  288. package/lib/typescript/src/domain/groupChannelBannedUsers/types.d.ts +3 -3
  289. package/lib/typescript/src/domain/groupChannelList/types.d.ts +2 -2
  290. package/lib/typescript/src/domain/groupChannelModeration/types.d.ts +3 -3
  291. package/lib/typescript/src/domain/groupChannelMutedMembers/types.d.ts +3 -3
  292. package/lib/typescript/src/domain/groupChannelNotifications/component/GroupChannelNotificationsHeader.d.ts +3 -0
  293. package/lib/typescript/src/domain/groupChannelNotifications/component/GroupChannelNotificationsView.d.ts +2 -0
  294. package/lib/typescript/src/domain/groupChannelNotifications/index.d.ts +4 -0
  295. package/lib/typescript/src/domain/groupChannelNotifications/module/createGroupChannelNotificationsModule.d.ts +3 -0
  296. package/lib/typescript/src/domain/groupChannelNotifications/module/moduleContext.d.ts +3 -0
  297. package/lib/typescript/src/domain/groupChannelNotifications/types.d.ts +33 -0
  298. package/lib/typescript/src/domain/groupChannelOperators/types.d.ts +3 -3
  299. package/lib/typescript/src/domain/groupChannelSettings/component/GroupChannelSettingsMenu.d.ts +1 -1
  300. package/lib/typescript/src/domain/groupChannelSettings/types.d.ts +3 -1
  301. package/lib/typescript/src/domain/groupChannelUserList/types.d.ts +6 -6
  302. package/lib/typescript/src/domain/userList/component/UserListList.d.ts +2 -1
  303. package/lib/typescript/src/domain/userList/module/createUserListModule.d.ts +2 -1
  304. package/lib/typescript/src/fragments/createGroupChannelCreateFragment.d.ts +1 -1
  305. package/lib/typescript/src/fragments/createGroupChannelInviteFragment.d.ts +1 -1
  306. package/lib/typescript/src/fragments/createGroupChannelNotificationsFragment.d.ts +3 -0
  307. package/lib/typescript/src/hooks/useConnection.d.ts +1 -1
  308. package/lib/typescript/src/hooks/useContext.d.ts +4 -0
  309. package/lib/typescript/src/hooks/useKeyboardStatus.d.ts +6 -0
  310. package/lib/typescript/src/hooks/useMentionSuggestion.d.ts +17 -0
  311. package/lib/typescript/src/hooks/useMentionTextInput.d.ts +18 -0
  312. package/lib/typescript/src/index.d.ts +7 -2
  313. package/lib/typescript/src/libs/EmojiManager.d.ts +16 -0
  314. package/lib/typescript/src/libs/MentionConfig.d.ts +24 -0
  315. package/lib/typescript/src/libs/MentionManager.d.ts +61 -0
  316. package/lib/typescript/src/localization/StringSet.type.d.ts +16 -2
  317. package/lib/typescript/src/platform/createMediaService.expo.d.ts +1 -1
  318. package/lib/typescript/src/platform/createMediaService.native.d.ts +1 -1
  319. package/lib/typescript/src/platform/dynamicModule.d.ts +1 -1
  320. package/lib/typescript/src/platform/types.d.ts +5 -5
  321. package/lib/typescript/src/types.d.ts +13 -4
  322. package/lib/typescript/src/utils/common.d.ts +1 -0
  323. package/lib/typescript/src/version.d.ts +1 -1
  324. package/package.json +6 -6
  325. package/src/components/ChatFlatList.tsx +2 -1
  326. package/src/components/MessageRenderer/FileMessage/BaseFileMessage.tsx +17 -12
  327. package/src/components/MessageRenderer/FileMessage/ImageFileMessage.tsx +25 -18
  328. package/src/components/MessageRenderer/FileMessage/VideoFileMessage.tsx +20 -10
  329. package/src/components/MessageRenderer/UserMessage/BaseUserMessage.tsx +82 -14
  330. package/src/components/MessageRenderer/UserMessage/OpenGraphUserMessage.tsx +135 -61
  331. package/src/components/MessageRenderer/UserMessage/index.tsx +8 -1
  332. package/src/components/MessageRenderer/index.tsx +39 -7
  333. package/src/components/ReactionAddons/BottomSheetReactionAddon.tsx +106 -0
  334. package/src/components/ReactionAddons/MessageReactionAddon.tsx +123 -0
  335. package/src/components/ReactionAddons/ReactionRoundedButton.tsx +71 -0
  336. package/src/components/ReactionAddons/index.tsx +7 -0
  337. package/src/components/ReactionBottomSheets/ReactionListBottomSheet.tsx +113 -0
  338. package/src/components/ReactionBottomSheets/ReactionUserListBottomSheet.tsx +249 -0
  339. package/src/components/ReactionBottomSheets/index.tsx +24 -0
  340. package/src/constants.ts +2 -0
  341. package/src/containers/GroupChannelPreviewContainer.tsx +3 -1
  342. package/src/containers/SendbirdUIKitContainer.tsx +141 -83
  343. package/src/contexts/ReactionCtx.tsx +102 -0
  344. package/src/contexts/SendbirdChatCtx.tsx +24 -2
  345. package/src/contexts/UserProfileCtx.tsx +9 -0
  346. package/src/domain/groupChannel/component/GroupChannelInput/EditInput.tsx +56 -30
  347. package/src/domain/groupChannel/component/GroupChannelInput/SendInput.tsx +48 -19
  348. package/src/domain/groupChannel/component/GroupChannelInput/index.tsx +137 -43
  349. package/src/domain/groupChannel/component/GroupChannelMessageList.tsx +27 -10
  350. package/src/domain/groupChannel/component/GroupChannelSuggestedMentionList.tsx +173 -0
  351. package/src/domain/groupChannel/index.ts +1 -0
  352. package/src/domain/groupChannel/module/createGroupChannelModule.tsx +12 -1
  353. package/src/domain/groupChannel/module/moduleContext.tsx +4 -4
  354. package/src/domain/groupChannel/types.ts +28 -5
  355. package/src/domain/groupChannelBannedUsers/component/GroupChannelBannedUsersList.tsx +3 -2
  356. package/src/domain/groupChannelList/component/GroupChannelListList.tsx +2 -1
  357. package/src/domain/groupChannelMutedMembers/component/GroupChannelMutedMembersList.tsx +2 -1
  358. package/src/domain/groupChannelNotifications/component/GroupChannelNotificationsHeader.tsx +14 -0
  359. package/src/domain/groupChannelNotifications/component/GroupChannelNotificationsView.tsx +128 -0
  360. package/src/domain/groupChannelNotifications/index.ts +4 -0
  361. package/src/domain/groupChannelNotifications/module/createGroupChannelNotificationsModule.tsx +15 -0
  362. package/src/domain/groupChannelNotifications/module/moduleContext.tsx +31 -0
  363. package/src/domain/groupChannelNotifications/types.ts +38 -0
  364. package/src/domain/groupChannelOperators/component/GroupChannelOperatorsList.tsx +2 -1
  365. package/src/domain/groupChannelSettings/component/GroupChannelSettingsMenu.tsx +42 -3
  366. package/src/domain/groupChannelSettings/module/moduleContext.tsx +9 -12
  367. package/src/domain/groupChannelSettings/types.ts +2 -0
  368. package/src/domain/groupChannelUserList/types.ts +7 -2
  369. package/src/domain/userList/component/UserListList.tsx +5 -1
  370. package/src/domain/userList/module/createUserListModule.tsx +3 -1
  371. package/src/fragments/createGroupChannelCreateFragment.tsx +8 -2
  372. package/src/fragments/createGroupChannelFragment.tsx +19 -6
  373. package/src/fragments/createGroupChannelInviteFragment.tsx +2 -2
  374. package/src/fragments/createGroupChannelNotificationsFragment.tsx +27 -0
  375. package/src/fragments/createGroupChannelSettingsFragment.tsx +2 -0
  376. package/src/hooks/useConnection.ts +3 -1
  377. package/src/hooks/useContext.ts +7 -0
  378. package/src/hooks/useKeyboardStatus.ts +54 -0
  379. package/src/hooks/useMentionSuggestion.ts +106 -0
  380. package/src/hooks/useMentionTextInput.ts +139 -0
  381. package/src/index.ts +13 -2
  382. package/src/libs/EmojiManager.ts +80 -0
  383. package/src/libs/MentionConfig.ts +40 -0
  384. package/src/libs/MentionManager.tsx +240 -0
  385. package/src/localization/StringSet.type.ts +38 -8
  386. package/src/types.ts +12 -0
  387. package/src/utils/common.ts +8 -0
  388. package/src/version.ts +1 -1
@@ -0,0 +1,102 @@
1
+ import React, { useCallback, useContext, useReducer, useRef, useState } from 'react';
2
+
3
+ import type { SendbirdBaseChannel, SendbirdBaseMessage } from '@sendbird/uikit-utils';
4
+ import { NOOP } from '@sendbird/uikit-utils';
5
+
6
+ import { ReactionBottomSheets } from '../components/ReactionBottomSheets';
7
+ import { LocalizationContext } from '../contexts/LocalizationCtx';
8
+ import { SendbirdChatContext } from '../contexts/SendbirdChatCtx';
9
+ import { UserProfileContext } from '../contexts/UserProfileCtx';
10
+
11
+ type State = {
12
+ message?: SendbirdBaseMessage;
13
+ channel?: SendbirdBaseChannel;
14
+ };
15
+ export type ReactionContextType = {
16
+ openReactionList(param: Required<State>): void;
17
+ openReactionUserList(param: Required<State> & { focusIndex?: number }): void;
18
+ updateReactionFocusedItem(param?: State): void;
19
+ focusIndex: number;
20
+ } & State;
21
+
22
+ type Props = React.PropsWithChildren<{}>;
23
+
24
+ export const ReactionContext = React.createContext<ReactionContextType | null>(null);
25
+ export const ReactionProvider = ({ children }: Props) => {
26
+ const chatCtx = useContext(SendbirdChatContext);
27
+ const localizationCtx = useContext(LocalizationContext);
28
+ const userProfileCtx = useContext(UserProfileContext);
29
+ if (!chatCtx) throw new Error('SendbirdChatContext is not provided');
30
+ if (!localizationCtx) throw new Error('LocalizationContext is not provided');
31
+ if (!userProfileCtx) throw new Error('UserProfileContext is not provided');
32
+
33
+ const [state, setState] = useReducer((prev: State, next: State) => ({ ...prev, ...next }), {});
34
+ const [reactionListVisible, setReactionListVisible] = useState(false);
35
+ const [reactionUserListVisible, setReactionUserListVisible] = useState(false);
36
+ const [reactionUserListFocusIndex, setReactionUserListFocusIndex] = useState(0);
37
+
38
+ const closeResolver = useRef<() => void>(NOOP);
39
+
40
+ const openReactionList: ReactionContextType['openReactionList'] = useCallback((params) => {
41
+ setState(params);
42
+ setReactionListVisible(true);
43
+ }, []);
44
+
45
+ const openReactionUserList: ReactionContextType['openReactionUserList'] = useCallback(
46
+ ({ channel, message, focusIndex = 0 }) => {
47
+ setState({ channel, message });
48
+ setReactionUserListFocusIndex(focusIndex);
49
+ setReactionUserListVisible(true);
50
+ },
51
+ [],
52
+ );
53
+
54
+ const updateReactionFocusedItem: ReactionContextType['updateReactionFocusedItem'] = useCallback((params) => {
55
+ if (params) setState(params);
56
+ else setState({});
57
+ }, []);
58
+
59
+ const createOnCloseWithResolver = (callback: () => void) => {
60
+ return () => {
61
+ return new Promise<void>((resolve) => {
62
+ closeResolver.current = resolve;
63
+ callback();
64
+ });
65
+ };
66
+ };
67
+
68
+ const reactionCtx = {
69
+ ...state,
70
+ openReactionList,
71
+ openReactionUserList,
72
+ updateReactionFocusedItem,
73
+ focusIndex: reactionUserListFocusIndex,
74
+ };
75
+
76
+ const sheetProps = {
77
+ chatCtx,
78
+ reactionCtx,
79
+ localizationCtx,
80
+ userProfileCtx,
81
+ onDismiss: () => {
82
+ setState({});
83
+ closeResolver.current?.();
84
+ },
85
+ };
86
+
87
+ return (
88
+ <ReactionContext.Provider value={reactionCtx}>
89
+ {children}
90
+ <ReactionBottomSheets.UserList
91
+ {...sheetProps}
92
+ visible={reactionUserListVisible}
93
+ onClose={createOnCloseWithResolver(() => setReactionUserListVisible(false))}
94
+ />
95
+ <ReactionBottomSheets.ReactionList
96
+ {...sheetProps}
97
+ visible={reactionListVisible}
98
+ onClose={createOnCloseWithResolver(() => setReactionListVisible(false))}
99
+ />
100
+ </ReactionContext.Provider>
101
+ );
102
+ };
@@ -10,6 +10,8 @@ import type {
10
10
  } from '@sendbird/uikit-utils';
11
11
  import { confirmAndMarkAsDelivered, useForceUpdate } from '@sendbird/uikit-utils';
12
12
 
13
+ import type EmojiManager from '../libs/EmojiManager';
14
+ import type MentionManager from '../libs/MentionManager';
13
15
  import type { FileType } from '../platform/types';
14
16
 
15
17
  export interface UIKitFeaturesInSendbirdChatContext {
@@ -17,14 +19,19 @@ export interface UIKitFeaturesInSendbirdChatContext {
17
19
  enableChannelListTypingIndicator: boolean;
18
20
  enableChannelListMessageReceiptStatus: boolean;
19
21
  enableUseUserIdForNickname: boolean;
22
+ enableUserMention: boolean;
20
23
  }
21
24
 
22
25
  interface Props extends UIKitFeaturesInSendbirdChatContext, React.PropsWithChildren {
23
26
  sdkInstance: SendbirdChatSDK;
27
+ emojiManager: EmojiManager;
28
+ mentionManager: MentionManager;
24
29
  }
25
30
 
26
31
  type Context = {
27
32
  sdk: SendbirdChatSDK;
33
+ emojiManager: EmojiManager;
34
+ mentionManager: MentionManager;
28
35
  currentUser?: SendbirdUser;
29
36
  setCurrentUser: React.Dispatch<React.SetStateAction<SendbirdUser | undefined>>;
30
37
 
@@ -38,6 +45,7 @@ type Context = {
38
45
  channelListTypingIndicatorEnabled: boolean;
39
46
  channelListMessageReceiptStatusEnabled: boolean;
40
47
  useUserIdForNicknameEnabled: boolean;
48
+ userMentionEnabled: boolean;
41
49
 
42
50
  // Sendbird application features
43
51
  deliveryReceiptEnabled: boolean;
@@ -51,10 +59,13 @@ export const SendbirdChatContext = React.createContext<Context | null>(null);
51
59
  export const SendbirdChatProvider = ({
52
60
  children,
53
61
  sdkInstance,
62
+ emojiManager,
63
+ mentionManager,
54
64
  enableAutoPushTokenRegistration,
55
65
  enableChannelListMessageReceiptStatus,
56
66
  enableChannelListTypingIndicator,
57
67
  enableUseUserIdForNickname,
68
+ enableUserMention,
58
69
  }: Props) => {
59
70
  const [currentUser, _setCurrentUser] = useState<SendbirdUser>();
60
71
  const forceUpdate = useForceUpdate();
@@ -72,9 +83,17 @@ export const SendbirdChatProvider = ({
72
83
 
73
84
  if (!user) throw new Error('Current user is not defined, please connect using `useConnection()` hook first');
74
85
 
75
- const params: SendbirdUserUpdateParams = { nickname };
86
+ const params: SendbirdUserUpdateParams = {};
76
87
 
77
- if (typeof profile === 'string') {
88
+ if (!nickname) {
89
+ params.nickname = user.nickname;
90
+ } else {
91
+ params.nickname = nickname;
92
+ }
93
+
94
+ if (!profile) {
95
+ params.profileUrl = user.profileUrl;
96
+ } else if (typeof profile === 'string') {
78
97
  params.profileUrl = profile;
79
98
  } else if (typeof profile === 'object') {
80
99
  params.profileImage = profile;
@@ -110,6 +129,8 @@ export const SendbirdChatProvider = ({
110
129
 
111
130
  const value: Context = {
112
131
  sdk: sdkInstance,
132
+ emojiManager,
133
+ mentionManager,
113
134
  currentUser,
114
135
  setCurrentUser,
115
136
 
@@ -122,6 +143,7 @@ export const SendbirdChatProvider = ({
122
143
  channelListTypingIndicatorEnabled: enableChannelListTypingIndicator,
123
144
  channelListMessageReceiptStatusEnabled: enableChannelListMessageReceiptStatus,
124
145
  useUserIdForNicknameEnabled: enableUseUserIdForNickname,
146
+ userMentionEnabled: enableUserMention,
125
147
  },
126
148
  };
127
149
 
@@ -29,6 +29,8 @@ type Props = React.PropsWithChildren<{
29
29
  onBeforeCreateChannel?: OnBeforeCreateChannel;
30
30
  }>;
31
31
 
32
+ let WARN_onCreateChannel = false;
33
+
32
34
  export const UserProfileContext = React.createContext<UserProfileContextType | null>(null);
33
35
  export const UserProfileProvider = ({ children, onCreateChannel, onBeforeCreateChannel = PASS }: Props) => {
34
36
  const chatContext = useContext(SendbirdChatContext);
@@ -37,6 +39,13 @@ export const UserProfileProvider = ({ children, onCreateChannel, onBeforeCreateC
37
39
  if (!chatContext) throw new Error('SendbirdChatContext is not provided');
38
40
  if (!localizationContext) throw new Error('LocalizationContext is not provided');
39
41
 
42
+ if (__DEV__ && !WARN_onCreateChannel && !onCreateChannel) {
43
+ Logger.warn(
44
+ 'You should pass `userProfile.onCreateChannel` prop to SendbirdUIKitContainer if want to use message in a user profile',
45
+ );
46
+ WARN_onCreateChannel = true;
47
+ }
48
+
40
49
  const { bottom, left, right } = useSafeAreaInsets();
41
50
 
42
51
  const [user, setUser] = useState<SendbirdUser | SendbirdMember>();
@@ -1,60 +1,86 @@
1
- import React, { useEffect, useRef } from 'react';
2
- import { Platform, TextInput as RNTextInput, View } from 'react-native';
1
+ import React, { forwardRef } from 'react';
2
+ import {
3
+ NativeSyntheticEvent,
4
+ Platform,
5
+ TextInput as RNTextInput,
6
+ TextInputSelectionChangeEventData,
7
+ View,
8
+ } from 'react-native';
3
9
 
10
+ import { MentionType } from '@sendbird/chat/message';
4
11
  import { Button, TextInput, createStyleSheet, useToast } from '@sendbird/uikit-react-native-foundation';
5
12
  import type { SendbirdFileMessage, SendbirdUserMessage } from '@sendbird/uikit-utils';
6
13
 
7
- import { useLocalization } from '../../../../hooks/useContext';
14
+ import { useLocalization, useSendbirdChat } from '../../../../hooks/useContext';
15
+ import type { MentionedUser } from '../../../../types';
8
16
  import type { GroupChannelProps } from '../../types';
9
17
 
10
18
  type EditInputProps = GroupChannelProps['Input'] & {
11
19
  text: string;
12
- setText: (val: string) => void;
13
- editMessage: SendbirdUserMessage | SendbirdFileMessage;
14
- setEditMessage: (msg?: SendbirdUserMessage | SendbirdFileMessage) => void;
20
+ onChangeText: (val: string) => void;
21
+ messageToEdit: SendbirdUserMessage | SendbirdFileMessage;
22
+ setMessageToEdit: (msg?: SendbirdUserMessage | SendbirdFileMessage) => void;
23
+ onSelectionChange: (e: NativeSyntheticEvent<TextInputSelectionChangeEventData>) => void;
15
24
  disabled: boolean;
25
+ autoFocus: boolean;
26
+ mentionedUsers: MentionedUser[];
16
27
  };
17
28
 
18
- const AUTO_FOCUS = Platform.select({ ios: false, android: true, default: false });
19
- const EditInput = ({ text, setText, editMessage, setEditMessage, onUpdateUserMessage, disabled }: EditInputProps) => {
29
+ const EditInput = forwardRef<RNTextInput, EditInputProps>(function EditInput(
30
+ {
31
+ text,
32
+ onChangeText,
33
+ messageToEdit,
34
+ setMessageToEdit,
35
+ onUpdateUserMessage,
36
+ onSelectionChange,
37
+ disabled,
38
+ autoFocus,
39
+ mentionedUsers,
40
+ },
41
+ ref,
42
+ ) {
43
+ const { mentionManager } = useSendbirdChat();
20
44
  const { STRINGS } = useLocalization();
21
- const inputRef = useRef<RNTextInput>(null);
22
45
  const toast = useToast();
23
46
 
24
- useEffect(() => {
25
- if (editMessage.isUserMessage()) {
26
- setText(editMessage.message ?? '');
27
-
28
- if (!AUTO_FOCUS) setTimeout(() => inputRef.current?.focus(), 500);
29
- }
30
- }, [editMessage]);
31
-
32
47
  const onPressCancel = () => {
33
- setEditMessage();
34
- setText('');
48
+ setMessageToEdit();
49
+ onChangeText('');
35
50
  };
36
51
 
37
52
  const onPressSave = () => {
38
- if (editMessage.isUserMessage()) {
39
- onUpdateUserMessage(text, editMessage).catch(() => toast.show(STRINGS.TOAST.UPDATE_MSG_ERROR, 'error'));
53
+ if (messageToEdit.isUserMessage()) {
54
+ const mention = {
55
+ userIds: mentionedUsers.map((it) => it.user.userId),
56
+ messageTemplate: mentionManager.textToMentionedMessageTemplate(text, mentionedUsers),
57
+ type: MentionType.USERS,
58
+ };
59
+
60
+ onUpdateUserMessage(text, messageToEdit, mention).catch(() =>
61
+ toast.show(STRINGS.TOAST.UPDATE_MSG_ERROR, 'error'),
62
+ );
40
63
  }
41
- setEditMessage();
42
- setText('');
64
+ setMessageToEdit();
65
+ onChangeText('');
43
66
  };
44
67
 
45
68
  return (
46
69
  <View style={styles.editInputContainer}>
47
70
  <View style={styles.inputWrapper}>
48
71
  <TextInput
49
- editable={!disabled}
50
- autoFocus={AUTO_FOCUS}
51
- ref={inputRef}
72
+ ref={ref}
52
73
  multiline
53
- value={text}
54
- onChangeText={setText}
74
+ disableFullscreenUI
75
+ editable={!disabled}
76
+ autoFocus={autoFocus}
77
+ onChangeText={onChangeText}
55
78
  style={styles.input}
56
79
  placeholder={STRINGS.GROUP_CHANNEL.INPUT_PLACEHOLDER_ACTIVE}
57
- />
80
+ onSelectionChange={onSelectionChange}
81
+ >
82
+ {mentionManager.textToMentionedComponents(text, mentionedUsers)}
83
+ </TextInput>
58
84
  </View>
59
85
  <View style={{ marginTop: 8, flexDirection: 'row' }}>
60
86
  <Button variant={'text'} onPress={onPressCancel}>
@@ -67,7 +93,7 @@ const EditInput = ({ text, setText, editMessage, setEditMessage, onUpdateUserMes
67
93
  </View>
68
94
  </View>
69
95
  );
70
- };
96
+ });
71
97
 
72
98
  const styles = createStyleSheet({
73
99
  editInputContainer: {
@@ -1,6 +1,14 @@
1
- import React from 'react';
2
- import { Platform, TouchableOpacity, View } from 'react-native';
1
+ import React, { forwardRef } from 'react';
2
+ import {
3
+ NativeSyntheticEvent,
4
+ Platform,
5
+ TextInput as RNTextInput,
6
+ TextInputSelectionChangeEventData,
7
+ TouchableOpacity,
8
+ View,
9
+ } from 'react-native';
3
10
 
11
+ import { MentionType } from '@sendbird/chat/message';
4
12
  import {
5
13
  Icon,
6
14
  TextInput,
@@ -12,27 +20,36 @@ import {
12
20
  } from '@sendbird/uikit-react-native-foundation';
13
21
  import { conditionChaining } from '@sendbird/uikit-utils';
14
22
 
15
- import { useLocalization, usePlatformService } from '../../../../hooks/useContext';
23
+ import { useLocalization, usePlatformService, useSendbirdChat } from '../../../../hooks/useContext';
16
24
  import SBUError from '../../../../libs/SBUError';
17
25
  import SBUUtils from '../../../../libs/SBUUtils';
26
+ import type { MentionedUser } from '../../../../types';
18
27
  import type { GroupChannelProps } from '../../types';
19
28
 
20
29
  type SendInputProps = GroupChannelProps['Input'] & {
21
30
  text: string;
22
- setText: (val: string) => void;
31
+ onChangeText: (val: string) => void;
23
32
  frozen: boolean;
24
33
  muted: boolean;
25
34
  disabled: boolean;
35
+ onSelectionChange: (e: NativeSyntheticEvent<TextInputSelectionChangeEventData>) => void;
36
+ mentionedUsers: MentionedUser[];
26
37
  };
27
- const SendInput = ({
28
- onSendUserMessage,
29
- onSendFileMessage,
30
- text,
31
- setText,
32
- disabled,
33
- frozen,
34
- muted,
35
- }: SendInputProps) => {
38
+ const SendInput = forwardRef<RNTextInput, SendInputProps>(function SendInput(
39
+ {
40
+ onSendUserMessage,
41
+ onSendFileMessage,
42
+ text,
43
+ onChangeText,
44
+ disabled,
45
+ frozen,
46
+ muted,
47
+ onSelectionChange,
48
+ mentionedUsers,
49
+ },
50
+ ref,
51
+ ) {
52
+ const { mentionManager } = useSendbirdChat();
36
53
  const { STRINGS } = useLocalization();
37
54
  const { fileService } = usePlatformService();
38
55
  const { colors } = useUIKitTheme();
@@ -41,9 +58,16 @@ const SendInput = ({
41
58
  const toast = useToast();
42
59
 
43
60
  const onPressSend = () => {
44
- onSendUserMessage(text).catch(() => toast.show(STRINGS.TOAST.SEND_MSG_ERROR, 'error'));
45
- setText('');
61
+ const mention = {
62
+ userIds: mentionedUsers.map((it) => it.user.userId),
63
+ messageTemplate: mentionManager.textToMentionedMessageTemplate(text, mentionedUsers),
64
+ type: MentionType.USERS,
65
+ };
66
+
67
+ onSendUserMessage(text, mention).catch(() => toast.show(STRINGS.TOAST.SEND_MSG_ERROR, 'error'));
68
+ onChangeText('');
46
69
  };
70
+
47
71
  const onPressAttachment = () => {
48
72
  openSheet({
49
73
  sheetItems: [
@@ -124,10 +148,12 @@ const SendInput = ({
124
148
  />
125
149
  </TouchableOpacity>
126
150
  <TextInput
151
+ ref={ref}
127
152
  multiline
153
+ disableFullscreenUI
154
+ onSelectionChange={onSelectionChange}
128
155
  editable={!disabled}
129
- value={text}
130
- onChangeText={setText}
156
+ onChangeText={onChangeText}
131
157
  style={styles.input}
132
158
  placeholder={conditionChaining(
133
159
  [frozen, muted],
@@ -137,7 +163,10 @@ const SendInput = ({
137
163
  STRINGS.GROUP_CHANNEL.INPUT_PLACEHOLDER_ACTIVE,
138
164
  ],
139
165
  )}
140
- />
166
+ >
167
+ {mentionManager.textToMentionedComponents(text, mentionedUsers)}
168
+ </TextInput>
169
+
141
170
  {Boolean(text.trim()) && (
142
171
  <TouchableOpacity onPress={onPressSend} disabled={disabled}>
143
172
  <Icon
@@ -150,7 +179,7 @@ const SendInput = ({
150
179
  )}
151
180
  </View>
152
181
  );
153
- };
182
+ });
154
183
 
155
184
  const styles = createStyleSheet({
156
185
  sendInputContainer: {
@@ -1,75 +1,169 @@
1
- import React, { useContext, useEffect, useRef, useState } from 'react';
2
- import { KeyboardAvoidingView, Platform, View } from 'react-native';
1
+ import React, { MutableRefObject, useContext, useEffect, useRef, useState } from 'react';
2
+ import { KeyboardAvoidingView, Platform, TextInput, View } from 'react-native';
3
3
  import { useSafeAreaInsets } from 'react-native-safe-area-context';
4
4
 
5
- import { useUIKitTheme } from '@sendbird/uikit-react-native-foundation';
6
- import { getGroupChannelChatAvailableState, useIIFE } from '@sendbird/uikit-utils';
5
+ import { createStyleSheet, useUIKitTheme } from '@sendbird/uikit-react-native-foundation';
6
+ import {
7
+ SendbirdFileMessage,
8
+ SendbirdGroupChannel,
9
+ SendbirdUserMessage,
10
+ getGroupChannelChatAvailableState,
11
+ replace,
12
+ useIIFE,
13
+ } from '@sendbird/uikit-utils';
7
14
 
15
+ import { useSendbirdChat } from '../../../../hooks/useContext';
16
+ import useMentionTextInput from '../../../../hooks/useMentionTextInput';
8
17
  import { GroupChannelContexts } from '../../module/moduleContext';
9
18
  import type { GroupChannelProps } from '../../types';
10
19
  import EditInput from './EditInput';
11
20
  import SendInput from './SendInput';
12
21
 
22
+ const AUTO_FOCUS = Platform.select({ ios: false, android: true, default: false });
13
23
  const KEYBOARD_AVOID_VIEW_BEHAVIOR = Platform.select({ ios: 'padding' as const, default: undefined });
24
+
25
+ // FIXME(iOS): Dynamic style does not work properly when typing the CJK. (https://github.com/facebook/react-native/issues/26107)
26
+ // To workaround temporarily, change the key for re-mount the component.
27
+ const GET_INPUT_KEY = (shouldReset: boolean) => (shouldReset ? 'uikit-input-clear' : 'uikit-input');
28
+
29
+ // TODO: Refactor 'Edit' mode to clearly
14
30
  const GroupChannelInput = (props: GroupChannelProps['Input']) => {
15
- const { left, right, bottom } = useSafeAreaInsets();
31
+ const { top, left, right, bottom } = useSafeAreaInsets();
16
32
  const { colors } = useUIKitTheme();
17
- const { channel, editMessage, setEditMessage, keyboardAvoidOffset = 0 } = useContext(GroupChannelContexts.Fragment);
33
+ const { features, mentionManager } = useSendbirdChat();
34
+ const {
35
+ channel,
36
+ messageToEdit,
37
+ setMessageToEdit,
38
+ keyboardAvoidOffset = 0,
39
+ } = useContext(GroupChannelContexts.Fragment);
18
40
 
19
- const [text, setText] = useState('');
20
- const textTmpRef = useRef('');
21
41
  const chatAvailableState = getGroupChannelChatAvailableState(channel);
42
+ const mentionAvailable = features.userMentionEnabled && channel.isGroupChannel() && !channel.isBroadcast;
43
+ const inputMode = useIIFE(() => {
44
+ if (!messageToEdit) return 'send';
45
+ if (messageToEdit.isFileMessage()) return 'send';
46
+ return 'edit';
47
+ });
48
+
49
+ const [inputHeight, setInputHeight] = useState(styles.inputDefault.height);
50
+
51
+ const { selection, onSelectionChange, textInputRef, text, onChangeText, mentionedUsers } = useMentionTextInput({
52
+ messageToEdit: messageToEdit,
53
+ });
54
+
55
+ useTypingTrigger(text, channel);
56
+ useTextPersistenceOnDisabled(text, onChangeText, chatAvailableState.disabled);
57
+ useAutoFocusOnEditMode(textInputRef, messageToEdit);
58
+
59
+ const onPressToMention: GroupChannelProps['SuggestedMentionList']['onPressToMention'] = (user, searchStringRange) => {
60
+ const mentionedMessageText = mentionManager.asMentionedMessageText(user, true);
61
+ const range = { start: searchStringRange.start, end: searchStringRange.start + mentionedMessageText.length - 1 };
62
+
63
+ onChangeText(replace(text, searchStringRange.start, searchStringRange.end, mentionedMessageText), { user, range });
64
+ };
65
+
66
+ if (!props.shouldRenderInput) {
67
+ return <SafeAreaBottom height={bottom} />;
68
+ }
69
+
70
+ return (
71
+ <>
72
+ <KeyboardAvoidingView
73
+ keyboardVerticalOffset={-bottom + keyboardAvoidOffset}
74
+ behavior={KEYBOARD_AVOID_VIEW_BEHAVIOR}
75
+ >
76
+ <View style={{ paddingLeft: left, paddingRight: right, backgroundColor: colors.background }}>
77
+ <View onLayout={(e) => setInputHeight(e.nativeEvent.layout.height)} style={styles.inputContainer}>
78
+ {inputMode === 'send' && (
79
+ <SendInput
80
+ {...props}
81
+ {...chatAvailableState}
82
+ key={GET_INPUT_KEY(mentionedUsers.length === 0)}
83
+ ref={textInputRef as never}
84
+ text={text}
85
+ onChangeText={onChangeText}
86
+ onSelectionChange={onSelectionChange}
87
+ mentionedUsers={mentionedUsers}
88
+ />
89
+ )}
90
+ {inputMode === 'edit' && messageToEdit && (
91
+ <EditInput
92
+ {...props}
93
+ key={GET_INPUT_KEY(mentionedUsers.length === 0)}
94
+ ref={textInputRef as never}
95
+ autoFocus={AUTO_FOCUS}
96
+ text={text}
97
+ onChangeText={onChangeText}
98
+ messageToEdit={messageToEdit}
99
+ setMessageToEdit={setMessageToEdit}
100
+ disabled={chatAvailableState.disabled}
101
+ onSelectionChange={onSelectionChange}
102
+ mentionedUsers={mentionedUsers}
103
+ />
104
+ )}
105
+ </View>
106
+ <SafeAreaBottom height={bottom} />
107
+ </View>
108
+ </KeyboardAvoidingView>
109
+ {mentionAvailable && (
110
+ <props.SuggestedMentionList
111
+ text={text}
112
+ selection={selection}
113
+ inputHeight={inputHeight}
114
+ topInset={top}
115
+ bottomInset={bottom}
116
+ onPressToMention={onPressToMention}
117
+ mentionedUsers={mentionedUsers}
118
+ />
119
+ )}
120
+ </>
121
+ );
122
+ };
22
123
 
124
+ const useTypingTrigger = (text: string, channel: SendbirdGroupChannel) => {
23
125
  useEffect(() => {
24
126
  if (text.length === 0) channel.endTyping();
25
127
  else channel.startTyping();
26
128
  }, [text]);
129
+ };
130
+
131
+ const useTextPersistenceOnDisabled = (text: string, setText: (val: string) => void, chatDisabled: boolean) => {
132
+ const textTmpRef = useRef('');
27
133
 
28
134
  useEffect(() => {
29
- if (chatAvailableState.disabled) {
135
+ if (chatDisabled) {
30
136
  textTmpRef.current = text;
31
137
  setText('');
32
138
  } else {
33
139
  setText(textTmpRef.current);
34
140
  }
35
- }, [chatAvailableState.disabled]);
36
-
37
- const inputMode = useIIFE(() => {
38
- if (!editMessage) return 'send';
39
- if (editMessage.isFileMessage()) return 'send';
40
- return 'edit';
41
- });
42
-
43
- if (!props.shouldRenderInput) {
44
- return <SafeAreaBottom height={bottom} />;
45
- }
141
+ }, [chatDisabled]);
142
+ };
46
143
 
47
- return (
48
- <KeyboardAvoidingView
49
- keyboardVerticalOffset={-bottom + keyboardAvoidOffset}
50
- behavior={KEYBOARD_AVOID_VIEW_BEHAVIOR}
51
- >
52
- <View style={{ paddingLeft: left, paddingRight: right, backgroundColor: colors.background }}>
53
- <View style={{ justifyContent: 'center', width: '100%' }}>
54
- {inputMode === 'send' && <SendInput {...props} text={text} setText={setText} {...chatAvailableState} />}
55
- {inputMode === 'edit' && editMessage && (
56
- <EditInput
57
- {...props}
58
- text={text}
59
- setText={setText}
60
- editMessage={editMessage}
61
- setEditMessage={setEditMessage}
62
- disabled={chatAvailableState.disabled}
63
- />
64
- )}
65
- </View>
66
- <SafeAreaBottom height={bottom} />
67
- </View>
68
- </KeyboardAvoidingView>
69
- );
144
+ const useAutoFocusOnEditMode = (
145
+ textInputRef: MutableRefObject<TextInput | undefined>,
146
+ messageToEdit?: SendbirdUserMessage | SendbirdFileMessage,
147
+ ) => {
148
+ useEffect(() => {
149
+ if (messageToEdit?.isUserMessage()) {
150
+ if (!AUTO_FOCUS) setTimeout(() => textInputRef.current?.focus(), 500);
151
+ }
152
+ }, [messageToEdit]);
70
153
  };
154
+
71
155
  const SafeAreaBottom = ({ height }: { height: number }) => {
72
156
  return <View style={{ height }} />;
73
157
  };
74
158
 
159
+ const styles = createStyleSheet({
160
+ inputContainer: {
161
+ justifyContent: 'center',
162
+ width: '100%',
163
+ },
164
+ inputDefault: {
165
+ height: 56,
166
+ },
167
+ });
168
+
75
169
  export default React.memo(GroupChannelInput);