define([ 'jquery', '/customize/messages.js', '/common/common-util.js', '/common/common-interface.js', '/common/common-ui-elements.js', '/common/hyperscript.js', '/common/diffMarked.js', ], function ($, Messages, Util, UI, UIElements, h, DiffMd) { 'use strict'; var debug = console.log; debug = function () {}; var MessengerUI = {}; var mutedUsers = {}; var dataQuery = function (id) { return '[data-key="' + id + '"]'; }; var userQuery = function (curve) { return '[data-user="' + curve + '"]'; }; var initChannel = function (state, info) { console.debug('initializing channel for [%s]', info.id); var h, t; if (Array.isArray(info.messages) && info.messages.length) { h = info.messages[info.messages.length -1].sig; t = info.messages[0].sig; } state.channels[info.id] = { messages: info.messages || [], name: info.name, isFriendChat: info.isFriendChat, isPadChat: info.isPadChat, isTeamChat: info.isTeamChat, needMoreHistory: !info.isPadChat && !info.isTeamChat, curvePublic: info.curvePublic, HEAD: h || info.lastKnownHash, TAIL: t || null, }; }; MessengerUI.create = function ($container, common, toolbar) { var metadataMgr = common.getMetadataMgr(); var origin = metadataMgr.getPrivateData().origin; var readOnly = metadataMgr.getPrivateData().readOnly || (toolbar && toolbar.readOnly); var isApp = typeof(toolbar) !== "undefined"; $container.addClass('cp-app-contacts-initializing'); var messaging = h('div#cp-app-contacts-messaging', [ h('span.fa.fa-spinner.fa-pulse.fa-4x.fa-fw.cp-app-contacts-spinner'), h('div.cp-app-contacts-info', [ h('h2', Messages.contacts_info1), h('ul', [ h('li', Messages.contacts_info2), h('li', Messages.contacts_info3), h('li', Messages.contacts_info4), ]) ]) ]); var friendList = h('div#cp-app-contacts-friendlist', [ h('span.fa.fa-spinner.fa-pulse.fa-4x.fa-fw.cp-app-contacts-spinner'), h('div.cp-app-contacts-padchat.cp-app-contacts-category', [ h('div.cp-app-contacts-category-content') ]), h('div.cp-app-contacts-friends.cp-app-contacts-category', [ h('button.cp-app-contacts-muted-button',[ h('i.fa.fa-bell-slash'), Messages.contacts_manageMuted || 'MANAGE MUTED' // XXX ]), h('div.cp-app-contacts-category-content.cp-contacts-friends') ]), h('div.cp-app-contacts-rooms.cp-app-contacts-category', [ h('div.cp-app-contacts-category-content'), h('h2.cp-app-contacts-category-title', Messages.contacts_rooms), ]), ]); var execCommand = function () { console.warn(arguments); }; /* var execCommand = function (cmd, data, cb) { sframeChan.query('Q_CHAT_COMMAND', {cmd: cmd, data: data}, function (err, obj) { if (err || (obj && obj.error)) { return void cb(err || (obj && obj.error)); } cb(void 0, obj); }); }; */ var $userlist = $(friendList).appendTo($container); var $messages = $(messaging).appendTo($container); var state = window.state = { active: '', channels: {} }; var contactsData = state.contactsData = {}; var avatars = state.avatars = {}; var setActive = function (id) { state.active = id; }; var isActive = function (id) { return id === state.active; }; var find = {}; find.inList = function (id) { return $userlist.find(dataQuery(id)); }; var notifyToolbar = function () { if (!toolbar || !toolbar['chat']) { return; } if (!toolbar['chat'].find('button').hasClass('cp-toolbar-button-active')) { toolbar['chat'].find('button').addClass('cp-toolbar-notification'); } if (!toolbar['chat'].hasClass('cp-leftside-active')) { toolbar['chat'].find('span.fa').addClass('cp-team-chat-notification'); } }; var notify = function (id) { find.inList(id).addClass('cp-app-contacts-notify'); }; var unnotify = function (id) { find.inList(id).removeClass('cp-app-contacts-notify'); }; var onResize = function () { // Don't update the width if we are in the contacts app if (!isApp) { return; } var w = $userlist[0].offsetWidth - $userlist[0].clientWidth; $userlist.css('width', (68 + w)+'px'); }; var reorderRooms = function () { var channels = Object.keys(state.channels).sort(function (a, b) { var m1 = state.channels[a].messages.slice(-1)[0]; var m2 = state.channels[b].messages.slice(-1)[0]; if (!m2) { return !m1 ? 0 : 1; } if (!m1) { return -1; } return m1.time - m2.time; }); channels.forEach(function (c, i) { $userlist.find(dataQuery(c)).css('order', i); }); // Make sure the width is correct even if there is a scrollbar onResize(); }; $(window).on('resize', onResize); var m = function (md, hour) { var id = Util.createRandomInteger(); var d = h('div', { id: 'msg-'+id }); try { var $d = $(d); DiffMd.apply(DiffMd.render(md || '', true, true), $d, common); $d.addClass("cp-app-contacts-content"); // override link clicking, because we're in an iframe $d.find('a').each(function () { var href = $(this).click(function (e) { e.preventDefault(); common.openUnsafeURL(href); }).attr('href'); }); var time = h('div.cp-app-contacts-time', hour); $d.append(time); } catch (e) { console.error(md); console.error(e); } return d; }; var markup = {}; markup.message = function (msg) { if (msg.type !== 'MSG') { return; } var curvePublic = msg.author; var name = (typeof msg.name !== "undefined" || !contactsData[msg.author]) ? (msg.name || Messages.anonymous) : contactsData[msg.author].displayName; var d = msg.time ? new Date(msg.time) : undefined; var day = d ? d.toLocaleDateString() : ''; var hour = d ? d.toLocaleTimeString() : ''; return h('div.cp-app-contacts-message', { //title: time || '?', 'data-user': curvePublic || name, 'data-day': day }, [ name? h('div.cp-app-contacts-sender', [ h('span.cp-app-contacts-sender-name', name), h('span.cp-app-contacts-sender-time', day) ]): undefined, m(msg.text, hour), ]); }; var getChat = function (id) { return $messages.find(dataQuery(id)); }; var scrollChatToBottom = function () { var $messagebox = $('.cp-app-contacts-messages'); $messagebox.scrollTop($messagebox[0].scrollHeight); }; var normalizeLabels = function ($messagebox) { $messagebox.find('div.cp-app-contacts-message').toArray().reduce(function (a, b) { var $b = $(b); if ($(a).data('user') === $b.data('user') && $(a).data('day') === $b.data('day')) { $b.find('.cp-app-contacts-sender').hide(); return a; } return b; }, []); }; var clearChannel = function (id) { $(getChat(id)).find('.cp-app-contacts-messages').html(''); }; markup.chatbox = function (id, data, curvePublic) { var moreHistory = h('span.cp-app-contacts-more-history.fa.fa-history', { title: Messages.contacts_fetchHistory, }); var chan = state.channels[id]; var displayName = chan.name; var fetching = false; var $moreHistory = $(moreHistory).click(function () { if (fetching) { return; } // get oldest known message... var channel = state.channels[id]; if (channel.exhausted) { return void $moreHistory.addClass('cp-app-contacts-faded'); } debug('getting history'); var sig = channel.TAIL || channel.HEAD; fetching = true; var $messagebox = $(getChat(id)).find('.cp-app-contacts-messages'); execCommand('GET_MORE_HISTORY', { id: id, sig: sig, count: 10 }, function (e, history) { fetching = false; if (e) { return void console.error(e); } if (history.length === 0) { channel.exhausted = true; return; } history.forEach(function (msg, i) { if (channel.exhausted) { return; } if (msg.sig) { if (i === 0 && history.length > 1 && msg.sig === channel.TAIL) { // First message is usually the lastKnownHash, ignore it return; } if (msg.sig === channel.TAIL) { console.error('No more messages to fetch'); channel.exhausted = true; return void $moreHistory.addClass('cp-app-contacts-faded'); } else { channel.TAIL = msg.sig; } } else { return void console.error('expected signature'); } if (msg.type !== 'MSG') { return; } // FIXME Schlameil the painter (performance does not scale well) channel.messages.unshift(msg); var el_message = markup.message(msg); $messagebox.prepend(el_message); }); normalizeLabels($messagebox); }); }); var removeHistory = h('span.cp-app-contacts-remove-history.fa.fa-eraser', { title: Messages.contacts_removeHistoryTitle }); $(removeHistory).click(function () { UI.confirm(Messages.contacts_confirmRemoveHistory, function (yes) { if (!yes) { return; } execCommand('CLEAR_OWNED_CHANNEL', id, function (e) { if (e) { console.error(e); UI.alert(Messages.contacts_removeHistoryServerError); return; } }); }); }); var avatar = h('div.cp-avatar'); var headerContent = [avatar, moreHistory, data.isFriendChat ? removeHistory : undefined]; if (isApp) { headerContent = [ h('div.cp-app-contacts-header-title', Messages.contacts_padTitle), moreHistory ]; } var header = h('div.cp-app-contacts-header', headerContent); var priv = metadataMgr.getPrivateData(); var closeTips = h('span.fa.fa-window-close.cp-app-contacts-tips-close'); var tips; if (isApp && Util.find(priv.settings, ['general', 'hidetips', 'chat']) !== true) { tips = h('div.cp-app-contacts-tips', [ closeTips, Messages.contacts_warning ]); } $(closeTips).click(function () { $(tips).hide(); common.setAttribute(['general', 'hidetips', 'chat'], true); }); var messages = h('div.cp-app-contacts-messages'); var input = h('textarea', { placeholder: Messages.contacts_typeHere }); var sendButton = h('button.btn.btn-primary.fa.fa-paper-plane', { title: Messages.contacts_send, }); var rightCol = h('span.cp-app-contacts-right-col', [ h('span.cp-app-contacts-name', displayName), ]); var $avatar = $(avatar); var friend = contactsData[curvePublic] || {}; if (friend.avatar && avatars[friend.avatar]) { $avatar.append(avatars[friend.avatar]).append(rightCol); } else { common.displayAvatar($avatar, friend.avatar, displayName, function ($img) { if (friend.avatar && $img) { avatars[friend.avatar] = $img[0].outerHTML; } $(rightCol).insertAfter($avatar); }); } var sending = false; var send = function (content) { if (typeof(content) !== 'string' || !content.trim()) { return; } if (sending) { return false; } sending = true; execCommand('SEND_MESSAGE', { id: id, content: content }, function (e) { if (e) { // failed to send return void console.error('failed to send', e); } input.value = ''; sending = false; debug('sent successfully'); scrollChatToBottom(); }); }; var onKeyDown = function (e) { // ignore anything that isn't 'enter' if (e.keyCode !== 13) { return; } // send unless they're holding a ctrl-key or shift if (!e.ctrlKey && !e.shiftKey) { send(this.value); return false; } // insert a newline if they're holding either var val = this.value; var start = this.selectionStart; var end = this.selectionEnd; if (![start,end].some(function (x) { return typeof(x) !== 'number'; })) { this.value = val.slice(0, start) + '\n' + val.slice(end); this.selectionStart = this.selectionEnd = start + 1; } else if (document.selection && document.selection.createRange) { this.focus(); var range = document.selection.createRange(); range.text = '\r\n'; range.collapse(false); range.select(); } return false; }; $(input).on('keydown', onKeyDown); $(sendButton).click(function () { send(input.value); }); return h('div.cp-app-contacts-chat', { 'data-key': id, 'data-user': data.isFriendChat && curvePublic }, [ header, readOnly ? undefined : tips, messages, readOnly ? undefined : h('div.cp-app-contacts-input', [ input, sendButton, ]), ]); }; var hideInfo = function () { $messages.find('.cp-app-contacts-info').hide(); }; var showInfo = function () { $messages.find('.cp-app-contacts-info').show(); }; var updateStatus = function (id) { if (!state.channels[id]) { return; } var $status = find.inList(id).find('.cp-app-contacts-status'); execCommand('GET_STATUS', id, function (e, online) { // if error maybe you shouldn't display this friend... if (e) { find.inList(id).hide(); getChat(id).hide(); return void console.error(id, e); } if (online) { return void $status .removeClass('cp-app-contacts-offline').addClass('cp-app-contacts-online'); } $status.removeClass('cp-app-contacts-online').addClass('cp-app-contacts-offline'); }); }; var display = function (chanId) { var channel = state.channels[chanId]; var lastMsg = channel.messages.slice(-1)[0]; if (lastMsg) { channel.HEAD = lastMsg.sig; execCommand('SET_CHANNEL_HEAD', { id: chanId, sig: channel.HEAD }, function (e) { if (e) { console.error(e); } }); } setActive(chanId); unnotify(chanId); var $chat = getChat(chanId); hideInfo(); $messages.find('div.cp-app-contacts-chat[data-key]').hide(); if ($chat.length) { var $chat_messages = $chat.find('div.cp-app-contacts-message'); if ($chat_messages.length < 10) { delete channel.needMoreHistory; var $more = $chat.find('.cp-app-contacts-more-history'); $more.click(); } $chat.show(); return; } else { console.error("Chat is missing... Please reload the page and try again."); } }; var unmuteUser = function (curve) { execCommand('UNMUTE_USER', curve, function (e) { if (e) { return void console.error(e); } }); }; var muteUser = function (data) { execCommand('MUTE_USER', { curvePublic: data.curvePublic, name: data.displayName || data.name, avatar: data.avatar }, function (e /*, removed */) { if (e) { return void console.error(e); } }); }; var removeFriend = function (curvePublic) { execCommand('REMOVE_FRIEND', curvePublic, function (e /*, removed */) { if (e) { return void console.error(e); } }); }; markup.room = function (id, room, userlist) { var roomEl = h('div.cp-app-contacts-friend.cp-avatar', { 'data-key': id, 'data-user': room.isFriendChat ? userlist[0].curvePublic : '', title: room.name }); var curve; if (room.isFriendChat) { var __channel = state.channels[id]; curve = __channel.curvePublic; } var unmute = h('span.cp-app-contacts-remove.fa.fa-bell.cp-unmute-icon', { title: Messages.contacts_unmute || 'unmute', style: (curve && mutedUsers[curve]) ? undefined : 'display: none;' }); var mute = h('span.cp-app-contacts-remove.fa.fa-bell-slash.cp-mute-icon', { title: Messages.contacts_mute || 'mute', style: (curve && mutedUsers[curve]) ? 'display: none;' : undefined }); var remove = h('span.cp-app-contacts-remove.fa.fa-user-times', { title: Messages.contacts_remove }); var leaveRoom = h('span.cp-app-contacts-remove.fa.fa-sign-out', { title: Messages.contacts_leaveRoom }); var status = h('span.cp-app-contacts-status', { title: Messages.contacts_online }); var rightCol = h('span.cp-app-contacts-right-col', [ h('span.cp-app-contacts-name', [room.name]), h('span.cp-app-contacts-icons', [ room.isFriendChat ? mute : undefined, room.isFriendChat ? unmute : undefined, room.isFriendChat ? remove : (room.isPadChat || room.isTeamChat) ? undefined : leaveRoom, ]) ]); var friendData = room.isFriendChat ? userlist[0] : {}; var $room = $(roomEl).click(function () { display(id); }).dblclick(function () { if (friendData.profile) { window.open(origin + '/profile/#' + friendData.profile); } }); $(unmute).on('click dblclick', function (e) { e.stopPropagation(); var channel = state.channels[id]; if (!channel.isFriendChat) { return; } var curvePublic = channel.curvePublic; $(mute).show(); $(unmute).hide(); unmuteUser(curvePublic); }); $(mute).on('click dblclick', function (e) { e.stopPropagation(); var channel = state.channels[id]; if (!channel.isFriendChat) { return; } var curvePublic = channel.curvePublic; var friend = contactsData[curvePublic] || friendData; $(mute).hide(); $(unmute).show(); muteUser(friend); }); $(remove).click(function (e) { e.stopPropagation(); var channel = state.channels[id]; if (!channel.isFriendChat) { return; } var curvePublic = channel.curvePublic; var friend = contactsData[curvePublic] || friendData; var content = h('div', [ UI.setHTML(h('p'), Messages._getKey('contacts_confirmRemove', [Util.fixHTML(friend.name)])), ]); UI.confirm(content, function (yes) { if (!yes) { return; } removeFriend(curvePublic); // TODO remove friend from userlist ui // FIXME seems to trigger EJOINED from netflux-websocket (from server); // (tried to join a channel in which you were already present) }); }); if (friendData.avatar && avatars[friendData.avatar]) { $room.append(avatars[friendData.avatar]); $room.append(rightCol); } else { common.displayAvatar($room, friendData.avatar, room.name, function ($img) { if (friendData.avatar && $img) { avatars[friendData.avatar] = $img[0].outerHTML; } $room.append(rightCol); }); } $room.append(status); return $room; }; var isBottomedOut = function ($elem) { return ($elem[0].scrollHeight - $elem.scrollTop() === $elem[0].clientHeight); }; var onMessage = function (message) { var chanId = message.channel; var channel = state.channels[chanId]; if (!channel) { return; } var chat = getChat(chanId); debug(message); var el_message = markup.message(message); if (message.type === 'MSG') { var name = typeof message.name !== "undefined" ? (message.name || Messages.anonymous) : contactsData[message.author].displayName; common.notify({ title: name, msg: message.text, force: toolbar && toolbar.team && !toolbar['chat'].hasClass('cp-leftside-active') }); } notifyToolbar(); channel.messages.push(message); var $chat = $(chat); if (!$chat.length) { console.error("Got a message but the chat isn't open"); } var $messagebox = $chat.find('.cp-app-contacts-messages'); var shouldScroll = isBottomedOut($messagebox); $messagebox.append(el_message); if (shouldScroll) { scrollChatToBottom(); } normalizeLabels($messagebox); reorderRooms(); if (isActive(chanId)) { channel.HEAD = message.sig; execCommand('SET_CHANNEL_HEAD', { id: chanId, sig: message.sig }, function (e) { if (e) { return void console.error(e); } }); return; } var lastMsg = channel.messages.slice(-1)[0]; if (lastMsg.sig !== channel.HEAD) { return void notify(chanId, message); } unnotify(chanId); }; var onJoin = function (obj) { var channel = obj.id; var data = obj.info; if (data.curvePublic) { contactsData[data.curvePublic] = data; } updateStatus(channel); }; var onLeave = function (obj) { var channel = obj.id; var chan = state.channels[channel]; var data = obj.info; if (contactsData[data.curvePublic] && !(chan && chan.isFriendChat)) { delete contactsData[data.curvePublic]; } updateStatus(channel); }; // change in your friend list var onUpdateData = function (data) { var info = data.info; var types = data.types; var channel = data.channel; if (!info || !info.curvePublic) { return; } // Make sure we don't store useless data (friends data in pad chat or the other way) if (channel && !state.channels[channel]) { return; } var curvePublic = info.curvePublic; contactsData[curvePublic] = info; if (types.indexOf('displayName') !== -1) { var name = info.displayName; // update label in friend list $userlist.find(userQuery(curvePublic)).find('.cp-app-contacts-name').text(name); $userlist.find(userQuery(curvePublic)).attr('title', name); // update title bar and messages $messages.find(userQuery(curvePublic) + ' .cp-app-contacts-header ' + '.cp-app-contacts-name, div.cp-app-contacts-message'+ userQuery(curvePublic) + ' div.cp-app-contacts-sender').text(name); } if (types.indexOf('profile') !== -1) { // update dblclick event in friend list $userlist.find(userQuery(curvePublic)).off('dblclick').dblclick(function () { if (info.profile) { window.open(origin + '/profile/#' + info.profile); } }); } if (types.indexOf('avatar') !== -1) { var $mAvatar = $messages .find(userQuery(curvePublic) +' .cp-app-contacts-header .cp-avatar'); var $lAvatar = $userlist.find(userQuery(curvePublic)); $lAvatar.find('.cp-avatar-default, media-tag').remove(); var $div = $('