VLCMovieViewController.m 65 KB

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