[Feature] Website live chat (#187)
Co-authored-by: Nithin David Thomas <webofnithin@gmail.com> Co-authored-by: Sojan Jose <sojan@pepalo.com>
This commit is contained in:
parent
a4114288f3
commit
16fe912fbd
80 changed files with 2040 additions and 106 deletions
160
app/javascript/packs/sdk.js
Executable file
160
app/javascript/packs/sdk.js
Executable file
|
@ -0,0 +1,160 @@
|
|||
import sdkStyles from '../widget/assets/scss/sdk.css';
|
||||
/* eslint-disable no-param-reassign */
|
||||
const bubbleImg =
|
||||
'';
|
||||
const closeImg =
|
||||
'';
|
||||
|
||||
const body = document.getElementsByTagName('body')[0];
|
||||
const iframe = document.createElement('iframe');
|
||||
const holder = document.createElement('div');
|
||||
|
||||
const bubbleHolder = document.createElement('div');
|
||||
const chatBubble = document.createElement('div');
|
||||
const closeBubble = document.createElement('div');
|
||||
|
||||
const notification_bubble = document.createElement('span');
|
||||
const bodyOverFlowStyle = document.body.style.overflow;
|
||||
|
||||
function addClass(elm, classes) {
|
||||
if (classes) {
|
||||
elm.className += ` ${classes}`;
|
||||
}
|
||||
}
|
||||
|
||||
function loadCSS() {
|
||||
const css = document.createElement('style');
|
||||
css.type = 'text/css';
|
||||
css.innerHTML = sdkStyles;
|
||||
document.body.appendChild(css);
|
||||
}
|
||||
|
||||
function wootOn(elm, event, fn) {
|
||||
if (document.addEventListener) {
|
||||
elm.addEventListener(event, fn, false);
|
||||
} else if (document.attachEvent) {
|
||||
// <= IE 8 loses scope so need to apply, we add this to object so we
|
||||
// can detach later (can't detach anonymous functions)
|
||||
// eslint-disable-next-line
|
||||
elm[event + fn] = function() {
|
||||
// eslint-disable-next-line
|
||||
return fn.apply(elm, arguments);
|
||||
};
|
||||
elm.attachEvent(`on${event}`, elm[event + fn]);
|
||||
}
|
||||
}
|
||||
|
||||
function classHelper(classes, action, elm) {
|
||||
let classarray;
|
||||
let search;
|
||||
let replace;
|
||||
let i;
|
||||
let has = false;
|
||||
if (classes) {
|
||||
// Trim any whitespace
|
||||
classarray = classes.split(/\s+/);
|
||||
for (i = 0; i < classarray.length; i += 1) {
|
||||
search = new RegExp(`\\b${classarray[i]}\\b`, 'g');
|
||||
replace = new RegExp(` *${classarray[i]}\\b`, 'g');
|
||||
if (action === 'remove') {
|
||||
// eslint-disable-next-line
|
||||
elm.className = elm.className.replace(replace, '');
|
||||
} else if (action === 'toggle') {
|
||||
// eslint-disable-next-line
|
||||
elm.className = elm.className.match(search)
|
||||
? elm.className.replace(replace, '')
|
||||
: `${elm.className} ${classarray[i]}`;
|
||||
} else if (action === 'has') {
|
||||
if (elm.className.match(search)) {
|
||||
has = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return has;
|
||||
}
|
||||
|
||||
// Toggle class
|
||||
function toggleClass(elm, classes) {
|
||||
classHelper(classes, 'toggle', elm);
|
||||
}
|
||||
|
||||
const createBubbleIcon = ({ className, src, target }) => {
|
||||
target.className = className;
|
||||
const bubbleIcon = document.createElement('img');
|
||||
bubbleIcon.src = src;
|
||||
target.appendChild(bubbleIcon);
|
||||
return target;
|
||||
};
|
||||
|
||||
function createBubbleHolder() {
|
||||
addClass(bubbleHolder, 'woot--bubble-holder');
|
||||
body.appendChild(bubbleHolder);
|
||||
}
|
||||
|
||||
function createNotificationBubble() {
|
||||
addClass(notification_bubble, 'woot--notification');
|
||||
return notification_bubble;
|
||||
}
|
||||
|
||||
function bubbleClickCallback() {
|
||||
toggleClass(chatBubble, 'woot--hide');
|
||||
toggleClass(closeBubble, 'woot--hide');
|
||||
toggleClass(holder, 'woot--hide');
|
||||
}
|
||||
|
||||
function onClickChatBubble() {
|
||||
wootOn(chatBubble, 'click', bubbleClickCallback);
|
||||
wootOn(closeBubble, 'click', bubbleClickCallback);
|
||||
}
|
||||
|
||||
function disableScroll() {
|
||||
document.body.style.overflow = 'hidden';
|
||||
}
|
||||
|
||||
function enableScroll() {
|
||||
document.body.style.overflow = bodyOverFlowStyle;
|
||||
}
|
||||
|
||||
function loadCallback() {
|
||||
iframe.style.visibility = '';
|
||||
iframe.setAttribute('id', `chatwoot_live_chat_widget`);
|
||||
iframe.onmouseenter = disableScroll;
|
||||
iframe.onmouseleave = enableScroll;
|
||||
|
||||
loadCSS();
|
||||
createBubbleHolder();
|
||||
|
||||
bubbleHolder.appendChild(
|
||||
createBubbleIcon({
|
||||
className: 'woot-widget-bubble',
|
||||
src: bubbleImg,
|
||||
target: chatBubble,
|
||||
})
|
||||
);
|
||||
bubbleHolder.appendChild(
|
||||
createBubbleIcon({
|
||||
className: 'woot-widget-bubble woot--close woot--hide',
|
||||
src: closeImg,
|
||||
target: closeBubble,
|
||||
})
|
||||
);
|
||||
bubbleHolder.appendChild(createNotificationBubble());
|
||||
onClickChatBubble();
|
||||
}
|
||||
|
||||
function loadIframe({ websiteToken, baseUrl }) {
|
||||
iframe.style.visibility = 'hidden';
|
||||
iframe.src = `${baseUrl}/widgets?website_token=${websiteToken}`;
|
||||
iframe.onload = loadCallback;
|
||||
|
||||
holder.className = 'woot-widget-holder woot--hide';
|
||||
holder.appendChild(iframe);
|
||||
|
||||
body.appendChild(holder);
|
||||
}
|
||||
|
||||
window.chatwootSDK = {
|
||||
run: loadIframe,
|
||||
};
|
Loading…
Add table
Add a link
Reference in a new issue