[Enhancement] Use JS-Cookie to set Cookies (#193)
* Update js-cookie * Add sdk css * Remove conversation padding
This commit is contained in:
parent
e32b6bf6d4
commit
3d3aefb197
14 changed files with 170 additions and 68 deletions
1
Gemfile
1
Gemfile
|
@ -36,6 +36,7 @@ gem 'webpacker'
|
||||||
gem 'devise', git: 'https://github.com/plataformatec/devise'
|
gem 'devise', git: 'https://github.com/plataformatec/devise'
|
||||||
gem 'devise_token_auth', git: 'https://github.com/lynndylanhurley/devise_token_auth'
|
gem 'devise_token_auth', git: 'https://github.com/lynndylanhurley/devise_token_auth'
|
||||||
# authorization
|
# authorization
|
||||||
|
gem 'jwt'
|
||||||
gem 'pundit'
|
gem 'pundit'
|
||||||
|
|
||||||
##--- gems for pubsub service ---##
|
##--- gems for pubsub service ---##
|
||||||
|
|
|
@ -220,6 +220,7 @@ GEM
|
||||||
jmespath (1.4.0)
|
jmespath (1.4.0)
|
||||||
json (2.2.0)
|
json (2.2.0)
|
||||||
json_pure (2.2.0)
|
json_pure (2.2.0)
|
||||||
|
jwt (2.2.1)
|
||||||
kaminari (1.1.1)
|
kaminari (1.1.1)
|
||||||
activesupport (>= 4.1.0)
|
activesupport (>= 4.1.0)
|
||||||
kaminari-actionview (= 1.1.1)
|
kaminari-actionview (= 1.1.1)
|
||||||
|
@ -459,6 +460,7 @@ DEPENDENCIES
|
||||||
haikunator
|
haikunator
|
||||||
hashie
|
hashie
|
||||||
jbuilder (~> 2.5)
|
jbuilder (~> 2.5)
|
||||||
|
jwt
|
||||||
kaminari
|
kaminari
|
||||||
koala
|
koala
|
||||||
letter_opener
|
letter_opener
|
||||||
|
|
|
@ -45,7 +45,9 @@ class Api::V1::Widget::MessagesController < ActionController::Base
|
||||||
end
|
end
|
||||||
|
|
||||||
def cookie_params
|
def cookie_params
|
||||||
JSON.parse(cookies.signed[cookie_name]).symbolize_keys
|
@cookie_params ||= JWT.decode(
|
||||||
|
request.headers[header_name], secret_key, true, algorithm: 'HS256'
|
||||||
|
).first.symbolize_keys
|
||||||
end
|
end
|
||||||
|
|
||||||
def message_finder_params
|
def message_finder_params
|
||||||
|
@ -58,11 +60,15 @@ class Api::V1::Widget::MessagesController < ActionController::Base
|
||||||
@message_finder ||= MessageFinder.new(conversation, message_finder_params)
|
@message_finder ||= MessageFinder.new(conversation, message_finder_params)
|
||||||
end
|
end
|
||||||
|
|
||||||
def cookie_name
|
def header_name
|
||||||
'cw_conversation_' + params[:website_token]
|
'X-Auth-Token'
|
||||||
end
|
end
|
||||||
|
|
||||||
def permitted_params
|
def permitted_params
|
||||||
params.fetch(:message).permit(:content)
|
params.fetch(:message).permit(:content)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def secret_key
|
||||||
|
Rails.application.secrets.secret_key_base
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,14 +1,11 @@
|
||||||
class WidgetsController < ActionController::Base
|
class WidgetsController < ActionController::Base
|
||||||
before_action :set_web_widget
|
before_action :set_web_widget
|
||||||
|
before_action :set_token
|
||||||
before_action :set_contact
|
before_action :set_contact
|
||||||
before_action :build_contact
|
before_action :build_contact
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def set_web_widget
|
|
||||||
@web_widget = ::Channel::WebWidget.find_by!(website_token: permitted_params[:website_token])
|
|
||||||
end
|
|
||||||
|
|
||||||
def set_contact
|
def set_contact
|
||||||
return if cookie_params[:source_id].nil?
|
return if cookie_params[:source_id].nil?
|
||||||
|
|
||||||
|
@ -20,28 +17,49 @@ class WidgetsController < ActionController::Base
|
||||||
@contact = contact_inbox.contact
|
@contact = contact_inbox.contact
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def set_token
|
||||||
|
@token = conversation_token
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_web_widget
|
||||||
|
@web_widget = ::Channel::WebWidget.find_by!(website_token: permitted_params[:website_token])
|
||||||
|
end
|
||||||
|
|
||||||
def build_contact
|
def build_contact
|
||||||
return if @contact.present?
|
return if @contact.present?
|
||||||
|
|
||||||
contact_inbox = @web_widget.create_contact_inbox
|
contact_inbox = @web_widget.create_contact_inbox
|
||||||
@contact = contact_inbox.contact
|
@contact = contact_inbox.contact
|
||||||
|
|
||||||
cookies.signed[cookie_name] = JSON.generate(
|
payload = {
|
||||||
source_id: contact_inbox.source_id,
|
source_id: contact_inbox.source_id,
|
||||||
contact_id: @contact.id,
|
contact_id: @contact.id,
|
||||||
inbox_id: @web_widget.inbox.id
|
inbox_id: @web_widget.inbox.id
|
||||||
).to_s
|
}
|
||||||
|
@token = JWT.encode payload, secret_key, 'HS256'
|
||||||
end
|
end
|
||||||
|
|
||||||
def cookie_params
|
def cookie_params
|
||||||
cookies.signed[cookie_name] ? JSON.parse(cookies.signed[cookie_name]).symbolize_keys : {}
|
return @cookie_params if @cookie_params.present?
|
||||||
|
|
||||||
|
if conversation_token.present?
|
||||||
|
@cookie_params = JWT.decode(
|
||||||
|
conversation_token, secret_key, true, algorithm: 'HS256'
|
||||||
|
).first.symbolize_keys
|
||||||
|
return @cookie_params
|
||||||
|
end
|
||||||
|
{}
|
||||||
|
end
|
||||||
|
|
||||||
|
def conversation_token
|
||||||
|
permitted_params[:cw_conversation]
|
||||||
end
|
end
|
||||||
|
|
||||||
def permitted_params
|
def permitted_params
|
||||||
params.permit(:website_token)
|
params.permit(:website_token, :cw_conversation)
|
||||||
end
|
end
|
||||||
|
|
||||||
def cookie_name
|
def secret_key
|
||||||
'cw_conversation_' + permitted_params[:website_token]
|
Rails.application.secrets.secret_key_base
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
import sdkStyles from '../widget/assets/scss/sdk.css';
|
import Cookies from 'js-cookie';
|
||||||
|
|
||||||
|
import { SDK_CSS } from '../widget/assets/scss/sdk';
|
||||||
/* eslint-disable no-param-reassign */
|
/* eslint-disable no-param-reassign */
|
||||||
const bubbleImg =
|
const bubbleImg =
|
||||||
'';
|
'';
|
||||||
|
@ -6,7 +8,6 @@ const closeImg =
|
||||||
'';
|
'';
|
||||||
|
|
||||||
const body = document.getElementsByTagName('body')[0];
|
const body = document.getElementsByTagName('body')[0];
|
||||||
const iframe = document.createElement('iframe');
|
|
||||||
const holder = document.createElement('div');
|
const holder = document.createElement('div');
|
||||||
|
|
||||||
const bubbleHolder = document.createElement('div');
|
const bubbleHolder = document.createElement('div');
|
||||||
|
@ -16,16 +17,10 @@ const closeBubble = document.createElement('div');
|
||||||
const notification_bubble = document.createElement('span');
|
const notification_bubble = document.createElement('span');
|
||||||
const bodyOverFlowStyle = document.body.style.overflow;
|
const bodyOverFlowStyle = document.body.style.overflow;
|
||||||
|
|
||||||
function addClass(elm, classes) {
|
|
||||||
if (classes) {
|
|
||||||
elm.className += ` ${classes}`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function loadCSS() {
|
function loadCSS() {
|
||||||
const css = document.createElement('style');
|
const css = document.createElement('style');
|
||||||
css.type = 'text/css';
|
css.type = 'text/css';
|
||||||
css.innerHTML = sdkStyles;
|
css.innerHTML = `${SDK_CSS}`;
|
||||||
document.body.appendChild(css);
|
document.body.appendChild(css);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -45,14 +40,13 @@ function wootOn(elm, event, fn) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function classHelper(classes, action, elm) {
|
function classHelper(classes, action, elm) {
|
||||||
let classarray;
|
|
||||||
let search;
|
let search;
|
||||||
let replace;
|
let replace;
|
||||||
let i;
|
let i;
|
||||||
let has = false;
|
let has = false;
|
||||||
if (classes) {
|
if (classes) {
|
||||||
// Trim any whitespace
|
// Trim any whitespace
|
||||||
classarray = classes.split(/\s+/);
|
const classarray = classes.split(/\s+/);
|
||||||
for (i = 0; i < classarray.length; i += 1) {
|
for (i = 0; i < classarray.length; i += 1) {
|
||||||
search = new RegExp(`\\b${classarray[i]}\\b`, 'g');
|
search = new RegExp(`\\b${classarray[i]}\\b`, 'g');
|
||||||
replace = new RegExp(` *${classarray[i]}\\b`, 'g');
|
replace = new RegExp(` *${classarray[i]}\\b`, 'g');
|
||||||
|
@ -75,6 +69,12 @@ function classHelper(classes, action, elm) {
|
||||||
return has;
|
return has;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function addClass(elm, classes) {
|
||||||
|
if (classes) {
|
||||||
|
elm.className += ` ${classes}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Toggle class
|
// Toggle class
|
||||||
function toggleClass(elm, classes) {
|
function toggleClass(elm, classes) {
|
||||||
classHelper(classes, 'toggle', elm);
|
classHelper(classes, 'toggle', elm);
|
||||||
|
@ -117,7 +117,49 @@ function enableScroll() {
|
||||||
document.body.style.overflow = bodyOverFlowStyle;
|
document.body.style.overflow = bodyOverFlowStyle;
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadCallback() {
|
const IFrameHelper = {
|
||||||
|
createFrame: ({ baseUrl, websiteToken }) => {
|
||||||
|
const iframe = document.createElement('iframe');
|
||||||
|
const cwCookie = Cookies.get('cw_conversation');
|
||||||
|
let widgetUrl = `${baseUrl}/widgets?website_token=${websiteToken}`;
|
||||||
|
if (cwCookie) {
|
||||||
|
widgetUrl = `${widgetUrl}&cw_conversation=${cwCookie}`;
|
||||||
|
}
|
||||||
|
iframe.src = widgetUrl;
|
||||||
|
|
||||||
|
iframe.id = 'chatwoot_web_widget';
|
||||||
|
iframe.style.visibility = 'hidden';
|
||||||
|
holder.className = 'woot-widget-holder woot--hide';
|
||||||
|
holder.appendChild(iframe);
|
||||||
|
body.appendChild(holder);
|
||||||
|
IFrameHelper.initPostMessageCommunication();
|
||||||
|
},
|
||||||
|
getAppFrame: () => document.getElementById('chatwoot_web_widget'),
|
||||||
|
sendMessage: (key, value) => {
|
||||||
|
const element = IFrameHelper.getAppFrame();
|
||||||
|
element.contentWindow.postMessage(
|
||||||
|
`chatwoot-widget:${JSON.stringify({ event: key, ...value })}`,
|
||||||
|
'*'
|
||||||
|
);
|
||||||
|
},
|
||||||
|
initPostMessageCommunication: () => {
|
||||||
|
window.onmessage = e => {
|
||||||
|
if (
|
||||||
|
typeof e.data !== 'string' ||
|
||||||
|
e.data.indexOf('chatwoot-widget:') !== 0
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const message = JSON.parse(e.data.replace('chatwoot-widget:', ''));
|
||||||
|
if (message.event === 'loaded') {
|
||||||
|
Cookies.set('cw_conversation', message.config.authToken);
|
||||||
|
IFrameHelper.sendMessage('config-set', {});
|
||||||
|
IFrameHelper.onLoad();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
},
|
||||||
|
onLoad: () => {
|
||||||
|
const iframe = IFrameHelper.getAppFrame();
|
||||||
iframe.style.visibility = '';
|
iframe.style.visibility = '';
|
||||||
iframe.setAttribute('id', `chatwoot_live_chat_widget`);
|
iframe.setAttribute('id', `chatwoot_live_chat_widget`);
|
||||||
iframe.onmouseenter = disableScroll;
|
iframe.onmouseenter = disableScroll;
|
||||||
|
@ -142,17 +184,14 @@ function loadCallback() {
|
||||||
);
|
);
|
||||||
bubbleHolder.appendChild(createNotificationBubble());
|
bubbleHolder.appendChild(createNotificationBubble());
|
||||||
onClickChatBubble();
|
onClickChatBubble();
|
||||||
}
|
},
|
||||||
|
};
|
||||||
|
|
||||||
function loadIframe({ websiteToken, baseUrl }) {
|
function loadIframe({ baseUrl, websiteToken }) {
|
||||||
iframe.style.visibility = 'hidden';
|
IFrameHelper.createFrame({
|
||||||
iframe.src = `${baseUrl}/widgets?website_token=${websiteToken}`;
|
baseUrl,
|
||||||
iframe.onload = loadCallback;
|
websiteToken,
|
||||||
|
});
|
||||||
holder.className = 'woot-widget-holder woot--hide';
|
|
||||||
holder.appendChild(iframe);
|
|
||||||
|
|
||||||
body.appendChild(holder);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
window.chatwootSDK = {
|
window.chatwootSDK = {
|
||||||
|
|
|
@ -6,6 +6,17 @@
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapActions } from 'vuex';
|
import { mapActions } from 'vuex';
|
||||||
|
import { setHeader } from './helpers/axios';
|
||||||
|
|
||||||
|
export const IFrameHelper = {
|
||||||
|
isIFrame: () => window.self !== window.top,
|
||||||
|
sendMessage: msg => {
|
||||||
|
window.parent.postMessage(
|
||||||
|
`chatwoot-widget:${JSON.stringify({ ...msg })}`,
|
||||||
|
'*'
|
||||||
|
);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'App',
|
name: 'App',
|
||||||
|
@ -15,7 +26,28 @@ export default {
|
||||||
},
|
},
|
||||||
|
|
||||||
mounted() {
|
mounted() {
|
||||||
|
if (IFrameHelper.isIFrame()) {
|
||||||
|
IFrameHelper.sendMessage({
|
||||||
|
event: 'loaded',
|
||||||
|
config: {
|
||||||
|
authToken: window.authToken,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
setHeader('X-Auth-Token', window.authToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener('message', e => {
|
||||||
|
if (
|
||||||
|
typeof e.data !== 'string' ||
|
||||||
|
e.data.indexOf('chatwoot-widget:') !== 0
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const message = JSON.parse(e.data.replace('chatwoot-widget:', ''));
|
||||||
|
if (message.event === 'config-set') {
|
||||||
this.fetchOldConversations();
|
this.fetchOldConversations();
|
||||||
|
}
|
||||||
|
});
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
export const SDK_CSS = `
|
||||||
.woot-widget-holder {
|
.woot-widget-holder {
|
||||||
z-index: 2147483000!important;
|
z-index: 2147483000!important;
|
||||||
position: fixed!important;
|
position: fixed!important;
|
||||||
|
@ -63,3 +64,4 @@
|
||||||
.woot--hide {
|
.woot--hide {
|
||||||
display: none !important;
|
display: none !important;
|
||||||
}
|
}
|
||||||
|
`;
|
|
@ -28,6 +28,5 @@ export default {
|
||||||
|
|
||||||
.conversation {
|
.conversation {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
padding: $space-large $space-small $space-large $space-normal;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -6,6 +6,7 @@ class ActionCableConnector extends BaseActionCableConnector {
|
||||||
this.events = {
|
this.events = {
|
||||||
'message.created': this.onMessageCreated,
|
'message.created': this.onMessageCreated,
|
||||||
};
|
};
|
||||||
|
console.log('joined', app, pubsubToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
onMessageCreated = data => {
|
onMessageCreated = data => {
|
||||||
|
|
|
@ -69,6 +69,7 @@ export default {
|
||||||
.conversation-wrap {
|
.conversation-wrap {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
|
padding: $space-large $space-small $space-large $space-normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
.footer-wrap {
|
.footer-wrap {
|
||||||
|
|
|
@ -42,7 +42,7 @@ class Message < ApplicationRecord
|
||||||
private
|
private
|
||||||
|
|
||||||
def dispatch_event
|
def dispatch_event
|
||||||
Rails.configuration.dispatcher.dispatch(MESSAGE_CREATED, Time.zone.now, message: self) unless conversation.messages.count == 1
|
Rails.configuration.dispatcher.dispatch(MESSAGE_CREATED, Time.zone.now, message: self)
|
||||||
|
|
||||||
if outgoing? && conversation.messages.outgoing.count == 1
|
if outgoing? && conversation.messages.outgoing.count == 1
|
||||||
Rails.configuration.dispatcher.dispatch(FIRST_REPLY_CREATED, Time.zone.now, message: self)
|
Rails.configuration.dispatcher.dispatch(FIRST_REPLY_CREATED, Time.zone.now, message: self)
|
||||||
|
|
|
@ -4,15 +4,16 @@
|
||||||
<title>Chatwoot</title>
|
<title>Chatwoot</title>
|
||||||
<%= csrf_meta_tags %>
|
<%= csrf_meta_tags %>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=0" />
|
||||||
|
<script>
|
||||||
|
window.chatwootWebChannel = '<%= @web_widget.website_name %>'
|
||||||
|
window.chatwootPubsubToken = '<%= @contact.pubsub_token %>'
|
||||||
|
window.authToken = '<%= @token %>'
|
||||||
|
</script>
|
||||||
<%= javascript_pack_tag 'widget' %>
|
<%= javascript_pack_tag 'widget' %>
|
||||||
<%= stylesheet_pack_tag 'widget' %>
|
<%= stylesheet_pack_tag 'widget' %>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="app"></div>
|
<div id="app"></div>
|
||||||
<%= yield %>
|
<%= yield %>
|
||||||
<script>
|
|
||||||
window.chatwootWebChannel = '<%= @web_widget.website_name %>'
|
|
||||||
window.chatwootPubsubToken = '<%= @contact.pubsub_token %>'
|
|
||||||
</script>
|
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
"foundation-sites": "6.3.0",
|
"foundation-sites": "6.3.0",
|
||||||
"highlight.js": "^9.15.10",
|
"highlight.js": "^9.15.10",
|
||||||
"ionicons": "~2.0.1",
|
"ionicons": "~2.0.1",
|
||||||
"js-cookie": "~2.1.3",
|
"js-cookie": "^2.2.1",
|
||||||
"md5": "~2.2.1",
|
"md5": "~2.2.1",
|
||||||
"moment": "~2.19.3",
|
"moment": "~2.19.3",
|
||||||
"query-string": "5",
|
"query-string": "5",
|
||||||
|
|
|
@ -5855,10 +5855,10 @@ js-beautify@^1.6.12, js-beautify@^1.6.14:
|
||||||
mkdirp "~0.5.1"
|
mkdirp "~0.5.1"
|
||||||
nopt "~4.0.1"
|
nopt "~4.0.1"
|
||||||
|
|
||||||
js-cookie@~2.1.3:
|
js-cookie@^2.2.1:
|
||||||
version "2.1.4"
|
version "2.2.1"
|
||||||
resolved "https://registry.yarnpkg.com/js-cookie/-/js-cookie-2.1.4.tgz#da4ec503866f149d164cf25f579ef31015025d8d"
|
resolved "https://registry.yarnpkg.com/js-cookie/-/js-cookie-2.2.1.tgz#69e106dc5d5806894562902aa5baec3744e9b2b8"
|
||||||
integrity sha1-2k7FA4ZvFJ0WTPJfV57zEBUCXY0=
|
integrity sha512-HvdH2LzI/EAZcUwA8+0nKNtWHqS+ZmijLA30RwZA0bo7ToCckjK5MkGhjED9KoRcXO6BaGI3I9UIzSA1FKFPOQ==
|
||||||
|
|
||||||
js-levenshtein@^1.1.3:
|
js-levenshtein@^1.1.3:
|
||||||
version "1.1.6"
|
version "1.1.6"
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue