VLCMovieViewController.m 63 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632
  1. /*****************************************************************************
  2. * VLCMovieViewController.m
  3. * VLC for iOS
  4. *****************************************************************************
  5. * Copyright (c) 2013-2014 VideoLAN. All rights reserved.
  6. * $Id$
  7. *
  8. * Authors: Felix Paul Kühne <fkuehne # videolan.org>
  9. * Gleb Pinigin <gpinigin # gmail.com>
  10. * Ahmad Harb <harb.dev.leb # gmail.com>
  11. * Fabio Ritrovato <sephiroth87 # videolan.org>
  12. * Pierre SAGASPE <pierre.sagaspe # me.com>
  13. * Jean-Baptiste Kempf <jb # videolan.org>
  14. *
  15. * Refer to the COPYING file of the official project for license.
  16. *****************************************************************************/
  17. #import "VLCMovieViewController.h"
  18. #import "VLCExternalDisplayController.h"
  19. #import <AVFoundation/AVFoundation.h>
  20. #import <CommonCrypto/CommonDigest.h>
  21. #import "UIDevice+SpeedCategory.h"
  22. #import "VLCBugreporter.h"
  23. #import "VLCThumbnailsCache.h"
  24. #import "OBSlider.h"
  25. #import "VLCStatusLabel.h"
  26. #define INPUT_RATE_DEFAULT 1000.
  27. #define FORWARD_SWIPE_DURATION 30
  28. #define BACKWARD_SWIPE_DURATION 10
  29. @interface VLCMovieViewController () <UIGestureRecognizerDelegate, AVAudioSessionDelegate, VLCMediaDelegate>
  30. {
  31. VLCMediaListPlayer *_listPlayer;
  32. VLCMediaPlayer *_mediaPlayer;
  33. BOOL _controlsHidden;
  34. BOOL _videoFiltersHidden;
  35. BOOL _playbackSpeedViewHidden;
  36. UIActionSheet *_subtitleActionSheet;
  37. UIActionSheet *_audiotrackActionSheet;
  38. float _currentPlaybackRate;
  39. NSArray *_aspectRatios;
  40. NSUInteger _currentAspectRatioMask;
  41. NSTimer *_idleTimer;
  42. BOOL _shouldResumePlaying;
  43. BOOL _viewAppeared;
  44. BOOL _displayRemainingTime;
  45. BOOL _positionSet;
  46. BOOL _playerIsSetup;
  47. BOOL _isScrubbing;
  48. BOOL _swipeGesturesEnabled;
  49. NSString * panType;
  50. UIPinchGestureRecognizer *_pinchRecognizer;
  51. UIPanGestureRecognizer *_panRecognizer;
  52. UISwipeGestureRecognizer *_swipeRecognizerLeft;
  53. UISwipeGestureRecognizer *_swipeRecognizerRight;
  54. UITapGestureRecognizer *_tapRecognizer;
  55. UITapGestureRecognizer *_tapOnVideoRecognizer;
  56. }
  57. @property (nonatomic, strong) UIPopoverController *masterPopoverController;
  58. @property (nonatomic, strong) UIWindow *externalWindow;
  59. @end
  60. @implementation VLCMovieViewController
  61. + (void)initialize
  62. {
  63. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  64. NSDictionary *appDefaults = @{kVLCShowRemainingTime : @(YES)};
  65. [defaults registerDefaults:appDefaults];
  66. }
  67. - (void)dealloc
  68. {
  69. [[NSNotificationCenter defaultCenter] removeObserver:self];
  70. if (_tapRecognizer)
  71. [self.view removeGestureRecognizer:_tapRecognizer];
  72. if (_swipeRecognizerLeft)
  73. [self.view removeGestureRecognizer:_swipeRecognizerLeft];
  74. if (_swipeRecognizerRight)
  75. [self.view removeGestureRecognizer:_swipeRecognizerRight];
  76. if (_panRecognizer)
  77. [self.view removeGestureRecognizer:_panRecognizer];
  78. if (_pinchRecognizer)
  79. [self.view removeGestureRecognizer:_pinchRecognizer];
  80. [self.view removeGestureRecognizer:_tapOnVideoRecognizer];
  81. _tapRecognizer = nil;
  82. _swipeRecognizerLeft = nil;
  83. _swipeRecognizerRight = nil;
  84. _panRecognizer = nil;
  85. _pinchRecognizer = nil;
  86. _tapOnVideoRecognizer = nil;
  87. [[NSNotificationCenter defaultCenter] removeObserver:self];
  88. }
  89. #pragma mark - Managing the media item
  90. - (void)setFileFromMediaLibrary:(id)newFile
  91. {
  92. if (_fileFromMediaLibrary != newFile) {
  93. [self _stopPlayback];
  94. _fileFromMediaLibrary = newFile;
  95. if (_viewAppeared)
  96. [self _startPlayback];
  97. }
  98. if (self.masterPopoverController != nil)
  99. [self.masterPopoverController dismissPopoverAnimated:YES];
  100. }
  101. - (void)setUrl:(NSURL *)url
  102. {
  103. [self _stopPlayback];
  104. _url = url;
  105. _playerIsSetup = NO;
  106. if (_viewAppeared)
  107. [self _startPlayback];
  108. }
  109. - (void)setMediaList:(VLCMediaList *)mediaList
  110. {
  111. [self _stopPlayback];
  112. _mediaList = mediaList;
  113. _playerIsSetup = NO;
  114. if (_viewAppeared)
  115. [self _startPlayback];
  116. }
  117. - (void)viewDidLoad
  118. {
  119. [super viewDidLoad];
  120. self.wantsFullScreenLayout = YES;
  121. self.videoFilterView.hidden = YES;
  122. _videoFiltersHidden = YES;
  123. _hueLabel.text = NSLocalizedString(@"VFILTER_HUE", nil);
  124. _hueSlider.accessibilityLabel = _hueLabel.text;
  125. _hueSlider.isAccessibilityElement = YES;
  126. _contrastLabel.text = NSLocalizedString(@"VFILTER_CONTRAST", nil);
  127. _contrastSlider.accessibilityLabel = _contrastLabel.text;
  128. _contrastSlider.isAccessibilityElement = YES;
  129. _brightnessLabel.text = NSLocalizedString(@"VFILTER_BRIGHTNESS", nil);
  130. _brightnessSlider.accessibilityLabel = _brightnessLabel.text;
  131. _brightnessSlider.isAccessibilityElement = YES;
  132. _saturationLabel.text = NSLocalizedString(@"VFILTER_SATURATION", nil);
  133. _saturationSlider.accessibilityLabel = _saturationLabel.text;
  134. _saturationSlider.isAccessibilityElement = YES;
  135. _gammaLabel.text = NSLocalizedString(@"VFILTER_GAMMA", nil);
  136. _gammaSlider.accessibilityLabel = _gammaLabel.text;
  137. _gammaSlider.isAccessibilityElement = YES;
  138. _playbackSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SPEED", nil);
  139. _playbackSpeedSlider.accessibilityLabel = _playbackSpeedLabel.text;
  140. _playbackSpeedSlider.isAccessibilityElement = YES;
  141. _positionSlider.accessibilityLabel = NSLocalizedString(@"PLAYBACK_POSITION", nil);
  142. _positionSlider.isAccessibilityElement = YES;
  143. _timeDisplay.isAccessibilityElement = YES;
  144. _audioSwitcherButton.accessibilityLabel = NSLocalizedString(@"CHOOSE_AUDIO_TRACK", nil);
  145. _audioSwitcherButton.isAccessibilityElement = YES;
  146. _audioSwitcherButtonLandscape.accessibilityLabel = NSLocalizedString(@"CHOOSE_AUDIO_TRACK", nil);
  147. _audioSwitcherButtonLandscape.isAccessibilityElement = YES;
  148. _subtitleSwitcherButton.accessibilityLabel = NSLocalizedString(@"CHOOSE_SUBTITLE_TRACK", nil);
  149. _subtitleSwitcherButton.isAccessibilityElement = YES;
  150. _subtitleSwitcherButtonLandscape.accessibilityLabel = NSLocalizedString(@"CHOOSE_SUBTITLE_TRACK", nil);
  151. _subtitleSwitcherButtonLandscape.isAccessibilityElement = YES;
  152. _playbackSpeedButton.accessibilityLabel = _playbackSpeedLabel.text;
  153. _playbackSpeedButton.isAccessibilityElement = YES;
  154. _playbackSpeedButtonLandscape.accessibilityLabel = _playbackSpeedLabel.text;
  155. _playbackSpeedButtonLandscape.isAccessibilityElement = YES;
  156. _videoFilterButton.accessibilityLabel = NSLocalizedString(@"VIDEO_FILTER", nil);
  157. _videoFilterButton.isAccessibilityElement = YES;
  158. _videoFilterButtonLandscape.accessibilityLabel = NSLocalizedString(@"VIDEO_FILTER", nil);
  159. _videoFilterButtonLandscape.isAccessibilityElement = YES;
  160. _resetVideoFilterButton.accessibilityLabel = NSLocalizedString(@"VIDEO_FILTER_RESET_BUTTON", nil);
  161. _resetVideoFilterButton.isAccessibilityElement = YES;
  162. _aspectRatioButton.accessibilityLabel = NSLocalizedString(@"VIDEO_ASPECT_RATIO_BUTTON", nil);
  163. _aspectRatioButton.isAccessibilityElement = YES;
  164. _playPauseButton.accessibilityLabel = NSLocalizedString(@"PLAY_PAUSE_BUTTON", nil);
  165. _playPauseButton.isAccessibilityElement = YES;
  166. _playPauseButtonLandscape.accessibilityLabel = NSLocalizedString(@"PLAY_PAUSE_BUTTON", nil);
  167. _playPauseButtonLandscape.isAccessibilityElement = YES;
  168. _bwdButton.accessibilityLabel = NSLocalizedString(@"BWD_BUTTON", nil);
  169. _bwdButton.isAccessibilityElement = YES;
  170. _bwdButtonLandscape.accessibilityLabel = NSLocalizedString(@"BWD_BUTTON", nil);
  171. _bwdButtonLandscape.isAccessibilityElement = YES;
  172. _fwdButton.accessibilityLabel = NSLocalizedString(@"FWD_BUTTON", nil);
  173. _fwdButton.isAccessibilityElement = YES;
  174. _fwdButtonLandscape.accessibilityLabel = NSLocalizedString(@"FWD_BUTTON", nil);
  175. _fwdButtonLandscape.isAccessibilityElement = YES;
  176. _repeatButton.accessibilityLabel = NSLocalizedString(@"BUTTON_REPEAT", nil);
  177. _repeatButton.isAccessibilityElement = YES;
  178. _scrubHelpLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HELP", nil);
  179. self.playbackSpeedView.hidden = YES;
  180. _playbackSpeedViewHidden = YES;
  181. NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
  182. [center addObserver:self selector:@selector(handleExternalScreenDidConnect:)
  183. name:UIScreenDidConnectNotification object:nil];
  184. [center addObserver:self selector:@selector(handleExternalScreenDidDisconnect:)
  185. name:UIScreenDidDisconnectNotification object:nil];
  186. [center addObserver:self selector:@selector(applicationWillResignActive:)
  187. name:UIApplicationWillResignActiveNotification object:nil];
  188. [center addObserver:self selector:@selector(applicationDidBecomeActive:)
  189. name:UIApplicationDidBecomeActiveNotification object:nil];
  190. [center addObserver:self selector:@selector(applicationDidEnterBackground:)
  191. name:UIApplicationDidEnterBackgroundNotification object:nil];
  192. [center addObserver:self selector:@selector(audioSessionRouteChange:)
  193. name:AVAudioSessionRouteChangeNotification object:nil];
  194. _playingExternallyTitle.text = NSLocalizedString(@"PLAYING_EXTERNALLY_TITLE", nil);
  195. _playingExternallyDescription.text = NSLocalizedString(@"PLAYING_EXTERNALLY_DESC", nil);
  196. if ([self hasExternalDisplay])
  197. [self showOnExternalDisplay];
  198. self.trackNameLabel.text = self.artistNameLabel.text = self.albumNameLabel.text = @"";
  199. _movieView.userInteractionEnabled = NO;
  200. _tapOnVideoRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(toggleControlsVisible)];
  201. _tapOnVideoRecognizer.delegate = self;
  202. [self.view addGestureRecognizer:_tapOnVideoRecognizer];
  203. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  204. _displayRemainingTime = [[defaults objectForKey:kVLCShowRemainingTime] boolValue];
  205. _swipeGesturesEnabled = [[defaults objectForKey:kVLCSettingPlaybackGestures] boolValue];
  206. _pinchRecognizer = [[UIPinchGestureRecognizer alloc] initWithTarget:self action:@selector(handlePinchGesture:)];
  207. _pinchRecognizer.delegate = self;
  208. [self.view addGestureRecognizer:_pinchRecognizer];
  209. _tapRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(tapRecognized)];
  210. [_tapRecognizer setNumberOfTouchesRequired:2];
  211. _panRecognizer = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(panRecognized:)];
  212. [_panRecognizer setMinimumNumberOfTouches:1];
  213. [_panRecognizer setMaximumNumberOfTouches:1];
  214. _swipeRecognizerLeft = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(swipeRecognized:)];
  215. _swipeRecognizerLeft.direction = UISwipeGestureRecognizerDirectionLeft;
  216. _swipeRecognizerRight = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(swipeRecognized:)];
  217. _swipeRecognizerRight.direction = UISwipeGestureRecognizerDirectionRight;
  218. [self.view addGestureRecognizer:_swipeRecognizerLeft];
  219. [self.view addGestureRecognizer:_swipeRecognizerRight];
  220. [self.view addGestureRecognizer:_panRecognizer];
  221. [self.view addGestureRecognizer:_tapRecognizer];
  222. [_panRecognizer requireGestureRecognizerToFail:_swipeRecognizerLeft];
  223. [_panRecognizer requireGestureRecognizerToFail:_swipeRecognizerRight];
  224. _panRecognizer.delegate = self;
  225. _swipeRecognizerRight.delegate = self;
  226. _swipeRecognizerLeft.delegate = self;
  227. _tapRecognizer.delegate = self;
  228. _aspectRatios = @[@"DEFAULT", @"FILL_TO_SCREEN", @"4:3", @"16:9", @"16:10", @"2.21:1"];
  229. [self.aspectRatioButton setImage:[UIImage imageNamed:@"ratioIcon"] forState:UIControlStateNormal];
  230. if (SYSTEM_RUNS_IOS7_OR_LATER) {
  231. self.backButton.tintColor = [UIColor colorWithRed:(190.0f/255.0f) green:(190.0f/255.0f) blue:(190.0f/255.0f) alpha:1.];
  232. self.toolbar.tintColor = [UIColor whiteColor];
  233. self.toolbar.barStyle = UIBarStyleBlack;
  234. CGRect rect = self.resetVideoFilterButton.frame;
  235. rect.origin.y = rect.origin.y + 5.;
  236. self.resetVideoFilterButton.frame = rect;
  237. rect = self.toolbar.frame;
  238. rect.size.height = rect.size.height + rect.origin.y;
  239. rect.origin.y = 0;
  240. self.toolbar.frame = rect;
  241. rect = self.aspectRatioButton.frame;
  242. rect.size.width -= 19.;
  243. rect.origin.x += 19.;
  244. self.aspectRatioButton.frame = rect;
  245. rect = self.timeDisplay.frame;
  246. rect.origin.x += 19.;
  247. self.timeDisplay.frame = rect;
  248. rect = self.positionSlider.frame;
  249. rect.size.width += 19.;
  250. self.positionSlider.frame = rect;
  251. } else {
  252. CGRect rect = self.positionSlider.frame;
  253. rect.origin.y = rect.origin.y + 3.;
  254. self.positionSlider.frame = rect;
  255. [self.aspectRatioButton setBackgroundImage:[UIImage imageNamed:@"ratioButton"] forState:UIControlStateNormal];
  256. [self.aspectRatioButton setBackgroundImage:[UIImage imageNamed:@"ratioButtonHighlight"] forState:UIControlStateHighlighted];
  257. [self.toolbar setBackgroundImage:[UIImage imageNamed:@"seekbarBg"] forBarMetrics:UIBarMetricsDefault];
  258. [self.backButton setBackgroundImage:[UIImage imageNamed:@"playbackDoneButton"] forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
  259. [self.backButton setBackgroundImage:[UIImage imageNamed:@"playbackDoneButtonHighlight"] forState:UIControlStateHighlighted barMetrics:UIBarMetricsDefault];
  260. }
  261. /* FIXME: there is a saner iOS 6+ API for this! */
  262. /* this looks a bit weird, but we need to support iOS 5 and should show the same appearance */
  263. void (^initVolumeSlider)(MPVolumeView *) = ^(MPVolumeView *volumeView){
  264. UISlider *volumeSlider = nil;
  265. for (id aView in volumeView.subviews){
  266. if ([[[aView class] description] isEqualToString:@"MPVolumeSlider"]){
  267. volumeSlider = (UISlider *)aView;
  268. break;
  269. }
  270. }
  271. if (!SYSTEM_RUNS_IOS7_OR_LATER) {
  272. [volumeSlider setMinimumTrackImage:[[UIImage imageNamed:@"sliderminiValue"]resizableImageWithCapInsets:UIEdgeInsetsMake(0, 4, 0, 0)] forState:UIControlStateNormal];
  273. [volumeSlider setMaximumTrackImage:[[UIImage imageNamed:@"slidermaxValue"] resizableImageWithCapInsets:UIEdgeInsetsMake(0, 0, 0, 4)] forState:UIControlStateNormal];
  274. [volumeSlider setThumbImage:[UIImage imageNamed:@"volumeballslider"] forState:UIControlStateNormal];
  275. } else
  276. [volumeView setVolumeThumbImage:[UIImage imageNamed:@"modernSliderKnob"] forState:UIControlStateNormal];
  277. [volumeSlider addTarget:self
  278. action:@selector(volumeSliderAction:)
  279. forControlEvents:UIControlEventValueChanged];
  280. };
  281. initVolumeSlider(self.volumeView);
  282. initVolumeSlider(self.volumeViewLandscape);
  283. [[AVAudioSession sharedInstance] setDelegate:self];
  284. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad)
  285. self.positionSlider.scrubbingSpeedChangePositions = @[@(0.), @(100.), @(200.), @(300)];
  286. _playerIsSetup = NO;
  287. [self.movieView setAccessibilityLabel:NSLocalizedString(@"VO_VIDEOPLAYER_TITLE", nil)];
  288. [self.movieView setAccessibilityHint:NSLocalizedString(@"VO_VIDEOPLAYER_DOUBLETAP", nil)];
  289. }
  290. - (BOOL)_blobCheck
  291. {
  292. NSArray *searchPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
  293. NSString *directoryPath = searchPaths[0];
  294. if (![[NSFileManager defaultManager] fileExistsAtPath:[directoryPath stringByAppendingPathComponent:@"blob.bin"]])
  295. return NO;
  296. NSData *data = [NSData dataWithContentsOfFile:[directoryPath stringByAppendingPathComponent:@"blob.bin"]];
  297. uint8_t digest[CC_SHA1_DIGEST_LENGTH];
  298. CC_SHA1(data.bytes, (unsigned int)data.length, digest);
  299. NSMutableString *hash = [NSMutableString stringWithCapacity:CC_SHA1_DIGEST_LENGTH * 2];
  300. for (unsigned int u = 0; u < CC_SHA1_DIGEST_LENGTH; u++)
  301. [hash appendFormat:@"%02x", digest[u]];
  302. if ([hash isEqualToString:kBlobHash])
  303. return YES;
  304. else
  305. return NO;
  306. }
  307. - (void)viewWillAppear:(BOOL)animated
  308. {
  309. [super viewWillAppear:animated];
  310. /* reset audio meta data views */
  311. self.artworkImageView.image = nil;
  312. self.trackNameLabel.text = nil;
  313. self.artistNameLabel.text = nil;
  314. self.albumNameLabel.text = nil;
  315. _swipeGesturesEnabled = [[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingPlaybackGestures] boolValue];
  316. [self.navigationController setNavigationBarHidden:YES animated:YES];
  317. if (!SYSTEM_RUNS_IOS7_OR_LATER) {
  318. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone)
  319. [UIApplication sharedApplication].statusBarStyle = UIStatusBarStyleBlackTranslucent;
  320. }
  321. [self _startPlayback];
  322. [self setControlsHidden:NO animated:YES];
  323. _viewAppeared = YES;
  324. }
  325. - (void)viewWillLayoutSubviews
  326. {
  327. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  328. CGSize viewSize = self.view.frame.size;
  329. if (UIDeviceOrientationIsLandscape([UIDevice currentDevice].orientation)) {
  330. [_controllerPanel removeFromSuperview];
  331. _controllerPanelLandscape.frame = (CGRect){CGPointMake(0, viewSize.height - _controllerPanelLandscape.frame.size.height), CGSizeMake(viewSize.width, _controllerPanelLandscape.frame.size.height)};
  332. [self.view addSubview:_controllerPanelLandscape];
  333. } else {
  334. [_controllerPanelLandscape removeFromSuperview];
  335. _controllerPanel.frame = (CGRect){CGPointMake(0, viewSize.height - _controllerPanel.frame.size.height), CGSizeMake(viewSize.width, _controllerPanel.frame.size.height)};
  336. [self.view addSubview:_controllerPanel];
  337. }
  338. }
  339. }
  340. - (void)_startPlayback
  341. {
  342. if (_playerIsSetup)
  343. return;
  344. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  345. if (!self.fileFromMediaLibrary && !self.url && !self.mediaList) {
  346. [self _stopPlayback];
  347. return;
  348. }
  349. if (self.pathToExternalSubtitlesFile)
  350. _listPlayer = [[VLCMediaListPlayer alloc] initWithOptions:@[[NSString stringWithFormat:@"--%@=%@", kVLCSettingSubtitlesFilePath, self.pathToExternalSubtitlesFile]]];
  351. else
  352. _listPlayer = [[VLCMediaListPlayer alloc] init];
  353. _mediaPlayer = _listPlayer.mediaPlayer;
  354. [_mediaPlayer setDelegate:self];
  355. [_mediaPlayer setDrawable:self.movieView];
  356. if ([[defaults objectForKey:kVLCSettingPlaybackSpeedDefaultValue] floatValue] != 0)
  357. [_mediaPlayer setRate: [[defaults objectForKey:kVLCSettingPlaybackSpeedDefaultValue] floatValue]];
  358. if ([[defaults objectForKey:kVLCSettingDeinterlace] intValue] != 0)
  359. [_mediaPlayer setDeinterlaceFilter:@"blend"];
  360. else
  361. [_mediaPlayer setDeinterlaceFilter:nil];
  362. if (self.pathToExternalSubtitlesFile)
  363. [_mediaPlayer openVideoSubTitlesFromFile:self.pathToExternalSubtitlesFile];
  364. self.trackNameLabel.text = self.artistNameLabel.text = self.albumNameLabel.text = @"";
  365. VLCMedia *media;
  366. if (self.fileFromMediaLibrary) {
  367. MLFile *item = self.fileFromMediaLibrary;
  368. media = [VLCMedia mediaWithURL:[NSURL URLWithString:item.url]];
  369. } else if (self.mediaList) {
  370. media = [self.mediaList mediaAtIndex:self.itemInMediaListToBePlayedFirst];
  371. [media parse];
  372. } else {
  373. media = [VLCMedia mediaWithURL:self.url];
  374. [media parse];
  375. }
  376. NSMutableDictionary *mediaDictionary = [[NSMutableDictionary alloc] init];
  377. [mediaDictionary setObject:[defaults objectForKey:kVLCSettingNetworkCaching] forKey:kVLCSettingNetworkCaching];
  378. [mediaDictionary setObject:[[defaults objectForKey:kVLCSettingStretchAudio] boolValue] ? kVLCSettingStretchAudioOnValue : kVLCSettingStretchAudioOffValue forKey:kVLCSettingStretchAudio];
  379. [mediaDictionary setObject:[defaults objectForKey:kVLCSettingTextEncoding] forKey:kVLCSettingTextEncoding];
  380. [mediaDictionary setObject:[defaults objectForKey:kVLCSettingSkipLoopFilter] forKey:kVLCSettingSkipLoopFilter];
  381. [NSTimeZone resetSystemTimeZone];
  382. NSString *tzName = [[NSTimeZone systemTimeZone] name];
  383. NSArray *tzNames = @[@"America/Adak", @"America/Anchorage", @"America/Boise", @"America/Chicago", @"America/Denver", @"America/Detroit", @"America/Indiana/Indianapolis", @"America/Indiana/Knox", @"America/Indiana/Marengo", @"America/Indiana/Petersburg", @"America/Indiana/Tell_City", @"America/Indiana/Vevay", @"America/Indiana/Vincennes", @"America/Indiana/Winamac", @"America/Juneau", @"America/Kentucky/Louisville", @"America/Kentucky/Monticello", @"America/Los_Angeles", @"America/Menominee", @"America/Metlakatla", @"America/New_York", @"America/Nome", @"America/North_Dakota/Beulah", @"America/North_Dakota/Center", @"America/North_Dakota/New_Salem", @"America/Phoenix", @"America/Puerto_Rico", @"America/Shiprock", @"America/Sitka", @"America/St_Thomas", @"America/Thule", @"America/Yakutat", @"Pacific/Guam", @"Pacific/Honolulu", @"Pacific/Johnston", @"Pacific/Kwajalein", @"Pacific/Midway", @"Pacific/Pago_Pago", @"Pacific/Saipan", @"Pacific/Wake"];
  384. if ([tzNames containsObject:tzName] || [[tzName stringByDeletingLastPathComponent] isEqualToString:@"US"]) {
  385. NSArray *tracksInfo = media.tracksInformation;
  386. for (NSUInteger x = 0; x < tracksInfo.count; x++) {
  387. if ([[tracksInfo[x] objectForKey:VLCMediaTracksInformationType] isEqualToString:VLCMediaTracksInformationTypeAudio])
  388. {
  389. NSInteger fourcc = [[tracksInfo[x] objectForKey:VLCMediaTracksInformationCodec] integerValue];
  390. switch (fourcc) {
  391. case 540161377:
  392. case 1647457633:
  393. case 858612577:
  394. case 862151027:
  395. case 862151013:
  396. case 1684566644:
  397. case 2126701:
  398. {
  399. if (![self _blobCheck]) {
  400. [mediaDictionary setObject:[NSNull null] forKey:@"no-audio"];
  401. APLog(@"audio playback disabled because an unsupported codec was found");
  402. }
  403. break;
  404. }
  405. default:
  406. break;
  407. }
  408. }
  409. }
  410. }
  411. if (self.mediaList) {
  412. VLCMediaList *list = self.mediaList;
  413. NSUInteger count = list.count;
  414. for (NSUInteger x = 0; x < count; x++)
  415. [[list mediaAtIndex:x] addOptions:mediaDictionary];
  416. [_listPlayer setMediaList:self.mediaList];
  417. } else {
  418. [media addOptions:mediaDictionary];
  419. [_listPlayer setRootMedia:media];
  420. }
  421. [_listPlayer setRepeatMode:VLCDoNotRepeat];
  422. self.positionSlider.value = 0.;
  423. [self.timeDisplay setTitle:@"" forState:UIControlStateNormal];
  424. self.timeDisplay.accessibilityLabel = @"";
  425. [self.repeatButton setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  426. [self.repeatButtonLandscape setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  427. if (![self _isMediaSuitableForDevice]) {
  428. UIAlertView * alert = [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"DEVICE_TOOSLOW_TITLE", nil) message:[NSString stringWithFormat:NSLocalizedString(@"DEVICE_TOOSLOW", nil), [[UIDevice currentDevice] model], self.fileFromMediaLibrary.title] delegate:self cancelButtonTitle:NSLocalizedString(@"BUTTON_CANCEL", nil) otherButtonTitles:NSLocalizedString(@"BUTTON_OPEN", nil), nil];
  429. [alert show];
  430. } else
  431. [self _playNewMedia];
  432. if (![self hasExternalDisplay])
  433. self.brightnessSlider.value = [UIScreen mainScreen].brightness * 2.;
  434. }
  435. - (BOOL)_isMediaSuitableForDevice
  436. {
  437. if (!self.fileFromMediaLibrary)
  438. return YES;
  439. NSUInteger totalNumberOfPixels = [[[self.fileFromMediaLibrary videoTrack] valueForKey:@"width"] doubleValue] * [[[self.fileFromMediaLibrary videoTrack] valueForKey:@"height"] doubleValue];
  440. NSInteger speedCategory = [[UIDevice currentDevice] speedCategory];
  441. if (speedCategory == 1) {
  442. // iPhone 3GS, iPhone 4, first gen. iPad, 3rd and 4th generation iPod touch
  443. return (totalNumberOfPixels < 600000); // between 480p and 720p
  444. } else if (speedCategory == 2) {
  445. // iPhone 4S, iPad 2 and 3, iPod 4 and 5
  446. return (totalNumberOfPixels < 922000); // 720p
  447. } else if (speedCategory == 3) {
  448. // iPhone 5, iPad 4
  449. return (totalNumberOfPixels < 2074000); // 1080p
  450. } else if (speedCategory == 4) {
  451. // iPhone 6, 2014 iPads
  452. return (totalNumberOfPixels < 8850000); // 4K
  453. }
  454. return YES;
  455. }
  456. - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
  457. {
  458. if (buttonIndex == 1)
  459. [self _playNewMedia];
  460. else {
  461. [self _stopPlayback];
  462. [self closePlayback:nil];
  463. }
  464. }
  465. - (void)_playNewMedia
  466. {
  467. NSNumber *playbackPositionInTime = @(0);
  468. CGFloat lastPosition = .0;
  469. NSInteger duration = 0;
  470. MLFile *matchedFile;
  471. if (self.fileFromMediaLibrary)
  472. matchedFile = self.fileFromMediaLibrary;
  473. else if (self.mediaList) {
  474. NSArray *matches = [MLFile fileForURL:[[[self.mediaList mediaAtIndex:self.itemInMediaListToBePlayedFirst] url] absoluteString]];
  475. if (matches.count > 0) {
  476. matchedFile = matches[0];
  477. lastPosition = matchedFile.lastPosition.floatValue;
  478. }
  479. }
  480. if (matchedFile.lastPosition)
  481. lastPosition = matchedFile.lastPosition.floatValue;
  482. duration = matchedFile.duration.intValue;
  483. if (lastPosition < .95) {
  484. if (duration != 0)
  485. playbackPositionInTime = @(lastPosition * (duration / 1000.));
  486. }
  487. if (playbackPositionInTime.intValue > 0 && (duration * lastPosition - duration) < -60000) {
  488. [_mediaPlayer.media addOptions:@{@"start-time": playbackPositionInTime}];
  489. APLog(@"set starttime to %i", playbackPositionInTime.intValue);
  490. }
  491. [_mediaPlayer addObserver:self forKeyPath:@"time" options:0 context:nil];
  492. [_mediaPlayer addObserver:self forKeyPath:@"remainingTime" options:0 context:nil];
  493. if (self.mediaList)
  494. [_listPlayer playItemAtIndex:self.itemInMediaListToBePlayedFirst];
  495. else
  496. [_listPlayer playMedia:_listPlayer.rootMedia];
  497. if (matchedFile) {
  498. if (matchedFile.lastAudioTrack.intValue > 0)
  499. _mediaPlayer.currentAudioTrackIndex = matchedFile.lastAudioTrack.intValue;
  500. if (matchedFile.lastSubtitleTrack.intValue > 0)
  501. _mediaPlayer.currentVideoSubTitleIndex = matchedFile.lastSubtitleTrack.intValue;
  502. }
  503. self.playbackSpeedSlider.value = [self _playbackSpeed];
  504. [self _updatePlaybackSpeedIndicator];
  505. _currentAspectRatioMask = 0;
  506. _mediaPlayer.videoAspectRatio = NULL;
  507. /* some demuxers don't respect :start-time, so re-try here */
  508. if (lastPosition < .95 && _mediaPlayer.position < lastPosition && (duration * lastPosition - duration) < -60000)
  509. _mediaPlayer.position = lastPosition;
  510. [self _resetIdleTimer];
  511. _playerIsSetup = YES;
  512. }
  513. - (void)viewWillDisappear:(BOOL)animated
  514. {
  515. [self _stopPlayback];
  516. _viewAppeared = NO;
  517. if (_idleTimer) {
  518. [_idleTimer invalidate];
  519. _idleTimer = nil;
  520. }
  521. [self.navigationController setNavigationBarHidden:NO animated:YES];
  522. if (!SYSTEM_RUNS_IOS7_OR_LATER)
  523. [UIApplication sharedApplication].statusBarStyle = UIStatusBarStyleBlackOpaque;
  524. [[UIApplication sharedApplication] setStatusBarHidden:NO withAnimation:UIStatusBarAnimationFade];
  525. [super viewWillDisappear:animated];
  526. // hide filter UI for next run
  527. if (!_videoFiltersHidden)
  528. _videoFiltersHidden = YES;
  529. if (!_playbackSpeedViewHidden)
  530. _playbackSpeedViewHidden = YES;
  531. }
  532. - (void)_stopPlayback
  533. {
  534. if (_mediaPlayer) {
  535. @try {
  536. [_mediaPlayer removeObserver:self forKeyPath:@"time"];
  537. [_mediaPlayer removeObserver:self forKeyPath:@"remainingTime"];
  538. }
  539. @catch (NSException *exception) {
  540. APLog(@"we weren't an observer yet");
  541. }
  542. if (_mediaPlayer.media) {
  543. [_mediaPlayer pause];
  544. [self _saveCurrentState];
  545. [_mediaPlayer stop];
  546. }
  547. if (_mediaPlayer)
  548. _mediaPlayer = nil;
  549. if (_listPlayer)
  550. _listPlayer = nil;
  551. }
  552. if (_fileFromMediaLibrary)
  553. _fileFromMediaLibrary = nil;
  554. if (_mediaList)
  555. _mediaList = nil;
  556. if (_url)
  557. _url = nil;
  558. if (_pathToExternalSubtitlesFile) {
  559. NSFileManager *fileManager = [NSFileManager defaultManager];
  560. if ([fileManager fileExistsAtPath:_pathToExternalSubtitlesFile])
  561. [fileManager removeItemAtPath:_pathToExternalSubtitlesFile error:nil];
  562. _pathToExternalSubtitlesFile = nil;
  563. }
  564. _playerIsSetup = NO;
  565. }
  566. - (void)_saveCurrentState
  567. {
  568. if (self.fileFromMediaLibrary) {
  569. @try {
  570. MLFile *item = self.fileFromMediaLibrary;
  571. item.lastPosition = @([_mediaPlayer position]);
  572. item.lastAudioTrack = @(_mediaPlayer.currentAudioTrackIndex);
  573. item.lastSubtitleTrack = @(_mediaPlayer.currentVideoSubTitleIndex);
  574. }
  575. @catch (NSException *exception) {
  576. APLog(@"failed to save current media state - file removed?");
  577. }
  578. } else {
  579. NSArray *files = [MLFile fileForURL:[[_mediaPlayer.media url] absoluteString]];
  580. if (files.count > 0) {
  581. MLFile *fileFromList = files[0];
  582. fileFromList.lastPosition = @([_mediaPlayer position]);
  583. fileFromList.lastAudioTrack = @(_mediaPlayer.currentAudioTrackIndex);
  584. fileFromList.lastSubtitleTrack = @(_mediaPlayer.currentVideoSubTitleIndex);
  585. }
  586. }
  587. }
  588. - (NSString *)_resolveFontName
  589. {
  590. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  591. BOOL bold = [[defaults objectForKey:kVLCSettingSubtitlesBoldFont] boolValue];
  592. NSString *font = [defaults objectForKey:kVLCSettingSubtitlesFont];
  593. NSDictionary *fontMap = @{
  594. @"AmericanTypewriter": @"AmericanTypewriter-Bold",
  595. @"ArialMT": @"Arial-BoldMT",
  596. @"ArialHebrew": @"ArialHebrew-Bold",
  597. @"ChalkboardSE-Regular": @"ChalkboardSE-Bold",
  598. @"CourierNewPSMT": @"CourierNewPS-BoldMT",
  599. @"Georgia": @"Georgia-Bold",
  600. @"GillSans": @"GillSans-Bold",
  601. @"GujaratiSangamMN": @"GujaratiSangamMN-Bold",
  602. @"STHeitiSC-Light": @"STHeitiSC-Medium",
  603. @"STHeitiTC-Light": @"STHeitiTC-Medium",
  604. @"HelveticaNeue": @"HelveticaNeue-Bold",
  605. @"HiraKakuProN-W3": @"HiraKakuProN-W6",
  606. @"HiraMinProN-W3": @"HiraMinProN-W6",
  607. @"HoeflerText-Regular": @"HoeflerText-Black",
  608. @"Kailasa": @"Kailasa-Bold",
  609. @"KannadaSangamMN": @"KannadaSangamMN-Bold",
  610. @"MalayalamSangamMN": @"MalayalamSangamMN-Bold",
  611. @"OriyaSangamMN": @"OriyaSangamMN-Bold",
  612. @"SinhalaSangamMN": @"SinhalaSangamMN-Bold",
  613. @"SnellRoundhand": @"SnellRoundhand-Bold",
  614. @"TamilSangamMN": @"TamilSangamMN-Bold",
  615. @"TeluguSangamMN": @"TeluguSangamMN-Bold",
  616. @"TimesNewRomanPSMT": @"TimesNewRomanPS-BoldMT",
  617. @"Zapfino": @"Zapfino"
  618. };
  619. if (!bold) {
  620. return font;
  621. } else {
  622. return fontMap[font];
  623. }
  624. }
  625. #pragma mark - remote events
  626. - (void)viewDidAppear:(BOOL)animated
  627. {
  628. [super viewDidAppear:animated];
  629. [[UIApplication sharedApplication] beginReceivingRemoteControlEvents];
  630. [self becomeFirstResponder];
  631. }
  632. - (void)viewDidDisappear:(BOOL)animated
  633. {
  634. [super viewDidDisappear:animated];
  635. [[UIApplication sharedApplication] endReceivingRemoteControlEvents];
  636. [self resignFirstResponder];
  637. [[NSUserDefaults standardUserDefaults] setBool:_displayRemainingTime forKey:kVLCShowRemainingTime];
  638. }
  639. - (BOOL)canBecomeFirstResponder
  640. {
  641. return YES;
  642. }
  643. - (void)remoteControlReceivedWithEvent:(UIEvent *)event
  644. {
  645. switch (event.subtype) {
  646. case UIEventSubtypeRemoteControlPlay:
  647. [_listPlayer play];
  648. break;
  649. case UIEventSubtypeRemoteControlPause:
  650. [_listPlayer pause];
  651. break;
  652. case UIEventSubtypeRemoteControlTogglePlayPause:
  653. [self playPause];
  654. break;
  655. case UIEventSubtypeRemoteControlNextTrack:
  656. [self forward:nil];
  657. break;
  658. case UIEventSubtypeRemoteControlPreviousTrack:
  659. [self backward:nil];
  660. break;
  661. case UIEventSubtypeRemoteControlStop:
  662. [self closePlayback:nil];
  663. break;
  664. default:
  665. break;
  666. }
  667. }
  668. #pragma mark - controls visibility
  669. - (void)handlePinchGesture:(UIPinchGestureRecognizer *)recognizer
  670. {
  671. if (!_swipeGesturesEnabled)
  672. return;
  673. if (recognizer.velocity < 0.)
  674. [self closePlayback:nil];
  675. }
  676. - (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch
  677. {
  678. if (touch.view != self.view)
  679. return NO;
  680. return YES;
  681. }
  682. - (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer
  683. {
  684. return YES;
  685. }
  686. - (void)setControlsHidden:(BOOL)hidden animated:(BOOL)animated
  687. {
  688. _controlsHidden = hidden;
  689. CGFloat alpha = _controlsHidden? 0.0f: 1.0f;
  690. if (!_controlsHidden) {
  691. _controllerPanel.alpha = 0.0f;
  692. _controllerPanel.hidden = !_videoFiltersHidden;
  693. _controllerPanelLandscape.alpha = 0.0f;
  694. _controllerPanelLandscape.hidden = !_videoFiltersHidden;
  695. _toolbar.alpha = 0.0f;
  696. _toolbar.hidden = NO;
  697. _videoFilterView.alpha = 0.0f;
  698. _videoFilterView.hidden = _videoFiltersHidden;
  699. _playbackSpeedView.alpha = 0.0f;
  700. _playbackSpeedView.hidden = _playbackSpeedViewHidden;
  701. }
  702. void (^animationBlock)() = ^() {
  703. _controllerPanel.alpha = alpha;
  704. _controllerPanelLandscape.alpha = alpha;
  705. _toolbar.alpha = alpha;
  706. _videoFilterView.alpha = alpha;
  707. _playbackSpeedView.alpha = alpha;
  708. };
  709. void (^completionBlock)(BOOL finished) = ^(BOOL finished) {
  710. _controllerPanel.hidden = _videoFiltersHidden ? _controlsHidden : NO;
  711. _controllerPanelLandscape.hidden = _videoFiltersHidden ? _controlsHidden : NO;
  712. _toolbar.hidden = _controlsHidden;
  713. _videoFilterView.hidden = _videoFiltersHidden;
  714. _playbackSpeedView.hidden = _playbackSpeedViewHidden;
  715. };
  716. UIStatusBarAnimation animationType = animated? UIStatusBarAnimationFade: UIStatusBarAnimationNone;
  717. NSTimeInterval animationDuration = animated? 0.3: 0.0;
  718. [[UIApplication sharedApplication] setStatusBarHidden:_viewAppeared ? _controlsHidden : NO withAnimation:animationType];
  719. [UIView animateWithDuration:animationDuration animations:animationBlock completion:completionBlock];
  720. _volumeView.hidden = _volumeViewLandscape.hidden = _controllerPanel.hidden;
  721. }
  722. - (void)toggleControlsVisible
  723. {
  724. if (_controlsHidden && !_videoFiltersHidden)
  725. _videoFiltersHidden = YES;
  726. [self setControlsHidden:!_controlsHidden animated:YES];
  727. }
  728. - (void)_resetIdleTimer
  729. {
  730. if (!_idleTimer)
  731. _idleTimer = [NSTimer scheduledTimerWithTimeInterval:4.
  732. target:self
  733. selector:@selector(idleTimerExceeded)
  734. userInfo:nil
  735. repeats:NO];
  736. else {
  737. if (fabs([_idleTimer.fireDate timeIntervalSinceNow]) < 4.)
  738. [_idleTimer setFireDate:[NSDate dateWithTimeIntervalSinceNow:4.]];
  739. }
  740. }
  741. - (void)idleTimerExceeded
  742. {
  743. _idleTimer = nil;
  744. if (!_controlsHidden)
  745. [self toggleControlsVisible];
  746. if (!_videoFiltersHidden)
  747. _videoFiltersHidden = YES;
  748. if (!_playbackSpeedViewHidden)
  749. _playbackSpeedViewHidden = YES;
  750. if (self.scrubIndicatorView.hidden == NO)
  751. self.scrubIndicatorView.hidden = YES;
  752. }
  753. - (UIResponder *)nextResponder
  754. {
  755. [self _resetIdleTimer];
  756. return [super nextResponder];
  757. }
  758. #pragma mark - controls
  759. - (IBAction)closePlayback:(id)sender
  760. {
  761. [self setControlsHidden:NO animated:NO];
  762. [self.navigationController dismissViewControllerAnimated:YES completion:nil];
  763. }
  764. - (IBAction)positionSliderAction:(UISlider *)sender
  765. {
  766. /* we need to limit the number of events sent by the slider, since otherwise, the user
  767. * wouldn't see the I-frames when seeking on current mobile devices. This isn't a problem
  768. * within the Simulator, but especially on older ARMv7 devices, it's clearly noticeable. */
  769. [self performSelector:@selector(_setPositionForReal) withObject:nil afterDelay:0.3];
  770. VLCTime *newPosition = [VLCTime timeWithInt:(int)(_positionSlider.value * self.fileFromMediaLibrary.duration.intValue)];
  771. [self.timeDisplay setTitle:newPosition.stringValue forState:UIControlStateNormal];
  772. self.timeDisplay.accessibilityLabel = [NSString stringWithFormat:@"%@: %@", NSLocalizedString(@"PLAYBACK_POSITION", nil), newPosition.stringValue];
  773. _positionSet = NO;
  774. [self _resetIdleTimer];
  775. }
  776. - (void)_setPositionForReal
  777. {
  778. if (!_positionSet) {
  779. _mediaPlayer.position = _positionSlider.value;
  780. _positionSet = YES;
  781. }
  782. }
  783. - (IBAction)positionSliderTouchDown:(id)sender
  784. {
  785. [self _updateScrubLabel];
  786. self.scrubIndicatorView.hidden = NO;
  787. _isScrubbing = YES;
  788. }
  789. - (IBAction)positionSliderTouchUp:(id)sender
  790. {
  791. self.scrubIndicatorView.hidden = YES;
  792. _isScrubbing = NO;
  793. }
  794. - (void)_updateScrubLabel
  795. {
  796. float speed = self.positionSlider.scrubbingSpeed;
  797. if (speed == 1.)
  798. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HIGH", nil);
  799. else if (speed == .5)
  800. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HALF", nil);
  801. else if (speed == .25)
  802. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_QUARTER", nil);
  803. else
  804. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_FINE", nil);
  805. [self _resetIdleTimer];
  806. }
  807. - (IBAction)positionSliderDrag:(id)sender
  808. {
  809. [self _updateScrubLabel];
  810. }
  811. - (IBAction)volumeSliderAction:(id)sender
  812. {
  813. [self _resetIdleTimer];
  814. }
  815. - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
  816. {
  817. if (!_isScrubbing) {
  818. self.positionSlider.value = [_mediaPlayer position];
  819. }
  820. if (_displayRemainingTime)
  821. [self.timeDisplay setTitle:[[_mediaPlayer remainingTime] stringValue] forState:UIControlStateNormal];
  822. else
  823. [self.timeDisplay setTitle:[[_mediaPlayer time] stringValue] forState:UIControlStateNormal];
  824. }
  825. - (void)mediaPlayerStateChanged:(NSNotification *)aNotification
  826. {
  827. VLCMediaPlayerState currentState = _mediaPlayer.state;
  828. if (currentState == VLCMediaPlayerStateBuffering) {
  829. /* attach delegate */
  830. _mediaPlayer.media.delegate = self;
  831. /* let's update meta data */
  832. [self _updateDisplayedMetadata];
  833. /* on-the-fly values through private API */
  834. [_mediaPlayer performSelector:@selector(setTextRendererFont:) withObject:[self _resolveFontName]];
  835. [_mediaPlayer performSelector:@selector(setTextRendererFontSize:) withObject:[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingSubtitlesFontSize]];
  836. [_mediaPlayer performSelector:@selector(setTextRendererFontColor:) withObject:[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingSubtitlesFontColor]];
  837. }
  838. if (currentState == VLCMediaPlayerStateError) {
  839. [self.statusLabel showStatusMessage:NSLocalizedString(@"PLAYBACK_FAILED", nil)];
  840. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  841. }
  842. if ((currentState == VLCMediaPlayerStateEnded || currentState == VLCMediaPlayerStateStopped) && _listPlayer.repeatMode == VLCDoNotRepeat) {
  843. if ([_listPlayer.mediaList indexOfMedia:_mediaPlayer.media] == _listPlayer.mediaList.count - 1)
  844. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  845. }
  846. UIImage *playPauseImage = [_mediaPlayer isPlaying]? [UIImage imageNamed:@"pauseIcon"] : [UIImage imageNamed:@"playIcon"];
  847. [_playPauseButton setImage:playPauseImage forState:UIControlStateNormal];
  848. [_playPauseButtonLandscape setImage:playPauseImage forState:UIControlStateNormal];
  849. if ([[_mediaPlayer audioTrackIndexes] count] > 2) {
  850. self.audioSwitcherButton.hidden = NO;
  851. self.audioSwitcherButtonLandscape.hidden = NO;
  852. } else {
  853. self.audioSwitcherButton.hidden = YES;
  854. self.audioSwitcherButtonLandscape.hidden = YES;
  855. }
  856. if ([[_mediaPlayer videoSubTitlesIndexes] count] > 1) {
  857. self.subtitleContainer.hidden = NO;
  858. self.subtitleContainerLandscape.hidden = NO;
  859. } else {
  860. self.subtitleContainer.hidden = YES;
  861. self.subtitleContainerLandscape.hidden = YES;
  862. }
  863. }
  864. - (IBAction)playPause
  865. {
  866. if ([_mediaPlayer isPlaying])
  867. [_listPlayer pause];
  868. else
  869. [_listPlayer play];
  870. }
  871. - (IBAction)forward:(id)sender
  872. {
  873. if (self.mediaList)
  874. [_listPlayer next];
  875. else
  876. [_mediaPlayer mediumJumpForward];
  877. }
  878. - (IBAction)backward:(id)sender
  879. {
  880. if (self.mediaList)
  881. [_listPlayer previous];
  882. else
  883. [_mediaPlayer mediumJumpBackward];
  884. }
  885. - (void)toggleRepeatMode:(id)sender
  886. {
  887. if (_listPlayer.repeatMode == VLCDoNotRepeat) {
  888. _listPlayer.repeatMode = VLCRepeatCurrentItem;
  889. [self.repeatButton setImage:[UIImage imageNamed:@"repeatOne"] forState:UIControlStateNormal];
  890. [self.repeatButtonLandscape setImage:[UIImage imageNamed:@"repeatOne"] forState:UIControlStateNormal];
  891. } else {
  892. _listPlayer.repeatMode = VLCDoNotRepeat;
  893. [self.repeatButton setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  894. [self.repeatButtonLandscape setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  895. }
  896. }
  897. - (IBAction)switchAudioTrack:(id)sender
  898. {
  899. _audiotrackActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_AUDIO_TRACK", @"audio track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  900. NSArray *audioTracks = [_mediaPlayer audioTrackNames];
  901. NSArray *audioTrackIndexes = [_mediaPlayer audioTrackIndexes];
  902. NSUInteger count = [audioTracks count];
  903. for (NSUInteger i = 0; i < count; i++) {
  904. NSString *indexIndicator = ([audioTrackIndexes[i] intValue] == [_mediaPlayer currentAudioTrackIndex])? @"\u2713": @"";
  905. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, audioTracks[i]];
  906. [_audiotrackActionSheet addButtonWithTitle:buttonTitle];
  907. }
  908. [_audiotrackActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  909. [_audiotrackActionSheet setCancelButtonIndex:[_audiotrackActionSheet numberOfButtons] - 1];
  910. [_audiotrackActionSheet showInView:(UIButton *)sender];
  911. }
  912. - (IBAction)switchSubtitleTrack:(id)sender
  913. {
  914. NSArray *spuTracks = [_mediaPlayer videoSubTitlesNames];
  915. NSArray *spuTrackIndexes = [_mediaPlayer videoSubTitlesIndexes];
  916. NSUInteger count = [spuTracks count];
  917. if (count <= 1)
  918. return;
  919. _subtitleActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_SUBTITLE_TRACK", @"subtitle track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  920. for (NSUInteger i = 0; i < count; i++) {
  921. NSString *indexIndicator = ([spuTrackIndexes[i] intValue] == [_mediaPlayer currentVideoSubTitleIndex])? @"\u2713": @"";
  922. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, spuTracks[i]];
  923. [_subtitleActionSheet addButtonWithTitle:buttonTitle];
  924. }
  925. [_subtitleActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  926. [_subtitleActionSheet setCancelButtonIndex:[_subtitleActionSheet numberOfButtons] - 1];
  927. [_subtitleActionSheet showInView:(UIButton *)sender];
  928. }
  929. - (void)actionSheet:(UIActionSheet *)actionSheet clickedButtonAtIndex:(NSInteger)buttonIndex
  930. {
  931. if (buttonIndex == [actionSheet cancelButtonIndex])
  932. return;
  933. NSArray *indexArray;
  934. if (actionSheet == _subtitleActionSheet) {
  935. indexArray = _mediaPlayer.videoSubTitlesIndexes;
  936. if (buttonIndex <= indexArray.count) {
  937. _mediaPlayer.currentVideoSubTitleIndex = [indexArray[buttonIndex] intValue];
  938. }
  939. } else if (actionSheet == _audiotrackActionSheet) {
  940. indexArray = _mediaPlayer.audioTrackIndexes;
  941. if (buttonIndex <= indexArray.count) {
  942. _mediaPlayer.currentAudioTrackIndex = [indexArray[buttonIndex] intValue];
  943. }
  944. }
  945. }
  946. - (IBAction)toggleTimeDisplay:(id)sender
  947. {
  948. _displayRemainingTime = !_displayRemainingTime;
  949. [self _resetIdleTimer];
  950. }
  951. #pragma mark - multi-touch gestures
  952. - (void)tapRecognized
  953. {
  954. if (!_swipeGesturesEnabled)
  955. return;
  956. if ([_mediaPlayer isPlaying]) {
  957. [_listPlayer pause];
  958. [self.statusLabel showStatusMessage:@" ▌▌"];
  959. } else {
  960. [_listPlayer play];
  961. [self.statusLabel showStatusMessage:@" ►"];
  962. }
  963. }
  964. - (NSString*)detectPanTypeForPan:(UIPanGestureRecognizer*)panRecognizer
  965. {
  966. NSString * type;
  967. NSString * deviceType = [[UIDevice currentDevice] model];
  968. type = @"Volume"; // default in case of error
  969. CGPoint location = [panRecognizer locationInView:self.view];
  970. CGFloat position = location.x;
  971. UIInterfaceOrientation orientation = [[UIApplication sharedApplication] statusBarOrientation];
  972. CGRect screenRect = [[UIScreen mainScreen] bounds];
  973. CGFloat screenWidth = .0;
  974. if (orientation == UIDeviceOrientationPortrait)
  975. screenWidth = screenRect.size.width;
  976. else
  977. screenWidth = screenRect.size.height;
  978. if (position < screenWidth / 2)
  979. type = @"Brightness";
  980. if (position > screenWidth / 2)
  981. type = @"Volume";
  982. // only check for seeking gesture if on iPad , will overwrite last statements if true
  983. if ([deviceType isEqualToString:@"iPad"]) {
  984. if (location.y < 110)
  985. type = @"Seek";
  986. }
  987. return type;
  988. }
  989. - (void)panRecognized:(UIPanGestureRecognizer*)panRecognizer
  990. {
  991. if (!_swipeGesturesEnabled)
  992. return;
  993. CGFloat panDirectionX = [panRecognizer velocityInView:self.view].x;
  994. CGFloat panDirectionY = [panRecognizer velocityInView:self.view].y;
  995. if (panRecognizer.state == UIGestureRecognizerStateBegan) // Only Detect pantype when began to allow more freedom
  996. panType = [self detectPanTypeForPan:panRecognizer];
  997. if ([panType isEqual:@"Seek"]) {
  998. double timeRemainingDouble = (-_mediaPlayer.remainingTime.intValue*0.001);
  999. int timeRemaining = timeRemainingDouble;
  1000. if (panDirectionX > 0) {
  1001. if (timeRemaining > 2 ) // to not go outside duration , video will stop
  1002. [_mediaPlayer jumpForward:1];
  1003. } else
  1004. [_mediaPlayer jumpBackward:1];
  1005. } else if ([panType isEqual:@"Volume"]) {
  1006. MPMusicPlayerController *musicPlayer = [MPMusicPlayerController applicationMusicPlayer];
  1007. if (panDirectionY > 0)
  1008. musicPlayer.volume -= 0.01;
  1009. else
  1010. musicPlayer.volume += 0.01;
  1011. } else if ([panType isEqual:@"Brightness"]) {
  1012. CGFloat brightness = [UIScreen mainScreen].brightness;
  1013. if (panDirectionY > 0)
  1014. brightness = brightness - 0.01;
  1015. else
  1016. brightness = brightness + 0.01;
  1017. // Sanity check since -[UIScreen brightness] does not go by 0.01 steps
  1018. if (brightness > 1.0)
  1019. brightness = 1.0;
  1020. else if (brightness < 0.0)
  1021. brightness = 0.0;
  1022. NSAssert(brightness >= 0 && brightness <= 1, @"Brightness must be within 0 and 1 (it is %f)", brightness);
  1023. [[UIScreen mainScreen] setBrightness:brightness];
  1024. self.brightnessSlider.value = brightness * 2.;
  1025. NSString *brightnessHUD = [NSString stringWithFormat:@"%@: %@ %%", NSLocalizedString(@"VFILTER_BRIGHTNESS", nil), [[[NSString stringWithFormat:@"%f",(brightness*100)] componentsSeparatedByString:@"."] objectAtIndex:0]];
  1026. [self.statusLabel showStatusMessage:brightnessHUD];
  1027. }
  1028. if (panRecognizer.state == UIGestureRecognizerStateEnded) {
  1029. if ([_mediaPlayer isPlaying])
  1030. [_listPlayer play];
  1031. }
  1032. }
  1033. - (void)swipeRecognized:(UISwipeGestureRecognizer*)swipeRecognizer
  1034. {
  1035. if (!_swipeGesturesEnabled)
  1036. return;
  1037. NSString * hudString = @" ";
  1038. if (swipeRecognizer.direction == UISwipeGestureRecognizerDirectionRight) {
  1039. double timeRemainingDouble = (-_mediaPlayer.remainingTime.intValue*0.001);
  1040. int timeRemaining = timeRemainingDouble;
  1041. if (FORWARD_SWIPE_DURATION < timeRemaining) {
  1042. [_mediaPlayer jumpForward:FORWARD_SWIPE_DURATION];
  1043. hudString = [NSString stringWithFormat:@"⇒ %is", FORWARD_SWIPE_DURATION];
  1044. } else {
  1045. [_mediaPlayer jumpForward:(timeRemaining - 5)];
  1046. hudString = [NSString stringWithFormat:@"⇒ %is",(timeRemaining - 5)];
  1047. }
  1048. }
  1049. else if (swipeRecognizer.direction == UISwipeGestureRecognizerDirectionLeft) {
  1050. [_mediaPlayer jumpBackward:BACKWARD_SWIPE_DURATION];
  1051. hudString = [NSString stringWithFormat:@"⇐ %is",BACKWARD_SWIPE_DURATION];
  1052. }
  1053. if (swipeRecognizer.state == UIGestureRecognizerStateEnded) {
  1054. if ([_mediaPlayer isPlaying])
  1055. [_listPlayer play];
  1056. [self.statusLabel showStatusMessage:hudString];
  1057. }
  1058. }
  1059. #pragma mark - Video Filter UI
  1060. - (IBAction)videoFilterToggle:(id)sender
  1061. {
  1062. if (!_playbackSpeedViewHidden)
  1063. self.playbackSpeedView.hidden = _playbackSpeedViewHidden = YES;
  1064. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  1065. if (!_controlsHidden) {
  1066. self.controllerPanel.hidden = _controlsHidden = YES;
  1067. self.controllerPanelLandscape.hidden = YES;
  1068. }
  1069. }
  1070. self.videoFilterView.hidden = !_videoFiltersHidden;
  1071. _videoFiltersHidden = self.videoFilterView.hidden;
  1072. }
  1073. - (IBAction)videoFilterSliderAction:(id)sender
  1074. {
  1075. if (sender == self.hueSlider)
  1076. _mediaPlayer.hue = (int)self.hueSlider.value;
  1077. else if (sender == self.contrastSlider)
  1078. _mediaPlayer.contrast = self.contrastSlider.value;
  1079. else if (sender == self.brightnessSlider) {
  1080. if ([self hasExternalDisplay])
  1081. _mediaPlayer.brightness = self.brightnessSlider.value;
  1082. else
  1083. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  1084. } else if (sender == self.saturationSlider)
  1085. _mediaPlayer.saturation = self.saturationSlider.value;
  1086. else if (sender == self.gammaSlider)
  1087. _mediaPlayer.gamma = self.gammaSlider.value;
  1088. else if (sender == self.resetVideoFilterButton) {
  1089. _mediaPlayer.hue = self.hueSlider.value = 0.;
  1090. _mediaPlayer.contrast = self.contrastSlider.value = 1.;
  1091. _mediaPlayer.brightness = self.brightnessSlider.value = 1.;
  1092. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  1093. _mediaPlayer.saturation = self.saturationSlider.value = 1.;
  1094. _mediaPlayer.gamma = self.gammaSlider.value = 1.;
  1095. } else
  1096. APLog(@"unknown sender for videoFilterSliderAction");
  1097. [self _resetIdleTimer];
  1098. }
  1099. #pragma mark - playback view
  1100. - (IBAction)playbackSpeedSliderAction:(UISlider *)sender
  1101. {
  1102. double speed = pow(2, sender.value / 17.);
  1103. float rate = INPUT_RATE_DEFAULT / speed;
  1104. if (_currentPlaybackRate != rate)
  1105. [_mediaPlayer setRate:INPUT_RATE_DEFAULT / rate];
  1106. _currentPlaybackRate = rate;
  1107. [self _updatePlaybackSpeedIndicator];
  1108. [self _resetIdleTimer];
  1109. }
  1110. - (void)_updatePlaybackSpeedIndicator
  1111. {
  1112. float f_value = self.playbackSpeedSlider.value;
  1113. double speed = pow(2, f_value / 17.);
  1114. self.playbackSpeedIndicator.text = [NSString stringWithFormat:@"%.2fx", speed];
  1115. /* rate changed, so update the exported info */
  1116. [self performSelectorInBackground:@selector(_updateDisplayedMetadata) withObject:nil];
  1117. }
  1118. - (float)_playbackSpeed
  1119. {
  1120. float f_rate = _mediaPlayer.rate;
  1121. double value = 17 * log(f_rate) / log(2.);
  1122. float returnValue = (int) ((value > 0) ? value + .5 : value - .5);
  1123. if (returnValue < -34.)
  1124. returnValue = -34.;
  1125. else if (returnValue > 34.)
  1126. returnValue = 34.;
  1127. _currentPlaybackRate = returnValue;
  1128. return returnValue;
  1129. }
  1130. - (IBAction)videoDimensionAction:(id)sender
  1131. {
  1132. if (sender == self.playbackSpeedButton || sender == self.playbackSpeedButtonLandscape) {
  1133. if (!_videoFiltersHidden)
  1134. self.videoFilterView.hidden = _videoFiltersHidden = YES;
  1135. self.playbackSpeedView.hidden = !_playbackSpeedViewHidden;
  1136. _playbackSpeedViewHidden = self.playbackSpeedView.hidden;
  1137. [self _resetIdleTimer];
  1138. } else if (sender == self.aspectRatioButton) {
  1139. NSUInteger count = [_aspectRatios count];
  1140. if (_currentAspectRatioMask + 1 > count - 1) {
  1141. _mediaPlayer.videoAspectRatio = NULL;
  1142. _mediaPlayer.videoCropGeometry = NULL;
  1143. _currentAspectRatioMask = 0;
  1144. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", nil), NSLocalizedString(@"DEFAULT", nil)]];
  1145. } else {
  1146. _currentAspectRatioMask++;
  1147. if ([_aspectRatios[_currentAspectRatioMask] isEqualToString:@"FILL_TO_SCREEN"]) {
  1148. UIScreen *screen;
  1149. if (![self hasExternalDisplay])
  1150. screen = [UIScreen mainScreen];
  1151. else
  1152. screen = [UIScreen screens][1];
  1153. float f_ar = screen.bounds.size.width / screen.bounds.size.height;
  1154. if (f_ar == (float)(640./1136.)) // iPhone 5 aka 16:9.01
  1155. _mediaPlayer.videoCropGeometry = "16:9";
  1156. else if (f_ar == (float)(2./3.)) // all other iPhones
  1157. _mediaPlayer.videoCropGeometry = "16:10"; // libvlc doesn't support 2:3 crop
  1158. else if (f_ar == .75) // all iPads
  1159. _mediaPlayer.videoCropGeometry = "4:3";
  1160. else if (f_ar == .5625) // AirPlay
  1161. _mediaPlayer.videoCropGeometry = "16:9";
  1162. else
  1163. APLog(@"unknown screen format %f, can't crop", f_ar);
  1164. [self.statusLabel showStatusMessage:NSLocalizedString(@"FILL_TO_SCREEN", nil)];
  1165. return;
  1166. }
  1167. _mediaPlayer.videoCropGeometry = NULL;
  1168. _mediaPlayer.videoAspectRatio = (char *)[_aspectRatios[_currentAspectRatioMask] UTF8String];
  1169. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", nil), _aspectRatios[_currentAspectRatioMask]]];
  1170. }
  1171. }
  1172. }
  1173. #pragma mark - background interaction
  1174. - (void)applicationWillResignActive:(NSNotification *)aNotification
  1175. {
  1176. [self _saveCurrentState];
  1177. _mediaPlayer.currentVideoTrackIndex = 0;
  1178. if (![[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingContinueAudioInBackgroundKey] boolValue]) {
  1179. if ([_mediaPlayer isPlaying]) {
  1180. [_mediaPlayer pause];
  1181. _shouldResumePlaying = YES;
  1182. }
  1183. }
  1184. }
  1185. - (void)applicationDidEnterBackground:(NSNotification *)notification
  1186. {
  1187. _shouldResumePlaying = NO;
  1188. }
  1189. - (void)applicationDidBecomeActive:(NSNotification *)notification
  1190. {
  1191. _mediaPlayer.currentVideoTrackIndex = 1;
  1192. if (_shouldResumePlaying) {
  1193. _shouldResumePlaying = NO;
  1194. [_listPlayer play];
  1195. }
  1196. }
  1197. - (void)audioSessionRouteChange:(NSNotification *)notification
  1198. {
  1199. NSArray *outputs = [[AVAudioSession sharedInstance] currentRoute].outputs;
  1200. NSString *portName = [[outputs objectAtIndex:0] portName];
  1201. if (![portName isEqualToString:@"Headphones"] && [_mediaPlayer isPlaying])
  1202. [_listPlayer pause];
  1203. }
  1204. - (void)mediaDidFinishParsing:(VLCMedia *)aMedia
  1205. {
  1206. [self _updateDisplayedMetadata];
  1207. }
  1208. - (void)mediaMetaDataDidChange:(VLCMedia*)aMedia
  1209. {
  1210. [self _updateDisplayedMetadata];
  1211. }
  1212. - (void)_updateDisplayedMetadata
  1213. {
  1214. MLFile *item;
  1215. NSString *title;
  1216. NSString *artist;
  1217. NSString *albumName;
  1218. NSString *trackNumber;
  1219. BOOL mediaIsAudioOnly = YES;
  1220. if (self.fileFromMediaLibrary)
  1221. item = self.fileFromMediaLibrary;
  1222. else if (self.mediaList) {
  1223. NSArray *matches = [MLFile fileForURL:[_mediaPlayer.media.url absoluteString]];
  1224. if (matches.count > 0)
  1225. item = matches[0];
  1226. }
  1227. if (item) {
  1228. if (item.isAlbumTrack) {
  1229. title = item.albumTrack.title;
  1230. artist = item.albumTrack.artist;
  1231. albumName = item.albumTrack.album.name;
  1232. } else
  1233. title = item.title;
  1234. /* MLKit knows better than us if this thing is audio only or not */
  1235. mediaIsAudioOnly = [item isSupportedAudioFile];
  1236. if (mediaIsAudioOnly)
  1237. self.artworkImageView.image = [VLCThumbnailsCache thumbnailForMediaFile:item];
  1238. } else {
  1239. NSDictionary * metaDict = _mediaPlayer.media.metaDictionary;
  1240. /* this is a non file media, so we need to actually check if there is there is
  1241. * a video track included or not */
  1242. NSArray *tracks = _mediaPlayer.media.tracksInformation;
  1243. NSUInteger trackCount = tracks.count;
  1244. for (NSUInteger x = 0 ; x < trackCount; x++) {
  1245. if ([[tracks[x] objectForKey:VLCMediaTracksInformationType] isEqualToString:VLCMediaTracksInformationTypeVideo]) {
  1246. mediaIsAudioOnly = NO;
  1247. break;
  1248. }
  1249. }
  1250. if (metaDict) {
  1251. title = metaDict[VLCMetaInformationNowPlaying] ? metaDict[VLCMetaInformationNowPlaying] : metaDict[VLCMetaInformationTitle];
  1252. artist = metaDict[VLCMetaInformationArtist];
  1253. albumName = metaDict[VLCMetaInformationAlbum];
  1254. trackNumber = metaDict[VLCMetaInformationTrackNumber];
  1255. if (mediaIsAudioOnly)
  1256. self.artworkImageView.image = [VLCThumbnailsCache thumbnailForMediaItemWithTitle:title Artist:artist andAlbumName:albumName];
  1257. }
  1258. }
  1259. if (mediaIsAudioOnly) {
  1260. if (!self.artworkImageView.image) {
  1261. self.trackNameLabel.text = title;
  1262. self.artistNameLabel.text = artist;
  1263. self.albumNameLabel.text = albumName;
  1264. } else {
  1265. NSString *trackName = title;
  1266. if (artist)
  1267. trackName = [trackName stringByAppendingFormat:@" — %@", artist];
  1268. if (albumName)
  1269. trackName = [trackName stringByAppendingFormat:@" — %@", albumName];
  1270. self.trackNameLabel.text = trackName;
  1271. }
  1272. if (self.trackNameLabel.text.length < 1)
  1273. self.trackNameLabel.text = [[_mediaPlayer.media url] lastPathComponent];
  1274. if (!self.aspectRatioButton.hidden) {
  1275. CGRect rect = self.timeDisplay.frame;
  1276. rect.origin.x += self.aspectRatioButton.frame.size.width;
  1277. self.timeDisplay.frame = rect;
  1278. rect = self.positionSlider.frame;
  1279. rect.size.width += self.aspectRatioButton.frame.size.width;
  1280. self.positionSlider.frame = rect;
  1281. self.aspectRatioButton.hidden = YES;
  1282. }
  1283. } else {
  1284. if (self.aspectRatioButton.hidden) {
  1285. CGRect rect = self.timeDisplay.frame;
  1286. rect.origin.x -= self.aspectRatioButton.frame.size.width;
  1287. self.timeDisplay.frame = rect;
  1288. rect = self.positionSlider.frame;
  1289. rect.size.width -= self.aspectRatioButton.frame.size.width;
  1290. self.positionSlider.frame = rect;
  1291. self.aspectRatioButton.hidden = NO;
  1292. }
  1293. }
  1294. self.videoFilterButton.hidden = mediaIsAudioOnly;
  1295. /* don't leak sensitive information to the OS, if passcode lock is enabled */
  1296. BOOL passcodeLockEnabled = [[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingPasscodeOnKey] boolValue];
  1297. NSMutableDictionary *currentlyPlayingTrackInfo;
  1298. if (passcodeLockEnabled)
  1299. currentlyPlayingTrackInfo = [NSMutableDictionary dictionaryWithObjectsAndKeys:@(_mediaPlayer.media.length.intValue / 1000.), MPMediaItemPropertyPlaybackDuration, @(_mediaPlayer.time.intValue / 1000.), MPNowPlayingInfoPropertyElapsedPlaybackTime, @(_mediaPlayer.rate), MPNowPlayingInfoPropertyPlaybackRate, nil];
  1300. else {
  1301. currentlyPlayingTrackInfo = [NSMutableDictionary dictionaryWithObjectsAndKeys: title, MPMediaItemPropertyTitle, @(_mediaPlayer.media.length.intValue / 1000.), MPMediaItemPropertyPlaybackDuration, @(_mediaPlayer.time.intValue / 1000.), MPNowPlayingInfoPropertyElapsedPlaybackTime, @(_mediaPlayer.rate), MPNowPlayingInfoPropertyPlaybackRate, nil];
  1302. if (artist.length > 0)
  1303. [currentlyPlayingTrackInfo setObject:artist forKey:MPMediaItemPropertyArtist];
  1304. if (albumName.length > 0)
  1305. [currentlyPlayingTrackInfo setObject:albumName forKey:MPMediaItemPropertyAlbumTitle];
  1306. [currentlyPlayingTrackInfo setObject:[NSNumber numberWithInt:[trackNumber intValue]] forKey:MPMediaItemPropertyAlbumTrackNumber];
  1307. if (self.artworkImageView.image) {
  1308. MPMediaItemArtwork *mpartwork = [[MPMediaItemArtwork alloc] initWithImage:self.artworkImageView.image];
  1309. [currentlyPlayingTrackInfo setObject:mpartwork forKey:MPMediaItemPropertyArtwork];
  1310. }
  1311. }
  1312. [MPNowPlayingInfoCenter defaultCenter].nowPlayingInfo = currentlyPlayingTrackInfo;
  1313. }
  1314. #pragma mark - autorotation
  1315. - (BOOL)shouldAutorotate
  1316. {
  1317. UIInterfaceOrientation toInterfaceOrientation = [[UIApplication sharedApplication] statusBarOrientation];
  1318. return UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad
  1319. || toInterfaceOrientation != UIInterfaceOrientationPortraitUpsideDown;
  1320. }
  1321. - (void)willRotateToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation duration:(NSTimeInterval)duration
  1322. {
  1323. [super willRotateToInterfaceOrientation:toInterfaceOrientation duration:duration];
  1324. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  1325. if (self.artworkImageView.image)
  1326. self.trackNameLabel.hidden = UIInterfaceOrientationIsLandscape(toInterfaceOrientation);
  1327. }
  1328. }
  1329. #pragma mark - AVSession delegate
  1330. - (void)beginInterruption
  1331. {
  1332. if ([_mediaPlayer isPlaying]) {
  1333. [_mediaPlayer pause];
  1334. _shouldResumePlaying = YES;
  1335. }
  1336. }
  1337. - (void)endInterruption
  1338. {
  1339. if (_shouldResumePlaying) {
  1340. [_mediaPlayer play];
  1341. _shouldResumePlaying = NO;
  1342. }
  1343. }
  1344. #pragma mark - External Display
  1345. - (BOOL)hasExternalDisplay
  1346. {
  1347. return ([[UIScreen screens] count] > 1);
  1348. }
  1349. - (void)showOnExternalDisplay
  1350. {
  1351. UIScreen *screen = [UIScreen screens][1];
  1352. screen.overscanCompensation = UIScreenOverscanCompensationInsetApplicationFrame;
  1353. self.externalWindow = [[UIWindow alloc] initWithFrame:screen.bounds];
  1354. UIViewController *controller = [[VLCExternalDisplayController alloc] init];
  1355. self.externalWindow.rootViewController = controller;
  1356. [controller.view addSubview:_movieView];
  1357. controller.view.frame = screen.bounds;
  1358. _movieView.frame = screen.bounds;
  1359. self.playingExternallyView.hidden = NO;
  1360. self.externalWindow.screen = screen;
  1361. self.externalWindow.hidden = NO;
  1362. }
  1363. - (void)hideFromExternalDisplay
  1364. {
  1365. [self.view addSubview:_movieView];
  1366. [self.view sendSubviewToBack:_movieView];
  1367. _movieView.frame = self.view.frame;
  1368. self.playingExternallyView.hidden = YES;
  1369. self.externalWindow.hidden = YES;
  1370. self.externalWindow = nil;
  1371. }
  1372. - (void)handleExternalScreenDidConnect:(NSNotification *)notification
  1373. {
  1374. [self showOnExternalDisplay];
  1375. }
  1376. - (void)handleExternalScreenDidDisconnect:(NSNotification *)notification
  1377. {
  1378. [self hideFromExternalDisplay];
  1379. }
  1380. @end