2017-06-14 16:39:34 +02:00
/ * *
* Conversation chat window functions
*
* @ author Pierre HUBERT
* /
2020-04-02 19:13:10 +02:00
const ConvChatWindow = {
2017-06-18 09:15:50 +02:00
2017-06-18 11:32:31 +02:00
/ * *
* @ var { Object } _ _conversationsCache Chat windows cache
* /
_ _conversationsCache : { } ,
2017-06-18 09:15:50 +02:00
/ * *
* Open a new conversation window
*
2017-06-18 11:14:26 +02:00
* @ param { Integer } conversationID The ID of the window to open
2017-06-18 09:15:50 +02:00
* @ return { Boolean } True for a success
* /
openConversation : function ( conversationID ) {
//Log action
ComunicWeb . debug . logMessage ( "Opening conversation " + conversationID ) ;
//Create a conversation window
var conversationWindow = this . create ( {
2018-03-25 09:43:39 +02:00
target : byId ( ComunicWeb . components . conversations . manager . _ _conversationsContainerID ) ,
2017-06-18 11:14:26 +02:00
conversationID : conversationID ,
2017-06-18 09:15:50 +02:00
} ) ;
2017-06-18 11:14:26 +02:00
//Load the conversation
this . load ( conversationID , conversationWindow ) ;
//Success
return true ;
} ,
2017-06-21 16:44:10 +02:00
/ * *
* Create a new chat window
*
* @ param { Object } infos Informations required for the new chat window
* @ info { HTMLElement } target The target of the new chat window
* @ info { Integer } conversationID The ID of the target conversation
* @ return { Object } Informations about the new chat window
* /
create : function ( infos ) {
//Log action
ComunicWeb . debug . logMessage ( "Create a new chat window" ) ;
//First, create the generic conversation window
var infosBox = ComunicWeb . components . conversations . windows . create ( infos . target . children [ 0 ] ) ;
//Save conversation ID
infosBox . conversationID = infos . conversationID ;
//Change box root class name
2019-01-23 15:45:19 +01:00
infosBox . rootElem . className += " chat-window direct-chat direct-chat-primary" ;
2017-06-21 16:44:10 +02:00
//Adapt close button behaviour
infosBox . closeFunction = function ( ) {
//Remove root element
infosBox . rootElem . remove ( ) ;
//Remove the conversation from opened ones
ComunicWeb . components . conversations . cachingOpened . remove ( infosBox . conversationID ) ;
//Unload conversation
ComunicWeb . components . conversations . chatWindows . unload ( infosBox . conversationID ) ;
}
infosBox . closeButton . onclick = infosBox . closeFunction ;
//Debug
2018-03-25 09:43:39 +02:00
//Create messages container
2017-06-24 19:23:55 +02:00
infosBox . messagesArea = createElem2 ( {
appendTo : infosBox . boxBody ,
type : "div" ,
class : "direct-chat-messages" ,
2017-06-25 19:10:26 +02:00
innerHTML : "" ,
2017-06-24 19:23:55 +02:00
} ) ;
2017-06-21 16:44:10 +02:00
//Add button to get conversation members
2019-01-23 15:45:19 +01:00
infosBox . membersButton = createElem2 ( {
type : "button" ,
insertBefore : infosBox . closeButton ,
elemType : "button" ,
class : "btn btn-box-tool" ,
title : "Conversation members"
} ) ;
2017-06-21 16:44:10 +02:00
infosBox . membersButton . setAttribute ( "data-toggle" , "tooltip" ) ;
infosBox . membersButton . setAttribute ( "data-widget" , "chat-pane-toggle" ) ;
//Add button icon
var buttonIcon = createElem ( "i" , infosBox . membersButton ) ;
buttonIcon . className = "fa fa-users" ;
2019-01-23 15:45:19 +01:00
2017-06-21 16:44:10 +02:00
//Add conversation members pane
var membersPane = createElem ( "div" , infosBox . boxBody ) ;
membersPane . className = "direct-chat-contacts" ;
//Add conversation members list
infosBox . membersList = createElem ( "ul" , membersPane ) ;
infosBox . membersList . className = "contacts-list" ;
//Add send a message form
this . addMessageform ( infosBox ) ;
//Return informations about the chat window
return infosBox ;
} ,
/ * *
* Add a message form to the chat windows
*
* @ param { Object } infosBox Informations about the chat box
* @ return { Boolean } True for a success
* /
addMessageform : function ( infosBox ) {
2018-03-25 09:43:39 +02:00
//Create form container
var conversationFormContainer = createElem2 ( {
2017-06-21 16:44:10 +02:00
appendTo : infosBox . boxFooter ,
2017-06-21 18:29:14 +02:00
type : "form" ,
2017-06-21 16:44:10 +02:00
class : "create-message-form"
} ) ;
2021-03-08 19:01:51 +01:00
new ConversationWritingNotifier ( conversationFormContainer , infosBox . conversationID )
2017-06-21 18:29:14 +02:00
//Create input group
2017-06-21 16:44:10 +02:00
var inputGroup = createElem2 ( {
2018-03-25 09:43:39 +02:00
appendTo : conversationFormContainer ,
2017-06-21 16:44:10 +02:00
type : "div" ,
class : "input-group"
} ) ;
//Create text input (for message)
var inputText = createElem2 ( {
appendTo : inputGroup ,
2017-06-21 18:29:14 +02:00
type : "textarea" ,
2017-06-21 16:44:10 +02:00
class : "form-control" ,
2021-03-06 14:37:18 +01:00
placeholder : tr ( "New message..." ) ,
2017-06-21 16:44:10 +02:00
} ) ;
2021-03-12 16:10:32 +01:00
inputText . maxLength = ServerConfig . conf . conversations _policy . max _message _len ;
2017-06-21 16:44:10 +02:00
2021-03-08 18:09:56 +01:00
// Notify other users when this user is writing a message
ConversationsUtils . listenToInputChangeEvents ( inputText , infosBox . conversationID )
2017-06-21 18:29:14 +02:00
//Enable textarea 2.0 on the message
var textarea2 = new ComunicWeb . components . textarea ( ) ;
textarea2 . init ( {
element : inputText ,
minHeight : "34px" ,
autosize : true ,
} ) ;
2021-03-06 14:37:18 +01:00
//Create file input (for optionnal file)
var fileInput = createElem2 ( {
2017-06-21 18:29:14 +02:00
type : "input" ,
elemType : "file" ,
} ) ;
2021-03-12 16:10:32 +01:00
fileInput . accept = ServerConfig . conf . conversations _policy . allowed _files _type . join ( ", " ) ;
2017-06-21 18:29:14 +02:00
2017-06-21 16:44:10 +02:00
//Create button group
var buttonGroup = createElem2 ( {
appendTo : inputGroup ,
type : "span" ,
class : "input-group-btn" ,
} ) ;
2018-04-21 08:38:45 +02:00
//Add emojie button
var emojiButton = createElem2 ( {
appendTo : buttonGroup ,
type : "button" ,
elemType : "button" ,
class : "btn btn-flat btn-add-emoji" ,
} ) ;
createElem2 ( {
type : "i" ,
appendTo : emojiButton ,
class : "fa fa-smile-o"
} ) ;
//Make emojie button lives
ComunicWeb . components . emoji . picker . addPicker ( inputText , emojiButton , function ( ) {
//Make the emojie picker visible
wdtEmojiBundle . popup . style . top = ( abs _height _bottom _screen ( ) - 357 ) + "px" ;
//Make the smile button visible
var currLeft = Number ( wdtEmojiBundle . popup . style . left . replace ( "px" , "" ) ) ;
var potentialLeft = currLeft - 20 ;
if ( potentialLeft > 0 )
wdtEmojiBundle . popup . style . left = potentialLeft + "px" ;
} ) ;
2021-03-06 15:04:10 +01:00
// =========== SEND FILES ===========
2017-06-21 18:29:14 +02:00
//Add image button
2021-03-06 14:37:18 +01:00
const fileButton = createElem2 ( {
2017-06-21 18:29:14 +02:00
appendTo : buttonGroup ,
type : "button" ,
elemType : "button" ,
class : "btn btn-flat btn-add-image" ,
} ) ;
2021-03-06 14:37:18 +01:00
fileButton . onclick = function ( ) {
2017-06-21 18:29:14 +02:00
//Call file selector
2021-03-06 14:37:18 +01:00
fileInput . click ( ) ;
2017-06-21 18:29:14 +02:00
} ;
//Add image icon
createElem2 ( {
type : "i" ,
2021-03-06 14:37:18 +01:00
appendTo : fileButton ,
2021-03-06 15:06:54 +01:00
class : "fa fa-plus"
2017-06-21 18:29:14 +02:00
} ) ;
2017-06-21 16:44:10 +02:00
//Add send button
var sendButton = createElem2 ( {
appendTo : buttonGroup ,
2017-06-23 17:25:29 +02:00
type : "button" ,
2017-06-21 16:44:10 +02:00
class : "btn btn-primary btn-flat" ,
elemType : "submit" ,
2017-06-21 18:29:14 +02:00
} ) ;
2017-06-23 17:25:29 +02:00
//Add send icon
createElem2 ( {
appendTo : sendButton ,
type : "i" ,
class : "fa fa-send-o" ,
} ) ;
2021-03-06 15:04:10 +01:00
2021-03-06 15:06:54 +01:00
ConversationsUtils . registerInputToSendFile ( infosBox . conversationID , fileInput , conversationFormContainer ) ;
2021-03-06 15:04:10 +01:00
// =========== /SEND FILES ===========
2017-06-23 17:25:29 +02:00
2017-06-21 18:29:14 +02:00
//Prevent textarea from adding a new line when pressing enter
$ ( inputText ) . keypress ( function ( event ) {
if ( event . keyCode == 13 ) {
event . preventDefault ( ) ;
sendButton . click ( ) ;
}
2017-06-21 16:44:10 +02:00
} ) ;
2017-06-21 18:29:14 +02:00
//Add required elements to infosBox
infosBox . sendMessageForm = {
2018-03-25 09:43:39 +02:00
formRoot : conversationFormContainer ,
2017-06-21 18:29:14 +02:00
sendButton : sendButton ,
inputText : inputText ,
textarea2 : textarea2 ,
} ;
2017-06-21 16:44:10 +02:00
//Success
return true ;
} ,
2017-06-18 11:14:26 +02:00
/ * *
* Load ( or reload ) a conversation
*
* @ param { Integer } conversationID The ID of the conversation to load
* @ param { Object } conversationWindow Informations about the conversation window
* @ return { Boolean } True for a success
* /
2021-03-06 17:58:20 +01:00
load : async function ( conversationID , conversationWindow , forceRefresh ) {
2017-06-24 19:23:55 +02:00
2021-03-06 12:04:01 +01:00
try {
2017-06-24 19:23:55 +02:00
2021-03-06 12:04:01 +01:00
//Change conversation window name (loading state)
this . changeName ( "Loading" , conversationWindow ) ;
2017-06-18 09:15:50 +02:00
2021-03-06 12:04:01 +01:00
/** @type {Conversation} */
const conv = await new Promise ( ( res , rej ) => {
ConversationsInterface . getInfosOne ( conversationID , ( info ) => {
if ( info . error )
rej ( info )
else
res ( info )
2021-03-06 17:58:20 +01:00
} , forceRefresh ) ;
2021-03-06 12:04:01 +01:00
} )
2017-06-18 09:15:50 +02:00
2021-03-06 12:04:01 +01:00
const users = await getUsers ( conv . members . map ( m => m . user _id ) ) ;
2017-06-18 09:15:50 +02:00
2021-03-06 12:04:01 +01:00
// Create conversation informations root object
var conversationInfos = {
box : conversationWindow ,
membersInfos : users ,
infos : conv
} ;
2017-06-18 09:15:50 +02:00
2021-03-06 12:04:01 +01:00
// Save conversation informations in the cache
this . _ _conversationsCache [ "conversation-" + conversationID ] = conversationInfos ;
2017-06-18 09:15:50 +02:00
2021-03-06 12:04:01 +01:00
//Change the name of the conversation
this . changeName ( await getConvName ( conv ) , conversationWindow ) ;
2017-06-18 09:15:50 +02:00
2021-03-07 14:36:17 +01:00
// Apply the color of the conversation (if any)
if ( conv . color )
conversationWindow . rootElem . setAttribute ( "style" , "--primary-blue: #" + conv . color )
2021-03-06 12:04:01 +01:00
// Update conversation members informations
this . updateMembersList ( conversationInfos ) ;
2017-06-18 09:15:50 +02:00
2021-03-06 12:04:01 +01:00
// Display conversation settings pane
this . showConversationSettings ( conversationInfos ) ;
2017-06-24 19:23:55 +02:00
2021-03-06 12:04:01 +01:00
// Register the conversation in the service
ConvService . registerConversation ( conversationID ) ;
2017-06-21 18:29:14 +02:00
2021-03-06 12:04:01 +01:00
// Make send a message button lives
conversationInfos . box . sendMessageForm . formRoot . onsubmit = ( e ) => {
e . preventDefault ( ) ;
//Submit new message
this . submitMessageForm ( conversationInfos ) ;
2017-06-21 18:29:14 +02:00
2021-03-06 12:04:01 +01:00
} ;
2019-01-23 15:45:19 +01:00
2021-03-06 12:04:01 +01:00
//Add call button (if possible)
this . showCallButton ( conversationInfos ) ;
2017-06-18 09:15:50 +02:00
2021-03-06 12:04:01 +01:00
}
catch ( e ) {
console . error ( e ) ;
notify ( tr ( "Failed to load conversation!" ) , "danger" ) ;
}
2017-06-18 09:15:50 +02:00
} ,
2017-06-18 11:32:31 +02:00
/ * *
* Unload a chat window
*
* @ param { Integer } conversationID The ID of the conversation to unload
* @ param { Boolean } keepInfos Keep informations about the chat window
* @ return { Boolean } True for a success
* /
unload : function ( conversationID , keepInfos ) {
if ( ! this . _ _conversationsCache [ "conversation-" + conversationID ] ) {
ComunicWeb . debug . logMessage ( "Couldn't unload conversation: " + conversationID + ". It seems not to be loaded..." ) ;
return false ;
}
//Log action
ComunicWeb . debug . logMessage ( "Unloading a conversation: " + conversationID ) ;
2017-06-24 19:23:55 +02:00
//Get informations
var conversationInfos = this . _ _conversationsCache [ "conversation-" + conversationID ] ;
//Empty messages area
emptyElem ( conversationInfos . box . messagesArea ) ;
2017-06-25 19:10:26 +02:00
conversationInfos . box . messagesArea . innerHTML = "" ;
2017-06-24 19:23:55 +02:00
//Un-register conversation
ComunicWeb . components . conversations . service . unregisterConversation ( conversationID ) ;
2017-06-18 11:32:31 +02:00
//Remove informations if required
if ( ! keepInfos ) {
delete this . _ _conversationsCache [ "conversation-" + conversationID ] ;
}
//Success
return true ;
} ,
/ * *
* Unload all chat windows
*
* @ return { Boolean } True for a success
* /
unloadAll : function ( ) {
//Clear conversation object
clearObject ( this . _ _conversationsCache ) ;
//Success
return true ;
} ,
2017-06-16 11:42:28 +02:00
/ * *
* Change the name of the converation at the top of the windows
*
* @ param { String } newName The new name for the conversation window
2019-05-16 18:40:39 +02:00
* @ param { Ojbect } info Information about the conversation window
2017-06-16 11:42:28 +02:00
* @ return { Boolean } True for a success
* /
2019-05-16 18:40:39 +02:00
changeName : function ( newName , info ) {
2017-06-16 11:42:28 +02:00
2018-08-16 12:37:03 +02:00
//Reduce new name
if ( newName . length > 18 )
newName = newName . slice ( 0 , 17 ) + "..." ;
2017-06-16 11:42:28 +02:00
//Empty name field
2019-05-16 18:40:39 +02:00
emptyElem ( info . boxTitle ) ;
2017-06-16 11:42:28 +02:00
//Create conversation icon
2019-05-16 18:40:39 +02:00
createElem2 ( {
type : "i" ,
appendTo : info . boxTitle ,
class : "fa fa-comments" ,
ondblclick : ( ) => {
openConversation ( info . conversationID , true ) ;
info . closeFunction ( ) ;
}
} ) ;
2017-06-16 11:42:28 +02:00
//Add conversation title
2019-05-16 18:40:39 +02:00
var conversationTitle = createElem ( "span" , info . boxTitle ) ;
2017-06-16 11:42:28 +02:00
conversationTitle . innerHTML = " " + newName ;
2017-06-17 10:09:37 +02:00
//Success
return true ;
} ,
/ * *
* Update conversation members list
*
2021-03-06 17:43:27 +01:00
* @ param { Object } conv Information about the conversation
2017-06-17 10:09:37 +02:00
* @ return { Boolean } True for a success
* /
2021-03-06 17:43:27 +01:00
updateMembersList : function ( info ) {
2017-06-17 10:09:37 +02:00
//First, make sure conversation members pane is empty
2021-03-06 17:43:27 +01:00
emptyElem ( info . box . membersList ) ;
/** @type {Conversation} */
const conv = info . infos ;
2021-03-06 19:06:40 +01:00
let isAdmin = conv . members . find ( m => m . user _id == userID ( ) ) . is _admin ;
2021-03-06 17:43:27 +01:00
let canAddUser = conv . group _id == null && ( conv . can _everyone _add _members || isAdmin ) ;
2021-03-06 18:05:53 +01:00
let canRemoveUsers = isAdmin && canAddUser ;
2021-03-06 17:43:27 +01:00
// =================== Add a member ===================
if ( canAddUser ) {
//Create form container
var addUserForm = createElem2 ( {
appendTo : info . box . membersList ,
type : "form" ,
class : "invite-user-form"
} ) ;
//Form input
let userInput = createFormGroup ( {
target : addUserForm ,
multiple : false ,
placeholder : "Select user" ,
type : "select2" } ) ;
userInput . parentNode . className = "input-group" ;
ComunicWeb . components . userSelect . init ( userInput ) ;
//Add submit button
var groupsButton = createElem2 ( {
appendTo : userInput . parentNode ,
type : "div" ,
class : "input-group-btn"
} ) ;
createElem2 ( {
appendTo : groupsButton ,
type : "button" ,
elemType : "submit" ,
class : "btn btn-primary" ,
innerHTML : "Add"
} ) ;
addUserForm . addEventListener ( "submit" , async e => {
try {
e . preventDefault ( ) ;
//Get the list of selected users
var usersToInvite = ComunicWeb . components . userSelect . getResults ( userInput ) ;
//Check if there is not any user to invite
if ( usersToInvite . length == 0 ) {
notify ( tr ( "Please choose a user to add!" ) , "danger" ) ;
return ;
}
await ConversationsInterface . addUser ( conv . id , usersToInvite [ 0 ] ) ;
ConvChatWindow . reload ( info )
}
2017-06-17 10:09:37 +02:00
2021-03-06 17:43:27 +01:00
catch ( e )
{
console . error ( e ) ;
notify ( tr ( "Failed to update conversation settings!" ) , "danger" )
}
} )
}
// =================== / Add a member ===================
// Then process each user
for ( let member of conv . members ) {
let user = info . membersInfos . get ( member . user _id ) ;
2021-03-06 12:04:01 +01:00
if ( ! user )
continue ;
//Display user informations
2021-03-06 17:43:27 +01:00
var userLi = createElem ( "li" , info . box . membersList ) ;
2021-03-06 12:04:01 +01:00
var userLink = createElem ( "a" , userLi ) ;
//Add user account image
var userImage = createElem ( "img" , userLink ) ;
userImage . className = "contacts-list-img" ;
userImage . src = user . image ;
//Add member informations
var memberInfosList = createElem2 ( {
type : "div" ,
appendTo : userLink ,
class : "contacts-list-info" ,
} ) ;
2017-06-17 10:09:37 +02:00
2021-03-06 12:04:01 +01:00
//Add user name
var memberName = createElem2 ( {
type : "span" ,
appendTo : memberInfosList ,
class : "contacts-list-name" ,
innerHTML : user . fullName ,
} ) ;
2017-06-17 10:09:37 +02:00
2021-03-06 12:04:01 +01:00
//Add member status
2021-03-06 18:05:53 +01:00
let status = createElem2 ( {
2021-03-06 12:04:01 +01:00
type : "span" ,
appendTo : memberInfosList ,
class : "contacts-list-msg" ,
2021-03-06 18:05:53 +01:00
innerHTML : ( member . is _admin ? tr ( "Admin" ) : tr ( "Member" ) ) + " "
2021-03-06 12:04:01 +01:00
} ) ;
2021-03-06 18:05:53 +01:00
2021-03-06 18:15:21 +01:00
// Set / unset admin
if ( canRemoveUsers && member . user _id != userID ( ) ) {
let removeLink = createElem2 ( {
type : "a" ,
appendTo : status ,
innerHTML : ( member . is _admin ? tr ( "Unset admin" ) : tr ( "Set admin" ) )
} )
removeLink . addEventListener ( "click" , async e => {
e . preventDefault ( ) ;
try {
await ConversationsInterface . toggleAdminStatus ( conv . id , member . user _id , ! member . is _admin ) ;
ConvChatWindow . reload ( info ) ;
} catch ( e ) {
console . error ( e ) ;
notify ( tr ( "Failed to toggle admin status of %1%!" , { "1" : user . fullName } ) , "danger" ) ;
}
} )
}
add _space ( status )
2021-03-06 18:05:53 +01:00
// Remove user
if ( canRemoveUsers && member . user _id != userID ( ) ) {
let removeLink = createElem2 ( {
type : "a" ,
appendTo : status ,
2021-03-06 19:06:40 +01:00
innerHTML : tr ( "Remove" )
2021-03-06 18:05:53 +01:00
} )
removeLink . addEventListener ( "click" , async e => {
e . preventDefault ( ) ;
2021-03-07 14:28:38 +01:00
if ( ! await showConfirmDialog ( tr ( "Do you really want to remove %1% from the conversation?" , { "1" : user . fullName } ) ) )
2021-03-06 18:05:53 +01:00
return ;
try {
await ConversationsInterface . removeUser ( conv . id , member . user _id ) ;
ConvChatWindow . reload ( info ) ;
} catch ( e ) {
console . error ( e ) ;
notify ( tr ( "Failed to remove %1% from the conversation!" , { "1" : user . fullName } ) , "danger" ) ;
}
} )
}
2021-03-07 14:28:38 +01:00
// Leave conversation
if ( member . user _id == userID ( ) ) {
let removeLink = createElem2 ( {
type : "a" ,
appendTo : status ,
innerHTML : tr ( "Leave" )
} )
removeLink . addEventListener ( "click" , async e => {
e . preventDefault ( ) ;
const isLastAdmin = conv . members . filter ( m => m . is _admin && m . user _id != userID ( ) ) . length == 0 ;
const msg = isLastAdmin ? tr ( "As you are its last admin, if you leave this conversation, it will be permanently deleted!" )
: tr ( "Do you really want to leave this conversation?" ) ;
if ( ! await showConfirmDialog ( msg ) )
return ;
try {
await ConversationsInterface . leaveConversation ( conv . id , member . user _id ) ;
// Close the conversation
info . box . closeFunction ( ) ;
} catch ( e ) {
console . error ( e ) ;
notify ( tr ( "Failed to leave conversation!" ) , "danger" ) ;
}
} )
}
2021-03-06 12:04:01 +01:00
2017-06-17 10:09:37 +02:00
}
//Enable slimscrooll
2021-03-06 17:43:27 +01:00
$ ( info . box . membersList ) . slimscroll ( {
2017-06-17 10:09:37 +02:00
height : "100%" ,
color : "#FFFFFF"
} ) ;
2017-06-16 11:42:28 +02:00
//Success
return true ;
2017-06-17 11:20:54 +02:00
} ,
/ * *
* Show conversation settings ( button + pane )
*
2017-06-17 17:24:45 +02:00
* @ param { Object } conversation Informations about the conversation
2017-06-17 11:20:54 +02:00
* @ return { Boolean } True for a success
* /
2017-06-17 17:24:45 +02:00
showConversationSettings : function ( conversation ) {
2017-06-17 11:40:57 +02:00
2017-06-18 11:14:26 +02:00
//First, check conversation settings button and pane don't exists yet
2021-03-07 14:08:01 +01:00
if ( conversation . box . settingsButton && conversation . box . settingsButton . remove ) {
conversation . box . settingsButton . remove ( ) ;
2017-06-18 11:14:26 +02:00
}
2021-03-07 14:08:01 +01:00
if ( conversation . box . settingsPane && conversation . box . settingsPane . remove ) {
conversation . box . settingsPane . remove ( ) ;
2017-06-18 11:14:26 +02:00
}
2017-06-17 11:20:54 +02:00
//Create and display conversation settings button wheel
2017-06-17 17:24:45 +02:00
conversation . box . settingsButton = createElem2 ( {
2017-06-17 11:20:54 +02:00
type : "button" ,
2017-06-17 17:24:45 +02:00
insertBefore : conversation . box . membersButton ,
2017-06-17 11:20:54 +02:00
class : "btn btn-box-tool" ,
type : "button"
} ) ;
//Add button icon
createElem2 ( {
type : "i" ,
2017-06-17 17:24:45 +02:00
appendTo : conversation . box . settingsButton ,
2017-06-17 11:20:54 +02:00
class : "fa fa-gear" ,
} ) ;
//Create settings pane
var settingsPane = createElem2 ( {
type : "div" ,
2017-06-17 17:24:45 +02:00
appendTo : conversation . box . boxBody ,
2017-06-17 11:20:54 +02:00
class : "conversation-settings-pane" ,
} ) ;
2017-06-18 11:14:26 +02:00
conversation . box . settingsPane = settingsPane ;
2017-06-17 11:20:54 +02:00
//Make the settings button lives
2017-06-17 17:24:45 +02:00
conversation . box . settingsButton . onclick = function ( ) {
2017-06-17 11:40:57 +02:00
//Update settings pane classname
2017-06-17 11:20:54 +02:00
if ( settingsPane . className . includes ( " open" ) )
settingsPane . className = settingsPane . className . replace ( " open" , "" ) ; //Close the pane
else
settingsPane . className += " open" ; //Open the pane
} ;
2017-06-17 11:40:57 +02:00
//Create the conversation form
2020-04-25 18:20:29 +02:00
const settingsForm = ConversationsUtils . createConversationForm ( settingsPane ) ;
2017-06-17 11:40:57 +02:00
//Update form informations
2017-06-17 17:24:45 +02:00
settingsForm . createButton . innerHTML = "Update settings" ;
2017-06-17 11:40:57 +02:00
//Update conversation name
2017-06-17 17:24:45 +02:00
if ( conversation . infos . name )
settingsForm . conversationNameInput . value = conversation . infos . name ;
2021-03-07 14:08:01 +01:00
// Apply conversation color
if ( conversation . infos . color ) {
settingsForm . conversationColorInput . value = "#" + conversation . infos . color ;
settingsForm . conversationColorInput . dispatchEvent ( new CustomEvent ( "change" ) )
}
2017-06-17 17:24:45 +02:00
2020-04-25 18:39:01 +02:00
//Update conversation members
2021-03-06 12:04:01 +01:00
ComunicWeb . components . userSelect . pushEntries ( settingsForm . usersElement , conversation . infos . members . map ( m => m . user _id ) ) ;
2020-04-25 18:24:47 +02:00
2020-04-25 18:39:01 +02:00
// Update checkbox to allow or not everyone to add members
2021-03-07 14:09:36 +01:00
$ ( settingsForm . allowEveryoneToAddMembers ) . iCheck ( conversation . infos . can _everyone _add _members ? "check" : "uncheck" ) ;
2020-04-25 18:39:01 +02:00
2021-03-06 18:41:42 +01:00
settingsForm . usersElement . parentNode . style . display = "none" ;
2020-04-25 18:39:01 +02:00
//Check if user is a conversation moderator or not
2021-03-06 12:04:01 +01:00
if ( ! conversation . infos . members . find ( m => m . user _id == userID ( ) ) . is _admin ) {
2021-03-07 14:08:01 +01:00
settingsForm . conversationNameInput . disabled = true ;
settingsForm . conversationColorInput . parentNode . parentNode . style . display = "none" ;
2017-06-17 11:40:57 +02:00
2020-04-25 18:20:29 +02:00
settingsForm . allowEveryoneToAddMembers . parentNode . parentNode . remove ( ) ;
2017-06-17 17:24:45 +02:00
}
2017-06-17 11:40:57 +02:00
2017-06-18 11:14:26 +02:00
//Update follow conversation checkbox status
2021-03-07 14:09:36 +01:00
$ ( settingsForm . followConversationInput ) . iCheck ( conversation . infos . members . find ( m => m . user _id == userID ( ) ) . following ? "check" : "uncheck" ) ;
2017-06-17 17:24:45 +02:00
//Save settings form in global form
conversation . settingsForm = settingsForm ;
2017-06-17 11:40:57 +02:00
2017-06-17 17:24:45 +02:00
//Make update settings button lives
2020-04-25 18:20:29 +02:00
settingsForm . createButton . onclick = ( ) => {
this . submitUpdateForm ( conversation ) ;
2017-06-17 17:24:45 +02:00
} ;
2017-06-17 11:40:57 +02:00
2021-03-07 14:55:00 +01:00
// Add conversation image section
if ( conversation . infos . members . find ( m => m . user _id == userID ( ) ) . is _admin ) {
const convImageSection = createElem2 ( {
appendTo : settingsForm . rootElem ,
type : "div" ,
2021-03-07 15:25:18 +01:00
class : "conversation-image-settings" ,
2021-03-07 15:33:29 +01:00
innerHTML : "<br/><br/><p><strong>" + tr ( "Conversation image" ) + "</strong></p>"
2021-03-07 14:55:00 +01:00
} )
2021-03-07 15:33:29 +01:00
// Show current
if ( conversation . infos . logo != null ) {
createElem2 ( {
appendTo : convImageSection ,
type : "img" ,
src : conversation . infos . logo ,
class : "current-image"
} )
}
2021-03-07 15:25:18 +01:00
// Upload a new logo
2021-03-07 14:55:00 +01:00
const newConvImagebutton = createElem2 ( {
appendTo : convImageSection ,
type : "button" ,
class : "btn btn-default" ,
2021-03-07 15:25:18 +01:00
innerHTML : tr ( "Upload a new conversation logo" )
2021-03-07 14:55:00 +01:00
} ) ;
newConvImagebutton . addEventListener ( "click" , async e => {
e . preventDefault ( ) ;
try {
await ConversationsUtils . uploadNewConversationImage ( conversation . infos . id ) ;
ConvChatWindow . reload ( conversation )
} catch ( e ) {
console . error ( e ) ;
notify ( tr ( "Failed to change conversation image!" ) , "danger" ) ;
}
} )
2021-03-07 15:25:18 +01:00
// Delete current image
if ( conversation . infos . logo != null ) {
const deleteConvImage = createElem2 ( {
appendTo : convImageSection ,
type : "button" ,
class : "btn btn-danger" ,
innerHTML : tr ( "Delete current logo" )
} ) ;
deleteConvImage . addEventListener ( "click" , async e => {
e . preventDefault ( ) ;
try {
if ( ! await showConfirmDialog ( tr ( "Do you really want to delete this image ?" ) ) )
return ;
await ConversationsInterface . deleteConversationImage ( conversation . infos . id ) ;
ConvChatWindow . reload ( conversation )
} catch ( e ) {
console . error ( e ) ;
notify ( tr ( "Failed to remove conversation image!" ) , "danger" ) ;
}
} )
}
2021-03-07 14:55:00 +01:00
}
2017-06-17 11:20:54 +02:00
} ,
2017-06-17 10:09:37 +02:00
2019-01-23 15:45:19 +01:00
/ * *
* Add a call button to the conversation , if possible
*
* @ param { Object } conversation Information about the conversation
* /
showCallButton : function ( conversation ) {
2021-03-07 19:40:16 +01:00
// Remove previous button (if any)
let previousButton = conversation . box . boxTools . querySelector ( ".phone-button" ) ;
if ( previousButton )
previousButton . remove ( ) ;
2019-01-23 15:45:19 +01:00
//Check if calls are disabled
2020-04-10 11:24:46 +02:00
if ( ! conversation . infos . can _have _call )
2019-01-23 15:45:19 +01:00
return ;
//Add the call button
2020-04-10 11:24:46 +02:00
const button = createElem2 ( {
2019-01-23 15:45:19 +01:00
insertBefore : conversation . box . boxTools . firstChild ,
type : "button" ,
2021-03-07 19:40:16 +01:00
class : "btn btn-box-tool phone-button" ,
2019-01-23 15:45:19 +01:00
innerHTML : "<i class='fa fa-phone'></i>"
} ) ;
conversation . box . callButton = button ;
2019-01-24 14:40:36 +01:00
button . addEventListener ( "click" , function ( ) {
2020-04-10 11:24:46 +02:00
CallsController . Open ( conversation . infos )
2019-01-24 14:40:36 +01:00
} ) ;
2019-01-23 15:45:19 +01:00
} ,
2017-06-17 17:24:45 +02:00
/ * *
* Process submited update conversation form
*
2019-01-23 15:45:19 +01:00
* @ param { Object } conversation Information about the conversation
2017-06-17 17:24:45 +02:00
* @ return { Boolean } True for a success
* /
submitUpdateForm : function ( conversation ) {
2019-01-23 15:45:19 +01:00
//Then, get information about the input
2017-06-17 17:24:45 +02:00
var newValues = {
2021-03-06 18:41:42 +01:00
conversationID : conversation . infos . id ,
2017-06-17 17:24:45 +02:00
following : conversation . settingsForm . followConversationInput . checked ,
}
//Add other fields if the user is a conversation moderator
2021-03-07 14:08:01 +01:00
if ( conversation . infos . members . find ( m => m . user _id == userID ( ) ) . is _admin ) {
2017-06-17 17:24:45 +02:00
//Specify conversation name
2021-03-07 14:08:01 +01:00
let nameValue = conversation . settingsForm . conversationNameInput . value
2021-03-07 14:12:47 +01:00
newValues . name = ( nameValue === "" ? null : nameValue ) ;
2021-03-07 14:08:01 +01:00
let colorValue = conversation . settingsForm . conversationColorInput . value
newValues . color = ( colorValue == "" ? null : colorValue )
2017-06-17 17:24:45 +02:00
2020-04-25 18:20:29 +02:00
newValues . canEveryoneAddMembers = conversation . settingsForm . allowEveryoneToAddMembers . checked ;
2017-06-17 17:24:45 +02:00
}
//Now, freeze the submit button
2017-06-18 11:14:26 +02:00
conversation . settingsForm . createButton . disabled = true ;
2017-06-17 17:24:45 +02:00
//Peform a request through the interface
2020-04-25 18:20:29 +02:00
ConversationsInterface . updateSettings ( newValues , function ( result ) {
2017-06-18 11:14:26 +02:00
//Enable again update button
conversation . settingsForm . createButton . disabled = false ;
//Check for errors
if ( result . error )
2021-03-06 12:04:01 +01:00
notify ( "An error occured while trying to update conversation settings !" , "danger" , 4 ) ;
2017-06-18 11:14:26 +02:00
//Reload the conversation
2021-03-06 17:43:27 +01:00
ConvChatWindow . reload ( conversation ) ;
2017-06-17 17:32:21 +02:00
} ) ;
2021-03-06 17:43:27 +01:00
} ,
2017-06-17 17:24:45 +02:00
2021-03-06 17:43:27 +01:00
/ * *
* Reload the conversation
*
* @ param { Object } conversation Information about the conversation
* /
reload : function ( conversation ) {
ConvChatWindow . unload ( conversation . infos . id , true ) ;
2021-03-06 17:58:20 +01:00
ConvChatWindow . load ( conversation . infos . id , conversation . box , true ) ;
2017-06-17 17:24:45 +02:00
} ,
2017-06-21 18:29:14 +02:00
/ * *
* Submit a new message form
*
2019-01-23 15:45:19 +01:00
* @ param { Object } convInfos Information about the conversation
2017-06-21 18:29:14 +02:00
* @ return { Boolean } True for a success
* /
2021-03-06 14:37:18 +01:00
submitMessageForm : async function ( convInfos ) {
2017-06-22 11:16:53 +02:00
2021-03-06 14:37:18 +01:00
try {
2017-06-22 11:16:53 +02:00
2021-03-06 14:37:18 +01:00
//Extract main fields
var form = convInfos . box . sendMessageForm ;
2017-06-22 11:16:53 +02:00
2021-03-06 14:37:18 +01:00
//Check if message is empty.
let message = form . inputText . value ;
2017-06-22 11:16:53 +02:00
2021-03-06 14:37:18 +01:00
if ( message . length == 0 )
return ;
2017-06-22 11:16:53 +02:00
2021-03-12 16:10:32 +01:00
if ( message . length < ServerConfig . conf . conversations _policy . min _message _len
|| message . length > ServerConfig . conf . conversations _policy . max _message _len ) {
2021-03-06 14:37:18 +01:00
notify ( tr ( "Invalid message length!" ) , "danger" , 2 ) ;
return ;
2017-06-22 11:16:53 +02:00
}
2021-03-06 14:37:18 +01:00
//Lock send button
form . sendButton . disabled = true ;
2017-06-22 11:16:53 +02:00
2021-03-06 14:37:18 +01:00
await ConversationsInterface . sendMessage ( convInfos . infos . id , message ) ;
2017-06-22 11:16:53 +02:00
//Reset the form
2021-03-06 14:37:18 +01:00
ConvChatWindow . resetCreateMessageForm ( convInfos ) ;
2017-06-22 11:16:53 +02:00
}
2021-03-06 14:37:18 +01:00
catch ( e )
{
console . error ( e )
2021-03-06 17:43:27 +01:00
notify ( tr ( "An error occured while trying to send message! Please try again..." ) , "danger" , 2 ) ;
2021-03-06 14:37:18 +01:00
}
//Unlock send button
form . sendButton . disabled = false ;
2017-06-22 11:16:53 +02:00
} ,
/ * *
* Reset a create a message form
*
2019-01-23 15:45:19 +01:00
* @ param { Object } infos Information about the conversation
2017-06-22 11:16:53 +02:00
* @ return { Boolean } True for a success
* /
resetCreateMessageForm : function ( infos ) {
//Extract form informations
var form = infos . box . sendMessageForm ;
//Unlock send button and reset its value
form . sendButton . disabled = false ;
//Empty textarea
form . inputText . value = "" ;
form . textarea2 . resetHeight ( ) ;
2017-06-21 18:29:14 +02:00
} ,
2017-06-25 19:10:26 +02:00
/ * *
* Add a message to a conversation window
*
* @ param { Integer } conversationID The ID of the conversation to update
2018-04-27 18:26:16 +02:00
* @ param { Object } messageInfo Information about the message to add
2017-06-25 19:10:26 +02:00
* @ return { Boolean } True for a success
* /
2018-04-27 18:26:16 +02:00
addMessage : function ( conversationID , messageInfo ) {
2017-06-25 19:10:26 +02:00
2018-04-27 19:05:06 +02:00
//First, check if the conversation information can be found
2017-06-25 19:10:26 +02:00
if ( ! this . _ _conversationsCache [ "conversation-" + conversationID ] ) {
ComunicWeb . debug . logMessage ( "Conversation Chat Windows : Error ! Couldn't add a message to the conversation because the conversation was not found !" ) ;
return false ;
}
//Else extract conversation informations
var convInfos = this . _ _conversationsCache [ "conversation-" + conversationID ] ;
2017-06-26 11:16:14 +02:00
//Check if this is the first message of the conversation or not
2018-04-27 18:26:16 +02:00
if ( ! convInfos . messages ) {
convInfos . messages = [ ] ;
2017-06-26 11:16:14 +02:00
}
2017-06-25 19:10:26 +02:00
2018-04-27 18:41:57 +02:00
//Get message HTML element add append it
var uiMessageInfo = this . _get _message _element ( convInfos , messageInfo ) ;
convInfos . box . messagesArea . appendChild ( uiMessageInfo . rootElem ) ;
//Perform post-processing operations
var num = convInfos . messages . push ( uiMessageInfo ) ;
//Check if it is not the first message from the current user
this . _makeMessageFollowAnotherMessage ( convInfos , num - 1 ) ;
//Enable slimscroll
$ ( convInfos . box . messagesArea ) . slimscroll ( {
height : "250px" ,
} ) ;
//Scroll to the bottom of the conversation
var scrollBottom = $ ( convInfos . box . messagesArea ) . prop ( "scrollHeight" ) + "px" ;
$ ( convInfos . box . messagesArea ) . slimScroll ( {
scrollTo : scrollBottom
} ) ;
//Initialize top scroll detection if required
this . initTopScrollDetection ( conversationID ) ;
//Success
return true ;
} ,
2018-04-27 19:05:06 +02:00
/ * *
* Add old messages to a conversation window
*
* @ param { number } conversationID The ID of the target conversation
* @ param { array } messages The list of messages to add
* /
addOldMessages : function ( conversationID , messages ) {
//First, check if the conversation information can be found
if ( ! this . _ _conversationsCache [ "conversation-" + conversationID ] ) {
ComunicWeb . debug . logMessage ( "Conversation Chat Windows : Error ! Couldn't add a message to the conversation because the conversation was not found !" ) ;
return false ;
}
//Else extract conversation informations
var conv = this . _ _conversationsCache [ "conversation-" + conversationID ] ;
2018-04-28 10:04:12 +02:00
//Save the position of the oldest message element
var currOldestMessageElem = conv . messages [ 0 ] . rootElem ;
2018-04-27 19:05:06 +02:00
//Process the list of messages in reverse order
messages . reverse ( ) ;
messages . forEach ( function ( message ) {
//Get message element
var uiMessageInfo = ComunicWeb . components . conversations . chatWindows . _get _message _element ( conv , message ) ;
//Add the messages at the begining of the conversation
conv . box . messagesArea . insertBefore ( uiMessageInfo . rootElem , conv . messages [ 0 ] . rootElem ) ;
//Add the message to the list
conv . messages . unshift ( uiMessageInfo ) ;
//Check if some information about the post can be updated
ComunicWeb . components . conversations . chatWindows . _makeMessageFollowAnotherMessage ( conv , 1 ) ;
} ) ;
2018-04-28 10:04:12 +02:00
//Update slimscroll
newScrollPos = currOldestMessageElem . offsetTop - 30 ;
if ( newScrollPos < 0 )
newScrollPos = 0 ;
$ ( conv . box . messagesArea ) . slimScroll ( {
scrollTo : newScrollPos + "px"
} ) ;
2018-04-27 19:05:06 +02:00
} ,
2018-04-27 18:41:57 +02:00
/ * *
* Generate message HTML node based on given information
*
* @ param { object } conversationInfo Information about the created conversation
2021-03-06 12:04:01 +01:00
* @ param { ConversationMessage } message Information about the target message
2018-04-27 18:41:57 +02:00
* @ return { object } Information about the created message element
* /
_get _message _element : function ( conversationInfo , message ) {
2021-03-06 14:25:41 +01:00
if ( message . user _id != null && message . user _id > 0 )
return this . _get _user _message ( conversationInfo , message ) ;
else
return this . _get _server _message ( conversationInfo , message ) ;
} ,
2018-04-27 18:41:57 +02:00
2021-03-06 14:25:41 +01:00
/ * *
* @ param { Object } conversationInfo
* @ param { ConversationMessage } message
* /
_get _user _message : ( conversationInfo , message ) => {
2018-04-27 18:26:16 +02:00
//Check if it is the current user who sent the message
2021-03-06 12:04:01 +01:00
var userIsPoster = message . user _id == userID ( ) ;
2017-06-25 19:10:26 +02:00
2018-04-27 18:26:16 +02:00
//Create message element
2020-04-02 19:13:10 +02:00
const messageContainer = createElem2 ( {
2018-04-27 18:26:16 +02:00
type : "div" ,
class : "direct-chat-msg " + ( userIsPoster ? "right" : "" )
} ) ;
2021-03-06 14:25:41 +01:00
messageContainer . setAttribute ( "data-chatwin-msg-id" , message . id )
2017-06-25 19:10:26 +02:00
2018-04-27 18:26:16 +02:00
//Display message header
var messageHeader = createElem2 ( {
appendTo : messageContainer ,
type : "div" ,
class : "direct-chat-info clearfix"
} ) ;
2017-06-26 11:16:14 +02:00
2018-05-02 16:16:18 +02:00
//Add top information
var topInfosElem = createElem2 ( {
appendTo : messageHeader ,
type : "div" ,
class : "direct-chat-name pull-" + ( userIsPoster ? "right" : "left" ) ,
} ) ;
2018-04-27 18:26:16 +02:00
//Add user name
var usernameElem = createElem2 ( {
2018-05-02 16:16:18 +02:00
appendTo : topInfosElem ,
2018-04-27 18:26:16 +02:00
type : "span" ,
innerHTML : "Loading" ,
} ) ;
2017-06-26 11:16:14 +02:00
2018-04-27 18:26:16 +02:00
//Hide user name if it is the current user
if ( userIsPoster )
usernameElem . style . display = "none" ;
2017-06-26 11:16:14 +02:00
2018-04-27 18:26:16 +02:00
//Add user account image
var userAccountImage = createElem2 ( {
appendTo : messageContainer ,
type : "img" ,
class : "direct-chat-img" ,
src : ComunicWeb . _ _config . assetsURL + "img/defaultAvatar.png" ,
alt : "User account image" ,
} ) ;
2017-06-26 11:16:14 +02:00
2018-04-27 18:26:16 +02:00
//Load user informations
2021-03-06 12:04:01 +01:00
let userInfo = conversationInfo . membersInfos . get ( message . user _id ) ;
if ( userInfo ) {
usernameElem . innerHTML = userInfo . fullName ;
userAccountImage . src = userInfo . image ;
2017-06-26 11:16:14 +02:00
}
2017-06-25 19:10:26 +02:00
2021-03-06 14:09:19 +01:00
else {
async ( ) => {
try {
const userInfo = await userInfo ( message . user _id ) ;
usernameElem . innerHTML = userInfo . fullName ;
userAccountImage . src = userInfo . image ;
} catch ( e ) {
console . error ( e ) ;
}
}
}
2017-06-25 19:10:26 +02:00
//Add message
var messageTargetElem = createElem2 ( {
2017-06-26 11:16:14 +02:00
appendTo : messageContainer ,
2017-06-25 19:10:26 +02:00
type : "div" ,
2018-04-27 18:26:16 +02:00
class : "direct-chat-text " ,
2017-06-25 19:10:26 +02:00
} ) ;
//Add text message
2017-06-26 10:56:12 +02:00
var textMessage = createElem2 ( {
2017-06-25 19:10:26 +02:00
appendTo : messageTargetElem ,
type : "span" ,
2018-04-27 18:41:57 +02:00
innerHTML : removeHtmlTags ( message . message ) , //Remove HTML tags
2017-06-25 19:10:26 +02:00
} ) ;
//Check if an image has to be added
2021-03-06 15:09:36 +01:00
if ( message . file != null ) {
const messageFile = message . file ;
if ( messageFile . type == "image/png" ) {
var imageLink = createElem2 ( {
appendTo : messageTargetElem ,
type : "a" ,
href : messageFile . url
} ) ;
2017-06-26 15:36:29 +02:00
2021-03-06 15:09:36 +01:00
//Apply image
createElem2 ( {
appendTo : imageLink ,
type : "img" ,
2021-03-06 15:17:57 +01:00
class : "conversation-msg-image" ,
2021-03-06 15:09:36 +01:00
src : messageFile . thumbnail == null ? messageFile . url : messageFile . thumbnail
2017-06-26 15:36:29 +02:00
} ) ;
2021-03-06 15:09:36 +01:00
imageLink . onclick = function ( ) {
$ ( this ) . ekkoLightbox ( {
alwaysShowClose : true ,
} ) ;
return false ;
} ;
}
else if ( messageFile . type == "audio/mpeg" ) {
new SmallMediaPlayer ( messageTargetElem , messageFile . url , false )
}
else if ( messageFile . type == "video/mp4" ) {
new SmallMediaPlayer ( messageTargetElem , messageFile . url , true )
}
// Fallback
else {
let letFileLink = createElem2 ( {
appendTo : messageTargetElem ,
type : "a" ,
href : messageFile . url ,
2021-03-06 19:09:54 +01:00
innerHTML : "<i class='fa fa-download'></i> " + messageFile . name + " (" + fileSizeToHuman ( messageFile . size ) + ")" ,
2021-03-06 15:09:36 +01:00
} )
letFileLink . target = "_blank"
2017-06-26 15:36:29 +02:00
}
2017-06-25 19:10:26 +02:00
}
2018-05-02 16:16:18 +02:00
//Add date
var dateElem = createElem2 ( {
appendTo : messageContainer ,
type : "div" ,
class : "date-conversation-message" ,
2021-03-06 14:07:20 +01:00
innerHTML : ComunicWeb . common . date . timeDiffToStr ( message . time _sent )
2018-05-02 16:16:18 +02:00
} ) ;
2017-06-26 10:56:12 +02:00
//Parse emojies in text message
2018-05-03 22:05:06 +02:00
ComunicWeb . components . textParser . parse ( {
2017-06-26 10:56:12 +02:00
element : textMessage ,
2021-03-06 12:04:01 +01:00
user : userInfo ,
2017-06-26 10:56:12 +02:00
} ) ;
2019-01-11 17:57:55 +01:00
//Add message dropdown menu
messageContainer . className += " dropdown" ;
var dropdownToggle = createElem2 ( {
insertBefore : dateElem ,
type : "i" ,
class : "hidden"
} ) ;
dropdownToggle . setAttribute ( "data-toggle" , "dropdown" ) ;
var dropdownMenu = createElem2 ( {
insertBefore : dateElem ,
type : "ul" ,
class : "dropdown-menu"
} ) ;
dropdownMenu . setAttribute ( "role" , "menu" ) ;
messageTargetElem . addEventListener ( "dblclick" , function ( ) {
$ ( dropdownToggle ) . dropdown ( "toggle" ) ;
} ) ;
//Add message options
if ( userIsPoster ) {
2021-03-06 14:52:35 +01:00
if ( message . file == null )
{
//Update message content
var updateLi = createElem2 ( {
type : "li" ,
appendTo : dropdownMenu
} ) ;
2019-01-11 18:17:37 +01:00
2021-03-06 14:52:35 +01:00
var updateLink = createElem2 ( {
type : "a" ,
appendTo : updateLi ,
innerHTML : "Edit"
} ) ;
2019-01-11 18:17:37 +01:00
2021-03-06 14:52:35 +01:00
updateLink . addEventListener ( "click" , function ( ) {
ComunicWeb . components . conversations . messageEditor . open ( message , function ( newContent ) {
/ *
DEPRECATED WITH WEBSOCKETS
* /
} ) ;
2019-01-11 18:17:37 +01:00
} ) ;
2021-03-06 14:52:35 +01:00
}
2019-01-11 18:17:37 +01:00
2019-01-11 17:57:55 +01:00
//Delete the message
var deleteLi = createElem2 ( {
type : "li" ,
appendTo : dropdownMenu
} ) ;
var deleteLink = createElem2 ( {
type : "a" ,
appendTo : deleteLi ,
innerHTML : "Delete"
} ) ;
deleteLink . addEventListener ( "click" , function ( ) {
ComunicWeb . common . messages . confirm (
"Do you really want to delete this message? The operation can not be reverted!" ,
function ( confirm ) {
if ( ! confirm ) return ;
//Hide the message
messageTargetElem . style . display = "none" ;
//Execute the request
2021-03-06 14:25:41 +01:00
ConversationsInterface . DeleteSingleMessage (
2021-03-06 14:37:59 +01:00
message . id ,
2019-01-11 17:57:55 +01:00
function ( result ) {
if ( ! result ) {
messageTargetElem . style . display = "block" ;
notify ( "Could delete conversation message!" , "danger" ) ;
}
}
) ;
}
)
} ) ;
}
2018-04-27 18:41:57 +02:00
//Return information about the message
return {
2021-03-06 12:04:01 +01:00
userID : message . user _id ,
2018-04-27 18:26:16 +02:00
rootElem : messageContainer ,
userNameElem : usernameElem ,
2018-05-02 16:16:18 +02:00
dateElem : dateElem ,
2021-03-06 14:07:20 +01:00
time _sent : message . time _sent ,
2018-04-27 18:26:16 +02:00
messageTargetElem : messageTargetElem ,
accountImage : userAccountImage
} ;
2018-04-27 18:41:57 +02:00
} ,
2018-04-27 18:26:16 +02:00
2021-03-06 14:25:41 +01:00
/ * *
* Apply a server message
*
* @ param { Object } convInfo Information about the conversation
* @ param { ConversationMessage } message The message
* /
_get _server _message : ( convInfo , message ) => {
//Create message element
const messageContainer = createElem2 ( {
type : "div" ,
class : "direct-chat-msg "
} ) ;
messageContainer . setAttribute ( "data-chatwin-msg-id" , message . id ) ;
//Add message
let messageTargetElem = createElem2 ( {
appendTo : messageContainer ,
type : "div" ,
class : "server_message" ,
} ) ;
( async ( ) => {
try {
let ids = ConversationsUtils . getUsersIDForMessage ( message )
let users = await getUsers ( ids ) ;
let msg = ConversationsUtils . getServerMessage ( message , users ) ;
messageTargetElem . innerHTML = msg
}
catch ( e ) {
console . error ( e ) ;
notify ( tr ( "Failed to load a server message!" ) )
}
} ) ( ) ;
return {
userID : null ,
rootElem : messageContainer ,
userNameElem : document . createElement ( "span" ) ,
dateElem : document . createElement ( "span" ) ,
time _sent : message . time _sent ,
messageTargetElem : messageTargetElem ,
accountImage : document . createElement ( "span" )
} ;
} ,
2018-04-27 18:41:57 +02:00
/ * *
* Make a conversation message "follow" another conversation message from the
* same user
*
* @ param { object } conv Information about the target conversation
* @ param { number } num The number of the conversation message to update
* /
_makeMessageFollowAnotherMessage : function ( conv , num ) {
2018-04-27 18:26:16 +02:00
2018-04-27 18:41:57 +02:00
if ( conv . messages [ num - 1 ] ) {
2018-04-27 18:26:16 +02:00
2018-04-27 18:41:57 +02:00
if ( conv . messages [ num - 1 ] . userID == conv . messages [ num ] . userID ) {
2018-04-27 18:26:16 +02:00
//Update object class name
2018-04-27 18:41:57 +02:00
conv . messages [ num ] . messageTargetElem . className += " not-first-message" ;
2018-04-27 18:26:16 +02:00
//Hide user name and account image
2018-04-27 18:41:57 +02:00
conv . messages [ num ] . userNameElem . style . display = "none" ;
conv . messages [ num ] . accountImage . style . display = "none" ;
2018-04-27 18:26:16 +02:00
//Update the class of the previous message
2018-04-27 18:41:57 +02:00
conv . messages [ num - 1 ] . rootElem . className += " not-last-message-from-user" ;
2018-04-27 18:26:16 +02:00
}
2018-05-02 16:16:18 +02:00
//Check the difference of time between the two messages
2021-03-06 14:07:20 +01:00
if ( conv . messages [ num ] . time _sent - conv . messages [ num - 1 ] . time _sent < 3600
2018-05-02 16:16:18 +02:00
|| conv . messages [ num ] . dateElem . innerHTML == conv . messages [ num - 1 ] . dateElem . innerHTML )
conv . messages [ num ] . dateElem . style . display = "none" ;
2018-04-27 18:26:16 +02:00
}
2017-06-25 19:10:26 +02:00
} ,
2018-04-26 06:50:23 +02:00
/ * *
* Init top scroll detection ( if required )
*
* @ param { number } conversationID The ID of the target conversation
* /
initTopScrollDetection : function ( conversationID ) {
//Extract conversation informations
var convInfo = this . _ _conversationsCache [ "conversation-" + conversationID ] ;
//Check if nothing has to be done
if ( convInfo . box . initializedScrollDetection )
return ;
//Mark scroll detection as initialized
convInfo . box . initializedScrollDetection = true ;
var scrollDetectionLocked = false ;
var scrollTopCount = 0 ;
$ ( convInfo . box . messagesArea ) . slimScroll ( ) . bind ( "slimscrolling" , function ( e , pos ) {
if ( scrollDetectionLocked )
return ;
if ( pos != 0 ) {
scrollTopCount = 0 ;
}
scrollTopCount ++ ;
//Check top count
if ( scrollTopCount < 3 )
return ;
//Lock the detection
scrollDetectionLocked = true ;
//Fetch older messages
2021-03-06 14:25:41 +01:00
ConversationsInterface . getOlderMessages (
2018-04-26 17:55:38 +02:00
conversationID ,
2021-03-06 14:25:41 +01:00
ConvService . getOldestMessageID ( conversationID ) ,
2018-04-26 17:55:38 +02:00
10 ,
function ( result ) {
//Unlock scroll detection
scrollDetectionLocked = false ;
//Check for errors
if ( result . error ) {
notify ( "An error occured while trying to fetch older messages for the conversation !" ) ;
return ;
}
//Check for results
if ( result . length == 0 ) {
//Lock scroll detection in order to avoid useless traffic
scrollDetectionLocked = true ;
return ;
}
//Save the ID of the oldest message
2021-03-06 14:29:15 +01:00
ConvService . setOldestMessageID ( conversationID , result [ 0 ] . id ) ;
2018-04-26 17:55:38 +02:00
2018-04-27 19:05:06 +02:00
//Display the list of messages
ComunicWeb . components . conversations . chatWindows . addOldMessages ( conversationID , result ) ;
2018-04-26 17:55:38 +02:00
}
) ;
2018-04-26 06:50:23 +02:00
} ) ;
}
2017-06-18 11:32:31 +02:00
}
2020-04-02 19:13:10 +02:00
ComunicWeb . components . conversations . chatWindows = ConvChatWindow ;
2017-06-18 11:32:31 +02:00
//Register conversations cache cleaning function
2020-04-02 19:30:11 +02:00
ComunicWeb . common . cacheManager . registerCacheCleaner ( "ComunicWeb.components.conversations.chatWindows.unloadAll" ) ;
// Register to messages update events
document . addEventListener ( "updatedConvMessage" , ( e ) => {
const msg = e . detail ;
// Get message target
2021-03-06 12:04:01 +01:00
const target = document . querySelector ( "[data-chatwin-msg-id='" + msg . id + "']" ) ;
2020-04-02 19:30:11 +02:00
if ( ! target )
return ;
// Get conversation info
2021-03-06 12:04:01 +01:00
const convInfo = ConvChatWindow . _ _conversationsCache [ "conversation-" + msg . conv _id ] ;
2020-04-02 19:30:11 +02:00
if ( ! convInfo )
return ;
target . replaceWith ( ConvChatWindow . _get _message _element ( convInfo , msg ) . rootElem )
2020-04-03 09:04:41 +02:00
} ) ;
// Register to message deletion events
document . addEventListener ( "deletedConvMessage" , ( e ) => {
const msgID = e . detail ;
// Get message target
const target = document . querySelector ( "[data-chatwin-msg-id='" + msgID + "'] .direct-chat-text" ) ;
if ( ! target )
return ;
target . style . display = "none" ;
2021-03-07 19:49:34 +01:00
} )
// Register to conversation removal
document . addEventListener ( "removedUserFromConv" , e => {
const msg = e . detail ;
if ( msg . user _id != userID ( ) )
return ;
if ( ! ConvService . _ _serviceCache . hasOwnProperty ( "conversation-" + msg . conv _id ) )
return ;
ConvChatWindow . _ _conversationsCache [ "conversation-" + msg . conv _id ] . box . closeFunction ( ) ;
2021-03-13 13:39:03 +01:00
} ) ;
document . addEventListener ( "deletedConversation" , e => {
const convID = e . detail ;
if ( ! ConvService . _ _serviceCache . hasOwnProperty ( "conversation-" + convID ) )
return ;
ConvChatWindow . _ _conversationsCache [ "conversation-" + convID ] . box . closeFunction ( ) ;
2021-03-07 19:49:34 +01:00
} ) ;