433 lines
12 KiB
Objective-C
433 lines
12 KiB
Objective-C
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
Copyright 2017 Vector Creations Ltd
|
|
Copyright 2015 OpenMarket Ltd
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
Please see LICENSE in the repository root for full details.
|
|
*/
|
|
|
|
#import "PublicRoomsDirectoryDataSource.h"
|
|
|
|
#import "PublicRoomTableViewCell.h"
|
|
|
|
#import "GeneratedInterface-Swift.h"
|
|
|
|
#pragma mark - Constants definitions
|
|
|
|
// Time in seconds from which public rooms data is considered as obsolete
|
|
double const kPublicRoomsDirectoryDataExpiration = 10;
|
|
|
|
static NSString *const kNSFWKeyword = @"nsfw";
|
|
|
|
#pragma mark - PublicRoomsDirectoryDataSource
|
|
|
|
@interface PublicRoomsDirectoryDataSource ()
|
|
{
|
|
// The pending request to refresh public rooms data.
|
|
MXHTTPOperation *publicRoomsRequest;
|
|
|
|
/**
|
|
All public rooms fetched so far.
|
|
*/
|
|
NSMutableArray<MXPublicRoom*> *rooms;
|
|
|
|
/**
|
|
The next token to use for pagination.
|
|
*/
|
|
NSString *nextBatch;
|
|
}
|
|
|
|
@property (nonatomic, strong) NSRegularExpression *forbiddenTermsRegex;
|
|
|
|
@end
|
|
|
|
@implementation PublicRoomsDirectoryDataSource
|
|
|
|
- (instancetype)init
|
|
{
|
|
self = [super init];
|
|
if (self)
|
|
{
|
|
rooms = [NSMutableArray array];
|
|
_paginationLimit = 20;
|
|
|
|
NSString *path = [[NSBundle mainBundle] pathForResource:@"forbidden_terms" ofType:@"txt"];
|
|
NSString *fileContents = [NSString stringWithContentsOfFile:path encoding: NSUTF8StringEncoding error:nil];
|
|
NSArray *forbiddenTerms = [fileContents componentsSeparatedByCharactersInSet: NSCharacterSet.whitespaceAndNewlineCharacterSet];
|
|
|
|
NSString *pattern = [NSString stringWithFormat:@"\\b(%@)\\b", [forbiddenTerms componentsJoinedByString:@"|"]];
|
|
pattern = [pattern stringByAppendingString:@"|(\\b18\\+)"]; // Special case "18+"
|
|
|
|
_forbiddenTermsRegex = [[NSRegularExpression alloc] initWithPattern:pattern options:NSRegularExpressionCaseInsensitive error:nil];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (NSString *)directoryServerDisplayname
|
|
{
|
|
NSString *directoryServerDisplayname;
|
|
|
|
if (_homeserver)
|
|
{
|
|
directoryServerDisplayname = _homeserver;
|
|
}
|
|
else if (_thirdpartyProtocolInstance)
|
|
{
|
|
directoryServerDisplayname = _thirdpartyProtocolInstance.desc;
|
|
}
|
|
else
|
|
{
|
|
if (_includeAllNetworks)
|
|
{
|
|
// We display all rooms, included bridged ones, of the user's HS
|
|
directoryServerDisplayname = self.mxSession.matrixRestClient.credentials.homeServerName;
|
|
}
|
|
else
|
|
{
|
|
// We display only Matrix rooms of the user's HS
|
|
directoryServerDisplayname = [VectorL10n matrix];
|
|
}
|
|
}
|
|
|
|
return directoryServerDisplayname;
|
|
}
|
|
|
|
- (void)setHomeserver:(NSString *)homeserver
|
|
{
|
|
if ([homeserver isEqualToString:self.mxSession.matrixRestClient.credentials.homeServerName])
|
|
{
|
|
// The CS API does not like we pass the user's HS as parameter
|
|
homeserver = nil;
|
|
}
|
|
|
|
_thirdpartyProtocolInstance = nil;
|
|
|
|
if (homeserver != _homeserver)
|
|
{
|
|
_homeserver = homeserver;
|
|
|
|
// Reset data
|
|
[self resetPagination];
|
|
}
|
|
}
|
|
|
|
- (void)setIncludeAllNetworks:(BOOL)includeAllNetworks
|
|
{
|
|
if (includeAllNetworks != _includeAllNetworks)
|
|
{
|
|
_includeAllNetworks = includeAllNetworks;
|
|
|
|
// Reset data
|
|
[self resetPagination];
|
|
}
|
|
}
|
|
|
|
- (void)setThirdpartyProtocolInstance:(MXThirdPartyProtocolInstance *)thirdpartyProtocolInstance
|
|
{
|
|
if (thirdpartyProtocolInstance != _thirdpartyProtocolInstance)
|
|
{
|
|
_homeserver = nil;
|
|
_includeAllNetworks = NO;
|
|
_thirdpartyProtocolInstance = thirdpartyProtocolInstance;
|
|
|
|
// Reset data
|
|
[self resetPagination];
|
|
}
|
|
}
|
|
|
|
- (void)setSearchPattern:(NSString *)searchPattern
|
|
{
|
|
if (searchPattern)
|
|
{
|
|
if (![searchPattern isEqualToString:_searchPattern])
|
|
{
|
|
_searchPattern = searchPattern;
|
|
[self resetPagination];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Refresh if the previous search was not nil
|
|
// or if it is the first time we make a search
|
|
if (_searchPattern || rooms.count == 0)
|
|
{
|
|
_searchPattern = searchPattern;
|
|
[self resetPagination];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (NSUInteger)roomsCount
|
|
{
|
|
return rooms.count;
|
|
}
|
|
|
|
- (NSIndexPath*)cellIndexPathWithRoomId:(NSString*)roomId andMatrixSession:(MXSession*)matrixSession
|
|
{
|
|
NSIndexPath *indexPath = nil;
|
|
|
|
// Look for the public room
|
|
for (NSInteger index = 0; index < rooms.count; index ++)
|
|
{
|
|
MXPublicRoom *room = rooms[index];
|
|
if ([roomId isEqualToString:room.roomId])
|
|
{
|
|
// Got it
|
|
indexPath = [NSIndexPath indexPathForRow:index inSection:0];
|
|
break;
|
|
}
|
|
}
|
|
|
|
return indexPath;
|
|
}
|
|
|
|
- (MXPublicRoom *)roomAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
MXPublicRoom *room;
|
|
|
|
if (indexPath.row < rooms.count)
|
|
{
|
|
room = rooms[indexPath.row];
|
|
}
|
|
|
|
return room;
|
|
}
|
|
|
|
- (CGFloat)cellHeightAtIndexPath:(NSIndexPath*)indexPath
|
|
{
|
|
if (indexPath.row < rooms.count)
|
|
{
|
|
return PublicRoomTableViewCell.cellHeight;
|
|
}
|
|
|
|
return 50.0;
|
|
}
|
|
|
|
- (void)resetPagination
|
|
{
|
|
// Cancel the previous request
|
|
if (publicRoomsRequest)
|
|
{
|
|
[publicRoomsRequest cancel];
|
|
}
|
|
|
|
// Reset all pagination vars
|
|
[rooms removeAllObjects];
|
|
nextBatch = nil;
|
|
_searchResultsCount = 0;
|
|
_searchResultsCountIsLimited = NO;
|
|
_hasReachedPaginationEnd = NO;
|
|
}
|
|
|
|
- (MXHTTPOperation *)paginate:(void (^)(NSUInteger))complete failure:(void (^)(NSError *))failure
|
|
{
|
|
if (_hasReachedPaginationEnd)
|
|
{
|
|
if (complete)
|
|
{
|
|
complete(0);
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
[self setState:MXKDataSourceStatePreparing];
|
|
|
|
__weak typeof(self) weakSelf = self;
|
|
|
|
// Get the public rooms from the server
|
|
MXHTTPOperation *newPublicRoomsRequest;
|
|
newPublicRoomsRequest = [self.mxSession.matrixRestClient publicRoomsOnServer:_homeserver limit:_paginationLimit since:nextBatch filter:_searchPattern thirdPartyInstanceId:_thirdpartyProtocolInstance.instanceId includeAllNetworks:_includeAllNetworks success:^(MXPublicRoomsResponse *publicRoomsResponse) {
|
|
|
|
if (weakSelf)
|
|
{
|
|
typeof(self) self = weakSelf;
|
|
|
|
self->publicRoomsRequest = nil;
|
|
|
|
NSArray<MXPublicRoom*> *publicRooms;
|
|
|
|
publicRooms = [self filterPublicRooms:publicRoomsResponse.chunk];
|
|
|
|
[self->rooms addObjectsFromArray:publicRooms];
|
|
self->nextBatch = publicRoomsResponse.nextBatch;
|
|
|
|
if (!self->_searchPattern)
|
|
{
|
|
// When there is no search, we can use totalRoomCountEstimate returned by the server
|
|
self->_searchResultsCount = publicRoomsResponse.totalRoomCountEstimate;
|
|
self->_searchResultsCountIsLimited = NO;
|
|
}
|
|
else
|
|
{
|
|
// Else we can only display something like ">20 matching rooms"
|
|
self->_searchResultsCount = self->rooms.count;
|
|
self->_searchResultsCountIsLimited = publicRoomsResponse.nextBatch ? YES : NO;
|
|
}
|
|
|
|
// Detect pagination end
|
|
if (!publicRoomsResponse.nextBatch)
|
|
{
|
|
self->_hasReachedPaginationEnd = YES;
|
|
}
|
|
|
|
[self setState:MXKDataSourceStateReady];
|
|
|
|
if (complete)
|
|
{
|
|
complete(publicRoomsResponse.chunk.count);
|
|
}
|
|
}
|
|
|
|
} failure:^(NSError *error) {
|
|
|
|
if (weakSelf)
|
|
{
|
|
typeof(self) self = weakSelf;
|
|
|
|
if (error.domain == NSURLErrorDomain && error.code == NSURLErrorCancelled)
|
|
{
|
|
// Do not take into account error coming from a cancellation
|
|
return;
|
|
}
|
|
|
|
self->publicRoomsRequest = nil;
|
|
|
|
MXLogDebug(@"[PublicRoomsDirectoryDataSource] Failed to fecth public rooms.");
|
|
|
|
[self setState:MXKDataSourceStateFailed];
|
|
|
|
// Alert user
|
|
[[AppDelegate theDelegate] showErrorAsAlert:error];
|
|
|
|
if (failure)
|
|
{
|
|
failure(error);
|
|
}
|
|
}
|
|
}];
|
|
|
|
publicRoomsRequest = newPublicRoomsRequest;
|
|
|
|
return publicRoomsRequest;
|
|
}
|
|
|
|
#pragma mark - Private methods
|
|
|
|
// Update the MXKDataSource state and the delegate
|
|
- (void)setState:(MXKDataSourceState)newState
|
|
{
|
|
state = newState;
|
|
if (self.delegate && [self.delegate respondsToSelector:@selector(dataSource:didStateChange:)])
|
|
{
|
|
[self.delegate dataSource:self didStateChange:state];
|
|
}
|
|
}
|
|
|
|
- (NSArray<MXPublicRoom*>*)filterPublicRooms:(NSArray<MXPublicRoom*>*)publicRooms
|
|
{
|
|
NSMutableArray *filteredRooms = [NSMutableArray new];
|
|
|
|
for (MXPublicRoom *publicRoom in publicRooms)
|
|
{
|
|
BOOL shouldAllow = YES;
|
|
|
|
if (publicRoom.name != nil) {
|
|
shouldAllow &= [self.forbiddenTermsRegex numberOfMatchesInString:publicRoom.name options:0 range:NSMakeRange(0, publicRoom.name.length)] == 0;
|
|
}
|
|
|
|
if (publicRoom.topic != nil) {
|
|
shouldAllow &= [self.forbiddenTermsRegex numberOfMatchesInString:publicRoom.topic options:0 range:NSMakeRange(0, publicRoom.topic.length)] == 0;
|
|
}
|
|
|
|
if (shouldAllow) {
|
|
[filteredRooms addObject:publicRoom];
|
|
}
|
|
}
|
|
|
|
return filteredRooms;
|
|
}
|
|
|
|
#pragma mark - UITableViewDataSource
|
|
|
|
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
|
|
{
|
|
// Display a default cell when no rooms is available.
|
|
return rooms.count ? rooms.count : 1;
|
|
}
|
|
|
|
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
// Sanity check
|
|
if (indexPath.row < rooms.count)
|
|
{
|
|
PublicRoomTableViewCell *publicRoomCell = [tableView dequeueReusableCellWithIdentifier:[PublicRoomTableViewCell defaultReuseIdentifier]];
|
|
if (!publicRoomCell)
|
|
{
|
|
publicRoomCell = [[PublicRoomTableViewCell alloc] init];
|
|
}
|
|
|
|
[publicRoomCell render:rooms[indexPath.row] withMatrixSession:self.mxSession];
|
|
return publicRoomCell;
|
|
}
|
|
else
|
|
{
|
|
MXKTableViewCell *tableViewCell = [tableView dequeueReusableCellWithIdentifier:[MXKTableViewCell defaultReuseIdentifier]];
|
|
if (!tableViewCell)
|
|
{
|
|
tableViewCell = [[MXKTableViewCell alloc] init];
|
|
tableViewCell.textLabel.textColor = ThemeService.shared.theme.textSecondaryColor;
|
|
tableViewCell.textLabel.font = [UIFont systemFontOfSize:15.0];
|
|
tableViewCell.selectionStyle = UITableViewCellSelectionStyleNone;
|
|
}
|
|
|
|
if (state == MXKDataSourceStateReady)
|
|
{
|
|
if (_searchPattern.length)
|
|
{
|
|
tableViewCell.textLabel.text = [VectorL10n searchNoResult];
|
|
}
|
|
else
|
|
{
|
|
tableViewCell.textLabel.text = [VectorL10n roomDirectoryNoPublicRoom];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (_searchPattern.length)
|
|
{
|
|
tableViewCell.textLabel.text = [VectorL10n searchInProgress];
|
|
}
|
|
else
|
|
{
|
|
// Show nothing in other cases
|
|
tableViewCell.textLabel.text = @"";
|
|
}
|
|
}
|
|
|
|
return tableViewCell;
|
|
}
|
|
}
|
|
|
|
#pragma mark - NSCopying
|
|
|
|
- (id)copyWithZone:(NSZone *)zone
|
|
{
|
|
PublicRoomsDirectoryDataSource *source = [[[self class] allocWithZone:zone] initWithMatrixSession:self.mxSession];
|
|
|
|
source.homeserver = [self.homeserver copyWithZone:zone];
|
|
source.includeAllNetworks = self.includeAllNetworks;
|
|
if (self.thirdpartyProtocolInstance)
|
|
{
|
|
source.thirdpartyProtocolInstance = [MXThirdPartyProtocolInstance modelFromJSON:self.thirdpartyProtocolInstance.JSONDictionary];
|
|
}
|
|
source.paginationLimit = self.paginationLimit;
|
|
source.searchPattern = [self.searchPattern copyWithZone:zone];
|
|
source->rooms = [rooms mutableCopyWithZone:zone];
|
|
source->nextBatch = [nextBatch copyWithZone:zone];
|
|
|
|
return source;
|
|
}
|
|
|
|
@end
|