metro4
Version:
The front-end framework for Build responsive, mobile-first projects on the web with the first front-end component library in Metro Style
287 lines (235 loc) • 11.1 kB
JavaScript
/* global Metro, Datetime, datetime, Cake, METRO_LOCALE */
(function(Metro, $) {
'use strict';
var Utils = Metro.utils;
var defaultAvatar = "";
var ChatDefaultConfig = {
chatDeferred: 0,
inputTimeFormat: null,
timeFormat: "D MMM hh:mm A",
name: "John Doe",
avatar: defaultAvatar,
welcome: null,
welcomeAvatar: defaultAvatar,
title: null,
width: "100%",
height: "auto",
messages: null,
sendButtonTitle: "Send",
readonly: false,
locale: METRO_LOCALE,
clsChat: "",
clsName: "",
clsTime: "",
clsInput: "",
clsSendButton: "",
clsMessageLeft: "default",
clsMessageRight: "default",
onMessage: Metro.noop,
onSend: Metro.noop,
onSendButtonClick: Metro.noop,
onChatCreate: Metro.noop
};
Metro.chatSetup = function (options) {
ChatDefaultConfig = $.extend({}, ChatDefaultConfig, options);
};
if (typeof window["metroChatSetup"] !== undefined) {
Metro.chatSetup(window["metroChatSetup"]);
}
Metro.Component('chat', {
init: function( options, elem ) {
this._super(elem, options, ChatDefaultConfig, {
input: null,
classes: "primary secondary success alert warning yellow info dark light".split(" "),
lastMessage: null
});
return this;
},
_create: function(){
var element = this.element;
this._createStructure();
this._createEvents();
this._fireEvent("chat-create", {
element: element
});
},
_createStructure: function(){
var that = this, element = this.element, o = this.options;
var messages, messageInput, input;
var customButtons = [
{
html: o.sendButtonTitle,
cls: o.clsSendButton+" js-chat-send-button",
onclick: o.onSendButtonClick
}
];
element.addClass("chat").addClass(o.clsChat);
element.css({
width: o.width,
height: o.height
});
if (Utils.isValue(o.title)) {
$("<div>").addClass("title").html(o.title).appendTo(element);
}
messages = $("<div>").addClass("messages");
messages.appendTo(element);
messageInput = $("<div>").addClass("message-input").appendTo(element);
input = $("<input type='text'>");
input.appendTo(messageInput);
setTimeout(() => {
Metro.makePlugin(input[0], 'input', {
customButtons: customButtons,
clsInput: o.clsInput
})
})
if (o.welcome) {
this.add({
text: o.welcome,
time: datetime(),
position: "left",
name: "Chat Bot",
avatar: o.welcomeAvatar
})
}
if (Utils.isValue(o.messages) && typeof o.messages === "string") {
o.messages = Utils.isObject(o.messages);
}
if (!Utils.isNull(o.messages) && typeof o.messages === "object" && Utils.objectLength(o.messages) > 0) {
$.each(o.messages, function(){
that.add(this);
});
}
element.find(".message-input")[o.readonly ? 'addClass':'removeClass']("disabled");
},
_createEvents: function(){
var that = this, element = this.element, o = this.options;
var sendButton = element.find(".js-chat-send-button");
var input = element.find("input[type=text]");
var send = function(){
var msg = ""+input.val(), m;
if (msg.trim() === "") {return false;}
m = {
id: Utils.elementId("chat-message"),
name: o.name,
avatar: o.avatar,
text: msg,
position: "right",
time: datetime()
};
that.add(m);
input.val("");
that._fireEvent("send", {
msg: m
});
input.focus();
};
sendButton.on(Metro.events.click, function () {
send();
});
input.on(Metro.events.keyup, function(e){
if (e.keyCode === Metro.keyCode.ENTER) {
send();
}
})
},
add: function(msg){
var that = this, element = this.element, o = this.options;
var message, sender, time, item, avatar, text;
var messages = element.find(".messages");
var messageDate;
messageDate = o.inputTimeFormat ? Datetime.from(msg.time, o.inputTimeFormat, o.locale) : datetime(msg.time);
message = $("<div>").addClass("message").addClass(msg.position).appendTo(messages);
item = $("<div>").addClass("message-item").appendTo(message);
avatar = $("<img>").attr("src", msg.avatar).addClass("message-avatar").appendTo(item);
text = $("<div>").addClass("message-text").append($("<div>").addClass("message-text-inner").html(Str.escapeHtml(msg.text))).appendTo(item);
time = $("<div>").addClass("message-time").addClass(o.clsTime).text(messageDate.format(o.timeFormat)).appendTo(text);
sender = $("<div>").addClass("message-sender").addClass(o.clsName).text(msg.name).appendTo(text);
if (Utils.isValue(msg.id)) {
message.attr("id", msg.id);
}
if (msg.position === 'left' && Utils.isValue(o.clsMessageLeft)) {
text.addClass(o.clsMessageLeft);
}
if (msg.position === 'right' && Utils.isValue(o.clsMessageRight)) {
text.addClass(o.clsMessageRight);
}
if (this.lastMessage && this.lastMessage.position === msg.position) {
text.addClass("--next");
avatar.visible(false);
sender.hide();
}
that._fireEvent("message", {
msg: msg,
el: {
message: message,
sender: sender,
time: time,
item: item,
avatar: avatar,
text: text
}
});
messages.animate({
draw: {
scrollTop: messages[0].scrollHeight
},
dur: 1000
});
this.lastMessage = msg;
return this;
},
addMessages: function(messages){
var that = this;
if (Utils.isValue(messages) && typeof messages === "string") {
messages = Utils.isObject(messages);
}
if (typeof messages === "object" && Utils.objectLength(messages) > 0) {
$.each(messages, function(){
that.add(this);
});
}
return this;
},
delMessage: function(id){
var element = this.element;
element.find(".messages").find("#"+id).remove();
return this;
},
updMessage: function(msg){
var element = this.element;
var message = element.find(".messages").find("#"+msg.id);
let o = this.options;
if (message.length === 0) return this;
let messageDate = o.inputTimeFormat ? Datetime.from(msg.time, o.inputTimeFormat, o.locale) : datetime(msg.time);
message.find(".message-text-inner").html(msg.text);
message.find(".message-time").html(messageDate.format(o.timeFormat));
return this;
},
clear: function(){
var element = this.element;
var messages = element.find(".messages");
messages.html("");
this.lastMessage = null;
},
toggleReadonly: function(readonly){
var element = this.element, o = this.options;
o.readonly = typeof readonly === "undefined" ? !o.readonly : readonly;
element.find(".message-input")[o.readonly ? 'addClass':'removeClass']("disabled");
},
changeAttribute: function(attributeName){
switch (attributeName) {
case "data-readonly": this.toggleReadonly(); break;
}
},
destroy: function(){
var element = this.element;
var sendButton = element.find(".js-chat-send-button");
var input = element.find("input[type=text]");
sendButton.off(Metro.events.click);
input.off(Metro.events.keyup);
return element;
}
});
Metro.defaults.Chat = ChatDefaultConfig;
Metro.defaults.ChatAvatar = defaultAvatar;
}(Metro, m4q));