From 09536a5cf913dab0a5544442bd4577df8e490cd4 Mon Sep 17 00:00:00 2001 From: Corina Date: Fri, 19 Apr 2019 19:35:33 -0700 Subject: [PATCH] Add Reconnect UI to ConnectivityStatus when connection has been interrupted (#1895) * Move HowToQuestions.md to root .github * Minor copy update to English strings * Add Reconnecting component to ConectivityStatus * Add tests and screencaptures * Rewrite ConnectivityStatus * Refactor * More refactoring * Refactor 3 * Add transparency to offlineUI tests * More ConnectivityStatus refactor * Fix 1831: ability to change Spinner and Typing indicator via defaultStyles * Fix offlineUI tests with static spinner * Fix test with static spinner --- .../.github => .github}/HowToQuestions.md | 0 ...-when-connection-is-interrupted-1-snap.png | Bin 0 -> 10156 bytes ...n-connecting-for-the-first-time-1-snap.png | Bin 8075 -> 8069 bytes ...ect-ui-when-connection-is-slow-1-snap.png} | Bin ...ct-ui-when-reconnection-is-slow-1-snap.png | Bin 0 -> 9020 bytes __tests__/offlineUI.js | 122 ++++++++++++- __tests__/setup/assets/staticSpinner.js | 3 + __tests__/setup/setupTestFramework.js | 5 +- package-lock.json | 84 +++++---- .../src/Attachment/Assets/SpinnerAnimation.js | 19 +- .../src/Attachment/Assets/TypingAnimation.js | 17 +- packages/component/src/Localization/en-US.js | 2 +- .../src/SendBox/ConnectivityStatus.js | 168 ++++++++---------- .../src/Styles/StyleSet/SpinnerAnimation.js | 19 ++ .../src/Styles/StyleSet/TypingAnimation.js | 15 ++ .../component/src/Styles/createStyleSet.js | 4 + .../src/Styles/defaultStyleSetOptions.js | 9 + packages/core/src/actions/connect.js | 4 +- .../core/src/reducers/connectivityStatus.js | 59 ++---- packages/core/src/sagas/connectSaga.js | 7 +- .../src/sagas/detectSlowConnectionSaga.js | 10 +- 21 files changed, 338 insertions(+), 209 deletions(-) rename {__tests__/.github => .github}/HowToQuestions.md (100%) create mode 100644 __tests__/__image_snapshots__/chrome-docker/offline-ui-js-offline-ui-should-display-network-interruption-occurred-reconnecting-status-when-connection-is-interrupted-1-snap.png rename __tests__/__image_snapshots__/chrome-docker/{offline-ui-js-offline-ui-should-show-slow-to-connect-ui-when-connection-is-slow-1-snap.png => offline-ui-js-offline-ui-should-show-taking-longer-than-usual-to-connect-ui-when-connection-is-slow-1-snap.png} (100%) create mode 100644 __tests__/__image_snapshots__/chrome-docker/offline-ui-js-offline-ui-should-show-taking-longer-than-usual-to-connect-ui-when-reconnection-is-slow-1-snap.png create mode 100644 __tests__/setup/assets/staticSpinner.js create mode 100644 packages/component/src/Styles/StyleSet/SpinnerAnimation.js create mode 100644 packages/component/src/Styles/StyleSet/TypingAnimation.js diff --git a/__tests__/.github/HowToQuestions.md b/.github/HowToQuestions.md similarity index 100% rename from __tests__/.github/HowToQuestions.md rename to .github/HowToQuestions.md diff --git a/__tests__/__image_snapshots__/chrome-docker/offline-ui-js-offline-ui-should-display-network-interruption-occurred-reconnecting-status-when-connection-is-interrupted-1-snap.png b/__tests__/__image_snapshots__/chrome-docker/offline-ui-js-offline-ui-should-display-network-interruption-occurred-reconnecting-status-when-connection-is-interrupted-1-snap.png new file mode 100644 index 0000000000000000000000000000000000000000..3ddfe723b3ff908f8974eb684d0a18d3e132579b GIT binary patch literal 10156 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 literal 0 HcmV?d00001 diff --git a/__tests__/__image_snapshots__/chrome-docker/offline-ui-js-offline-ui-should-display-the-connecting-connectivity-status-when-connecting-for-the-first-time-1-snap.png b/__tests__/__image_snapshots__/chrome-docker/offline-ui-js-offline-ui-should-display-the-connecting-connectivity-status-when-connecting-for-the-first-time-1-snap.png index 7bb5bf430f04b01c3e37b73b28ab0643079f928d..2299115761ef34913b98fae0218b0e5f6cd82520 100644 GIT binary patch literal 8069 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 cq_+f0QY?h(cT8Hsm+UCCj%s$RU3tR`(f*C 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 literal 0 HcmV?d00001 diff --git a/__tests__/offlineUI.js b/__tests__/offlineUI.js index a344021295..1b00a64cb6 100644 --- a/__tests__/offlineUI.js +++ b/__tests__/offlineUI.js @@ -1,6 +1,7 @@ import { By, Condition, Key } from 'selenium-webdriver'; import { imageSnapshotOptions, timeouts } from './constants.json'; +import staticSpinner from './setup/assets/staticSpinner'; import uiConnected from './setup/conditions/uiConnected'; // selenium-webdriver API doc: @@ -18,9 +19,17 @@ const allOutgoingMessagesFailed = new Condition('All outgoing messages to fail s }); describe('offline UI', async () => { - test('should show "slow to connect" UI when connection is slow', async () => { + test('should show "taking longer than usual to connect" UI when connection is slow', async () => { + + const WEB_CHAT_PROPS = { styleOptions: { spinnerAnimationBackgroundImage: staticSpinner } }; + const { driver } = await setupWebDriver({ + props: { WEB_CHAT_PROPS }, createDirectLine: options => { + // This part of code is running in the JavaScript VM in Chromium. + // This variable must be declared within scope + const ONLINE = 2; + const workingDirectLine = window.WebChat.createDirectLine(options); return { @@ -32,7 +41,7 @@ describe('offline UI', async () => { complete: () => observer.complete(), error: err => observer.error(err), next: connectionStatus => { - connectionStatus !== 2 && observer.next(connectionStatus); + connectionStatus !== ONLINE && observer.next(connectionStatus); } }); @@ -177,8 +186,15 @@ describe('offline UI', async () => { }); test('should display the "Connecting..." connectivity status when connecting for the first time', async() => { + const WEB_CHAT_PROPS = { spinnerAnimationBackgroundImage: staticSpinner }; + const { driver } = await setupWebDriver({ + props: WEB_CHAT_PROPS, createDirectline: options => { + // This part of code is running in the JavaScript VM in Chromium. + // This Direct Line Connection Status variable must be declared within scope + const UNINITIALIZED = 0; + const workingDirectLine = window.WebChat.createDirectLine(options); return { @@ -186,15 +202,63 @@ describe('offline UI', async () => { postActivity: workingDirectLine.postActivity.bind(workingDirectLine), connectionStatus$: new Observable(observer => { - const subscription = workingDirectLine.connectionStatus$.subscribe( { + const subscription = workingDirectLine.connectionStatus$.subscribe({ + complete: () => observer.complete(), + error: err => observer.error(err), + next: connectionStatus => { + connectionStatus === UNINITIALIZED && observer.next(connectionStatus); + } + }); + + return () => subscription.unsubscribe(); + }) + }; + }, + pingBotOnLoad: false, + setup: () => new Promise(resolve => { + const scriptElement = document.createElement('script'); + + scriptElement.onload = resolve; + scriptElement.setAttribute('src', 'https://unpkg.com/core-js@2.6.3/client/core.min.js'); + + document.head.appendChild(scriptElement); + + }) + }); + + const base64PNG = await driver.takeScreenshot(); + expect(base64PNG).toMatchImageSnapshot(imageSnapshotOptions); + }); + + test('should display "Network interruption occurred. Reconnecting…" status when connection is interrupted', async () => { + const WEB_CHAT_PROPS = { styleOptions: { spinnerAnimationBackgroundImage: staticSpinner } }; + + const { driver } = await setupWebDriver({ + props: { WEB_CHAT_PROPS }, + createDirectLine: options => { + // This part of code is running in the JavaScript VM in Chromium. + // These Direct Line Connection Status variables must be declared within scope + const CONNECTING = 1; + + const ONLINE = 2; + + const reconnectingDirectLine = window.WebChat.createDirectLine(options); + + return { + activity$: reconnectingDirectLine.activity$, + postActivity: reconnectingDirectLine.postActivity.bind(reconnectingDirectLine), + + connectionStatus$: new Observable(observer => { + const subscription = reconnectingDirectLine.connectionStatus$.subscribe({ complete: () => observer.complete(), error: err => observer.error(err), next: connectionStatus => { - connectionStatus == 1 && observer.next(connectionStatus); + observer.next(connectionStatus); + connectionStatus === ONLINE && observer.next(CONNECTING); } }); - return subscription.unsubscribe(); + return () => subscription.unsubscribe(); }) }; }, @@ -206,12 +270,58 @@ describe('offline UI', async () => { scriptElement.setAttribute('src', 'https://unpkg.com/core-js@2.6.3/client/core.min.js'); document.head.appendChild(scriptElement); + }) + }); + + await driver.sleep(600); + const base64PNG = await driver.takeScreenshot(); + expect(base64PNG).toMatchImageSnapshot(imageSnapshotOptions); + }); + + test('should show "Taking longer than usual to connect" UI when reconnection is slow', async () => { + const { driver } = await setupWebDriver({ + createDirectLine: options => { + // This part of code is running in the JavaScript VM in Chromium. + // These Direct Line Connection Status variables must be declared within scope + const CONNECTING = 1; + + const ONLINE = 2; + + const reconnectingDirectLine = window.WebChat.createDirectLine(options); + + return { + activity$: reconnectingDirectLine.activity$, + postActivity: reconnectingDirectLine.postActivity.bind(reconnectingDirectLine), + + connectionStatus$: new Observable(observer => { + const subscription = reconnectingDirectLine.connectionStatus$.subscribe({ + complete: () => observer.complete(), + error: err => observer.error(err), + next: connectionStatus => { + observer.next(connectionStatus); + connectionStatus === ONLINE && observer.next(CONNECTING); + } + }); + return () => subscription.unsubscribe(); + }) + }; + }, + pingBotOnLoad: false, + setup: () => new Promise(resolve => { + const scriptElement = document.createElement('script'); + + scriptElement.onload = resolve; + scriptElement.setAttribute('src', 'https://unpkg.com/core-js@2.6.3/client/core.min.js'); + + document.head.appendChild(scriptElement); }) }); + await driver.sleep(17000); + const base64PNG = await driver.takeScreenshot(); - // Snapshots are intentionally not compared because the spinner will cause the snapshot to fail regularly + expect(base64PNG).toMatchImageSnapshot(imageSnapshotOptions); }); }); diff --git a/__tests__/setup/assets/staticSpinner.js b/__tests__/setup/assets/staticSpinner.js new file mode 100644 index 0000000000..a30654ba22 --- /dev/null +++ b/__tests__/setup/assets/staticSpinner.js @@ -0,0 +1,3 @@ + +export default () => +'url(\'\')'; diff --git a/__tests__/setup/setupTestFramework.js b/__tests__/setup/setupTestFramework.js index dd9605769a..50db2cde4e 100644 --- a/__tests__/setup/setupTestFramework.js +++ b/__tests__/setup/setupTestFramework.js @@ -66,7 +66,10 @@ global.setupWebDriver = async options => { (coverage, options, callback) => { window.__coverage__ = coverage; - main(options).then(() => callback(), callback); + main(options).then(() => callback(), err => { + console.error(err); + callback(err); + }); }, global.__coverage__, marshal({ diff --git a/package-lock.json b/package-lock.json index fa2f5c3fce..7dd4b0b29e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1767,9 +1767,9 @@ } }, "@octokit/rest": { - "version": "16.24.3", - "resolved": "https://registry.npmjs.org/@octokit/rest/-/rest-16.24.3.tgz", - "integrity": "sha512-fBr2ziN4WT9G9sYTfnNVI/0wCb68ZI5isNU48lfWXQDyAy4ftlrh0SkIbhL7aigXUjcY0cX5J46ypyRPH0/U0g==", + "version": "16.24.2", + "resolved": "https://registry.npmjs.org/@octokit/rest/-/rest-16.24.2.tgz", + "integrity": "sha512-s0q08+1UgX2Kc5SOciE03nPUrWjMQoiYh0YaU1IxGhjPZ0kjmzD1NUPmm7Zo1pQKuo1CBxG2alyFYvtqFwE1NA==", "dev": true, "requires": { "@octokit/request": "3.0.0", @@ -2056,12 +2056,12 @@ "dev": true }, "async": { - "version": "2.6.1", - "resolved": "https://registry.npmjs.org/async/-/async-2.6.1.tgz", - "integrity": "sha512-fNEiL2+AZt6AlAw/29Cr0UDe4sRAHCpEHh54WMz+Bb7QfNcFw4h3loofyJpLeQs4Yx7yuqu/2dLgM5hKOs6HlQ==", + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.2.tgz", + "integrity": "sha512-H1qVYh1MYhEEFLsP97cVKqCGo7KfCyTt6uEWqsTBr9SO84oK9Uwbyd/yCW+6rKJLHksBNUVWZDAjfS+Ccx0Bbg==", "dev": true, "requires": { - "lodash": "^4.17.10" + "lodash": "^4.17.11" } }, "async-each": { @@ -5652,9 +5652,9 @@ "dev": true }, "istanbul-lib-hook": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/istanbul-lib-hook/-/istanbul-lib-hook-2.0.3.tgz", - "integrity": "sha512-CLmEqwEhuCYtGcpNVJjLV1DQyVnIqavMLFHV/DP+np/g3qvdxu3gsPqYoJMXm15sN84xOlckFB3VNvRbf5yEgA==", + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/istanbul-lib-hook/-/istanbul-lib-hook-2.0.6.tgz", + "integrity": "sha512-829DKONApZ7UCiPXcOYWSgkFXa4+vNYoNOt3F+4uDJLKL1OotAoVwvThoEj1i8jmOj7odbYcR3rnaHu+QroaXg==", "dev": true, "requires": { "append-transform": "^1.0.0" @@ -5676,16 +5676,44 @@ } }, "istanbul-lib-report": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-2.0.4.tgz", - "integrity": "sha512-sOiLZLAWpA0+3b5w5/dq0cjm2rrNdAfHWaGhmn7XEFW6X++IV9Ohn+pnELAl9K3rfpaeBfbmH9JU5sejacdLeA==", + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-2.0.7.tgz", + "integrity": "sha512-wLH6beJBFbRBLiTlMOBxmb85cnVM1Vyl36N48e4e/aTKSM3WbOx7zbVIH1SQ537fhhsPbX0/C5JB4qsmyRXXyA==", "dev": true, "requires": { - "istanbul-lib-coverage": "^2.0.3", - "make-dir": "^1.3.0", + "istanbul-lib-coverage": "^2.0.4", + "make-dir": "^2.1.0", "supports-color": "^6.0.0" }, "dependencies": { + "istanbul-lib-coverage": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz", + "integrity": "sha512-LXTBICkMARVgo579kWDm8SqfB6nvSDKNqIOBEjmJRnL04JvoMHCYGWaMddQnseJYtkEuEvO/sIcOxPLk9gERug==", + "dev": true + }, + "make-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", + "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", + "dev": true, + "requires": { + "pify": "^4.0.1", + "semver": "^5.6.0" + } + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "semver": { + "version": "5.7.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.0.tgz", + "integrity": "sha512-Ya52jSX2u7QKghxeoFGpLwCtGlt7j0oY9DYb5apt9nPlJ42ID+ulTXESnt/qAQcoSERyZ5sl3LDIOw0nAn/5DA==", + "dev": true + }, "supports-color": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", @@ -5734,32 +5762,12 @@ } }, "istanbul-reports": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-2.1.1.tgz", - "integrity": "sha512-FzNahnidyEPBCI0HcufJoSEoKykesRlFcSzQqjH9x0+LC8tnnE/p/90PBLu8iZTxr8yYZNyTtiAujUqyN+CIxw==", + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-2.2.3.tgz", + "integrity": "sha512-T6EbPuc8Cb620LWAYyZ4D8SSn06dY9i1+IgUX2lTH8gbwflMc9Obd33zHTyNX653ybjpamAHS9toKS3E6cGhTw==", "dev": true, "requires": { "handlebars": "^4.1.0" - }, - "dependencies": { - "handlebars": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.1.2.tgz", - "integrity": "sha512-nvfrjqvt9xQ8Z/w0ijewdD/vvWDTOweBUm96NTr66Wfvo1mJenBLwcYmPs3TIBP5ruzYGD7Hx/DaM9RmhroGPw==", - "dev": true, - "requires": { - "neo-async": "^2.6.0", - "optimist": "^0.6.1", - "source-map": "^0.6.1", - "uglify-js": "^3.1.4" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } } }, "jest": { diff --git a/packages/component/src/Attachment/Assets/SpinnerAnimation.js b/packages/component/src/Attachment/Assets/SpinnerAnimation.js index 81f63d9f19..982587901d 100644 --- a/packages/component/src/Attachment/Assets/SpinnerAnimation.js +++ b/packages/component/src/Attachment/Assets/SpinnerAnimation.js @@ -1,14 +1,11 @@ -import { css } from 'glamor'; import React from 'react'; -const ROOT_CSS = css( { - backgroundImage: 'url(\'\')', - backgroundRepeat: 'no-repeat', - backgroundSize: 'contain', - height: 16, - paddingRight: 12, - width: 16 -} ); +import connectToWebChat from '../../connectToWebChat'; -export default () => -
+const ConnectSpinnerAnimation = connectToWebChat( + ({ styleSet }) => ({ styleSet }) +)(({ styleSet }) => +
+) + +export default () => diff --git a/packages/component/src/Attachment/Assets/TypingAnimation.js b/packages/component/src/Attachment/Assets/TypingAnimation.js index 66ecfa8f3e..0434574361 100644 --- a/packages/component/src/Attachment/Assets/TypingAnimation.js +++ b/packages/component/src/Attachment/Assets/TypingAnimation.js @@ -1,12 +1,11 @@ -import { css } from 'glamor'; import React from 'react'; -const ROOT_CSS = css({ - backgroundImage: 'url(\'\')', - backgroundRepeat: 'no-repeat', - height: 20, - width: 64 -}); +import connectToWebChat from '../../connectToWebChat'; -export default () => -
+const ConnectTypingAnimation = connectToWebChat( + ({ styleSet }) => ({ styleSet }) +)(({ styleSet }) => +
+) + +export default () => diff --git a/packages/component/src/Localization/en-US.js b/packages/component/src/Localization/en-US.js index fabff4a213..3e329d4ee1 100644 --- a/packages/component/src/Localization/en-US.js +++ b/packages/component/src/Localization/en-US.js @@ -49,7 +49,7 @@ function xMinutesAgo(dateStr) { export default { FAILED_CONNECTION_NOTIFICATION: 'Unable to connect.', - INITIAL_CONNECTION_NOTIFICATION: 'Connecting...', + INITIAL_CONNECTION_NOTIFICATION: 'Connecting…', INTERRUPTED_CONNECTION_NOTIFICATION: 'Network interruption occurred. Reconnecting…', // Do not localize {Retry}; it is a placeholder for "Retry". English translation should be, "Send failed. Retry." SEND_FAILED_KEY: 'Send failed. {Retry}.', diff --git a/packages/component/src/SendBox/ConnectivityStatus.js b/packages/component/src/SendBox/ConnectivityStatus.js index 58e67e8b39..e7370bf56c 100644 --- a/packages/component/src/SendBox/ConnectivityStatus.js +++ b/packages/component/src/SendBox/ConnectivityStatus.js @@ -1,110 +1,100 @@ +import classNames from 'classnames'; import React from 'react'; -import connectToWebChat from '../connectToWebChat'; import { localize } from '../Localization/Localize'; +import connectToWebChat from '../connectToWebChat'; import ErrorNotificationIcon from '../Attachment/Assets/ErrorNotificationIcon'; import SpinnerAnimation from '../Attachment/Assets/SpinnerAnimation'; import WarningNotificationIcon from '../Attachment/Assets/WarningNotificationIcon'; -import classnames from 'classnames'; - -class connectConnectivityAlert extends React.Component { - connectivityStatusText() { - const { connectivityStatus: { status, timeoutCompleted }, language } = this.props; - const { connectingSlow } = this.state; - - if ( this.timerId ) { - clearTimeout( this.timerId ); - } - - if ( status === 'uninitialized' || ( status === 'connected' && timeoutCompleted === false && connectingSlow === false ) ) { - this.timerId = setTimeout( () => { - if ( status === 'connected' ) { - this.props.dispatch( { type: 'DIRECT_LINE/CONNECT_TIMEOUT_COMPLETE' } ); - } - this.timerId = null; - }, 400 ); - return ( - - - {localize( 'INITIAL_CONNECTION_NOTIFICATION', language )} - - ); - } else if ( status === 'reconnecting' ) { - return ( - - - {localize( 'INTERRUPTED_CONNECTION_NOTIFICATION', language )} - - ); - } else if ( status === 'connectingslow' ) { - return ( - - - {localize( 'SLOW_CONNECTION_NOTIFICATION', language )} - - ); - } else if ( status === 'error' ) { - return ( - - - {localize( 'FAILED_CONNECTION_NOTIFICATION', language )} - - ); - } - } - - constructor( props ) { - super( props ); +class DebouncedConnectivityStatus extends React.Component { + constructor(props) { + super(props); this.state = { - connectivityStatus: props.connectivityStatus, - connectingSlow: false, - language: props.language, - styleSet: props.styleSet, - } + children: props.children, + since: Date.now() + }; } - componentDidUpdate( props ) { - const { connectivityStatus: { status } } = props; - const { connectingSlow } = this.state + componentWillReceiveProps(nextProps) { + const { children, interval } = nextProps; - if ( status === 'connectingslow' && connectingSlow === false ) { - this.setState( () => ( { connectingSlow: true } ) ); + if ( + children !== this.props.children + || interval !== this.props.interval + ) { + clearTimeout(this.timeout); + + this.timeout = setTimeout(() => { + this.setState(() => ({ + children, + since: Date.now() + })); + }, Math.max(0, interval - Date.now() + this.state.since)); } } - static getDerivedStateFromProps( props, state ) { - if ( props !== state ) { - state = { - connectivityStatus: props.connectivityStatus, - connectingSlow: false, - language: props.language, - styleSet: props.styleSet, - } - } - return state; + componentWillUnmount() { + clearTimeout(this.timeout); } render() { - const { connectivityStatus, styleSet } = this.state; - const { status } = connectivityStatus; - return ( -
- {this.connectivityStatusText()} -
- ); + return typeof this.state.children === 'function' ? this.state.children() : false; } } -export default connectToWebChat( - ( { connectivityStatus, language, styleSet } ) => ( { connectivityStatus, language, styleSet } ) -)( connectConnectivityAlert ) +const connectConnectivityStatus = (...selectors) => connectToWebChat( + ({ connectivityStatus, language }) => ({ connectivityStatus, language }), + ...selectors +) + +export default connectConnectivityStatus( + ({ styleSet }) => ({ styleSet }) +)( + ({ connectivityStatus, language, styleSet }) => +
+ + { () => + connectivityStatus === 'connectingslow' ? + + + { localize('SLOW_CONNECTION_NOTIFICATION', language) } + + : (connectivityStatus === 'error' || connectivityStatus === 'notconnected') ? + + + { localize('FAILED_CONNECTION_NOTIFICATION', language) } + + : connectivityStatus === 'uninitialized' ? + + + { localize('INITIAL_CONNECTION_NOTIFICATION', language) } + + : connectivityStatus === 'reconnecting' && + + + { localize('INTERRUPTED_CONNECTION_NOTIFICATION', language) } + + } + +
+ ) diff --git a/packages/component/src/Styles/StyleSet/SpinnerAnimation.js b/packages/component/src/Styles/StyleSet/SpinnerAnimation.js new file mode 100644 index 0000000000..8e47fc067c --- /dev/null +++ b/packages/component/src/Styles/StyleSet/SpinnerAnimation.js @@ -0,0 +1,19 @@ +export default function ( { + spinnerAnimationBackgroundImage, + spinnerAnimationHeight, + spinnerAnimationWidth, + spinnerAnimationPaddingRight + +} ) { + + const backgroundImage = spinnerAnimationBackgroundImage ? spinnerAnimationBackgroundImage : 'url(\'\')'; + + return { + backgroundImage: backgroundImage, + backgroundRepeat: 'no-repeat', + backgroundSize: 'contain', + height: spinnerAnimationHeight, + paddingRight: spinnerAnimationPaddingRight, + width: spinnerAnimationWidth + }; +} diff --git a/packages/component/src/Styles/StyleSet/TypingAnimation.js b/packages/component/src/Styles/StyleSet/TypingAnimation.js new file mode 100644 index 0000000000..3e473f9946 --- /dev/null +++ b/packages/component/src/Styles/StyleSet/TypingAnimation.js @@ -0,0 +1,15 @@ +export default function ( { + typingAnimationBackgroundImage, + typingAnimationHeight, + typingAnimationWidth, +} ) { + + const backgroundImage = typingAnimationBackgroundImage ? typingAnimationBackgroundImage : 'url(\'\')' + + return { + backgroundImage: backgroundImage, + backgroundRepeat: 'no-repeat', + height: typingAnimationHeight, + width: typingAnimationWidth + }; +} diff --git a/packages/component/src/Styles/createStyleSet.js b/packages/component/src/Styles/createStyleSet.js index c397104b98..555d292e4a 100644 --- a/packages/component/src/Styles/createStyleSet.js +++ b/packages/component/src/Styles/createStyleSet.js @@ -23,6 +23,7 @@ import createSendBoxTextareaStyle from './StyleSet/SendBoxTextarea'; import createSendBoxTextBoxStyle from './StyleSet/SendBoxTextBox'; import createSendStatusStyle from './StyleSet/SendStatus'; import createSingleAttachmentActivityStyle from './StyleSet/SingleAttachmentActivity'; +import createSpinnerAnimationStyle from './StyleSet/SpinnerAnimation'; import createStackedLayoutStyle from './StyleSet/StackedLayout'; import createSuggestedActionsStyle from './StyleSet/SuggestedActions'; import createSuggestedActionsStyleSet from './StyleSet/SuggestedActionsStyleSet'; @@ -30,6 +31,7 @@ import createSuggestedActionStyle from './StyleSet/SuggestedAction'; import createTextContentStyle from './StyleSet/TextContent'; import createTimestampStyle from './StyleSet/Timestamp'; import createTypingActivityStyle from './StyleSet/TypingActivity'; +import createTypingAnimationStyle from './StyleSet/TypingAnimation'; import createUploadButtonStyle from './StyleSet/UploadButton'; import createVideoAttachmentStyle from './StyleSet/VideoAttachment'; import createVideoContentStyle from './StyleSet/VideoContent'; @@ -74,12 +76,14 @@ export default function createStyleSet(options) { sendBoxTextBox: createSendBoxTextBoxStyle(options), sendStatus: createSendStatusStyle(options), singleAttachmentActivity: createSingleAttachmentActivityStyle(options), + spinnerAnimation: createSpinnerAnimationStyle(options), stackedLayout: createStackedLayoutStyle(options), suggestedAction: createSuggestedActionStyle(options), suggestedActions: createSuggestedActionsStyle(options), textContent: createTextContentStyle(options), timestamp: createTimestampStyle(options), typingActivity: createTypingActivityStyle(options), + typingAnimation: createTypingAnimationStyle(options), uploadButton: createUploadButtonStyle(options), videoAttachment: createVideoAttachmentStyle(options), videoContent: createVideoContentStyle(options), diff --git a/packages/component/src/Styles/defaultStyleSetOptions.js b/packages/component/src/Styles/defaultStyleSetOptions.js index ad42865087..8c1b01458a 100644 --- a/packages/component/src/Styles/defaultStyleSetOptions.js +++ b/packages/component/src/Styles/defaultStyleSetOptions.js @@ -102,6 +102,15 @@ const DEFAULT_OPTIONS = { failedConnectivity: '#C50F1F', slowConnectivity: '#EAA300', notificationText: '#5E5E5E', + + typingAnimationBackgroundImage: null, + typingAnimationHeight: 20, + typingAnimationWidth: 64, + + spinnerAnimationBackgroundImage: null, + spinnerAnimationHeight: 16, + spinnerAnimationWidth: 16, + spinnerAnimationPaddingRight: 12 }; export default DEFAULT_OPTIONS diff --git a/packages/core/src/actions/connect.js b/packages/core/src/actions/connect.js index b650f364c0..a0795c2780 100644 --- a/packages/core/src/actions/connect.js +++ b/packages/core/src/actions/connect.js @@ -4,7 +4,6 @@ const CONNECT_FULFILLING = `${ CONNECT }_FULFILLING`; const CONNECT_PENDING = `${ CONNECT }_PENDING`; const CONNECT_REJECTED = `${ CONNECT }_REJECTED`; const CONNECT_STILL_PENDING = `${ CONNECT }_STILL_PENDING`; -const CONNECT_TIMEOUT_COMPLETE = `${ CONNECT }_TIMEOUT_COMPLETE` export default function ( { directLine, userID, username } ) { return { @@ -23,6 +22,5 @@ export { CONNECT_FULFILLING, CONNECT_PENDING, CONNECT_REJECTED, - CONNECT_STILL_PENDING, - CONNECT_TIMEOUT_COMPLETE + CONNECT_STILL_PENDING } diff --git a/packages/core/src/reducers/connectivityStatus.js b/packages/core/src/reducers/connectivityStatus.js index a3053d587f..8df27e5ceb 100644 --- a/packages/core/src/reducers/connectivityStatus.js +++ b/packages/core/src/reducers/connectivityStatus.js @@ -3,71 +3,44 @@ import { CONNECT_PENDING, CONNECT_REJECTED, CONNECT_STILL_PENDING, - CONNECT_TIMEOUT_COMPLETE } from '../actions/connect'; +import { + RECONNECT_PENDING, + RECONNECT_FULFILLED +} from '../actions/reconnect'; + import { DISCONNECT_FULFILLED } from '../../lib/actions/disconnect'; -const DEFAULT_STATE = { - status: 'uninitialized', - timeoutCompleted: false -}; +const DEFAULT_STATE = 'uninitialized'; export default function (state = DEFAULT_STATE, { type, meta }) { switch (type) { case CONNECT_PENDING: - if (state.status !== 'uninitialized') { - state = { - ...state, - status: 'reconnecting', - timeoutCompleted: false - }; + case RECONNECT_PENDING: + if (state !== 'uninitialized') { + state = 'reconnecting'; } - break; case CONNECT_FULFILLED: - state = { - ...state, - status: 'connected', - timeoutCompleted: false - }; + state = 'connected'; + break; + case RECONNECT_FULFILLED: + state = 'reconnected'; break; case CONNECT_REJECTED: - state = { - ...state, - status: 'error', - timeoutCompleted: false - }; - + state = 'error'; break; case CONNECT_STILL_PENDING: - state = { - ...state, - status: 'connectingslow', - timeoutCompleted: false - }; - + state = 'connectingslow'; break; case DISCONNECT_FULFILLED: - state = { - ...state, - status: meta.error ? 'error' : 'notconnected', - timeoutCompleted: false - }; - - break; - - case CONNECT_TIMEOUT_COMPLETE: - state = { - ...state, - timeoutCompleted: true - }; - + state = meta.error ? 'error' : 'notconnected' break; default: break; diff --git a/packages/core/src/sagas/connectSaga.js b/packages/core/src/sagas/connectSaga.js index 3aa3e4d91b..b59953063e 100644 --- a/packages/core/src/sagas/connectSaga.js +++ b/packages/core/src/sagas/connectSaga.js @@ -1,5 +1,4 @@ import { - all, call, cancel, cancelled, @@ -128,14 +127,14 @@ function* reconnectSaga() { if (connectionStatus === ONLINE) { break; } else if (connectionStatus !== CONNECTING) { - throw new Error(`Failed to reconnect, DirectLineJS returned ${ connectionStatus }.`); + throw new Error(`Failed to reconnect. DirectLineJS returned ${ connectionStatus }.`); } } } // This is similar to behavior of redux-promise-middleware, but using saga instead of Promise. // We guarantee PENDING -> FULFILLING -> FULFILLED, or PENDING -> REJECTED. This will help us simplify logic in other part of code. -// Note that after the saga is cancelled, subsequent call to put() will silently ignored. +// Note that after the saga is cancelled, subsequent call to put() will be ignored silently. function* runAsyncEffect({ type, meta, payload }, callEffectFactory) { try { yield forkPut({ type: `${ type }_PENDING`, meta, payload }); @@ -163,7 +162,7 @@ function* takeDisconnectAsError() { function runAsyncEffectUntilDisconnect(baseAction, callEffectFactory) { // We cannot use saga cancel() here, because cancelling saga will prohibit us from sending *_REJECTED. - // Without REJECTED, it impact our assumptions around PENDING/FULFILLED/REJECTED. + // Without REJECTED, it impacts our assumptions around PENDING/FULFILLED/REJECTED. return runAsyncEffect( baseAction, function* () { diff --git a/packages/core/src/sagas/detectSlowConnectionSaga.js b/packages/core/src/sagas/detectSlowConnectionSaga.js index 41796843b8..d64e026017 100644 --- a/packages/core/src/sagas/detectSlowConnectionSaga.js +++ b/packages/core/src/sagas/detectSlowConnectionSaga.js @@ -1,16 +1,18 @@ -import { call, put, race, take } from "redux-saga/effects"; +import { call, put, race, take } from 'redux-saga/effects'; + import { CONNECT_FULFILLED, CONNECT_PENDING, CONNECT_REJECTED, CONNECT_STILL_PENDING -} from "../actions/connect"; -import sleep from "../utils/sleep"; +} from '../actions/connect'; +import { RECONNECT_PENDING } from '../actions/reconnect'; +import sleep from '../utils/sleep'; export default function* detectSlowConnectionSaga() { for (;;) { - yield take(CONNECT_PENDING); + yield take([CONNECT_PENDING, RECONNECT_PENDING]); const connectivityRace = yield race({ fulfilled: take(CONNECT_FULFILLED),