VLCMovieViewController.m 65 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658
  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. // switch back to the caller when user presses "Done"
  774. if (self.successCallback && [sender isKindOfClass:[UIBarButtonItem class]]) {
  775. [[UIApplication sharedApplication] openURL:self.successCallback];
  776. }
  777. }
  778. - (IBAction)positionSliderAction:(UISlider *)sender
  779. {
  780. /* we need to limit the number of events sent by the slider, since otherwise, the user
  781. * wouldn't see the I-frames when seeking on current mobile devices. This isn't a problem
  782. * within the Simulator, but especially on older ARMv7 devices, it's clearly noticeable. */
  783. [self performSelector:@selector(_setPositionForReal) withObject:nil afterDelay:0.3];
  784. VLCTime *newPosition = [VLCTime timeWithInt:(int)(_positionSlider.value * self.fileFromMediaLibrary.duration.intValue)];
  785. [self.timeDisplay setTitle:newPosition.stringValue forState:UIControlStateNormal];
  786. self.timeDisplay.accessibilityLabel = [NSString stringWithFormat:@"%@: %@", NSLocalizedString(@"PLAYBACK_POSITION", nil), newPosition.stringValue];
  787. _positionSet = NO;
  788. [self _resetIdleTimer];
  789. }
  790. - (void)_setPositionForReal
  791. {
  792. if (!_positionSet) {
  793. _mediaPlayer.position = _positionSlider.value;
  794. _positionSet = YES;
  795. }
  796. }
  797. - (IBAction)positionSliderTouchDown:(id)sender
  798. {
  799. [self _updateScrubLabel];
  800. self.scrubIndicatorView.hidden = NO;
  801. _isScrubbing = YES;
  802. }
  803. - (IBAction)positionSliderTouchUp:(id)sender
  804. {
  805. self.scrubIndicatorView.hidden = YES;
  806. _isScrubbing = NO;
  807. }
  808. - (void)_updateScrubLabel
  809. {
  810. float speed = self.positionSlider.scrubbingSpeed;
  811. if (speed == 1.)
  812. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HIGH", nil);
  813. else if (speed == .5)
  814. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HALF", nil);
  815. else if (speed == .25)
  816. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_QUARTER", nil);
  817. else
  818. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_FINE", nil);
  819. [self _resetIdleTimer];
  820. }
  821. - (IBAction)positionSliderDrag:(id)sender
  822. {
  823. [self _updateScrubLabel];
  824. }
  825. - (IBAction)volumeSliderAction:(id)sender
  826. {
  827. [self _resetIdleTimer];
  828. }
  829. - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
  830. {
  831. if (!_isScrubbing) {
  832. self.positionSlider.value = [_mediaPlayer position];
  833. }
  834. if (_displayRemainingTime)
  835. [self.timeDisplay setTitle:[[_mediaPlayer remainingTime] stringValue] forState:UIControlStateNormal];
  836. else
  837. [self.timeDisplay setTitle:[[_mediaPlayer time] stringValue] forState:UIControlStateNormal];
  838. }
  839. - (void)mediaPlayerStateChanged:(NSNotification *)aNotification
  840. {
  841. VLCMediaPlayerState currentState = _mediaPlayer.state;
  842. if (currentState == VLCMediaPlayerStateBuffering) {
  843. /* attach delegate */
  844. _mediaPlayer.media.delegate = self;
  845. /* let's update meta data */
  846. [self _updateDisplayedMetadata];
  847. /* on-the-fly values through private API */
  848. [_mediaPlayer performSelector:@selector(setTextRendererFont:) withObject:[self _resolveFontName]];
  849. [_mediaPlayer performSelector:@selector(setTextRendererFontSize:) withObject:[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingSubtitlesFontSize]];
  850. [_mediaPlayer performSelector:@selector(setTextRendererFontColor:) withObject:[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingSubtitlesFontColor]];
  851. }
  852. if (currentState == VLCMediaPlayerStateError) {
  853. [self.statusLabel showStatusMessage:NSLocalizedString(@"PLAYBACK_FAILED", nil)];
  854. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  855. }
  856. if ((currentState == VLCMediaPlayerStateEnded || currentState == VLCMediaPlayerStateStopped) && _listPlayer.repeatMode == VLCDoNotRepeat) {
  857. if ([_listPlayer.mediaList indexOfMedia:_mediaPlayer.media] == _listPlayer.mediaList.count - 1)
  858. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  859. }
  860. UIImage *playPauseImage = [_mediaPlayer isPlaying]? [UIImage imageNamed:@"pauseIcon"] : [UIImage imageNamed:@"playIcon"];
  861. [_playPauseButton setImage:playPauseImage forState:UIControlStateNormal];
  862. [_playPauseButtonLandscape setImage:playPauseImage forState:UIControlStateNormal];
  863. if ([[_mediaPlayer audioTrackIndexes] count] > 2) {
  864. self.audioSwitcherButton.hidden = NO;
  865. self.audioSwitcherButtonLandscape.hidden = NO;
  866. } else {
  867. self.audioSwitcherButton.hidden = YES;
  868. self.audioSwitcherButtonLandscape.hidden = YES;
  869. }
  870. if ([[_mediaPlayer videoSubTitlesIndexes] count] > 1) {
  871. self.subtitleContainer.hidden = NO;
  872. self.subtitleContainerLandscape.hidden = NO;
  873. } else {
  874. self.subtitleContainer.hidden = YES;
  875. self.subtitleContainerLandscape.hidden = YES;
  876. }
  877. }
  878. - (IBAction)playPause
  879. {
  880. if ([_mediaPlayer isPlaying])
  881. [_listPlayer pause];
  882. else
  883. [_listPlayer play];
  884. }
  885. - (IBAction)forward:(id)sender
  886. {
  887. if (self.mediaList)
  888. [_listPlayer next];
  889. else
  890. [_mediaPlayer mediumJumpForward];
  891. }
  892. - (IBAction)backward:(id)sender
  893. {
  894. if (self.mediaList)
  895. [_listPlayer previous];
  896. else
  897. [_mediaPlayer mediumJumpBackward];
  898. }
  899. - (void)toggleRepeatMode:(id)sender
  900. {
  901. if (_listPlayer.repeatMode == VLCDoNotRepeat) {
  902. _listPlayer.repeatMode = VLCRepeatCurrentItem;
  903. [self.repeatButton setImage:[UIImage imageNamed:@"repeatOne"] forState:UIControlStateNormal];
  904. [self.repeatButtonLandscape setImage:[UIImage imageNamed:@"repeatOne"] forState:UIControlStateNormal];
  905. } else {
  906. _listPlayer.repeatMode = VLCDoNotRepeat;
  907. [self.repeatButton setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  908. [self.repeatButtonLandscape setImage:[UIImage imageNamed:@"repeat"] forState:UIControlStateNormal];
  909. }
  910. }
  911. - (IBAction)switchAudioTrack:(id)sender
  912. {
  913. _audiotrackActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_AUDIO_TRACK", @"audio track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  914. NSArray *audioTracks = [_mediaPlayer audioTrackNames];
  915. NSArray *audioTrackIndexes = [_mediaPlayer audioTrackIndexes];
  916. NSUInteger count = [audioTracks count];
  917. for (NSUInteger i = 0; i < count; i++) {
  918. NSString *indexIndicator = ([audioTrackIndexes[i] intValue] == [_mediaPlayer currentAudioTrackIndex])? @"\u2713": @"";
  919. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, audioTracks[i]];
  920. [_audiotrackActionSheet addButtonWithTitle:buttonTitle];
  921. }
  922. [_audiotrackActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  923. [_audiotrackActionSheet setCancelButtonIndex:[_audiotrackActionSheet numberOfButtons] - 1];
  924. [_audiotrackActionSheet showInView:(UIButton *)sender];
  925. }
  926. - (IBAction)switchSubtitleTrack:(id)sender
  927. {
  928. NSArray *spuTracks = [_mediaPlayer videoSubTitlesNames];
  929. NSArray *spuTrackIndexes = [_mediaPlayer videoSubTitlesIndexes];
  930. NSUInteger count = [spuTracks count];
  931. if (count <= 1)
  932. return;
  933. _subtitleActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_SUBTITLE_TRACK", @"subtitle track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  934. for (NSUInteger i = 0; i < count; i++) {
  935. NSString *indexIndicator = ([spuTrackIndexes[i] intValue] == [_mediaPlayer currentVideoSubTitleIndex])? @"\u2713": @"";
  936. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, spuTracks[i]];
  937. [_subtitleActionSheet addButtonWithTitle:buttonTitle];
  938. }
  939. [_subtitleActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  940. [_subtitleActionSheet setCancelButtonIndex:[_subtitleActionSheet numberOfButtons] - 1];
  941. [_subtitleActionSheet showInView:(UIButton *)sender];
  942. }
  943. - (void)actionSheet:(UIActionSheet *)actionSheet clickedButtonAtIndex:(NSInteger)buttonIndex
  944. {
  945. if (buttonIndex == [actionSheet cancelButtonIndex])
  946. return;
  947. NSArray *indexArray;
  948. if (actionSheet == _subtitleActionSheet) {
  949. indexArray = _mediaPlayer.videoSubTitlesIndexes;
  950. if (buttonIndex <= indexArray.count) {
  951. _mediaPlayer.currentVideoSubTitleIndex = [indexArray[buttonIndex] intValue];
  952. }
  953. } else if (actionSheet == _audiotrackActionSheet) {
  954. indexArray = _mediaPlayer.audioTrackIndexes;
  955. if (buttonIndex <= indexArray.count) {
  956. _mediaPlayer.currentAudioTrackIndex = [indexArray[buttonIndex] intValue];
  957. }
  958. }
  959. }
  960. - (IBAction)toggleTimeDisplay:(id)sender
  961. {
  962. _displayRemainingTime = !_displayRemainingTime;
  963. [self _resetIdleTimer];
  964. }
  965. #pragma mark - multi-touch gestures
  966. - (void)tapRecognized
  967. {
  968. if (!_swipeGesturesEnabled)
  969. return;
  970. if ([_mediaPlayer isPlaying]) {
  971. [_listPlayer pause];
  972. [self.statusLabel showStatusMessage:@" ▌▌"];
  973. } else {
  974. [_listPlayer play];
  975. [self.statusLabel showStatusMessage:@" ►"];
  976. }
  977. }
  978. - (NSString*)detectPanTypeForPan:(UIPanGestureRecognizer*)panRecognizer
  979. {
  980. NSString * type;
  981. NSString * deviceType = [[UIDevice currentDevice] model];
  982. type = @"Volume"; // default in case of error
  983. CGPoint location = [panRecognizer locationInView:self.view];
  984. CGFloat position = location.x;
  985. UIInterfaceOrientation orientation = [[UIApplication sharedApplication] statusBarOrientation];
  986. CGRect screenRect = [[UIScreen mainScreen] bounds];
  987. CGFloat screenWidth = .0;
  988. if (orientation == UIDeviceOrientationPortrait)
  989. screenWidth = screenRect.size.width;
  990. else
  991. screenWidth = screenRect.size.height;
  992. if (position < screenWidth / 2)
  993. type = @"Brightness";
  994. if (position > screenWidth / 2)
  995. type = @"Volume";
  996. // only check for seeking gesture if on iPad , will overwrite last statements if true
  997. if ([deviceType isEqualToString:@"iPad"]) {
  998. if (location.y < 110)
  999. type = @"Seek";
  1000. }
  1001. return type;
  1002. }
  1003. - (void)panRecognized:(UIPanGestureRecognizer*)panRecognizer
  1004. {
  1005. if (!_swipeGesturesEnabled)
  1006. return;
  1007. CGFloat panDirectionX = [panRecognizer velocityInView:self.view].x;
  1008. CGFloat panDirectionY = [panRecognizer velocityInView:self.view].y;
  1009. if (panRecognizer.state == UIGestureRecognizerStateBegan) // Only Detect pantype when began to allow more freedom
  1010. panType = [self detectPanTypeForPan:panRecognizer];
  1011. if ([panType isEqual:@"Seek"]) {
  1012. double timeRemainingDouble = (-_mediaPlayer.remainingTime.intValue*0.001);
  1013. int timeRemaining = timeRemainingDouble;
  1014. if (panDirectionX > 0) {
  1015. if (timeRemaining > 2 ) // to not go outside duration , video will stop
  1016. [_mediaPlayer jumpForward:1];
  1017. } else
  1018. [_mediaPlayer jumpBackward:1];
  1019. } else if ([panType isEqual:@"Volume"]) {
  1020. MPMusicPlayerController *musicPlayer = [MPMusicPlayerController applicationMusicPlayer];
  1021. if (panDirectionY > 0)
  1022. musicPlayer.volume -= 0.01;
  1023. else
  1024. musicPlayer.volume += 0.01;
  1025. } else if ([panType isEqual:@"Brightness"]) {
  1026. CGFloat brightness = [UIScreen mainScreen].brightness;
  1027. if (panDirectionY > 0)
  1028. brightness = brightness - 0.01;
  1029. else
  1030. brightness = brightness + 0.01;
  1031. // Sanity check since -[UIScreen brightness] does not go by 0.01 steps
  1032. if (brightness > 1.0)
  1033. brightness = 1.0;
  1034. else if (brightness < 0.0)
  1035. brightness = 0.0;
  1036. NSAssert(brightness >= 0 && brightness <= 1, @"Brightness must be within 0 and 1 (it is %f)", brightness);
  1037. [[UIScreen mainScreen] setBrightness:brightness];
  1038. self.brightnessSlider.value = brightness * 2.;
  1039. NSString *brightnessHUD = [NSString stringWithFormat:@"%@: %@ %%", NSLocalizedString(@"VFILTER_BRIGHTNESS", nil), [[[NSString stringWithFormat:@"%f",(brightness*100)] componentsSeparatedByString:@"."] objectAtIndex:0]];
  1040. [self.statusLabel showStatusMessage:brightnessHUD];
  1041. }
  1042. if (panRecognizer.state == UIGestureRecognizerStateEnded) {
  1043. if ([_mediaPlayer isPlaying])
  1044. [_listPlayer play];
  1045. }
  1046. }
  1047. - (void)swipeRecognized:(UISwipeGestureRecognizer*)swipeRecognizer
  1048. {
  1049. if (!_swipeGesturesEnabled)
  1050. return;
  1051. NSString * hudString = @" ";
  1052. if (swipeRecognizer.direction == UISwipeGestureRecognizerDirectionRight) {
  1053. double timeRemainingDouble = (-_mediaPlayer.remainingTime.intValue*0.001);
  1054. int timeRemaining = timeRemainingDouble;
  1055. if (FORWARD_SWIPE_DURATION < timeRemaining) {
  1056. [_mediaPlayer jumpForward:FORWARD_SWIPE_DURATION];
  1057. hudString = [NSString stringWithFormat:@"⇒ %is", FORWARD_SWIPE_DURATION];
  1058. } else {
  1059. [_mediaPlayer jumpForward:(timeRemaining - 5)];
  1060. hudString = [NSString stringWithFormat:@"⇒ %is",(timeRemaining - 5)];
  1061. }
  1062. }
  1063. else if (swipeRecognizer.direction == UISwipeGestureRecognizerDirectionLeft) {
  1064. [_mediaPlayer jumpBackward:BACKWARD_SWIPE_DURATION];
  1065. hudString = [NSString stringWithFormat:@"⇐ %is",BACKWARD_SWIPE_DURATION];
  1066. }
  1067. if (swipeRecognizer.state == UIGestureRecognizerStateEnded) {
  1068. if ([_mediaPlayer isPlaying])
  1069. [_listPlayer play];
  1070. [self.statusLabel showStatusMessage:hudString];
  1071. }
  1072. }
  1073. #pragma mark - Video Filter UI
  1074. - (IBAction)videoFilterToggle:(id)sender
  1075. {
  1076. if (!_playbackSpeedViewHidden)
  1077. self.playbackSpeedView.hidden = _playbackSpeedViewHidden = YES;
  1078. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  1079. if (!_controlsHidden) {
  1080. self.controllerPanel.hidden = _controlsHidden = YES;
  1081. self.controllerPanelLandscape.hidden = YES;
  1082. }
  1083. }
  1084. self.videoFilterView.hidden = !_videoFiltersHidden;
  1085. _videoFiltersHidden = self.videoFilterView.hidden;
  1086. }
  1087. - (IBAction)videoFilterSliderAction:(id)sender
  1088. {
  1089. if (sender == self.hueSlider)
  1090. _mediaPlayer.hue = (int)self.hueSlider.value;
  1091. else if (sender == self.contrastSlider)
  1092. _mediaPlayer.contrast = self.contrastSlider.value;
  1093. else if (sender == self.brightnessSlider) {
  1094. if ([self hasExternalDisplay])
  1095. _mediaPlayer.brightness = self.brightnessSlider.value;
  1096. else
  1097. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  1098. } else if (sender == self.saturationSlider)
  1099. _mediaPlayer.saturation = self.saturationSlider.value;
  1100. else if (sender == self.gammaSlider)
  1101. _mediaPlayer.gamma = self.gammaSlider.value;
  1102. else if (sender == self.resetVideoFilterButton) {
  1103. _mediaPlayer.hue = self.hueSlider.value = 0.;
  1104. _mediaPlayer.contrast = self.contrastSlider.value = 1.;
  1105. _mediaPlayer.brightness = self.brightnessSlider.value = 1.;
  1106. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  1107. _mediaPlayer.saturation = self.saturationSlider.value = 1.;
  1108. _mediaPlayer.gamma = self.gammaSlider.value = 1.;
  1109. } else
  1110. APLog(@"unknown sender for videoFilterSliderAction");
  1111. [self _resetIdleTimer];
  1112. }
  1113. #pragma mark - playback view
  1114. - (IBAction)playbackSliderAction:(UISlider *)sender
  1115. {
  1116. if (sender == _playbackSpeedSlider) {
  1117. double speed = pow(2, sender.value / 17.);
  1118. float rate = INPUT_RATE_DEFAULT / speed;
  1119. if (_currentPlaybackRate != rate)
  1120. [_mediaPlayer setRate:INPUT_RATE_DEFAULT / rate];
  1121. _currentPlaybackRate = rate;
  1122. [self _updatePlaybackSpeedIndicator];
  1123. } else if (sender == _audioDelaySlider) {
  1124. _mediaPlayer.currentAudioPlaybackDelay = _audioDelaySlider.value * 1000000;
  1125. _audioDelayIndicator.text = [NSString stringWithFormat:@"%1.2f s", _audioDelaySlider.value];
  1126. } else if (sender == _spuDelaySlider) {
  1127. _mediaPlayer.currentVideoSubTitleDelay = _spuDelaySlider.value * 1000000;
  1128. _spuDelayIndicator.text = [NSString stringWithFormat:@"%1.00f s", _spuDelaySlider.value];
  1129. }
  1130. [self _resetIdleTimer];
  1131. }
  1132. - (void)_updatePlaybackSpeedIndicator
  1133. {
  1134. float f_value = self.playbackSpeedSlider.value;
  1135. double speed = pow(2, f_value / 17.);
  1136. self.playbackSpeedIndicator.text = [NSString stringWithFormat:@"%.2fx", speed];
  1137. /* rate changed, so update the exported info */
  1138. [self performSelectorInBackground:@selector(_updateDisplayedMetadata) withObject:nil];
  1139. }
  1140. - (float)_playbackSpeed
  1141. {
  1142. float f_rate = _mediaPlayer.rate;
  1143. double value = 17 * log(f_rate) / log(2.);
  1144. float returnValue = (int) ((value > 0) ? value + .5 : value - .5);
  1145. if (returnValue < -34.)
  1146. returnValue = -34.;
  1147. else if (returnValue > 34.)
  1148. returnValue = 34.;
  1149. _currentPlaybackRate = returnValue;
  1150. return returnValue;
  1151. }
  1152. - (IBAction)videoDimensionAction:(id)sender
  1153. {
  1154. if (sender == self.playbackSpeedButton || sender == self.playbackSpeedButtonLandscape) {
  1155. if (!_videoFiltersHidden)
  1156. self.videoFilterView.hidden = _videoFiltersHidden = YES;
  1157. self.playbackSpeedView.hidden = !_playbackSpeedViewHidden;
  1158. _playbackSpeedViewHidden = self.playbackSpeedView.hidden;
  1159. [self _resetIdleTimer];
  1160. } else if (sender == self.aspectRatioButton) {
  1161. NSUInteger count = [_aspectRatios count];
  1162. if (_currentAspectRatioMask + 1 > count - 1) {
  1163. _mediaPlayer.videoAspectRatio = NULL;
  1164. _mediaPlayer.videoCropGeometry = NULL;
  1165. _currentAspectRatioMask = 0;
  1166. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", nil), NSLocalizedString(@"DEFAULT", nil)]];
  1167. } else {
  1168. _currentAspectRatioMask++;
  1169. if ([_aspectRatios[_currentAspectRatioMask] isEqualToString:@"FILL_TO_SCREEN"]) {
  1170. UIScreen *screen;
  1171. if (![self hasExternalDisplay])
  1172. screen = [UIScreen mainScreen];
  1173. else
  1174. screen = [UIScreen screens][1];
  1175. float f_ar = screen.bounds.size.width / screen.bounds.size.height;
  1176. if (f_ar == (float)(640./1136.)) // iPhone 5 aka 16:9.01
  1177. _mediaPlayer.videoCropGeometry = "16:9";
  1178. else if (f_ar == (float)(2./3.)) // all other iPhones
  1179. _mediaPlayer.videoCropGeometry = "16:10"; // libvlc doesn't support 2:3 crop
  1180. else if (f_ar == .75) // all iPads
  1181. _mediaPlayer.videoCropGeometry = "4:3";
  1182. else if (f_ar == .5625) // AirPlay
  1183. _mediaPlayer.videoCropGeometry = "16:9";
  1184. else
  1185. APLog(@"unknown screen format %f, can't crop", f_ar);
  1186. [self.statusLabel showStatusMessage:NSLocalizedString(@"FILL_TO_SCREEN", nil)];
  1187. return;
  1188. }
  1189. _mediaPlayer.videoCropGeometry = NULL;
  1190. _mediaPlayer.videoAspectRatio = (char *)[_aspectRatios[_currentAspectRatioMask] UTF8String];
  1191. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", nil), _aspectRatios[_currentAspectRatioMask]]];
  1192. }
  1193. }
  1194. }
  1195. #pragma mark - background interaction
  1196. - (void)applicationWillResignActive:(NSNotification *)aNotification
  1197. {
  1198. [self _saveCurrentState];
  1199. _mediaPlayer.currentVideoTrackIndex = 0;
  1200. if (![[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingContinueAudioInBackgroundKey] boolValue]) {
  1201. if ([_mediaPlayer isPlaying]) {
  1202. [_mediaPlayer pause];
  1203. _shouldResumePlaying = YES;
  1204. }
  1205. }
  1206. }
  1207. - (void)applicationDidEnterBackground:(NSNotification *)notification
  1208. {
  1209. _shouldResumePlaying = NO;
  1210. }
  1211. - (void)applicationDidBecomeActive:(NSNotification *)notification
  1212. {
  1213. _mediaPlayer.currentVideoTrackIndex = 1;
  1214. if (_shouldResumePlaying) {
  1215. _shouldResumePlaying = NO;
  1216. [_listPlayer play];
  1217. }
  1218. }
  1219. - (void)audioSessionRouteChange:(NSNotification *)notification
  1220. {
  1221. NSArray *outputs = [[AVAudioSession sharedInstance] currentRoute].outputs;
  1222. NSString *portName = [[outputs objectAtIndex:0] portName];
  1223. if (![portName isEqualToString:@"Headphones"] && [_mediaPlayer isPlaying])
  1224. [_listPlayer pause];
  1225. }
  1226. - (void)mediaDidFinishParsing:(VLCMedia *)aMedia
  1227. {
  1228. [self _updateDisplayedMetadata];
  1229. }
  1230. - (void)mediaMetaDataDidChange:(VLCMedia*)aMedia
  1231. {
  1232. [self _updateDisplayedMetadata];
  1233. }
  1234. - (void)_updateDisplayedMetadata
  1235. {
  1236. MLFile *item;
  1237. NSString *title;
  1238. NSString *artist;
  1239. NSString *albumName;
  1240. NSString *trackNumber;
  1241. BOOL mediaIsAudioOnly = YES;
  1242. if (self.fileFromMediaLibrary)
  1243. item = self.fileFromMediaLibrary;
  1244. else if (self.mediaList) {
  1245. NSArray *matches = [MLFile fileForURL:[_mediaPlayer.media.url absoluteString]];
  1246. if (matches.count > 0)
  1247. item = matches[0];
  1248. }
  1249. if (item) {
  1250. if (item.isAlbumTrack) {
  1251. title = item.albumTrack.title;
  1252. artist = item.albumTrack.artist;
  1253. albumName = item.albumTrack.album.name;
  1254. } else
  1255. title = item.title;
  1256. /* MLKit knows better than us if this thing is audio only or not */
  1257. mediaIsAudioOnly = [item isSupportedAudioFile];
  1258. if (mediaIsAudioOnly)
  1259. self.artworkImageView.image = [VLCThumbnailsCache thumbnailForMediaFile:item];
  1260. } else {
  1261. NSDictionary * metaDict = _mediaPlayer.media.metaDictionary;
  1262. /* this is a non file media, so we need to actually check if there is there is
  1263. * a video track included or not */
  1264. NSArray *tracks = _mediaPlayer.media.tracksInformation;
  1265. NSUInteger trackCount = tracks.count;
  1266. for (NSUInteger x = 0 ; x < trackCount; x++) {
  1267. if ([[tracks[x] objectForKey:VLCMediaTracksInformationType] isEqualToString:VLCMediaTracksInformationTypeVideo]) {
  1268. mediaIsAudioOnly = NO;
  1269. break;
  1270. }
  1271. }
  1272. if (metaDict) {
  1273. title = metaDict[VLCMetaInformationNowPlaying] ? metaDict[VLCMetaInformationNowPlaying] : metaDict[VLCMetaInformationTitle];
  1274. artist = metaDict[VLCMetaInformationArtist];
  1275. albumName = metaDict[VLCMetaInformationAlbum];
  1276. trackNumber = metaDict[VLCMetaInformationTrackNumber];
  1277. if (mediaIsAudioOnly)
  1278. self.artworkImageView.image = [VLCThumbnailsCache thumbnailForMediaItemWithTitle:title Artist:artist andAlbumName:albumName];
  1279. }
  1280. }
  1281. if (mediaIsAudioOnly) {
  1282. if (!self.artworkImageView.image) {
  1283. self.trackNameLabel.text = title;
  1284. self.artistNameLabel.text = artist;
  1285. self.albumNameLabel.text = albumName;
  1286. } else {
  1287. NSString *trackName = title;
  1288. if (artist)
  1289. trackName = [trackName stringByAppendingFormat:@" — %@", artist];
  1290. if (albumName)
  1291. trackName = [trackName stringByAppendingFormat:@" — %@", albumName];
  1292. self.trackNameLabel.text = trackName;
  1293. }
  1294. if (self.trackNameLabel.text.length < 1)
  1295. self.trackNameLabel.text = [[_mediaPlayer.media url] lastPathComponent];
  1296. if (!self.aspectRatioButton.hidden) {
  1297. CGRect rect = self.timeDisplay.frame;
  1298. rect.origin.x += self.aspectRatioButton.frame.size.width;
  1299. self.timeDisplay.frame = rect;
  1300. rect = self.positionSlider.frame;
  1301. rect.size.width += self.aspectRatioButton.frame.size.width;
  1302. self.positionSlider.frame = rect;
  1303. self.aspectRatioButton.hidden = YES;
  1304. }
  1305. } else {
  1306. if (self.aspectRatioButton.hidden) {
  1307. CGRect rect = self.timeDisplay.frame;
  1308. rect.origin.x -= self.aspectRatioButton.frame.size.width;
  1309. self.timeDisplay.frame = rect;
  1310. rect = self.positionSlider.frame;
  1311. rect.size.width -= self.aspectRatioButton.frame.size.width;
  1312. self.positionSlider.frame = rect;
  1313. self.aspectRatioButton.hidden = NO;
  1314. }
  1315. }
  1316. self.videoFilterButton.hidden = mediaIsAudioOnly;
  1317. /* don't leak sensitive information to the OS, if passcode lock is enabled */
  1318. BOOL passcodeLockEnabled = [[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingPasscodeOnKey] boolValue];
  1319. NSMutableDictionary *currentlyPlayingTrackInfo;
  1320. if (passcodeLockEnabled)
  1321. currentlyPlayingTrackInfo = [NSMutableDictionary dictionaryWithObjectsAndKeys:@(_mediaPlayer.media.length.intValue / 1000.), MPMediaItemPropertyPlaybackDuration, @(_mediaPlayer.time.intValue / 1000.), MPNowPlayingInfoPropertyElapsedPlaybackTime, @(_mediaPlayer.rate), MPNowPlayingInfoPropertyPlaybackRate, nil];
  1322. else {
  1323. currentlyPlayingTrackInfo = [NSMutableDictionary dictionaryWithObjectsAndKeys: title, MPMediaItemPropertyTitle, @(_mediaPlayer.media.length.intValue / 1000.), MPMediaItemPropertyPlaybackDuration, @(_mediaPlayer.time.intValue / 1000.), MPNowPlayingInfoPropertyElapsedPlaybackTime, @(_mediaPlayer.rate), MPNowPlayingInfoPropertyPlaybackRate, nil];
  1324. if (artist.length > 0)
  1325. [currentlyPlayingTrackInfo setObject:artist forKey:MPMediaItemPropertyArtist];
  1326. if (albumName.length > 0)
  1327. [currentlyPlayingTrackInfo setObject:albumName forKey:MPMediaItemPropertyAlbumTitle];
  1328. [currentlyPlayingTrackInfo setObject:[NSNumber numberWithInt:[trackNumber intValue]] forKey:MPMediaItemPropertyAlbumTrackNumber];
  1329. if (self.artworkImageView.image) {
  1330. MPMediaItemArtwork *mpartwork = [[MPMediaItemArtwork alloc] initWithImage:self.artworkImageView.image];
  1331. [currentlyPlayingTrackInfo setObject:mpartwork forKey:MPMediaItemPropertyArtwork];
  1332. }
  1333. }
  1334. [MPNowPlayingInfoCenter defaultCenter].nowPlayingInfo = currentlyPlayingTrackInfo;
  1335. }
  1336. #pragma mark - autorotation
  1337. - (BOOL)shouldAutorotate
  1338. {
  1339. UIInterfaceOrientation toInterfaceOrientation = [[UIApplication sharedApplication] statusBarOrientation];
  1340. return UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad
  1341. || toInterfaceOrientation != UIInterfaceOrientationPortraitUpsideDown;
  1342. }
  1343. - (void)willRotateToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation duration:(NSTimeInterval)duration
  1344. {
  1345. [super willRotateToInterfaceOrientation:toInterfaceOrientation duration:duration];
  1346. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  1347. if (self.artworkImageView.image)
  1348. self.trackNameLabel.hidden = UIInterfaceOrientationIsLandscape(toInterfaceOrientation);
  1349. }
  1350. }
  1351. #pragma mark - AVSession delegate
  1352. - (void)beginInterruption
  1353. {
  1354. if ([_mediaPlayer isPlaying]) {
  1355. [_mediaPlayer pause];
  1356. _shouldResumePlaying = YES;
  1357. }
  1358. }
  1359. - (void)endInterruption
  1360. {
  1361. if (_shouldResumePlaying) {
  1362. [_mediaPlayer play];
  1363. _shouldResumePlaying = NO;
  1364. }
  1365. }
  1366. #pragma mark - External Display
  1367. - (BOOL)hasExternalDisplay
  1368. {
  1369. return ([[UIScreen screens] count] > 1);
  1370. }
  1371. - (void)showOnExternalDisplay
  1372. {
  1373. UIScreen *screen = [UIScreen screens][1];
  1374. screen.overscanCompensation = UIScreenOverscanCompensationInsetApplicationFrame;
  1375. self.externalWindow = [[UIWindow alloc] initWithFrame:screen.bounds];
  1376. UIViewController *controller = [[VLCExternalDisplayController alloc] init];
  1377. self.externalWindow.rootViewController = controller;
  1378. [controller.view addSubview:_movieView];
  1379. controller.view.frame = screen.bounds;
  1380. _movieView.frame = screen.bounds;
  1381. self.playingExternallyView.hidden = NO;
  1382. self.externalWindow.screen = screen;
  1383. self.externalWindow.hidden = NO;
  1384. }
  1385. - (void)hideFromExternalDisplay
  1386. {
  1387. [self.view addSubview:_movieView];
  1388. [self.view sendSubviewToBack:_movieView];
  1389. _movieView.frame = self.view.frame;
  1390. self.playingExternallyView.hidden = YES;
  1391. self.externalWindow.hidden = YES;
  1392. self.externalWindow = nil;
  1393. }
  1394. - (void)handleExternalScreenDidConnect:(NSNotification *)notification
  1395. {
  1396. [self showOnExternalDisplay];
  1397. }
  1398. - (void)handleExternalScreenDidDisconnect:(NSNotification *)notification
  1399. {
  1400. [self hideFromExternalDisplay];
  1401. }
  1402. @end