diff --git a/CHANGELOG.md b/CHANGELOG.md index 6cb785fabe..7a469b93e4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -20,6 +20,8 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0. - Resolves [#XXX](https://github.com/microsoft/BotFramework-WebChat/issues/XXX). Added something, by [@johndoe](https://github.com/johndoe), in PR [#XXX](https://github.com/microsoft/BotFramework-WebChat/pull/XXX) --> +Notes: web developers are advised to use [`~` (tilde range)](https://github.com/npm/node-semver?tab=readme-ov-file#tilde-ranges-123-12-1) to select minor versions, which contains new features and/or fixes. Use [`^` (caret range)](https://github.com/npm/node-semver?tab=readme-ov-file#caret-ranges-123-025-004) to select major versions, which may contains breaking changes. + ## [Unreleased] ### Fixed @@ -122,6 +124,8 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0. - Added `useMakeThumbnail` hook option to create a thumbnail from the file given, by [@compulim](https://github.com/compulim), in PR [#5123](https://github.com/microsoft/BotFramework-WebChat/pull/5123) and [#5122](https://github.com/microsoft/BotFramework-WebChat/pull/5122) - Added `moduleFormat` and `transpiler` build info to `` tag, in PR [#5148](https://github.com/microsoft/BotFramework-WebChat/pull/5148), by [@compulim](https://github.com/compulim) - Added support of rendering HTML-in-Markdown, in PR [#5161](https://github.com/microsoft/BotFramework-WebChat/pull/5161) and PR [#5164](https://github.com/microsoft/BotFramework-WebChat/pull/5164), by [@compulim](https://github.com/compulim), [@beyackle2](https://github.com/beyackle2), and [@OEvgeny](https://github.com/OEvgeny) +- Resolves [#5184](https://github.com/microsoft/BotFramework-WebChat/issues/5184). Added `channelData.webChat.styleOptions.typingIndicatorDuration` to override the default typing indicator duration on a per-activity basis, by [@compulim](https://github.com/compulim), in PR [#5141](https://github.com/microsoft/BotFramework-WebChat/pull/5141) +- Resolves [#4876](https://github.com/microsoft/BotFramework-WebChat/issues/4876) and [#4939](https://github.com/microsoft/BotFramework-WebChat/issues/4939). Added support of livestreaming, by [@compulim](https://github.com/compulim), in PR [#5141](https://github.com/microsoft/BotFramework-WebChat/pull/5141) ### Fixed diff --git a/README.md b/README.md index 63008a8195..8c52afd2df 100644 --- a/README.md +++ b/README.md @@ -19,21 +19,29 @@ Web Chat supports [Content Security Policy (CSP)](https://developer.mozilla.org/ > This section points out important version notes. For further information, please see the related links and check the [`CHANGELOG.md`](https://github.com/microsoft/BotFramework-WebChat/blob/main/CHANGELOG.md) -### 4.17.0 notable changes +Notes: web developers are advised to use [`~` (tilde range)](https://github.com/npm/node-semver?tab=readme-ov-file#tilde-ranges-123-12-1) to select minor versions, which contains new features and/or fixes. Use [`^` (caret range)](https://github.com/npm/node-semver?tab=readme-ov-file#caret-ranges-123-025-004) to select major versions, which may contains breaking changes. -#### Debut of ES Modules +## 4.18.0 notable changes + +### Support livestreaming response + +Bots can now livestream their responses. Before Bot Framework SDK support this feature, bot authors can follow the details in [this pull request](https://github.com/microsoft/BotFramework-WebChat/pull/5141) to construct the livestream responses. + +## 4.17.0 notable changes + +### Debut of ES Modules Web Chat now exports as ES Modules (named exports) along with CommonJS (named and unnamed exports). -#### Improvement to file upload experience +### Improvement to file upload experience End-user can now add a message and confirm before uploading their file to the bot. To opt-out of the new experience, pass `sendAttachmentOn: 'send'` in style options. -#### Theme pack support +### Theme pack support We are excited to add theme pack support. Developers can now pack all their customization in a single package and publish it to NPM. -#### Experimental Fluent UI theme pack +### Experimental Fluent UI theme pack We are excited to announce Fluent UI theme pack is in the work and is currently in experimental phase. This theme pack is designed for web developers who want to bring a native Copilot user experience to their customers. @@ -60,15 +68,15 @@ Web Chat will now render HTML-in-Markdown. We have ported our sanitizer and acce You can turn off this option by setting `styleOptions.markdownRenderHTML` to `false`. -### 4.16.1 notable changes +## 4.16.1 notable changes Web Chat now supports [Adaptive Cards schema up to 1.6](https://adaptivecards.io/explorer/). Some features in Adaptive Cards are in preview or designed to use outside of Bot Framework. Web Chat does not support these features. -### 4.16.0 notable changes +## 4.16.0 notable changes Starting from 4.16.0, Internet Explorer is no longer supported. After more than a year of the Internet Explorer 11 officially retirement, we decided to stop supporting Internet Explorer. This will help us to bring new features to Web Chat. 4.15.9 is the last version which supports Internet Explorer in limited fashion. -### 4.12.1 patch: New style property `adaptiveCardsParserMaxVersion` +## 4.12.1 patch: New style property `adaptiveCardsParserMaxVersion` Web Chat 4.12.1 patch includes a new style property allowing developers to choose the max Adaptive Cards schema version. See [PR #3778](https://github.com/microsoft/BotFramework-WebChat/pull/3778) for code changes. @@ -220,7 +228,7 @@ See the working sample of the [full Web Chat bundle](https://github.com/microsof For full customizability, you can use React to recompose components of Web Chat. -To install the production build from NPM, run `npm install botframework-webchat`. +To install the production build from NPM, run `npm install botframework-webchat`. See our [version notes](#version-notes) on how to select a version. ```js diff --git a/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-1-snap.png b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-1-snap.png new file mode 100644 index 0000000000..c91bccadfa Binary files /dev/null and b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-1-snap.png differ diff --git a/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-2-snap.png b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-2-snap.png new file mode 100644 index 0000000000..db74935b8a Binary files /dev/null and b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-2-snap.png differ diff --git a/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-3-snap.png b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-3-snap.png new file mode 100644 index 0000000000..93656dfaa3 Binary files /dev/null and b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-3-snap.png differ diff --git a/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-4-snap.png b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-4-snap.png new file mode 100644 index 0000000000..5f46270717 Binary files /dev/null and b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-4-snap.png differ diff --git a/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-5-snap.png b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-5-snap.png new file mode 100644 index 0000000000..359ca556cb Binary files /dev/null and b/__tests__/__image_snapshots__/html/activity-order-js-bot-sending-multiple-messages-should-sort-typing-activity-in-its-original-order-5-snap.png differ diff --git a/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-1-snap.png b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-1-snap.png new file mode 100644 index 0000000000..c91bccadfa Binary files /dev/null and b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-1-snap.png differ diff --git a/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-2-snap.png b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-2-snap.png new file mode 100644 index 0000000000..0fdaee80c4 Binary files /dev/null and b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-2-snap.png differ diff --git a/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-3-snap.png b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-3-snap.png new file mode 100644 index 0000000000..d5221f0420 Binary files /dev/null and b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-3-snap.png differ diff --git a/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-4-snap.png b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-4-snap.png new file mode 100644 index 0000000000..b881212408 Binary files /dev/null and b/__tests__/__image_snapshots__/html/chunk-js-bot-typing-with-chunks-should-display-partial-message-4-snap.png differ diff --git a/__tests__/__image_snapshots__/html/informative-js-informative-typing-message-should-be-shown-as-typing-indicator-1-snap.png b/__tests__/__image_snapshots__/html/informative-js-informative-typing-message-should-be-shown-as-typing-indicator-1-snap.png new file mode 100644 index 0000000000..16093c5e60 Binary files /dev/null and b/__tests__/__image_snapshots__/html/informative-js-informative-typing-message-should-be-shown-as-typing-indicator-1-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-1-snap.png b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-1-snap.png new file mode 100644 index 0000000000..c91bccadfa Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-1-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-2-snap.png b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-2-snap.png new file mode 100644 index 0000000000..d5221f0420 Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-2-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-3-snap.png b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-3-snap.png new file mode 100644 index 0000000000..d5221f0420 Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-3-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-4-snap.png b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-4-snap.png new file mode 100644 index 0000000000..b881212408 Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-in-its-original-order-4-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-1-snap.png b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-1-snap.png new file mode 100644 index 0000000000..c91bccadfa Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-1-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-2-snap.png b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-2-snap.png new file mode 100644 index 0000000000..d5221f0420 Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-2-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-3-snap.png b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-3-snap.png new file mode 100644 index 0000000000..d5221f0420 Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-3-snap.png differ diff --git a/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-4-snap.png b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-4-snap.png new file mode 100644 index 0000000000..b881212408 Binary files /dev/null and b/__tests__/__image_snapshots__/html/out-of-order-sequence-number-js-bot-typing-message-in-out-of-order-fashion-should-sort-typing-activity-based-on-channel-data-sequence-number-4-snap.png differ diff --git a/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-1-snap.png b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-1-snap.png new file mode 100644 index 0000000000..c91bccadfa Binary files /dev/null and b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-1-snap.png differ diff --git a/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-2-snap.png b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-2-snap.png new file mode 100644 index 0000000000..5681a7d0fd Binary files /dev/null and b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-2-snap.png differ diff --git a/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-3-snap.png b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-3-snap.png new file mode 100644 index 0000000000..7db8d32f92 Binary files /dev/null and b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-3-snap.png differ diff --git a/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-4-snap.png b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-4-snap.png new file mode 100644 index 0000000000..72833bad3e Binary files /dev/null and b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-4-snap.png differ diff --git a/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-5-snap.png b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-5-snap.png new file mode 100644 index 0000000000..19dd3dc767 Binary files /dev/null and b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-5-snap.png differ diff --git a/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-6-snap.png b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-6-snap.png new file mode 100644 index 0000000000..0e66ecafdd Binary files /dev/null and b/__tests__/__image_snapshots__/html/simultaneous-js-bot-typing-multiple-messages-should-work-properly-6-snap.png differ diff --git a/__tests__/hooks/useTypingIndicatorVisible.js b/__tests__/hooks/useTypingIndicatorVisible.js deleted file mode 100644 index fe81aa1380..0000000000 --- a/__tests__/hooks/useTypingIndicatorVisible.js +++ /dev/null @@ -1,47 +0,0 @@ -import { timeouts } from '../constants.json'; - -import minNumActivitiesShown from '../setup/conditions/minNumActivitiesShown'; -import uiConnected from '../setup/conditions/uiConnected'; - -// selenium-webdriver API doc: -// https://seleniumhq.github.io/selenium/docs/api/javascript/module/selenium-webdriver/index_exports_WebDriver.html - -jest.setTimeout(timeouts.test); - -test('getter should return true when typing indicator is shown', async () => { - const { driver, pageObjects } = await setupWebDriver(); - - await driver.wait(uiConnected(), timeouts.directLine); - await pageObjects.sendMessageViaSendBox('typing 1', { waitForSend: true }); - - await driver.wait(minNumActivitiesShown(2), timeouts.directLine); - - const [typingIndicatorVisible] = await pageObjects.runHook('useTypingIndicatorVisible'); - - expect(typingIndicatorVisible).toBeTruthy(); -}); - -test('getter should return false when typing indicator is not shown', async () => { - const { pageObjects } = await setupWebDriver(); - - const [typingIndicatorVisible] = await pageObjects.runHook('useTypingIndicatorVisible'); - - expect(typingIndicatorVisible).toBeFalsy(); -}); - -test('getter should return false when user is typing', async () => { - const { pageObjects } = await setupWebDriver({ props: { sendTypingIndicator: true } }); - - await pageObjects.typeInSendBox('Hello, World!'); - - const [typingIndicatorVisible] = await pageObjects.runHook('useTypingIndicatorVisible'); - - expect(typingIndicatorVisible).toBeFalsy(); -}); - -test('setter should be falsy', async () => { - const { pageObjects } = await setupWebDriver(); - const [_, setTypingIndicatorVisible] = await pageObjects.runHook('useTypingIndicatorVisible'); - - expect(setTypingIndicatorVisible).toBeFalsy(); -}); diff --git a/__tests__/html/hooks.useActiveTyping.html b/__tests__/html/hooks.useActiveTyping.html index 58e23169da..26f07fc6ba 100644 --- a/__tests__/html/hooks.useActiveTyping.html +++ b/__tests__/html/hooks.useActiveTyping.html @@ -1,4 +1,4 @@ - + @@ -63,7 +63,8 @@ at: 600, expireAt: 5600, name: expect.any(String), - role: 'user' + role: 'user', + type: 'busy' } ]); @@ -80,7 +81,8 @@ at: 600, expireAt: 5600, name: expect.any(String), - role: 'bot' + role: 'bot', + type: 'busy' } ]); @@ -88,23 +90,20 @@ await pageObjects.typeInSendBox('.'); // THEN: `useActiveTyping` should return both. - await expect( - renderWithFunction(() => - Object - .values(useActiveTyping()[0]) - ) - ).resolves.toEqual([ + await expect(renderWithFunction(() => Object.values(useActiveTyping()[0]))).resolves.toEqual([ { at: 600, expireAt: 5600, name: expect.any(String), - role: 'bot' + role: 'bot', + type: 'busy' }, { at: 600, expireAt: 5600, name: expect.any(String), - role: 'user' + role: 'user', + type: 'busy' } ]); }); diff --git a/__tests__/html/hooks.useActiveTyping.livestream.html b/__tests__/html/hooks.useActiveTyping.livestream.html new file mode 100644 index 0000000000..80a4b8b97e --- /dev/null +++ b/__tests__/html/hooks.useActiveTyping.livestream.html @@ -0,0 +1,82 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/hooks.useActiveTyping.livestream.js b/__tests__/html/hooks.useActiveTyping.livestream.js new file mode 100644 index 0000000000..b69b7c2981 --- /dev/null +++ b/__tests__/html/hooks.useActiveTyping.livestream.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('useActiveTyping', () => { + test('should get bot livestream', () => runHTML('hooks.useActiveTyping.livestream.html')); +}); diff --git a/__tests__/html/hooks.useActiveTyping.variable.html b/__tests__/html/hooks.useActiveTyping.variable.html index 4e4754244d..51eb270eaa 100644 --- a/__tests__/html/hooks.useActiveTyping.variable.html +++ b/__tests__/html/hooks.useActiveTyping.variable.html @@ -1,4 +1,4 @@ - + @@ -63,7 +63,8 @@ at: 600, expireAt: 5600, name: expect.any(String), - role: 'user' + role: 'user', + type: 'busy' } ]); @@ -88,7 +89,8 @@ at: 600, expireAt: 8610, name: expect.any(String), - role: 'user' + role: 'user', + type: 'busy' } ]); @@ -104,7 +106,8 @@ at: 600, expireAt: 13610, name: expect.any(String), - role: 'user' + role: 'user', + type: 'busy' } ]); }); diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/getter.botTyping.html b/__tests__/html/hooks/useTypingIndicatorVisible/getter.botTyping.html new file mode 100644 index 0000000000..9fec70d3af --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/getter.botTyping.html @@ -0,0 +1,51 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/getter.botTyping.js b/__tests__/html/hooks/useTypingIndicatorVisible/getter.botTyping.js new file mode 100644 index 0000000000..ab7f3ad115 --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/getter.botTyping.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('useTypingIndicatorVisible', () => { + test('getter should return true when typing indicator is shown', () => runHTML('hooks/useTypingIndicatorVisible/getter.botTyping.html')); +}); diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/getter.initial.html b/__tests__/html/hooks/useTypingIndicatorVisible/getter.initial.html new file mode 100644 index 0000000000..acd27c938e --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/getter.initial.html @@ -0,0 +1,49 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/getter.initial.js b/__tests__/html/hooks/useTypingIndicatorVisible/getter.initial.js new file mode 100644 index 0000000000..e54c905378 --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/getter.initial.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('useTypingIndicatorVisible', () => { + test('getter should return false when typing indicator is not shown', () => runHTML('hooks/useTypingIndicatorVisible/getter.initial.html')); +}); diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/getter.userTyping.html b/__tests__/html/hooks/useTypingIndicatorVisible/getter.userTyping.html new file mode 100644 index 0000000000..df7d633a8f --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/getter.userTyping.html @@ -0,0 +1,50 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/getter.userTyping.js b/__tests__/html/hooks/useTypingIndicatorVisible/getter.userTyping.js new file mode 100644 index 0000000000..3e036521ac --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/getter.userTyping.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('useTypingIndicatorVisible', () => { + test('getter should return false when user is typing', () => runHTML('hooks/useTypingIndicatorVisible/getter.userTyping.html')); +}); diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/setter.html b/__tests__/html/hooks/useTypingIndicatorVisible/setter.html new file mode 100644 index 0000000000..5a1e529d79 --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/setter.html @@ -0,0 +1,49 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/hooks/useTypingIndicatorVisible/setter.js b/__tests__/html/hooks/useTypingIndicatorVisible/setter.js new file mode 100644 index 0000000000..fd2ef61950 --- /dev/null +++ b/__tests__/html/hooks/useTypingIndicatorVisible/setter.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('useTypingIndicatorVisible', () => { + test('setter should be falsy', () => runHTML('hooks/useTypingIndicatorVisible/setter.html')); +}); diff --git a/__tests__/html/scrollToEndButton.typingChunk.html b/__tests__/html/scrollToEndButton.typingChunk.html new file mode 100644 index 0000000000..efb001a72f --- /dev/null +++ b/__tests__/html/scrollToEndButton.typingChunk.html @@ -0,0 +1,71 @@ + + + + + + + + + + +
+ + + diff --git a/__tests__/html/scrollToEndButton.typingChunk.js b/__tests__/html/scrollToEndButton.typingChunk.js new file mode 100644 index 0000000000..0467fa36f7 --- /dev/null +++ b/__tests__/html/scrollToEndButton.typingChunk.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('scroll to end button', () => { + test('should show for typing chunks.', () => runHTML('scrollToEndButton.typingChunk.html')); +}); diff --git a/__tests__/html/typing/activityOrder.html b/__tests__/html/typing/activityOrder.html new file mode 100644 index 0000000000..6ddb20f265 --- /dev/null +++ b/__tests__/html/typing/activityOrder.html @@ -0,0 +1,222 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/activityOrder.js b/__tests__/html/typing/activityOrder.js new file mode 100644 index 0000000000..fcbd2a8e48 --- /dev/null +++ b/__tests__/html/typing/activityOrder.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('bot sending multiple messages', () => { + test('should sort typing activity in its original order', () => runHTML('typing/activityOrder')); +}); diff --git a/__tests__/html/typing/chunk.html b/__tests__/html/typing/chunk.html new file mode 100644 index 0000000000..ed03cbbee7 --- /dev/null +++ b/__tests__/html/typing/chunk.html @@ -0,0 +1,173 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/chunk.js b/__tests__/html/typing/chunk.js new file mode 100644 index 0000000000..cb795c1f3b --- /dev/null +++ b/__tests__/html/typing/chunk.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('bot typing with chunks', () => { + test('should display partial message', () => runHTML('typing/chunk')); +}); diff --git a/__tests__/html/typing/informative.html b/__tests__/html/typing/informative.html new file mode 100644 index 0000000000..a931a02b32 --- /dev/null +++ b/__tests__/html/typing/informative.html @@ -0,0 +1,64 @@ + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/informative.js b/__tests__/html/typing/informative.js new file mode 100644 index 0000000000..742379c45e --- /dev/null +++ b/__tests__/html/typing/informative.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('informative typing message', () => { + test('should be shown as typing indicator', () => runHTML('typing/informative')); +}); diff --git a/__tests__/html/typing/outOfOrder.html b/__tests__/html/typing/outOfOrder.html new file mode 100644 index 0000000000..4ab028974b --- /dev/null +++ b/__tests__/html/typing/outOfOrder.html @@ -0,0 +1,219 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/outOfOrder.js b/__tests__/html/typing/outOfOrder.js new file mode 100644 index 0000000000..5e5d8ab0f0 --- /dev/null +++ b/__tests__/html/typing/outOfOrder.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('bot typing message in out-of-order fashion', () => { + test('should sort typing activity in its original order', () => runHTML('typing/outOfOrder')); +}); diff --git a/__tests__/html/typing/outOfOrder.sequenceNumber.html b/__tests__/html/typing/outOfOrder.sequenceNumber.html new file mode 100644 index 0000000000..5be58ebf5e --- /dev/null +++ b/__tests__/html/typing/outOfOrder.sequenceNumber.html @@ -0,0 +1,219 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/outOfOrder.sequenceNumber.js b/__tests__/html/typing/outOfOrder.sequenceNumber.js new file mode 100644 index 0000000000..ac2e421ac2 --- /dev/null +++ b/__tests__/html/typing/outOfOrder.sequenceNumber.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('bot typing message in out-of-order fashion', () => { + test('should sort typing activity based on channelData.sequenceNumber', () => runHTML('typing/outOfOrder.sequenceNumber')); +}); diff --git a/__tests__/html/typing/perActivityStyleOptions.html b/__tests__/html/typing/perActivityStyleOptions.html new file mode 100644 index 0000000000..e548da140f --- /dev/null +++ b/__tests__/html/typing/perActivityStyleOptions.html @@ -0,0 +1,118 @@ + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/perActivityStyleOptions.js b/__tests__/html/typing/perActivityStyleOptions.js new file mode 100644 index 0000000000..2175f83b8f --- /dev/null +++ b/__tests__/html/typing/perActivityStyleOptions.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('bot typing message with a custom typing indicator in channelData', () => { + test('should only show/hide typing indicator accordingly', () => runHTML('typing/perActivityStyleOptions')); +}); diff --git a/__tests__/html/typing/simultaneous.html b/__tests__/html/typing/simultaneous.html new file mode 100644 index 0000000000..ea2a8863d0 --- /dev/null +++ b/__tests__/html/typing/simultaneous.html @@ -0,0 +1,245 @@ + + + + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/simultaneous.js b/__tests__/html/typing/simultaneous.js new file mode 100644 index 0000000000..c467feadc0 --- /dev/null +++ b/__tests__/html/typing/simultaneous.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('bot typing multiple messages', () => { + test('should work properly', () => runHTML('typing/simultaneous')); +}); diff --git a/__tests__/html/typing/typingIndicator.shouldNotRevive.html b/__tests__/html/typing/typingIndicator.shouldNotRevive.html new file mode 100644 index 0000000000..b1c1b396a8 --- /dev/null +++ b/__tests__/html/typing/typingIndicator.shouldNotRevive.html @@ -0,0 +1,71 @@ + + + + + + + + + +
+ + + diff --git a/__tests__/html/typing/typingIndicator.shouldNotRevive.js b/__tests__/html/typing/typingIndicator.shouldNotRevive.js new file mode 100644 index 0000000000..a506e9bf39 --- /dev/null +++ b/__tests__/html/typing/typingIndicator.shouldNotRevive.js @@ -0,0 +1,5 @@ +/** @jest-environment ./packages/test/harness/src/host/jest/WebDriverEnvironment.js */ + +describe('expired typing indicator', () => { + test('should not revive when an OOO message is received', () => runHTML('typing/typingIndicator.shouldNotRevive')); +}); diff --git a/__tests__/setup/conditions/typingIndicatorShown.js b/__tests__/setup/conditions/typingIndicatorShown.js index 534d36928d..50c523307b 100644 --- a/__tests__/setup/conditions/typingIndicatorShown.js +++ b/__tests__/setup/conditions/typingIndicatorShown.js @@ -1,5 +1,5 @@ import { By, until } from 'selenium-webdriver'; export default function typingIndicatorShown() { - return until.elementLocated(By.css('.webchat__typingIndicator')); + return until.elementLocated(By.css('.webchat__typing-indicator')); } diff --git a/docs/HOOKS.md b/docs/HOOKS.md index 92f10d6aaf..884bc104de 100644 --- a/docs/HOOKS.md +++ b/docs/HOOKS.md @@ -53,6 +53,7 @@ Following is the list of hooks supported by Web Chat API. - [`useActiveTyping`](#useactivetyping) - [`useActivities`](#useactivities) +- [`useActivityKeysByRead`](#useactivitykeysbyread) - [`useAdaptiveCardsHostConfig`](#useadaptivecardshostconfig) - [`useAdaptiveCardsPackage`](#useadaptivecardspackage) - [`useAvatarForBot`](#useavatarforbot) @@ -75,15 +76,23 @@ Following is the list of hooks supported by Web Chat API. - [`useEmitTypingIndicator`](#useemittypingindicator) - [`useFocus`](#usefocus) - [`useFocusSendBox`](#usefocussendbox) +- [`useGetActivitiesByKey`](#usegetactivitiesbykey) +- [`useGetActivityByKey`](#usegetactivitybykey) +- [`useGetHasAcknowledgedByActivityKey`](#usegethasacknowledgedbyactivitykey) +- [`useGetKeyByActivity`](#usegetkeybyactivity) +- [`useGetKeyByActivityId`](#usegetkeybyactivityid) - [`useGetSendTimeoutForActivity`](#usegetsendtimeoutforactivity) - [`useGrammars`](#usegrammars) - [`useGroupTimestamp`](#usegrouptimestamp) - [`useLanguage`](#uselanguage) -- [`useLastTypingAt`](#uselasttypingat) +- [`useLastAcknowledgedActivityKey`](#uselastacknowledgedactivitykey) +- [`useLastReadActivityKey`](#uselastreadactivitykey) - [`useLastTypingAt`](#uselasttypingat) (Deprecated) - [`useLocalize`](#uselocalize) (Deprecated) - [`useLocalizer`](#useLocalizer) - [`useMarkActivityAsSpoken`](#usemarkactivityasspoken) +- [`useMarkActivityKeyAsRead`](#usemarkactivitykeyasread) +- [`useMarkAllAsAcknowledged`](#usemarkallasacknowledged) - [`useNotification`](#usenotification) - [`useObserveScrollPosition`](#useobservescrollposition) - [`useObserveTranscriptFocus`](#useobservetranscriptfocus) @@ -139,20 +148,28 @@ interface Typing { expireAt: number; name: string; role: 'bot' | 'user'; + type: 'busy' | 'livestream'; } -useActiveTyping(expireAfter?: number): [{ [id: string]: Typing }] +useActiveTyping(expireAfter?: number): readonly [Readonly>] ``` > On or before 4.15.1, there is [an issue](https://github.com/microsoft/BotFramework-WebChat/issues/4209) which the `at` field is not accurately reflecting the time when the participant start typing. +> New in 4.18.0: Added `type` property. The returned type is marked as read-only to prevent accidental modification. + This hook will return a list of participants who are actively typing, including the start typing time (`at`) and expiration time (`expireAt`), the name and the role of the participant. Both time values are based on local clock. If the participant sends a message after the typing activity, the participant will be explicitly removed from the list. If no messages or typing activities are received, the participant is considered inactive and not listed in the result. To keep the typing indicator active, participants should continuously send the typing activity. The `expireAfter` argument can override the inactivity timer. If `expireAfter` is `Infinity`, it will return all participants who did not explicitly remove from the list. In other words, it will return participants who sent a typing activity, but did not send a message activity afterward. +The `type` property will tell if the participant is livestreaming or busy preparing its response: + +- `busy` indicates the participant is busy preparing the response +- `livestream` indicates the participant is sending its response as it is being prepared + > This hook will trigger render of your component if one or more typing information is expired or removed. ## `useActivities` @@ -165,6 +182,16 @@ useActivities(): [Activity[]] This hook will return a list of activities. +## `useActivityKeysByRead` + + +```ts +useActivityKeysByRead(): readonly [readonly string[], readonly string[]] +``` + + +This hook will subscribe and return two lists of activities: read and unread. + ## `useAdaptiveCardsHostConfig` @@ -229,7 +256,7 @@ useByteFormatter() => (bytes: number) => string ``` -This hook will return a function that, when called with a file size, will return a localized representation of the size in bytes, kilobytes, megabytes, or gigabytes. It honors the language settings from the `useLanguage` hook. +When the returned function is called with a file size, will return a localized representation of the size in bytes, kilobytes, megabytes, or gigabytes. It honors the language settings from the [`useLanguage` hook](#uselanguage). ## `useConnectivityStatus` @@ -365,7 +392,7 @@ useDateFormatter() => (dateOrString: (Date | number | string)) => string ``` -This hook will return a function that, when called with a `Date` object, `number`, or `string`, will return a localized representation of the date in absolute time. It honors the language settings from the `useLanguage` hook. +When the returned function is called with a `Date` object, `number`, or `string`, will return a localized representation of the date in absolute time. It honors the language settings from the [`useLanguage` hook](#uselanguage). ## `useDebouncedNotification` @@ -511,6 +538,68 @@ useFocusSendBox(): () => void When called, this function will send focus to the send box. +## `useGetActivitiesByKey` + + +```ts +useGetActivitiesByKey(): (key?: string) => readonly WebChatActivity[] | undefined +``` + + +> Please refer to [the activity key section](#what-is-activity-key) for details about how Web Chat use activity keys. + +When the returned function is called, will return a chronologically sorted list of activities which share the same activity key. These activities represent different revisions of the same activity. For example, a livestreaming activity is made up of multiple revisions. + +## `useGetActivityByKey` + + +```ts +useGetActivityByKey(): (key?: string) => undefined | WebChatActivity +``` + + +> Please refer to [the activity key section](#what-is-activity-key) for details about how Web Chat use activity keys. + +When called, this hook will return a function to get the latest activity which share the same activity key. + +This hook is same as getting the last element from the result of the [`useGetActivitiesByKey`](#usegetactivitiesbykey) hook. + +## `useGetHasAcknowledgedByActivityKey` + + +```ts +useGetHasAcknowledgedByActivityKey(): (activityKey: string) => boolean | undefined +``` + + +> Please refer to [this section](#what-is-acknowledged-activity) for details about acknowledged activity. + +When the returned function is called with an activity key, will evaluate whether the activity is acknowledged by the user or not. + +## `useGetKeyByActivity` + + +```ts +useGetKeyByActivity(): (activity?: WebChatActivity | undefined) => string | undefined +``` + + +> Please refer to [the activity key section](#what-is-activity-key) for details about how Web Chat use activity keys. + +When called, this hook will return a function to get the activity key of the passing activity. + +## `useGetKeyByActivityId` + + +```ts +useGetKeyByActivityId(): (activityId?: string | undefined) => string | undefined +``` + + +> Please refer to [the activity key section](#what-is-activity-key) for details about how Web Chat use activity keys. + +When called, this hook will return a function to get the activity key of the passing activity ID. + ## `useGetSendTimeoutForActivity` @@ -519,7 +608,7 @@ useGetSendTimeoutForActivity(): ({ activity: Activity }) => number ``` -When called, This hook will return a function to evaluate the timeout (in milliseconds) for sending a specific activity. +When called, this hook will return a function to evaluate the timeout (in milliseconds) for sending a specific activity. ## `useGrammars` @@ -565,6 +654,28 @@ If `"speech"` is passed to `options`, the return value will be the oral language To modify this value, change the value in the `locale` prop passed to Web Chat. +## `useLastAcknowledgedActivityKey` + + +```ts +useLastAcknowledgedActivityKey(): readonly [string | undefined] +``` + + +> Please refer to [this section](#what-is-acknowledged-activity) for details about acknowledged activity. + +This hook will subscribe and return the activity key of the last acknowledged activity in the chat history. + +## `useLastReadActivityKey` + + +```ts +useLastReadActivityKey(): readonly [string | undefined] +``` + + +This hook will subscribe and return the activity key of the last read activity in the chat history. + ## `useLastTypingAt` @@ -664,6 +775,28 @@ useMarkActivityAsSpoken(): (activity: Activity) => void When called, this function will mark the activity as spoken and remove it from the text-to-speech queue. +## `useMarkActivityKeyAsRead` + + +```ts +useMarkActivityKeyAsRead(): (activityKey: string) => void +``` + + +When the returned function is called, will mark the activity as read. + +## `useMarkAllAsAcknowledged` + + +```ts +useMarkAllAsAcknowledged(): () => void +``` + + +> Please refer to [this section](#what-is-acknowledged-activity) for details about acknowledged activity. + +When the returned function is called, will mark all activities in the chat history as acknowledged. + ## `useNotifications` @@ -1473,3 +1606,28 @@ useTrackTiming(): (name: string, promise: Promise) => void This function will emit timing measurements for the execution of a synchronous or asynchronous function. Before the execution, the `onTelemetry` handler will be triggered with a `timingstart` event. After completion, regardless of resolve or reject, the `onTelemetry` handler will be triggered again with a `timingend` event. If the function throws an exception while executing, the exception will be reported to [`useTrackException`](#usetrackexception) hook as a non-fatal error. + +## What is activity key? + +Activity ID is a service-assigned ID that is unique in the conversation. However, not every activity has an activity ID. Therefore, it is not possible to reference every activities in the chat history by solely using activity ID. + +Web Chat introduces activity key as an alternative method to reference activity in the system. + +Activity key is an opaque string. When the activity first appear in Web Chat, they will be assigned an activity key and never be reassigned to another key again until Web Chat is restarted. + +Multiple activities could share the same activity key if they are revision of each others. For example, a livestreaming activity could made up of different revisions of the same activity. Thus, these activities would share the same activity key. + +Following hooks are designed to help navigating between activity, activity ID and activity keys: + +- [`useGetActivitiesByKey`](#usegetactivitiesbykey) +- [`useGetActivityByKey`](#usegetactivitybykey) +- [`useGetKeyByActivity`](#usegetkeybyactivity) +- [`useGetKeyByActivityId`](#usegetkeybyactivityid) + +## What is acknowledged activity? + +Chat history normally would scroll to the bottom when message arrive and remains stick to the bottom. However, in some circumstances, such as the bot sending more than a page of message, the chat history will pause the auto-scroll and unstick from the bottom. + +The pause helps users to read the long text sent by the bot without explicitly scrolling up from the very bottom of the chat history. + +Activities are being acknowledged when the chat history view is being scroll to the end, either by auto-scroll or manually after a pause. It can also be programmatically acknowledged using the [`useMarkAllAsAcknowledged` hook](#usemarkallasacknowledged). diff --git a/packages/api/src/hooks/Composer.tsx b/packages/api/src/hooks/Composer.tsx index c4f21f11e6..31c2f8fff2 100644 --- a/packages/api/src/hooks/Composer.tsx +++ b/packages/api/src/hooks/Composer.tsx @@ -43,8 +43,10 @@ import normalizeStyleOptions from '../normalizeStyleOptions'; import patchStyleOptionsFromDeprecatedProps from '../patchStyleOptionsFromDeprecatedProps'; import ActivityAcknowledgementComposer from '../providers/ActivityAcknowledgement/ActivityAcknowledgementComposer'; import ActivityKeyerComposer from '../providers/ActivityKeyer/ActivityKeyerComposer'; +import ActivityListenerComposer from '../providers/ActivityListener/ActivityListenerComposer'; import ActivitySendStatusComposer from '../providers/ActivitySendStatus/ActivitySendStatusComposer'; import ActivitySendStatusTelemetryComposer from '../providers/ActivitySendStatusTelemetry/ActivitySendStatusTelemetryComposer'; +import ActivityTypingComposer from '../providers/ActivityTyping/ActivityTypingComposer'; import PonyfillComposer from '../providers/Ponyfill/PonyfillComposer'; import ActivityMiddleware from '../types/ActivityMiddleware'; import { type ActivityStatusMiddleware, type RenderActivityStatus } from '../types/ActivityStatusMiddleware'; @@ -109,6 +111,8 @@ const DISPATCHERS = { submitSendBox }; +const EMPTY_ARRAY: readonly [] = Object.freeze([]); + function createCardActionContext({ cardActionMiddleware, directLine, @@ -589,14 +593,18 @@ const ComposerCore = ({ return ( - - - - {typeof children === 'function' ? children(context) : children} - - - - + + + + + + {typeof children === 'function' ? children(context) : children} + + + + + + {onTelemetry && } ); diff --git a/packages/api/src/hooks/index.ts b/packages/api/src/hooks/index.ts index bafd77a6c0..95764a88c6 100644 --- a/packages/api/src/hooks/index.ts +++ b/packages/api/src/hooks/index.ts @@ -19,6 +19,7 @@ import useDirection from './useDirection'; import useDisabled from './useDisabled'; import useDismissNotification from './useDismissNotification'; import useEmitTypingIndicator from './useEmitTypingIndicator'; +import useGetActivitiesByKey from './useGetActivitiesByKey'; import useGetActivityByKey from './useGetActivityByKey'; import useGetHasAcknowledgedByActivityKey from './useGetHasAcknowledgedByActivityKey'; import useGetKeyByActivity from './useGetKeyByActivity'; @@ -91,6 +92,7 @@ export { useDisabled, useDismissNotification, useEmitTypingIndicator, + useGetActivitiesByKey, useGetActivityByKey, useGetHasAcknowledgedByActivityKey, useGetKeyByActivity, diff --git a/packages/api/src/hooks/internal/usePrevious.ts b/packages/api/src/hooks/internal/usePrevious.ts index fdec6c16c1..c49f55bfb5 100644 --- a/packages/api/src/hooks/internal/usePrevious.ts +++ b/packages/api/src/hooks/internal/usePrevious.ts @@ -1,7 +1,10 @@ import { useEffect, useRef } from 'react'; -export default function usePrevious(value: T): T { - const ref = useRef(); +export default function usePrevious(value: T): T | undefined; +export default function usePrevious(value: T, initialValue: T): T; + +export default function usePrevious(value: T, initialValue?: T | undefined): T | undefined { + const ref = useRef(initialValue); useEffect(() => { ref.current = value; diff --git a/packages/api/src/hooks/private/numberWithInfinity.spec.ts b/packages/api/src/hooks/private/numberWithInfinity.spec.ts new file mode 100644 index 0000000000..6c9c98bbc1 --- /dev/null +++ b/packages/api/src/hooks/private/numberWithInfinity.spec.ts @@ -0,0 +1,9 @@ +import numberWithInfinity from './numberWithInfinity'; + +test('passing "Infinity" should return Infinity', () => expect(numberWithInfinity('Infinity')).toBe(Infinity)); +test('passing "-Infinity" should return -Infinity', () => expect(numberWithInfinity('-Infinity')).toBe(-Infinity)); +test('passing 0 should return 0', () => expect(numberWithInfinity(0)).toBe(0)); +test('passing -0 should return -0', () => expect(numberWithInfinity(-0)).toBe(-0)); +test('passing 1 should return 1', () => expect(numberWithInfinity(1)).toBe(1)); +test('passing "1" should return undefined', () => expect(numberWithInfinity('1' as any)).toBeUndefined()); +test('passing "ABC" should return undefined', () => expect(numberWithInfinity('ABC' as any)).toBeUndefined()); diff --git a/packages/api/src/hooks/private/numberWithInfinity.ts b/packages/api/src/hooks/private/numberWithInfinity.ts new file mode 100644 index 0000000000..74179103eb --- /dev/null +++ b/packages/api/src/hooks/private/numberWithInfinity.ts @@ -0,0 +1,15 @@ +export default function numberWithInfinity(value: number | 'Infinity' | '-Infinity'): number; +export default function numberWithInfinity(value: unknown): undefined; + +export default function numberWithInfinity(value: number | 'Infinity' | '-Infinity' | unknown): number | undefined { + switch (value) { + case 'Infinity': + return Infinity; + + case '-Infinity': + return -Infinity; + + default: + return typeof value === 'number' ? value : undefined; + } +} diff --git a/packages/api/src/hooks/private/reduceIterable.spec.ts b/packages/api/src/hooks/private/reduceIterable.spec.ts new file mode 100644 index 0000000000..3e7b70a84f --- /dev/null +++ b/packages/api/src/hooks/private/reduceIterable.spec.ts @@ -0,0 +1,31 @@ +/* eslint-disable no-magic-numbers */ + +import reduceIterable from './reduceIterable'; + +describe('when called with a summation reducer', () => { + let reducer: jest.Mock; + let actual: number; + + beforeEach(() => { + reducer = jest.fn((intermediate, value) => intermediate + +value); + actual = reduceIterable(['1', '2', '3'].values(), reducer, 100); + }); + + test('should return summation', () => expect(actual).toBe(106)); + test('should have called reducer 3 times', () => expect(reducer).toHaveBeenCalledTimes(3)); + test("should have called reducer with (100, '1')", () => expect(reducer).toHaveBeenNthCalledWith(1, 100, '1')); + test("should have called reducer with (101, '2')", () => expect(reducer).toHaveBeenNthCalledWith(2, 101, '2')); + test("should have called reducer with (103, '3')", () => expect(reducer).toHaveBeenNthCalledWith(3, 103, '3')); +}); + +describe('when called with an empty array', () => { + let reducer: jest.Mock; + let actual: number; + + beforeEach(() => { + reducer = jest.fn(); + actual = reduceIterable([].values(), reducer, 100); + }); + + test('should return initial value', () => expect(actual).toBe(100)); +}); diff --git a/packages/api/src/hooks/private/reduceIterable.ts b/packages/api/src/hooks/private/reduceIterable.ts new file mode 100644 index 0000000000..c5c660a20e --- /dev/null +++ b/packages/api/src/hooks/private/reduceIterable.ts @@ -0,0 +1,13 @@ +export default function reduceIterable( + iterable: Iterable, + reducer: (intermediate: U, item: T) => U, + initial: U +): U { + let intermediate = initial; + + for (const item of iterable) { + intermediate = reducer(intermediate, item); + } + + return intermediate; +} diff --git a/packages/api/src/hooks/useActiveTyping.ts b/packages/api/src/hooks/useActiveTyping.ts index c4c1f0d4eb..6201f88a24 100644 --- a/packages/api/src/hooks/useActiveTyping.ts +++ b/packages/api/src/hooks/useActiveTyping.ts @@ -1,39 +1,41 @@ import { useEffect } from 'react'; -import type { Typing } from '../types/Typing'; -import { useSelector } from './internal/WebChatReduxContext'; +import useAllTyping from '../providers/ActivityTyping/useAllTyping'; +import { type Typing } from '../types/Typing'; import useForceRender from './internal/useForceRender'; +import reduceIterable from './private/reduceIterable'; import usePonyfill from './usePonyfill'; import useStyleOptions from './useStyleOptions'; -function useActiveTyping(expireAfter?: number): [{ [userId: string]: Typing }] { +function useActiveTyping(expireAfter?: number): readonly [Readonly>] { const [{ clearTimeout, Date, setTimeout }] = usePonyfill(); const [{ typingAnimationDuration }] = useStyleOptions(); + const [typing] = useAllTyping(); const forceRender = useForceRender(); - const typing: { [userId: string]: { at: number; last: number; name: string; role: string } } = useSelector( - ({ typing }) => typing - ); - const now = Date.now(); - if (typeof expireAfter !== 'number') { - expireAfter = typingAnimationDuration; - } - - const activeTyping: { [userId: string]: Typing } = Object.entries(typing).reduce( - (activeTyping, [id, { at, last, name, role }]) => { - const until = last + expireAfter; - - if (until > now) { - return { ...activeTyping, [id]: { at, expireAt: until, name, role } }; - } - - return activeTyping; - }, - {} - ); - - const earliestExpireAt = Math.min(...Object.values(activeTyping).map(({ expireAt }) => expireAt)); + // TODO: We should use useState to simplify the force render part. + const activeTypingState: readonly [Readonly>] = Object.freeze([ + Object.freeze( + Object.fromEntries( + reduceIterable( + typing.entries(), + (activeTypingMap, [id, { firstReceivedAt, lastActivityDuration, lastReceivedAt, name, role, type }]) => { + const expireAt = lastReceivedAt + (expireAfter ?? lastActivityDuration ?? typingAnimationDuration); + + if (expireAt > now) { + activeTypingMap.set(id, { at: firstReceivedAt, expireAt, name, role, type }); + } + + return activeTypingMap; + }, + new Map() + ).entries() + ) + ) + ]); + + const earliestExpireAt = Math.min(...Object.values(activeTypingState[0]).map(({ expireAt }) => expireAt)); const timeToRender = earliestExpireAt && earliestExpireAt - now; useEffect(() => { @@ -44,7 +46,7 @@ function useActiveTyping(expireAfter?: number): [{ [userId: string]: Typing }] { } }, [clearTimeout, forceRender, setTimeout, timeToRender]); - return [activeTyping]; + return activeTypingState; } export default useActiveTyping; diff --git a/packages/api/src/hooks/useGetActivitiesByKey.ts b/packages/api/src/hooks/useGetActivitiesByKey.ts new file mode 100644 index 0000000000..61f5cdebd8 --- /dev/null +++ b/packages/api/src/hooks/useGetActivitiesByKey.ts @@ -0,0 +1,3 @@ +import useGetActivitiesByKey from '../providers/ActivityKeyer/useGetActivitiesByKey'; + +export default useGetActivitiesByKey; diff --git a/packages/api/src/providers/ActivityAcknowledgement/useLastReadActivityKey.ts b/packages/api/src/providers/ActivityAcknowledgement/useLastReadActivityKey.ts index b1b9e8f8d4..53b5b22d52 100644 --- a/packages/api/src/providers/ActivityAcknowledgement/useLastReadActivityKey.ts +++ b/packages/api/src/providers/ActivityAcknowledgement/useLastReadActivityKey.ts @@ -1,5 +1,5 @@ import useActivityAcknowledgementContext from './private/useContext'; -export default function useLastReadActivityKey(): readonly [string] { +export default function useLastReadActivityKey(): readonly [string | undefined] { return useActivityAcknowledgementContext().lastReadActivityKeyState; } diff --git a/packages/api/src/providers/ActivityKeyer/ActivityKeyerComposer.tsx b/packages/api/src/providers/ActivityKeyer/ActivityKeyerComposer.tsx index c0508b9ca3..a24f29189d 100644 --- a/packages/api/src/providers/ActivityKeyer/ActivityKeyerComposer.tsx +++ b/packages/api/src/providers/ActivityKeyer/ActivityKeyerComposer.tsx @@ -1,18 +1,35 @@ import type { WebChatActivity } from 'botframework-webchat-core'; import React, { useCallback, useMemo, useRef, type ReactNode } from 'react'; +import reduceIterable from '../../hooks/private/reduceIterable'; import useActivities from '../../hooks/useActivities'; import type { ActivityKeyerContextType } from './private/Context'; import ActivityKeyerContext from './private/Context'; import getActivityId from './private/getActivityId'; import getClientActivityId from './private/getClientActivityId'; +import lastOf from './private/lastOf'; +import someIterable from './private/someIterable'; import uniqueId from './private/uniqueId'; import useActivityKeyerContext from './private/useContext'; type ActivityIdToKeyMap = Map; type ActivityToKeyMap = Map; type ClientActivityIdToKeyMap = Map; -type KeyToActivityMap = Map; +type KeyToActivitiesMap = Map; + +function getTypingActivityId(activity: WebChatActivity): string | undefined { + const { type } = activity; + + if ( + (type === 'message' || type === 'typing') && + 'text' in activity && + typeof activity.text === 'string' && + 'streamId' in activity.channelData && + activity.channelData.streamId + ) { + return activity.channelData.streamId; + } +} /** * React context composer component to assign a perma-key to every activity. @@ -39,7 +56,7 @@ const ActivityKeyerComposer = ({ children }: Readonly<{ children?: ReactNode | u const activityIdToKeyMapRef = useRef>(Object.freeze(new Map())); const activityToKeyMapRef = useRef>(Object.freeze(new Map())); const clientActivityIdToKeyMapRef = useRef>(Object.freeze(new Map())); - const keyToActivityMapRef = useRef>(Object.freeze(new Map())); + const keyToActivitiesMapRef = useRef>(Object.freeze(new Map())); // TODO: [P1] `useMemoWithPrevious` to check and cache the resulting array if it hasn't changed. const activityKeysState = useMemo(() => { @@ -47,17 +64,19 @@ const ActivityKeyerComposer = ({ children }: Readonly<{ children?: ReactNode | u const { current: activityToKeyMap } = activityToKeyMapRef; const { current: clientActivityIdToKeyMap } = clientActivityIdToKeyMapRef; const nextActivityIdToKeyMap: ActivityIdToKeyMap = new Map(); - const nextActivityKeys: string[] = []; + const nextActivityKeys: Set = new Set(); const nextActivityToKeyMap: ActivityToKeyMap = new Map(); const nextClientActivityIdToKeyMap: ClientActivityIdToKeyMap = new Map(); - const nextKeyToActivityMap: KeyToActivityMap = new Map(); + const nextKeyToActivitiesMap: KeyToActivitiesMap = new Map(); activities.forEach(activity => { const activityId = getActivityId(activity); const clientActivityId = getClientActivityId(activity); + const typingActivityId = getTypingActivityId(activity); const key = (clientActivityId && clientActivityIdToKeyMap.get(clientActivityId)) || + (typingActivityId && activityIdToKeyMap.get(typingActivityId)) || (activityId && activityIdToKeyMap.get(activityId)) || activityToKeyMap.get(activity) || uniqueId(); @@ -65,31 +84,40 @@ const ActivityKeyerComposer = ({ children }: Readonly<{ children?: ReactNode | u activityId && nextActivityIdToKeyMap.set(activityId, key); clientActivityId && nextClientActivityIdToKeyMap.set(clientActivityId, key); nextActivityToKeyMap.set(activity, key); - nextKeyToActivityMap.set(key, activity); - nextActivityKeys.push(key); + nextActivityKeys.add(key); + + const activities = nextKeyToActivitiesMap.has(key) ? [...nextKeyToActivitiesMap.get(key)] : []; + + activities.push(activity); + nextKeyToActivitiesMap.set(key, Object.freeze(activities)); }); activityIdToKeyMapRef.current = Object.freeze(nextActivityIdToKeyMap); activityToKeyMapRef.current = Object.freeze(nextActivityToKeyMap); clientActivityIdToKeyMapRef.current = Object.freeze(nextClientActivityIdToKeyMap); - keyToActivityMapRef.current = Object.freeze(nextKeyToActivityMap); + keyToActivitiesMapRef.current = Object.freeze(nextKeyToActivitiesMap); // `nextActivityKeys` could potentially same as `prevActivityKeys` despite reference differences, we should memoize it. - return Object.freeze([Object.freeze(nextActivityKeys)]) as readonly [readonly string[]]; - }, [activities, activityIdToKeyMapRef, activityToKeyMapRef, clientActivityIdToKeyMapRef, keyToActivityMapRef]); + return Object.freeze([Object.freeze([...nextActivityKeys.values()])]) as readonly [readonly string[]]; + }, [activities, activityIdToKeyMapRef, activityToKeyMapRef, clientActivityIdToKeyMapRef, keyToActivitiesMapRef]); + + const getActivitiesByKey: (key?: string | undefined) => readonly WebChatActivity[] | undefined = useCallback( + (key?: string | undefined): readonly WebChatActivity[] | undefined => key && keyToActivitiesMapRef.current.get(key), + [keyToActivitiesMapRef] + ); - const getActivityByKey: (key?: string) => undefined | WebChatActivity = useCallback( - (key?: string): undefined | WebChatActivity => key && keyToActivityMapRef.current.get(key), - [keyToActivityMapRef] + const getActivityByKey: (key?: string | undefined) => undefined | WebChatActivity = useCallback( + (key?: string | undefined): undefined | WebChatActivity => lastOf(getActivitiesByKey(key)), + [getActivitiesByKey] ); - const getKeyByActivity: (activity?: WebChatActivity) => string | undefined = useCallback( - (activity?: WebChatActivity) => activity && activityToKeyMapRef.current.get(activity), + const getKeyByActivity: (activity?: WebChatActivity | undefined) => string | undefined = useCallback( + (activity?: WebChatActivity | undefined) => activity && activityToKeyMapRef.current.get(activity), [activityToKeyMapRef] ); - const getKeyByActivityId: (activityId?: string) => string | undefined = useCallback( - (activityId?: string) => activityId && activityIdToKeyMapRef.current.get(activityId), + const getKeyByActivityId: (activityId?: string | undefined) => string | undefined = useCallback( + (activityId?: string | undefined) => activityId && activityIdToKeyMapRef.current.get(activityId), [activityIdToKeyMapRef] ); @@ -97,10 +125,11 @@ const ActivityKeyerComposer = ({ children }: Readonly<{ children?: ReactNode | u () => ({ activityKeysState, getActivityByKey, + getActivitiesByKey, getKeyByActivity, getKeyByActivityId }), - [activityKeysState, getActivityByKey, getKeyByActivity, getKeyByActivityId] + [activityKeysState, getActivitiesByKey, getActivityByKey, getKeyByActivity, getKeyByActivityId] ); const { length: numActivities } = activities; @@ -123,15 +152,23 @@ const ActivityKeyerComposer = ({ children }: Readonly<{ children?: ReactNode | u ); } - if (keyToActivityMapRef.current.size !== numActivities) { + if (someIterable(keyToActivitiesMapRef.current.values(), ({ length }) => !length)) { + console.warn( + 'botframework-webchat internal assertion: all values in "keyToActivitiesMap" should have at least one item.' + ); + } + + if ( + reduceIterable(keyToActivitiesMapRef.current.values(), (total, { length }) => total + length, 0) !== numActivities + ) { console.warn( - 'botframework-webchat internal assertion: "keyToActivityMap.size" should be same as "activities.length".' + 'botframework-webchat internal assertion: "keyToActivitiesMap.size" should be same as "activities.length".' ); } - if (activityKeysState[0].length !== numActivities) { + if (activityKeysState[0].length !== keyToActivitiesMapRef.current.size) { console.warn( - 'botframework-webchat internal assertion: "activityKeys.length" should be same as "activities.length".' + 'botframework-webchat internal assertion: "activityKeys.length" should be same as "keyToActivitiesMap.size".' ); } diff --git a/packages/api/src/providers/ActivityKeyer/private/Context.ts b/packages/api/src/providers/ActivityKeyer/private/Context.ts index ff13be45be..2c4d505717 100644 --- a/packages/api/src/providers/ActivityKeyer/private/Context.ts +++ b/packages/api/src/providers/ActivityKeyer/private/Context.ts @@ -1,8 +1,9 @@ -import { createContext } from 'react'; import type { WebChatActivity } from 'botframework-webchat-core'; +import { createContext } from 'react'; type ActivityKeyerContextType = { activityKeysState: readonly [readonly string[]]; + getActivitiesByKey: (key?: string) => readonly WebChatActivity[] | undefined; getActivityByKey: (key?: string) => undefined | WebChatActivity; getKeyByActivity: (activity?: WebChatActivity) => string | undefined; getKeyByActivityId: (activityKey?: string) => string | undefined; diff --git a/packages/api/src/providers/ActivityKeyer/private/lastOf.spec.ts b/packages/api/src/providers/ActivityKeyer/private/lastOf.spec.ts new file mode 100644 index 0000000000..5e2090a4c6 --- /dev/null +++ b/packages/api/src/providers/ActivityKeyer/private/lastOf.spec.ts @@ -0,0 +1,6 @@ +import lastOf from './lastOf'; + +test('should return last element', () => expect(lastOf(Object.freeze(['1', '2', '3']))).toBe('3')); +test('when passed empty array should return undefined', () => expect(lastOf([])).toBeUndefined()); +test('when passed undefined should return undefined', () => expect(lastOf(undefined)).toBeUndefined()); +test('when passed null should return undefined', () => expect(lastOf(null as any)).toBeUndefined()); diff --git a/packages/api/src/providers/ActivityKeyer/private/lastOf.ts b/packages/api/src/providers/ActivityKeyer/private/lastOf.ts new file mode 100644 index 0000000000..4c0fb30ff7 --- /dev/null +++ b/packages/api/src/providers/ActivityKeyer/private/lastOf.ts @@ -0,0 +1,3 @@ +export default function lastOf(array: readonly T[] | undefined): T | undefined { + return array?.[array.length - 1]; +} diff --git a/packages/api/src/providers/ActivityKeyer/private/someIterable.spec.ts b/packages/api/src/providers/ActivityKeyer/private/someIterable.spec.ts new file mode 100644 index 0000000000..0bfc7187cf --- /dev/null +++ b/packages/api/src/providers/ActivityKeyer/private/someIterable.spec.ts @@ -0,0 +1,46 @@ +/* eslint-disable no-magic-numbers */ + +import someIterable from './someIterable'; + +describe('when predicate return true should return true', () => { + let predicate: jest.Mock; + let actual: boolean; + + beforeEach(() => { + predicate = jest.fn(() => true); + actual = someIterable(['1', '2', '3'], predicate); + }); + + test('should return true', () => expect(actual).toBe(true)); + test('should have called predicate once', () => expect(predicate).toHaveBeenCalledTimes(1)); + test("should have called predicate with '1'", () => expect(predicate).toHaveBeenNthCalledWith(1, '1')); +}); + +describe('when predicate return false should return false', () => { + let predicate: jest.Mock; + let actual: boolean; + + beforeEach(() => { + predicate = jest.fn(() => false); + actual = someIterable(['1', '2', '3'], predicate); + }); + + test('should return false', () => expect(actual).toBe(false)); + test('should have called predicate 3 times', () => expect(predicate).toHaveBeenCalledTimes(3)); + test("should have called predicate with '1'", () => expect(predicate).toHaveBeenNthCalledWith(1, '1')); + test("should have called predicate with '2'", () => expect(predicate).toHaveBeenNthCalledWith(2, '2')); + test("should have called predicate with '3'", () => expect(predicate).toHaveBeenNthCalledWith(3, '3')); +}); + +describe('when passed an empty array', () => { + let predicate: jest.Mock; + let actual: boolean; + + beforeEach(() => { + predicate = jest.fn(); + actual = someIterable([], predicate); + }); + + test('should return false', () => expect(actual).toBe(false)); + test('should not call predicate', () => expect(predicate).toHaveBeenCalledTimes(0)); +}); diff --git a/packages/api/src/providers/ActivityKeyer/private/someIterable.ts b/packages/api/src/providers/ActivityKeyer/private/someIterable.ts new file mode 100644 index 0000000000..0b8ce0081d --- /dev/null +++ b/packages/api/src/providers/ActivityKeyer/private/someIterable.ts @@ -0,0 +1,9 @@ +export default function someIterable(iterable: Iterable, predicate: (item: T) => boolean): boolean { + for (const item of iterable) { + if (predicate(item)) { + return true; + } + } + + return false; +} diff --git a/packages/api/src/providers/ActivityKeyer/useGetActivitiesByKey.ts b/packages/api/src/providers/ActivityKeyer/useGetActivitiesByKey.ts new file mode 100644 index 0000000000..a513e9cd34 --- /dev/null +++ b/packages/api/src/providers/ActivityKeyer/useGetActivitiesByKey.ts @@ -0,0 +1,7 @@ +import type { WebChatActivity } from 'botframework-webchat-core'; + +import useActivityKeyerContext from './private/useContext'; + +export default function useGetActivitiesByKey(): (key?: string) => readonly WebChatActivity[] | undefined { + return useActivityKeyerContext().getActivitiesByKey; +} diff --git a/packages/api/src/providers/ActivityListener/ActivityListenerComposer.tsx b/packages/api/src/providers/ActivityListener/ActivityListenerComposer.tsx new file mode 100644 index 0000000000..fc36762b68 --- /dev/null +++ b/packages/api/src/providers/ActivityListener/ActivityListenerComposer.tsx @@ -0,0 +1,28 @@ +import type { WebChatActivity } from 'botframework-webchat-core'; +import React, { memo, useMemo, type ReactNode } from 'react'; +import usePrevious from '../../hooks/internal/usePrevious'; +import useActivities from '../../hooks/useActivities'; +import ActivityListenerContext, { type ActivityListenerContextType } from './private/Context'; + +type Props = Readonly<{ children?: ReactNode | undefined }>; + +const ActivityListenerComposer = memo(({ children }: Props) => { + const [activities] = useActivities(); + const prevActivities = usePrevious(activities, []); + + const upsertedActivitiesState = useMemo(() => { + const upserts: WebChatActivity[] = []; + + for (const activity of activities) { + prevActivities.includes(activity) || upserts.push(activity); + } + + return Object.freeze([Object.freeze(upserts)]); + }, [activities, prevActivities]); + + const context = useMemo(() => ({ upsertedActivitiesState }), [upsertedActivitiesState]); + + return {children}; +}); + +export default ActivityListenerComposer; diff --git a/packages/api/src/providers/ActivityListener/private/Context.tsx b/packages/api/src/providers/ActivityListener/private/Context.tsx new file mode 100644 index 0000000000..b9062610b2 --- /dev/null +++ b/packages/api/src/providers/ActivityListener/private/Context.tsx @@ -0,0 +1,16 @@ +import { type WebChatActivity } from 'botframework-webchat-core'; +import { createContext } from 'react'; + +export type ActivityListenerContextType = { + upsertedActivitiesState: readonly [readonly WebChatActivity[]]; +}; + +const ActivityListenerContext = createContext( + new Proxy({} as ActivityListenerContextType, { + get() { + throw new Error('botframework-webchat internal: This hook can only used under .'); + } + }) +); + +export default ActivityListenerContext; diff --git a/packages/api/src/providers/ActivityListener/private/useContext.ts b/packages/api/src/providers/ActivityListener/private/useContext.ts new file mode 100644 index 0000000000..4425511379 --- /dev/null +++ b/packages/api/src/providers/ActivityListener/private/useContext.ts @@ -0,0 +1,6 @@ +import { useContext } from 'react'; +import ActivityListenerContext from './Context'; + +export default function useActivityListenerContext() { + return useContext(ActivityListenerContext); +} diff --git a/packages/api/src/providers/ActivityListener/useUpsertedActivities.ts b/packages/api/src/providers/ActivityListener/useUpsertedActivities.ts new file mode 100644 index 0000000000..34489ed90f --- /dev/null +++ b/packages/api/src/providers/ActivityListener/useUpsertedActivities.ts @@ -0,0 +1,6 @@ +import type { WebChatActivity } from 'botframework-webchat-core'; +import useActivityListenerContext from './private/useContext'; + +export default function useUpsertedActivities(): readonly [readonly WebChatActivity[]] { + return useActivityListenerContext().upsertedActivitiesState; +} diff --git a/packages/api/src/providers/ActivityTyping/ActivityTypingComposer.tsx b/packages/api/src/providers/ActivityTyping/ActivityTypingComposer.tsx new file mode 100644 index 0000000000..404ddd939f --- /dev/null +++ b/packages/api/src/providers/ActivityTyping/ActivityTypingComposer.tsx @@ -0,0 +1,85 @@ +import type { WebChatActivity } from 'botframework-webchat-core'; +import React, { memo, useMemo, type ReactNode } from 'react'; +import { useRefFrom } from 'use-ref-from'; + +import numberWithInfinity from '../../hooks/private/numberWithInfinity'; +import useActivities from '../../hooks/useActivities'; +import usePonyfill from '../../hooks/usePonyfill'; +import useUpsertedActivities from '../../providers/ActivityListener/useUpsertedActivities'; +import ActivityTypingContext, { ActivityTypingContextType } from './private/Context'; +import useMemoWithPrevious from './private/useMemoWithPrevious'; +import { type AllTyping } from './types/AllTyping'; + +const INITIAL_ALL_TYPING_STATE = Object.freeze([Object.freeze(new Map())] as const); + +type Props = Readonly<{ children?: ReactNode | undefined }>; + +function isLivestreamChunk(activity: WebChatActivity): boolean { + return ( + activity.type === 'typing' && + 'text' in activity && + typeof activity.text === 'string' && + activity.channelData.streamType !== 'informative' + ); +} + +const ActivityTypingComposer = ({ children }: Props) => { + const [{ Date }] = usePonyfill(); + const [activities] = useActivities(); + const [upsertedActivities] = useUpsertedActivities(); + const activitiesRef = useRefFrom(activities); + + const allTypingState = useMemoWithPrevious]>( + (prevAllTypingState = INITIAL_ALL_TYPING_STATE) => { + const { current: activities } = activitiesRef; + const nextTyping = new Map(prevAllTypingState[0]); + let changed = false; + + const firstIndex = upsertedActivities.reduce( + (firstIndex, upsertedActivity) => Math.min(firstIndex, activities.indexOf(upsertedActivity)), + Infinity + ); + + for (const activity of activities.slice(firstIndex)) { + const { + from, + from: { id, role }, + type + } = activity; + + if (type === 'typing' && (role === 'bot' || role === 'user')) { + const currentTyping = nextTyping.get(id); + // TODO: When we rework on types of DLActivity, we will make sure all activities has "webChat.receivedAt", this coalesces can be removed. + const receivedAt = activity.channelData.webChat?.receivedAt || Date.now(); + + nextTyping.set(id, { + firstReceivedAt: currentTyping?.firstReceivedAt || receivedAt, + lastActivityDuration: numberWithInfinity( + activity.channelData.webChat?.styleOptions?.typingAnimationDuration + ), + lastReceivedAt: receivedAt, + name: from.name, + role, + type: isLivestreamChunk(activity) ? 'livestream' : 'busy' // Informative message means the bot is busy. + }); + + changed = true; + } else if (type === 'message') { + nextTyping.delete(id); + changed = true; + } + } + + return changed ? Object.freeze([nextTyping]) : prevAllTypingState; + }, + [activitiesRef, upsertedActivities] + ); + + const context = useMemo(() => ({ allTypingState }), [allTypingState]); + + return {children}; +}; + +ActivityTypingComposer.displayName = 'ActivityTypingComposer'; + +export default memo(ActivityTypingComposer); diff --git a/packages/api/src/providers/ActivityTyping/private/Context.ts b/packages/api/src/providers/ActivityTyping/private/Context.ts new file mode 100644 index 0000000000..0b6d1f8656 --- /dev/null +++ b/packages/api/src/providers/ActivityTyping/private/Context.ts @@ -0,0 +1,16 @@ +import { createContext } from 'react'; +import { type AllTyping } from '../types/AllTyping'; + +export type ActivityTypingContextType = { + allTypingState: readonly [ReadonlyMap]; +}; + +const ActivityTypingContext = createContext( + new Proxy({} as ActivityTypingContextType, { + get() { + throw new Error('botframework-webchat internal: This hook can only be used under .'); + } + }) +); + +export default ActivityTypingContext; diff --git a/packages/api/src/providers/ActivityTyping/private/useContext.ts b/packages/api/src/providers/ActivityTyping/private/useContext.ts new file mode 100644 index 0000000000..cd88d9bf09 --- /dev/null +++ b/packages/api/src/providers/ActivityTyping/private/useContext.ts @@ -0,0 +1,6 @@ +import { useContext } from 'react'; +import ActivityTypingContext, { type ActivityTypingContextType } from './Context'; + +export default function useActivityTypingContext(): ActivityTypingContextType { + return useContext(ActivityTypingContext); +} diff --git a/packages/api/src/providers/ActivityTyping/private/useMemoWithPrevious.ts b/packages/api/src/providers/ActivityTyping/private/useMemoWithPrevious.ts new file mode 100644 index 0000000000..967dc2b828 --- /dev/null +++ b/packages/api/src/providers/ActivityTyping/private/useMemoWithPrevious.ts @@ -0,0 +1,17 @@ +// TODO: [P1] Dedupe. +import { useEffect, useMemo, useRef } from 'react'; + +import type { DependencyList } from 'react'; + +export default function useMemoWithPrevious(factory: (prevValue: T | undefined) => T, deps: DependencyList): T { + const prevValueRef = useRef(undefined); + // We are building a `useMemo`-like hook, `deps` is passed as-is and `factory` is not one fo the dependencies. + // eslint-disable-next-line react-hooks/exhaustive-deps + const value = useMemo(() => factory(prevValueRef.current), deps); + + useEffect(() => { + prevValueRef.current = value; + }); + + return value; +} diff --git a/packages/api/src/providers/ActivityTyping/types/AllTyping.ts b/packages/api/src/providers/ActivityTyping/types/AllTyping.ts new file mode 100644 index 0000000000..7357c338d0 --- /dev/null +++ b/packages/api/src/providers/ActivityTyping/types/AllTyping.ts @@ -0,0 +1,8 @@ +export type AllTyping = { + firstReceivedAt: number; + lastActivityDuration: number; + lastReceivedAt: number; + name: string | undefined; + role: 'bot' | 'user'; + type: 'busy' | 'livestream'; +}; diff --git a/packages/api/src/providers/ActivityTyping/useAllTyping.ts b/packages/api/src/providers/ActivityTyping/useAllTyping.ts new file mode 100644 index 0000000000..1780324a2a --- /dev/null +++ b/packages/api/src/providers/ActivityTyping/useAllTyping.ts @@ -0,0 +1,6 @@ +import useActivityTypingContext from './private/useContext'; +import type { AllTyping } from './types/AllTyping'; + +export default function useAllTyping(): readonly [ReadonlyMap] { + return useActivityTypingContext().allTypingState; +} diff --git a/packages/api/src/types/Typing.ts b/packages/api/src/types/Typing.ts index b16a8adde3..c7096e55d5 100644 --- a/packages/api/src/types/Typing.ts +++ b/packages/api/src/types/Typing.ts @@ -1,8 +1,7 @@ -type Typing = { +export type Typing = { at: number; expireAt: number; name: string; role: 'bot' | 'user'; + type: 'busy' | 'livestream'; }; - -export type { Typing }; diff --git a/packages/component/src/Assets/TypingAnimation.js b/packages/component/src/Assets/TypingAnimation.js index 97f38f2b03..2b0e5e2ceb 100644 --- a/packages/component/src/Assets/TypingAnimation.js +++ b/packages/component/src/Assets/TypingAnimation.js @@ -3,14 +3,13 @@ import classNames from 'classnames'; import React from 'react'; import ScreenReaderText from '../ScreenReaderText'; - -import useStyleSet from '../hooks/useStyleSet'; import useStyleToEmotionObject from '../hooks/internal/useStyleToEmotionObject'; +import useStyleSet from '../hooks/useStyleSet'; const { useDirection, useLocalizer } = hooks; const ROOT_STYLE = { - '&.webchat__typingIndicator.webchat__typingIndicator--rtl': { transform: 'scale(-1, 1)' } + '&.webchat__typing-indicator.webchat__typing-indicator--rtl': { transform: 'scale(-1, 1)' } }; const TypingAnimation = () => { @@ -25,9 +24,9 @@ const TypingAnimation = () => {
role !== 'user').length]; + return useMemo( + () => + Object.freeze([ + !!Object.values(activeTyping).some( + // Show typing indicator if anyone is typing and not livestreaming. + ({ role, type }) => role !== 'user' && type !== 'livestream' + ) + ]), + [activeTyping] + ); } -const BasicTypingIndicator: VFC<{}> = () => { +const BasicTypingIndicator = () => { const [activeTyping] = useActiveTyping(); const [visible] = useTypingIndicatorVisible(); const [typing] = useActiveTyping(Infinity); diff --git a/packages/component/src/Middleware/Activity/createCoreMiddleware.tsx b/packages/component/src/Middleware/Activity/createCoreMiddleware.tsx index f5e0dd2070..e97741e068 100644 --- a/packages/component/src/Middleware/Activity/createCoreMiddleware.tsx +++ b/packages/component/src/Middleware/Activity/createCoreMiddleware.tsx @@ -15,8 +15,37 @@ export default function createCoreMiddleware(): ActivityMiddleware[] { const { type } = activity; + if (type === 'typing') { + if ( + !( + 'text' in activity && + typeof activity.text === 'string' && + activity.channelData.streamType !== 'informative' + ) + ) { + // If it is an informative message, hide it until we have a design for informative message. + return false; + } + + // Should show if this is useActiveTyping()[0][*].firstActivity, and render it with the content of lastActivity. + return function renderStackedLayout(renderAttachment, props) { + typeof props === 'undefined' && + console.warn( + 'botframework-webchat: One or more arguments were missing after passing through the activity middleware. Please check your custom activity middleware to make sure it passes all arguments.' + ); + + return ( + + ); + }; + } + // Filter out activities that should not be visible - if (type === 'conversationUpdate' || type === 'event' || type === 'invoke' || type === 'typing') { + if (type === 'conversationUpdate' || type === 'event' || type === 'invoke') { return false; } else if (type === 'message') { const { attachments, channelData, text } = activity; diff --git a/packages/component/src/Middleware/TypingIndicator/createCoreMiddleware.tsx b/packages/component/src/Middleware/TypingIndicator/createCoreMiddleware.tsx index f482efaad3..e711878eb4 100644 --- a/packages/component/src/Middleware/TypingIndicator/createCoreMiddleware.tsx +++ b/packages/component/src/Middleware/TypingIndicator/createCoreMiddleware.tsx @@ -13,7 +13,7 @@ const DotIndicator = () => { const localize = useLocalizer(); return ( -
+
); diff --git a/packages/component/src/Styles/StyleSet/TypingIndicator.ts b/packages/component/src/Styles/StyleSet/TypingIndicator.ts index 3101161da4..9f964c810f 100644 --- a/packages/component/src/Styles/StyleSet/TypingIndicator.ts +++ b/packages/component/src/Styles/StyleSet/TypingIndicator.ts @@ -4,11 +4,11 @@ export default function createTypingIndicatorStyle({ paddingRegular }: StrictSty return { paddingBottom: paddingRegular, - '&:not(.webchat__typingIndicator--rtl)': { + '&:not(.webchat__typing-indicator--rtl)': { paddingLeft: paddingRegular }, - '&.webchat__typingIndicator--rtl': { + '&.webchat__typing-indicator--rtl': { paddingRight: paddingRegular } }; diff --git a/packages/component/src/providers/ActivityTree/ActivityTreeComposer.tsx b/packages/component/src/providers/ActivityTree/ActivityTreeComposer.tsx index 4d5640cf3c..09daea67b2 100644 --- a/packages/component/src/providers/ActivityTree/ActivityTreeComposer.tsx +++ b/packages/component/src/providers/ActivityTree/ActivityTreeComposer.tsx @@ -1,31 +1,47 @@ -import { hooks } from 'botframework-webchat-api'; -import React, { useMemo } from 'react'; - -import type { ActivityComponentFactory } from 'botframework-webchat-api'; -import type { FC, PropsWithChildren } from 'react'; +import { hooks, type ActivityComponentFactory } from 'botframework-webchat-api'; import type { WebChatActivity } from 'botframework-webchat-core'; +import React, { useMemo, type ReactNode } from 'react'; -import { ActivityWithRenderer, ReadonlyActivityTree } from './private/types'; +import useMemoWithPrevious from '../../hooks/internal/useMemoWithPrevious'; import ActivityTreeContext from './private/Context'; +import { ActivityWithRenderer, ReadonlyActivityTree } from './private/types'; import useActivitiesWithRenderer from './private/useActivitiesWithRenderer'; -import useActivityTreeContext from './private/useContext'; import useActivityTreeWithRenderer from './private/useActivityTreeWithRenderer'; -import useMemoWithPrevious from '../../hooks/internal/useMemoWithPrevious'; +import useActivityTreeContext from './private/useContext'; import type { ActivityTreeContextType } from './private/Context'; -type ActivityTreeComposerProps = PropsWithChildren<{}>; +type ActivityTreeComposerProps = Readonly<{ children?: ReactNode | undefined }>; -const { useActivities, useCreateActivityRenderer } = hooks; +const { useActivities, useCreateActivityRenderer, useGetActivitiesByKey, useGetKeyByActivity } = hooks; -const ActivityTreeComposer: FC = ({ children }) => { +const ActivityTreeComposer = ({ children }: ActivityTreeComposerProps) => { const existingContext = useActivityTreeContext(false); if (existingContext) { throw new Error('botframework-webchat internal: should not be nested.'); } - const [activities]: [WebChatActivity[]] = useActivities(); + const [rawActivities] = useActivities(); + const getActivitiesByKey = useGetActivitiesByKey(); + const getKeyByActivity = useGetKeyByActivity(); + + const activities = useMemo(() => { + const activities: WebChatActivity[] = []; + + for (const activity of rawActivities) { + // If an activity has multiple revisions, display the latest revision only at the position of the first revision. + + // "Activities with same key" means "multiple revisions of same activity." + const activitiesWithSameKey = getActivitiesByKey(getKeyByActivity(activity)); + + // TODO: We may want to send all revisions of activity to the middleware so they can render UI to see previous revisions. + activitiesWithSameKey[0] === activity && activities.push(activitiesWithSameKey[activitiesWithSameKey.length - 1]); + } + + return Object.freeze(activities); + }, [getActivitiesByKey, getKeyByActivity, rawActivities]); + const createActivityRenderer: ActivityComponentFactory = useCreateActivityRenderer(); const activitiesWithRenderer = useActivitiesWithRenderer(activities, createActivityRenderer); diff --git a/packages/core/src/reducers/createActivitiesReducer.ts b/packages/core/src/reducers/createActivitiesReducer.ts index 5ecec37c13..eb005bab18 100644 --- a/packages/core/src/reducers/createActivitiesReducer.ts +++ b/packages/core/src/reducers/createActivitiesReducer.ts @@ -11,10 +11,11 @@ import { POST_ACTIVITY_PENDING, POST_ACTIVITY_REJECTED } from '../actions/postActivity'; -import { SEND_FAILED, SENDING, SENT } from '../types/internal/SendStatus'; +import { SENDING, SEND_FAILED, SENT } from '../types/internal/SendStatus'; +import findBeforeAfter from './private/findBeforeAfter'; +import type { Reducer } from 'redux'; import type { DeleteActivityAction } from '../actions/deleteActivity'; -import type { GlobalScopePonyfill } from '../types/GlobalScopePonyfill'; import type { IncomingActivityAction } from '../actions/incomingActivity'; import type { MarkActivityAction } from '../actions/markActivity'; import type { @@ -23,7 +24,7 @@ import type { PostActivityPendingAction, PostActivityRejectedAction } from '../actions/postActivity'; -import type { Reducer } from 'redux'; +import type { GlobalScopePonyfill } from '../types/GlobalScopePonyfill'; import type { WebChatActivity } from '../types/WebChatActivity'; type ActivitiesAction = @@ -49,9 +50,15 @@ function findByClientActivityID(clientActivityID: string): (activity: WebChatAct return (activity: WebChatActivity) => getClientActivityID(activity) === clientActivityID; } +function isPartOfLivestreamSession( + activity: WebChatActivity +): activity is WebChatActivity & { text: string; type: 'typing' } { + return activity.type === 'typing' && 'text' in activity && typeof activity.text === 'string'; +} + function patchActivity( activity: WebChatActivity, - lastActivity: WebChatActivity, + activities: WebChatActivity[], { Date }: GlobalScopePonyfill ): WebChatActivity { // Direct Line channel will return a placeholder image for the user-uploaded image. @@ -69,26 +76,54 @@ function patchActivity( } }); + activity = updateIn(activity, ['channelData'], channelData => ({ ...channelData })); + activity = updateIn(activity, ['channelData', 'webChat', 'receivedAt'], () => Date.now()); + + const { + channelData: { 'webchat:sequence-id': sequenceId } + } = activity; + + // TODO: [P1] #3953 We should move this patching logic to a DLJS wrapper for simplicity. // If the message does not have sequence ID, use these fallback values: - // 1. "timestamp" field + // 1. "channelData.streamSequence" field (if available) + // - 0.0001 * streamSequence should be good + // 2. "timestamp" field // - outgoing activity will not have "timestamp" field - // 2. last activity sequence ID (or 0) + 0.001 + // 3. last activity sequence ID (or 0) + 0.001 // - best effort to put this message the last one in the chat history - activity = updateIn(activity, ['channelData', 'webchat:sequence-id'], (sequenceId?: number) => - typeof sequenceId === 'number' - ? sequenceId - : typeof activity.timestamp !== 'undefined' - ? +new Date(activity.timestamp) - : // We assume there will be no more than 1,000 messages sent before receiving server response. - // If there are more than 1,000 messages, some messages will get reordered and appear jumpy after receiving server response. - // eslint-disable-next-line no-magic-numbers - (lastActivity?.channelData?.['webchat:sequence-id'] || 0) + 0.001 - ); + if (typeof sequenceId !== 'number') { + let after: WebChatActivity; + let before: WebChatActivity; + + if (isPartOfLivestreamSession(activity) && typeof activity.channelData.streamSequence === 'number') { + [before, after] = findBeforeAfter(activities, target => { + if (target.type === 'typing' && target.channelData.streamId === activity.channelData.streamId) { + return target.channelData.streamSequence < activity.channelData.streamSequence ? 'before' : 'after'; + } - // TODO: [P1] #3953 We should move this patching logic to a DLJS wrapper for simplicity. - activity = updateIn(activity, ['channelData', 'webchat:sequence-id'], (sequenceId: number) => - typeof sequenceId === 'number' ? sequenceId : +new Date(activity.timestamp || 0) || 0 - ); + return 'unknown'; + }); + } + + let sequenceId: number; + + if (before) { + // eslint-disable-next-line no-magic-numbers + sequenceId = before.channelData['webchat:sequence-id'] + 0.001; + } else if (after) { + // eslint-disable-next-line no-magic-numbers + sequenceId = after.channelData['webchat:sequence-id'] - 0.001; + } else if (typeof activity.timestamp !== 'undefined') { + sequenceId = +new Date(activity.timestamp); + } else { + // We assume there will be no more than 1,000 messages sent before receiving server response. + // If there are more than 1,000 messages, some messages will get reordered and appear jumpy after receiving server response. + // eslint-disable-next-line no-magic-numbers + sequenceId = (activities[activities.length - 1]?.channelData['webchat:sequence-id'] || 0) + 0.001; + } + + activity = updateIn(activity, ['channelData', 'webchat:sequence-id'], () => sequenceId); + } return activity; } @@ -98,7 +133,7 @@ function upsertActivityWithSort( nextActivity: WebChatActivity, ponyfill: GlobalScopePonyfill ): WebChatActivity[] { - nextActivity = patchActivity(nextActivity, activities[activities.length - 1], ponyfill); + nextActivity = patchActivity(nextActivity, activities, ponyfill); const { channelData: { clientActivityID: nextClientActivityID, 'webchat:sequence-id': nextSequenceId } = {} } = nextActivity; @@ -180,18 +215,22 @@ export default function createActivitiesReducer( break; case POST_ACTIVITY_FULFILLED: - state = updateIn(state, [findByClientActivityID(action.meta.clientActivityID)], () => { + { // We will replace the activity with the version from the server const activity = updateIn( - patchActivity(action.payload.activity, state[state.length - 1], ponyfill), - // `channelData.state` is being deprecated in favor of `channelData['webchat:send-status']`. - // Please refer to #4362 for details. Remove on or after 2024-07-31. - ['channelData', 'state'], + updateIn( + patchActivity(action.payload.activity, state, ponyfill), + // `channelData.state` is being deprecated in favor of `channelData['webchat:send-status']`. + // Please refer to #4362 for details. Remove on or after 2024-07-31. + ['channelData', 'state'], + () => SENT + ), + ['channelData', 'webchat:send-status'], () => SENT ); - return updateIn(activity, ['channelData', 'webchat:send-status'], () => SENT); - }); + state = updateIn(state, [findByClientActivityID(action.meta.clientActivityID)], () => activity); + } break; diff --git a/packages/core/src/reducers/private/findBeforeAfter.spec.ts b/packages/core/src/reducers/private/findBeforeAfter.spec.ts new file mode 100644 index 0000000000..87362b2b48 --- /dev/null +++ b/packages/core/src/reducers/private/findBeforeAfter.spec.ts @@ -0,0 +1,90 @@ +/* eslint no-magic-numbers: "off" */ + +import findBeforeAfter from './findBeforeAfter'; + +let before: number | string | undefined; +let after: number | string | undefined; +let index: number | undefined; +const getPosition = jest.fn<'after' | 'before' | 'unknown', [unknown]>().mockImplementation(() => { + throw new Error('This function should not be called.'); +}); + +describe('when passing an empty array', () => { + beforeEach(() => ([before, after, index] = findBeforeAfter([] as (number | string)[], getPosition))); + + test('before should be undefined', () => expect(before).toBeUndefined()); + test('after should be undefined', () => expect(after).toBeUndefined()); + test('index should be undefined', () => expect(index).toBeUndefined()); +}); + +describe('when passing a value of 2 to [a, 1, 3, b]', () => { + beforeEach(() => { + getPosition.mockImplementation(value => (typeof value === 'number' ? (value < 2 ? 'before' : 'after') : 'unknown')); + + [before, after, index] = findBeforeAfter(['a', 1, 3, 'b'], getPosition); + }); + + test('before should be 1', () => expect(before).toBe(1)); + test('after should be 3', () => expect(after).toBe(3)); + test('index should be 2', () => expect(index).toBe(2)); +}); + +describe('when passing a value of 2 to [3, b]', () => { + beforeEach(() => { + getPosition.mockImplementation(value => (typeof value === 'number' ? (value < 2 ? 'before' : 'after') : 'unknown')); + + [before, after, index] = findBeforeAfter([3, 'b'], getPosition); + }); + + test('before should be undefined', () => expect(before).toBeUndefined()); + test('after should be 3', () => expect(after).toBe(3)); + test('index should be 0', () => expect(index).toBe(0)); +}); + +describe('when passing a value of 2 to [a, 1]', () => { + beforeEach(() => { + getPosition.mockImplementation(value => (typeof value === 'number' ? (value < 2 ? 'before' : 'after') : 'unknown')); + + [before, after, index] = findBeforeAfter(['a', 1], getPosition); + }); + + test('before should be 1', () => expect(before).toBe(1)); + test('after should be undefined', () => expect(after).toBeUndefined()); + test('index should be 2', () => expect(index).toBe(2)); +}); + +describe('when passing a value of 2 to [a, 1, b]', () => { + beforeEach(() => { + getPosition.mockImplementation(value => (typeof value === 'number' ? (value < 2 ? 'before' : 'after') : 'unknown')); + + [before, after, index] = findBeforeAfter(['a', 1, 'b'], getPosition); + }); + + test('before should be 1', () => expect(before).toBe(1)); + test('after should be "b"', () => expect(after).toBe('b')); + test('index should be 2', () => expect(index).toBe(2)); +}); + +describe('when passing a value of 2 to [a, 3, b]', () => { + beforeEach(() => { + getPosition.mockImplementation(value => (typeof value === 'number' ? (value < 2 ? 'before' : 'after') : 'unknown')); + + [before, after, index] = findBeforeAfter(['a', 3, 'b'], getPosition); + }); + + test('before should be "a"', () => expect(before).toBe('a')); + test('after should be 3', () => expect(after).toBe(3)); + test('index should be 1', () => expect(index).toBe(1)); +}); + +describe('when passing a value of 2 to [a, b]', () => { + beforeEach(() => { + getPosition.mockImplementation(value => (typeof value === 'number' ? (value < 2 ? 'before' : 'after') : 'unknown')); + + [before, after, index] = findBeforeAfter(['a', 'b'], getPosition); + }); + + test('before should be undefined', () => expect(before).toBeUndefined()); + test('after should be undefined', () => expect(after).toBeUndefined()); + test('index should be undefined', () => expect(index).toBeUndefined()); +}); diff --git a/packages/core/src/reducers/private/findBeforeAfter.ts b/packages/core/src/reducers/private/findBeforeAfter.ts new file mode 100644 index 0000000000..bec4e48452 --- /dev/null +++ b/packages/core/src/reducers/private/findBeforeAfter.ts @@ -0,0 +1,30 @@ +type Position = 'after' | 'before' | 'unknown'; + +export default function findBeforeAfter( + array: T[], + getPosition: (value: T) => Position +): [T | undefined, T | undefined, number | undefined] { + let lastValue: T | undefined; + let lastPosition: Position = 'unknown'; + + for (let index = 0; index < array.length; index++) { + const value = array[+index]; + const currentPosition = getPosition(value); + + if ( + ((lastPosition === 'before' || lastPosition === 'unknown') && currentPosition === 'after') || + (lastPosition === 'before' && (currentPosition === 'after' || currentPosition === 'unknown')) + ) { + return [lastValue, value, index]; + } + + lastValue = value; + lastPosition = currentPosition; + } + + if (lastPosition === 'before') { + return [lastValue, undefined, array.length]; + } + + return [undefined, undefined, undefined]; +} diff --git a/packages/core/src/types/WebChatActivity.ts b/packages/core/src/types/WebChatActivity.ts index eeaf4321cb..3d41d78744 100644 --- a/packages/core/src/types/WebChatActivity.ts +++ b/packages/core/src/types/WebChatActivity.ts @@ -25,6 +25,22 @@ type ChannelData; + }; } & (SendStatus extends SupportedSendStatus ? { /** @@ -135,6 +151,11 @@ type EventActivityEssence = { type MessageActivityEssence = { attachmentLayout?: 'carousel' | 'stacked'; attachments?: DirectLineAttachment[]; + channelData: { + streamId?: string; + streamSequence?: number; + streamType?: 'final'; + }; inputHint?: 'accepting' | 'expecting' | 'ignoring'; locale?: string; speak?: string; @@ -146,9 +167,19 @@ type MessageActivityEssence = { }; // https://github.com/Microsoft/botframework-sdk/blob/main/specs/botframework-activity/botframework-activity.md#typing-activity -type TypingActivityEssence = { - type: 'typing'; -}; +type TypingActivityEssence = + | { + type: 'typing'; + } + | { + channelData: { + streamId: string; + streamSequence: number; + streamType: 'informative' | 'streaming'; + }; + text: string; + type: 'typing'; + }; // Abstract - timestamps diff --git a/packages/test/page-object/src/globals/pageElements/index.js b/packages/test/page-object/src/globals/pageElements/index.js index f1d2bc190a..0e1451a897 100644 --- a/packages/test/page-object/src/globals/pageElements/index.js +++ b/packages/test/page-object/src/globals/pageElements/index.js @@ -20,6 +20,7 @@ import transcript from './transcript'; import transcriptLiveRegion from './transcriptLiveRegion'; import transcriptScrollable from './transcriptScrollable'; import transcriptTerminator from './transcriptTerminator'; +import typingIndicator from './typingIndicator'; import uploadButton from './uploadButton'; export { @@ -45,5 +46,6 @@ export { transcriptLiveRegion, transcriptScrollable, transcriptTerminator, + typingIndicator, uploadButton }; diff --git a/packages/test/page-object/src/globals/pageElements/typingIndicator.js b/packages/test/page-object/src/globals/pageElements/typingIndicator.js index 69a73b62ec..ac85876d28 100644 --- a/packages/test/page-object/src/globals/pageElements/typingIndicator.js +++ b/packages/test/page-object/src/globals/pageElements/typingIndicator.js @@ -1,3 +1,3 @@ -export default function typeFocusSink() { - return document.querySelector('.webchat__typingIndicator'); +export default function typingIndicator() { + return document.querySelector('.webchat__typing-indicator'); } diff --git a/samples/05.custom-components/j.typing-indicator/README.md b/samples/05.custom-components/j.typing-indicator/README.md index fc2ef9cca0..abd90c0764 100644 --- a/samples/05.custom-components/j.typing-indicator/README.md +++ b/samples/05.custom-components/j.typing-indicator/README.md @@ -59,7 +59,7 @@ Then, register a custom component to override the existing typing indicator: + + return ( + !!activeTyping.length && ( -+ ++ + Currently typing:{' '} + {activeTyping + .map(({ role }) => role) @@ -105,7 +105,7 @@ Add the following CSS for styling the typing indicator: ```css -.webchat__typingIndicator { +.webchat__typing-indicator { font-family: 'Calibri', 'Helvetica Neue', 'Arial', 'sans-serif'; font-size: 14px; padding: 10px; @@ -143,7 +143,7 @@ Here is the finished `index.html`: width: 100%; } - .webchat__typingIndicator { + .webchat__typing-indicator { font-family: 'Calibri', 'Helvetica Neue', 'Arial', 'sans-serif'; font-size: 14px; padding: 10px; @@ -168,7 +168,7 @@ Here is the finished `index.html`: return ( !!activeTyping.length && ( - + Currently typing:{' '} {activeTyping .map(({ role }) => role) diff --git a/samples/05.custom-components/j.typing-indicator/index.html b/samples/05.custom-components/j.typing-indicator/index.html index 9890465d47..06d8602639 100644 --- a/samples/05.custom-components/j.typing-indicator/index.html +++ b/samples/05.custom-components/j.typing-indicator/index.html @@ -1,4 +1,4 @@ - + Web Chat: Customizing typing indicator @@ -29,7 +29,7 @@ width: 100%; } - .webchat__typingIndicator { + .webchat__typing-indicator { font-family: 'Calibri', 'Helvetica Neue', 'Arial', 'sans-serif'; font-size: 14px; padding: 10px; @@ -53,21 +53,23 @@ next => ({ activeTyping }) => { - activeTyping = Object.values(activeTyping); + typingIndicatorMiddleware={() => + next => + ({ activeTyping }) => { + activeTyping = Object.values(activeTyping); - return ( - !!activeTyping.length && ( - - Currently typing:{' '} - {activeTyping - .map(({ role }) => role) - .sort() - .join(', ')} - - ) - ); - }} + return ( + !!activeTyping.length && ( + + Currently typing:{' '} + {activeTyping + .map(({ role }) => role) + .sort() + .join(', ')} + + ) + ); + }} />, document.getElementById('webchat') );