|  |  |  | /* global Whisper, i18n, textsecure, _ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // eslint-disable-next-line func-names
 | 
					
						
							|  |  |  | (function() { | 
					
						
							|  |  |  |   'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   window.Whisper = window.Whisper || {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Whisper.UpdateGroupMembersDialogView = Whisper.View.extend({ | 
					
						
							|  |  |  |     className: 'loki-dialog modal', | 
					
						
							|  |  |  |     initialize(groupConvo) { | 
					
						
							|  |  |  |       const ourPK = textsecure.storage.user.getNumber(); | 
					
						
							|  |  |  |       this.groupName = groupConvo.get('name'); | 
					
						
							|  |  |  |       this.close = this.close.bind(this); | 
					
						
							|  |  |  |       this.onSubmit = this.onSubmit.bind(this); | 
					
						
							|  |  |  |       this.isPublic = groupConvo.isPublic(); | 
					
						
							|  |  |  |       this.groupId = groupConvo.id; | 
					
						
							|  |  |  |       this.avatarPath = groupConvo.getAvatarPath(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (this.isPublic) { | 
					
						
							|  |  |  |         this.titleText = `${i18n('updatePublicGroupDialogTitle')}: ${ | 
					
						
							|  |  |  |           this.groupName | 
					
						
							|  |  |  |         }`;
 | 
					
						
							|  |  |  |         // I'd much prefer to integrate mods with groupAdmins
 | 
					
						
							|  |  |  |         // but lets discuss first...
 | 
					
						
							|  |  |  |         this.isAdmin = groupConvo.isModerator( | 
					
						
							|  |  |  |           window.storage.get('primaryDevicePubKey') | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         // zero out contactList for now
 | 
					
						
							|  |  |  |         this.contactsAndMembers = []; | 
					
						
							|  |  |  |         this.existingMembers = []; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         this.titleText = i18n('updateGroupDialogTitle'); | 
					
						
							|  |  |  |         this.isAdmin = groupConvo.get('groupAdmins').includes(ourPK); | 
					
						
							|  |  |  |         const convos = window.getConversations().models.filter(d => !!d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.existingMembers = groupConvo.get('members') || []; | 
					
						
							|  |  |  |         // Show a contact if they are our friend or if they are a member
 | 
					
						
							|  |  |  |         this.contactsAndMembers = convos.filter( | 
					
						
							|  |  |  |           d => this.existingMembers.includes(d.id) && d.isPrivate() && !d.isMe() | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         this.contactsAndMembers = _.uniq( | 
					
						
							|  |  |  |           this.contactsAndMembers, | 
					
						
							|  |  |  |           true, | 
					
						
							|  |  |  |           d => d.id | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // at least make sure it's an array
 | 
					
						
							|  |  |  |         if (!Array.isArray(this.existingMembers)) { | 
					
						
							|  |  |  |           this.existingMembers = []; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       this.$el.focus(); | 
					
						
							|  |  |  |       this.render(); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     render() { | 
					
						
							|  |  |  |       this.dialogView = new Whisper.ReactWrapperView({ | 
					
						
							|  |  |  |         className: 'create-group-dialog', | 
					
						
							|  |  |  |         Component: window.Signal.Components.UpdateGroupMembersDialog, | 
					
						
							|  |  |  |         props: { | 
					
						
							|  |  |  |           titleText: this.titleText, | 
					
						
							|  |  |  |           okText: i18n('ok'), | 
					
						
							|  |  |  |           cancelText: i18n('cancel'), | 
					
						
							|  |  |  |           isPublic: this.isPublic, | 
					
						
							|  |  |  |           existingMembers: this.existingMembers, | 
					
						
							|  |  |  |           contactList: this.contactsAndMembers, | 
					
						
							|  |  |  |           isAdmin: this.isAdmin, | 
					
						
							|  |  |  |           onClose: this.close, | 
					
						
							|  |  |  |           onSubmit: this.onSubmit, | 
					
						
							|  |  |  |           groupId: this.groupId, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       this.$el.append(this.dialogView.el); | 
					
						
							|  |  |  |       return this; | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     async onSubmit(newMembers) { | 
					
						
							|  |  |  |       const _ = window.Lodash; | 
					
						
							|  |  |  |       const ourPK = textsecure.storage.user.getNumber(); | 
					
						
							|  |  |  |       const allMembers = window.Lodash.concat(newMembers, [ourPK]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // We need to NOT trigger an group update if the list of member is the same.
 | 
					
						
							|  |  |  |       const notPresentInOld = allMembers.filter( | 
					
						
							|  |  |  |         m => !this.existingMembers.includes(m) | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const notPresentInNew = this.existingMembers.filter( | 
					
						
							|  |  |  |         m => !allMembers.includes(m) | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Filter out all linked devices for cases in which one device
 | 
					
						
							|  |  |  |       // exists in group, but hasn't yet synced with its other devices.
 | 
					
						
							|  |  |  |       const getDevicesForRemoved = async () => { | 
					
						
							|  |  |  |         const promises = notPresentInNew.map(member => | 
					
						
							|  |  |  |           window.libsession.Protocols.MultiDeviceProtocol.getAllDevices(member) | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         const devices = _.flatten(await Promise.all(promises)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return devices; | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Get all devices for notPresentInNew
 | 
					
						
							|  |  |  |       const allDevicesOfMembersToRemove = await getDevicesForRemoved(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // If any extra devices of removed exist in newMembers, ensure that you filter them
 | 
					
						
							|  |  |  |       const filteredMemberes = allMembers.filter( | 
					
						
							|  |  |  |         member => !_.includes(allDevicesOfMembersToRemove, member) | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const xor = _.xor(notPresentInNew, notPresentInOld); | 
					
						
							|  |  |  |       if (xor.length === 0) { | 
					
						
							|  |  |  |         window.console.log( | 
					
						
							|  |  |  |           'skipping group update: no detected changes in group member list' | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       window.doUpdateGroup( | 
					
						
							|  |  |  |         this.groupId, | 
					
						
							|  |  |  |         this.groupName, | 
					
						
							|  |  |  |         filteredMemberes, | 
					
						
							|  |  |  |         this.avatarPath | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     close() { | 
					
						
							|  |  |  |       this.remove(); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | })(); |