/***************************************************************************** * VLCOpenNetworkStreamViewController.m * VLC for iOS ***************************************************************************** * Copyright (c) 2013-2018 VideoLAN. All rights reserved. * $Id$ * * Authors: Felix Paul Kühne * Gleb Pinigin * Pierre Sagaspe * Adam Viaud * Refer to the COPYING file of the official project for license. *****************************************************************************/ #import "VLCOpenNetworkStreamViewController.h" #import "VLCPlaybackController.h" #import "VLCStreamingHistoryCell.h" #import "UIDevice+VLC.h" #import "VLC_iOS-Swift.h" @interface VLCOpenNetworkStreamViewController () { NSMutableArray *_recentURLs; NSMutableDictionary *_recentURLTitles; } @end @implementation VLCOpenNetworkStreamViewController + (void)initialize { NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; NSDictionary *appDefaults = @{kVLCRecentURLs : @[], kVLCRecentURLTitles : @{}, kVLCPrivateWebStreaming : @(NO)}; [defaults registerDefaults:appDefaults]; } - (instancetype)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil { self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil]; if (self) { self.title = NSLocalizedString(@"OPEN_NETWORK", comment: ""); } return self; } - (void)applicationDidBecomeActive:(NSNotification *)notification { [self updatePasteboardTextInURLField]; } - (void)ubiquitousKeyValueStoreDidChange:(NSNotification *)notification { /* TODO: don't blindly trust that the Cloud knows best */ _recentURLs = [NSMutableArray arrayWithArray:[[NSUbiquitousKeyValueStore defaultStore] arrayForKey:kVLCRecentURLs]]; _recentURLTitles = [NSMutableDictionary dictionaryWithDictionary:[[NSUbiquitousKeyValueStore defaultStore] dictionaryForKey:kVLCRecentURLTitles]]; [self.historyTableView reloadData]; } - (void)viewDidLoad { [super viewDidLoad]; NSNotificationCenter *notificationCenter = [NSNotificationCenter defaultCenter]; [notificationCenter addObserver:self selector:@selector(ubiquitousKeyValueStoreDidChange:) name:NSUbiquitousKeyValueStoreDidChangeExternallyNotification object:[NSUbiquitousKeyValueStore defaultStore]]; [notificationCenter addObserver:self selector:@selector(updateForTheme) name:kVLCThemeDidChangeNotification object:nil]; /* force store update */ NSUbiquitousKeyValueStore *ubiquitousKeyValueStore = [NSUbiquitousKeyValueStore defaultStore]; [ubiquitousKeyValueStore synchronize]; /* fetch data from cloud */ _recentURLs = [NSMutableArray arrayWithArray:[[NSUbiquitousKeyValueStore defaultStore] arrayForKey:kVLCRecentURLs]]; _recentURLTitles = [NSMutableDictionary dictionaryWithDictionary:[[NSUbiquitousKeyValueStore defaultStore] dictionaryForKey:kVLCRecentURLTitles]]; /* merge data from local storage (aka legacy VLC versions) */ NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; NSArray *localRecentUrls = [defaults objectForKey:kVLCRecentURLs]; if (localRecentUrls != nil) { if (localRecentUrls.count != 0) { [_recentURLs addObjectsFromArray:localRecentUrls]; [defaults setObject:nil forKey:kVLCRecentURLs]; [ubiquitousKeyValueStore setArray:_recentURLs forKey:kVLCRecentURLs]; [ubiquitousKeyValueStore synchronize]; } } /* * Observe changes to the pasteboard so we can automatically paste it into the URL field. * Do not use UIPasteboardChangedNotification because we have copy actions that will trigger it on this screen. * Instead when the user comes back to the application from the background (or the inactive state by pulling down notification center), update the URL field. * Using the 'active' rather than 'foreground' notification for future proofing if iOS ever allows running multiple apps on the same screen (which would allow the pasteboard to be changed without truly backgrounding the app). */ [notificationCenter addObserver:self selector:@selector(applicationDidBecomeActive:) name:UIApplicationDidBecomeActiveNotification object:[UIApplication sharedApplication]]; [self.openButton setTitle:NSLocalizedString(@"OPEN_NETWORK", nil) forState:UIControlStateNormal]; [self.openButton setAccessibilityIdentifier:@"Open Network Stream"]; [self.privateModeLabel setText:NSLocalizedString(@"PRIVATE_PLAYBACK_TOGGLE", nil)]; UILabel *scanSubModelabel = self.ScanSubModeLabel; [scanSubModelabel setText:NSLocalizedString(@"SCAN_SUBTITLE_TOGGLE", nil)]; [scanSubModelabel setAdjustsFontSizeToFitWidth:YES]; [scanSubModelabel setNumberOfLines:0]; [self.whatToOpenHelpLabel setText:NSLocalizedString(@"OPEN_NETWORK_HELP", nil)]; self.urlField.delegate = self; self.urlField.keyboardType = UIKeyboardTypeURL; self.edgesForExtendedLayout = UIRectEdgeNone; // This will be called every time this VC is opened by the side menu controller [self updatePasteboardTextInURLField]; // Registering a custom menu item for renaming streams NSString *renameTitle = NSLocalizedString(@"BUTTON_RENAME", nil); SEL renameStreamSelector = @selector(renameStream:); UIMenuItem *renameItem = [[UIMenuItem alloc] initWithTitle:renameTitle action:renameStreamSelector]; UIMenuController *sharedMenuController = [UIMenuController sharedMenuController]; [sharedMenuController setMenuItems:@[renameItem]]; [sharedMenuController update]; [self updateForTheme]; } - (NSString *)detailText { return NSLocalizedString(@"STREAMVC_DETAILTEXT", nil); } - (UIImage *)cellImage { return [UIImage imageNamed:@"OpenNetStream"]; } - (void)updateForTheme { self.historyTableView.backgroundColor = PresentationTheme.current.colors.background; self.view.backgroundColor = PresentationTheme.current.colors.background; NSAttributedString *coloredAttributedPlaceholder = [[NSAttributedString alloc] initWithString:@"http://myserver.com/file.mkv" attributes:@{NSForegroundColorAttributeName: PresentationTheme.current.colors.lightTextColor}]; self.urlField.attributedPlaceholder = coloredAttributedPlaceholder; self.urlField.backgroundColor = PresentationTheme.current.colors.cellBackgroundB; self.urlField.textColor = PresentationTheme.current.colors.cellTextColor; self.privateModeLabel.textColor = PresentationTheme.current.colors.lightTextColor; self.ScanSubModeLabel.textColor = PresentationTheme.current.colors.lightTextColor; self.whatToOpenHelpLabel.textColor = PresentationTheme.current.colors.lightTextColor; self.openButton.backgroundColor = PresentationTheme.current.colors.orangeUI; [self.historyTableView reloadData]; } - (void)updatePasteboardTextInURLField { UIPasteboard *pasteboard = [UIPasteboard generalPasteboard]; if ([pasteboard containsPasteboardTypes:@[@"public.url"]]) self.urlField.text = [[pasteboard valueForPasteboardType:@"public.url"] absoluteString]; } - (void)viewWillAppear:(BOOL)animated { NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; self.privateToggleSwitch.on = [defaults boolForKey:kVLCPrivateWebStreaming]; self.ScanSubToggleSwitch.on = [defaults boolForKey:kVLChttpScanSubtitle]; [super viewWillAppear:animated]; } - (void)viewWillDisappear:(BOOL)animated { [[NSNotificationCenter defaultCenter] removeObserver:self name:UIApplicationDidBecomeActiveNotification object:[UIApplication sharedApplication]]; NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; [defaults setBool:self.privateToggleSwitch.on forKey:kVLCPrivateWebStreaming]; [defaults setBool:self.ScanSubToggleSwitch.on forKey:kVLChttpScanSubtitle]; [self.view endEditing:YES]; /* force update before we leave */ [[NSUbiquitousKeyValueStore defaultStore] synchronize]; [super viewWillDisappear:animated]; } - (CGSize)contentSizeForViewInPopover { return [self.view sizeThatFits:CGSizeMake(320, 800)]; } #pragma mark - UI interaction - (BOOL)shouldAutorotate { UIInterfaceOrientation toInterfaceOrientation = [[UIApplication sharedApplication] statusBarOrientation]; if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone && toInterfaceOrientation == UIInterfaceOrientationPortraitUpsideDown) return NO; return YES; } - (IBAction)openButtonAction:(id)sender { if ([self.urlField.text length] <= 0 || [NSURL URLWithString:self.urlField.text] == nil) { NSMutableArray *buttonsAction = [[NSMutableArray alloc] init]; ButtonAction *cancelAction = [[ButtonAction alloc] initWithButtonTitle: NSLocalizedString(@"BUTTON_OK", nil) buttonAction: ^(UIAlertAction* action){}]; [buttonsAction addObject: cancelAction]; [VLCAlertViewController alertViewManagerWithTitle:NSLocalizedString(@"URL_NOT_SUPPORTED", nil) errorMessage: NSLocalizedString(@"PROTOCOL_NOT_SELECTED", nil) viewController:self buttonsAction:buttonsAction]; return; } if (!self.privateToggleSwitch.on) { NSString *urlString = self.urlField.text; if ([_recentURLs indexOfObject:urlString] != NSNotFound) [_recentURLs removeObject:urlString]; if (_recentURLs.count >= 100) [_recentURLs removeLastObject]; [_recentURLs addObject:urlString]; [[NSUbiquitousKeyValueStore defaultStore] setArray:_recentURLs forKey:kVLCRecentURLs]; [self.historyTableView reloadData]; } [self.urlField resignFirstResponder]; [self _openURLStringAndDismiss:self.urlField.text]; } - (void)renameStreamFromCell:(UITableViewCell *)cell { NSIndexPath *cellIndexPath = [self.historyTableView indexPathForCell:cell]; NSString *renameString = NSLocalizedString(@"BUTTON_RENAME", nil); NSString *cancelString = NSLocalizedString(@"BUTTON_CANCEL", nil); UIAlertController *alertController = [UIAlertController alertControllerWithTitle:renameString message:nil preferredStyle:UIAlertControllerStyleAlert]; UIAlertAction *cancelAction = [UIAlertAction actionWithTitle:cancelString style:UIAlertActionStyleCancel handler:nil]; UIAlertAction *okAction = [UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) { NSString *streamTitle = alertController.textFields.firstObject.text; [self renameStreamWithTitle:streamTitle atIndex:cellIndexPath.row]; }]; [alertController addTextFieldWithConfigurationHandler:^(UITextField * _Nonnull textField) { textField.text = cell.textLabel.text; [[NSNotificationCenter defaultCenter] addObserverForName:UITextFieldTextDidChangeNotification object:textField queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification * _Nonnull note) { okAction.enabled = (textField.text.length != 0); }]; }]; [alertController addAction:cancelAction]; [alertController addAction:okAction]; [self presentViewController:alertController animated:YES completion:nil]; } - (void)renameStreamWithTitle:(NSString *)title atIndex:(NSInteger)index { [_recentURLTitles setObject:title forKey:[@(index) stringValue]]; [[NSUbiquitousKeyValueStore defaultStore] setDictionary:_recentURLTitles forKey:kVLCRecentURLTitles]; [[NSOperationQueue mainQueue] addOperationWithBlock:^{ [self.historyTableView reloadSections:[NSIndexSet indexSetWithIndex:0] withRowAnimation:UITableViewRowAnimationAutomatic]; }]; } #pragma mark - table view data source - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView { return 1; } - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { return _recentURLs.count; } - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *CellIdentifier = @"StreamingHistoryCell"; VLCStreamingHistoryCell *cell = (VLCStreamingHistoryCell *)[tableView dequeueReusableCellWithIdentifier:CellIdentifier]; if (cell == nil) { cell = [[VLCStreamingHistoryCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:CellIdentifier]; cell.delegate = self; [cell customizeAppearance]; } NSString *content = [_recentURLs[indexPath.row] stringByRemovingPercentEncoding]; NSString *possibleTitle = _recentURLTitles[[@(indexPath.row) stringValue]]; cell.detailTextLabel.text = content; cell.textLabel.text = possibleTitle ?: [content lastPathComponent]; return cell; } #pragma mark - table view delegate - (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath { cell.backgroundColor = (indexPath.row % 2 == 0)? PresentationTheme.current.colors.cellBackgroundB : PresentationTheme.current.colors.cellBackgroundA; cell.textLabel.textColor = PresentationTheme.current.colors.cellTextColor; cell.detailTextLabel.textColor = PresentationTheme.current.colors.cellDetailTextColor; } - (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath { return YES; } - (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath { if (editingStyle == UITableViewCellEditingStyleDelete) { [_recentURLs removeObjectAtIndex:indexPath.row]; [_recentURLTitles removeObjectForKey:[@(indexPath.row) stringValue]]; [[NSUbiquitousKeyValueStore defaultStore] setArray:_recentURLs forKey:kVLCRecentURLs]; [[NSUbiquitousKeyValueStore defaultStore] setDictionary:_recentURLTitles forKey:kVLCRecentURLTitles]; [tableView reloadData]; } } - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath { [self.historyTableView deselectRowAtIndexPath:indexPath animated:NO]; [self _openURLStringAndDismiss:_recentURLs[indexPath.row]]; } - (void)tableView:(UITableView *)tableView performAction:(SEL)action forRowAtIndexPath:(NSIndexPath *)indexPath withSender:(id)sender { NSString *actionText = NSStringFromSelector(action); if ([actionText isEqualToString:@"copy:"]) [UIPasteboard generalPasteboard].string = _recentURLs[indexPath.row]; } - (BOOL)tableView:(UITableView *)tableView canPerformAction:(SEL)action forRowAtIndexPath:(NSIndexPath *)indexPath withSender:(id)sender { NSString *actionText = NSStringFromSelector(action); if ([actionText isEqualToString:@"copy:"]) return YES; return NO; } - (BOOL)tableView:(UITableView *)tableView shouldShowMenuForRowAtIndexPath:(NSIndexPath *)indexPath { return YES; } #pragma mark - internals - (void)_openURLStringAndDismiss:(NSString *)url { NSURL *URLscheme = [NSURL URLWithString:url]; NSString *URLofSubtitle = nil; if ([URLscheme.scheme isEqualToString:@"http"] && self.ScanSubToggleSwitch.on) { URLofSubtitle = [self _checkURLofSubtitle:url]; } VLCMedia *media = [VLCMedia mediaWithURL:[NSURL URLWithString:url]]; VLCMediaList *medialist = [[VLCMediaList alloc] init]; [medialist addMedia:media]; [[VLCPlaybackController sharedInstance] playMediaList:medialist firstIndex:0 subtitlesFilePath:URLofSubtitle]; } - (NSString *)_checkURLofSubtitle:(NSString *)url { NSCharacterSet *characterFilter = [NSCharacterSet characterSetWithCharactersInString:@"\\.():$"]; NSString *subtitleFileExtensions = [[kSupportedSubtitleFileExtensions componentsSeparatedByCharactersInSet:characterFilter] componentsJoinedByString:@""]; NSArray *arraySubtitleFileExtensions = [subtitleFileExtensions componentsSeparatedByString:@"|"]; NSString *urlTemp = [[url stringByDeletingPathExtension] stringByAppendingString:@"."]; NSUInteger count = arraySubtitleFileExtensions.count; for (int i = 0; i < count; i++) { NSString *checkAddress = [urlTemp stringByAppendingString:arraySubtitleFileExtensions[i]]; NSURL *checkURL = [NSURL URLWithString:checkAddress]; NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:checkURL]; request.HTTPMethod = @"HEAD"; NSURLResponse *response = nil; NSError *error = nil; [self sendSynchronousRequest:request returningResponse:&response error:&error]; NSInteger httpStatus = [(NSHTTPURLResponse *)response statusCode]; if (httpStatus == 200) { NSString *fileSubtitlePath = [self _getFileSubtitleFromServer:checkURL]; return fileSubtitlePath; } } return nil; } - (NSString *)_getFileSubtitleFromServer:(NSURL *)url { NSString *fileSubtitlePath = nil; NSString *fileName = [[url path] lastPathComponent]; NSData *receivedSub = [NSData dataWithContentsOfURL:url]; if (receivedSub.length < [[UIDevice currentDevice] VLCFreeDiskSpace].longLongValue) { NSArray *searchPaths = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES); NSString *directoryPath = [searchPaths objectAtIndex:0]; fileSubtitlePath = [directoryPath stringByAppendingPathComponent:fileName]; NSFileManager *fileManager = [NSFileManager defaultManager]; if (![fileManager fileExistsAtPath:fileSubtitlePath]) { [fileManager createFileAtPath:fileSubtitlePath contents:nil attributes:nil]; if (![fileManager fileExistsAtPath:fileSubtitlePath]) APLog(@"file creation failed, no data was saved"); } [receivedSub writeToFile:fileSubtitlePath atomically:YES]; } else { NSMutableArray *buttonsAction = [[NSMutableArray alloc] init]; ButtonAction *okAction = [[ButtonAction alloc] initWithButtonTitle: NSLocalizedString(@"BUTTON_OK", nil) buttonAction: ^(UIAlertAction* action){}]; [buttonsAction addObject: okAction]; [VLCAlertViewController alertViewManagerWithTitle:NSLocalizedString(@"DISK_FULL", nil) errorMessage: [NSString stringWithFormat:NSLocalizedString(@"DISK_FULL_FORMAT", nil), fileName, [[UIDevice currentDevice] model]] viewController:self buttonsAction:buttonsAction]; } return fileSubtitlePath; } - (NSData *)sendSynchronousRequest:(NSURLRequest *)request returningResponse:(NSURLResponse **)response error:(NSError **)error { NSError __block *erreur = NULL; NSData __block *data; BOOL __block reqProcessed = false; NSURLResponse __block *urlResponse; [[[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^(NSData * _Nullable _data, NSURLResponse * _Nullable _response, NSError * _Nullable _error) { urlResponse = _response; erreur = _error; data = _data; reqProcessed = true; }] resume]; while (!reqProcessed) { [NSThread sleepForTimeInterval:0]; } *response = urlResponse; *error = erreur; return data; } #pragma mark - text view delegate - (BOOL)textFieldShouldReturn:(UITextField *)textField { [self.urlField resignFirstResponder]; return NO; } @end