mirror of
https://github.com/danog/Telegram.git
synced 2024-12-12 09:29:55 +01:00
1121 lines
40 KiB
Objective-C
1121 lines
40 KiB
Objective-C
#import "TGPrivateModernConversationCompanion.h"
|
|
|
|
#import "TGDatabase.h"
|
|
#import "TGDateUtils.h"
|
|
#import "TGStringUtils.h"
|
|
#import "TGPhoneUtils.h"
|
|
#import "TGDialogListCompanion.h"
|
|
|
|
#import "ActionStage.h"
|
|
#import "SGraphObjectNode.h"
|
|
|
|
#import "TGAppDelegate.h"
|
|
#import "TGTelegraph.h"
|
|
#import "TGPopoverController.h"
|
|
|
|
#import "TGInterfaceManager.h"
|
|
#import "TGNavigationBar.h"
|
|
#import "TGTelegraphUserInfoController.h"
|
|
|
|
#import "TGModernConversationController.h"
|
|
#import "TGModernConversationActionInputPanel.h"
|
|
#import "TGModernConversationPrivateTitlePanel.h"
|
|
#import "TGModernConversationContactLinkTitlePanel.h"
|
|
|
|
#import "TGModernConversationTitleIcon.h"
|
|
|
|
#import "TGModernConversationTitleView.h"
|
|
|
|
#import "TGProgressWindow.h"
|
|
|
|
#import "TGBotUserInfoController.h"
|
|
|
|
#import "TGBotSignals.h"
|
|
#import "TGPeerInfoSignals.h"
|
|
|
|
#import "TGBotConversationHeaderView.h"
|
|
|
|
#import "TGModernViewContext.h"
|
|
|
|
#import "TGServiceSignals.h"
|
|
|
|
#import "TGMessageModernConversationItem.h"
|
|
|
|
#import "TGActionSheet.h"
|
|
|
|
#import "TGRecentContextBotsSignal.h"
|
|
|
|
typedef enum {
|
|
TGPhoneSharingStatusUnknown = 0,
|
|
TGPhoneSharingStatusNotShared = 1,
|
|
TGPhoneSharingStatusMyShared = 2
|
|
} TGPhoneSharingStatus;
|
|
|
|
static NSMutableDictionary *dismissedContactLinkPanelsByUserId()
|
|
{
|
|
static NSMutableDictionary *dict = nil;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^
|
|
{
|
|
dict = [[NSMutableDictionary alloc] init];
|
|
});
|
|
return dict;
|
|
}
|
|
|
|
@interface TGPrivateModernConversationCompanion () <TGModernConversationContactLinkTitlePanelDelegate>
|
|
{
|
|
NSString *_initialActivity;
|
|
|
|
NSString *_cachedPhone;
|
|
|
|
bool _isLoadingFirstMessages;
|
|
|
|
bool _hasUnblockPanel;
|
|
bool _hasOutgoingMessages;
|
|
bool _hasIncomingMessages;
|
|
|
|
bool _isBlocked; // Main Thread
|
|
bool _isContact; // Main Thread
|
|
|
|
bool _isMuted; // Main Thread
|
|
|
|
NSArray *_additionalTitleIcons; // Main Thread
|
|
|
|
TGPhoneSharingStatus _phoneSharingStatus; // Main Thread
|
|
|
|
TGBotConversationHeaderView *_botHeaderView; // Main Thread
|
|
|
|
bool _isBot;
|
|
id<SDisposable> _botInfoDisposable;
|
|
TGBotInfo *_botInfo;
|
|
|
|
TGModernConversationActionInputPanel *_unblockPanel;
|
|
TGModernConversationActionInputPanel *_botStartPanel; // Main Thread
|
|
bool _botStartPanelDismissed;
|
|
|
|
SMetaDisposable *_botStartDisposable;
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation TGPrivateModernConversationCompanion
|
|
|
|
- (instancetype)initWithUid:(int)uid activity:(NSString *)activity mayHaveUnreadMessages:(bool)mayHaveUnreadMessages
|
|
{
|
|
return [self initWithConversationId:uid uid:uid activity:activity mayHaveUnreadMessages:mayHaveUnreadMessages];
|
|
}
|
|
|
|
- (instancetype)initWithConversationId:(int64_t)conversationId uid:(int)uid activity:(NSString *)activity mayHaveUnreadMessages:(bool)mayHaveUnreadMessages
|
|
{
|
|
self = [super initWithConversationId:conversationId mayHaveUnreadMessages:mayHaveUnreadMessages];
|
|
if (self != nil)
|
|
{
|
|
_uid = uid;
|
|
_initialActivity = activity;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[_botInfoDisposable dispose];
|
|
[_botStartDisposable dispose];
|
|
}
|
|
|
|
- (void)setAdditionalTitleIcons:(NSArray *)additionalTitleIcons
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
_additionalTitleIcons = additionalTitleIcons;
|
|
[self _updateTitleIcons];
|
|
});
|
|
}
|
|
|
|
- (bool)shouldDisplayContactLinkPanel
|
|
{
|
|
if (_conversationId == [TGTelegraphInstance serviceUserUid])
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
- (void)_updateTitleIcons
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
NSMutableArray *icons = [[NSMutableArray alloc] initWithArray:_additionalTitleIcons];
|
|
|
|
if (_isMuted)
|
|
{
|
|
TGModernConversationTitleIcon *muteIcon = [[TGModernConversationTitleIcon alloc] init];
|
|
muteIcon.bounds = CGRectMake(0.0f, 0.0f, 16, 16);
|
|
muteIcon.offsetWeight = 0.5f;
|
|
muteIcon.imageOffset = CGPointMake(4.0f, 7.0f);
|
|
|
|
static UIImage *muteImage = nil;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^
|
|
{
|
|
muteImage = [UIImage imageNamed:@"ModernConversationTitleIconMute.png"];
|
|
});
|
|
|
|
muteIcon.image = muteImage;
|
|
muteIcon.iconPosition = TGModernConversationTitleIconPositionAfterTitle;
|
|
|
|
[icons addObject:muteIcon];
|
|
}
|
|
|
|
[self _setTitleIcons:icons];
|
|
});
|
|
}
|
|
|
|
- (void)_updateUserMute:(bool)isMuted
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
if (_isMuted != isMuted)
|
|
{
|
|
_isMuted = isMuted;
|
|
[self _updateTitleIcons];
|
|
}
|
|
});
|
|
}
|
|
|
|
- (void)_updatePhoneSharingStatusFromUserLink:(int)userLink
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
TGPhoneSharingStatus phoneSharingStatus = TGPhoneSharingStatusUnknown;
|
|
if (userLink & TGUserLinkKnown)
|
|
{
|
|
if (userLink & (TGUserLinkForeignHasPhone | TGUserLinkForeignMutual | TGUserLinkMyRequested))
|
|
phoneSharingStatus = TGPhoneSharingStatusMyShared;
|
|
else
|
|
phoneSharingStatus = TGPhoneSharingStatusNotShared;
|
|
}
|
|
|
|
if (phoneSharingStatus != _phoneSharingStatus)
|
|
{
|
|
_phoneSharingStatus = phoneSharingStatus;
|
|
[self _updateContactLinkPanel];
|
|
}
|
|
});
|
|
}
|
|
|
|
- (void)_updateContactLinkPanel
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
if (![self shouldDisplayContactLinkPanel])
|
|
return;
|
|
|
|
TGModernConversationController *controller = self.controller;
|
|
|
|
TGModernConversationContactLinkTitlePanel *panel = [controller.secondaryTitlePanel isKindOfClass:[TGModernConversationContactLinkTitlePanel class]] ? (TGModernConversationContactLinkTitlePanel *)controller.secondaryTitlePanel : nil;
|
|
TGUser *user = [TGDatabaseInstance() loadUser:_uid];
|
|
|
|
if (!_isContact && user.phoneNumber.length != 0)
|
|
{
|
|
NSMutableDictionary *dict = dismissedContactLinkPanelsByUserId()[@(TGTelegraphInstance.clientUserId)];
|
|
if (dict == nil)
|
|
{
|
|
dict = [[NSMutableDictionary alloc] init];
|
|
dismissedContactLinkPanelsByUserId()[@(TGTelegraphInstance.clientUserId)] = dict;
|
|
}
|
|
if (![dict[[[NSString alloc] initWithFormat:@"%" PRId32 "_%@", _uid, @"add"]] boolValue])
|
|
{
|
|
if (panel == nil)
|
|
{
|
|
panel = [[TGModernConversationContactLinkTitlePanel alloc] init];
|
|
panel.delegate = self;
|
|
}
|
|
|
|
[panel setShareContact:false addContact:true reportSpam:_hasIncomingMessages && !_hasOutgoingMessages && !_isBot];
|
|
}
|
|
}
|
|
else if (!_isContact && _hasIncomingMessages && !_hasOutgoingMessages && !_isBot) {
|
|
NSMutableDictionary *dict = dismissedContactLinkPanelsByUserId()[@(TGTelegraphInstance.clientUserId)];
|
|
if (dict == nil)
|
|
{
|
|
dict = [[NSMutableDictionary alloc] init];
|
|
dismissedContactLinkPanelsByUserId()[@(TGTelegraphInstance.clientUserId)] = dict;
|
|
}
|
|
if (![dict[[[NSString alloc] initWithFormat:@"%" PRId32 "_%@", _uid, @"add"]] boolValue])
|
|
{
|
|
if (panel == nil)
|
|
{
|
|
panel = [[TGModernConversationContactLinkTitlePanel alloc] init];
|
|
panel.delegate = self;
|
|
}
|
|
|
|
[panel setShareContact:false addContact:false reportSpam:true];
|
|
}
|
|
} else {
|
|
panel = nil;
|
|
}
|
|
|
|
[controller setSecondaryTitlePanel:panel];
|
|
});
|
|
}
|
|
|
|
- (void)contactLinkTitlePanelShareContactPressed:(TGModernConversationContactLinkTitlePanel *)panel
|
|
{
|
|
[self contactLinkTitlePanelDismissed:panel];
|
|
|
|
[self shareVCard];
|
|
}
|
|
|
|
- (void)contactLinkTitlePanelAddContactPressed:(TGModernConversationContactLinkTitlePanel *)__unused panel
|
|
{
|
|
TGUser *contact = [TGDatabaseInstance() loadUser:_uid];
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
if (contact != nil)
|
|
{
|
|
TGModernConversationController *controller = self.controller;
|
|
[controller showAddContactMenu:contact];
|
|
}
|
|
});
|
|
}
|
|
|
|
- (void)contactLinkTitlePanelBlockContactPressed:(TGModernConversationContactLinkTitlePanel *)__unused panel {
|
|
TGModernConversationController *controller = self.controller;
|
|
|
|
int64_t conversationId = _conversationId;
|
|
TGActionSheet *actionSheet = [[TGActionSheet alloc] initWithTitle:TGLocalized(@"Conversation.ReportSpamConfirmation") actions:@[[[TGActionSheetAction alloc] initWithTitle:TGLocalized(@"Conversation.ReportSpam") action:@"reportSpam" type:TGActionSheetActionTypeDestructive],
|
|
[[TGActionSheetAction alloc] initWithTitle:TGLocalized(@"Common.Cancel") action:@"cancel" type:TGActionSheetActionTypeCancel]
|
|
] actionBlock:^(TGModernConversationController *controller, NSString *action) {
|
|
if ([action isEqualToString:@"reportSpam"]) {
|
|
TGUser *user = [TGDatabaseInstance() loadUser:(int32_t)conversationId];
|
|
SMetaDisposable *metaDisposable = [[SMetaDisposable alloc] init];
|
|
id<SDisposable> disposable = [[[TGServiceSignals reportSpam:conversationId accessHash:user.phoneNumberHash] onDispose:^{
|
|
[TGTelegraphInstance.disposeOnLogout remove:metaDisposable];
|
|
}] startWithNext:nil];
|
|
[metaDisposable setDisposable:disposable];
|
|
[TGTelegraphInstance.disposeOnLogout add:metaDisposable];
|
|
|
|
[TGAppDelegateInstance.rootController.dialogListController.dialogListCompanion deleteItem:[[TGConversation alloc] initWithConversationId:_conversationId unreadCount:0 serviceUnreadCount:0] animated:false];
|
|
|
|
[controller.navigationController popToRootViewControllerAnimated:true];
|
|
}
|
|
} target:controller];
|
|
|
|
if (TGAppDelegateInstance.rootController.currentSizeClass == UIUserInterfaceSizeClassRegular) {
|
|
[actionSheet showFromRect:[controller.view convertRect:panel.bounds fromView:panel] inView:controller.view animated:true];
|
|
} else {
|
|
[actionSheet showInView:controller.view];
|
|
}
|
|
}
|
|
|
|
- (void)contactLinkTitlePanelDismissed:(TGModernConversationContactLinkTitlePanel *)panel
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
NSMutableDictionary *dict = dismissedContactLinkPanelsByUserId()[@(TGTelegraphInstance.clientUserId)];
|
|
if (dict == nil)
|
|
{
|
|
dict = [[NSMutableDictionary alloc] init];
|
|
dismissedContactLinkPanelsByUserId()[@(TGTelegraphInstance.clientUserId)] = dict;
|
|
}
|
|
dict[[[NSString alloc] initWithFormat:@"%" PRId32 "_%@", _uid, panel.shareContact ? @"share" : @"add"]] = @(true);
|
|
|
|
TGModernConversationController *controller = self.controller;
|
|
[controller setSecondaryTitlePanel:nil];
|
|
});
|
|
}
|
|
|
|
#pragma mark -
|
|
|
|
- (void)_controllerWillAppearAnimated:(bool)animated firstTime:(bool)firstTime
|
|
{
|
|
[super _controllerWillAppearAnimated:animated firstTime:firstTime];
|
|
|
|
if (firstTime)
|
|
{
|
|
[ActionStageInstance() dispatchOnStageQueue:^
|
|
{
|
|
[ActionStageInstance() requestActor:[[NSString alloc] initWithFormat:@"/tg/blockedUsers/(%" PRId32 ",cached)", _uid] options:@{@"uid": @(_uid)} watcher:self];
|
|
[ActionStageInstance() requestActor:[[NSString alloc] initWithFormat:@"/tg/completeUsers/(%" PRId32 ",cached)", _uid] options:@{@"uid": @(_uid)} watcher:TGTelegraphInstance];
|
|
}];
|
|
|
|
[TGDatabaseInstance() dispatchOnDatabaseThread:^
|
|
{
|
|
bool outdated = false;
|
|
int userLink = [TGDatabaseInstance() loadUserLink:_uid outdated:&outdated];
|
|
[self _updatePhoneSharingStatusFromUserLink:userLink];
|
|
} synchronous:false];
|
|
}
|
|
}
|
|
|
|
#pragma mark -
|
|
|
|
- (void)_controllerAvatarPressed
|
|
{
|
|
__weak TGPrivateModernConversationCompanion *weakSelf = self;
|
|
void (^shareVCard)() = ^
|
|
{
|
|
__strong TGPrivateModernConversationCompanion *strongSelf = weakSelf;
|
|
if (strongSelf != nil)
|
|
{
|
|
[strongSelf shareVCard];
|
|
TGProgressWindow *progressWindow = [[TGProgressWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
|
|
[progressWindow show:false];
|
|
[progressWindow dismissWithSuccess];
|
|
}
|
|
};
|
|
|
|
TGModernConversationController *controller = self.controller;
|
|
|
|
if (controller.currentSizeClass == UIUserInterfaceSizeClassCompact)
|
|
{
|
|
TGUser *user = [TGDatabaseInstance() loadUser:_uid];
|
|
|
|
TGCollectionMenuController *userInfoController = nil;
|
|
|
|
if (user.kind == TGUserKindBot || user.kind == TGUserKindSmartBot)
|
|
{
|
|
__weak TGPrivateModernConversationCompanion *weakSelf = self;
|
|
userInfoController = [[TGBotUserInfoController alloc] initWithUid:_uid sendCommand:^(NSString *command)
|
|
{
|
|
__strong TGPrivateModernConversationCompanion *strongSelf = weakSelf;
|
|
if (strongSelf != nil)
|
|
{
|
|
[strongSelf controllerWantsToSendTextMessage:command entities:nil asReplyToMessageId:0 withAttachedMessages:nil disableLinkPreviews:false botContextResult:nil];
|
|
}
|
|
}];
|
|
}
|
|
else
|
|
{
|
|
userInfoController = [[TGTelegraphUserInfoController alloc] initWithUid:_uid];
|
|
((TGTelegraphUserInfoController *)userInfoController).shareVCard = shareVCard;
|
|
}
|
|
|
|
[controller.navigationController pushViewController:userInfoController animated:true];
|
|
}
|
|
else
|
|
{
|
|
if (controller != nil)
|
|
{
|
|
TGUser *user = [TGDatabaseInstance() loadUser:_uid];
|
|
|
|
TGCollectionMenuController *userInfoController = nil;
|
|
|
|
if (user.kind == TGUserKindBot || user.kind == TGUserKindSmartBot)
|
|
{
|
|
__weak TGPrivateModernConversationCompanion *weakSelf = self;
|
|
userInfoController = [[TGBotUserInfoController alloc] initWithUid:_uid sendCommand:^(NSString *command)
|
|
{
|
|
__strong TGPrivateModernConversationCompanion *strongSelf = weakSelf;
|
|
if (strongSelf != nil)
|
|
{
|
|
[strongSelf controllerWantsToSendTextMessage:command entities:nil asReplyToMessageId:0 withAttachedMessages:nil disableLinkPreviews:false botContextResult:nil];
|
|
}
|
|
}];
|
|
}
|
|
else
|
|
{
|
|
userInfoController = [[TGTelegraphUserInfoController alloc] initWithUid:_uid withoutCompose:true];
|
|
((TGTelegraphUserInfoController *)userInfoController).shareVCard = shareVCard;
|
|
}
|
|
|
|
TGNavigationController *navigationController = [TGNavigationController navigationControllerWithControllers:@[userInfoController] navigationBarClass:[TGWhiteNavigationBar class]];
|
|
navigationController.presentationStyle = TGNavigationControllerPresentationStyleRootInPopover;
|
|
TGPopoverController *popoverController = [[TGPopoverController alloc] initWithContentViewController:navigationController];
|
|
navigationController.parentPopoverController = popoverController;
|
|
navigationController.detachFromPresentingControllerInCompactMode = true;
|
|
[popoverController setContentSize:CGSizeMake(320.0f, 528.0f)];
|
|
|
|
controller.associatedPopoverController = popoverController;
|
|
[popoverController presentPopoverFromBarButtonItem:controller.navigationItem.rightBarButtonItem permittedArrowDirections:UIPopoverArrowDirectionAny animated:true];
|
|
userInfoController.collectionView.contentOffset = CGPointMake(0.0f, -userInfoController.collectionView.contentInset.top);
|
|
}
|
|
}
|
|
}
|
|
|
|
- (NSString *)stringForTitle:(TGUser *)user isContact:(bool)isContact
|
|
{
|
|
if (user.uid == [TGTelegraphInstance serviceUserUid])
|
|
return @"Telegram";
|
|
|
|
if (isContact || user.phoneNumber.length == 0)
|
|
return user.displayName;
|
|
|
|
if (_cachedPhone == nil)
|
|
_cachedPhone = [TGPhoneUtils formatPhone:user.phoneNumber forceInternational:true];
|
|
|
|
return _cachedPhone;
|
|
}
|
|
|
|
- (NSString *)statusStringForUser:(TGUser *)user accentColored:(bool *)accentColored
|
|
{
|
|
if (user.kind == TGUserKindBot || user.kind == TGUserKindSmartBot)
|
|
return TGLocalized(@"Bot.GenericBotStatus");
|
|
|
|
if (user.uid == [TGTelegraphInstance serviceUserUid])
|
|
return TGLocalized(@"Core.ServiceUserStatus");
|
|
|
|
if (user.presence.online)
|
|
{
|
|
if (accentColored != NULL)
|
|
*accentColored = true;
|
|
return TGLocalizedStatic(@"Presence.online");
|
|
}
|
|
else if (user.presence.lastSeen != 0)
|
|
return [TGDateUtils stringForRelativeLastSeen:user.presence.lastSeen];
|
|
|
|
return TGLocalized(@"Presence.offline");
|
|
}
|
|
|
|
- (NSString *)stringForActivity:(NSString *)activity
|
|
{
|
|
if ([activity isEqualToString:@"recordingAudio"])
|
|
return TGLocalized(@"Activity.RecordingAudio");
|
|
else if ([activity isEqualToString:@"uploadingPhoto"])
|
|
return TGLocalized(@"Activity.UploadingPhoto");
|
|
else if ([activity isEqualToString:@"uploadingVideo"])
|
|
return TGLocalized(@"Activity.UploadingVideo");
|
|
else if ([activity isEqualToString:@"uploadingDocument"])
|
|
return TGLocalized(@"Activity.UploadingDocument");
|
|
else if ([activity isEqualToString:@"pickingLocation"])
|
|
return nil;
|
|
|
|
return TGLocalized(@"Conversation.typing");
|
|
}
|
|
|
|
- (int)activityTypeForActivity:(NSString *)activity
|
|
{
|
|
if ([activity isEqualToString:@"recordingAudio"])
|
|
return TGModernConversationTitleViewActivityAudioRecording;
|
|
else if ([activity isEqualToString:@"uploadingPhoto"])
|
|
return TGModernConversationTitleViewActivityUploading;
|
|
else if ([activity isEqualToString:@"uploadingVideo"])
|
|
return TGModernConversationTitleViewActivityUploading;
|
|
else if ([activity isEqualToString:@"uploadingDocument"])
|
|
return TGModernConversationTitleViewActivityUploading;
|
|
else if ([activity isEqualToString:@"pickingLocation"])
|
|
return 0;
|
|
|
|
return TGModernConversationTitleViewActivityTyping;
|
|
}
|
|
|
|
- (void)loadInitialState
|
|
{
|
|
TGUser *user = [TGDatabaseInstance() loadUser:_uid];
|
|
if (user.kind == TGUserKindBot || user.kind == TGUserKindSmartBot)
|
|
_isBot = true;
|
|
_isContact = [TGDatabaseInstance() uidIsRemoteContact:_uid];
|
|
|
|
self.viewContext.commandsEnabled = _isBot;
|
|
self.viewContext.isBot = _isBot;
|
|
|
|
_isLoadingFirstMessages = true;
|
|
[super loadInitialState];
|
|
_isLoadingFirstMessages = false;
|
|
|
|
[self _setTitle:[self stringForTitle:user isContact:_isContact]];
|
|
[self _setAvatarConversationId:_uid firstName:user.firstName lastName:user.lastName];
|
|
[self _setAvatarUrl:user.photoUrlSmall];
|
|
bool accentColored = false;
|
|
NSString *statusString = [self statusStringForUser:user accentColored:&accentColored];
|
|
[self _setStatus:statusString accentColored:accentColored allowAnimation:false toggleMode:TGModernConversationControllerTitleToggleNone];
|
|
|
|
if (_initialActivity != nil)
|
|
[self _setTypingStatus:[self stringForActivity:_initialActivity] activity:[self activityTypeForActivity:_initialActivity]];
|
|
|
|
if (_isBot)
|
|
{
|
|
__weak TGPrivateModernConversationCompanion *weakSelf = self;
|
|
_botInfoDisposable = [[[TGBotSignals botInfoForUserId:_uid] deliverOn:[SQueue mainQueue]] startWithNext:^(TGBotInfo *botInfo)
|
|
{
|
|
__strong TGPrivateModernConversationCompanion *strongSelf = weakSelf;
|
|
if (strongSelf != nil)
|
|
{
|
|
strongSelf->_botInfo = botInfo;
|
|
TGModernConversationController *controller = strongSelf.controller;
|
|
if (strongSelf->_botHeaderView == nil)
|
|
[controller setConversationHeader:[strongSelf _conversationHeader]];
|
|
}
|
|
}];
|
|
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
TGModernConversationController *controller = self.controller;
|
|
[controller setHasBots:true];
|
|
});
|
|
}
|
|
|
|
[self _updateContactLinkPanel];
|
|
}
|
|
|
|
#pragma mark -
|
|
|
|
- (TGModernConversationEmptyListPlaceholderView *)_conversationEmptyListPlaceholder
|
|
{
|
|
if (_isBot)
|
|
return nil;
|
|
else
|
|
return [super _conversationEmptyListPlaceholder];
|
|
}
|
|
|
|
- (TGModernConversationInputPanel *)_conversationEmptyListInputPanel
|
|
{
|
|
if (_unblockPanel != nil)
|
|
return _unblockPanel;
|
|
|
|
if (_isBot)
|
|
{
|
|
if (_botStartPanel == nil && !_botStartPanelDismissed)
|
|
{
|
|
TGModernConversationController *controller = self.controller;
|
|
_botStartPanel = [[TGModernConversationActionInputPanel alloc] init];
|
|
[_botStartPanel setActionWithTitle:TGLocalized(@"Bot.Start") action:@"botstart" color:TGAccentColor()];
|
|
_botStartPanel.companionHandle = self.actionHandle;
|
|
_botStartPanel.delegate = controller;
|
|
}
|
|
return _botStartPanel;
|
|
}
|
|
else
|
|
return _unblockPanel;
|
|
}
|
|
|
|
- (TGModernConversationInputPanel *)_conversationGenericInputPanel
|
|
{
|
|
if (_unblockPanel != nil)
|
|
return _unblockPanel;
|
|
|
|
if (_isBot && _botStartPayload != nil && !_botStartPanelDismissed)
|
|
{
|
|
if (_botStartPanel == nil)
|
|
{
|
|
TGModernConversationController *controller = self.controller;
|
|
_botStartPanel = [[TGModernConversationActionInputPanel alloc] init];
|
|
[_botStartPanel setActionWithTitle:TGLocalized(@"Bot.Start") action:@"botstart" color:TGAccentColor()];
|
|
_botStartPanel.companionHandle = self.actionHandle;
|
|
_botStartPanel.delegate = controller;
|
|
}
|
|
return _botStartPanel;
|
|
}
|
|
return _unblockPanel;
|
|
}
|
|
|
|
- (UIView *)_conversationHeader
|
|
{
|
|
if (_botInfo != nil && _botInfo.botDescription.length != 0)
|
|
{
|
|
if (_botHeaderView == nil)
|
|
{
|
|
_botHeaderView = [[TGBotConversationHeaderView alloc] initWithContext:self.viewContext botInfo:_botInfo];
|
|
[_botHeaderView sizeToFit];
|
|
}
|
|
return _botHeaderView;
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
- (bool)imageDownloadsShouldAutosavePhotos
|
|
{
|
|
return TGAppDelegateInstance.autosavePhotos;
|
|
}
|
|
|
|
- (bool)shouldAutomaticallyDownloadPhotos
|
|
{
|
|
return TGAppDelegateInstance.autoDownloadPhotosInPrivateChats;
|
|
}
|
|
|
|
- (bool)shouldAutomaticallyDownloadAnimations
|
|
{
|
|
return TGAppDelegateInstance.autoPlayAnimations;
|
|
}
|
|
|
|
- (bool)shouldAutomaticallyDownloadAudios
|
|
{
|
|
return TGAppDelegateInstance.autoDownloadAudioInPrivateChats;
|
|
}
|
|
|
|
- (NSString *)_sendMessagePathForMessageId:(int32_t)mid
|
|
{
|
|
return [[NSString alloc] initWithFormat:@"/tg/sendCommonMessage/(%@)/(%d)", [self _conversationIdPathComponent], mid];
|
|
}
|
|
|
|
- (NSString *)_sendMessagePathPrefix
|
|
{
|
|
return [[NSString alloc] initWithFormat:@"/tg/sendCommonMessage/(%@)/", [self _conversationIdPathComponent]];
|
|
}
|
|
|
|
- (NSDictionary *)_optionsForMessageActions
|
|
{
|
|
return @{@"conversationId": @(_conversationId)};
|
|
}
|
|
|
|
- (void)subscribeToUpdates
|
|
{
|
|
[ActionStageInstance() watchForPaths:@[
|
|
[[NSString alloc] initWithFormat:@"/tg/conversation/(%" PRId64 ")/typing", _conversationId],
|
|
[[NSString alloc] initWithFormat:@"/tg/userLink/(%" PRId32 ")", _uid],
|
|
@"/tg/blockedUsers"
|
|
] watcher:self];
|
|
|
|
[ActionStageInstance() watchForPath:[NSString stringWithFormat:@"/tg/peerSettings/(%" PRId32 ")", _uid] watcher:self];
|
|
[ActionStageInstance() requestActor:[NSString stringWithFormat:@"/tg/peerSettings/(%" PRId32 ",cachedOnly)", _uid] options:@{@"peerId": @(_uid)} watcher:self];
|
|
|
|
[super subscribeToUpdates];
|
|
}
|
|
|
|
#pragma mark -
|
|
|
|
- (void)_createOrUpdatePrimaryTitlePanel:(bool)createIfNeeded
|
|
{
|
|
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPhone)
|
|
{
|
|
TGModernConversationController *controller = self.controller;
|
|
|
|
TGModernConversationPrivateTitlePanel *privateTitlePanel = nil;
|
|
if ([[controller primaryTitlePanel] isKindOfClass:[TGModernConversationPrivateTitlePanel class]])
|
|
privateTitlePanel = (TGModernConversationPrivateTitlePanel *)[controller primaryTitlePanel];
|
|
else
|
|
{
|
|
if (createIfNeeded)
|
|
{
|
|
privateTitlePanel = [[TGModernConversationPrivateTitlePanel alloc] init];
|
|
privateTitlePanel.companionHandle = self.actionHandle;
|
|
}
|
|
else
|
|
return;
|
|
}
|
|
|
|
if (_isContact)
|
|
{
|
|
[privateTitlePanel setButtonsWithTitlesAndActions:@[
|
|
@{@"title": TGLocalized(@"Conversation.Search"), @"action": @"search"},
|
|
//@{@"title": TGLocalized(@"Conversation.Call"), @"action": @"call"},
|
|
@{@"title": TGLocalized(@"Common.Edit"), @"action": @"edit"},
|
|
@{@"title": TGLocalized(@"Conversation.Info"), @"action": @"info"},
|
|
]];
|
|
}
|
|
else
|
|
{
|
|
NSMutableArray *actions = [[NSMutableArray alloc] init];
|
|
/*if (_isBlocked)
|
|
[actions addObject:@{@"title": TGLocalized(@"Conversation.Unblock"), @"action": @"unblock"}];
|
|
else
|
|
[actions addObject:@{@"title": TGLocalized(@"Conversation.Block"), @"action": @"block"}];*/
|
|
[actions addObject:@{@"title": TGLocalized(@"Conversation.Search"), @"action": @"search"}];
|
|
[actions addObject:@{@"title": TGLocalized(@"Common.Edit"), @"action": @"edit"}];
|
|
[actions addObject:@{@"title": TGLocalized(@"Conversation.Info"), @"action": @"info"}];
|
|
[privateTitlePanel setButtonsWithTitlesAndActions:actions];
|
|
}
|
|
|
|
[controller setPrimaryTitlePanel:privateTitlePanel];
|
|
}
|
|
}
|
|
|
|
- (void)_loadControllerPrimaryTitlePanel
|
|
{
|
|
[self _createOrUpdatePrimaryTitlePanel:true];
|
|
}
|
|
|
|
- (void)controllerDidUpdateTypingActivity
|
|
{
|
|
[ActionStageInstance() dispatchOnStageQueue:^
|
|
{
|
|
if ((![TGDatabaseInstance() uidIsRemoteContact:_uid] || [TGDatabaseInstance() loadUser:_uid].presence.online || [TGDatabaseInstance() loadUser:_uid].presence.lastSeen <= 0))
|
|
{
|
|
[[TGTelegraphInstance activityManagerForConversationId:_conversationId accessHash:[self requestAccessHash]] addActivityWithType:@"typing" priority:10 timeout:5.0];
|
|
}
|
|
}];
|
|
}
|
|
|
|
- (void)controllerDidCancelTypingActivity
|
|
{
|
|
[ActionStageInstance() dispatchOnStageQueue:^
|
|
{
|
|
[[TGTelegraphInstance activityManagerForConversationId:_conversationId accessHash:[self requestAccessHash]] removeActivityWithType:@"typing"];
|
|
}];
|
|
}
|
|
|
|
- (void)requestUserBlocked:(bool)blocked
|
|
{
|
|
[ActionStageInstance() dispatchOnStageQueue:^
|
|
{
|
|
_hasUnblockPanel = blocked;
|
|
|
|
static int actionId = 0;
|
|
[ActionStageInstance() requestActor:[[NSString alloc] initWithFormat:@"/tg/changePeerBlockedStatus/(cb%d)", actionId++] options:@{@"peerId": @(_uid), @"block": @(blocked)} watcher:TGTelegraphInstance];
|
|
|
|
if (!blocked && _isBot)
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
[self requestBotStart];
|
|
});
|
|
}
|
|
}];
|
|
|
|
[self updateUserBlocked:blocked];
|
|
}
|
|
|
|
- (void)requestBotStart
|
|
{
|
|
if (_botStartPayload != nil)
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
[_botStartPanel setActivity:true];
|
|
if (_botStartDisposable == nil)
|
|
_botStartDisposable = [[SMetaDisposable alloc] init];
|
|
__weak TGPrivateModernConversationCompanion *weakSelf = self;
|
|
[_botStartDisposable setDisposable:[[[[TGBotSignals botStartForUserId:_uid payload:_botStartPayload] deliverOn:[SQueue mainQueue]] onDispose:^
|
|
{
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
__strong TGPrivateModernConversationCompanion *strongSelf = weakSelf;
|
|
if (strongSelf != nil)
|
|
{
|
|
[strongSelf->_botStartPanel setActivity:false];
|
|
}
|
|
});
|
|
}] startWithNext:nil completed:^
|
|
{
|
|
__strong TGPrivateModernConversationCompanion *strongSelf = weakSelf;
|
|
if (strongSelf != nil)
|
|
{
|
|
strongSelf->_botStartPanel = nil;
|
|
strongSelf->_botStartPanelDismissed = true;
|
|
[strongSelf _updateInputPanel];
|
|
}
|
|
}]];
|
|
});
|
|
}
|
|
else
|
|
{
|
|
[self controllerWantsToSendTextMessage:@"/start" entities:nil asReplyToMessageId:0 withAttachedMessages:nil disableLinkPreviews:false botContextResult:nil];
|
|
}
|
|
}
|
|
|
|
- (void)updateUserBlocked:(bool)blocked
|
|
{
|
|
if (_hasUnblockPanel != blocked)
|
|
{
|
|
_hasUnblockPanel = blocked;
|
|
|
|
ASHandle *actionHandle = self.actionHandle;
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
_isBlocked = blocked;
|
|
[self _createOrUpdatePrimaryTitlePanel:false];
|
|
|
|
TGModernConversationController *controller = self.controller;
|
|
if (blocked)
|
|
{
|
|
if (_unblockPanel == nil)
|
|
{
|
|
_unblockPanel = [[TGModernConversationActionInputPanel alloc] init];
|
|
[_unblockPanel setActionWithTitle:_isBot ? TGLocalized(@"Bot.Unblock") : TGLocalized(@"Conversation.Unblock") action:@"unblock"];
|
|
_unblockPanel.delegate = controller;
|
|
_unblockPanel.companionHandle = actionHandle;
|
|
}
|
|
}
|
|
else
|
|
_unblockPanel = nil;
|
|
|
|
[self _updateInputPanel];
|
|
});
|
|
}
|
|
}
|
|
|
|
- (NSString *)_controllerInfoButtonText
|
|
{
|
|
return TGLocalized(@"Conversation.InfoPrivate");
|
|
}
|
|
|
|
- (NSDictionary *)userActivityData
|
|
{
|
|
NSMutableDictionary *peerDict = [[NSMutableDictionary alloc] init];
|
|
peerDict[@"type"] = @"user";
|
|
peerDict[@"id"] = @((int32_t)_conversationId);
|
|
TGUser *user = [TGDatabaseInstance() loadUser:(int32_t)_conversationId];
|
|
if (user.userName.length != 0)
|
|
peerDict[@"username"] = user.userName;
|
|
return @{@"user_id": @(TGTelegraphInstance.clientUserId), @"peer": peerDict};
|
|
}
|
|
|
|
- (TGApplicationFeaturePeerType)applicationFeaturePeerType
|
|
{
|
|
return TGApplicationFeaturePeerPrivate;
|
|
}
|
|
|
|
#pragma mark -
|
|
|
|
- (void)actionStageActionRequested:(NSString *)action options:(id)options
|
|
{
|
|
if ([action isEqualToString:@"actionPanelAction"])
|
|
{
|
|
NSString *panelAction = options[@"action"];
|
|
if ([panelAction isEqualToString:@"unblock"])
|
|
[self requestUserBlocked:false];
|
|
else if ([panelAction isEqualToString:@"botstart"])
|
|
[self requestBotStart];
|
|
}
|
|
else if ([action isEqualToString:@"titlePanelAction"])
|
|
{
|
|
NSString *panelAction = options[@"action"];
|
|
|
|
if ([panelAction isEqualToString:@"block"])
|
|
[self requestUserBlocked:true];
|
|
else if ([panelAction isEqualToString:@"unblock"])
|
|
[self requestUserBlocked:false];
|
|
else if ([panelAction isEqualToString:@"call"])
|
|
{
|
|
TGUser *user = [TGDatabaseInstance() loadUser:_uid];
|
|
|
|
NSMutableArray *phoneNumbers = [[NSMutableArray alloc] init];
|
|
|
|
if (user.phoneNumber != nil && user.phoneNumber.length != 0)
|
|
{
|
|
[phoneNumbers addObject:@[@"mobile", user.phoneNumber, [TGPhoneUtils formatPhone:user.phoneNumber forceInternational:true]]];
|
|
}
|
|
|
|
TGPhonebookContact *contact = [TGDatabaseInstance() phonebookContactByPhoneId:[user contactId]];
|
|
int mainPhoneHash = phoneMatchHash(user.phoneNumber);
|
|
for (TGPhoneNumber *number in contact.phoneNumbers)
|
|
{
|
|
if (number.phoneId != mainPhoneHash)
|
|
{
|
|
[phoneNumbers addObject:@[number.label == nil ? @"" : number.label, number.number, [TGPhoneUtils formatPhone:number.number forceInternational:false]]];
|
|
}
|
|
}
|
|
|
|
if (phoneNumbers.count != 0)
|
|
{
|
|
if (phoneNumbers.count == 1)
|
|
{
|
|
NSString *telephoneScheme = @"tel:";
|
|
if (![[UIApplication sharedApplication] canOpenURL:[NSURL URLWithString:@"tel://"]])
|
|
telephoneScheme = @"facetime:";
|
|
|
|
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:[NSString stringWithFormat:@"%@%@", telephoneScheme, [TGPhoneUtils formatPhoneUrl:[[phoneNumbers objectAtIndex:0] objectAtIndex:1]]]]];
|
|
}
|
|
else if (phoneNumbers.count > 1)
|
|
{
|
|
TGModernConversationController *controller = self.controller;
|
|
[controller showCallNumberMenu:phoneNumbers];
|
|
}
|
|
}
|
|
}
|
|
else if ([panelAction isEqualToString:@"edit"])
|
|
{
|
|
TGModernConversationController *controller = self.controller;
|
|
[controller enterEditingMode];
|
|
}
|
|
else if ([panelAction isEqualToString:@"info"])
|
|
[self _controllerAvatarPressed];
|
|
else if ([panelAction isEqualToString:@"search"])
|
|
[self navigateToMessageSearch];
|
|
}
|
|
|
|
[super actionStageActionRequested:action options:options];
|
|
}
|
|
|
|
- (void)actionStageResourceDispatched:(NSString *)path resource:(id)resource arguments:(id)arguments
|
|
{
|
|
if ([path isEqualToString:@"/tg/userdatachanges"] || [path isEqualToString:@"/tg/userpresencechanges"])
|
|
{
|
|
NSArray *users = ((SGraphObjectNode *)resource).object;
|
|
|
|
for (TGUser *user in users)
|
|
{
|
|
if (user.uid == _uid)
|
|
{
|
|
bool accentColored = false;
|
|
NSString *statusString = [self statusStringForUser:user accentColored:&accentColored];
|
|
[self _setTitle:[self stringForTitle:user isContact:[TGDatabaseInstance() uidIsRemoteContact:_uid]] andStatus:statusString accentColored:accentColored allowAnimatioon:true toggleMode:TGModernConversationControllerTitleToggleNone];
|
|
[self _setAvatarConversationId:_uid firstName:user.firstName lastName:user.lastName];
|
|
[self _setAvatarUrl:user.photoUrlSmall];
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if ([path isEqualToString:@"/as/updateRelativeTimestamps"])
|
|
{
|
|
TGUser *user = [TGDatabaseInstance() loadUser:_uid];
|
|
|
|
bool accentColored = false;
|
|
NSString *statusString = [self statusStringForUser:user accentColored:&accentColored];
|
|
[self _setTitle:[self stringForTitle:user isContact:[TGDatabaseInstance() uidIsRemoteContact:_uid]] andStatus:statusString accentColored:accentColored allowAnimatioon:false toggleMode:TGModernConversationControllerTitleToggleNone];
|
|
[self _setAvatarConversationId:_uid firstName:user.firstName lastName:user.lastName];
|
|
[self _setAvatarUrl:user.photoUrlSmall];
|
|
}
|
|
else if ([path isEqualToString:[[NSString alloc] initWithFormat:@"/tg/conversation/(%lld)/typing", _conversationId]])
|
|
{
|
|
NSDictionary *userActivities = ((SGraphObjectNode *)resource).object;
|
|
if (userActivities.count == 0)
|
|
[self _setTypingStatus:nil activity:0];
|
|
else
|
|
{
|
|
NSString *activity = userActivities[userActivities.allKeys.firstObject];
|
|
[self _setTypingStatus:[self stringForActivity:activity] activity:[self activityTypeForActivity:activity]];
|
|
}
|
|
}
|
|
else if ([path hasPrefix:@"/tg/blockedUsers"])
|
|
{
|
|
[self actorCompleted:ASStatusSuccess path:path result:resource];
|
|
}
|
|
else if ([path hasPrefix:@"/tg/contactlist"])
|
|
{
|
|
bool isContact = [TGDatabaseInstance() uidIsRemoteContact:_uid];
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
if (_isContact != isContact)
|
|
{
|
|
_isContact = isContact;
|
|
|
|
[self _createOrUpdatePrimaryTitlePanel:false];
|
|
[self _updateContactLinkPanel];
|
|
}
|
|
});
|
|
}
|
|
else if ([path hasPrefix:@"/tg/peerSettings/"])
|
|
{
|
|
[self actorCompleted:ASStatusSuccess path:path result:resource];
|
|
}
|
|
else if ([path hasPrefix:@"/tg/userLink/"])
|
|
{
|
|
int userLink = [(NSNumber *)((SGraphObjectNode *)resource).object intValue];
|
|
TGDispatchOnMainThread(^
|
|
{
|
|
[self _updatePhoneSharingStatusFromUserLink:userLink];
|
|
});
|
|
}
|
|
|
|
[super actionStageResourceDispatched:path resource:resource arguments:arguments];
|
|
}
|
|
|
|
- (void)actorCompleted:(int)status path:(NSString *)path result:(id)result
|
|
{
|
|
if ([path hasPrefix:@"/tg/blockedUsers"])
|
|
{
|
|
id blockedResult = ((SGraphObjectNode *)result).object;
|
|
|
|
bool blocked = false;
|
|
|
|
if ([blockedResult respondsToSelector:@selector(boolValue)])
|
|
blocked = [blockedResult boolValue];
|
|
else if ([blockedResult isKindOfClass:[NSArray class]])
|
|
{
|
|
for (TGUser *user in blockedResult)
|
|
{
|
|
if (user.uid == _uid)
|
|
{
|
|
blocked = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
[self updateUserBlocked:blocked];
|
|
}
|
|
else if ([path hasPrefix:@"/tg/peerSettings/"])
|
|
{
|
|
bool isMuted = [[((SGraphObjectNode *)result).object objectForKey:@"muteUntil"] intValue] != 0;
|
|
[self _updateUserMute:isMuted];
|
|
}
|
|
|
|
[super actorCompleted:status path:path result:result];
|
|
}
|
|
|
|
- (bool)allowReplies
|
|
{
|
|
return true;
|
|
}
|
|
|
|
- (bool)isASingleBotGroup
|
|
{
|
|
return _isBot;
|
|
}
|
|
|
|
- (SSignal *)commandListForCommand:(NSString *)command
|
|
{
|
|
if (_isBot)
|
|
{
|
|
TGUser *user = [TGDatabaseInstance() loadUser:_uid];
|
|
if (user == nil)
|
|
return nil;
|
|
|
|
NSString *normalizedCommand = [command lowercaseString];
|
|
if ([normalizedCommand hasPrefix:@"/"])
|
|
normalizedCommand = [normalizedCommand substringFromIndex:1];
|
|
return [[TGBotSignals botInfoForUserId:_uid] map:^id(TGBotInfo *botInfo)
|
|
{
|
|
NSMutableArray *commands = [[NSMutableArray alloc] init];
|
|
for (TGBotComandInfo *commandInfo in botInfo.commandList)
|
|
{
|
|
if (normalizedCommand.length == 0 || [[commandInfo.command lowercaseString] hasPrefix:normalizedCommand])
|
|
[commands addObject:commandInfo];
|
|
}
|
|
if (commands.count == 1 && [[((TGBotComandInfo *)commands[0]).command lowercaseString] isEqualToString:normalizedCommand])
|
|
[commands removeAllObjects];
|
|
return @[@[user, commands]];
|
|
}];
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
|
|
- (void)standaloneSendBotStartPayload:(NSString *)payload
|
|
{
|
|
_botStartPayload = payload;
|
|
[self requestBotStart];
|
|
}
|
|
|
|
- (void)_itemsUpdated {
|
|
[super _itemsUpdated];
|
|
|
|
if (!_hasOutgoingMessages || !_hasIncomingMessages) {
|
|
bool foundOutgoing = false;
|
|
bool foundIncoming = false;
|
|
|
|
for (TGMessageModernConversationItem *item in _items) {
|
|
if (item->_message.outgoing) {
|
|
foundOutgoing = true;
|
|
if (foundIncoming) {
|
|
break;
|
|
}
|
|
} else if (item->_message.mid < TGMessageLocalMidBaseline) {
|
|
foundIncoming = true;
|
|
if (foundOutgoing) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (foundIncoming != _hasIncomingMessages || foundOutgoing != _hasOutgoingMessages) {
|
|
TGDispatchOnMainThread(^{
|
|
_hasOutgoingMessages = foundOutgoing;
|
|
_hasIncomingMessages = foundIncoming;
|
|
|
|
if (!_isLoadingFirstMessages) {
|
|
[self _updateContactLinkPanel];
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
- (SSignal *)userListForMention:(NSString *)mention canBeContextBot:(bool)canBeContextBot {
|
|
return [[canBeContextBot ? [TGRecentContextBotsSignal recentBots] : [SSignal single:@[]] mapToSignal:^SSignal *(NSArray *userIds) {
|
|
return [TGDatabaseInstance() modify:^id{
|
|
NSString *normalizedMention = [mention lowercaseString];
|
|
NSMutableArray *users = [[NSMutableArray alloc] init];
|
|
for (NSNumber *nUserId in userIds) {
|
|
TGUser *user = [TGDatabaseInstance() loadUser:[nUserId intValue]];
|
|
if (user != nil && (normalizedMention.length == 0 || [[user.userName lowercaseString] hasPrefix:normalizedMention])) {
|
|
[users addObject:user];
|
|
}
|
|
}
|
|
return users;
|
|
}];
|
|
}] deliverOn:[SQueue mainQueue]];
|
|
}
|
|
|
|
@end
|