From 022dc040237e0faafccb4a709378a627337892e4 Mon Sep 17 00:00:00 2001 From: Lukas Podmelle Date: Wed, 10 Jan 2024 10:33:14 +0100 Subject: [PATCH 1/9] Redesign --- package-lock.json | 2 +- samples/feedyou/newdesign.html | 30 + src/App/App.tsx | 4 +- .../templates/ExpandableTemplate/Header.tsx | 113 +- .../Signature/Signature.tsx | 103 +- .../Signature/SignatureLink.tsx | 40 +- .../Signature/SignatureTemplate.tsx | 17 +- src/Shell.tsx | 488 +++---- src/scss/botchat-redesign.scss | 1122 +++++++++++++++++ src/themes/ExpandableBarTheme.tsx | 6 +- src/themes/ExpandableKnobTheme.tsx | 15 +- src/themes/ExpandableKnobThemeNew.tsx | 21 + src/themes/SidebarTheme.tsx | 14 +- src/themes/index.ts | 65 +- typing-indicator.gif | Bin 0 -> 14372 bytes 15 files changed, 1664 insertions(+), 376 deletions(-) create mode 100644 samples/feedyou/newdesign.html create mode 100644 src/scss/botchat-redesign.scss create mode 100644 src/themes/ExpandableKnobThemeNew.tsx create mode 100644 typing-indicator.gif diff --git a/package-lock.json b/package-lock.json index 9b278a2f98..f2b49f66be 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "@feedyou-ai/feedbot-webchat-v0", - "version": "0.14.1-10", + "version": "0.14.1-12", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/samples/feedyou/newdesign.html b/samples/feedyou/newdesign.html new file mode 100644 index 0000000000..41bcd8798d --- /dev/null +++ b/samples/feedyou/newdesign.html @@ -0,0 +1,30 @@ + + + + + + Chatbot + + + + + + + + diff --git a/src/App/App.tsx b/src/App/App.tsx index 9ee0b42cd7..d3035ca4d5 100644 --- a/src/App/App.tsx +++ b/src/App/App.tsx @@ -10,7 +10,7 @@ import { generateUserId } from '../utils/generateUserId' export type AppProps = ChatProps & { theme?: Theme; // option to override theme settings from remote config defaultTheme?: Theme; // option to set default template when no remote config found (on default microsite for example) - header?: { textWhenCollapsed?: string; text: string, extraHtml?: string }; + header?: { textWhenCollapsed?: string; text: string, extraHtml?: string, image?: string }; channel?: { index?: number, id?: string }; autoExpandTimeout?: number; enableScreenshotUpload?: boolean; @@ -213,7 +213,7 @@ export const App = async (props: AppProps, container?: HTMLElement) => { // FEEDYOU configurable theming if (props.theme || !container) { - const theme = { mainColor: "#D83838", ...props.theme }; + const theme = { mainColor: "#0063f8", ...props.theme }; props.theme && (props.theme.enableScreenshotUpload = !!props.enableScreenshotUpload) const themeStyle = document.createElement("style"); themeStyle.type = "text/css"; diff --git a/src/App/templates/ExpandableTemplate/Header.tsx b/src/App/templates/ExpandableTemplate/Header.tsx index 0780a63180..2d25725345 100644 --- a/src/App/templates/ExpandableTemplate/Header.tsx +++ b/src/App/templates/ExpandableTemplate/Header.tsx @@ -1,11 +1,12 @@ -import * as React from 'react' -import { AppProps } from '../../App' +import * as React from 'react'; +import { AppProps } from '../../App'; +import { Theme } from '../../../themes'; export type Props = { - appProps: AppProps - onClick(): void - isCollapsed: boolean -} + appProps: AppProps; + onClick(): void; + isCollapsed: boolean; +}; export const Header: React.StatelessComponent = ({ appProps, @@ -13,39 +14,79 @@ export const Header: React.StatelessComponent = ({ isCollapsed, }) => { const { - theme: { mainColor }, + theme: { mainColor, template }, header: { extraHtml }, - } = appProps - - const backgroundColor = mainColor || '#e51836' - const title = getTitle(appProps, isCollapsed) - - + } = appProps; + + let backgroundColor; + {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} + if (template.type !== 'expandable-knob') { + backgroundColor = mainColor || '#e51836'; + } + + const title = getTitle(appProps, isCollapsed); + const avatar = + template.avatar || + "data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='36' height='36' viewBox='0 0 36 36' fill='none'%3E%3Cpath d='M36 18c0 9.941-8.059 18-18 18S0 27.941 0 18 8.059 0 18 0s18 8.059 18 18Z' fill='%23F7F9FB'/%3E%3Cpath d='M14.024 18.026h-.003v-.005h.004l.001.002v.001l-.002.002Zm3.978 0v-.001l.001-.002v-.002h-.006v.004l.002.001h.003Zm3.976 0h-.003l-.001-.002v-.002l.001-.001h.004v.003l-.001.002Z' fill='%23385B75'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M18.718 9.023a8.912 8.912 0 0 1 5.669 2.59 8.91 8.91 0 0 1-10.583 14.12l-3.073 1.182a1.274 1.274 0 0 1-1.646-1.646l1.182-3.073a8.91 8.91 0 0 1 8.45-13.173ZM14.023 17a1.024 1.024 0 1 0 0 2.047 1.024 1.024 0 0 0 0-2.047Zm3.409.172a1.023 1.023 0 1 1 1.136 1.702 1.023 1.023 0 0 1-1.136-1.702ZM21.977 17a1.023 1.023 0 1 0 0 2.047 1.023 1.023 0 0 0 0-2.047Z' fill='%23385B75'/%3E%3C/svg%3E"; + return ( -
- - {title} - - - {extraHtml && } - +
+
+ {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} + {template.type === 'expandable-knob' ? ( +
+
+ ) : null} +
+ {title} + {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} + {template.type === 'expandable-knob' ? ( + + {/* LP: for dev; pak text odstranit */} + S čím vám dnes mohu pomoci? + {template.supportiveTitle} + + ): null} +
+
+ + {extraHtml && ( + + )} + e.preventDefault()} - className="feedbot-minimize" - href="#" - >_ + onClick={(e) => e.preventDefault()} + className='feedbot-minimize' + href='#'> + {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} + {template.type === 'expandable-knob' ? ( + + + + + + ) : ( + '_' + )} +
- ) -} + ); +}; const getTitle = (props: AppProps, isCollapsed: boolean) => { - const { text, textWhenCollapsed } = props.header - - const titleWhenExpanded = text || 'Chatbot' - const titleWhenCollapsed = textWhenCollapsed || titleWhenExpanded - const titleToShow = isCollapsed ? titleWhenCollapsed : titleWhenExpanded - - return titleToShow -} - -export type HeaderProps = Props + const { text, textWhenCollapsed } = props.header; + + const titleWhenExpanded = text || 'Chatbot'; + const titleWhenCollapsed = textWhenCollapsed || titleWhenExpanded; + const titleToShow = isCollapsed ? titleWhenCollapsed : titleWhenExpanded; + + return titleToShow; +}; + +export type HeaderProps = Props; diff --git a/src/App/templates/ExpandableTemplate/Signature/Signature.tsx b/src/App/templates/ExpandableTemplate/Signature/Signature.tsx index c90ac10954..b43f797cc5 100644 --- a/src/App/templates/ExpandableTemplate/Signature/Signature.tsx +++ b/src/App/templates/ExpandableTemplate/Signature/Signature.tsx @@ -1,64 +1,67 @@ -import * as React from 'react' -import { SignatureLink } from './SignatureLink' -import { Theme } from '../../../../themes' -import { SignatureTemplate } from './SignatureTemplate' +import * as React from 'react'; +import { SignatureLink } from './SignatureLink'; +import { Theme } from '../../../../themes'; +import { SignatureTemplate } from './SignatureTemplate'; -type SignatureSchema = Theme['signature'] -const FEEDYOU_LOGO_IMG_SRC = 'https://cdn.feedyou.ai/webchat/feedyou_logo_red.png' +type SignatureSchema = Theme['signature']; +const FEEDYOU_LOGO_IMG_SRC = + 'https://cdn.feedyou.ai/webchat/feedyou_logo_red.png'; export type Props = { - signature: SignatureSchema - botId: string -} + signature: SignatureSchema; + botId: string; +}; -const getLinkQueryString = (botId: string) => `?utm_source=webchat&utm_medium=chatbot&utm_campaign=${botId}` +const getLinkQueryString = (botId: string) => + `?utm_source=webchat&utm_medium=chatbot&utm_campaign=${botId}`; -export const Signature: React.StatelessComponent = ({ signature, botId }) => { - const { partnerLogoUrl, partnerLinkUrl, partnerLogoStyle, mode } = signature - const attachQueryStringToUrl = (url: string) => `${url}${getLinkQueryString(botId)}` - - const enhancedFeedyouUrl = attachQueryStringToUrl('https://feedyou.ai') - const enhancedPartnerUrl = partnerLinkUrl ? attachQueryStringToUrl(partnerLinkUrl) : enhancedFeedyouUrl - - const feedyouLink = - - const partnerLink = - - if(mode === 'none') { - return null +export const Signature: React.StatelessComponent = ({ + signature, + botId, +}) => { + const { partnerLogoUrl, partnerLinkUrl, partnerLogoStyle, mode } = + signature; + const attachQueryStringToUrl = (url: string) => + `${url}${getLinkQueryString(botId)}`; + + const enhancedFeedyouUrl = attachQueryStringToUrl('https://feedyou.ai'); + const enhancedPartnerUrl = partnerLinkUrl + ? attachQueryStringToUrl(partnerLinkUrl) + : enhancedFeedyouUrl; + + const feedyouLink = ( + + ); + + const partnerLink = ( + + ); + + if (mode === 'none') { + return null; } - - if (partnerLogoUrl && mode === 'both') { + + if (mode === 'both') { return ( {partnerLink} -
&
+
&
{feedyouLink}
- ) + ); } - - if (partnerLogoUrl && mode === 'partner') { - return ( - - {partnerLink} - - ) + + if (mode === 'partner') { + return {partnerLink}; } - - return ( - - {feedyouLink} - - ) -} -export type SignatureProps = Props + return {feedyouLink}; +}; + +export type SignatureProps = Props; diff --git a/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx b/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx index c6b587f913..1edc3d8af1 100644 --- a/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx +++ b/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx @@ -1,28 +1,36 @@ -import * as React from 'react' -import cx from 'classnames' -import { CustomStylesForCssClass } from '../../../CustomStylesForCssClass' +import * as React from 'react'; +import cx from 'classnames'; +import { CustomStylesForCssClass } from '../../../CustomStylesForCssClass'; export type Props = { - href: string, - imgSrc: string - className?: string - customStyles?: string -} + href: string; + imgSrc?: string; + className?: string; + customStyles?: string; + text?: string; +}; export const SignatureLink: React.StatelessComponent = ({ href, imgSrc, className, customStyles, + text, }) => { return ( - - Logo - {className && customStyles && - } + + {imgSrc ? : text} + {className && customStyles && ( + + )} - ) -} + ); +}; - -export type SignatureLinkProps = Props +export type SignatureLinkProps = Props; diff --git a/src/App/templates/ExpandableTemplate/Signature/SignatureTemplate.tsx b/src/App/templates/ExpandableTemplate/Signature/SignatureTemplate.tsx index 12a439fe13..eb508dd803 100644 --- a/src/App/templates/ExpandableTemplate/Signature/SignatureTemplate.tsx +++ b/src/App/templates/ExpandableTemplate/Signature/SignatureTemplate.tsx @@ -1,17 +1,16 @@ -import * as React from 'react' +import * as React from 'react'; export type Props = { - botId: string -} + botId: string; +}; export const SignatureTemplate: React.StatelessComponent = ({ children }) => { return ( -
-
-
with ❤️ by
+
+
+
powered by
{children}
- ) -} - + ); +}; diff --git a/src/Shell.tsx b/src/Shell.tsx index 66e4a3ebab..c1b8305d1b 100644 --- a/src/Shell.tsx +++ b/src/Shell.tsx @@ -4,255 +4,307 @@ import { User } from 'botframework-directlinejs'; import { classList } from './Chat'; import { Dispatch, connect } from 'react-redux'; import { Strings } from './Strings'; -import { Speech } from './SpeechModule' -import { ChatActions, ListeningState, sendMessage, sendFiles, sendScreenshot } from './Store'; +import { Speech } from './SpeechModule'; +import { + ChatActions, + ListeningState, + sendMessage, + sendFiles, + sendScreenshot, +} from './Store'; -import * as html2canvas from 'html2canvas' +import * as html2canvas from 'html2canvas'; interface Props { - inputText: string, - strings: Strings, - listeningState: ListeningState, - showUploadButton: boolean, - uploadCapture: 'image/*' | 'video/*' | 'audio/*' | string, - disableInput: boolean + inputText: string; + strings: Strings; + listeningState: ListeningState; + showUploadButton: boolean; + uploadCapture: 'image/*' | 'video/*' | 'audio/*' | string; + disableInput: boolean; - onChangeText: (inputText: string) => void + onChangeText: (inputText: string) => void; - sendMessage: (inputText: string) => void, - sendFiles: (files: FileList) => void, - sendScreenshot: (screen: string) => void, - stopListening: () => void, - startListening: () => void + sendMessage: (inputText: string) => void; + sendFiles: (files: FileList) => void; + sendScreenshot: (screen: string) => void; + stopListening: () => void; + startListening: () => void; } export interface ShellFunctions { - focus: (appendKey?: string) => void + focus: (appendKey?: string) => void; } class ShellContainer extends React.Component implements ShellFunctions { - private textInput: HTMLTextAreaElement; - private fileInput: HTMLInputElement; + private textInput: HTMLTextAreaElement; + private fileInput: HTMLInputElement; - componentDidUpdate(prevProps: Props) { - if (prevProps.disableInput === true && this.props.disableInput === false) { - this.textInput.focus(); - } - } + componentDidUpdate(prevProps: Props) { + if ( + prevProps.disableInput === true && + this.props.disableInput === false + ) { + this.textInput.focus(); + } + } - private sendMessage() { - if (this.props.inputText.trim().length > 0) { - this.props.sendMessage(this.props.inputText); - } - } + private sendMessage() { + if (this.props.inputText.trim().length > 0) { + this.props.sendMessage(this.props.inputText); + } + } - private handleSendButtonKeyPress(evt: React.KeyboardEvent) { - if (evt.key === 'Enter' || evt.key === ' ') { - evt.preventDefault(); - this.sendMessage(); - this.textInput.focus(); - } - } + private handleSendButtonKeyPress( + evt: React.KeyboardEvent + ) { + if (evt.key === 'Enter' || evt.key === ' ') { + evt.preventDefault(); + this.sendMessage(); + this.textInput.focus(); + } + } - private handleUploadButtonKeyPress(evt: React.KeyboardEvent) { - if (evt.key === 'Enter' || evt.key === ' ') { - evt.preventDefault(); - this.fileInput.click(); - } - } + private handleUploadButtonKeyPress( + evt: React.KeyboardEvent + ) { + if (evt.key === 'Enter' || evt.key === ' ') { + evt.preventDefault(); + this.fileInput.click(); + } + } - private onKeyPress(e: React.KeyboardEvent) { - if (e.key === 'Enter' && !e.shiftKey) { - this.sendMessage(); - e.stopPropagation() - e.preventDefault() - } - } + private onKeyPress(e: React.KeyboardEvent) { + if (e.key === 'Enter' && !e.shiftKey) { + this.sendMessage(); + e.stopPropagation(); + e.preventDefault(); + } + } - private onClickSend() { - this.sendMessage(); - } + private onClickSend() { + this.sendMessage(); + } - private onChangeFile() { - this.props.sendFiles(this.fileInput.files); - this.fileInput.value = null; - this.textInput.focus(); - } + private onChangeFile() { + this.props.sendFiles(this.fileInput.files); + this.fileInput.value = null; + this.textInput.focus(); + } - private onTextInputFocus() { - if (this.props.listeningState === ListeningState.STARTED) { - this.props.stopListening(); - } - } + private onTextInputFocus() { + if (this.props.listeningState === ListeningState.STARTED) { + this.props.stopListening(); + } + } - private onClickMic() { - if (this.props.listeningState === ListeningState.STARTED) { - this.props.stopListening(); - } else if (this.props.listeningState === ListeningState.STOPPED) { - this.props.startListening(); - } - } + private onClickMic() { + if (this.props.listeningState === ListeningState.STARTED) { + this.props.stopListening(); + } else if (this.props.listeningState === ListeningState.STOPPED) { + this.props.startListening(); + } + } - public focus(appendKey?: string) { - this.textInput.focus(); + public focus(appendKey?: string) { + this.textInput.focus(); - if (appendKey) { - this.props.onChangeText(this.props.inputText + appendKey); - } - } + if (appendKey) { + this.props.onChangeText(this.props.inputText + appendKey); + } + } - private async takeScreenshot() { - const screen = await html2canvas(document.body, { allowTaint: true, useCORS: true }).then((canvas) => { - const dataURI = canvas.toDataURL("image/png"); + private async takeScreenshot() { + const screen = await html2canvas(document.body, { + allowTaint: true, + useCORS: true, + }).then((canvas) => { + const dataURI = canvas.toDataURL('image/png'); - return dataURI - }) - this.props.sendScreenshot(screen); - } + return dataURI; + }); + this.props.sendScreenshot(screen); + } - render() { - const className = classList( - 'wc-console', - this.props.inputText.length > 0 && 'has-text', - this.props.showUploadButton && 'has-upload-button', - this.props.disableInput && 'disable-input' - ); + render() { + const className = classList( + 'wc-console', + this.props.inputText.length > 0 && 'has-text', + this.props.showUploadButton && 'has-upload-button', + this.props.disableInput && 'disable-input' + ); - const showMicButton = this.props.listeningState !== ListeningState.STOPPED || (Speech.SpeechRecognizer.speechIsAvailable() && !this.props.inputText.length); + const showMicButton = + this.props.listeningState !== ListeningState.STOPPED || + (Speech.SpeechRecognizer.speechIsAvailable() && + !this.props.inputText.length); - const sendButtonClassName = classList( - 'wc-send', - showMicButton && 'hidden' - ); + const sendButtonClassName = classList( + 'wc-send', + showMicButton && 'hidden' + ); - const micButtonClassName = classList( - 'wc-mic', - !showMicButton && 'hidden', - this.props.listeningState === ListeningState.STARTED && 'active', - this.props.listeningState !== ListeningState.STARTED && 'inactive' - ); + const micButtonClassName = classList( + 'wc-mic', + !showMicButton && 'hidden', + this.props.listeningState === ListeningState.STARTED && 'active', + this.props.listeningState !== ListeningState.STARTED && 'inactive' + ); - const placeholder = this.props.listeningState === ListeningState.STARTED ? this.props.strings.listeningIndicator : this.props.strings.consolePlaceholder; + const placeholder = + this.props.listeningState === ListeningState.STARTED + ? this.props.strings.listeningIndicator + : this.props.strings.consolePlaceholder; - return ( -
- { - this.props.showUploadButton && - - } - { - this.props.showUploadButton && - this.fileInput = input} - // multiple - onChange={() => this.onChangeFile()} - aria-label={this.props.strings.uploadFile} - role="button" - capture={!!this.props.uploadCapture} - accept={this.props.uploadCapture} - /> - } - { - this.props.showUploadButton && - - } -
- -
- - -
- ); - } + return ( +
+ {this.props.showUploadButton && ( + + )} + {this.props.showUploadButton && ( + (this.fileInput = input)} + // multiple + onChange={() => this.onChangeFile()} + aria-label={this.props.strings.uploadFile} + role='button' + capture={!!this.props.uploadCapture} + accept={this.props.uploadCapture} + /> + )} + {this.props.showUploadButton && ( + + )} +
+ +
+ + +
+ ); + } } export const Shell = connect( - (state: ChatState) => ({ - // passed down to ShellContainer - inputText: state.shell.input, - showUploadButton: state.format.showUploadButton, - uploadCapture: state.format.uploadCapture, - disableInput: state.format.disableInput, - strings: state.format.strings, - // only used to create helper functions below - locale: state.format.locale, - user: state.connection.user, - listeningState: state.shell.listeningState - }), { - // passed down to ShellContainer - onChangeText: (input: string) => ({ type: 'Update_Input', input, source: "text" } as ChatActions), - stopListening: () => ({ type: 'Listening_Stopping' }), - startListening: () => ({ type: 'Listening_Starting' }), - // only used to create helper functions below - sendMessage, - sendFiles, - sendScreenshot -}, (stateProps: any, dispatchProps: any, ownProps: any): Props => ({ - // from stateProps - inputText: stateProps.inputText, - showUploadButton: stateProps.showUploadButton, - uploadCapture: stateProps.uploadCapture, - disableInput: stateProps.disableInput, - strings: stateProps.strings, - listeningState: stateProps.listeningState, - // from dispatchProps - onChangeText: dispatchProps.onChangeText, - // helper functions - sendMessage: (text: string) => dispatchProps.sendMessage(text, stateProps.user, stateProps.locale), - sendFiles: (files: FileList) => dispatchProps.sendFiles(files, stateProps.user, stateProps.locale), - sendScreenshot: (screen: string) => dispatchProps.sendScreenshot(screen, stateProps.user, stateProps.locale), - startListening: () => dispatchProps.startListening(), - stopListening: () => dispatchProps.stopListening() -}), { - withRef: true -} + (state: ChatState) => ({ + // passed down to ShellContainer + inputText: state.shell.input, + showUploadButton: state.format.showUploadButton, + uploadCapture: state.format.uploadCapture, + disableInput: state.format.disableInput, + strings: state.format.strings, + // only used to create helper functions below + locale: state.format.locale, + user: state.connection.user, + listeningState: state.shell.listeningState, + }), + { + // passed down to ShellContainer + onChangeText: (input: string) => + ({ type: 'Update_Input', input, source: 'text' } as ChatActions), + stopListening: () => ({ type: 'Listening_Stopping' }), + startListening: () => ({ type: 'Listening_Starting' }), + // only used to create helper functions below + sendMessage, + sendFiles, + sendScreenshot, + }, + (stateProps: any, dispatchProps: any, ownProps: any): Props => ({ + // from stateProps + inputText: stateProps.inputText, + showUploadButton: stateProps.showUploadButton, + uploadCapture: stateProps.uploadCapture, + disableInput: stateProps.disableInput, + strings: stateProps.strings, + listeningState: stateProps.listeningState, + // from dispatchProps + onChangeText: dispatchProps.onChangeText, + // helper functions + sendMessage: (text: string) => + dispatchProps.sendMessage(text, stateProps.user, stateProps.locale), + sendFiles: (files: FileList) => + dispatchProps.sendFiles(files, stateProps.user, stateProps.locale), + sendScreenshot: (screen: string) => + dispatchProps.sendScreenshot( + screen, + stateProps.user, + stateProps.locale + ), + startListening: () => dispatchProps.startListening(), + stopListening: () => dispatchProps.stopListening(), + }), + { + withRef: true, + } )(ShellContainer); diff --git a/src/scss/botchat-redesign.scss b/src/scss/botchat-redesign.scss new file mode 100644 index 0000000000..04fff57e7b --- /dev/null +++ b/src/scss/botchat-redesign.scss @@ -0,0 +1,1122 @@ +@import 'includes/card-size'; +@import url('https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700;800&display=swap'); + +// Colors +$black: #16364d; +$white: #ffffff; +$silver: #c7d0d7; +$silverLight: #f7f9fb; +$silverMedium: #dae0e5; +$grey: #628198; +$greyDark: #385b75; +$blue: #0063f8; +$blueLight: #ebf6ff; +$blueDark: #0a4fbd; +$shadow: rgba(0, 0, 0, 0.05); +$c_messageFromMe: $blue; +$c_messageFromThem: $white; + +// Settings +$chatWindowWidth: 420px; +$chatWindowHeight: 620px; + +$actionsHeight: 40px; +$consoleHeight: 70px; + +$fontFamily: 'Inter', sans-serif; +$fontSize: 15px; + +$headerTotalHeight: 38px; + +$actionTransition: 0.2s cubic-bezier(0, 0, 0.5, 1); + +$card_narrow: 216px; +$card_normal: 320px; +$card_wide: 416px; + +$card_borderWidth: 1px; +$card_padding: 8px; + +.feedbot-wrapper { + width: $chatWindowWidth; + height: $chatWindowHeight; + bottom: 24px; + right: 24px; + border-radius: 10px; + background-color: $white; + min-width: 275px; + max-width: 87%; + max-height: 90%; + position: fixed; + z-index: 100000; +} +.feedbot-wrapper:not(.collapsed) { + border: 1px solid $silverMedium; +} +.feedbot-header { + border-top-left-radius: 10px; + border-top-right-radius: 10px; + color: $greyDark; + z-index: 10; + padding: 18px; + cursor: pointer; + font-size: 1em; + display: flex; + align-items: center; + font-family: $fontFamily; +} +.feedbot-header .feedbot-title { + font-weight: 500; + flex-grow: 1; +} +.feedbot-header .feedbot-minimize { + font-weight: bolder; + font-family: Verdana; + color: $greyDark; + font-size: 1.2em; + line-height: 0.9em; +} +.feedbot-wrapper .wc-chatview-panel { + border-bottom-right-radius: 10px; + border-bottom-left-radius: 10px; +} +.wc-app .wc-chatview-panel { + top: 75px; +} +.wc-app .wc-header { + display: none; +} +.wc-message-wrapper:not([data-activity-id='retry']) .wc-message-from { + display: none; +} +.wc-carousel + .wc-hscroll + > ul + > li + > .wc-card + > div + > .ac-container + > .ac-container + .ac-image { + border-radius: 10px 10px 0 0; +} +.wc-carousel .wc-hscroll ul { + display: flex; + align-items: stretch; +} +.wc-carousel .wc-hscroll > ul > li > .wc-card, +.wc-carousel .wc-hscroll > ul > li > .wc-card > div { + height: 100%; +} +.feedbot-wrapper .wc-adaptive-card, +.feedbot-wrapper .wc-card { + max-width: 100%; +} +.wc-carousel .wc-hscroll > ul > li > .wc-card > div > .ac-container { + padding: 0 0 10px 0 !important; + height: 100%; + justify-content: space-between !important; +} +.wc-carousel + .wc-hscroll + > ul + > li + > .wc-card + > div + > .ac-container + > .ac-container + .ac-textBlock { + padding: 0px 13px; + white-space: unset !important; + text-overflow: unset !important; + overflow: unset !important; + font-family: $fontFamily !important; +} +.wc-carousel .wc-hscroll > ul > li > .wc-card > div .ac-actionSet { + margin: 4px 13px 13px !important; +} +.feedbot-wrapper .wc-app .wc-card button, +.feedbot-wrapper .wc-app .wc-card button > div { + text-overflow: initial !important; + white-space: initial !important; +} +.feedbot-wrapper .wc-suggested-actions .wc-hscroll > ul > li button:focus, +.feedbot-wrapper .wc-console .wc-mic, +.wc-console .wc-send, +.feedbot-wrapper .wc-app .wc-card button { + outline: 0; +} +.wc-carousel .wc-hscroll > ul > li { + padding: 4px !important; +} +.feedbot-wrapper.collapsed { + border-radius: 40px; + width: 75px; + min-width: auto; + height: 75px; +} +.feedbot-wrapper.collapsed .feedbot-header { + border-radius: 40px; + height: 100%; + width: 100%; + padding: 0px; + text-indent: 999%; + white-space: nowrap; + overflow: hidden; + font-size: 0px; + /* background image (chat icon) is set in redesign template */ + background-size: 45px; + background-position: center; + background-repeat: no-repeat; + transition: transform 0.3s ease; +} +.feedbot-wrapper.collapsed .feedbot-header:hover { + transform: scale(1.05); + transition: transform 0.3s ease; +} +.feedbot-wrapper.collapsed .wc-chatview-panel { + display: none; +} +.feedbot-wrapper.collapsed .feedbot-signature, +.feedbot-wrapper.collapsed .feedbot-header .feedbot-extra-html, +.feedbot-disabled div.feedbot { + display: none; +} +.wc-list.tiles .ac-actionSet { + flex-direction: row !important; + flex-wrap: wrap; + justify-content: center; +} + +.wc-list.tiles .ac-container { + padding-top: 0px; + margin-top: -5px; +} + +.wc-list.tiles .ac-pushButton { + flex-basis: 44% !important; + min-height: 120px !important; + margin: 3% !important; + flex-direction: column !important; + position: relative; + padding: 16px; + top: 0; +} + +.feedbot-wrapper .wc-app .wc-list.tiles .wc-card .ac-pushButton:active { + background-color: unset; +} + +.wc-list.tiles .ac-pushButton img { + width: 36px !important; + height: 36px !important; + margin-right: 0px !important; + margin-bottom: 10px !important; +} + +.wc-list.tiles .ac-pushButton div { + overflow: unset !important; + text-overflow: unset !important; + white-space: unset !important; +} + +@media (max-width: 450px) { + .feedbot-wrapper .wc-list.tiles .wc-card { + border: none; + width: 100%; + } + .wc-list.tiles .ac-pushButton { + min-height: 95px !important; + } + .wc-list.tiles .ac-container { + padding: 0 !important; + margin-top: 0 !important; + } +} +.wc-upload-screenshot { + display: none !important; +} +.feedbot-wrapper.collapsed .feedbot-signature { + display: none; +} + +.feedbot-wrapper .feedbot-signature { + position: absolute; + bottom: -23px; + font-size: 13px; + right: -5px; + opacity: 0.3; + font-family: $fontFamily; + + height: 22px; + display: block; + align-items: center; + -webkit-transition: opacity 0.3s ease-in-out; + -moz-transition: opacity 0.3s ease-in-out; + -ms-transition: opacity 0.3s ease-in-out; + -o-transition: opacity 0.3s ease-in-out; + transition: opacity 0.3s ease-in-out; +} + +.feedbot-wrapper .feedbot-signature:hover { + opacity: 0.8; +} + +.feedbot-signature a { + transition: 0.3s; + color: black; + text-decoration: none; + margin: 0 4px; + display: flex; + align-items: center; +} + +.feedbot-signature a:hover { + cursor: pointer; +} + +.feedbot-signature a img { + height: 22px; +} + +.feedbot-signature-row { + display: flex; + height: 100%; +} +/* +* +* +/* reset */ +body .wc-app, +.wc-app button, +.wc-app input, +.wc-app textarea, +.popup-message { + font-family: $fontFamily; + font-size: $fontSize; +} + +.wc-app button { + background-color: $white; + border: 1px solid $silverMedium; + border-radius: 1px; + color: $blueDark; + cursor: pointer; + transition: color 0.2s ease, background-color 0.2s ease; +} + +.wc-app h1, +.wc-app h2, +.wc-app h3, +.wc-app h4, +.wc-app p, +.wc-app ul, +.wc-app ol { + margin: 0; + padding: 0; +} + +.wc-app audio, +.wc-app video { + display: block; +} + +/* docking */ + +.wc-hidden { + visibility: hidden; +} + +.wc-time { + color: $silver; + margin-bottom: 10px; +} + +.wc-message-groups { + background: $silverLight; + bottom: $consoleHeight; + left: 0; + transform: translateY(0); + outline: 0; + overflow-x: hidden; + overflow-y: scroll; + padding: 10px 5px 0px 10px; + position: absolute; + right: 0; + top: 0; + transition: transform $actionTransition; + border-top: 1px solid $silverMedium; + + &.no-header { + top: 0; + } +} + +.wc-message-group-content { + overflow: hidden; +} + +.wc-suggested-actions { + background-color: $silverLight; + bottom: $consoleHeight; + height: 0; + left: 0; + overflow: hidden; + position: absolute; + right: 0; + transition: height $actionTransition; + + .wc-hscroll > ul { + height: $actionsHeight; + padding: 0px 18px; + + > li { + display: inline-block; + margin-right: 6px; + max-width: 40%; + + button { + background-color: $white; + color: $blueDark; + min-height: 40px; + overflow: hidden; + padding: 0 16px; + text-overflow: ellipsis; + white-space: nowrap; + width: 100%; + border-radius: 5px; + border-width: 1px; + font-weight: bold; + } + + button:focus, + button:hover { + background-color: $blueLight; + border-color: $blueDark; + border-width: 1px; + color: $blueDark; + } + + button:active { + background-color: $blueLight; + border-color: $blueDark; + border-width: 1px; + color: $blueDark; + } + } + } + + button.scroll { + background-color: $silver; + height: $actionsHeight; + overflow: hidden; + padding: 0; + position: absolute; + top: 0; + width: 28px; + } + + button.scroll:disabled { + display: none; + } + + button.scroll:focus, + button.scroll:hover { + background-color: $grey; + } + + button.scroll svg { + fill: $white; + + //TODO: use proper svg coords in the path data + path { + transform: translateY(6px); + } + } + + button.scroll.previous { + left: 0; + } + + button.scroll.next { + right: 0; + } +} + +.wc-message-pane.show-actions { + .wc-message-groups { + top: $headerTotalHeight + $actionsHeight; + transform: translateY(-$actionsHeight); + } + .wc-suggested-actions { + height: $actionsHeight; + } +} + +/* views */ + +.wc-chatview-panel { + overflow: hidden; + position: absolute; + right: 0; + left: 0; + top: 0; + bottom: 0; + background: $silverLight; +} + +/* messages */ + +.wc-message-wrapper { + animation: animationFrames 2s; + animation-iteration-count: 1; + clear: both; + margin-bottom: 10px; + overflow: hidden; + position: relative; + /*transition: max-height 2s ease-in-out;*/ +} + +@keyframes animationFrames { + 0% { + /*max-height: 0;*/ + opacity: 0; + } + 20% { + opacity: 1; + } + 100% { + /*max-height: 2000px;*/ + } +} + +.wc-message { + position: relative; +} + +.wc-message-wrapper.carousel .wc-message { + max-width: none; + padding-right: 8px; +} + +.wc-message svg.wc-message-callout { + display: none; +} + +.wc-message-content { + box-shadow: 0px 1px 2px 0px $shadow; + padding: 13px; + word-break: break-word; + border: 1px solid $silverMedium; +} + +.wc-message-content.clickable { + cursor: pointer; +} + +.wc-message-content.selected { + box-shadow: 0px 1px 1px 0px $blueDark; +} + +.wc-message-content img { + max-height: 320px; + max-width: 100%; + border-radius: 4px; +} + +.wc-message-content .video iframe { + border: 0; +} + +.wc-message-content audio, +.wc-message-content video { + max-width: 100%; +} + +.wc-message-content audio + h1, +.wc-message-content video + h1 { + margin-top: 11px; +} + +.wc-message-from { + clear: both; + color: $silver; + font-size: 11px; + margin-top: 5px; +} + +/* cards */ + +.wc-card { + background-color: $white; + border-radius: 10px; + border: 1px solid $silverMedium; + + .non-adaptive-content { + margin: $card_padding $card_padding 0 $card_padding; + } + + button { + border: 1px solid $silverMedium; + color: $blueDark; + background: $white; + font-weight: bold; + min-height: 40px; + border-radius: 5px; + flex: auto; + } + + button:hover { + background-color: $blueLight; + border: 2px solid $blueDark; + color: $blueDark; + } + + button:active { + background-color: $blueLight; + border: 2px solid $blueDark; + color: $blueDark; + } + + &.receipt table { + border-collapse: collapse; + width: 100%; + } + + &.receipt th, + &.receipt td { + text-align: right; + vertical-align: top; + } + + &.receipt th:first-child, + &.receipt td:first-child { + text-align: left; + } + + &.receipt th { + color: $grey; + font-size: inherit; + font-weight: normal; + line-height: 1.75; + } + + &.receipt thead tr:last-child th { + padding-bottom: 16px; + } + + &.receipt th[colspan='2'] { + color: inherit; + font-size: 15px; + font-weight: 700; + } + + &.receipt td { + padding: 4px 8px 0 8px; + } + + &.receipt td img { + float: left; + margin: 5px 8px 8px 0; + max-height: 50px; + max-width: 50px; + } + + &.receipt div.title { + font-weight: bolder; + } + + &.receipt div.subtitle { + font-weight: lighter; + } + + &.receipt tbody tr, + &.receipt tfoot tr { + border-top: 1px solid $silver; + } + + &.receipt tbody tr:first-child, + &.receipt tfoot tr:first-child { + border-top-width: 2px; + } + + &.receipt tfoot td { + line-height: 2.25; + } + + &.receipt tfoot .total { + font-weight: bold; + } + + &.thumbnail img { + float: right; + margin-bottom: 10px; + margin-left: 10px; + width: 100px; + } + + &.signin h1 { + margin: 10px 24px 16px 14px; + } + + &.error { + text-align: center; + + .error-icon { + fill: $silver; + height: 56px; + margin-bottom: 2px; + margin-top: 20px; + padding-left: 12px; + } + + .error-text { + color: $silver; + font-weight: 600; + letter-spacing: 0.5px; + margin-bottom: 20px; + text-align: inherit; + } + } +} + +/* alternate chat sizes */ + +.wc-message { + max-width: 91%; +} + +@include card-size($card_normal); + +.wc-wide { + @include card-size($card_wide); +} + +.wc-narrow { + @include card-size($card_narrow); +} + +/* adaptive card adjustments from wc-card */ +.wc-adaptive-card { + p { + margin-left: 0; + margin-right: 0; + } +} + +/* list */ +.wc-list { + .ac-container { + padding: 0 !important; + } +} +.wc-list > img { + margin-top: 10px; +} + +.wc-list > .wc-card { + margin-top: 8px; + border: 0 !important; +} + +.wc-list > .wc-card:first-child { + margin-top: 0; +} + +/* horizontal scroll */ + +.wc-hscroll-outer { + /* allow horizontal scrolling but hide the scrollbar */ + overflow: hidden; +} + +.wc-hscroll { + /* allow horizontal scrolling but hide the scrollbar */ + overflow-x: scroll; + overflow-y: hidden; +} + +.wc-hscroll > ul { + white-space: nowrap; +} + +.wc-hscroll > ul > li { + display: inline-block; + vertical-align: top; + white-space: normal; +} + +/* carousel */ + +.wc-carousel { + position: relative; + margin-top: 8px; + + button.scroll { + background-color: $blue; + height: 31px; + overflow: hidden; + padding: 0px; + position: absolute; + top: 50%; + width: 43px; + border-radius: 100px; + border: 2px solid $blueDark; + } + + button.scroll:disabled { + display: none; + } + + button.scroll:focus, + button.scroll:hover { + background-color: $blueDark; + border: 2px solid $blueDark; + } + + button.scroll svg { + fill: $white; + } + + button.scroll.previous { + left: -16px; + } + + button.scroll.next { + right: -16px; + } + + .wc-hscroll > ul { + margin-left: -4px; + } + + .wc-hscroll > ul > li { + padding: 0 4px; + } + + .wc-hscroll > ul > li:last-child { + padding-right: 0; + } + + li { + p { + min-height: 4em; + white-space: normal; + } + + .wc-adaptive-card p { + min-height: initial; + } + + .wc-adaptive-card-content > div > div > .ac-textBlock:first-child { + margin-top: 8px; + } + } +} + +/* from me */ + +.wc-message-from-me { + float: right; + margin-right: 6px; +} + +.wc-message-from-me.wc-message-from { + text-align: right; +} + +.wc-message-from-me .wc-message-content { + background-color: $c_messageFromMe; + border: 1px solid $blueDark; + color: $white; + border-radius: 10px 10px 4px 10px; +} + +/* from bot */ + +.wc-message-from-bot { + float: left; + margin-left: 8px; +} + +.wc-message-from-bot .wc-message-content { + background-color: $c_messageFromThem; + color: $black; + border-radius: 10px 10px 10px 4px; +} + +/* console */ + +.wc-console { + background: #ffffff; + border: 1px solid #c7d0d7; + border-radius: 5px; + margin: 18px; + bottom: 0; + box-sizing: border-box; + height: 45px; + left: 0; + position: absolute; + right: 0; + box-shadow: 0px 1px 2px 0px $shadow; + + > .wc-upload, + > .wc-textbox, + > .wc-send, + > .wc-mic { + position: absolute; + top: 0; + vertical-align: middle; + } + + label, + button { + cursor: pointer; + display: inline-block; + height: 43px; + } + + svg { + fill: $silver; + margin: 9px; + } + + input[type='text'], + textarea { + border: none; + height: calc(100% - 12px); + outline: none; + padding: 0; + resize: none; + width: 100%; + padding-top: 12px; + background-color: transparent; + color: $black; + min-height: initial; + box-shadow: none; + line-height: initial; + box-sizing: initial; + } + + textarea::placeholder { + color: $silver; + } + + &.has-text .wc-send svg { + fill: $blue; + } + &.has-text { + border: 1px solid $grey; + } + + .wc-upload { + cursor: pointer; + position: relative; + + svg { + height: 25px; + width: 25px; + } + } + + #wc-upload-input { + font-size: 0; + height: 0; + left: 0; + opacity: 0; + outline: 0; + position: absolute; + top: 0; + width: 0; + } + + .wc-send { + right: 0; + } + + .wc-send.hidden { + visibility: hidden; + } + + &.has-upload-button .wc-textbox { + left: 48px; + } + + .wc-textbox { + bottom: 0; + left: 11px; + right: 49px; + + input { + background-color: transparent; + word-break: break-word; + } + } + + .wc-mic, + .wc-send { + background-color: transparent; + border: 0; + padding: 0; + right: 0; + + &.hidden { + visibility: hidden; + } + } + + .wc-send { + svg { + height: 25px; + width: 25px; + } + } + + .wc-mic { + &.active path#micFilling { + fill: rgb(78, 55, 135); + } + + &.inactive path#micFilling { + visibility: hidden; + } + } +} + +.wc-console.disable-input { + display: none; +} + +.wc-console.has-text .wc-send svg { + fill: $blue; +} + +/* animation */ + +.wc-typing { + background-image: url(''); + background-repeat: no-repeat; + height: 22px; + width: 50px; + background-size: contain; + background-position: center; +} + +.wc-animate-scroll { + left: 0; + position: absolute; + transition: left 0.8s ease; +} + +.wc-animate-scroll-rapid { + left: 0; + position: absolute; + transition: left 0.4s ease; +} + +.wc-animate-scroll-near { + left: 0; + position: absolute; + transition: left 0.3s ease-in-out; +} + +/* text formats */ + +.format-markdown > p { + margin-bottom: 0px; +} + +.format-markdown code { + white-space: pre-wrap; +} + +.format-markdown ol { + padding-left: 30px !important; /* numbers are right-aligned to the period */ +} + +.format-markdown ul { + padding-left: 33px !important; +} + +/* browser scrollbar customization */ + +.wc-app ::-webkit-scrollbar { + width: 8px; +} + +.wc-app ::-webkit-scrollbar * { + background-color: transparent; +} + +.wc-app ::-webkit-scrollbar-thumb { + background-color: $silver; +} + +/* download button for Unknown media */ + +.wc-icon-download { + display: inline-block; + height: 20px; + margin-left: 12px; + vertical-align: middle; + width: 15px; + + background-image: url('data:image/svg+xml,'); + background-repeat: no-repeat; + background-size: contain; +} + +.wc-text-download { + display: inline-block; + font-weight: 500; + text-decoration: none; + color: $white; +} + +.ac-textBlock img.emoji { + width: auto !important; +} + +/* redesigned header */ +.feedbot-header { + .feedbot-header-name { + display: flex; + align-items: center; + gap: 10px; + } + + .feedbot-avatar, + .feedbot-avatar img { + width: 40px; + height: 40px; + border-radius: 100px; + background-position: center; + background-size: cover; + } +} +.feedbot-wrapper.collapsed .feedbot-avatar { + display: none; +} +.feedbot-wrapper.collapsed .feedbot-header:before { + position: absolute; + content: ''; + top: 0px; + left: 0; + right: 0; + bottom: 0; + border-radius: 100px; + z-index: 0; + filter: brightness(0.7); + /* border-color is set in redesign template */ + border-style: solid; + border-width: 2px; +} +.feedbot-name { + display: flex; + flex-direction: column; + gap: 3px; +} +.feedbot-supportive-title { + font-size: 13px; + color: $grey; +} diff --git a/src/themes/ExpandableBarTheme.tsx b/src/themes/ExpandableBarTheme.tsx index 8fdf46acd7..422b08601a 100644 --- a/src/themes/ExpandableBarTheme.tsx +++ b/src/themes/ExpandableBarTheme.tsx @@ -1,5 +1,5 @@ -import { BaseTheme } from './BaseTheme' -import { Theme } from './index' +import { BaseTheme } from './BaseTheme'; +import { Theme } from './index'; export const ExpandableBarTheme = (theme: Theme) => ` .feedbot-reset { @@ -92,4 +92,4 @@ export const ExpandableBarTheme = (theme: Theme) => ` .wc-carousel .wc-hscroll > ul > li > .wc-card > div > .ac-container > .ac-container .ac-image{ border-radius: 5px 5px 0 0; } -` +`; diff --git a/src/themes/ExpandableKnobTheme.tsx b/src/themes/ExpandableKnobTheme.tsx index b237e623d5..cff918bc36 100644 --- a/src/themes/ExpandableKnobTheme.tsx +++ b/src/themes/ExpandableKnobTheme.tsx @@ -1,5 +1,5 @@ -import { ExpandableBarTheme } from './ExpandableBarTheme' -import { Theme } from './index' +import { ExpandableBarTheme } from './ExpandableBarTheme'; +import { Theme } from './index'; export const ExpandableKnobTheme = (theme: Theme) => ` .feedbot-reset { @@ -56,10 +56,11 @@ export const ExpandableKnobTheme = (theme: Theme) => ` height: 100%; padding: 0px; - background-image: url(${(theme.template && theme.template.iconUrl) - ? theme.template.iconUrl - : 'https://cdn.feedyou.ai/webchat/message-icon.png' -}); + background-image: url(${ + theme.template && theme.template.iconUrl + ? theme.template.iconUrl + : 'https://cdn.feedyou.ai/webchat/message-icon.png' + }); background-size: 50px 50px; background-position: 12px 12px; background-repeat: no-repeat; @@ -124,4 +125,4 @@ export const ExpandableKnobTheme = (theme: Theme) => ` } ${ExpandableBarTheme(theme)} -` +`; diff --git a/src/themes/ExpandableKnobThemeNew.tsx b/src/themes/ExpandableKnobThemeNew.tsx new file mode 100644 index 0000000000..05a6eb6c52 --- /dev/null +++ b/src/themes/ExpandableKnobThemeNew.tsx @@ -0,0 +1,21 @@ +import { Theme } from './index'; + +export const ExpandableKnobThemeNew = (theme: Theme) => ` +.feedbot-wrapper.collapsed .feedbot-header { + background-image: url(${ + theme.template && theme.template.iconUrl + ? theme.template.iconUrl + : '"' + + "data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='36' height='36' viewBox='0 0 36 36' fill='none'%3E%3Cpath d='M11.291 18.562L11.2898 18.5622L11.2882 18.5618L11.2869 18.5606L11.2864 18.5589L11.2867 18.5576L11.2879 18.5561L11.2898 18.5556C11.2904 18.5557 11.2916 18.5561 11.2921 18.5565C11.2926 18.557 11.2929 18.5576 11.293 18.5582L11.2928 18.5602L11.291 18.562Z' fill='white'/%3E%3Cpath d='M18.0016 18.5617L18.0025 18.5607L18.0031 18.5589L18.0026 18.5572C18.0022 18.5568 18.001 18.5558 18.0004 18.5556L17.9985 18.5558L17.997 18.557L17.9964 18.5589L17.9967 18.5602L17.9974 18.5612C17.9979 18.5617 17.9985 18.562 17.9991 18.5622L18.0016 18.5617Z' fill='white'/%3E%3Cpath d='M24.711 18.562L24.7097 18.5622C24.7091 18.5621 24.7078 18.5617 24.7074 18.5612C24.7069 18.5608 24.7066 18.5602 24.7065 18.5595L24.7066 18.5576L24.7079 18.5561L24.7097 18.5556C24.71 18.5556 24.7109 18.5558 24.7112 18.5559C24.7115 18.5561 24.7109 18.5557 24.7112 18.5559C24.7117 18.5564 24.7129 18.5576 24.713 18.5582L24.7128 18.5602L24.711 18.562Z' fill='white'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M19.4346 0.0453898C23.7144 0.351612 27.7405 2.19149 30.7729 5.22729C33.8088 8.25967 35.6487 12.2851 35.9546 16.5651C36.2605 20.8451 35.0123 25.0922 32.4388 28.5258C29.865 31.9593 26.1395 34.3494 21.9456 35.2567C17.7523 36.1643 13.3715 35.5283 9.60866 33.4655L3.46231 35.8297C2.84593 36.0668 2.16128 36.0551 1.55306 35.7973C0.944834 35.5394 0.460637 35.0551 0.202461 34.4469C-0.0557144 33.8388 -0.0673678 33.1541 0.169763 32.5377L2.53452 26.3913C0.471719 22.6289 -0.164628 18.2483 0.742975 14.0544C1.65029 9.86051 4.04038 6.1346 7.47397 3.56116C10.9074 0.98764 15.1546 -0.260549 19.4346 0.0453898ZM11.2898 17.4372C11.0679 17.4372 10.8511 17.503 10.6666 17.6263C10.4821 17.7495 10.3384 17.9247 10.2535 18.1297C10.1686 18.3346 10.1464 18.5601 10.1897 18.7777C10.2329 18.9953 10.3398 19.1952 10.4966 19.352C10.6535 19.5089 10.8534 19.6157 11.0709 19.659C11.2885 19.7023 11.514 19.6801 11.719 19.5952C11.924 19.5103 12.0991 19.3665 12.2224 19.1821C12.3456 18.9976 12.4114 18.7807 12.4114 18.5589C12.4114 18.2614 12.2932 17.9761 12.0829 17.7658C11.8725 17.5554 11.5872 17.4372 11.2898 17.4372ZM17.3766 17.6263C17.561 17.503 17.7779 17.4372 17.9997 17.4372C18.2972 17.4372 18.5825 17.5554 18.7929 17.7658C19.0032 17.9761 19.1214 18.2614 19.1214 18.5589C19.1214 18.7807 19.0556 18.9976 18.9324 19.1821C18.8091 19.3665 18.6339 19.5103 18.429 19.5952C18.224 19.6801 17.9985 19.7023 17.7809 19.659C17.5633 19.6157 17.3635 19.5089 17.2066 19.352C17.0497 19.1952 16.9429 18.9953 16.8996 18.7777C16.8564 18.5601 16.8786 18.3346 16.9635 18.1297C17.0484 17.9247 17.1921 17.7495 17.3766 17.6263ZM24.7097 17.4372C24.4879 17.4372 24.271 17.503 24.0866 17.6263C23.9021 17.7495 23.7583 17.9247 23.6734 18.1297C23.5885 18.3346 23.5663 18.5601 23.6096 18.7777C23.6529 18.9953 23.7597 19.1952 23.9166 19.352C24.0735 19.5089 24.2733 19.6157 24.4909 19.659C24.7085 19.7023 24.934 19.6801 25.139 19.5952C25.3439 19.5103 25.5191 19.3665 25.6423 19.1821C25.7656 18.9976 25.8314 18.7807 25.8314 18.5589C25.8314 18.2614 25.7132 17.9761 25.5028 17.7658C25.2925 17.5554 25.0072 17.4372 24.7097 17.4372Z' fill='white'/%3E%3C/svg%3E" + + '"' + }); +} +.feedbot-wrapper.collapsed .feedbot-header { + background-color: ${theme.mainColor ? theme.mainColor : '#0063f8'}; +} +.feedbot-wrapper.collapsed .feedbot-header:before { + border-color: ${theme.mainColor ? theme.mainColor : '#0063f8'}; +} + +${theme.customCss || ''} +`; diff --git a/src/themes/SidebarTheme.tsx b/src/themes/SidebarTheme.tsx index cd3d95f13a..a313884be4 100644 --- a/src/themes/SidebarTheme.tsx +++ b/src/themes/SidebarTheme.tsx @@ -1,9 +1,9 @@ -import { ExpandableKnobTheme } from './ExpandableKnobTheme' -import { Theme } from './index' +import { ExpandableKnobTheme } from './ExpandableKnobTheme'; +import { Theme } from './index'; function getSidebarBackgroundColor(theme: Theme) { - return '#e1e1e1' - + return '#e1e1e1'; + // TODO make background tint configurable in theme /*const color = theme.mainColor if(color.startsWith("rgb")){ @@ -79,7 +79,9 @@ export const SidebarTheme = (theme: Theme) => ` @supports ((-webkit-backdrop-filter: blur(40px)) or (backdrop-filter: blur(40px))) { .feedbot-wrapper { max-height: 100%; - background: linear-gradient(45deg, ${getSidebarBackgroundColor(theme)}33, #E1E1E1CE); + background: linear-gradient(45deg, ${getSidebarBackgroundColor( + theme + )}33, #E1E1E1CE); backdrop-filter: blur(40px); -webkit-backdrop-filter: blur(40px); } @@ -137,4 +139,4 @@ export const SidebarTheme = (theme: Theme) => ` justify-content: center; } -` +`; diff --git a/src/themes/index.ts b/src/themes/index.ts index 9bd784270b..3924c3a7f2 100644 --- a/src/themes/index.ts +++ b/src/themes/index.ts @@ -1,49 +1,58 @@ -import { ExpandableBarTheme } from './ExpandableBarTheme' -import { FullScreenTheme } from './FullScreenTheme' -import { ExpandableKnobTheme } from './ExpandableKnobTheme' -import { SidebarTheme } from './SidebarTheme' +import { ExpandableBarTheme } from './ExpandableBarTheme'; +import { FullScreenTheme } from './FullScreenTheme'; +import { ExpandableKnobTheme } from './ExpandableKnobTheme'; +import { SidebarTheme } from './SidebarTheme'; +import { ExpandableKnobThemeNew } from './ExpandableKnobThemeNew'; export type Theme = { mainColor: string; template?: { // Dost možná tu nějaký propy chyběj, // tak je neváhej připsat! :) - autoExpandTimeout?: number, - type?: string, - headerText?: string, - collapsedHeaderText?: string, + autoExpandTimeout?: number; + type?: string; + headerText?: string; + collapsedHeaderText?: string; popupMessage?: { - title: string, - description: string, - timeout: number - }, - iconUrl?: string, - customScript?: string, - logoUrl?: string, + title: string; + description: string; + timeout: number; + }; + iconUrl?: string; + customScript?: string; + logoUrl?: string; + avatar?: string; + supportiveTitle?: string; }; customCss?: string; - showSignature?: boolean, - enableScreenshotUpload?: boolean + showSignature?: boolean; + enableScreenshotUpload?: boolean; signature?: { - partnerLogoUrl: string, - partnerLogoStyle: string, - partnerLinkUrl: string, - mode: string - } + partnerLogoUrl: string; + partnerLogoStyle: string; + partnerLinkUrl: string; + mode: string; + }; }; export function getStyleForTheme(theme: Theme, remoteConfig: boolean): string { switch (theme && theme.template && theme.template.type) { case 'expandable-bar': - return ExpandableBarTheme(theme) + return ExpandableBarTheme(theme); case 'full-screen': - return FullScreenTheme(theme) + return FullScreenTheme(theme); case 'expandable-knob': - return ExpandableKnobTheme(theme) + return ExpandableKnobThemeNew( + theme + ); /* LP: pro dev; vrátit zpět ExpandableKnobTheme(theme);*/ case 'sidebar': - return SidebarTheme(theme) + return SidebarTheme(theme); + case 'expandable-knob-new': + return ExpandableKnobThemeNew(theme); } - + // backward compatibility - knob is new default for remote config, old default is bar - return remoteConfig ? ExpandableKnobTheme(theme) : ExpandableBarTheme(theme) + return remoteConfig + ? ExpandableKnobTheme(theme) + : ExpandableBarTheme(theme); } diff --git a/typing-indicator.gif b/typing-indicator.gif new file mode 100644 index 0000000000000000000000000000000000000000..379608723e4e58c49ff8db83e5abb522aac2caab GIT binary patch literal 14372 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 Date: Wed, 10 Jan 2024 14:43:10 +0100 Subject: [PATCH 2/9] Redesign --- .prettierignore | 1 - .../ExpandableTemplate/ExpandableTemplate.tsx | 2 +- .../templates/ExpandableTemplate/Header.tsx | 18 ++---- .../Signature/Signature.tsx | 64 ++++++++++++++++--- .../Signature/SignatureLink.tsx | 4 +- src/scss/botchat-redesign.scss | 38 +++++------ src/themes/ExpandableKnobThemeNew.tsx | 10 ++- src/themes/SidebarThemeNew.tsx | 34 ++++++++++ src/themes/index.ts | 8 ++- 9 files changed, 127 insertions(+), 52 deletions(-) create mode 100644 src/themes/SidebarThemeNew.tsx diff --git a/.prettierignore b/.prettierignore index f59ec20aab..e69de29bb2 100644 --- a/.prettierignore +++ b/.prettierignore @@ -1 +0,0 @@ -* \ No newline at end of file diff --git a/src/App/templates/ExpandableTemplate/ExpandableTemplate.tsx b/src/App/templates/ExpandableTemplate/ExpandableTemplate.tsx index 3d7b5bf552..1026da6b68 100644 --- a/src/App/templates/ExpandableTemplate/ExpandableTemplate.tsx +++ b/src/App/templates/ExpandableTemplate/ExpandableTemplate.tsx @@ -94,7 +94,7 @@ export class ExpandableTemplate extends React.Component {
{signature && showSignature && - + } {this.doesTemplateSupportPopupMsg() && !initialized && popupMessage && ( diff --git a/src/App/templates/ExpandableTemplate/Header.tsx b/src/App/templates/ExpandableTemplate/Header.tsx index 2d25725345..9a8e80272c 100644 --- a/src/App/templates/ExpandableTemplate/Header.tsx +++ b/src/App/templates/ExpandableTemplate/Header.tsx @@ -19,9 +19,8 @@ export const Header: React.StatelessComponent = ({ } = appProps; let backgroundColor; - {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} - if (template.type !== 'expandable-knob') { - backgroundColor = mainColor || '#e51836'; + if (template.type !== 'expandable-knob-new' && template.type !== 'sidebar-new') { + backgroundColor = mainColor || '#fb584e'; } const title = getTitle(appProps, isCollapsed); @@ -33,20 +32,16 @@ export const Header: React.StatelessComponent = ({
+ style={{ backgroundColor }}>
- {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} - {template.type === 'expandable-knob' ? ( + {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? (
) : null}
{title} - {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} - {template.type === 'expandable-knob' ? ( + {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( - {/* LP: for dev; pak text odstranit */} - S čím vám dnes mohu pomoci? {template.supportiveTitle} ): null} @@ -64,8 +59,7 @@ export const Header: React.StatelessComponent = ({ onClick={(e) => e.preventDefault()} className='feedbot-minimize' href='#'> - {/* LP: for dev; pak změnit type na 'expandable-knob-new' */} - {template.type === 'expandable-knob' ? ( + {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( diff --git a/src/App/templates/ExpandableTemplate/Signature/Signature.tsx b/src/App/templates/ExpandableTemplate/Signature/Signature.tsx index b43f797cc5..c79bf7a509 100644 --- a/src/App/templates/ExpandableTemplate/Signature/Signature.tsx +++ b/src/App/templates/ExpandableTemplate/Signature/Signature.tsx @@ -1,6 +1,8 @@ import * as React from 'react'; import { SignatureLink } from './SignatureLink'; import { Theme } from '../../../../themes'; +import { AppProps } from '../../../App'; + import { SignatureTemplate } from './SignatureTemplate'; type SignatureSchema = Theme['signature']; @@ -9,18 +11,28 @@ const FEEDYOU_LOGO_IMG_SRC = export type Props = { signature: SignatureSchema; + appProps: AppProps; botId: string; }; - const getLinkQueryString = (botId: string) => `?utm_source=webchat&utm_medium=chatbot&utm_campaign=${botId}`; export const Signature: React.StatelessComponent = ({ signature, botId, + appProps, }) => { - const { partnerLogoUrl, partnerLinkUrl, partnerLogoStyle, mode } = - signature; + const { + partnerLogoUrl, + partnerLinkUrl, + partnerLogoStyle, + mode, + partnerName, + } = signature; + const { + theme: { template }, + } = appProps; + const attachQueryStringToUrl = (url: string) => `${url}${getLinkQueryString(botId)}`; @@ -30,16 +42,19 @@ export const Signature: React.StatelessComponent = ({ : enhancedFeedyouUrl; const feedyouLink = ( - + ); const partnerLink = ( ); @@ -47,17 +62,50 @@ export const Signature: React.StatelessComponent = ({ return null; } - if (mode === 'both') { + /* Webchat Redesign Signature */ + if ( + template.type === 'expandable-knob-new' || + template.type === 'sidebar-new' + ) { + if (mode === 'both') { + return ( + + {partnerLink} &{' '} + + Feedyou + + + ); + } + if (mode === 'partner') { + return {partnerLink}; + } + return ( + + + Feedyou + + + ); + } + + if (partnerLogoUrl && mode === 'both') { return ( {partnerLink} -
&
+
&
{feedyouLink} ); } - if (mode === 'partner') { + if (partnerLogoUrl && mode === 'partner') { return {partnerLink}; } diff --git a/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx b/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx index 1edc3d8af1..5b8811d85d 100644 --- a/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx +++ b/src/App/templates/ExpandableTemplate/Signature/SignatureLink.tsx @@ -4,7 +4,7 @@ import { CustomStylesForCssClass } from '../../../CustomStylesForCssClass'; export type Props = { href: string; - imgSrc?: string; + imgSrc: string; className?: string; customStyles?: string; text?: string; @@ -22,7 +22,7 @@ export const SignatureLink: React.StatelessComponent = ({ className={cx('signature-link', className)} target='_blank' href={href}> - {imgSrc ? : text} + {imgSrc ? Logo : text} {className && customStyles && ( ` +/* webchat redesign */ .feedbot-wrapper.collapsed .feedbot-header { background-image: url(${ theme.template && theme.template.iconUrl @@ -13,9 +14,14 @@ export const ExpandableKnobThemeNew = (theme: Theme) => ` .feedbot-wrapper.collapsed .feedbot-header { background-color: ${theme.mainColor ? theme.mainColor : '#0063f8'}; } -.feedbot-wrapper.collapsed .feedbot-header:before { - border-color: ${theme.mainColor ? theme.mainColor : '#0063f8'}; + +/* expandable knob specific */ +.feedbot-signature { + bottom: -23px; + font-size: 13px; + right: -5px; } +/* theme custom css */ ${theme.customCss || ''} `; diff --git a/src/themes/SidebarThemeNew.tsx b/src/themes/SidebarThemeNew.tsx new file mode 100644 index 0000000000..4c3ec5bd9a --- /dev/null +++ b/src/themes/SidebarThemeNew.tsx @@ -0,0 +1,34 @@ +import { Theme } from './index'; + +export const SidebarThemeNew = (theme: Theme) => ` +/* webchat redesign */ +.feedbot-wrapper.collapsed .feedbot-header { + background-image: url(${ + theme.template && theme.template.iconUrl + ? theme.template.iconUrl + : '"' + + "data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='36' height='36' viewBox='0 0 36 36' fill='none'%3E%3Cpath d='M11.291 18.562L11.2898 18.5622L11.2882 18.5618L11.2869 18.5606L11.2864 18.5589L11.2867 18.5576L11.2879 18.5561L11.2898 18.5556C11.2904 18.5557 11.2916 18.5561 11.2921 18.5565C11.2926 18.557 11.2929 18.5576 11.293 18.5582L11.2928 18.5602L11.291 18.562Z' fill='white'/%3E%3Cpath d='M18.0016 18.5617L18.0025 18.5607L18.0031 18.5589L18.0026 18.5572C18.0022 18.5568 18.001 18.5558 18.0004 18.5556L17.9985 18.5558L17.997 18.557L17.9964 18.5589L17.9967 18.5602L17.9974 18.5612C17.9979 18.5617 17.9985 18.562 17.9991 18.5622L18.0016 18.5617Z' fill='white'/%3E%3Cpath d='M24.711 18.562L24.7097 18.5622C24.7091 18.5621 24.7078 18.5617 24.7074 18.5612C24.7069 18.5608 24.7066 18.5602 24.7065 18.5595L24.7066 18.5576L24.7079 18.5561L24.7097 18.5556C24.71 18.5556 24.7109 18.5558 24.7112 18.5559C24.7115 18.5561 24.7109 18.5557 24.7112 18.5559C24.7117 18.5564 24.7129 18.5576 24.713 18.5582L24.7128 18.5602L24.711 18.562Z' fill='white'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M19.4346 0.0453898C23.7144 0.351612 27.7405 2.19149 30.7729 5.22729C33.8088 8.25967 35.6487 12.2851 35.9546 16.5651C36.2605 20.8451 35.0123 25.0922 32.4388 28.5258C29.865 31.9593 26.1395 34.3494 21.9456 35.2567C17.7523 36.1643 13.3715 35.5283 9.60866 33.4655L3.46231 35.8297C2.84593 36.0668 2.16128 36.0551 1.55306 35.7973C0.944834 35.5394 0.460637 35.0551 0.202461 34.4469C-0.0557144 33.8388 -0.0673678 33.1541 0.169763 32.5377L2.53452 26.3913C0.471719 22.6289 -0.164628 18.2483 0.742975 14.0544C1.65029 9.86051 4.04038 6.1346 7.47397 3.56116C10.9074 0.98764 15.1546 -0.260549 19.4346 0.0453898ZM11.2898 17.4372C11.0679 17.4372 10.8511 17.503 10.6666 17.6263C10.4821 17.7495 10.3384 17.9247 10.2535 18.1297C10.1686 18.3346 10.1464 18.5601 10.1897 18.7777C10.2329 18.9953 10.3398 19.1952 10.4966 19.352C10.6535 19.5089 10.8534 19.6157 11.0709 19.659C11.2885 19.7023 11.514 19.6801 11.719 19.5952C11.924 19.5103 12.0991 19.3665 12.2224 19.1821C12.3456 18.9976 12.4114 18.7807 12.4114 18.5589C12.4114 18.2614 12.2932 17.9761 12.0829 17.7658C11.8725 17.5554 11.5872 17.4372 11.2898 17.4372ZM17.3766 17.6263C17.561 17.503 17.7779 17.4372 17.9997 17.4372C18.2972 17.4372 18.5825 17.5554 18.7929 17.7658C19.0032 17.9761 19.1214 18.2614 19.1214 18.5589C19.1214 18.7807 19.0556 18.9976 18.9324 19.1821C18.8091 19.3665 18.6339 19.5103 18.429 19.5952C18.224 19.6801 17.9985 19.7023 17.7809 19.659C17.5633 19.6157 17.3635 19.5089 17.2066 19.352C17.0497 19.1952 16.9429 18.9953 16.8996 18.7777C16.8564 18.5601 16.8786 18.3346 16.9635 18.1297C17.0484 17.9247 17.1921 17.7495 17.3766 17.6263ZM24.7097 17.4372C24.4879 17.4372 24.271 17.503 24.0866 17.6263C23.9021 17.7495 23.7583 17.9247 23.6734 18.1297C23.5885 18.3346 23.5663 18.5601 23.6096 18.7777C23.6529 18.9953 23.7597 19.1952 23.9166 19.352C24.0735 19.5089 24.2733 19.6157 24.4909 19.659C24.7085 19.7023 24.934 19.6801 25.139 19.5952C25.3439 19.5103 25.5191 19.3665 25.6423 19.1821C25.7656 18.9976 25.8314 18.7807 25.8314 18.5589C25.8314 18.2614 25.7132 17.9761 25.5028 17.7658C25.2925 17.5554 25.0072 17.4372 24.7097 17.4372Z' fill='white'/%3E%3C/svg%3E" + + '"' + }); +} +.feedbot-wrapper.collapsed .feedbot-header { + background-color: ${theme.mainColor ? theme.mainColor : '#0063f8'}; +} + +/* sidebar specific */ +.feedbot-wrapper { + height: 100vh; + bottom: 0; + right: 0; + max-height: 100%; + border-radius: 0; +} +.feedbot-signature { + bottom: -2px; + font-size: 11px; + right: 13px; +} + +/* theme custom css */ +${theme.customCss || ''} +`; diff --git a/src/themes/index.ts b/src/themes/index.ts index 3924c3a7f2..abb877499c 100644 --- a/src/themes/index.ts +++ b/src/themes/index.ts @@ -3,6 +3,7 @@ import { FullScreenTheme } from './FullScreenTheme'; import { ExpandableKnobTheme } from './ExpandableKnobTheme'; import { SidebarTheme } from './SidebarTheme'; import { ExpandableKnobThemeNew } from './ExpandableKnobThemeNew'; +import { SidebarThemeNew } from './SidebarThemeNew'; export type Theme = { mainColor: string; @@ -31,6 +32,7 @@ export type Theme = { partnerLogoUrl: string; partnerLogoStyle: string; partnerLinkUrl: string; + partnerName: string; mode: string; }; }; @@ -42,13 +44,13 @@ export function getStyleForTheme(theme: Theme, remoteConfig: boolean): string { case 'full-screen': return FullScreenTheme(theme); case 'expandable-knob': - return ExpandableKnobThemeNew( - theme - ); /* LP: pro dev; vrátit zpět ExpandableKnobTheme(theme);*/ + return ExpandableKnobTheme(theme); case 'sidebar': return SidebarTheme(theme); case 'expandable-knob-new': return ExpandableKnobThemeNew(theme); + case 'sidebar-new': + return SidebarThemeNew(theme); } // backward compatibility - knob is new default for remote config, old default is bar From e11d93f38f0777439e2fdefaeff1ccce9b3580cb Mon Sep 17 00:00:00 2001 From: Lukas Podmelle Date: Sat, 13 Jan 2024 00:46:40 +0100 Subject: [PATCH 3/9] Redesign & Persistent menu --- .prettierignore | 1 + .../templates/ExpandableTemplate/Header.tsx | 145 ++++++++++++------ src/scss/botchat-redesign.scss | 51 +++++- src/themes/index.ts | 4 + 4 files changed, 153 insertions(+), 48 deletions(-) diff --git a/.prettierignore b/.prettierignore index e69de29bb2..f59ec20aab 100644 --- a/.prettierignore +++ b/.prettierignore @@ -0,0 +1 @@ +* \ No newline at end of file diff --git a/src/App/templates/ExpandableTemplate/Header.tsx b/src/App/templates/ExpandableTemplate/Header.tsx index 9a8e80272c..00f54d27d6 100644 --- a/src/App/templates/ExpandableTemplate/Header.tsx +++ b/src/App/templates/ExpandableTemplate/Header.tsx @@ -8,45 +8,59 @@ export type Props = { isCollapsed: boolean; }; -export const Header: React.StatelessComponent = ({ - appProps, - onClick, - isCollapsed, -}) => { - const { - theme: { mainColor, template }, - header: { extraHtml }, - } = appProps; +export type State = { + isMenuOpen: boolean; +} - let backgroundColor; - if (template.type !== 'expandable-knob-new' && template.type !== 'sidebar-new') { - backgroundColor = mainColor || '#fb584e'; - } +export class Header extends React.Component { + state: State = {isMenuOpen: false} - const title = getTitle(appProps, isCollapsed); - const avatar = - template.avatar || - "data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='36' height='36' viewBox='0 0 36 36' fill='none'%3E%3Cpath d='M36 18c0 9.941-8.059 18-18 18S0 27.941 0 18 8.059 0 18 0s18 8.059 18 18Z' fill='%23F7F9FB'/%3E%3Cpath d='M14.024 18.026h-.003v-.005h.004l.001.002v.001l-.002.002Zm3.978 0v-.001l.001-.002v-.002h-.006v.004l.002.001h.003Zm3.976 0h-.003l-.001-.002v-.002l.001-.001h.004v.003l-.001.002Z' fill='%23385B75'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M18.718 9.023a8.912 8.912 0 0 1 5.669 2.59 8.91 8.91 0 0 1-10.583 14.12l-3.073 1.182a1.274 1.274 0 0 1-1.646-1.646l1.182-3.073a8.91 8.91 0 0 1 8.45-13.173ZM14.023 17a1.024 1.024 0 1 0 0 2.047 1.024 1.024 0 0 0 0-2.047Zm3.409.172a1.023 1.023 0 1 1 1.136 1.702 1.023 1.023 0 0 1-1.136-1.702ZM21.977 17a1.023 1.023 0 1 0 0 2.047 1.023 1.023 0 0 0 0-2.047Z' fill='%23385B75'/%3E%3C/svg%3E"; + render() { + const { + appProps: { + theme: { mainColor, template }, + header: { extraHtml }, + }, + isCollapsed, + onClick, + } = this.props; - return ( -
-
- {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( -
+ const handlePersistentMenuToggle = () => { + this.setState((prevState) => ({ + isMenuOpen: !prevState.isMenuOpen + })) + } + + let backgroundColor; + if (template.type !== 'expandable-knob-new' && template.type !== 'sidebar-new') { + backgroundColor = mainColor || '#fb584e'; + } + + const title = getTitle(this.props.appProps, isCollapsed); + + const avatar = + template.avatar || + "data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='36' height='36' viewBox='0 0 36 36' fill='none'%3E%3Cpath d='M36 18c0 9.941-8.059 18-18 18S0 27.941 0 18 8.059 0 18 0s18 8.059 18 18Z' fill='%23F7F9FB'/%3E%3Cpath d='M14.024 18.026h-.003v-.005h.004l.001.002v.001l-.002.002Zm3.978 0v-.001l.001-.002v-.002h-.006v.004l.002.001h.003Zm3.976 0h-.003l-.001-.002v-.002l.001-.001h.004v.003l-.001.002Z' fill='%23385B75'/%3E%3Cpath fillRule='evenodd' clipRule='evenodd' d='M18.718 9.023a8.912 8.912 0 0 1 5.669 2.59 8.91 8.91 0 0 1-10.583 14.12l-3.073 1.182a1.274 1.274 0 0 1-1.646-1.646l1.182-3.073a8.91 8.91 0 0 1 8.45-13.173ZM14.023 17a1.024 1.024 0 1 0 0 2.047 1.024 1.024 0 0 0 0-2.047Zm3.409.172a1.023 1.023 0 1 1 1.136 1.702 1.023 1.023 0 0 1-1.136-1.702ZM21.977 17a1.023 1.023 0 1 0 0 2.047 1.023 1.023 0 0 0 0-2.047Z' fill='%23385B75'/%3E%3C/svg%3E"; + + return ( +
+
+ {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( +
+
+ ) : null} +
+ {title} + {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( + + {template.supportiveTitle} + + ): null}
- ) : null} -
- {title} - {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( - - {template.supportiveTitle} - - ): null}
-
{extraHtml && ( = ({ /> )} - e.preventDefault()} - className='feedbot-minimize' - href='#'> +
- ); -}; + ) + } + +} const getTitle = (props: AppProps, isCollapsed: boolean) => { const { text, textWhenCollapsed } = props.header; @@ -83,4 +126,12 @@ const getTitle = (props: AppProps, isCollapsed: boolean) => { return titleToShow; }; +/** + * Triggers a specified dialog for the persistent menu item from Channel settings + * @param dialogId Dialog ID to be triggered on persistent menu item click + */ +const handleTriggerDialog = (dialogId: string) => { + window.dispatchEvent(new CustomEvent('feedbot:trigger-dialog', { detail: dialogId })) +} + export type HeaderProps = Props; diff --git a/src/scss/botchat-redesign.scss b/src/scss/botchat-redesign.scss index 3a19e07886..ff8b68e319 100644 --- a/src/scss/botchat-redesign.scss +++ b/src/scss/botchat-redesign.scss @@ -59,7 +59,6 @@ $card_padding: 8px; color: $greyDark; z-index: 10; padding: 18px; - cursor: pointer; font-size: 1em; display: flex; justify-content: space-between; @@ -167,6 +166,7 @@ $card_padding: 8px; height: 100%; width: 100%; padding: 0px; + cursor: pointer; text-indent: 999%; white-space: nowrap; overflow: hidden; @@ -1098,6 +1098,7 @@ body .wc-app, border-radius: 100px; background-position: center; background-size: cover; + aspect-ratio: 1/1; } } .feedbot-wrapper.collapsed .feedbot-avatar { @@ -1112,3 +1113,51 @@ body .wc-app, font-size: 13px; color: $grey; } +.feedbot-header-actions { + display: flex; + align-items: center; + gap: 8px; +} +.feedbot-persistent-menu-toggle { + color: $greyDark; + padding: 5px 5px; + cursor: pointer; +} +.feedbot-persistent-menu-toggle:hover { + color: $black; +} +.feedbot-persistent-menu { + line-height: 0.9em; + position: relative; +} +.feedbot-persistent-menu-links { + position: absolute; + top: 35px; + right: -35px; + background-color: $white; + border: 1px solid $silver; + border-radius: 6px; + box-shadow: 0px 1px 2px 0px $shadow; + z-index: 1; + list-style: none; + padding: 10px 0; + margin: 0; + min-width: 10em; + max-width: 17em; +} +.feedbot-persistent-menu-links a { + padding: 10px 20px; + display: block; + box-sizing: border-box; + white-space: nowrap; + overflow: hidden; + text-overflow: ellipsis; + cursor: pointer; + transition: background-color .2s ease, border .2 ease; +} +.feedbot-persistent-menu-links a:hover { + background-color: $blueLight; + outline: 2px solid $blueDark; + border-radius: 4px; + transition: background-color .2s ease, border .2 ease; +} \ No newline at end of file diff --git a/src/themes/index.ts b/src/themes/index.ts index abb877499c..d8eef09d6c 100644 --- a/src/themes/index.ts +++ b/src/themes/index.ts @@ -24,6 +24,10 @@ export type Theme = { logoUrl?: string; avatar?: string; supportiveTitle?: string; + persistentMenu?: [{ + title: string; + dialog: string; + }] }; customCss?: string; showSignature?: boolean; From 9370da8aaaa8866bad6c23dbb352c1036c05de9b Mon Sep 17 00:00:00 2001 From: Lukas Podmelle Date: Tue, 16 Jan 2024 12:29:02 +0100 Subject: [PATCH 4/9] Redesign & Persistent menu --- .../templates/ExpandableTemplate/Header.tsx | 34 +++++++++++++++++-- src/scss/botchat-redesign.scss | 20 +++++++++-- src/themes/ExpandableKnobThemeNew.tsx | 8 ++--- src/themes/SidebarThemeNew.tsx | 6 ++-- src/themes/index.ts | 1 + 5 files changed, 58 insertions(+), 11 deletions(-) diff --git a/src/App/templates/ExpandableTemplate/Header.tsx b/src/App/templates/ExpandableTemplate/Header.tsx index 00f54d27d6..06c64c171b 100644 --- a/src/App/templates/ExpandableTemplate/Header.tsx +++ b/src/App/templates/ExpandableTemplate/Header.tsx @@ -42,6 +42,12 @@ export class Header extends React.Component { template.avatar || "data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='36' height='36' viewBox='0 0 36 36' fill='none'%3E%3Cpath d='M36 18c0 9.941-8.059 18-18 18S0 27.941 0 18 8.059 0 18 0s18 8.059 18 18Z' fill='%23F7F9FB'/%3E%3Cpath d='M14.024 18.026h-.003v-.005h.004l.001.002v.001l-.002.002Zm3.978 0v-.001l.001-.002v-.002h-.006v.004l.002.001h.003Zm3.976 0h-.003l-.001-.002v-.002l.001-.001h.004v.003l-.001.002Z' fill='%23385B75'/%3E%3Cpath fillRule='evenodd' clipRule='evenodd' d='M18.718 9.023a8.912 8.912 0 0 1 5.669 2.59 8.91 8.91 0 0 1-10.583 14.12l-3.073 1.182a1.274 1.274 0 0 1-1.646-1.646l1.182-3.073a8.91 8.91 0 0 1 8.45-13.173ZM14.023 17a1.024 1.024 0 1 0 0 2.047 1.024 1.024 0 0 0 0-2.047Zm3.409.172a1.023 1.023 0 1 1 1.136 1.702 1.023 1.023 0 0 1-1.136-1.702ZM21.977 17a1.023 1.023 0 1 0 0 2.047 1.023 1.023 0 0 0 0-2.047Z' fill='%23385B75'/%3E%3C/svg%3E"; + const startOverIcon = + + + + + return (
{ style={{ backgroundColor }}>
{template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( -
+
) : null}
@@ -70,7 +78,7 @@ export class Header extends React.Component { )}
- {template.type === 'expandable-knob-new' || template.type === 'sidebar-new' ? ( + {(template.type === 'expandable-knob-new' || template.type === 'sidebar-new') && (checkFeedbotTestMode() || template.persistentMenu.length > 0) ? (
@@ -79,6 +87,13 @@ export class Header extends React.Component { {this.state.isMenuOpen &&