VLCMovieViewController.m 63 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629
  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. }
  451. return YES;
  452. }
  453. - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
  454. {
  455. if (buttonIndex == 1)
  456. [self _playNewMedia];
  457. else {
  458. [self _stopPlayback];
  459. [self closePlayback:nil];
  460. }
  461. }
  462. - (void)_playNewMedia
  463. {
  464. NSNumber *playbackPositionInTime = @(0);
  465. CGFloat lastPosition = .0;
  466. NSInteger duration = 0;
  467. MLFile *matchedFile;
  468. if (self.fileFromMediaLibrary)
  469. matchedFile = self.fileFromMediaLibrary;
  470. else if (self.mediaList) {
  471. NSArray *matches = [MLFile fileForURL:[[[self.mediaList mediaAtIndex:self.itemInMediaListToBePlayedFirst] url] absoluteString]];
  472. if (matches.count > 0) {
  473. matchedFile = matches[0];
  474. lastPosition = matchedFile.lastPosition.floatValue;
  475. }
  476. }
  477. if (matchedFile.lastPosition)
  478. lastPosition = matchedFile.lastPosition.floatValue;
  479. duration = matchedFile.duration.intValue;
  480. if (lastPosition < .95) {
  481. if (duration != 0)
  482. playbackPositionInTime = @(lastPosition * (duration / 1000.));
  483. }
  484. if (playbackPositionInTime.intValue > 0 && (duration * lastPosition - duration) < -60000) {
  485. [_mediaPlayer.media addOptions:@{@"start-time": playbackPositionInTime}];
  486. APLog(@"set starttime to %i", playbackPositionInTime.intValue);
  487. }
  488. [_mediaPlayer addObserver:self forKeyPath:@"time" options:0 context:nil];
  489. [_mediaPlayer addObserver:self forKeyPath:@"remainingTime" options:0 context:nil];
  490. if (self.mediaList)
  491. [_listPlayer playItemAtIndex:self.itemInMediaListToBePlayedFirst];
  492. else
  493. [_listPlayer playMedia:_listPlayer.rootMedia];
  494. if (matchedFile) {
  495. if (matchedFile.lastAudioTrack.intValue > 0)
  496. _mediaPlayer.currentAudioTrackIndex = matchedFile.lastAudioTrack.intValue;
  497. if (matchedFile.lastSubtitleTrack.intValue > 0)
  498. _mediaPlayer.currentVideoSubTitleIndex = matchedFile.lastSubtitleTrack.intValue;
  499. }
  500. self.playbackSpeedSlider.value = [self _playbackSpeed];
  501. [self _updatePlaybackSpeedIndicator];
  502. _currentAspectRatioMask = 0;
  503. _mediaPlayer.videoAspectRatio = NULL;
  504. /* some demuxers don't respect :start-time, so re-try here */
  505. if (lastPosition < .95 && _mediaPlayer.position < lastPosition && (duration * lastPosition - duration) < -60000)
  506. _mediaPlayer.position = lastPosition;
  507. [self _resetIdleTimer];
  508. _playerIsSetup = YES;
  509. }
  510. - (void)viewWillDisappear:(BOOL)animated
  511. {
  512. [self _stopPlayback];
  513. _viewAppeared = NO;
  514. if (_idleTimer) {
  515. [_idleTimer invalidate];
  516. _idleTimer = nil;
  517. }
  518. [self.navigationController setNavigationBarHidden:NO animated:YES];
  519. if (!SYSTEM_RUNS_IOS7_OR_LATER)
  520. [UIApplication sharedApplication].statusBarStyle = UIStatusBarStyleBlackOpaque;
  521. [[UIApplication sharedApplication] setStatusBarHidden:NO withAnimation:UIStatusBarAnimationFade];
  522. [super viewWillDisappear:animated];
  523. // hide filter UI for next run
  524. if (!_videoFiltersHidden)
  525. _videoFiltersHidden = YES;
  526. if (!_playbackSpeedViewHidden)
  527. _playbackSpeedViewHidden = YES;
  528. }
  529. - (void)_stopPlayback
  530. {
  531. if (_mediaPlayer) {
  532. @try {
  533. [_mediaPlayer removeObserver:self forKeyPath:@"time"];
  534. [_mediaPlayer removeObserver:self forKeyPath:@"remainingTime"];
  535. }
  536. @catch (NSException *exception) {
  537. APLog(@"we weren't an observer yet");
  538. }
  539. if (_mediaPlayer.media) {
  540. [_mediaPlayer pause];
  541. [self _saveCurrentState];
  542. [_mediaPlayer stop];
  543. }
  544. if (_mediaPlayer)
  545. _mediaPlayer = nil;
  546. if (_listPlayer)
  547. _listPlayer = nil;
  548. }
  549. if (_fileFromMediaLibrary)
  550. _fileFromMediaLibrary = nil;
  551. if (_mediaList)
  552. _mediaList = nil;
  553. if (_url)
  554. _url = nil;
  555. if (_pathToExternalSubtitlesFile) {
  556. NSFileManager *fileManager = [NSFileManager defaultManager];
  557. if ([fileManager fileExistsAtPath:_pathToExternalSubtitlesFile])
  558. [fileManager removeItemAtPath:_pathToExternalSubtitlesFile error:nil];
  559. _pathToExternalSubtitlesFile = nil;
  560. }
  561. _playerIsSetup = NO;
  562. }
  563. - (void)_saveCurrentState
  564. {
  565. if (self.fileFromMediaLibrary) {
  566. @try {
  567. MLFile *item = self.fileFromMediaLibrary;
  568. item.lastPosition = @([_mediaPlayer position]);
  569. item.lastAudioTrack = @(_mediaPlayer.currentAudioTrackIndex);
  570. item.lastSubtitleTrack = @(_mediaPlayer.currentVideoSubTitleIndex);
  571. }
  572. @catch (NSException *exception) {
  573. APLog(@"failed to save current media state - file removed?");
  574. }
  575. } else {
  576. NSArray *files = [MLFile fileForURL:[[_mediaPlayer.media url] absoluteString]];
  577. if (files.count > 0) {
  578. MLFile *fileFromList = files[0];
  579. fileFromList.lastPosition = @([_mediaPlayer position]);
  580. fileFromList.lastAudioTrack = @(_mediaPlayer.currentAudioTrackIndex);
  581. fileFromList.lastSubtitleTrack = @(_mediaPlayer.currentVideoSubTitleIndex);
  582. }
  583. }
  584. }
  585. - (NSString *)_resolveFontName
  586. {
  587. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  588. BOOL bold = [[defaults objectForKey:kVLCSettingSubtitlesBoldFont] boolValue];
  589. NSString *font = [defaults objectForKey:kVLCSettingSubtitlesFont];
  590. NSDictionary *fontMap = @{
  591. @"AmericanTypewriter": @"AmericanTypewriter-Bold",
  592. @"ArialMT": @"Arial-BoldMT",
  593. @"ArialHebrew": @"ArialHebrew-Bold",
  594. @"ChalkboardSE-Regular": @"ChalkboardSE-Bold",
  595. @"CourierNewPSMT": @"CourierNewPS-BoldMT",
  596. @"Georgia": @"Georgia-Bold",
  597. @"GillSans": @"GillSans-Bold",
  598. @"GujaratiSangamMN": @"GujaratiSangamMN-Bold",
  599. @"STHeitiSC-Light": @"STHeitiSC-Medium",
  600. @"STHeitiTC-Light": @"STHeitiTC-Medium",
  601. @"HelveticaNeue": @"HelveticaNeue-Bold",
  602. @"HiraKakuProN-W3": @"HiraKakuProN-W6",
  603. @"HiraMinProN-W3": @"HiraMinProN-W6",
  604. @"HoeflerText-Regular": @"HoeflerText-Black",
  605. @"Kailasa": @"Kailasa-Bold",
  606. @"KannadaSangamMN": @"KannadaSangamMN-Bold",
  607. @"MalayalamSangamMN": @"MalayalamSangamMN-Bold",
  608. @"OriyaSangamMN": @"OriyaSangamMN-Bold",
  609. @"SinhalaSangamMN": @"SinhalaSangamMN-Bold",
  610. @"SnellRoundhand": @"SnellRoundhand-Bold",
  611. @"TamilSangamMN": @"TamilSangamMN-Bold",
  612. @"TeluguSangamMN": @"TeluguSangamMN-Bold",
  613. @"TimesNewRomanPSMT": @"TimesNewRomanPS-BoldMT",
  614. @"Zapfino": @"Zapfino"
  615. };
  616. if (!bold) {
  617. return font;
  618. } else {
  619. return fontMap[font];
  620. }
  621. }
  622. #pragma mark - remote events
  623. - (void)viewDidAppear:(BOOL)animated
  624. {
  625. [super viewDidAppear:animated];
  626. [[UIApplication sharedApplication] beginReceivingRemoteControlEvents];
  627. [self becomeFirstResponder];
  628. }
  629. - (void)viewDidDisappear:(BOOL)animated
  630. {
  631. [super viewDidDisappear:animated];
  632. [[UIApplication sharedApplication] endReceivingRemoteControlEvents];
  633. [self resignFirstResponder];
  634. [[NSUserDefaults standardUserDefaults] setBool:_displayRemainingTime forKey:kVLCShowRemainingTime];
  635. }
  636. - (BOOL)canBecomeFirstResponder
  637. {
  638. return YES;
  639. }
  640. - (void)remoteControlReceivedWithEvent:(UIEvent *)event
  641. {
  642. switch (event.subtype) {
  643. case UIEventSubtypeRemoteControlPlay:
  644. [_listPlayer play];
  645. break;
  646. case UIEventSubtypeRemoteControlPause:
  647. [_listPlayer pause];
  648. break;
  649. case UIEventSubtypeRemoteControlTogglePlayPause:
  650. [self playPause];
  651. break;
  652. case UIEventSubtypeRemoteControlNextTrack:
  653. [self forward:nil];
  654. break;
  655. case UIEventSubtypeRemoteControlPreviousTrack:
  656. [self backward:nil];
  657. break;
  658. case UIEventSubtypeRemoteControlStop:
  659. [self closePlayback:nil];
  660. break;
  661. default:
  662. break;
  663. }
  664. }
  665. #pragma mark - controls visibility
  666. - (void)handlePinchGesture:(UIPinchGestureRecognizer *)recognizer
  667. {
  668. if (!_swipeGesturesEnabled)
  669. return;
  670. if (recognizer.velocity < 0.)
  671. [self closePlayback:nil];
  672. }
  673. - (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch
  674. {
  675. if (touch.view != self.view)
  676. return NO;
  677. return YES;
  678. }
  679. - (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer
  680. {
  681. return YES;
  682. }
  683. - (void)setControlsHidden:(BOOL)hidden animated:(BOOL)animated
  684. {
  685. _controlsHidden = hidden;
  686. CGFloat alpha = _controlsHidden? 0.0f: 1.0f;
  687. if (!_controlsHidden) {
  688. _controllerPanel.alpha = 0.0f;
  689. _controllerPanel.hidden = !_videoFiltersHidden;
  690. _controllerPanelLandscape.alpha = 0.0f;
  691. _controllerPanelLandscape.hidden = !_videoFiltersHidden;
  692. _toolbar.alpha = 0.0f;
  693. _toolbar.hidden = NO;
  694. _videoFilterView.alpha = 0.0f;
  695. _videoFilterView.hidden = _videoFiltersHidden;
  696. _playbackSpeedView.alpha = 0.0f;
  697. _playbackSpeedView.hidden = _playbackSpeedViewHidden;
  698. }
  699. void (^animationBlock)() = ^() {
  700. _controllerPanel.alpha = alpha;
  701. _controllerPanelLandscape.alpha = alpha;
  702. _toolbar.alpha = alpha;
  703. _videoFilterView.alpha = alpha;
  704. _playbackSpeedView.alpha = alpha;
  705. };
  706. void (^completionBlock)(BOOL finished) = ^(BOOL finished) {
  707. _controllerPanel.hidden = _videoFiltersHidden ? _controlsHidden : NO;
  708. _controllerPanelLandscape.hidden = _videoFiltersHidden ? _controlsHidden : NO;
  709. _toolbar.hidden = _controlsHidden;
  710. _videoFilterView.hidden = _videoFiltersHidden;
  711. _playbackSpeedView.hidden = _playbackSpeedViewHidden;
  712. };
  713. UIStatusBarAnimation animationType = animated? UIStatusBarAnimationFade: UIStatusBarAnimationNone;
  714. NSTimeInterval animationDuration = animated? 0.3: 0.0;
  715. [[UIApplication sharedApplication] setStatusBarHidden:_viewAppeared ? _controlsHidden : NO withAnimation:animationType];
  716. [UIView animateWithDuration:animationDuration animations:animationBlock completion:completionBlock];
  717. _volumeView.hidden = _volumeViewLandscape.hidden = _controllerPanel.hidden;
  718. }
  719. - (void)toggleControlsVisible
  720. {
  721. if (_controlsHidden && !_videoFiltersHidden)
  722. _videoFiltersHidden = YES;
  723. [self setControlsHidden:!_controlsHidden animated:YES];
  724. }
  725. - (void)_resetIdleTimer
  726. {
  727. if (!_idleTimer)
  728. _idleTimer = [NSTimer scheduledTimerWithTimeInterval:4.
  729. target:self
  730. selector:@selector(idleTimerExceeded)
  731. userInfo:nil
  732. repeats:NO];
  733. else {
  734. if (fabs([_idleTimer.fireDate timeIntervalSinceNow]) < 4.)
  735. [_idleTimer setFireDate:[NSDate dateWithTimeIntervalSinceNow:4.]];
  736. }
  737. }
  738. - (void)idleTimerExceeded
  739. {
  740. _idleTimer = nil;
  741. if (!_controlsHidden)
  742. [self toggleControlsVisible];
  743. if (!_videoFiltersHidden)
  744. _videoFiltersHidden = YES;
  745. if (!_playbackSpeedViewHidden)
  746. _playbackSpeedViewHidden = YES;
  747. if (self.scrubIndicatorView.hidden == NO)
  748. self.scrubIndicatorView.hidden = YES;
  749. }
  750. - (UIResponder *)nextResponder
  751. {
  752. [self _resetIdleTimer];
  753. return [super nextResponder];
  754. }
  755. #pragma mark - controls
  756. - (IBAction)closePlayback:(id)sender
  757. {
  758. [self setControlsHidden:NO animated:NO];
  759. [self.navigationController dismissViewControllerAnimated:YES completion:nil];
  760. }
  761. - (IBAction)positionSliderAction:(UISlider *)sender
  762. {
  763. /* we need to limit the number of events sent by the slider, since otherwise, the user
  764. * wouldn't see the I-frames when seeking on current mobile devices. This isn't a problem
  765. * within the Simulator, but especially on older ARMv7 devices, it's clearly noticeable. */
  766. [self performSelector:@selector(_setPositionForReal) withObject:nil afterDelay:0.3];
  767. VLCTime *newPosition = [VLCTime timeWithInt:(int)(_positionSlider.value * self.fileFromMediaLibrary.duration.intValue)];
  768. [self.timeDisplay setTitle:newPosition.stringValue forState:UIControlStateNormal];
  769. self.timeDisplay.accessibilityLabel = [NSString stringWithFormat:@"%@: %@", NSLocalizedString(@"PLAYBACK_POSITION", nil), newPosition.stringValue];
  770. _positionSet = NO;
  771. [self _resetIdleTimer];
  772. }
  773. - (void)_setPositionForReal
  774. {
  775. if (!_positionSet) {
  776. _mediaPlayer.position = _positionSlider.value;
  777. _positionSet = YES;
  778. }
  779. }
  780. - (IBAction)positionSliderTouchDown:(id)sender
  781. {
  782. [self _updateScrubLabel];
  783. self.scrubIndicatorView.hidden = NO;
  784. _isScrubbing = YES;
  785. }
  786. - (IBAction)positionSliderTouchUp:(id)sender
  787. {
  788. self.scrubIndicatorView.hidden = YES;
  789. _isScrubbing = NO;
  790. }
  791. - (void)_updateScrubLabel
  792. {
  793. float speed = self.positionSlider.scrubbingSpeed;
  794. if (speed == 1.)
  795. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HIGH", nil);
  796. else if (speed == .5)
  797. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HALF", nil);
  798. else if (speed == .25)
  799. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_QUARTER", nil);
  800. else
  801. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_FINE", nil);
  802. [self _resetIdleTimer];
  803. }
  804. - (IBAction)positionSliderDrag:(id)sender
  805. {
  806. [self _updateScrubLabel];
  807. }
  808. - (IBAction)volumeSliderAction:(id)sender
  809. {
  810. [self _resetIdleTimer];
  811. }
  812. - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
  813. {
  814. if (!_isScrubbing) {
  815. self.positionSlider.value = [_mediaPlayer position];
  816. }
  817. if (_displayRemainingTime)
  818. [self.timeDisplay setTitle:[[_mediaPlayer remainingTime] stringValue] forState:UIControlStateNormal];
  819. else
  820. [self.timeDisplay setTitle:[[_mediaPlayer time] stringValue] forState:UIControlStateNormal];
  821. }
  822. - (void)mediaPlayerStateChanged:(NSNotification *)aNotification
  823. {
  824. VLCMediaPlayerState currentState = _mediaPlayer.state;
  825. if (currentState == VLCMediaPlayerStateBuffering) {
  826. /* attach delegate */
  827. _mediaPlayer.media.delegate = self;
  828. /* let's update meta data */
  829. [self _updateDisplayedMetadata];
  830. /* on-the-fly values through private API */
  831. [_mediaPlayer performSelector:@selector(setTextRendererFont:) withObject:[self _resolveFontName]];
  832. [_mediaPlayer performSelector:@selector(setTextRendererFontSize:) withObject:[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingSubtitlesFontSize]];
  833. [_mediaPlayer performSelector:@selector(setTextRendererFontColor:) withObject:[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingSubtitlesFontColor]];
  834. }
  835. if (currentState == VLCMediaPlayerStateError) {
  836. [self.statusLabel showStatusMessage:NSLocalizedString(@"PLAYBACK_FAILED", nil)];
  837. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  838. }
  839. if ((currentState == VLCMediaPlayerStateEnded || currentState == VLCMediaPlayerStateStopped) && _listPlayer.repeatMode == VLCDoNotRepeat) {
  840. if ([_listPlayer.mediaList indexOfMedia:_mediaPlayer.media] == _listPlayer.mediaList.count - 1)
  841. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  842. }
  843. UIImage *playPauseImage = [_mediaPlayer isPlaying]? [UIImage imageNamed:@"pauseIcon"] : [UIImage imageNamed:@"playIcon"];
  844. [_playPauseButton setImage:playPauseImage forState:UIControlStateNormal];
  845. [_playPauseButtonLandscape setImage:playPauseImage forState:UIControlStateNormal];
  846. if ([[_mediaPlayer audioTrackIndexes] count] > 2) {
  847. self.audioSwitcherButton.hidden = NO;
  848. self.audioSwitcherButtonLandscape.hidden = NO;
  849. } else {
  850. self.audioSwitcherButton.hidden = YES;
  851. self.audioSwitcherButtonLandscape.hidden = YES;
  852. }
  853. if ([[_mediaPlayer videoSubTitlesIndexes] count] > 1) {
  854. self.subtitleContainer.hidden = NO;
  855. self.subtitleContainerLandscape.hidden = NO;
  856. } else {
  857. self.subtitleContainer.hidden = YES;
  858. self.subtitleContainerLandscape.hidden = YES;
  859. }
  860. }
  861. - (IBAction)playPause
  862. {
  863. if ([_mediaPlayer isPlaying])
  864. [_listPlayer pause];
  865. else
  866. [_listPlayer play];
  867. }
  868. - (IBAction)forward:(id)sender
  869. {
  870. if (self.mediaList)
  871. [_listPlayer next];
  872. else
  873. [_mediaPlayer mediumJumpForward];
  874. }
  875. - (IBAction)backward:(id)sender
  876. {
  877. if (self.mediaList)
  878. [_listPlayer previous];
  879. else
  880. [_mediaPlayer mediumJumpBackward];
  881. }
  882. - (void)toggleRepeatMode:(id)sender
  883. {
  884. if (_listPlayer.repeatMode == VLCDoNotRepeat) {
  885. _listPlayer.repeatMode = VLCRepeatCurrentItem;
  886. [self.repeatButton setImage:[UIImage imageNamed:@"repeatOne"] forState:UIControlStateNormal];
  887. [self.repeatButtonLandscape setImage:[UIImage imageNamed:@"repeatOne"] forState:UIControlStateNormal];
  888. } else {
  889. _listPlayer.repeatMode = VLCDoNotRepeat;
  890. [self.repeatButton setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  891. [self.repeatButtonLandscape setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  892. }
  893. }
  894. - (IBAction)switchAudioTrack:(id)sender
  895. {
  896. _audiotrackActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_AUDIO_TRACK", @"audio track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  897. NSArray *audioTracks = [_mediaPlayer audioTrackNames];
  898. NSArray *audioTrackIndexes = [_mediaPlayer audioTrackIndexes];
  899. NSUInteger count = [audioTracks count];
  900. for (NSUInteger i = 0; i < count; i++) {
  901. NSString *indexIndicator = ([audioTrackIndexes[i] intValue] == [_mediaPlayer currentAudioTrackIndex])? @"\u2713": @"";
  902. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, audioTracks[i]];
  903. [_audiotrackActionSheet addButtonWithTitle:buttonTitle];
  904. }
  905. [_audiotrackActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  906. [_audiotrackActionSheet setCancelButtonIndex:[_audiotrackActionSheet numberOfButtons] - 1];
  907. [_audiotrackActionSheet showInView:(UIButton *)sender];
  908. }
  909. - (IBAction)switchSubtitleTrack:(id)sender
  910. {
  911. NSArray *spuTracks = [_mediaPlayer videoSubTitlesNames];
  912. NSArray *spuTrackIndexes = [_mediaPlayer videoSubTitlesIndexes];
  913. NSUInteger count = [spuTracks count];
  914. if (count <= 1)
  915. return;
  916. _subtitleActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_SUBTITLE_TRACK", @"subtitle track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  917. for (NSUInteger i = 0; i < count; i++) {
  918. NSString *indexIndicator = ([spuTrackIndexes[i] intValue] == [_mediaPlayer currentVideoSubTitleIndex])? @"\u2713": @"";
  919. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, spuTracks[i]];
  920. [_subtitleActionSheet addButtonWithTitle:buttonTitle];
  921. }
  922. [_subtitleActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  923. [_subtitleActionSheet setCancelButtonIndex:[_subtitleActionSheet numberOfButtons] - 1];
  924. [_subtitleActionSheet showInView:(UIButton *)sender];
  925. }
  926. - (void)actionSheet:(UIActionSheet *)actionSheet clickedButtonAtIndex:(NSInteger)buttonIndex
  927. {
  928. if (buttonIndex == [actionSheet cancelButtonIndex])
  929. return;
  930. NSArray *indexArray;
  931. if (actionSheet == _subtitleActionSheet) {
  932. indexArray = _mediaPlayer.videoSubTitlesIndexes;
  933. if (buttonIndex <= indexArray.count) {
  934. _mediaPlayer.currentVideoSubTitleIndex = [indexArray[buttonIndex] intValue];
  935. }
  936. } else if (actionSheet == _audiotrackActionSheet) {
  937. indexArray = _mediaPlayer.audioTrackIndexes;
  938. if (buttonIndex <= indexArray.count) {
  939. _mediaPlayer.currentAudioTrackIndex = [indexArray[buttonIndex] intValue];
  940. }
  941. }
  942. }
  943. - (IBAction)toggleTimeDisplay:(id)sender
  944. {
  945. _displayRemainingTime = !_displayRemainingTime;
  946. [self _resetIdleTimer];
  947. }
  948. #pragma mark - multi-touch gestures
  949. - (void)tapRecognized
  950. {
  951. if (!_swipeGesturesEnabled)
  952. return;
  953. if ([_mediaPlayer isPlaying]) {
  954. [_listPlayer pause];
  955. [self.statusLabel showStatusMessage:@" ▌▌"];
  956. } else {
  957. [_listPlayer play];
  958. [self.statusLabel showStatusMessage:@" ►"];
  959. }
  960. }
  961. - (NSString*)detectPanTypeForPan:(UIPanGestureRecognizer*)panRecognizer
  962. {
  963. NSString * type;
  964. NSString * deviceType = [[UIDevice currentDevice] model];
  965. type = @"Volume"; // default in case of error
  966. CGPoint location = [panRecognizer locationInView:self.view];
  967. CGFloat position = location.x;
  968. UIInterfaceOrientation orientation = [[UIApplication sharedApplication] statusBarOrientation];
  969. CGRect screenRect = [[UIScreen mainScreen] bounds];
  970. CGFloat screenWidth = .0;
  971. if (orientation == UIDeviceOrientationPortrait)
  972. screenWidth = screenRect.size.width;
  973. else
  974. screenWidth = screenRect.size.height;
  975. if (position < screenWidth / 2)
  976. type = @"Brightness";
  977. if (position > screenWidth / 2)
  978. type = @"Volume";
  979. // only check for seeking gesture if on iPad , will overwrite last statements if true
  980. if ([deviceType isEqualToString:@"iPad"]) {
  981. if (location.y < 110)
  982. type = @"Seek";
  983. }
  984. return type;
  985. }
  986. - (void)panRecognized:(UIPanGestureRecognizer*)panRecognizer
  987. {
  988. if (!_swipeGesturesEnabled)
  989. return;
  990. CGFloat panDirectionX = [panRecognizer velocityInView:self.view].x;
  991. CGFloat panDirectionY = [panRecognizer velocityInView:self.view].y;
  992. if (panRecognizer.state == UIGestureRecognizerStateBegan) // Only Detect pantype when began to allow more freedom
  993. panType = [self detectPanTypeForPan:panRecognizer];
  994. if ([panType isEqual:@"Seek"]) {
  995. double timeRemainingDouble = (-_mediaPlayer.remainingTime.intValue*0.001);
  996. int timeRemaining = timeRemainingDouble;
  997. if (panDirectionX > 0) {
  998. if (timeRemaining > 2 ) // to not go outside duration , video will stop
  999. [_mediaPlayer jumpForward:1];
  1000. } else
  1001. [_mediaPlayer jumpBackward:1];
  1002. } else if ([panType isEqual:@"Volume"]) {
  1003. MPMusicPlayerController *musicPlayer = [MPMusicPlayerController applicationMusicPlayer];
  1004. if (panDirectionY > 0)
  1005. musicPlayer.volume -= 0.01;
  1006. else
  1007. musicPlayer.volume += 0.01;
  1008. } else if ([panType isEqual:@"Brightness"]) {
  1009. CGFloat brightness = [UIScreen mainScreen].brightness;
  1010. if (panDirectionY > 0)
  1011. brightness = brightness - 0.01;
  1012. else
  1013. brightness = brightness + 0.01;
  1014. // Sanity check since -[UIScreen brightness] does not go by 0.01 steps
  1015. if (brightness > 1.0)
  1016. brightness = 1.0;
  1017. else if (brightness < 0.0)
  1018. brightness = 0.0;
  1019. NSAssert(brightness >= 0 && brightness <= 1, @"Brightness must be within 0 and 1 (it is %f)", brightness);
  1020. [[UIScreen mainScreen] setBrightness:brightness];
  1021. self.brightnessSlider.value = brightness * 2.;
  1022. NSString *brightnessHUD = [NSString stringWithFormat:@"%@: %@ %%", NSLocalizedString(@"VFILTER_BRIGHTNESS", nil), [[[NSString stringWithFormat:@"%f",(brightness*100)] componentsSeparatedByString:@"."] objectAtIndex:0]];
  1023. [self.statusLabel showStatusMessage:brightnessHUD];
  1024. }
  1025. if (panRecognizer.state == UIGestureRecognizerStateEnded) {
  1026. if ([_mediaPlayer isPlaying])
  1027. [_listPlayer play];
  1028. }
  1029. }
  1030. - (void)swipeRecognized:(UISwipeGestureRecognizer*)swipeRecognizer
  1031. {
  1032. if (!_swipeGesturesEnabled)
  1033. return;
  1034. NSString * hudString = @" ";
  1035. if (swipeRecognizer.direction == UISwipeGestureRecognizerDirectionRight) {
  1036. double timeRemainingDouble = (-_mediaPlayer.remainingTime.intValue*0.001);
  1037. int timeRemaining = timeRemainingDouble;
  1038. if (FORWARD_SWIPE_DURATION < timeRemaining) {
  1039. [_mediaPlayer jumpForward:FORWARD_SWIPE_DURATION];
  1040. hudString = [NSString stringWithFormat:@"⇒ %is", FORWARD_SWIPE_DURATION];
  1041. } else {
  1042. [_mediaPlayer jumpForward:(timeRemaining - 5)];
  1043. hudString = [NSString stringWithFormat:@"⇒ %is",(timeRemaining - 5)];
  1044. }
  1045. }
  1046. else if (swipeRecognizer.direction == UISwipeGestureRecognizerDirectionLeft) {
  1047. [_mediaPlayer jumpBackward:BACKWARD_SWIPE_DURATION];
  1048. hudString = [NSString stringWithFormat:@"⇐ %is",BACKWARD_SWIPE_DURATION];
  1049. }
  1050. if (swipeRecognizer.state == UIGestureRecognizerStateEnded) {
  1051. if ([_mediaPlayer isPlaying])
  1052. [_listPlayer play];
  1053. [self.statusLabel showStatusMessage:hudString];
  1054. }
  1055. }
  1056. #pragma mark - Video Filter UI
  1057. - (IBAction)videoFilterToggle:(id)sender
  1058. {
  1059. if (!_playbackSpeedViewHidden)
  1060. self.playbackSpeedView.hidden = _playbackSpeedViewHidden = YES;
  1061. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  1062. if (!_controlsHidden) {
  1063. self.controllerPanel.hidden = _controlsHidden = YES;
  1064. self.controllerPanelLandscape.hidden = YES;
  1065. }
  1066. }
  1067. self.videoFilterView.hidden = !_videoFiltersHidden;
  1068. _videoFiltersHidden = self.videoFilterView.hidden;
  1069. }
  1070. - (IBAction)videoFilterSliderAction:(id)sender
  1071. {
  1072. if (sender == self.hueSlider)
  1073. _mediaPlayer.hue = (int)self.hueSlider.value;
  1074. else if (sender == self.contrastSlider)
  1075. _mediaPlayer.contrast = self.contrastSlider.value;
  1076. else if (sender == self.brightnessSlider) {
  1077. if ([self hasExternalDisplay])
  1078. _mediaPlayer.brightness = self.brightnessSlider.value;
  1079. else
  1080. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  1081. } else if (sender == self.saturationSlider)
  1082. _mediaPlayer.saturation = self.saturationSlider.value;
  1083. else if (sender == self.gammaSlider)
  1084. _mediaPlayer.gamma = self.gammaSlider.value;
  1085. else if (sender == self.resetVideoFilterButton) {
  1086. _mediaPlayer.hue = self.hueSlider.value = 0.;
  1087. _mediaPlayer.contrast = self.contrastSlider.value = 1.;
  1088. _mediaPlayer.brightness = self.brightnessSlider.value = 1.;
  1089. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  1090. _mediaPlayer.saturation = self.saturationSlider.value = 1.;
  1091. _mediaPlayer.gamma = self.gammaSlider.value = 1.;
  1092. } else
  1093. APLog(@"unknown sender for videoFilterSliderAction");
  1094. [self _resetIdleTimer];
  1095. }
  1096. #pragma mark - playback view
  1097. - (IBAction)playbackSpeedSliderAction:(UISlider *)sender
  1098. {
  1099. double speed = pow(2, sender.value / 17.);
  1100. float rate = INPUT_RATE_DEFAULT / speed;
  1101. if (_currentPlaybackRate != rate)
  1102. [_mediaPlayer setRate:INPUT_RATE_DEFAULT / rate];
  1103. _currentPlaybackRate = rate;
  1104. [self _updatePlaybackSpeedIndicator];
  1105. [self _resetIdleTimer];
  1106. }
  1107. - (void)_updatePlaybackSpeedIndicator
  1108. {
  1109. float f_value = self.playbackSpeedSlider.value;
  1110. double speed = pow(2, f_value / 17.);
  1111. self.playbackSpeedIndicator.text = [NSString stringWithFormat:@"%.2fx", speed];
  1112. /* rate changed, so update the exported info */
  1113. [self performSelectorInBackground:@selector(_updateDisplayedMetadata) withObject:nil];
  1114. }
  1115. - (float)_playbackSpeed
  1116. {
  1117. float f_rate = _mediaPlayer.rate;
  1118. double value = 17 * log(f_rate) / log(2.);
  1119. float returnValue = (int) ((value > 0) ? value + .5 : value - .5);
  1120. if (returnValue < -34.)
  1121. returnValue = -34.;
  1122. else if (returnValue > 34.)
  1123. returnValue = 34.;
  1124. _currentPlaybackRate = returnValue;
  1125. return returnValue;
  1126. }
  1127. - (IBAction)videoDimensionAction:(id)sender
  1128. {
  1129. if (sender == self.playbackSpeedButton || sender == self.playbackSpeedButtonLandscape) {
  1130. if (!_videoFiltersHidden)
  1131. self.videoFilterView.hidden = _videoFiltersHidden = YES;
  1132. self.playbackSpeedView.hidden = !_playbackSpeedViewHidden;
  1133. _playbackSpeedViewHidden = self.playbackSpeedView.hidden;
  1134. [self _resetIdleTimer];
  1135. } else if (sender == self.aspectRatioButton) {
  1136. NSUInteger count = [_aspectRatios count];
  1137. if (_currentAspectRatioMask + 1 > count - 1) {
  1138. _mediaPlayer.videoAspectRatio = NULL;
  1139. _mediaPlayer.videoCropGeometry = NULL;
  1140. _currentAspectRatioMask = 0;
  1141. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", nil), NSLocalizedString(@"DEFAULT", nil)]];
  1142. } else {
  1143. _currentAspectRatioMask++;
  1144. if ([_aspectRatios[_currentAspectRatioMask] isEqualToString:@"FILL_TO_SCREEN"]) {
  1145. UIScreen *screen;
  1146. if (![self hasExternalDisplay])
  1147. screen = [UIScreen mainScreen];
  1148. else
  1149. screen = [UIScreen screens][1];
  1150. float f_ar = screen.bounds.size.width / screen.bounds.size.height;
  1151. if (f_ar == (float)(640./1136.)) // iPhone 5 aka 16:9.01
  1152. _mediaPlayer.videoCropGeometry = "16:9";
  1153. else if (f_ar == (float)(2./3.)) // all other iPhones
  1154. _mediaPlayer.videoCropGeometry = "16:10"; // libvlc doesn't support 2:3 crop
  1155. else if (f_ar == .75) // all iPads
  1156. _mediaPlayer.videoCropGeometry = "4:3";
  1157. else if (f_ar == .5625) // AirPlay
  1158. _mediaPlayer.videoCropGeometry = "16:9";
  1159. else
  1160. APLog(@"unknown screen format %f, can't crop", f_ar);
  1161. [self.statusLabel showStatusMessage:NSLocalizedString(@"FILL_TO_SCREEN", nil)];
  1162. return;
  1163. }
  1164. _mediaPlayer.videoCropGeometry = NULL;
  1165. _mediaPlayer.videoAspectRatio = (char *)[_aspectRatios[_currentAspectRatioMask] UTF8String];
  1166. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", nil), _aspectRatios[_currentAspectRatioMask]]];
  1167. }
  1168. }
  1169. }
  1170. #pragma mark - background interaction
  1171. - (void)applicationWillResignActive:(NSNotification *)aNotification
  1172. {
  1173. [self _saveCurrentState];
  1174. _mediaPlayer.currentVideoTrackIndex = 0;
  1175. if (![[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingContinueAudioInBackgroundKey] boolValue]) {
  1176. if ([_mediaPlayer isPlaying]) {
  1177. [_mediaPlayer pause];
  1178. _shouldResumePlaying = YES;
  1179. }
  1180. }
  1181. }
  1182. - (void)applicationDidEnterBackground:(NSNotification *)notification
  1183. {
  1184. _shouldResumePlaying = NO;
  1185. }
  1186. - (void)applicationDidBecomeActive:(NSNotification *)notification
  1187. {
  1188. _mediaPlayer.currentVideoTrackIndex = 1;
  1189. if (_shouldResumePlaying) {
  1190. _shouldResumePlaying = NO;
  1191. [_listPlayer play];
  1192. }
  1193. }
  1194. - (void)audioSessionRouteChange:(NSNotification *)notification
  1195. {
  1196. NSArray *outputs = [[AVAudioSession sharedInstance] currentRoute].outputs;
  1197. NSString *portName = [[outputs objectAtIndex:0] portName];
  1198. if (![portName isEqualToString:@"Headphones"])
  1199. [_listPlayer pause];
  1200. }
  1201. - (void)mediaDidFinishParsing:(VLCMedia *)aMedia
  1202. {
  1203. [self _updateDisplayedMetadata];
  1204. }
  1205. - (void)mediaMetaDataDidChange:(VLCMedia*)aMedia
  1206. {
  1207. [self _updateDisplayedMetadata];
  1208. }
  1209. - (void)_updateDisplayedMetadata
  1210. {
  1211. MLFile *item;
  1212. NSString *title;
  1213. NSString *artist;
  1214. NSString *albumName;
  1215. NSString *trackNumber;
  1216. BOOL mediaIsAudioOnly = YES;
  1217. if (self.fileFromMediaLibrary)
  1218. item = self.fileFromMediaLibrary;
  1219. else if (self.mediaList) {
  1220. NSArray *matches = [MLFile fileForURL:[_mediaPlayer.media.url absoluteString]];
  1221. if (matches.count > 0)
  1222. item = matches[0];
  1223. }
  1224. if (item) {
  1225. if (item.isAlbumTrack) {
  1226. title = item.albumTrack.title;
  1227. artist = item.albumTrack.artist;
  1228. albumName = item.albumTrack.album.name;
  1229. } else
  1230. title = item.title;
  1231. /* MLKit knows better than us if this thing is audio only or not */
  1232. mediaIsAudioOnly = [item isSupportedAudioFile];
  1233. if (mediaIsAudioOnly)
  1234. self.artworkImageView.image = [VLCThumbnailsCache thumbnailForMediaFile:item];
  1235. } else {
  1236. NSDictionary * metaDict = _mediaPlayer.media.metaDictionary;
  1237. /* this is a non file media, so we need to actually check if there is there is
  1238. * a video track included or not */
  1239. NSArray *tracks = _mediaPlayer.media.tracksInformation;
  1240. NSUInteger trackCount = tracks.count;
  1241. for (NSUInteger x = 0 ; x < trackCount; x++) {
  1242. if ([[tracks[x] objectForKey:VLCMediaTracksInformationType] isEqualToString:VLCMediaTracksInformationTypeVideo]) {
  1243. mediaIsAudioOnly = NO;
  1244. break;
  1245. }
  1246. }
  1247. if (metaDict) {
  1248. title = metaDict[VLCMetaInformationNowPlaying] ? metaDict[VLCMetaInformationNowPlaying] : metaDict[VLCMetaInformationTitle];
  1249. artist = metaDict[VLCMetaInformationArtist];
  1250. albumName = metaDict[VLCMetaInformationAlbum];
  1251. trackNumber = metaDict[VLCMetaInformationTrackNumber];
  1252. if (mediaIsAudioOnly)
  1253. self.artworkImageView.image = [VLCThumbnailsCache thumbnailForMediaItemWithTitle:title Artist:artist andAlbumName:albumName];
  1254. }
  1255. }
  1256. if (mediaIsAudioOnly) {
  1257. if (!self.artworkImageView.image) {
  1258. self.trackNameLabel.text = title;
  1259. self.artistNameLabel.text = artist;
  1260. self.albumNameLabel.text = albumName;
  1261. } else {
  1262. NSString *trackName = title;
  1263. if (artist)
  1264. trackName = [trackName stringByAppendingFormat:@" — %@", artist];
  1265. if (albumName)
  1266. trackName = [trackName stringByAppendingFormat:@" — %@", albumName];
  1267. self.trackNameLabel.text = trackName;
  1268. }
  1269. if (self.trackNameLabel.text.length < 1)
  1270. self.trackNameLabel.text = [[_mediaPlayer.media url] lastPathComponent];
  1271. if (!self.aspectRatioButton.hidden) {
  1272. CGRect rect = self.timeDisplay.frame;
  1273. rect.origin.x += self.aspectRatioButton.frame.size.width;
  1274. self.timeDisplay.frame = rect;
  1275. rect = self.positionSlider.frame;
  1276. rect.size.width += self.aspectRatioButton.frame.size.width;
  1277. self.positionSlider.frame = rect;
  1278. self.aspectRatioButton.hidden = YES;
  1279. }
  1280. } else {
  1281. if (self.aspectRatioButton.hidden) {
  1282. CGRect rect = self.timeDisplay.frame;
  1283. rect.origin.x -= self.aspectRatioButton.frame.size.width;
  1284. self.timeDisplay.frame = rect;
  1285. rect = self.positionSlider.frame;
  1286. rect.size.width -= self.aspectRatioButton.frame.size.width;
  1287. self.positionSlider.frame = rect;
  1288. self.aspectRatioButton.hidden = NO;
  1289. }
  1290. }
  1291. self.videoFilterButton.hidden = mediaIsAudioOnly;
  1292. /* don't leak sensitive information to the OS, if passcode lock is enabled */
  1293. BOOL passcodeLockEnabled = [[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingPasscodeOnKey] boolValue];
  1294. NSMutableDictionary *currentlyPlayingTrackInfo;
  1295. if (passcodeLockEnabled)
  1296. currentlyPlayingTrackInfo = [NSMutableDictionary dictionaryWithObjectsAndKeys:@(_mediaPlayer.media.length.intValue / 1000.), MPMediaItemPropertyPlaybackDuration, @(_mediaPlayer.time.intValue / 1000.), MPNowPlayingInfoPropertyElapsedPlaybackTime, @(_mediaPlayer.rate), MPNowPlayingInfoPropertyPlaybackRate, nil];
  1297. else {
  1298. currentlyPlayingTrackInfo = [NSMutableDictionary dictionaryWithObjectsAndKeys: title, MPMediaItemPropertyTitle, @(_mediaPlayer.media.length.intValue / 1000.), MPMediaItemPropertyPlaybackDuration, @(_mediaPlayer.time.intValue / 1000.), MPNowPlayingInfoPropertyElapsedPlaybackTime, @(_mediaPlayer.rate), MPNowPlayingInfoPropertyPlaybackRate, nil];
  1299. if (artist.length > 0)
  1300. [currentlyPlayingTrackInfo setObject:artist forKey:MPMediaItemPropertyArtist];
  1301. if (albumName.length > 0)
  1302. [currentlyPlayingTrackInfo setObject:albumName forKey:MPMediaItemPropertyAlbumTitle];
  1303. [currentlyPlayingTrackInfo setObject:[NSNumber numberWithInt:[trackNumber intValue]] forKey:MPMediaItemPropertyAlbumTrackNumber];
  1304. if (self.artworkImageView.image) {
  1305. MPMediaItemArtwork *mpartwork = [[MPMediaItemArtwork alloc] initWithImage:self.artworkImageView.image];
  1306. [currentlyPlayingTrackInfo setObject:mpartwork forKey:MPMediaItemPropertyArtwork];
  1307. }
  1308. }
  1309. [MPNowPlayingInfoCenter defaultCenter].nowPlayingInfo = currentlyPlayingTrackInfo;
  1310. }
  1311. #pragma mark - autorotation
  1312. - (BOOL)shouldAutorotate
  1313. {
  1314. UIInterfaceOrientation toInterfaceOrientation = [[UIApplication sharedApplication] statusBarOrientation];
  1315. return UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad
  1316. || toInterfaceOrientation != UIInterfaceOrientationPortraitUpsideDown;
  1317. }
  1318. - (void)willRotateToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation duration:(NSTimeInterval)duration
  1319. {
  1320. [super willRotateToInterfaceOrientation:toInterfaceOrientation duration:duration];
  1321. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  1322. if (self.artworkImageView.image)
  1323. self.trackNameLabel.hidden = UIInterfaceOrientationIsLandscape(toInterfaceOrientation);
  1324. }
  1325. }
  1326. #pragma mark - AVSession delegate
  1327. - (void)beginInterruption
  1328. {
  1329. if ([_mediaPlayer isPlaying]) {
  1330. [_mediaPlayer pause];
  1331. _shouldResumePlaying = YES;
  1332. }
  1333. }
  1334. - (void)endInterruption
  1335. {
  1336. if (_shouldResumePlaying) {
  1337. [_mediaPlayer play];
  1338. _shouldResumePlaying = NO;
  1339. }
  1340. }
  1341. #pragma mark - External Display
  1342. - (BOOL)hasExternalDisplay
  1343. {
  1344. return ([[UIScreen screens] count] > 1);
  1345. }
  1346. - (void)showOnExternalDisplay
  1347. {
  1348. UIScreen *screen = [UIScreen screens][1];
  1349. screen.overscanCompensation = UIScreenOverscanCompensationInsetApplicationFrame;
  1350. self.externalWindow = [[UIWindow alloc] initWithFrame:screen.bounds];
  1351. UIViewController *controller = [[VLCExternalDisplayController alloc] init];
  1352. self.externalWindow.rootViewController = controller;
  1353. [controller.view addSubview:_movieView];
  1354. controller.view.frame = screen.bounds;
  1355. _movieView.frame = screen.bounds;
  1356. self.playingExternallyView.hidden = NO;
  1357. self.externalWindow.screen = screen;
  1358. self.externalWindow.hidden = NO;
  1359. }
  1360. - (void)hideFromExternalDisplay
  1361. {
  1362. [self.view addSubview:_movieView];
  1363. [self.view sendSubviewToBack:_movieView];
  1364. _movieView.frame = self.view.frame;
  1365. self.playingExternallyView.hidden = YES;
  1366. self.externalWindow.hidden = YES;
  1367. self.externalWindow = nil;
  1368. }
  1369. - (void)handleExternalScreenDidConnect:(NSNotification *)notification
  1370. {
  1371. [self showOnExternalDisplay];
  1372. }
  1373. - (void)handleExternalScreenDidDisconnect:(NSNotification *)notification
  1374. {
  1375. [self hideFromExternalDisplay];
  1376. }
  1377. @end