@azure/communication-react 1.14.0-alpha-202403160012 → 1.14.0-alpha-202403190012

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 (68) hide show
  1. package/dist/communication-react.d.ts +1 -1
  2. package/dist/dist-cjs/communication-react/{RichTextSendBoxWrapper-BCdGFyar.js → RichTextSendBoxWrapper-Bw9vaT1f.js} +2 -2
  3. package/dist/dist-cjs/communication-react/{RichTextSendBoxWrapper-BCdGFyar.js.map → RichTextSendBoxWrapper-Bw9vaT1f.js.map} +1 -1
  4. package/dist/dist-cjs/communication-react/{index-BfFCLrEO.js → index-DyaTe_Bz.js} +738 -562
  5. package/dist/dist-cjs/communication-react/index-DyaTe_Bz.js.map +1 -0
  6. package/dist/dist-cjs/communication-react/index.js +1 -1
  7. package/dist/dist-esm/acs-ui-common/src/telemetryVersion.js +1 -1
  8. package/dist/dist-esm/acs-ui-common/src/telemetryVersion.js.map +1 -1
  9. package/dist/dist-esm/calling-component-bindings/src/handlers/createCommonHandlers.js +2 -2
  10. package/dist/dist-esm/calling-component-bindings/src/handlers/createCommonHandlers.js.map +1 -1
  11. package/dist/dist-esm/calling-component-bindings/src/hooks/useSelector.d.ts +1 -1
  12. package/dist/dist-esm/calling-component-bindings/src/utils/videoGalleryUtils.d.ts +1 -1
  13. package/dist/dist-esm/calling-component-bindings/src/utils/videoGalleryUtils.js +14 -12
  14. package/dist/dist-esm/calling-component-bindings/src/utils/videoGalleryUtils.js.map +1 -1
  15. package/dist/dist-esm/calling-component-bindings/src/videoGallerySelector.js +2 -0
  16. package/dist/dist-esm/calling-component-bindings/src/videoGallerySelector.js.map +1 -1
  17. package/dist/dist-esm/chat-component-bindings/src/hooks/useSelector.d.ts +1 -1
  18. package/dist/dist-esm/chat-stateful-client/src/ChatContext.js +1 -1
  19. package/dist/dist-esm/chat-stateful-client/src/ChatContext.js.map +1 -1
  20. package/dist/dist-esm/chat-stateful-client/src/ResourceDownloadQueue.js +1 -1
  21. package/dist/dist-esm/chat-stateful-client/src/ResourceDownloadQueue.js.map +1 -1
  22. package/dist/dist-esm/chat-stateful-client/src/iterators/createDecoratedIterator.d.ts +2 -2
  23. package/dist/dist-esm/communication-react/src/mergedHooks.d.ts +1 -1
  24. package/dist/dist-esm/react-components/src/components/ChatMessage/ChatMessageComponentWrapper.js +12 -3
  25. package/dist/dist-esm/react-components/src/components/ChatMessage/ChatMessageComponentWrapper.js.map +1 -1
  26. package/dist/dist-esm/react-components/src/components/ChatMessage/MessageComponents/ChatMessageComponentAsMessageBubble.d.ts +46 -0
  27. package/dist/dist-esm/react-components/src/components/ChatMessage/MessageComponents/ChatMessageComponentAsMessageBubble.js +91 -0
  28. package/dist/dist-esm/react-components/src/components/ChatMessage/MessageComponents/ChatMessageComponentAsMessageBubble.js.map +1 -0
  29. package/dist/dist-esm/react-components/src/components/ChatMessage/{FluentChatMessageComponentWrapper.d.ts → MessageComponents/FluentChatMessageComponent.d.ts} +6 -7
  30. package/dist/dist-esm/react-components/src/components/ChatMessage/MessageComponents/FluentChatMessageComponent.js +122 -0
  31. package/dist/dist-esm/react-components/src/components/ChatMessage/MessageComponents/FluentChatMessageComponent.js.map +1 -0
  32. package/dist/dist-esm/react-components/src/components/ChatMessage/{ChatMessageComponent.d.ts → MyMessageComponents/ChatMyMessageComponent.d.ts} +9 -9
  33. package/dist/dist-esm/react-components/src/components/ChatMessage/{ChatMessageComponent.js → MyMessageComponents/ChatMyMessageComponent.js} +5 -5
  34. package/dist/dist-esm/react-components/src/components/ChatMessage/MyMessageComponents/ChatMyMessageComponent.js.map +1 -0
  35. package/dist/dist-esm/react-components/src/components/ChatMessage/{ChatMessageComponentAsMessageBubble.d.ts → MyMessageComponents/ChatMyMessageComponentAsMessageBubble.d.ts} +10 -19
  36. package/dist/dist-esm/react-components/src/components/ChatMessage/MyMessageComponents/ChatMyMessageComponentAsMessageBubble.js +162 -0
  37. package/dist/dist-esm/react-components/src/components/ChatMessage/MyMessageComponents/ChatMyMessageComponentAsMessageBubble.js.map +1 -0
  38. package/dist/dist-esm/react-components/src/components/ChatMessage/MyMessageComponents/FluentChatMyMessageComponent.d.ts +12 -0
  39. package/dist/dist-esm/react-components/src/components/ChatMessage/{FluentChatMessageComponentWrapper.js → MyMessageComponents/FluentChatMyMessageComponent.js} +12 -54
  40. package/dist/dist-esm/react-components/src/components/ChatMessage/MyMessageComponents/FluentChatMyMessageComponent.js.map +1 -0
  41. package/dist/dist-esm/react-components/src/components/TextFieldWithMention/TextFieldWithMention.js +3 -3
  42. package/dist/dist-esm/react-components/src/components/TextFieldWithMention/TextFieldWithMention.js.map +1 -1
  43. package/dist/dist-esm/react-components/src/components/VideoGallery/utils/videoGalleryLayoutUtils.d.ts +3 -1
  44. package/dist/dist-esm/react-components/src/components/VideoGallery/utils/videoGalleryLayoutUtils.js +98 -118
  45. package/dist/dist-esm/react-components/src/components/VideoGallery/utils/videoGalleryLayoutUtils.js.map +1 -1
  46. package/dist/dist-esm/react-components/src/components/utils/ChatMessageComponentUtils.d.ts +29 -0
  47. package/dist/dist-esm/react-components/src/components/utils/ChatMessageComponentUtils.js +92 -0
  48. package/dist/dist-esm/react-components/src/components/utils/ChatMessageComponentUtils.js.map +1 -0
  49. package/dist/dist-esm/react-components/src/gallery/dominantSpeaker.d.ts +2 -3
  50. package/dist/dist-esm/react-components/src/gallery/dominantSpeaker.js +2 -2
  51. package/dist/dist-esm/react-components/src/gallery/dominantSpeaker.js.map +1 -1
  52. package/dist/dist-esm/react-composites/src/composites/CallComposite/adapter/AzureCommunicationCallAdapter.js +5 -5
  53. package/dist/dist-esm/react-composites/src/composites/CallComposite/adapter/AzureCommunicationCallAdapter.js.map +1 -1
  54. package/dist/dist-esm/react-composites/src/composites/CallComposite/hooks/useAdaptedSelector.d.ts +2 -2
  55. package/dist/dist-esm/react-composites/src/composites/CallComposite/hooks/usePropsFor.d.ts +1 -1
  56. package/dist/dist-esm/react-composites/src/composites/CallComposite/hooks/useSelector.d.ts +1 -1
  57. package/dist/dist-esm/react-composites/src/composites/CallWithChatComposite/adapter/AzureCommunicationCallWithChatAdapter.js +4 -4
  58. package/dist/dist-esm/react-composites/src/composites/CallWithChatComposite/adapter/AzureCommunicationCallWithChatAdapter.js.map +1 -1
  59. package/dist/dist-esm/react-composites/src/composites/ChatComposite/adapter/AzureCommunicationChatAdapter.js +4 -4
  60. package/dist/dist-esm/react-composites/src/composites/ChatComposite/adapter/AzureCommunicationChatAdapter.js.map +1 -1
  61. package/dist/dist-esm/react-composites/src/composites/ChatComposite/hooks/useAdaptedSelector.d.ts +2 -2
  62. package/dist/dist-esm/react-composites/src/composites/ChatComposite/hooks/useSelector.d.ts +1 -1
  63. package/package.json +1 -1
  64. package/dist/dist-cjs/communication-react/index-BfFCLrEO.js.map +0 -1
  65. package/dist/dist-esm/react-components/src/components/ChatMessage/ChatMessageComponent.js.map +0 -1
  66. package/dist/dist-esm/react-components/src/components/ChatMessage/ChatMessageComponentAsMessageBubble.js +0 -218
  67. package/dist/dist-esm/react-components/src/components/ChatMessage/ChatMessageComponentAsMessageBubble.js.map +0 -1
  68. package/dist/dist-esm/react-components/src/components/ChatMessage/FluentChatMessageComponentWrapper.js.map +0 -1
@@ -173,7 +173,7 @@ function getDefaultExportFromCjs (x) {
173
173
  // Copyright (c) Microsoft Corporation.
174
174
  // Licensed under the MIT License.
175
175
  // GENERATED FILE. DO NOT EDIT MANUALLY.
176
- var telemetryVersion = '1.14.0-alpha-202403160012';
176
+ var telemetryVersion = '1.14.0-alpha-202403190012';
177
177
 
178
178
 
179
179
  var telemetryVersion$1 = /*@__PURE__*/getDefaultExportFromCjs(telemetryVersion);
@@ -964,7 +964,7 @@ const createDefaultCommonCallingHandlers = memoizeOne((callClient, deviceManager
964
964
  const onHangUp = (forEveryone) => __awaiter$Q(void 0, void 0, void 0, function* () { return yield (call === null || call === void 0 ? void 0 : call.hangUp({ forEveryone: forEveryone === true ? true : false })); });
965
965
  /* @conditional-compile-remove(PSTN-calls) */
966
966
  const onToggleHold = () => __awaiter$Q(void 0, void 0, void 0, function* () { return (call === null || call === void 0 ? void 0 : call.state) === 'LocalHold' ? yield (call === null || call === void 0 ? void 0 : call.resume()) : yield (call === null || call === void 0 ? void 0 : call.hold()); });
967
- const onCreateLocalStreamView = (options = { scalingMode: 'Crop', isMirrored: true }) => __awaiter$Q(void 0, void 0, void 0, function* () {
967
+ const onCreateLocalStreamView = (...args_1) => __awaiter$Q(void 0, [...args_1], void 0, function* (options = { scalingMode: 'Crop', isMirrored: true }) {
968
968
  var _d;
969
969
  if (!call || call.localVideoStreams.length === 0) {
970
970
  return;
@@ -980,7 +980,7 @@ const createDefaultCommonCallingHandlers = memoizeOne((callClient, deviceManager
980
980
  const { view } = (_d = (yield callClient.createView(call.id, undefined, localStream, options))) !== null && _d !== void 0 ? _d : {};
981
981
  return view ? { view } : undefined;
982
982
  });
983
- const onCreateRemoteStreamView = (userId, options = { scalingMode: 'Crop' }) => __awaiter$Q(void 0, void 0, void 0, function* () {
983
+ const onCreateRemoteStreamView = (userId_1, ...args_2) => __awaiter$Q(void 0, [userId_1, ...args_2], void 0, function* (userId, options = { scalingMode: 'Crop' }) {
984
984
  if (!call) {
985
985
  return;
986
986
  }
@@ -8334,8 +8334,8 @@ const TextFieldWithMention = (props) => {
8334
8334
  });
8335
8335
  }
8336
8336
  }, [updateSelectionIndexesWithMentionIfNeeded, setSelectionStartValue, setSelectionEndValue]);
8337
- const handleOnChange = React.useCallback(({ currentSelectionEnd, currentSelectionStart, currentTriggerStartIndex, event, htmlTextValue, inputTextValue, previousSelectionEnd, previousSelectionStart, tagsValue, updatedValue }) => __awaiter$D(void 0, void 0, void 0, function* () {
8338
- var _b;
8337
+ const handleOnChange = React.useCallback((_b) => __awaiter$D(void 0, [_b], void 0, function* ({ currentSelectionEnd, currentSelectionStart, currentTriggerStartIndex, event, htmlTextValue, inputTextValue, previousSelectionEnd, previousSelectionStart, tagsValue, updatedValue }) {
8338
+ var _c;
8339
8339
  debouncedQueryUpdate.cancel();
8340
8340
  if (event.currentTarget === null) {
8341
8341
  return;
@@ -8344,7 +8344,7 @@ const TextFieldWithMention = (props) => {
8344
8344
  // onSelect is not called for backspace as selection is not changed and local caret index is outdated
8345
8345
  setCaretIndex(undefined);
8346
8346
  const newValue = updatedValue !== null && updatedValue !== void 0 ? updatedValue : '';
8347
- const triggerText = (_b = mentionLookupOptions === null || mentionLookupOptions === void 0 ? void 0 : mentionLookupOptions.trigger) !== null && _b !== void 0 ? _b : DEFAULT_MENTION_TRIGGER;
8347
+ const triggerText = (_c = mentionLookupOptions === null || mentionLookupOptions === void 0 ? void 0 : mentionLookupOptions.trigger) !== null && _c !== void 0 ? _c : DEFAULT_MENTION_TRIGGER;
8348
8348
  const newTextLength = newValue.length;
8349
8349
  // updating indexes to set between 0 and text length, otherwise selectionRange won't be set correctly
8350
8350
  const currentSelectionEndValue = getValidatedIndexInRange({
@@ -11129,6 +11129,63 @@ function createStyleFromV8Style(v8Style) {
11129
11129
  return result;
11130
11130
  }
11131
11131
 
11132
+ // Copyright (c) Microsoft Corporation.
11133
+ // Licensed under the MIT License.
11134
+ /**
11135
+ * @private
11136
+ */
11137
+ const systemMessageIconStyle = react.mergeStyles({
11138
+ marginInlineEnd: '0.688rem'
11139
+ });
11140
+
11141
+ // Copyright (c) Microsoft Corporation.
11142
+ // Licensed under the MIT License.
11143
+ /**
11144
+ * @private
11145
+ */
11146
+ const SystemMessage = (props) => {
11147
+ const { iconName, content } = props;
11148
+ const Icon = React.createElement(react.FontIcon, { iconName: iconName, className: react.mergeStyles(systemMessageIconStyle) });
11149
+ return (React.createElement(react.Stack, { horizontal: true, className: react.mergeStyles(props === null || props === void 0 ? void 0 : props.containerStyle), tabIndex: 0 },
11150
+ Icon,
11151
+ React.createElement(react.Text, { style: { wordBreak: 'break-word' }, role: "status", title: content, variant: 'small' }, content)));
11152
+ };
11153
+
11154
+ // Copyright (c) Microsoft Corporation.
11155
+ // Licensed under the MIT License.
11156
+ /**
11157
+ * @private
11158
+ */
11159
+ const DefaultSystemMessage = (props) => {
11160
+ var _a;
11161
+ const message = props.message;
11162
+ switch (message.messageType) {
11163
+ case 'system':
11164
+ switch (message.systemMessageType) {
11165
+ case 'content':
11166
+ return (React.createElement(SystemMessage, { iconName: (message.iconName ? message.iconName : ''), content: (_a = message.content) !== null && _a !== void 0 ? _a : '', containerStyle: props === null || props === void 0 ? void 0 : props.messageContainerStyle }));
11167
+ case 'participantAdded':
11168
+ case 'participantRemoved':
11169
+ return (React.createElement(ParticipantSystemMessageComponent, { message: message, style: props.messageContainerStyle, defaultName: props.strings.noDisplayNameSub }));
11170
+ }
11171
+ }
11172
+ return React.createElement(React.Fragment, null);
11173
+ };
11174
+ const ParticipantSystemMessageComponent = ({ message, style, defaultName }) => {
11175
+ const { strings } = useLocale$1();
11176
+ const participantsStr = generateParticipantsStr(message.participants, defaultName);
11177
+ const messageSuffix = message.systemMessageType === 'participantAdded'
11178
+ ? strings.messageThread.participantJoined
11179
+ : strings.messageThread.participantLeft;
11180
+ if (participantsStr !== '') {
11181
+ return (React.createElement(SystemMessage, { iconName: (message.iconName ? message.iconName : ''), content: `${participantsStr} ${messageSuffix}`, containerStyle: style }));
11182
+ }
11183
+ return React.createElement(React.Fragment, null);
11184
+ };
11185
+ const generateParticipantsStr = (participants, defaultName) => participants
11186
+ .map((participant) => `${!participant.displayName || participant.displayName === '' ? defaultName : participant.displayName}`)
11187
+ .join(', ');
11188
+
11132
11189
  // Copyright (c) Microsoft Corporation.
11133
11190
  // Licensed under the MIT License.
11134
11191
  /**
@@ -11224,85 +11281,6 @@ const ChatMessageComponentAsEditBox = (props) => {
11224
11281
  } }, getContent()));
11225
11282
  };
11226
11283
 
11227
- // Copyright (c) Microsoft Corporation.
11228
- // Licensed under the MIT License.
11229
- /**
11230
- * @private
11231
- */
11232
- const formatTimeForChatMessage = (messageDate) => {
11233
- return messageDate.toLocaleTimeString([], { hour: 'numeric', minute: '2-digit' });
11234
- };
11235
- /**
11236
- * @private
11237
- */
11238
- const formatDateForChatMessage = (messageDate) => {
11239
- return messageDate.toLocaleDateString();
11240
- };
11241
- /**
11242
- * Given a message date object in ISO8601 and a current date object, generates a user friendly timestamp text
11243
- * using the system locale.
11244
- * <time in locale format>.
11245
- * Yesterday <time in locale format>.
11246
- * <dateStrings day of week> <time in locale format>.
11247
- * <date in locale format> <time in locale format>.
11248
- *
11249
- * If message is after yesterday, then only show the time.
11250
- * If message is before yesterday and after day before yesterday, then show 'Yesterday' plus the time.
11251
- * If message is before day before yesterday and within the current week, then show 'Monday/Tuesday/etc' plus the time.
11252
- * - We consider start of the week as Sunday. If current day is Sunday, then any time before that is in previous week.
11253
- * If message is in previous or older weeks, then show date string plus the time.
11254
- *
11255
- * @param messageDate - date of message
11256
- * @param currentDate - date used as offset to create the user friendly timestamp (e.g. to create 'Yesterday' instead of an absolute date)
11257
- *
11258
- * @private
11259
- */
11260
- const formatTimestampForChatMessage = (messageDate, todayDate, dateStrings) => {
11261
- // If message was in the same day timestamp string is just the time like '1:30 p.m.'.
11262
- const startOfDay = new Date(todayDate.getFullYear(), todayDate.getMonth(), todayDate.getDate());
11263
- if (messageDate > startOfDay) {
11264
- return formatTimeForChatMessage(messageDate);
11265
- }
11266
- // If message was yesterday then timestamp string is like this 'Yesterday 1:30 p.m.'.
11267
- const yesterdayDate = new Date(todayDate.getFullYear(), todayDate.getMonth(), todayDate.getDate() - 1);
11268
- if (messageDate > yesterdayDate) {
11269
- return dateStrings.yesterday + ' ' + formatTimeForChatMessage(messageDate);
11270
- }
11271
- // If message was before Sunday and today is Sunday (start of week) then timestamp string is like
11272
- // '2021-01-10 1:30 p.m.'.
11273
- const weekDay = todayDate.getDay();
11274
- if (weekDay === 0) {
11275
- return formatDateForChatMessage(messageDate) + ' ' + formatTimeForChatMessage(messageDate);
11276
- }
11277
- // If message was before first day of the week then timestamp string is like Monday 1:30 p.m.
11278
- const firstDayOfTheWeekDate = new Date(todayDate.getFullYear(), todayDate.getMonth(), todayDate.getDate() - weekDay);
11279
- if (messageDate > firstDayOfTheWeekDate) {
11280
- return dayToDayName(messageDate.getDay(), dateStrings) + ' ' + formatTimeForChatMessage(messageDate);
11281
- }
11282
- // If message date is in previous or older weeks then timestamp string is like 2021-01-10 1:30 p.m.
11283
- return formatDateForChatMessage(messageDate) + ' ' + formatTimeForChatMessage(messageDate);
11284
- };
11285
- const dayToDayName = (day, dateStrings) => {
11286
- switch (day) {
11287
- case 0:
11288
- return dateStrings.sunday;
11289
- case 1:
11290
- return dateStrings.monday;
11291
- case 2:
11292
- return dateStrings.tuesday;
11293
- case 3:
11294
- return dateStrings.wednesday;
11295
- case 4:
11296
- return dateStrings.thursday;
11297
- case 5:
11298
- return dateStrings.friday;
11299
- case 6:
11300
- return dateStrings.saturday;
11301
- default:
11302
- throw new Error(`Invalid day [${day}] passed`);
11303
- }
11304
- };
11305
-
11306
11284
  // Copyright (c) Microsoft Corporation.
11307
11285
  // Licensed under the MIT License.
11308
11286
  /**
@@ -11475,147 +11453,6 @@ const preventUnwantedDismissProps = {
11475
11453
  };
11476
11454
  const calloutMenuProps = Object.assign(Object.assign({}, preventUnwantedDismissProps), { styles: { root: { marginRight: '3px' } } });
11477
11455
 
11478
- // Copyright (c) Microsoft Corporation.
11479
- // Licensed under the MIT License.
11480
- /**
11481
- * Provides the default implementation for rendering an Mention in a message thread
11482
- * @param mention - The mention to render
11483
- *
11484
- * @private
11485
- */
11486
- const defaultOnMentionRender = (mention) => {
11487
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
11488
- const MsftMention = 'msft-mention';
11489
- return (React.createElement(MsftMention, { id: mention.id, key: Math.random().toString() }, mention.displayText));
11490
- };
11491
-
11492
- // Copyright (c) Microsoft Corporation.
11493
- // Licensed under the MIT License.
11494
- /** @private */
11495
- const ChatMessageContent = (props) => {
11496
- switch (props.message.contentType) {
11497
- case 'text':
11498
- return MessageContentAsText(props);
11499
- case 'html':
11500
- return MessageContentAsRichTextHTML(props);
11501
- case 'richtext/html':
11502
- return MessageContentAsRichTextHTML(props);
11503
- default:
11504
- console.warn('unknown message content type');
11505
- return React.createElement(React.Fragment, null);
11506
- }
11507
- };
11508
- const MessageContentWithLiveAria = (props) => {
11509
- return (React.createElement("div", { "data-ui-status": props.message.status, role: "text", "aria-label": props.ariaLabel },
11510
- React.createElement(LiveMessage, { message: props.liveMessage, ariaLive: "polite" }),
11511
- props.content));
11512
- };
11513
- const MessageContentAsRichTextHTML = (props) => {
11514
- return (React.createElement(MessageContentWithLiveAria, { message: props.message, liveMessage: generateLiveMessage(props), ariaLabel: messageContentAriaText(props), content: processHtmlToReact(props) }));
11515
- };
11516
- const MessageContentAsText = (props) => {
11517
- return (React.createElement(MessageContentWithLiveAria, { message: props.message, liveMessage: generateLiveMessage(props), ariaLabel: messageContentAriaText(props), content: React.createElement(Linkify, { componentDecorator: (decoratedHref, decoratedText, key) => {
11518
- return (React.createElement(react.Link, { target: "_blank", href: decoratedHref, key: key }, decoratedText));
11519
- } }, props.message.content) }));
11520
- };
11521
- /* @conditional-compile-remove(data-loss-prevention) */
11522
- /**
11523
- * @private
11524
- */
11525
- const BlockedMessageContent = (props) => {
11526
- var _a;
11527
- const Icon = React.createElement(react.FontIcon, { iconName: 'DataLossPreventionProhibited' });
11528
- const blockedMessage = props.message.warningText === undefined ? props.strings.blockedWarningText : props.message.warningText;
11529
- const blockedMessageLink = props.message.link;
11530
- const blockedMessageLinkText = blockedMessageLink
11531
- ? (_a = props.message.linkText) !== null && _a !== void 0 ? _a : props.strings.blockedWarningLinkText
11532
- : '';
11533
- const liveAuthor = props.message.mine || props.message.senderDisplayName === undefined ? '' : props.message.senderDisplayName;
11534
- const liveBlockedWarningText = `${liveAuthor} ${blockedMessage} ${blockedMessageLinkText}`;
11535
- return (React.createElement(MessageContentWithLiveAria, { message: props.message, liveMessage: liveBlockedWarningText, ariaLabel: liveBlockedWarningText, content: React.createElement(react.Stack, { horizontal: true, wrap: true },
11536
- Icon,
11537
- blockedMessage && React.createElement("p", null, blockedMessage),
11538
- blockedMessageLink && (React.createElement(react.Link, { target: '_blank', href: blockedMessageLink }, blockedMessageLinkText))) }));
11539
- };
11540
- const extractContentForAllyMessage = (props) => {
11541
- if (props.message.content) {
11542
- // Replace all <img> tags with 'image' for aria.
11543
- const parsedContent = DOMPurify.sanitize(props.message.content, {
11544
- ALLOWED_TAGS: ['img'],
11545
- RETURN_DOM_FRAGMENT: true
11546
- });
11547
- parsedContent.childNodes.forEach((child) => {
11548
- if (child.nodeName.toLowerCase() !== 'img') {
11549
- return;
11550
- }
11551
- const imageTextNode = document.createElement('div');
11552
- imageTextNode.innerHTML = 'image ';
11553
- parsedContent.replaceChild(imageTextNode, child);
11554
- });
11555
- // Strip all html tags from the content for aria.
11556
- const message = DOMPurify.sanitize(parsedContent, { ALLOWED_TAGS: [] });
11557
- return message;
11558
- }
11559
- return '';
11560
- };
11561
- const generateLiveMessage = (props) => {
11562
- const liveAuthor = _formatString(props.strings.liveAuthorIntro, { author: `${props.message.senderDisplayName}` });
11563
- return `${props.message.editedOn ? props.strings.editedTag : ''} ${props.message.mine ? '' : liveAuthor} ${extractContentForAllyMessage(props)} `;
11564
- };
11565
- const messageContentAriaText = (props) => {
11566
- const message = extractContentForAllyMessage(props);
11567
- return props.message.mine
11568
- ? _formatString(props.strings.messageContentMineAriaText, {
11569
- message: message
11570
- })
11571
- : _formatString(props.strings.messageContentAriaText, {
11572
- author: `${props.message.senderDisplayName}`,
11573
- message: message
11574
- });
11575
- };
11576
- /* @conditional-compile-remove(image-overlay) */
11577
- const defaultOnRenderInlineImage = (inlineImage) => {
11578
- return (React.createElement("img", Object.assign({ key: inlineImage.imageAttributes.id, tabIndex: 0, "data-ui-id": inlineImage.imageAttributes.id }, inlineImage.imageAttributes)));
11579
- };
11580
- const processHtmlToReact = (props) => {
11581
- var _a;
11582
- const options = {
11583
- transform(reactNode, domNode) {
11584
- var _a, _b, _c;
11585
- if (domNode instanceof parse.Element && domNode.attribs) {
11586
- // Transform custom rendering of mentions
11587
- /* @conditional-compile-remove(mention) */
11588
- if (domNode.name === 'msft-mention') {
11589
- const { id } = domNode.attribs;
11590
- const mention = {
11591
- id: id,
11592
- displayText: (_a = domNode.children[0].nodeValue) !== null && _a !== void 0 ? _a : ''
11593
- };
11594
- if ((_b = props.mentionDisplayOptions) === null || _b === void 0 ? void 0 : _b.onRenderMention) {
11595
- return props.mentionDisplayOptions.onRenderMention(mention, defaultOnMentionRender);
11596
- }
11597
- return defaultOnMentionRender(mention);
11598
- }
11599
- // Transform inline images
11600
- /* @conditional-compile-remove(teams-inline-images-and-file-sharing) */
11601
- if (domNode.name && domNode.name === 'img' && domNode.attribs && domNode.attribs.id) {
11602
- domNode.attribs['aria-label'] = domNode.attribs.name;
11603
- const imgProps = parse.attributesToProps(domNode.attribs);
11604
- /* @conditional-compile-remove(image-overlay) */
11605
- const inlineImageProps = { messageId: props.message.messageId, imageAttributes: imgProps };
11606
- /* @conditional-compile-remove(image-overlay) */
11607
- return ((_c = props.inlineImageOptions) === null || _c === void 0 ? void 0 : _c.onRenderInlineImage)
11608
- ? props.inlineImageOptions.onRenderInlineImage(inlineImageProps, defaultOnRenderInlineImage)
11609
- : defaultOnRenderInlineImage(inlineImageProps);
11610
- }
11611
- }
11612
- // Pass through the original node
11613
- return reactNode;
11614
- }
11615
- };
11616
- return React.createElement(React.Fragment, null, parse((_a = props.message.content) !== null && _a !== void 0 ? _a : '', options));
11617
- };
11618
-
11619
11456
  // Copyright (c) Microsoft Corporation.
11620
11457
  // Licensed under the MIT License.
11621
11458
  /**
@@ -11639,6 +11476,147 @@ const chatMessageActionMenuProps = (menuProps) => {
11639
11476
  return actionMenuProps;
11640
11477
  };
11641
11478
 
11479
+ // Copyright (c) Microsoft Corporation.
11480
+ // Licensed under the MIT License.
11481
+ /**
11482
+ * Provides the default implementation for rendering an Mention in a message thread
11483
+ * @param mention - The mention to render
11484
+ *
11485
+ * @private
11486
+ */
11487
+ const defaultOnMentionRender = (mention) => {
11488
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
11489
+ const MsftMention = 'msft-mention';
11490
+ return (React.createElement(MsftMention, { id: mention.id, key: Math.random().toString() }, mention.displayText));
11491
+ };
11492
+
11493
+ // Copyright (c) Microsoft Corporation.
11494
+ // Licensed under the MIT License.
11495
+ /** @private */
11496
+ const ChatMessageContent = (props) => {
11497
+ switch (props.message.contentType) {
11498
+ case 'text':
11499
+ return MessageContentAsText(props);
11500
+ case 'html':
11501
+ return MessageContentAsRichTextHTML(props);
11502
+ case 'richtext/html':
11503
+ return MessageContentAsRichTextHTML(props);
11504
+ default:
11505
+ console.warn('unknown message content type');
11506
+ return React.createElement(React.Fragment, null);
11507
+ }
11508
+ };
11509
+ const MessageContentWithLiveAria = (props) => {
11510
+ return (React.createElement("div", { "data-ui-status": props.message.status, role: "text", "aria-label": props.ariaLabel },
11511
+ React.createElement(LiveMessage, { message: props.liveMessage, ariaLive: "polite" }),
11512
+ props.content));
11513
+ };
11514
+ const MessageContentAsRichTextHTML = (props) => {
11515
+ return (React.createElement(MessageContentWithLiveAria, { message: props.message, liveMessage: generateLiveMessage(props), ariaLabel: messageContentAriaText(props), content: processHtmlToReact(props) }));
11516
+ };
11517
+ const MessageContentAsText = (props) => {
11518
+ return (React.createElement(MessageContentWithLiveAria, { message: props.message, liveMessage: generateLiveMessage(props), ariaLabel: messageContentAriaText(props), content: React.createElement(Linkify, { componentDecorator: (decoratedHref, decoratedText, key) => {
11519
+ return (React.createElement(react.Link, { target: "_blank", href: decoratedHref, key: key }, decoratedText));
11520
+ } }, props.message.content) }));
11521
+ };
11522
+ /* @conditional-compile-remove(data-loss-prevention) */
11523
+ /**
11524
+ * @private
11525
+ */
11526
+ const BlockedMessageContent = (props) => {
11527
+ var _a;
11528
+ const Icon = React.createElement(react.FontIcon, { iconName: 'DataLossPreventionProhibited' });
11529
+ const blockedMessage = props.message.warningText === undefined ? props.strings.blockedWarningText : props.message.warningText;
11530
+ const blockedMessageLink = props.message.link;
11531
+ const blockedMessageLinkText = blockedMessageLink
11532
+ ? (_a = props.message.linkText) !== null && _a !== void 0 ? _a : props.strings.blockedWarningLinkText
11533
+ : '';
11534
+ const liveAuthor = props.message.mine || props.message.senderDisplayName === undefined ? '' : props.message.senderDisplayName;
11535
+ const liveBlockedWarningText = `${liveAuthor} ${blockedMessage} ${blockedMessageLinkText}`;
11536
+ return (React.createElement(MessageContentWithLiveAria, { message: props.message, liveMessage: liveBlockedWarningText, ariaLabel: liveBlockedWarningText, content: React.createElement(react.Stack, { horizontal: true, wrap: true },
11537
+ Icon,
11538
+ blockedMessage && React.createElement("p", null, blockedMessage),
11539
+ blockedMessageLink && (React.createElement(react.Link, { target: '_blank', href: blockedMessageLink }, blockedMessageLinkText))) }));
11540
+ };
11541
+ const extractContentForAllyMessage = (props) => {
11542
+ if (props.message.content) {
11543
+ // Replace all <img> tags with 'image' for aria.
11544
+ const parsedContent = DOMPurify.sanitize(props.message.content, {
11545
+ ALLOWED_TAGS: ['img'],
11546
+ RETURN_DOM_FRAGMENT: true
11547
+ });
11548
+ parsedContent.childNodes.forEach((child) => {
11549
+ if (child.nodeName.toLowerCase() !== 'img') {
11550
+ return;
11551
+ }
11552
+ const imageTextNode = document.createElement('div');
11553
+ imageTextNode.innerHTML = 'image ';
11554
+ parsedContent.replaceChild(imageTextNode, child);
11555
+ });
11556
+ // Strip all html tags from the content for aria.
11557
+ const message = DOMPurify.sanitize(parsedContent, { ALLOWED_TAGS: [] });
11558
+ return message;
11559
+ }
11560
+ return '';
11561
+ };
11562
+ const generateLiveMessage = (props) => {
11563
+ const liveAuthor = _formatString(props.strings.liveAuthorIntro, { author: `${props.message.senderDisplayName}` });
11564
+ return `${props.message.editedOn ? props.strings.editedTag : ''} ${props.message.mine ? '' : liveAuthor} ${extractContentForAllyMessage(props)} `;
11565
+ };
11566
+ const messageContentAriaText = (props) => {
11567
+ const message = extractContentForAllyMessage(props);
11568
+ return props.message.mine
11569
+ ? _formatString(props.strings.messageContentMineAriaText, {
11570
+ message: message
11571
+ })
11572
+ : _formatString(props.strings.messageContentAriaText, {
11573
+ author: `${props.message.senderDisplayName}`,
11574
+ message: message
11575
+ });
11576
+ };
11577
+ /* @conditional-compile-remove(image-overlay) */
11578
+ const defaultOnRenderInlineImage = (inlineImage) => {
11579
+ return (React.createElement("img", Object.assign({ key: inlineImage.imageAttributes.id, tabIndex: 0, "data-ui-id": inlineImage.imageAttributes.id }, inlineImage.imageAttributes)));
11580
+ };
11581
+ const processHtmlToReact = (props) => {
11582
+ var _a;
11583
+ const options = {
11584
+ transform(reactNode, domNode) {
11585
+ var _a, _b, _c;
11586
+ if (domNode instanceof parse.Element && domNode.attribs) {
11587
+ // Transform custom rendering of mentions
11588
+ /* @conditional-compile-remove(mention) */
11589
+ if (domNode.name === 'msft-mention') {
11590
+ const { id } = domNode.attribs;
11591
+ const mention = {
11592
+ id: id,
11593
+ displayText: (_a = domNode.children[0].nodeValue) !== null && _a !== void 0 ? _a : ''
11594
+ };
11595
+ if ((_b = props.mentionDisplayOptions) === null || _b === void 0 ? void 0 : _b.onRenderMention) {
11596
+ return props.mentionDisplayOptions.onRenderMention(mention, defaultOnMentionRender);
11597
+ }
11598
+ return defaultOnMentionRender(mention);
11599
+ }
11600
+ // Transform inline images
11601
+ /* @conditional-compile-remove(teams-inline-images-and-file-sharing) */
11602
+ if (domNode.name && domNode.name === 'img' && domNode.attribs && domNode.attribs.id) {
11603
+ domNode.attribs['aria-label'] = domNode.attribs.name;
11604
+ const imgProps = parse.attributesToProps(domNode.attribs);
11605
+ /* @conditional-compile-remove(image-overlay) */
11606
+ const inlineImageProps = { messageId: props.message.messageId, imageAttributes: imgProps };
11607
+ /* @conditional-compile-remove(image-overlay) */
11608
+ return ((_c = props.inlineImageOptions) === null || _c === void 0 ? void 0 : _c.onRenderInlineImage)
11609
+ ? props.inlineImageOptions.onRenderInlineImage(inlineImageProps, defaultOnRenderInlineImage)
11610
+ : defaultOnRenderInlineImage(inlineImageProps);
11611
+ }
11612
+ }
11613
+ // Pass through the original node
11614
+ return reactNode;
11615
+ }
11616
+ };
11617
+ return React.createElement(React.Fragment, null, parse((_a = props.message.content) !== null && _a !== void 0 ? _a : '', options));
11618
+ };
11619
+
11642
11620
  // Copyright (c) Microsoft Corporation.
11643
11621
  // Licensed under the MIT License.
11644
11622
  var __awaiter$C = (window && window.__awaiter) || function (thisArg, _arguments, P, generator) {
@@ -11726,6 +11704,138 @@ const useLocaleStringsTrampoline = () => {
11726
11704
 
11727
11705
  // Copyright (c) Microsoft Corporation.
11728
11706
  // Licensed under the MIT License.
11707
+ /**
11708
+ * @private
11709
+ */
11710
+ const formatTimeForChatMessage = (messageDate) => {
11711
+ return messageDate.toLocaleTimeString([], { hour: 'numeric', minute: '2-digit' });
11712
+ };
11713
+ /**
11714
+ * @private
11715
+ */
11716
+ const formatDateForChatMessage = (messageDate) => {
11717
+ return messageDate.toLocaleDateString();
11718
+ };
11719
+ /**
11720
+ * Given a message date object in ISO8601 and a current date object, generates a user friendly timestamp text
11721
+ * using the system locale.
11722
+ * <time in locale format>.
11723
+ * Yesterday <time in locale format>.
11724
+ * <dateStrings day of week> <time in locale format>.
11725
+ * <date in locale format> <time in locale format>.
11726
+ *
11727
+ * If message is after yesterday, then only show the time.
11728
+ * If message is before yesterday and after day before yesterday, then show 'Yesterday' plus the time.
11729
+ * If message is before day before yesterday and within the current week, then show 'Monday/Tuesday/etc' plus the time.
11730
+ * - We consider start of the week as Sunday. If current day is Sunday, then any time before that is in previous week.
11731
+ * If message is in previous or older weeks, then show date string plus the time.
11732
+ *
11733
+ * @param messageDate - date of message
11734
+ * @param currentDate - date used as offset to create the user friendly timestamp (e.g. to create 'Yesterday' instead of an absolute date)
11735
+ *
11736
+ * @private
11737
+ */
11738
+ const formatTimestampForChatMessage = (messageDate, todayDate, dateStrings) => {
11739
+ // If message was in the same day timestamp string is just the time like '1:30 p.m.'.
11740
+ const startOfDay = new Date(todayDate.getFullYear(), todayDate.getMonth(), todayDate.getDate());
11741
+ if (messageDate > startOfDay) {
11742
+ return formatTimeForChatMessage(messageDate);
11743
+ }
11744
+ // If message was yesterday then timestamp string is like this 'Yesterday 1:30 p.m.'.
11745
+ const yesterdayDate = new Date(todayDate.getFullYear(), todayDate.getMonth(), todayDate.getDate() - 1);
11746
+ if (messageDate > yesterdayDate) {
11747
+ return dateStrings.yesterday + ' ' + formatTimeForChatMessage(messageDate);
11748
+ }
11749
+ // If message was before Sunday and today is Sunday (start of week) then timestamp string is like
11750
+ // '2021-01-10 1:30 p.m.'.
11751
+ const weekDay = todayDate.getDay();
11752
+ if (weekDay === 0) {
11753
+ return formatDateForChatMessage(messageDate) + ' ' + formatTimeForChatMessage(messageDate);
11754
+ }
11755
+ // If message was before first day of the week then timestamp string is like Monday 1:30 p.m.
11756
+ const firstDayOfTheWeekDate = new Date(todayDate.getFullYear(), todayDate.getMonth(), todayDate.getDate() - weekDay);
11757
+ if (messageDate > firstDayOfTheWeekDate) {
11758
+ return dayToDayName(messageDate.getDay(), dateStrings) + ' ' + formatTimeForChatMessage(messageDate);
11759
+ }
11760
+ // If message date is in previous or older weeks then timestamp string is like 2021-01-10 1:30 p.m.
11761
+ return formatDateForChatMessage(messageDate) + ' ' + formatTimeForChatMessage(messageDate);
11762
+ };
11763
+ const dayToDayName = (day, dateStrings) => {
11764
+ switch (day) {
11765
+ case 0:
11766
+ return dateStrings.sunday;
11767
+ case 1:
11768
+ return dateStrings.monday;
11769
+ case 2:
11770
+ return dateStrings.tuesday;
11771
+ case 3:
11772
+ return dateStrings.wednesday;
11773
+ case 4:
11774
+ return dateStrings.thursday;
11775
+ case 5:
11776
+ return dateStrings.friday;
11777
+ case 6:
11778
+ return dateStrings.saturday;
11779
+ default:
11780
+ throw new Error(`Invalid day [${day}] passed`);
11781
+ }
11782
+ };
11783
+
11784
+ // Copyright (c) Microsoft Corporation.
11785
+ // Licensed under the MIT License.
11786
+ /** @private
11787
+ * Return the string value for the FluentUI message attached prop based on the message's attached status.
11788
+ * @param attached - The message's attached status.
11789
+ */
11790
+ const getFluentUIAttachedValue = (messageAttachedStatus) => {
11791
+ return messageAttachedStatus === 'top' || messageAttachedStatus === false ? 'top' : 'center';
11792
+ };
11793
+ /**
11794
+ * @private
11795
+ * Get the message bubble content for the message.
11796
+ */
11797
+ function getMessageBubbleContent(message, strings, userId,
11798
+ /* @conditional-compile-remove(image-overlay) */
11799
+ inlineImageOptions,
11800
+ /* @conditional-compile-remove(file-sharing) */
11801
+ onRenderFileDownloads,
11802
+ /* @conditional-compile-remove(mention) */
11803
+ mentionDisplayOptions,
11804
+ /* @conditional-compile-remove(file-sharing) */
11805
+ fileDownloadHandler) {
11806
+ /* @conditional-compile-remove(data-loss-prevention) */
11807
+ if (message.messageType === 'blocked') {
11808
+ return (React.createElement("div", { tabIndex: 0 },
11809
+ React.createElement(BlockedMessageContent, { message: message, strings: strings })));
11810
+ }
11811
+ return (React.createElement("div", { tabIndex: 0, className: "ui-chat__message__content" },
11812
+ React.createElement(ChatMessageContent, { message: message, strings: strings,
11813
+ /* @conditional-compile-remove(mention) */
11814
+ mentionDisplayOptions: mentionDisplayOptions,
11815
+ /* @conditional-compile-remove(image-overlay) */
11816
+ inlineImageOptions: inlineImageOptions }),
11817
+ /* @conditional-compile-remove(file-sharing) */ onRenderFileDownloads
11818
+ ? onRenderFileDownloads(userId, message)
11819
+ : defaultOnRenderFileDownloads(userId, message, strings,
11820
+ /* @conditional-compile-remove(file-sharing) */ fileDownloadHandler)));
11821
+ }
11822
+ /**
11823
+ * Default component for rendering file downloads.
11824
+ */
11825
+ /* @conditional-compile-remove(file-sharing) */
11826
+ const defaultOnRenderFileDownloads = (userId, message, strings,
11827
+ /* @conditional-compile-remove(file-sharing) */
11828
+ fileDownloadHandler) => {
11829
+ /* @conditional-compile-remove(file-sharing) */
11830
+ return (React.createElement(_FileDownloadCards, { userId: userId,
11831
+ /* @conditional-compile-remove(file-sharing) */
11832
+ fileMetadata: message.files || [],
11833
+ /* @conditional-compile-remove(file-sharing) */
11834
+ downloadHandler: fileDownloadHandler,
11835
+ /* @conditional-compile-remove(file-sharing) */
11836
+ strings: { downloadFile: strings.downloadFile, fileCardGroupMessage: strings.fileCardGroupMessage } }));
11837
+ };
11838
+ /** @private */
11729
11839
  const generateDefaultTimestamp = (createdOn, showDate, strings) => {
11730
11840
  const formattedTimestamp = showDate
11731
11841
  ? formatTimestampForChatMessage(createdOn, new Date(), strings)
@@ -11733,16 +11843,31 @@ const generateDefaultTimestamp = (createdOn, showDate, strings) => {
11733
11843
  return formattedTimestamp;
11734
11844
  };
11735
11845
  // onDisplayDateTimeString from props overwrite onDisplayDateTimeString from locale
11736
- const generateCustomizedTimestamp = (props, createdOn, locale) => {
11846
+ /** @private */
11847
+ const generateCustomizedTimestamp = (createdOn, locale, onDisplayDateTimeString) => {
11737
11848
  /* @conditional-compile-remove(date-time-customization) */
11738
- return props.onDisplayDateTimeString
11739
- ? props.onDisplayDateTimeString(createdOn)
11849
+ return onDisplayDateTimeString
11850
+ ? onDisplayDateTimeString(createdOn)
11740
11851
  : locale.onDisplayDateTimeString
11741
11852
  ? locale.onDisplayDateTimeString(createdOn)
11742
11853
  : '';
11743
11854
  };
11855
+ /**
11856
+ * @private
11857
+ * Get the edited tag for the message if it is edited.
11858
+ */
11859
+ const getMessageEditedDetails = (message, theme, editedTag) => {
11860
+ const editedOn = 'editedOn' in message ? message.editedOn : undefined;
11861
+ if (message.messageType === 'chat' && editedOn) {
11862
+ return React.createElement("div", { className: chatMessageEditedTagStyle(theme) }, editedTag);
11863
+ }
11864
+ return undefined;
11865
+ };
11866
+
11867
+ // Copyright (c) Microsoft Corporation.
11868
+ // Licensed under the MIT License.
11744
11869
  /** @private */
11745
- const MessageBubble = (props) => {
11870
+ const MessageBubble$1 = (props) => {
11746
11871
  var _a;
11747
11872
  const ids = useIdentifiers();
11748
11873
  const theme = useTheme();
@@ -11751,12 +11876,20 @@ const MessageBubble = (props) => {
11751
11876
  /* @conditional-compile-remove(file-sharing) */
11752
11877
  fileDownloadHandler,
11753
11878
  /* @conditional-compile-remove(image-overlay) */
11754
- inlineImageOptions, shouldOverlapAvatarAndMessage } = props;
11755
- const defaultTimeStamp = message.createdOn
11756
- ? generateDefaultTimestamp(message.createdOn, showDate, strings)
11757
- : undefined;
11758
- const customTimestamp = message.createdOn ? generateCustomizedTimestamp(props, message.createdOn, locale) : '';
11759
- const formattedTimestamp = customTimestamp || defaultTimeStamp;
11879
+ inlineImageOptions,
11880
+ /* @conditional-compile-remove(file-sharing) */
11881
+ onRenderFileDownloads,
11882
+ /* @conditional-compile-remove(mention) */
11883
+ mentionDisplayOptions, onDisplayDateTimeString } = props;
11884
+ const formattedTimestamp = React.useMemo(() => {
11885
+ const defaultTimeStamp = message.createdOn
11886
+ ? generateDefaultTimestamp(message.createdOn, showDate, strings)
11887
+ : undefined;
11888
+ const customTimestamp = message.createdOn
11889
+ ? generateCustomizedTimestamp(message.createdOn, locale, onDisplayDateTimeString)
11890
+ : '';
11891
+ return customTimestamp || defaultTimeStamp;
11892
+ }, [locale, message.createdOn, onDisplayDateTimeString, showDate, strings]);
11760
11893
  // Track if the action menu was opened by touch - if so we increase the touch targets for the items
11761
11894
  const [wasInteractionByTouch, setWasInteractionByTouch] = React.useState(false);
11762
11895
  // `focused` state is used for show/hide actionMenu
@@ -11790,139 +11923,95 @@ const MessageBubble = (props) => {
11790
11923
  // or not we need to set the target to undefined here to actually hide the action flyout
11791
11924
  setChatMessageActionFlyoutTarget(undefined);
11792
11925
  }, [setChatMessageActionFlyoutTarget]);
11793
- const defaultOnRenderFileDownloads = React.useCallback(() => {
11794
- /* @conditional-compile-remove(file-sharing) */
11795
- return (React.createElement(_FileDownloadCards, { userId: userId,
11796
- /* @conditional-compile-remove(file-sharing) */
11797
- fileMetadata: message.files || [],
11798
- /* @conditional-compile-remove(file-sharing) */
11799
- downloadHandler: fileDownloadHandler,
11800
- /* @conditional-compile-remove(file-sharing) */
11801
- strings: { downloadFile: strings.downloadFile, fileCardGroupMessage: strings.fileCardGroupMessage } }));
11802
- }, [
11803
- userId,
11804
- message,
11805
- /* @conditional-compile-remove(file-sharing) */
11806
- strings,
11807
- /* @conditional-compile-remove(file-sharing) */
11808
- fileDownloadHandler
11809
- ]);
11810
- const editedOn = 'editedOn' in message ? message.editedOn : undefined;
11811
11926
  const getMessageDetails = React.useCallback(() => {
11812
11927
  if (messageStatus === 'failed') {
11813
11928
  return React.createElement("div", { className: chatMessageFailedTagStyle(theme) }, strings.failToSendTag);
11814
11929
  }
11815
- else if (message.messageType === 'chat' && editedOn) {
11816
- return React.createElement("div", { className: chatMessageEditedTagStyle(theme) }, strings.editedTag);
11930
+ else {
11931
+ return getMessageEditedDetails(message, theme, strings.editedTag);
11817
11932
  }
11818
- return undefined;
11819
- }, [editedOn, message.messageType, messageStatus, strings.editedTag, strings.failToSendTag, theme]);
11933
+ }, [message, messageStatus, strings.editedTag, strings.failToSendTag, theme]);
11820
11934
  const getContent = React.useCallback(() => {
11821
- /* @conditional-compile-remove(data-loss-prevention) */
11822
- if (message.messageType === 'blocked') {
11823
- return (React.createElement("div", { tabIndex: 0 },
11824
- React.createElement(BlockedMessageContent, { message: message, strings: strings })));
11825
- }
11826
- return (React.createElement("div", { tabIndex: 0, className: "ui-chat__message__content" },
11827
- React.createElement(ChatMessageContent, { message: message, strings: strings,
11828
- /* @conditional-compile-remove(mention) */
11829
- mentionDisplayOptions: props.mentionDisplayOptions,
11830
- /* @conditional-compile-remove(image-overlay) */
11831
- inlineImageOptions: inlineImageOptions }),
11832
- /* @conditional-compile-remove(file-sharing) */ props.onRenderFileDownloads
11833
- ? props.onRenderFileDownloads(userId, message)
11834
- : defaultOnRenderFileDownloads()));
11935
+ return getMessageBubbleContent(message, strings, userId,
11936
+ /* @conditional-compile-remove(image-overlay) */
11937
+ inlineImageOptions,
11938
+ /* @conditional-compile-remove(file-sharing) */
11939
+ onRenderFileDownloads,
11940
+ /* @conditional-compile-remove(mention) */
11941
+ mentionDisplayOptions,
11942
+ /* @conditional-compile-remove(file-sharing) */
11943
+ fileDownloadHandler);
11835
11944
  }, [
11836
- defaultOnRenderFileDownloads,
11945
+ /* @conditional-compile-remove(file-sharing) */ fileDownloadHandler,
11837
11946
  /* @conditional-compile-remove(image-overlay) */ inlineImageOptions,
11947
+ /* @conditional-compile-remove(mention) */ mentionDisplayOptions,
11838
11948
  message,
11839
- props,
11949
+ /* @conditional-compile-remove(file-sharing) */ onRenderFileDownloads,
11840
11950
  strings,
11841
11951
  userId
11842
11952
  ]);
11843
11953
  const isBlockedMessage = /* @conditional-compile-remove(data-loss-prevention) */ message.messageType === 'blocked';
11844
11954
  const chatMyMessageStyles = useChatMyMessageStyles();
11845
11955
  const chatMessageCommonStyles = useChatMessageCommonStyles();
11846
- const chatMessageStyles = useChatMessageStyles();
11847
- const chatItemMessageContainerClassName = reactComponents.mergeClasses(
11848
- // messageContainerStyle used in className and style prop as style prop can't handle CSS selectors
11849
- chatMessageStyles.body,
11850
- // disable placeholder functionality for GA releases as it might confuse users
11851
- /* @conditional-compile-remove(teams-inline-images-and-file-sharing) */
11852
- chatMessageStyles.bodyWithPlaceholderImage, isBlockedMessage
11853
- ? chatMessageCommonStyles.blocked
11854
- : props.message.status === 'failed'
11855
- ? chatMessageCommonStyles.failed
11856
- : undefined, shouldOverlapAvatarAndMessage ? chatMessageStyles.avatarOverlap : chatMessageStyles.avatarNoOverlap, message.attached === 'top' || message.attached === false
11857
- ? chatMessageStyles.bodyWithAvatar
11858
- : chatMessageStyles.bodyWithoutAvatar, react.mergeStyles(messageContainerStyle));
11859
11956
  const attached = message.attached === true ? 'center' : message.attached === 'bottom' ? 'bottom' : 'top';
11860
11957
  const chatMessage = (React.createElement(React.Fragment, null,
11861
- React.createElement("div", { key: props.message.messageId }, message.mine ? (React.createElement(reactChat.ChatMyMessage, { attached: attached, key: props.message.messageId, body: {
11862
- // messageContainerStyle used in className and style prop as style prop can't handle CSS selectors
11863
- className: reactComponents.mergeClasses(chatMyMessageStyles.body, isBlockedMessage
11864
- ? chatMessageCommonStyles.blocked
11865
- : props.message.status === 'failed'
11866
- ? chatMessageCommonStyles.failed
11867
- : undefined, attached !== 'top' ? chatMyMessageStyles.bodyAttached : undefined, react.mergeStyles(messageContainerStyle)),
11868
- style: Object.assign({}, createStyleFromV8Style(messageContainerStyle)),
11869
- ref: messageRef
11870
- }, root: {
11871
- className: chatMyMessageStyles.root,
11872
- onBlur: (e) => {
11873
- // `focused` controls is focused the whole `ChatMessage` or any of its children. When we're navigating
11874
- // with keyboard the focused element will be changed and there is no way to use `:focus` selector
11875
- if (chatMessageActionFlyoutTarget === null || chatMessageActionFlyoutTarget === void 0 ? void 0 : chatMessageActionFlyoutTarget.current) {
11876
- // doesn't dismiss action button if flyout is open, otherwise, narrator's focus will stay on the closed action menu
11958
+ React.createElement("div", { key: props.message.messageId },
11959
+ React.createElement(reactChat.ChatMyMessage, { attached: attached, key: props.message.messageId, body: {
11960
+ // messageContainerStyle used in className and style prop as style prop can't handle CSS selectors
11961
+ className: reactComponents.mergeClasses(chatMyMessageStyles.body, isBlockedMessage
11962
+ ? chatMessageCommonStyles.blocked
11963
+ : props.message.status === 'failed'
11964
+ ? chatMessageCommonStyles.failed
11965
+ : undefined, attached !== 'top' ? chatMyMessageStyles.bodyAttached : undefined, react.mergeStyles(messageContainerStyle)),
11966
+ style: Object.assign({}, createStyleFromV8Style(messageContainerStyle)),
11967
+ ref: messageRef
11968
+ }, root: {
11969
+ className: chatMyMessageStyles.root,
11970
+ onBlur: (e) => {
11971
+ // `focused` controls is focused the whole `ChatMessage` or any of its children. When we're navigating
11972
+ // with keyboard the focused element will be changed and there is no way to use `:focus` selector
11973
+ if (chatMessageActionFlyoutTarget === null || chatMessageActionFlyoutTarget === void 0 ? void 0 : chatMessageActionFlyoutTarget.current) {
11974
+ // doesn't dismiss action button if flyout is open, otherwise, narrator's focus will stay on the closed action menu
11975
+ return;
11976
+ }
11977
+ const shouldPreserveFocusState = e.currentTarget.contains(e.relatedTarget);
11978
+ setFocused(shouldPreserveFocusState);
11979
+ },
11980
+ onFocus: () => {
11981
+ // react onFocus is called even when nested component receives focus (i.e. it bubbles)
11982
+ // so when focus moves within actionMenu, the `focus` state in chatMessage remains true, and keeps actionMenu visible
11983
+ setFocused(true);
11984
+ },
11985
+ // make body not focusable to remove repetitions from narrators.
11986
+ // inner components are already focusable
11987
+ role: 'none',
11988
+ tabIndex: -1
11989
+ }, "data-ui-id": "chat-composite-message", author: React.createElement(react.Text, { className: chatMessageDateStyle, tabIndex: 0 }, message.senderDisplayName), timestamp: React.createElement(react.Text, { className: chatMessageDateStyle, "data-ui-id": ids.messageTimestamp, tabIndex: 0 }, formattedTimestamp), details: getMessageDetails(), actions: {
11990
+ children: actionMenuProps === null || actionMenuProps === void 0 ? void 0 : actionMenuProps.children,
11991
+ className: reactComponents.mergeClasses(chatMyMessageStyles.menu,
11992
+ // Make actions menu visible when the message is focused or the flyout is shown
11993
+ focused || (chatMessageActionFlyoutTarget === null || chatMessageActionFlyoutTarget === void 0 ? void 0 : chatMessageActionFlyoutTarget.current)
11994
+ ? chatMyMessageStyles.menuVisible
11995
+ : chatMyMessageStyles.menuHidden, attached !== 'top' ? chatMyMessageStyles.menuAttached : undefined)
11996
+ }, onTouchStart: () => setWasInteractionByTouch(true), onPointerDown: () => setWasInteractionByTouch(false), onKeyDown: () => setWasInteractionByTouch(false), onClick: () => {
11997
+ if (!wasInteractionByTouch) {
11877
11998
  return;
11878
11999
  }
11879
- const shouldPreserveFocusState = e.currentTarget.contains(e.relatedTarget);
11880
- setFocused(shouldPreserveFocusState);
11881
- },
11882
- onFocus: () => {
11883
- // react onFocus is called even when nested component receives focus (i.e. it bubbles)
11884
- // so when focus moves within actionMenu, the `focus` state in chatMessage remains true, and keeps actionMenu visible
11885
- setFocused(true);
11886
- },
11887
- // make body not focusable to remove repetitions from narrators.
11888
- // inner components are already focusable
11889
- role: 'none',
11890
- tabIndex: -1
11891
- }, "data-ui-id": "chat-composite-message", author: React.createElement(react.Text, { className: chatMessageDateStyle, tabIndex: 0 }, message.senderDisplayName), timestamp: React.createElement(react.Text, { className: chatMessageDateStyle, "data-ui-id": ids.messageTimestamp, tabIndex: 0 }, formattedTimestamp), details: getMessageDetails(), actions: {
11892
- children: actionMenuProps === null || actionMenuProps === void 0 ? void 0 : actionMenuProps.children,
11893
- className: reactComponents.mergeClasses(chatMyMessageStyles.menu,
11894
- // Make actions menu visible when the message is focused or the flyout is shown
11895
- focused || (chatMessageActionFlyoutTarget === null || chatMessageActionFlyoutTarget === void 0 ? void 0 : chatMessageActionFlyoutTarget.current)
11896
- ? chatMyMessageStyles.menuVisible
11897
- : chatMyMessageStyles.menuHidden, attached !== 'top' ? chatMyMessageStyles.menuAttached : undefined)
11898
- }, onTouchStart: () => setWasInteractionByTouch(true), onPointerDown: () => setWasInteractionByTouch(false), onKeyDown: () => setWasInteractionByTouch(false), onClick: () => {
11899
- if (!wasInteractionByTouch) {
11900
- return;
11901
- }
11902
- // If the message was touched via touch we immediately open the menu
11903
- // flyout (when using mouse the 3-dot menu that appears on hover
11904
- // must be clicked to open the flyout).
11905
- // In doing so here we set the target of the flyout to be the message and
11906
- // not the 3-dot menu button to position the flyout correctly.
11907
- setChatMessageActionFlyoutTarget(messageRef);
11908
- if (message.messageType === 'chat') {
11909
- props.onActionButtonClick(message, setMessageReadBy);
11910
- }
11911
- } }, getContent())) : (React.createElement(reactChat.ChatMessage, { attached: attached, key: props.message.messageId, root: {
11912
- className: chatMessageStyles.root,
11913
- // make body not focusable to remove repetitions from narrators.
11914
- // inner components are already focusable
11915
- tabIndex: -1,
11916
- role: 'none'
11917
- }, author: React.createElement(react.Text, { className: chatMessageAuthorStyle }, message.senderDisplayName), body: {
11918
- className: chatItemMessageContainerClassName,
11919
- style: Object.assign({}, createStyleFromV8Style(messageContainerStyle))
11920
- }, "data-ui-id": "chat-composite-message", timestamp: React.createElement(react.Text, { className: chatMessageDateStyle, "data-ui-id": ids.messageTimestamp }, formattedTimestamp), details: getMessageDetails() }, getContent()))),
12000
+ // If the message was touched via touch we immediately open the menu
12001
+ // flyout (when using mouse the 3-dot menu that appears on hover
12002
+ // must be clicked to open the flyout).
12003
+ // In doing so here we set the target of the flyout to be the message and
12004
+ // not the 3-dot menu button to position the flyout correctly.
12005
+ setChatMessageActionFlyoutTarget(messageRef);
12006
+ if (message.messageType === 'chat') {
12007
+ props.onActionButtonClick(message, setMessageReadBy);
12008
+ }
12009
+ } }, getContent())),
11921
12010
  chatActionsEnabled && (React.createElement(ChatMessageActionFlyout, { hidden: !chatMessageActionFlyoutTarget, target: chatMessageActionFlyoutTarget, increaseFlyoutItemSize: wasInteractionByTouch, onDismiss: onActionFlyoutDismiss, onEditClick: onEditClick, onRemoveClick: onRemoveClick, onResendClick: onResendClick, strings: strings, messageReadBy: messageReadBy, messageStatus: messageStatus !== null && messageStatus !== void 0 ? messageStatus : 'failed', remoteParticipantsCount: remoteParticipantsCount, onRenderAvatar: onRenderAvatar, showMessageStatus: showMessageStatus }))));
11922
12011
  return chatMessage;
11923
12012
  };
11924
12013
  /** @private */
11925
- const ChatMessageComponentAsMessageBubble = React.memo(MessageBubble);
12014
+ const ChatMyMessageComponentAsMessageBubble = React.memo(MessageBubble$1);
11926
12015
 
11927
12016
  // Copyright (c) Microsoft Corporation.
11928
12017
  // Licensed under the MIT License.
@@ -11938,7 +12027,7 @@ var __awaiter$B = (window && window.__awaiter) || function (thisArg, _arguments,
11938
12027
  /**
11939
12028
  * @private
11940
12029
  */
11941
- const ChatMessageComponent = (props) => {
12030
+ const ChatMyMessageComponent = (props) => {
11942
12031
  var _a, _b;
11943
12032
  const { onDeleteMessage, onSendMessage, message } = props;
11944
12033
  const [isEditing, setIsEditing] = React.useState(false);
@@ -11972,7 +12061,7 @@ const ChatMessageComponent = (props) => {
11972
12061
  mentionLookupOptions: (_a = props.mentionOptions) === null || _a === void 0 ? void 0 : _a.lookupOptions }));
11973
12062
  }
11974
12063
  else {
11975
- return (React.createElement(ChatMessageComponentAsMessageBubble, Object.assign({}, props, { onRemoveClick: onRemoveClick, onEditClick: onEditClick, onResendClick: onResendClick, onRenderAvatar: props.onRenderAvatar,
12064
+ return (React.createElement(ChatMyMessageComponentAsMessageBubble, Object.assign({}, props, { onRemoveClick: onRemoveClick, onEditClick: onEditClick, onResendClick: onResendClick, onRenderAvatar: props.onRenderAvatar,
11976
12065
  /* @conditional-compile-remove(date-time-customization) */
11977
12066
  onDisplayDateTimeString: props.onDisplayDateTimeString, strings: props.strings,
11978
12067
  /* @conditional-compile-remove(image-overlay) */
@@ -11992,7 +12081,7 @@ const ChatMessageComponent = (props) => {
11992
12081
  *
11993
12082
  * @private
11994
12083
  */
11995
- const FluentChatMessageComponentWrapper = (props) => {
12084
+ const FluentChatMyMessageComponent = (props) => {
11996
12085
  const { message, styles, shouldOverlapAvatarAndMessage, onRenderMessage, onRenderAvatar, showMessageStatus, onRenderMessageStatus, participantCount, readCount, onActionButtonClick,
11997
12086
  /* @conditional-compile-remove(date-time-customization) */
11998
12087
  onDisplayDateTimeString,
@@ -12014,7 +12103,7 @@ const FluentChatMessageComponentWrapper = (props) => {
12014
12103
  const defaultChatMessageRenderer = React.useCallback((messageProps) => {
12015
12104
  if (messageProps.message.messageType === 'chat' ||
12016
12105
  /* @conditional-compile-remove(data-loss-prevention) */ messageProps.message.messageType === 'blocked') {
12017
- return (React.createElement(ChatMessageComponent, Object.assign({}, messageProps, {
12106
+ return (React.createElement(ChatMyMessageComponent, Object.assign({}, messageProps, {
12018
12107
  /* @conditional-compile-remove(file-sharing) */
12019
12108
  onRenderFileDownloads: onRenderFileDownloadsMemo,
12020
12109
  /* @conditional-compile-remove(file-sharing) */
@@ -12056,12 +12145,9 @@ const FluentChatMessageComponentWrapper = (props) => {
12056
12145
  const messageStatusRenderer = React.useCallback((onRenderMessageStatus, defaultStatusRenderer, showMessageStatus, participantCount, readCount) => {
12057
12146
  return showMessageStatus && statusToRender ? (onRenderMessageStatus ? (onRenderMessageStatus({ status: message.status })) : (defaultStatusRenderer(message, participantCount !== null && participantCount !== void 0 ? participantCount : 0, readCount !== null && readCount !== void 0 ? readCount : 0, message.status))) : (React.createElement("div", { className: react.mergeStyles(noMessageStatusStyle) }));
12058
12147
  }, [message, statusToRender]);
12059
- const shouldShowAvatar = React.useMemo(() => {
12060
- return message.attached === 'top' || message.attached === false;
12061
- }, [message.attached]);
12062
12148
  const attached = React.useMemo(() => {
12063
- return shouldShowAvatar ? 'top' : 'center';
12064
- }, [shouldShowAvatar]);
12149
+ return getFluentUIAttachedValue(message.attached);
12150
+ }, [message.attached]);
12065
12151
  const myMessageRootProps = React.useMemo(() => {
12066
12152
  return {
12067
12153
  // myChatItemMessageContainer used in className and style prop as style prop can't handle CSS selectors
@@ -12097,6 +12183,166 @@ const FluentChatMessageComponentWrapper = (props) => {
12097
12183
  showMessageStatus,
12098
12184
  styles
12099
12185
  ]);
12186
+ // Fluent UI message components are used here as for default message renderer,
12187
+ // timestamp and author name should be shown but they aren't shown for custom renderer.
12188
+ // More investigations are needed to check if this can be simplified with states.
12189
+ // Status and avatar should be shown for both custom and default renderers.
12190
+ return (React.createElement(reactChat.ChatMyMessage, { attached: attached, root: myMessageRootProps, body: myMessageBodyProps, statusIcon: myMessageStatusIcon }, messageRenderer(Object.assign({}, props))));
12191
+ };
12192
+
12193
+ // Copyright (c) Microsoft Corporation.
12194
+ // Licensed under the MIT License.
12195
+ /** @private */
12196
+ const MessageBubble = (props) => {
12197
+ const ids = useIdentifiers();
12198
+ const theme = useTheme();
12199
+ const locale = useLocale$1();
12200
+ const { userId, message, showDate, messageContainerStyle, strings,
12201
+ /* @conditional-compile-remove(file-sharing) */
12202
+ fileDownloadHandler,
12203
+ /* @conditional-compile-remove(image-overlay) */
12204
+ inlineImageOptions, shouldOverlapAvatarAndMessage,
12205
+ /* @conditional-compile-remove(file-sharing) */
12206
+ onRenderFileDownloads,
12207
+ /* @conditional-compile-remove(mention) */
12208
+ mentionDisplayOptions, onDisplayDateTimeString } = props;
12209
+ const formattedTimestamp = React.useMemo(() => {
12210
+ const defaultTimeStamp = message.createdOn
12211
+ ? generateDefaultTimestamp(message.createdOn, showDate, strings)
12212
+ : undefined;
12213
+ const customTimestamp = message.createdOn
12214
+ ? generateCustomizedTimestamp(message.createdOn, locale, onDisplayDateTimeString)
12215
+ : '';
12216
+ return customTimestamp || defaultTimeStamp;
12217
+ }, [locale, message.createdOn, onDisplayDateTimeString, showDate, strings]);
12218
+ const getMessageDetails = React.useCallback(() => {
12219
+ return getMessageEditedDetails(message, theme, strings.editedTag);
12220
+ }, [strings.editedTag, theme, message]);
12221
+ const getContent = React.useCallback(() => {
12222
+ return getMessageBubbleContent(message, strings, userId,
12223
+ /* @conditional-compile-remove(image-overlay) */
12224
+ inlineImageOptions,
12225
+ /* @conditional-compile-remove(file-sharing) */
12226
+ onRenderFileDownloads,
12227
+ /* @conditional-compile-remove(mention) */
12228
+ mentionDisplayOptions,
12229
+ /* @conditional-compile-remove(file-sharing) */
12230
+ fileDownloadHandler);
12231
+ }, [
12232
+ /* @conditional-compile-remove(file-sharing) */ fileDownloadHandler,
12233
+ /* @conditional-compile-remove(image-overlay) */ inlineImageOptions,
12234
+ /* @conditional-compile-remove(mention) */ mentionDisplayOptions,
12235
+ message,
12236
+ /* @conditional-compile-remove(file-sharing) */ onRenderFileDownloads,
12237
+ strings,
12238
+ userId
12239
+ ]);
12240
+ const isBlockedMessage = /* @conditional-compile-remove(data-loss-prevention) */ message.messageType === 'blocked';
12241
+ const chatMessageCommonStyles = useChatMessageCommonStyles();
12242
+ const chatMessageStyles = useChatMessageStyles();
12243
+ const chatItemMessageContainerClassName = reactComponents.mergeClasses(
12244
+ // messageContainerStyle used in className and style prop as style prop can't handle CSS selectors
12245
+ chatMessageStyles.body,
12246
+ // disable placeholder functionality for GA releases as it might confuse users
12247
+ /* @conditional-compile-remove(teams-inline-images-and-file-sharing) */
12248
+ chatMessageStyles.bodyWithPlaceholderImage, isBlockedMessage
12249
+ ? chatMessageCommonStyles.blocked
12250
+ : props.message.status === 'failed'
12251
+ ? chatMessageCommonStyles.failed
12252
+ : undefined, shouldOverlapAvatarAndMessage ? chatMessageStyles.avatarOverlap : chatMessageStyles.avatarNoOverlap, message.attached === 'top' || message.attached === false
12253
+ ? chatMessageStyles.bodyWithAvatar
12254
+ : chatMessageStyles.bodyWithoutAvatar, react.mergeStyles(messageContainerStyle));
12255
+ const attached = message.attached === true ? 'center' : message.attached === 'bottom' ? 'bottom' : 'top';
12256
+ const chatMessage = (React.createElement(React.Fragment, null,
12257
+ React.createElement("div", { key: props.message.messageId },
12258
+ React.createElement(reactChat.ChatMessage, { attached: attached, key: props.message.messageId, root: {
12259
+ className: chatMessageStyles.root,
12260
+ // make body not focusable to remove repetitions from narrators.
12261
+ // inner components are already focusable
12262
+ tabIndex: -1,
12263
+ role: 'none'
12264
+ }, author: React.createElement(react.Text, { className: chatMessageAuthorStyle }, message.senderDisplayName), body: {
12265
+ className: chatItemMessageContainerClassName,
12266
+ style: Object.assign({}, createStyleFromV8Style(messageContainerStyle))
12267
+ }, "data-ui-id": "chat-composite-message", timestamp: React.createElement(react.Text, { className: chatMessageDateStyle, "data-ui-id": ids.messageTimestamp }, formattedTimestamp), details: getMessageDetails() }, getContent()))));
12268
+ return chatMessage;
12269
+ };
12270
+ /** @private */
12271
+ const ChatMessageComponentAsMessageBubble = React.memo(MessageBubble);
12272
+
12273
+ // Copyright (c) Microsoft Corporation.
12274
+ // Licensed under the MIT License.
12275
+ /**
12276
+ * The component for rendering a chat message using Fluent UI components
12277
+ * and handling default and custom renderers.
12278
+ * This component handles rendering for chat message body, avatar and message status.
12279
+ * The chat message body, avatar and message status should be shown for both default and custom renderers.
12280
+ *
12281
+ * @private
12282
+ */
12283
+ const FluentChatMessageComponent = (props) => {
12284
+ const { message, styles, shouldOverlapAvatarAndMessage, onRenderMessage, onRenderAvatar,
12285
+ /* @conditional-compile-remove(date-time-customization) */
12286
+ onDisplayDateTimeString,
12287
+ /* @conditional-compile-remove(image-overlay) */
12288
+ inlineImageOptions,
12289
+ /* @conditional-compile-remove(file-sharing) */
12290
+ fileDownloadHandler, userId,
12291
+ /* @conditional-compile-remove(file-sharing) */
12292
+ onRenderFileDownloads,
12293
+ /* @conditional-compile-remove(mention) */
12294
+ mentionOptions } = props;
12295
+ const chatMessageRenderStyles = useChatMessageRenderStyles();
12296
+ const onRenderFileDownloadsMemo = React.useMemo(() => {
12297
+ /* @conditional-compile-remove(file-sharing) */
12298
+ return onRenderFileDownloads;
12299
+ }, [/* @conditional-compile-remove(file-sharing) */ onRenderFileDownloads]);
12300
+ // To rerender the defaultChatMessageRenderer if app running across days(every new day chat time stamp
12301
+ // needs to be regenerated), the dependency on "new Date().toDateString()"" is added.
12302
+ const defaultChatMessageRenderer = React.useCallback((messageProps) => {
12303
+ if (messageProps.message.messageType === 'chat' ||
12304
+ /* @conditional-compile-remove(data-loss-prevention) */ messageProps.message.messageType === 'blocked') {
12305
+ return (React.createElement(ChatMessageComponentAsMessageBubble, Object.assign({}, messageProps, {
12306
+ /* @conditional-compile-remove(file-sharing) */
12307
+ onRenderFileDownloads: onRenderFileDownloadsMemo,
12308
+ /* @conditional-compile-remove(file-sharing) */
12309
+ strings: messageProps.strings, message: messageProps.message, userId: userId, shouldOverlapAvatarAndMessage: shouldOverlapAvatarAndMessage,
12310
+ /* @conditional-compile-remove(date-time-customization) */
12311
+ onDisplayDateTimeString: onDisplayDateTimeString,
12312
+ /* @conditional-compile-remove(image-overlay) */
12313
+ inlineImageOptions: inlineImageOptions,
12314
+ /* @conditional-compile-remove(file-sharing) */
12315
+ fileDownloadHandler: fileDownloadHandler,
12316
+ /* @conditional-compile-remove(mention) */
12317
+ mentionDisplayOptions: mentionOptions === null || mentionOptions === void 0 ? void 0 : mentionOptions.displayOptions })));
12318
+ }
12319
+ return React.createElement(React.Fragment, null);
12320
+ }, [
12321
+ onRenderFileDownloadsMemo,
12322
+ shouldOverlapAvatarAndMessage,
12323
+ userId,
12324
+ /* @conditional-compile-remove(date-time-customization) */
12325
+ onDisplayDateTimeString,
12326
+ /* @conditional-compile-remove(image-overlay) */
12327
+ inlineImageOptions,
12328
+ /* @conditional-compile-remove(file-sharing) */
12329
+ fileDownloadHandler,
12330
+ /* @conditional-compile-remove(mention) */
12331
+ mentionOptions,
12332
+ // eslint-disable-next-line react-hooks/exhaustive-deps
12333
+ new Date().toDateString()
12334
+ ]);
12335
+ const messageRenderer = React.useCallback((messageProps) => {
12336
+ return onRenderMessage === undefined
12337
+ ? defaultChatMessageRenderer(Object.assign({}, messageProps))
12338
+ : onRenderMessage(messageProps, defaultChatMessageRenderer);
12339
+ }, [defaultChatMessageRenderer, onRenderMessage]);
12340
+ const shouldShowAvatar = React.useMemo(() => {
12341
+ return message.attached === 'top' || message.attached === false;
12342
+ }, [message.attached]);
12343
+ const attached = React.useMemo(() => {
12344
+ return getFluentUIAttachedValue(message.attached);
12345
+ }, [message.attached]);
12100
12346
  const messageRootProps = React.useMemo(() => {
12101
12347
  return { className: reactComponents.mergeClasses(chatMessageRenderStyles.rootMessage, chatMessageRenderStyles.rootCommon) };
12102
12348
  }, [chatMessageRenderStyles.rootCommon, chatMessageRenderStyles.rootMessage]);
@@ -12133,74 +12379,10 @@ const FluentChatMessageComponentWrapper = (props) => {
12133
12379
  // Fluent UI message components are used here as for default message renderer,
12134
12380
  // timestamp and author name should be shown but they aren't shown for custom renderer.
12135
12381
  // More investigations are needed to check if this can be simplified with states.
12136
- // Status and avatar should be shown for both custom and default renderers.
12137
- if (message.mine === true) {
12138
- return (React.createElement("div", null,
12139
- React.createElement(reactChat.ChatMyMessage, { attached: attached, root: myMessageRootProps, body: myMessageBodyProps, statusIcon: myMessageStatusIcon }, messageRenderer(Object.assign({}, props)))));
12140
- }
12141
- else {
12142
- return (React.createElement("div", null,
12143
- React.createElement(reactChat.ChatMessage, { attached: attached, root: messageRootProps, body: messageBodyProps, avatar: avatar }, messageRenderer(Object.assign({}, props)))));
12144
- }
12145
- };
12146
-
12147
- // Copyright (c) Microsoft Corporation.
12148
- // Licensed under the MIT License.
12149
- /**
12150
- * @private
12151
- */
12152
- const systemMessageIconStyle = react.mergeStyles({
12153
- marginInlineEnd: '0.688rem'
12154
- });
12155
-
12156
- // Copyright (c) Microsoft Corporation.
12157
- // Licensed under the MIT License.
12158
- /**
12159
- * @private
12160
- */
12161
- const SystemMessage = (props) => {
12162
- const { iconName, content } = props;
12163
- const Icon = React.createElement(react.FontIcon, { iconName: iconName, className: react.mergeStyles(systemMessageIconStyle) });
12164
- return (React.createElement(react.Stack, { horizontal: true, className: react.mergeStyles(props === null || props === void 0 ? void 0 : props.containerStyle), tabIndex: 0 },
12165
- Icon,
12166
- React.createElement(react.Text, { style: { wordBreak: 'break-word' }, role: "status", title: content, variant: 'small' }, content)));
12382
+ // Avatar should be shown for both custom and default renderers.
12383
+ return (React.createElement(reactChat.ChatMessage, { attached: attached, root: messageRootProps, body: messageBodyProps, avatar: avatar }, messageRenderer(Object.assign({}, props))));
12167
12384
  };
12168
12385
 
12169
- // Copyright (c) Microsoft Corporation.
12170
- // Licensed under the MIT License.
12171
- /**
12172
- * @private
12173
- */
12174
- const DefaultSystemMessage = (props) => {
12175
- var _a;
12176
- const message = props.message;
12177
- switch (message.messageType) {
12178
- case 'system':
12179
- switch (message.systemMessageType) {
12180
- case 'content':
12181
- return (React.createElement(SystemMessage, { iconName: (message.iconName ? message.iconName : ''), content: (_a = message.content) !== null && _a !== void 0 ? _a : '', containerStyle: props === null || props === void 0 ? void 0 : props.messageContainerStyle }));
12182
- case 'participantAdded':
12183
- case 'participantRemoved':
12184
- return (React.createElement(ParticipantSystemMessageComponent, { message: message, style: props.messageContainerStyle, defaultName: props.strings.noDisplayNameSub }));
12185
- }
12186
- }
12187
- return React.createElement(React.Fragment, null);
12188
- };
12189
- const ParticipantSystemMessageComponent = ({ message, style, defaultName }) => {
12190
- const { strings } = useLocale$1();
12191
- const participantsStr = generateParticipantsStr(message.participants, defaultName);
12192
- const messageSuffix = message.systemMessageType === 'participantAdded'
12193
- ? strings.messageThread.participantJoined
12194
- : strings.messageThread.participantLeft;
12195
- if (participantsStr !== '') {
12196
- return (React.createElement(SystemMessage, { iconName: (message.iconName ? message.iconName : ''), content: `${participantsStr} ${messageSuffix}`, containerStyle: style }));
12197
- }
12198
- return React.createElement(React.Fragment, null);
12199
- };
12200
- const generateParticipantsStr = (participants, defaultName) => participants
12201
- .map((participant) => `${!participant.displayName || participant.displayName === '' ? defaultName : participant.displayName}`)
12202
- .join(', ');
12203
-
12204
12386
  // Copyright (c) Microsoft Corporation.
12205
12387
  // Licensed under the MIT License.
12206
12388
  /**
@@ -12227,7 +12409,7 @@ const ChatMessageComponentWrapper = (props) => {
12227
12409
  : styles === null || styles === void 0 ? void 0 : styles.myChatMessageContainer;
12228
12410
  const blockedMessageStyle = styles === null || styles === void 0 ? void 0 : styles.blockedMessageContainer;
12229
12411
  const messageContainerStyle = message.mine ? myChatMessageStyle : blockedMessageStyle;
12230
- return (React.createElement(FluentChatMessageComponentWrapper, Object.assign({}, props, { message: message, messageContainerStyle: messageContainerStyle })));
12412
+ return fluentChatComponent(Object.assign(Object.assign({}, props), { message: message, messageContainerStyle: messageContainerStyle }));
12231
12413
  }
12232
12414
  switch (message.messageType) {
12233
12415
  case 'chat': {
@@ -12236,7 +12418,7 @@ const ChatMessageComponentWrapper = (props) => {
12236
12418
  : styles === null || styles === void 0 ? void 0 : styles.myChatMessageContainer;
12237
12419
  const chatMessageStyle = styles === null || styles === void 0 ? void 0 : styles.chatMessageContainer;
12238
12420
  const messageContainerStyle = message.mine ? myChatMessageStyle : chatMessageStyle;
12239
- return (React.createElement(FluentChatMessageComponentWrapper, Object.assign({}, props, { message: message, messageContainerStyle: messageContainerStyle })));
12421
+ return fluentChatComponent(Object.assign(Object.assign({}, props), { message: message, messageContainerStyle: messageContainerStyle }));
12240
12422
  }
12241
12423
  case 'system': {
12242
12424
  const messageContainerStyle = styles === null || styles === void 0 ? void 0 : styles.systemMessageContainer;
@@ -12250,6 +12432,14 @@ const ChatMessageComponentWrapper = (props) => {
12250
12432
  }
12251
12433
  }
12252
12434
  };
12435
+ const fluentChatComponent = (props) => {
12436
+ if (props.message.mine === true) {
12437
+ return React.createElement(FluentChatMyMessageComponent, Object.assign({}, props));
12438
+ }
12439
+ else {
12440
+ return React.createElement(FluentChatMessageComponent, Object.assign({}, props));
12441
+ }
12442
+ };
12253
12443
 
12254
12444
  // Copyright (c) Microsoft Corporation.
12255
12445
  // Licensed under the MIT License.
@@ -14406,7 +14596,7 @@ function useLongPress(props) {
14406
14596
  * @returns VideoGalleryRemoteParticipant[] {@link @azure/communication-react#VideoGalleryRemoteParticipant}
14407
14597
  */
14408
14598
  const smartDominantSpeakerParticipants = (args) => {
14409
- const { participants, dominantSpeakers = [], lastVisibleParticipants = [], maxDominantSpeakers } = args;
14599
+ const { participants, dominantSpeakers = [], currentParticipants = [], maxDominantSpeakers } = args;
14410
14600
  // Don't apply any logic if total number of video streams is less than max dominant speakers.
14411
14601
  if (participants.length <= maxDominantSpeakers) {
14412
14602
  return participants;
@@ -14414,7 +14604,7 @@ const smartDominantSpeakerParticipants = (args) => {
14414
14604
  const participantsMap = participantsById(participants);
14415
14605
  // Only use the Max allowed dominant speakers that exist in participants
14416
14606
  const dominantSpeakerIds = dominantSpeakers.filter((id) => !!participantsMap[id]).slice(0, maxDominantSpeakers);
14417
- const newVisibleParticipantIds = lastVisibleParticipants.map((p) => p.userId).slice(0, maxDominantSpeakers);
14607
+ const newVisibleParticipantIds = currentParticipants.map((p) => p.userId).slice(0, maxDominantSpeakers);
14418
14608
  const newDominantSpeakerIds = dominantSpeakerIds.filter((id) => !newVisibleParticipantIds.includes(id));
14419
14609
  // Remove participants that are no longer dominant and replace them with new dominant speakers.
14420
14610
  for (let index = 0; index < maxDominantSpeakers; index++) {
@@ -14450,99 +14640,61 @@ const participantsById = (participants) => {
14450
14640
  const DEFAULT_MAX_OVERFLOW_GALLERY_DOMINANT_SPEAKERS = 6;
14451
14641
  const DEFAULT_MAX_VIDEO_SREAMS = 4;
14452
14642
  const MAX_GRID_PARTICIPANTS_NOT_LARGE_GALLERY = 9;
14453
- const _useOrganizedParticipants = (props) => {
14454
- const visibleGridParticipants = React.useRef([]);
14455
- const visibleOverflowGalleryParticipants = React.useRef([]);
14456
- const { remoteParticipants = [], dominantSpeakers = [], maxRemoteVideoStreams = DEFAULT_MAX_VIDEO_SREAMS, maxOverflowGalleryDominantSpeakers = DEFAULT_MAX_OVERFLOW_GALLERY_DOMINANT_SPEAKERS, isScreenShareActive = false, pinnedParticipantUserIds = [], layout } = props;
14457
- const calculateMaxRemoteVideoStreams = () => {
14458
- if (maxRemoteVideoStreams > MAX_GRID_PARTICIPANTS_NOT_LARGE_GALLERY) {
14459
- return MAX_GRID_PARTICIPANTS_NOT_LARGE_GALLERY;
14460
- }
14461
- else {
14462
- return maxRemoteVideoStreams;
14463
- }
14464
- };
14465
- const maxRemoteVideoStreamsToUse = calculateMaxRemoteVideoStreams();
14643
+ const getOrganizedParticipants = (props) => {
14644
+ const { remoteParticipants = [], dominantSpeakers = [], maxRemoteVideoStreams = DEFAULT_MAX_VIDEO_SREAMS, maxOverflowGalleryDominantSpeakers = DEFAULT_MAX_OVERFLOW_GALLERY_DOMINANT_SPEAKERS, isScreenShareActive = false, layout, previousGridParticipants = [], previousOverflowParticipants = [] } = props;
14645
+ const maxRemoteVideoStreamsToUse = maxRemoteVideoStreams > MAX_GRID_PARTICIPANTS_NOT_LARGE_GALLERY
14646
+ ? MAX_GRID_PARTICIPANTS_NOT_LARGE_GALLERY
14647
+ : maxRemoteVideoStreams;
14648
+ const remoteParticipantsOrdered = putVideoParticipantsFirst(remoteParticipants);
14466
14649
  const videoParticipants = remoteParticipants.filter((p) => { var _a; return (_a = p.videoStream) === null || _a === void 0 ? void 0 : _a.isAvailable; });
14467
- const participantsToSortTrampoline = () => {
14468
- return layout !== 'floatingLocalVideo' ? putVideoParticipantsFirst(remoteParticipants) : videoParticipants;
14469
- };
14470
- visibleGridParticipants.current =
14471
- pinnedParticipantUserIds.length > 0 || isScreenShareActive
14472
- ? []
14473
- : smartDominantSpeakerParticipants({
14474
- participants: participantsToSortTrampoline(),
14475
- dominantSpeakers,
14476
- lastVisibleParticipants: visibleGridParticipants.current,
14477
- maxDominantSpeakers: maxRemoteVideoStreamsToUse
14478
- }).slice(0, maxRemoteVideoStreamsToUse);
14650
+ const participants = layout === 'floatingLocalVideo' && videoParticipants.length > 0 ? videoParticipants : remoteParticipantsOrdered;
14651
+ let newGridParticipants = smartDominantSpeakerParticipants({
14652
+ participants: participants,
14653
+ dominantSpeakers,
14654
+ currentParticipants: previousGridParticipants,
14655
+ maxDominantSpeakers: maxRemoteVideoStreamsToUse
14656
+ }).slice(0, maxRemoteVideoStreamsToUse);
14479
14657
  const dominantSpeakerToGrid = layout === 'speaker'
14480
14658
  ? dominantSpeakers && dominantSpeakers[0]
14481
- ? visibleGridParticipants.current.filter((p) => p.userId === dominantSpeakers[0])
14482
- : [visibleGridParticipants.current[0]]
14659
+ ? newGridParticipants.filter((p) => p.userId === dominantSpeakers[0])
14660
+ : [newGridParticipants[0]]
14483
14661
  : [];
14484
14662
  if (dominantSpeakerToGrid[0]) {
14485
- visibleGridParticipants.current = dominantSpeakerToGrid;
14663
+ newGridParticipants = dominantSpeakerToGrid;
14486
14664
  }
14487
- const visibleGridParticipantsSet = new Set(visibleGridParticipants.current.map((p) => p.userId));
14488
- const remoteParticipantsOrdered = putVideoParticipantsFirst(remoteParticipants);
14665
+ const gridParticipantSet = new Set(newGridParticipants.map((p) => p.userId));
14489
14666
  /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14490
14667
  const callingParticipants = remoteParticipantsOrdered.filter((p) => p.state === ('Connecting' ));
14491
14668
  /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14492
14669
  const callingParticipantsSet = new Set(callingParticipants.map((p) => p.userId));
14493
- visibleOverflowGalleryParticipants.current = smartDominantSpeakerParticipants({
14494
- participants: remoteParticipantsOrdered.filter((p) => !visibleGridParticipantsSet.has(p.userId) &&
14670
+ const newOverflowGalleryParticipants = smartDominantSpeakerParticipants({
14671
+ participants: remoteParticipantsOrdered.filter((p) => !gridParticipantSet.has(p.userId) &&
14495
14672
  /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */ !callingParticipantsSet.has(p.userId)),
14496
14673
  dominantSpeakers: dominantSpeakers,
14497
- lastVisibleParticipants: visibleOverflowGalleryParticipants.current,
14674
+ currentParticipants: previousOverflowParticipants,
14498
14675
  maxDominantSpeakers: maxOverflowGalleryDominantSpeakers
14499
14676
  });
14500
- const getGridParticipants = React.useCallback(() => {
14501
- if (isScreenShareActive) {
14502
- return [];
14503
- }
14504
- // if we have no grid participants we need to cap the max number of overflowGallery participants in the grid
14505
- // we will use the max streams provided to the function to find the max participants that can go in the grid
14506
- // if there are less participants than max streams then we will use all participants including joining in the grid
14507
- /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14508
- return visibleGridParticipants.current.length > 0
14509
- ? visibleGridParticipants.current
14510
- : visibleOverflowGalleryParticipants.current.length > maxRemoteVideoStreamsToUse
14511
- ? visibleOverflowGalleryParticipants.current.slice(0, maxRemoteVideoStreamsToUse)
14512
- : visibleOverflowGalleryParticipants.current.slice(0, maxRemoteVideoStreamsToUse).concat(callingParticipants);
14513
- }, [
14514
- /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */ callingParticipants,
14677
+ const gridParticipants = getGridParticipants({
14515
14678
  isScreenShareActive,
14516
- maxRemoteVideoStreamsToUse
14517
- ]);
14518
- const gridParticipants = getGridParticipants();
14519
- const getOverflowGalleryRemoteParticipants = React.useCallback(() => {
14520
- if (isScreenShareActive) {
14521
- // If screen sharing is active, assign video and audio participants as overflow gallery participants
14522
- /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14523
- return visibleGridParticipants.current.concat(visibleOverflowGalleryParticipants.current.concat(callingParticipants));
14524
- }
14525
- else {
14526
- // If screen sharing is not active, then assign all video tiles as grid tiles.
14527
- // If there are no video tiles, then assign audio tiles as grid tiles.
14528
- // if there are more overflow tiles than max streams then find the tiles that don't fit in the grid and put them in overflow
14529
- // overflow should be empty if total participants including calling participants is less than max streams
14530
- /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14531
- return visibleGridParticipants.current.length > 0
14532
- ? visibleOverflowGalleryParticipants.current.concat(callingParticipants)
14533
- : visibleOverflowGalleryParticipants.current.length > maxRemoteVideoStreamsToUse
14534
- ? visibleOverflowGalleryParticipants.current.slice(maxRemoteVideoStreamsToUse).concat(callingParticipants)
14535
- : [];
14536
- }
14537
- }, [
14538
- /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */ callingParticipants,
14679
+ gridParticipants: newGridParticipants,
14680
+ overflowGalleryParticipants: newOverflowGalleryParticipants,
14681
+ maxRemoteVideoStreams: maxRemoteVideoStreamsToUse,
14682
+ /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */ callingParticipants
14683
+ });
14684
+ const overflowGalleryParticipants = getOverflowGalleryRemoteParticipants({
14539
14685
  isScreenShareActive,
14540
- maxRemoteVideoStreamsToUse
14541
- ]);
14542
- const overflowGalleryParticipants = getOverflowGalleryRemoteParticipants();
14543
- return { gridParticipants, overflowGalleryParticipants: overflowGalleryParticipants };
14686
+ gridParticipants: newGridParticipants,
14687
+ overflowGalleryParticipants: newOverflowGalleryParticipants,
14688
+ maxRemoteVideoStreams: maxRemoteVideoStreamsToUse,
14689
+ /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */ callingParticipants
14690
+ });
14691
+ return { gridParticipants, overflowGalleryParticipants };
14544
14692
  };
14545
- const _useOrganizedParticipantsWithFocusedParticipants = (props) => {
14693
+ /**
14694
+ * Hook to determine which participants should be in grid and overflow gallery and their order respectively
14695
+ * @private
14696
+ */
14697
+ const useOrganizedParticipants = (props) => {
14546
14698
  var _a, _b;
14547
14699
  // map remote participants by userId
14548
14700
  const remoteParticipantMap = props.remoteParticipants.reduce((map, remoteParticipant) => {
@@ -14550,35 +14702,62 @@ const _useOrganizedParticipantsWithFocusedParticipants = (props) => {
14550
14702
  return map;
14551
14703
  }, {});
14552
14704
  const spotlightedParticipantUserIds = (_a = props.spotlightedParticipantUserIds) !== null && _a !== void 0 ? _a : [];
14553
- // declare focused participant user ids as spotlighted participants user ids followed by
14554
- // pinned participants user ids
14555
- const focusedParticipantUserIds = [
14556
- ...new Set(spotlightedParticipantUserIds.concat((_b = props.pinnedParticipantUserIds) !== null && _b !== void 0 ? _b : []))
14557
- ];
14705
+ const pinnedParticipantUserIds = (_b = props.pinnedParticipantUserIds) !== null && _b !== void 0 ? _b : [];
14706
+ // declare set of focused participant user ids as spotlighted participants user ids followed by
14707
+ // pinned participants user ids which is deduplicated while maintaining order
14708
+ const focusedParticipantUserIdSet = new Set(spotlightedParticipantUserIds.concat(pinnedParticipantUserIds).filter((p) => remoteParticipantMap[p]));
14558
14709
  // get focused participants from map of remote participants in the order of the user ids
14559
- const focusedParticipants = [];
14560
- focusedParticipantUserIds.forEach((id) => {
14561
- const pinnedParticipant = remoteParticipantMap[id];
14562
- if (pinnedParticipant) {
14563
- focusedParticipants.push(pinnedParticipant);
14564
- }
14565
- });
14566
- // get unfocused participants by filtering out set of focused participant user ids from all remote participants
14567
- const focusedParticipantUserIdSet = new Set(focusedParticipantUserIds);
14710
+ const focusedParticipants = [...focusedParticipantUserIdSet].map((p) => remoteParticipantMap[p]);
14711
+ const currentGridParticipants = React.useRef([]);
14712
+ const currentOverflowGalleryParticipants = React.useRef([]);
14568
14713
  const unfocusedParticipants = props.remoteParticipants.filter((p) => !focusedParticipantUserIdSet.has(p.userId));
14569
14714
  const useOrganizedParticipantsProps = Object.assign(Object.assign({}, props), {
14570
- // if there are pinned participants then we should only consider unpinned participants
14571
- remoteParticipants: unfocusedParticipants });
14572
- const useOrganizedParticipantsResult = _useOrganizedParticipants(useOrganizedParticipantsProps);
14573
- if (focusedParticipants.length === 0) {
14574
- return useOrganizedParticipantsResult;
14715
+ // if there are focused participants then leave no room in the grid by setting maxGridParticipants to 0
14716
+ maxRemoteVideoStreams: focusedParticipants.length > 0 || props.isScreenShareActive ? 0 : props.maxRemoteVideoStreams, remoteParticipants: unfocusedParticipants, previousGridParticipants: currentGridParticipants.current, previousOverflowParticipants: currentOverflowGalleryParticipants.current });
14717
+ const useOrganizedParticipantsResult = getOrganizedParticipants(useOrganizedParticipantsProps);
14718
+ currentGridParticipants.current = useOrganizedParticipantsResult.gridParticipants;
14719
+ currentOverflowGalleryParticipants.current = useOrganizedParticipantsResult.overflowGalleryParticipants;
14720
+ return focusedParticipants.length > 0
14721
+ ? {
14722
+ gridParticipants: props.isScreenShareActive ? [] : focusedParticipants,
14723
+ overflowGalleryParticipants: props.isScreenShareActive
14724
+ ? focusedParticipants.concat(useOrganizedParticipantsResult.overflowGalleryParticipants)
14725
+ : useOrganizedParticipantsResult.overflowGalleryParticipants
14726
+ }
14727
+ : useOrganizedParticipantsResult;
14728
+ };
14729
+ const getGridParticipants = (args) => {
14730
+ if (args.isScreenShareActive) {
14731
+ return [];
14732
+ }
14733
+ // if we have no grid participants we need to cap the max number of overflowGallery participants in the grid
14734
+ // we will use the max streams provided to the function to find the max participants that can go in the grid
14735
+ // if there are less participants than max streams then we will use all participants including joining in the grid
14736
+ /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14737
+ return args.gridParticipants.length > 0
14738
+ ? args.gridParticipants
14739
+ : args.overflowGalleryParticipants.length > args.maxRemoteVideoStreams
14740
+ ? args.overflowGalleryParticipants.slice(0, args.maxRemoteVideoStreams)
14741
+ : args.overflowGalleryParticipants.slice(0, args.maxRemoteVideoStreams).concat(args.callingParticipants);
14742
+ };
14743
+ const getOverflowGalleryRemoteParticipants = (args) => {
14744
+ if (args.isScreenShareActive) {
14745
+ // If screen sharing is active, assign video and audio participants as overflow gallery participants
14746
+ /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14747
+ return args.gridParticipants.concat(args.overflowGalleryParticipants.concat(args.callingParticipants));
14748
+ }
14749
+ else {
14750
+ // If screen sharing is not active, then assign all video tiles as grid tiles.
14751
+ // If there are no video tiles, then assign audio tiles as grid tiles.
14752
+ // if there are more overflow tiles than max streams then find the tiles that don't fit in the grid and put them in overflow
14753
+ // overflow should be empty if total participants including calling participants is less than max streams
14754
+ /* @conditional-compile-remove(PSTN-calls) */ /* @conditional-compile-remove(one-to-n-calling) */
14755
+ return args.gridParticipants.length > 0
14756
+ ? args.overflowGalleryParticipants.concat(args.callingParticipants)
14757
+ : args.overflowGalleryParticipants.length > args.maxRemoteVideoStreams
14758
+ ? args.overflowGalleryParticipants.slice(args.maxRemoteVideoStreams).concat(args.callingParticipants)
14759
+ : [];
14575
14760
  }
14576
- return {
14577
- gridParticipants: props.isScreenShareActive ? [] : focusedParticipants,
14578
- overflowGalleryParticipants: props.isScreenShareActive
14579
- ? focusedParticipants.concat(useOrganizedParticipantsResult.overflowGalleryParticipants)
14580
- : useOrganizedParticipantsResult.gridParticipants.concat(useOrganizedParticipantsResult.overflowGalleryParticipants)
14581
- };
14582
14761
  };
14583
14762
  const putVideoParticipantsFirst = (remoteParticipants) => {
14584
14763
  const videoParticipants = [];
@@ -14595,13 +14774,6 @@ const putVideoParticipantsFirst = (remoteParticipants) => {
14595
14774
  const remoteParticipantSortedByVideo = videoParticipants.concat(audioParticipants);
14596
14775
  return remoteParticipantSortedByVideo;
14597
14776
  };
14598
- /**
14599
- * Hook to determine which participants should be in grid and overflow gallery and their order respectively
14600
- * @private
14601
- */
14602
- const useOrganizedParticipants = (args) => {
14603
- return _useOrganizedParticipantsWithFocusedParticipants(args);
14604
- };
14605
14777
  /* @conditional-compile-remove(reaction) */
14606
14778
  /**
14607
14779
  * @private
@@ -21142,38 +21314,40 @@ const _videoGalleryRemoteParticipantsMemo = (remoteParticipants, isHideAttendeeN
21142
21314
  /* @conditional-compile-remove(reaction) */
21143
21315
  const remoteParticipantReaction = memoizedConvertToVideoTileReaction(participant.reactionState);
21144
21316
  return memoizedFn(toFlatCommunicationIdentifier(participant.identifier), participant.isMuted, checkIsSpeaking(participant), participant.videoStreams, state, displayName, participant.raisedHand,
21317
+ /* @conditional-compile-remove(ppt-live) */
21318
+ participant.contentSharingStream,
21145
21319
  /* @conditional-compile-remove(reaction) */
21146
21320
  remoteParticipantReaction,
21147
21321
  /* @conditional-compile-remove(spotlight) */
21148
- participant.spotlight,
21149
- /* @conditional-compile-remove(ppt-live) */
21150
- participant.contentSharingStream);
21322
+ participant.spotlight);
21151
21323
  }));
21152
21324
  });
21153
21325
  };
21154
21326
  const memoizedAllConvertRemoteParticipant = memoizeFnAll((userId, isMuted, isSpeaking, videoStreams, state, displayName, raisedHand, // temp unknown type to build stable
21327
+ /* @conditional-compile-remove(ppt-live) */
21328
+ contentSharingStream,
21155
21329
  /* @conditional-compile-remove(reaction) */
21156
21330
  reaction, // temp unknown type to build stable
21157
21331
  /* @conditional-compile-remove(spotlight) */
21158
- spotlight, // temp unknown type to build stable
21159
- /* @conditional-compile-remove(ppt-live) */
21160
- contentSharingStream) => {
21332
+ spotlight // temp unknown type to build stable
21333
+ ) => {
21161
21334
  return convertRemoteParticipantToVideoGalleryRemoteParticipant(userId, isMuted, isSpeaking, videoStreams, state, displayName, raisedHand,
21335
+ /* @conditional-compile-remove(ppt-live) */
21336
+ contentSharingStream,
21162
21337
  /* @conditional-compile-remove(reaction) */
21163
21338
  reaction,
21164
21339
  /* @conditional-compile-remove(spotlight) */
21165
- spotlight,
21166
- /* @conditional-compile-remove(ppt-live) */
21167
- contentSharingStream);
21340
+ spotlight);
21168
21341
  });
21169
21342
  /** @private */
21170
21343
  const convertRemoteParticipantToVideoGalleryRemoteParticipant = (userId, isMuted, isSpeaking, videoStreams, state, displayName, raisedHand, // temp unknown type to build stable
21344
+ /* @conditional-compile-remove(ppt-live) */
21345
+ contentSharingStream,
21171
21346
  /* @conditional-compile-remove(reaction) */
21172
21347
  reaction, // temp unknown type to build stable
21173
21348
  /* @conditional-compile-remove(spotlight) */
21174
- spotlight, // temp unknown type to build stable
21175
- /* @conditional-compile-remove(ppt-live) */
21176
- contentSharingStream) => {
21349
+ spotlight // temp unknown type to build stable
21350
+ ) => {
21177
21351
  const rawVideoStreamsArray = Object.values(videoStreams);
21178
21352
  let videoStream = undefined;
21179
21353
  let screenShareStream = undefined;
@@ -21333,6 +21507,8 @@ capabilities) => {
21333
21507
  return {
21334
21508
  screenShareParticipant: screenShareRemoteParticipant
21335
21509
  ? convertRemoteParticipantToVideoGalleryRemoteParticipant(toFlatCommunicationIdentifier(screenShareRemoteParticipant.identifier), screenShareRemoteParticipant.isMuted, checkIsSpeaking(screenShareRemoteParticipant), screenShareRemoteParticipant.videoStreams, screenShareRemoteParticipant.state, screenShareRemoteParticipant.displayName, screenShareRemoteParticipant.raisedHand,
21510
+ /* @conditional-compile-remove(ppt-live) */
21511
+ screenShareRemoteParticipant.contentSharingStream,
21336
21512
  /* @conditional-compile-remove(spotlight) */
21337
21513
  screenShareRemoteParticipant.spotlight)
21338
21514
  : undefined,
@@ -22896,8 +23072,8 @@ class ResourceDownloadQueue {
22896
23072
  });
22897
23073
  }
22898
23074
  downloadAllPreviewUrls(message, operation) {
22899
- var _a;
22900
23075
  return __awaiter$t(this, void 0, void 0, function* () {
23076
+ var _a;
22901
23077
  const attachments = (_a = message.content) === null || _a === void 0 ? void 0 : _a.attachments;
22902
23078
  if (message.type === 'html' && attachments) {
22903
23079
  if (message.resourceCache === undefined) {
@@ -23055,8 +23231,8 @@ let ChatContext$1 = class ChatContext {
23055
23231
  }
23056
23232
  /* @conditional-compile-remove(teams-inline-images-and-file-sharing) */
23057
23233
  downloadResourceToCache(threadId, messageId, resourceUrl) {
23058
- var _a;
23059
23234
  return __awaiter$s(this, void 0, void 0, function* () {
23235
+ var _a;
23060
23236
  let message = (_a = this.getState().threads[threadId]) === null || _a === void 0 ? void 0 : _a.chatMessages[messageId];
23061
23237
  if (message && this._fullsizeImageQueue) {
23062
23238
  if (!message.resourceCache) {
@@ -24650,8 +24826,8 @@ class AzureCommunicationChatAdapter {
24650
24826
  offStateChange(handler) {
24651
24827
  this.context.offStateChange(handler);
24652
24828
  }
24653
- sendMessage(content, options = {}) {
24654
- return __awaiter$n(this, void 0, void 0, function* () {
24829
+ sendMessage(content_1) {
24830
+ return __awaiter$n(this, arguments, void 0, function* (content, options = {}) {
24655
24831
  yield this.asyncTeeErrorToEventEmitter(() => __awaiter$n(this, void 0, void 0, function* () {
24656
24832
  /* @conditional-compile-remove(file-sharing) */
24657
24833
  options.metadata = Object.assign(Object.assign({}, options.metadata), convertFileUploadsUiStateToMessageMetadata(this.context.getState().fileUploads));
@@ -24880,7 +25056,7 @@ const convertEventType = (type) => {
24880
25056
  *
24881
25057
  * @public
24882
25058
  */
24883
- const createAzureCommunicationChatAdapter = ({ endpoint: endpointUrl, userId, displayName, credential, threadId }) => __awaiter$n(void 0, void 0, void 0, function* () {
25059
+ const createAzureCommunicationChatAdapter = (_a) => __awaiter$n(void 0, [_a], void 0, function* ({ endpoint: endpointUrl, userId, displayName, credential, threadId }) {
24884
25060
  return _createAzureCommunicationChatAdapterInner(endpointUrl, userId, displayName, credential, threadId);
24885
25061
  });
24886
25062
  /**
@@ -24888,7 +25064,7 @@ const createAzureCommunicationChatAdapter = ({ endpoint: endpointUrl, userId, di
24888
25064
  *
24889
25065
  * @internal
24890
25066
  */
24891
- const _createAzureCommunicationChatAdapterInner = (endpoint, userId, displayName, credential, threadId, telemetryImplementationHint = 'Chat') => __awaiter$n(void 0, void 0, void 0, function* () {
25067
+ const _createAzureCommunicationChatAdapterInner = (endpoint_1, userId_1, displayName_1, credential_1, threadId_1, ...args_1) => __awaiter$n(void 0, [endpoint_1, userId_1, displayName_1, credential_1, threadId_1, ...args_1], void 0, function* (endpoint, userId, displayName, credential, threadId, telemetryImplementationHint = 'Chat') {
24892
25068
  if (!_isValidIdentifier(userId)) {
24893
25069
  throw new Error('Provided userId is invalid. Please provide valid identifier object.');
24894
25070
  }
@@ -25676,7 +25852,7 @@ const useSelector$2 = (selector, selectorProps) => {
25676
25852
  // Copyright (c) Microsoft Corporation.
25677
25853
  // Licensed under the MIT License.
25678
25854
  /* @conditional-compile-remove(rich-text-editor) */
25679
- const RichTextSendBoxWrapper = React.lazy(() => Promise.resolve().then(function () { return require('./RichTextSendBoxWrapper-BCdGFyar.js'); }).then((module) => ({ default: module.RichTextSendBoxWrapper })));
25855
+ const RichTextSendBoxWrapper = React.lazy(() => Promise.resolve().then(function () { return require('./RichTextSendBoxWrapper-Bw9vaT1f.js'); }).then((module) => ({ default: module.RichTextSendBoxWrapper })));
25680
25856
  /**
25681
25857
  * @private
25682
25858
  */
@@ -35347,8 +35523,8 @@ class AzureCommunicationCallAdapter {
35347
35523
  }
35348
35524
  /* @conditional-compile-remove(PSTN-calls) */
35349
35525
  holdCall() {
35350
- var _a, _b;
35351
35526
  return __awaiter$5(this, void 0, void 0, function* () {
35527
+ var _a, _b;
35352
35528
  if (((_a = this.call) === null || _a === void 0 ? void 0 : _a.state) !== 'LocalHold') {
35353
35529
  if ((_b = this.call) === null || _b === void 0 ? void 0 : _b.isLocalVideoStarted) {
35354
35530
  this.stopCamera().then(() => {
@@ -35363,8 +35539,8 @@ class AzureCommunicationCallAdapter {
35363
35539
  }
35364
35540
  /* @conditional-compile-remove(PSTN-calls) */
35365
35541
  resumeCall() {
35366
- var _a;
35367
35542
  return __awaiter$5(this, void 0, void 0, function* () {
35543
+ var _a;
35368
35544
  if (((_a = this.call) === null || _a === void 0 ? void 0 : _a.state) === 'LocalHold') {
35369
35545
  this.handlers.onToggleHold().then(() => {
35370
35546
  var _a;
@@ -35691,9 +35867,9 @@ function createAzureCommunicationCallAdapter(args) {
35691
35867
  *
35692
35868
  * @internal
35693
35869
  */
35694
- const _createAzureCommunicationCallAdapterInner = ({ userId, displayName, credential, locator, targetCallees,
35870
+ const _createAzureCommunicationCallAdapterInner = (_a) => __awaiter$5(void 0, [_a], void 0, function* ({ userId, displayName, credential, locator, targetCallees,
35695
35871
  /* @conditional-compile-remove(PSTN-calls) */ alternateCallerId,
35696
- /* @conditional-compile-remove(video-background-effects) */ options, telemetryImplementationHint = 'Call' }) => __awaiter$5(void 0, void 0, void 0, function* () {
35872
+ /* @conditional-compile-remove(video-background-effects) */ options, telemetryImplementationHint = 'Call' }) {
35697
35873
  if (!_isValidIdentifier(userId)) {
35698
35874
  throw new Error('Invalid identifier. Please provide valid identifier object.');
35699
35875
  }
@@ -35720,7 +35896,7 @@ const _createAzureCommunicationCallAdapterInner = ({ userId, displayName, creden
35720
35896
  /**
35721
35897
  * @beta
35722
35898
  */
35723
- const createTeamsCallAdapter = ({ userId, credential, locator, options }) => __awaiter$5(void 0, void 0, void 0, function* () {
35899
+ const createTeamsCallAdapter = (_b) => __awaiter$5(void 0, [_b], void 0, function* ({ userId, credential, locator, options }) {
35724
35900
  if (communicationCommon.isCommunicationUserIdentifier(userId)) {
35725
35901
  throw new Error('Communication User identifier is not supported by TeamsCallAdapter, please use our AzureCommunicationCallAdapter.');
35726
35902
  }
@@ -37538,9 +37714,9 @@ class AzureCommunicationCallWithChatAdapter {
37538
37714
  *
37539
37715
  * @public
37540
37716
  */
37541
- const createAzureCommunicationCallWithChatAdapter = ({ userId, displayName, credential, endpoint, locator,
37717
+ const createAzureCommunicationCallWithChatAdapter = (_a) => __awaiter$2(void 0, [_a], void 0, function* ({ userId, displayName, credential, endpoint, locator,
37542
37718
  /* @conditional-compile-remove(PSTN-calls) */ alternateCallerId,
37543
- /* @conditional-compile-remove(video-background-effects) */ callAdapterOptions }) => __awaiter$2(void 0, void 0, void 0, function* () {
37719
+ /* @conditional-compile-remove(video-background-effects) */ callAdapterOptions }) {
37544
37720
  const callAdapterLocator = isTeamsMeetingLinkLocator(locator) ? locator : locator.callLocator;
37545
37721
  const createCallAdapterPromise = _createAzureCommunicationCallAdapterInner({
37546
37722
  userId,
@@ -37684,9 +37860,9 @@ beforeDispose) => {
37684
37860
  *
37685
37861
  * @public
37686
37862
  */
37687
- const createAzureCommunicationCallWithChatAdapterFromClients = ({ callClient, callAgent, callLocator, chatClient, chatThreadClient,
37863
+ const createAzureCommunicationCallWithChatAdapterFromClients = (_b) => __awaiter$2(void 0, [_b], void 0, function* ({ callClient, callAgent, callLocator, chatClient, chatThreadClient,
37688
37864
  /* @conditional-compile-remove(video-background-effects) */
37689
- callAdapterOptions }) => __awaiter$2(void 0, void 0, void 0, function* () {
37865
+ callAdapterOptions }) {
37690
37866
  const createCallAdapterPromise = createAzureCommunicationCallAdapterFromClient(callClient, callAgent, callLocator,
37691
37867
  /* @conditional-compile-remove(video-background-effects) */
37692
37868
  callAdapterOptions);
@@ -38253,4 +38429,4 @@ exports.useTeamsCall = useTeamsCall;
38253
38429
  exports.useTeamsCallAdapter = useTeamsCallAdapter;
38254
38430
  exports.useTeamsCallAgent = useTeamsCallAgent;
38255
38431
  exports.useTheme = useTheme;
38256
- //# sourceMappingURL=index-BfFCLrEO.js.map
38432
+ //# sourceMappingURL=index-DyaTe_Bz.js.map