feat: Add flat design for widget (#4065)

This commit is contained in:
Pranav Raj S 2022-02-25 16:18:18 +05:30 committed by GitHub
parent dcecbf4b80
commit 6c94768bdb
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
18 changed files with 251 additions and 109 deletions

View file

@ -1,5 +1,8 @@
class WidgetTestsController < ActionController::Base class WidgetTestsController < ActionController::Base
before_action :set_web_widget before_action :ensure_web_widget
before_action :ensure_widget_position
before_action :ensure_widget_type
before_action :ensure_widget_style
def index def index
render render
@ -7,7 +10,24 @@ class WidgetTestsController < ActionController::Base
private private
def set_web_widget def ensure_widget_style
@web_widget = Channel::WebWidget.first @widget_style = params[:widget_style] || 'standard'
end
def ensure_widget_position
@widget_position = params[:position] || 'left'
end
def ensure_widget_type
@widget_type = params[:type] || 'expanded_bubble'
end
def inbox_id
@inbox_id ||= params[:inbox_id] || Channel::WebWidget.first.inbox.id
end
def ensure_web_widget
@inbox = Inbox.find(inbox_id)
@web_widget = @inbox.channel
end end
end end

View file

@ -1,25 +1,11 @@
import Cookies from 'js-cookie'; import Cookies from 'js-cookie';
import { IFrameHelper } from '../sdk/IFrameHelper'; import { IFrameHelper } from '../sdk/IFrameHelper';
import { getBubbleView } from '../sdk/bubbleHelpers'; import { getBubbleView } from '../sdk/settingsHelper';
import md5 from 'md5'; import {
import { getUserCookieName } from '../sdk/cookieHelpers'; computeHashForUserData,
getUserCookieName,
const REQUIRED_USER_KEYS = ['avatar_url', 'email', 'name']; hasUserKeys,
} from '../sdk/cookieHelpers';
const ALLOWED_USER_ATTRIBUTES = [...REQUIRED_USER_KEYS, 'identifier_hash'];
export const getUserString = ({ identifier = '', user }) => {
const userStringWithSortedKeys = ALLOWED_USER_ATTRIBUTES.reduce(
(acc, key) => `${acc}${key}${user[key] || ''}`,
''
);
return `${userStringWithSortedKeys}identifier${identifier}`;
};
const computeHashForUserData = (...args) => md5(getUserString(...args));
export const hasUserKeys = user =>
REQUIRED_USER_KEYS.reduce((acc, key) => acc || !!user[key], false);
const runSDK = ({ baseUrl, websiteToken }) => { const runSDK = ({ baseUrl, websiteToken }) => {
if (window.$chatwoot) { if (window.$chatwoot) {
@ -38,6 +24,7 @@ const runSDK = ({ baseUrl, websiteToken }) => {
type: getBubbleView(chatwootSettings.type), type: getBubbleView(chatwootSettings.type),
launcherTitle: chatwootSettings.launcherTitle || '', launcherTitle: chatwootSettings.launcherTitle || '',
showPopoutButton: chatwootSettings.showPopoutButton || false, showPopoutButton: chatwootSettings.showPopoutButton || false,
widgetStyle: chatwootSettings.widgetStyle || 'standard',
toggle(state) { toggle(state) {
IFrameHelper.events.toggleBubble(state); IFrameHelper.events.toggleBubble(state);

View file

@ -26,6 +26,7 @@ import { dispatchWindowEvent } from 'shared/helpers/CustomEventHelper';
import { CHATWOOT_ERROR, CHATWOOT_READY } from '../widget/constants/sdkEvents'; import { CHATWOOT_ERROR, CHATWOOT_READY } from '../widget/constants/sdkEvents';
import { SET_USER_ERROR } from '../widget/constants/errorTypes'; import { SET_USER_ERROR } from '../widget/constants/errorTypes';
import { getUserCookieName } from './cookieHelpers'; import { getUserCookieName } from './cookieHelpers';
import { isFlatWidgetStyle } from './settingsHelper';
export const IFrameHelper = { export const IFrameHelper = {
getUrl({ baseUrl, websiteToken }) { getUrl({ baseUrl, websiteToken }) {
@ -52,6 +53,10 @@ export const IFrameHelper = {
if (window.$chatwoot.hideMessageBubble) { if (window.$chatwoot.hideMessageBubble) {
holderClassName += ` woot-widget--without-bubble`; holderClassName += ` woot-widget--without-bubble`;
} }
if (isFlatWidgetStyle(window.$chatwoot.widgetStyle)) {
holderClassName += ` woot-widget-holder--flat`;
}
addClass(widgetHolder, holderClassName); addClass(widgetHolder, holderClassName);
widgetHolder.appendChild(iframe); widgetHolder.appendChild(iframe);
body.appendChild(widgetHolder); body.appendChild(widgetHolder);
@ -121,6 +126,7 @@ export const IFrameHelper = {
position: window.$chatwoot.position, position: window.$chatwoot.position,
hideMessageBubble: window.$chatwoot.hideMessageBubble, hideMessageBubble: window.$chatwoot.hideMessageBubble,
showPopoutButton: window.$chatwoot.showPopoutButton, showPopoutButton: window.$chatwoot.showPopoutButton,
widgetStyle: window.$chatwoot.widgetStyle,
}); });
IFrameHelper.onLoad({ IFrameHelper.onLoad({
widgetColor: message.config.channelConfig.widgetColor, widgetColor: message.config.channelConfig.widgetColor,
@ -222,21 +228,27 @@ export const IFrameHelper = {
createBubbleHolder(); createBubbleHolder();
onLocationChangeListener(); onLocationChangeListener();
if (!window.$chatwoot.hideMessageBubble) { if (!window.$chatwoot.hideMessageBubble) {
let className = 'woot-widget-bubble';
let closeBtnClassName = `woot-elements--${window.$chatwoot.position} woot-widget-bubble woot--close woot--hide`;
if (isFlatWidgetStyle(window.$chatwoot.widgetStyle)) {
className += ' woot-widget-bubble--flat';
closeBtnClassName += ' woot-widget-bubble--flat';
}
const chatIcon = createBubbleIcon({ const chatIcon = createBubbleIcon({
className: 'woot-widget-bubble', className,
src: bubbleImg, src: bubbleImg,
target: chatBubble, target: chatBubble,
}); });
const closeIcon = closeBubble; addClass(closeBubble, closeBtnClassName);
const closeIconclassName = `woot-elements--${window.$chatwoot.position} woot-widget-bubble woot--close woot--hide`;
addClass(closeIcon, closeIconclassName);
chatIcon.style.background = widgetColor; chatIcon.style.background = widgetColor;
closeIcon.style.background = widgetColor; closeBubble.style.background = widgetColor;
bubbleHolder.appendChild(chatIcon); bubbleHolder.appendChild(chatIcon);
bubbleHolder.appendChild(closeIcon); bubbleHolder.appendChild(closeBubble);
bubbleHolder.appendChild(createNotificationBubble()); bubbleHolder.appendChild(createNotificationBubble());
onClickChatBubble(); onClickChatBubble();
} }

View file

@ -1,6 +1,6 @@
import { addClass, removeClass, toggleClass, wootOn } from './DOMHelpers'; import { addClass, removeClass, toggleClass, wootOn } from './DOMHelpers';
import { IFrameHelper } from './IFrameHelper'; import { IFrameHelper } from './IFrameHelper';
import { BUBBLE_DESIGN } from './constants'; import { isExpandedView } from './settingsHelper';
export const bubbleImg = export const bubbleImg =
''; '';
@ -13,10 +13,6 @@ export const chatBubble = document.createElement('button');
export const closeBubble = document.createElement('button'); export const closeBubble = document.createElement('button');
export const notificationBubble = document.createElement('span'); export const notificationBubble = document.createElement('span');
export const getBubbleView = type =>
BUBBLE_DESIGN.includes(type) ? type : BUBBLE_DESIGN[0];
export const isExpandedView = type => getBubbleView(type) === BUBBLE_DESIGN[1];
export const setBubbleText = bubbleText => { export const setBubbleText = bubbleText => {
if (isExpandedView(window.$chatwoot.type)) { if (isExpandedView(window.$chatwoot.type)) {
const textNode = document.getElementById('woot-widget--expanded__text'); const textNode = document.getElementById('woot-widget--expanded__text');

View file

@ -1 +1,2 @@
export const BUBBLE_DESIGN = ['standard', 'expanded_bubble']; export const BUBBLE_DESIGN = ['standard', 'expanded_bubble'];
export const WIDGET_DESIGN = ['standard', 'flat'];

View file

@ -1,5 +1,23 @@
import md5 from 'md5';
const REQUIRED_USER_KEYS = ['avatar_url', 'email', 'name'];
const ALLOWED_USER_ATTRIBUTES = [...REQUIRED_USER_KEYS, 'identifier_hash'];
export const getUserCookieName = () => { export const getUserCookieName = () => {
const SET_USER_COOKIE_PREFIX = 'cw_user_'; const SET_USER_COOKIE_PREFIX = 'cw_user_';
const { websiteToken: websiteIdentifier } = window.$chatwoot; const { websiteToken: websiteIdentifier } = window.$chatwoot;
return `${SET_USER_COOKIE_PREFIX}${websiteIdentifier}`; return `${SET_USER_COOKIE_PREFIX}${websiteIdentifier}`;
}; };
export const getUserString = ({ identifier = '', user }) => {
const userStringWithSortedKeys = ALLOWED_USER_ATTRIBUTES.reduce(
(acc, key) => `${acc}${key}${user[key] || ''}`,
''
);
return `${userStringWithSortedKeys}identifier${identifier}`;
};
export const computeHashForUserData = (...args) => md5(getUserString(...args));
export const hasUserKeys = user =>
REQUIRED_USER_KEYS.reduce((acc, key) => acc || !!user[key], false);

View file

@ -1,5 +1,10 @@
export const SDK_CSS = `.woot-widget-holder { export const SDK_CSS = `
box-shadow: 0 5px 40px rgba(0, 0, 0, .16) !important; :root {
--b-100: #F2F3F7;
}
.woot-widget-holder {
box-shadow: 0 5px 40px rgba(0, 0, 0, .16);
opacity: 1; opacity: 1;
will-change: transform, opacity; will-change: transform, opacity;
transform: translateY(0); transform: translateY(0);
@ -9,6 +14,12 @@ export const SDK_CSS = `.woot-widget-holder {
z-index: 2147483000 !important; z-index: 2147483000 !important;
} }
.woot-widget-holder.woot-widget-holder--flat {
box-shadow: none;
border-radius: 0;
border: 1px solid var(--b-100);
}
.woot-widget-holder iframe { .woot-widget-holder iframe {
border: 0; border: 0;
height: 100% !important; height: 100% !important;
@ -22,21 +33,45 @@ export const SDK_CSS = `.woot-widget-holder {
height: auto; height: auto;
bottom: 94px; bottom: 94px;
box-shadow: none !important; box-shadow: none !important;
border: 0;
} }
.woot-widget-bubble { .woot-widget-bubble {
background: #1f93ff; background: #1f93ff;
border-radius: 100px !important; border-radius: 100px;
border-width: 0px; border-width: 0px;
bottom: 20px; bottom: 20px;
padding: 0px;
box-shadow: 0 8px 24px rgba(0, 0, 0, .16) !important; box-shadow: 0 8px 24px rgba(0, 0, 0, .16) !important;
cursor: pointer; cursor: pointer;
height: 64px !important; height: 64px;
padding: 0px;
position: fixed; position: fixed;
width: 64px !important;
z-index: 2147483000 !important;
user-select: none; user-select: none;
width: 64px;
z-index: 2147483000 !important;
}
.woot-widget-bubble.woot-widget-bubble--flat {
border-radius: 0;
}
.woot-widget-holder.woot-widget-holder--flat {
bottom: 90px;
}
.woot-widget-bubble.woot-widget-bubble--flat {
height: 56px;
width: 56px;
}
.woot-widget-bubble.woot-widget-bubble--flat img {
margin: 16px;
}
.woot-widget-bubble.woot-widget-bubble--flat.woot--close::before,
.woot-widget-bubble.woot-widget-bubble--flat.woot--close::after {
left: 28px;
top: 16px;
} }
.woot-widget-bubble.unread-notification::after { .woot-widget-bubble.unread-notification::after {
@ -184,7 +219,7 @@ export const SDK_CSS = `.woot-widget-holder {
@media only screen and (min-width: 667px) { @media only screen and (min-width: 667px) {
.woot-widget-holder { .woot-widget-holder {
border-radius: 16px !important; border-radius: 16px;
bottom: 104px; bottom: 104px;
height: calc(85% - 64px - 20px); height: calc(85% - 64px - 20px);
max-height: 590px !important; max-height: 590px !important;

View file

@ -0,0 +1,11 @@
import { BUBBLE_DESIGN, WIDGET_DESIGN } from './constants';
export const getBubbleView = type =>
BUBBLE_DESIGN.includes(type) ? type : BUBBLE_DESIGN[0];
export const isExpandedView = type => getBubbleView(type) === BUBBLE_DESIGN[1];
export const getWidgetStyle = style =>
WIDGET_DESIGN.includes(style) ? style : WIDGET_DESIGN[0];
export const isFlatWidgetStyle = style => style === 'flat';

View file

@ -1,17 +0,0 @@
import { getBubbleView, isExpandedView } from '../bubbleHelpers';
describe('#getBubbleView', () => {
it('returns correct view', () => {
expect(getBubbleView('')).toEqual('standard');
expect(getBubbleView('standard')).toEqual('standard');
expect(getBubbleView('expanded_bubble')).toEqual('expanded_bubble');
});
});
describe('#isExpandedView', () => {
it('returns true if it is expanded view', () => {
expect(isExpandedView('')).toEqual(false);
expect(isExpandedView('standard')).toEqual(false);
expect(isExpandedView('expanded_bubble')).toEqual(true);
});
});

View file

@ -1,4 +1,8 @@
import { getUserCookieName } from '../cookieHelpers'; import {
getUserCookieName,
getUserString,
hasUserKeys,
} from '../cookieHelpers';
describe('#getUserCookieName', () => { describe('#getUserCookieName', () => {
it('returns correct cookie name', () => { it('returns correct cookie name', () => {
@ -6,3 +10,40 @@ describe('#getUserCookieName', () => {
expect(getUserCookieName()).toBe('cw_user_123456'); expect(getUserCookieName()).toBe('cw_user_123456');
}); });
}); });
describe('#getUserString', () => {
it('returns correct user string', () => {
expect(
getUserString({
user: {
name: 'Pranav',
email: 'pranav@example.com',
avatar_url: 'https://images.chatwoot.com/placeholder',
identifier_hash: '12345',
},
identifier: '12345',
})
).toBe(
'avatar_urlhttps://images.chatwoot.com/placeholderemailpranav@example.comnamePranavidentifier_hash12345identifier12345'
);
expect(
getUserString({
user: {
email: 'pranav@example.com',
avatar_url: 'https://images.chatwoot.com/placeholder',
},
})
).toBe(
'avatar_urlhttps://images.chatwoot.com/placeholderemailpranav@example.comnameidentifier_hashidentifier'
);
});
});
describe('#hasUserKeys', () => {
it('checks whether the allowed list of keys are present', () => {
expect(hasUserKeys({})).toBe(false);
expect(hasUserKeys({ randomKey: 'randomValue' })).toBe(false);
expect(hasUserKeys({ avatar_url: 'randomValue' })).toBe(true);
});
});

View file

@ -0,0 +1,38 @@
import {
getBubbleView,
getWidgetStyle,
isExpandedView,
isFlatWidgetStyle,
} from '../settingsHelper';
describe('#getBubbleView', () => {
it('returns correct view', () => {
expect(getBubbleView('')).toEqual('standard');
expect(getBubbleView('standard')).toEqual('standard');
expect(getBubbleView('expanded_bubble')).toEqual('expanded_bubble');
});
});
describe('#isExpandedView', () => {
it('returns true if it is expanded view', () => {
expect(isExpandedView('')).toEqual(false);
expect(isExpandedView('standard')).toEqual(false);
expect(isExpandedView('expanded_bubble')).toEqual(true);
});
});
describe('#getWidgetStyle', () => {
it('returns correct view', () => {
expect(getWidgetStyle('')).toEqual('standard');
expect(getWidgetStyle('standard')).toEqual('standard');
expect(getWidgetStyle('flat')).toEqual('flat');
});
});
describe('#isFlatWidgetStyle', () => {
it('returns true if it is expanded view', () => {
expect(isFlatWidgetStyle('')).toEqual(false);
expect(isFlatWidgetStyle('standard')).toEqual(false);
expect(isFlatWidgetStyle('flat')).toEqual(true);
});
});

View file

@ -1,38 +0,0 @@
import { getUserString, hasUserKeys } from '../../packs/sdk';
describe('#getUserString', () => {
it('returns correct user string', () => {
expect(
getUserString({
user: {
name: 'Pranav',
email: 'pranav@example.com',
avatar_url: 'https://images.chatwoot.com/placeholder',
identifier_hash: '12345',
},
identifier: '12345',
})
).toBe(
'avatar_urlhttps://images.chatwoot.com/placeholderemailpranav@example.comnamePranavidentifier_hash12345identifier12345'
);
expect(
getUserString({
user: {
email: 'pranav@example.com',
avatar_url: 'https://images.chatwoot.com/placeholder',
},
})
).toBe(
'avatar_urlhttps://images.chatwoot.com/placeholderemailpranav@example.comnameidentifier_hashidentifier'
);
});
});
describe('#hasUserKeys', () => {
it('checks whether the allowed list of keys are present', () => {
expect(hasUserKeys({})).toBe(false);
expect(hasUserKeys({ randomKey: 'randomValue' })).toBe(false);
expect(hasUserKeys({ avatar_url: 'randomValue' })).toBe(true);
});
});

View file

@ -12,6 +12,7 @@
'is-mobile': isMobile, 'is-mobile': isMobile,
'is-widget-right': isRightAligned, 'is-widget-right': isRightAligned,
'is-bubble-hidden': hideMessageBubble, 'is-bubble-hidden': hideMessageBubble,
'is-flat-design': isWidgetStyleFlat,
}" }"
> >
<router-view></router-view> <router-view></router-view>
@ -61,6 +62,7 @@ export default {
isWidgetOpen: 'appConfig/getIsWidgetOpen', isWidgetOpen: 'appConfig/getIsWidgetOpen',
messageCount: 'conversation/getMessageCount', messageCount: 'conversation/getMessageCount',
unreadMessageCount: 'conversation/getUnreadMessageCount', unreadMessageCount: 'conversation/getUnreadMessageCount',
isWidgetStyleFlat: 'appConfig/isWidgetStyleFlat',
}), }),
isIFrame() { isIFrame() {
return IFrameHelper.isIFrame(); return IFrameHelper.isIFrame();

View file

@ -57,3 +57,30 @@ body {
padding-left: $space-normal; padding-left: $space-normal;
} }
} }
.is-flat-design {
.chat-bubble {
border-bottom-left-radius: 0 !important;
border-bottom-right-radius: 0 !important;
border-top-left-radius: 0 !important;
border-top-right-radius: 0 !important;
box-shadow: none;
}
button {
border-radius: 0 !important;
}
input {
border-radius: 0;
}
.chat-message--input {
border-radius: 0 !important;
box-shadow: none !important;
&.is-focused {
box-shadow: none !important;
}
}
}

View file

@ -1,7 +1,8 @@
<template> <template>
<footer <footer
v-if="!hideReplyBox" v-if="!hideReplyBox"
class="shadow-sm rounded-lg bg-white mb-1 z-50 relative" class="shadow-sm bg-white mb-1 z-50 relative"
:class="{ 'rounded-lg': !isWidgetStyleFlat }"
> >
<chat-input-wrap <chat-input-wrap
:on-send-message="handleSendMessage" :on-send-message="handleSendMessage"
@ -54,6 +55,7 @@ export default {
widgetColor: 'appConfig/getWidgetColor', widgetColor: 'appConfig/getWidgetColor',
getConversationSize: 'conversation/getConversationSize', getConversationSize: 'conversation/getConversationSize',
currentUser: 'contacts/getCurrentUser', currentUser: 'contacts/getCurrentUser',
isWidgetStyleFlat: 'appConfig/isWidgetStyleFlat',
}), }),
textColor() { textColor() {
return getContrastingTextColor(this.widgetColor); return getContrastingTextColor(this.widgetColor);

View file

@ -97,7 +97,6 @@ export default {
@import '~widget/assets/scss/mixins'; @import '~widget/assets/scss/mixins';
.header-wrap { .header-wrap {
border-radius: $space-normal $space-normal 0 0;
flex-shrink: 0; flex-shrink: 0;
transition: max-height 300ms; transition: max-height 300ms;
z-index: 99; z-index: 99;

View file

@ -6,14 +6,15 @@ import {
} from '../types'; } from '../types';
const state = { const state = {
showPopoutButton: false,
hideMessageBubble: false, hideMessageBubble: false,
position: 'right',
isWebWidgetTriggered: false,
isCampaignViewClicked: false, isCampaignViewClicked: false,
isWebWidgetTriggered: false,
isWidgetOpen: false, isWidgetOpen: false,
widgetColor: '', position: 'right',
referrerHost: '', referrerHost: '',
showPopoutButton: false,
widgetColor: '',
widgetStyle: 'standard',
}; };
export const getters = { export const getters = {
@ -23,14 +24,19 @@ export const getters = {
getIsWidgetOpen: $state => $state.isWidgetOpen, getIsWidgetOpen: $state => $state.isWidgetOpen,
getWidgetColor: $state => $state.widgetColor, getWidgetColor: $state => $state.widgetColor,
getReferrerHost: $state => $state.referrerHost, getReferrerHost: $state => $state.referrerHost,
isWidgetStyleFlat: $state => $state.widgetStyle === 'flat',
}; };
export const actions = { export const actions = {
setAppConfig({ commit }, { showPopoutButton, position, hideMessageBubble }) { setAppConfig(
{ commit },
{ showPopoutButton, position, hideMessageBubble, widgetStyle = 'rounded' }
) {
commit(SET_WIDGET_APP_CONFIG, { commit(SET_WIDGET_APP_CONFIG, {
showPopoutButton: !!showPopoutButton,
position: position || 'right',
hideMessageBubble: !!hideMessageBubble, hideMessageBubble: !!hideMessageBubble,
position: position || 'right',
showPopoutButton: !!showPopoutButton,
widgetStyle,
}); });
}, },
toggleWidgetOpen({ commit }, isWidgetOpen) { toggleWidgetOpen({ commit }, isWidgetOpen) {
@ -49,6 +55,7 @@ export const mutations = {
$state.showPopoutButton = data.showPopoutButton; $state.showPopoutButton = data.showPopoutButton;
$state.position = data.position; $state.position = data.position;
$state.hideMessageBubble = data.hideMessageBubble; $state.hideMessageBubble = data.hideMessageBubble;
$state.widgetStyle = data.widgetStyle;
}, },
[TOGGLE_WIDGET_OPEN]($state, isWidgetOpen) { [TOGGLE_WIDGET_OPEN]($state, isWidgetOpen) {
$state.isWidgetOpen = isWidgetOpen; $state.isWidgetOpen = isWidgetOpen;

View file

@ -14,10 +14,11 @@
window.chatwootSettings = { window.chatwootSettings = {
hideMessageBubble: false, hideMessageBubble: false,
position: 'left', position: '<%= @widget_position %>',
locale: 'en', locale: 'en',
type: 'expanded_bubble', type: '<%= @widget_type %>',
showPopoutButton: true, showPopoutButton: true,
widgetStyle: '<%= @widget_style %>',
}; };
(function(d,t) { (function(d,t) {