@ -31,6 +31,7 @@ class EditGroupViewModel: SessionTableViewModel, NavigatableStateHolder, Editabl
fileprivate var newGroupDescription : String ?
private var editDisplayPictureModal : ConfirmationModal ?
private var editDisplayPictureModalInfo : ConfirmationModal . Info ?
private var inviteByIdValue : String ?
// MARK: - I n i t i a l i z a t i o n
@ -68,6 +69,7 @@ class EditGroupViewModel: SessionTableViewModel, NavigatableStateHolder, Editabl
case groupDescription
case invite
case inviteById
case member ( String )
}
@ -301,8 +303,20 @@ class EditGroupViewModel: SessionTableViewModel, NavigatableStateHolder, Editabl
label : " Invite Contacts "
) ,
onTap : { [ weak self ] in self ? . inviteContacts ( currentGroupName : state . group . name ) }
) ,
( ! isUpdatedGroup || ! dependencies [ feature : . updatedGroupsAllowInviteById ] ? nil :
SessionCell . Info (
id : . inviteById ,
leadingAccessory : . icon ( UIImage ( named : " ic_plus_24 " ) ? . withRenderingMode ( . alwaysTemplate ) ) ,
title : " Invite Account ID or ONS " , // FIXME: L o c a l i s e t h i s
accessibility : Accessibility (
identifier : " Invite by id " ,
label : " Invite by id "
) ,
onTap : { [ weak self ] in self ? . inviteById ( ) }
)
)
]
] . compactMap { $0 }
) ,
SectionModel (
model : . members ,
@ -777,6 +791,127 @@ class EditGroupViewModel: SessionTableViewModel, NavigatableStateHolder, Editabl
)
}
private func inviteById ( ) {
// C o n v e n i e n c e f u n c t i o n s t o a v o i d d u p l i c a t e c o d e
func showError ( _ errorString : String ) {
let modal : ConfirmationModal = ConfirmationModal (
info : ConfirmationModal . Info (
title : " ALERT_ERROR_TITLE " . localized ( ) ,
body : . text ( errorString ) ,
cancelTitle : " BUTTON_OK " . localized ( ) ,
cancelStyle : . alert_text ,
dismissType : . single
)
)
self . transitionToScreen ( modal , transitionType : . present )
}
func inviteMember ( _ accountId : String , _ modal : UIViewController ) {
guard ! currentMemberIds . contains ( accountId ) else {
// FIXME: L o c a l i s e t h i s
return showError ( " This Account ID or ONS belongs to an existing member " )
}
MessageSender . addGroupMembers (
groupSessionId : threadId ,
members : [ ( accountId , nil ) ] ,
allowAccessToHistoricMessages : dependencies [ feature : . updatedGroupsAllowHistoricAccessOnInvite ] ,
using : dependencies
)
modal . dismiss ( animated : true ) { [ weak self ] in
self ? . showToast (
text : " GROUP_ACTION_INVITE_SENDING " . localized ( ) ,
backgroundColor : . backgroundSecondary
)
}
}
let currentMemberIds : Set < String > = ( tableData
. first ( where : { $0 . model = = . members } ) ?
. elements
. compactMap { item -> String ? in
switch item . id {
case . member ( let profileId ) : return profileId
default : return nil
}
} )
. defaulting ( to : [ ] )
. asSet ( )
// M a k e s u r e i n v i t i n g a n o t h e r m e m b e r w o u l d n ' t h i t t h e m e m b e r l i m i t
guard ( currentMemberIds . count + 1 ) <= SessionUtil . sizeMaxGroupMemberCount else {
return showError ( " vc_create_closed_group_too_many_group_members_error " . localized ( ) )
}
self . transitionToScreen (
ConfirmationModal (
info : ConfirmationModal . Info (
title : " Invite Account ID or ONS " , // FIXME: L o c a l i s e t h i s
body : . input (
explanation : nil ,
info : ConfirmationModal . Info . Body . InputInfo (
placeholder : " Enter Account ID or ONS " // FIXME: L o c a l i s e t h i s
) ,
onChange : { [ weak self ] updatedString in self ? . inviteByIdValue = updatedString }
) ,
confirmTitle : " Invite " , // FIXME: L o c a l i s e t h i s
confirmStyle : . danger ,
cancelStyle : . alert_text ,
dismissOnConfirm : false ,
onConfirm : { [ weak self ] modal in
// FIXME: C o n s o l i d a t e t h i s w i t h t h e l o g i c i n ` N e w D M V C `
switch Result ( catching : { try SessionId ( from : self ? . inviteByIdValue ) } ) {
case . success ( let sessionId ) where sessionId . prefix = = . standard :
inviteMember ( sessionId . hexString , modal )
case . success ( let sessionId ) where ( sessionId . prefix = = . blinded15 || sessionId . prefix = = . blinded25 ) :
// FIXME: L o c a l i s e t h i s
return showError ( " Unable to invite members using their Blinded IDs " )
case . success :
// FIXME: L o c a l i s e t h i s
return showError ( " The value entered is not a valid Account ID or ONS " )
case . failure :
guard let inviteByIdValue : String = self ? . inviteByIdValue else {
// FIXME: L o c a l i s e t h i s
return showError ( " Please enter a valid Account ID or ONS " )
}
// T h i s c o u l d b e a n O N S n a m e
let viewController = ModalActivityIndicatorViewController ( ) { modalActivityIndicator in
SnodeAPI
. getSessionID ( for : inviteByIdValue )
. subscribe ( on : DispatchQueue . global ( qos : . userInitiated ) )
. receive ( on : DispatchQueue . main )
. sinkUntilComplete (
receiveCompletion : { result in
switch result {
case . finished : break
case . failure :
modalActivityIndicator . dismiss {
// FIXME: L o c a l i s e t h i s
return showError ( " Unable to find ONS provided. " )
}
}
} ,
receiveValue : { sessionIdHexString in
modalActivityIndicator . dismiss {
inviteMember ( sessionIdHexString , modal )
}
}
)
}
self ? . transitionToScreen ( viewController , transitionType : . present )
}
} ,
afterClosed : { [ weak self ] in self ? . inviteByIdValue = nil }
)
) ,
transitionType : . present
)
}
private func resendInvitation ( memberId : String ) {
MessageSender . resendInvitation (
groupSessionId : threadId ,