VLCMovieViewController.m 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978
  1. //
  2. // VLCMovieViewController.m
  3. // AspenProject
  4. //
  5. // Created by Felix Paul Kühne on 27.02.13.
  6. // Copyright (c) 2013 VideoLAN. All rights reserved.
  7. //
  8. // Refer to the COPYING file of the official project for license.
  9. //
  10. #import "VLCMovieViewController.h"
  11. #import "VLCExternalDisplayController.h"
  12. #import <AVFoundation/AVFoundation.h>
  13. #import <CommonCrypto/CommonDigest.h>
  14. #import "UIDevice+SpeedCategory.h"
  15. #import "VLCBugreporter.h"
  16. #import <MediaPlayer/MediaPlayer.h>
  17. #define INPUT_RATE_DEFAULT 1000.
  18. @interface VLCMovieViewController () <UIGestureRecognizerDelegate, AVAudioSessionDelegate>
  19. {
  20. VLCMediaPlayer *_mediaPlayer;
  21. BOOL _controlsHidden;
  22. BOOL _videoFiltersHidden;
  23. BOOL _playbackSpeedViewHidden;
  24. UIActionSheet *_subtitleActionSheet;
  25. UIActionSheet *_audiotrackActionSheet;
  26. float _currentPlaybackRate;
  27. NSArray *_aspectRatios;
  28. NSUInteger _currentAspectRatioMask;
  29. NSTimer *_idleTimer;
  30. BOOL _shouldResumePlaying;
  31. BOOL _viewAppeared;
  32. BOOL _displayRemainingTime;
  33. BOOL _positionSet;
  34. }
  35. @property (nonatomic, strong) UIPopoverController *masterPopoverController;
  36. @property (nonatomic, strong) UIWindow *externalWindow;
  37. @end
  38. @implementation VLCMovieViewController
  39. + (void)initialize
  40. {
  41. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  42. NSDictionary *appDefaults = @{kVLCShowRemainingTime : @(YES)};
  43. [defaults registerDefaults:appDefaults];
  44. }
  45. - (void)dealloc
  46. {
  47. [[NSNotificationCenter defaultCenter] removeObserver:self];
  48. }
  49. #pragma mark - Managing the media item
  50. - (void)setMediaItem:(id)newMediaItem
  51. {
  52. if (_mediaItem != newMediaItem)
  53. _mediaItem = newMediaItem;
  54. if (self.masterPopoverController != nil)
  55. [self.masterPopoverController dismissPopoverAnimated:YES];
  56. }
  57. - (void)viewDidLoad
  58. {
  59. [super viewDidLoad];
  60. self.wantsFullScreenLayout = YES;
  61. self.videoFilterView.hidden = YES;
  62. _videoFiltersHidden = YES;
  63. _hueLabel.text = NSLocalizedString(@"VFILTER_HUE", @"");
  64. _contrastLabel.text = NSLocalizedString(@"VFILTER_CONTRAST", @"");
  65. _brightnessLabel.text = NSLocalizedString(@"VFILTER_BRIGHTNESS", @"");
  66. _saturationLabel.text = NSLocalizedString(@"VFILTER_SATURATION", @"");
  67. _gammaLabel.text = NSLocalizedString(@"VFILTER_GAMMA", @"");
  68. _playbackSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SPEED", @"");
  69. _scrubHelpLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HELP", @"");
  70. self.playbackSpeedView.hidden = YES;
  71. _playbackSpeedViewHidden = YES;
  72. NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
  73. [center addObserver:self selector:@selector(handleExternalScreenDidConnect:)
  74. name:UIScreenDidConnectNotification object:nil];
  75. [center addObserver:self selector:@selector(handleExternalScreenDidDisconnect:)
  76. name:UIScreenDidDisconnectNotification object:nil];
  77. [center addObserver:self selector:@selector(applicationWillResignActive:)
  78. name:UIApplicationWillResignActiveNotification object:nil];
  79. [center addObserver:self selector:@selector(applicationDidBecomeActive:)
  80. name:UIApplicationDidBecomeActiveNotification object:nil];
  81. [center addObserver:self selector:@selector(applicationDidEnterBackground:)
  82. name:UIApplicationDidEnterBackgroundNotification object:nil];
  83. _playingExternallyTitle.text = NSLocalizedString(@"PLAYING_EXTERNALLY_TITLE", @"");
  84. _playingExternallyDescription.text = NSLocalizedString(@"PLAYING_EXTERNALLY_DESC", @"");
  85. if ([self hasExternalDisplay])
  86. [self showOnExternalDisplay];
  87. _movieView.userInteractionEnabled = NO;
  88. UITapGestureRecognizer *tapOnVideoRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(toggleControlsVisible)];
  89. tapOnVideoRecognizer.delegate = self;
  90. [self.view addGestureRecognizer:tapOnVideoRecognizer];
  91. _displayRemainingTime = [[[NSUserDefaults standardUserDefaults] objectForKey:kVLCShowRemainingTime] boolValue];
  92. UIPinchGestureRecognizer *pinchRecognizer = [[UIPinchGestureRecognizer alloc] initWithTarget:self action:@selector(handlePinchGesture:)];
  93. pinchRecognizer.delegate = self;
  94. [self.view addGestureRecognizer:pinchRecognizer];
  95. #if 0 // FIXME: trac #8742
  96. UISwipeGestureRecognizer *leftSwipeRecognizer = [[VLCHorizontalSwipeGestureRecognizer alloc] initWithTarget:self action:nil];
  97. leftSwipeRecognizer.direction = UISwipeGestureRecognizerDirectionLeft;
  98. leftSwipeRecognizer.delegate = self;
  99. [self.view addGestureRecognizer:leftSwipeRecognizer];
  100. UISwipeGestureRecognizer *rightSwipeRecognizer = [[VLCHorizontalSwipeGestureRecognizer alloc] initWithTarget:self action:nil];
  101. rightSwipeRecognizer.direction = UISwipeGestureRecognizerDirectionRight;
  102. rightSwipeRecognizer.delegate = self;
  103. [self.view addGestureRecognizer:rightSwipeRecognizer];
  104. UISwipeGestureRecognizer *upSwipeRecognizer = [[VLCVerticalSwipeGestureRecognizer alloc] initWithTarget:self action:nil];
  105. upSwipeRecognizer.direction = UISwipeGestureRecognizerDirectionUp;
  106. upSwipeRecognizer.delegate = self;
  107. [self.view addGestureRecognizer:upSwipeRecognizer];
  108. UISwipeGestureRecognizer *downSwipeRecognizer = [[VLCVerticalSwipeGestureRecognizer alloc] initWithTarget:self action:nil];
  109. downSwipeRecognizer.direction = UISwipeGestureRecognizerDirectionDown;
  110. downSwipeRecognizer.delegate = self;
  111. [self.view addGestureRecognizer:downSwipeRecognizer];
  112. #endif
  113. _aspectRatios = @[@"DEFAULT", @"4:3", @"16:9", @"16:10", @"2.21:1", @"FILL_TO_SCREEN"];
  114. [self.aspectRatioButton setBackgroundImage:[UIImage imageNamed:@"ratioButton"] forState:UIControlStateNormal];
  115. [self.aspectRatioButton setBackgroundImage:[UIImage imageNamed:@"ratioButtonHighlight"] forState:UIControlStateHighlighted];
  116. [self.aspectRatioButton setImage:[UIImage imageNamed:@"ratioIcon"] forState:UIControlStateNormal];
  117. [self.toolbar setBackgroundImage:[UIImage imageNamed:@"seekbarBg"] forBarMetrics:UIBarMetricsDefault];
  118. [self.backButton setBackgroundImage:[UIImage imageNamed:@"playbackDoneButton"] forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
  119. [self.backButton setBackgroundImage:[UIImage imageNamed:@"playbackDoneButtonHighlight"] forState:UIControlStateHighlighted barMetrics:UIBarMetricsDefault];
  120. /* this looks a bit weird, but we need to support iOS 5 and should show the same appearance */
  121. UISlider *volumeSlider = nil;
  122. for (id aView in self.volumeView.subviews){
  123. if ([[[aView class] description] isEqualToString:@"MPVolumeSlider"]){
  124. volumeSlider = (UISlider *)aView;
  125. break;
  126. }
  127. }
  128. [volumeSlider setMinimumTrackImage:[[UIImage imageNamed:@"sliderminiValue"]resizableImageWithCapInsets:UIEdgeInsetsMake(0, 4, 0, 0)] forState:UIControlStateNormal];
  129. [volumeSlider setMaximumTrackImage:[[UIImage imageNamed:@"slidermaxValue"] resizableImageWithCapInsets:UIEdgeInsetsMake(0, 0, 0, 4)] forState:UIControlStateNormal];
  130. [volumeSlider setThumbImage:[UIImage imageNamed:@"volumeballslider"] forState:UIControlStateNormal];
  131. [volumeSlider addTarget:self
  132. action:@selector(volumeSliderAction:)
  133. forControlEvents:UIControlEventValueChanged];
  134. [[AVAudioSession sharedInstance] setDelegate:self];
  135. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad)
  136. self.positionSlider.scrubbingSpeedChangePositions = @[@(0.), @(100.), @(200.), @(300)];
  137. }
  138. - (BOOL)_blobCheck
  139. {
  140. NSArray *searchPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
  141. NSString *directoryPath = searchPaths[0];
  142. if (![[NSFileManager defaultManager] fileExistsAtPath:[directoryPath stringByAppendingPathComponent:@"blob.bin"]])
  143. return NO;
  144. NSData *data = [NSData dataWithContentsOfFile:[directoryPath stringByAppendingPathComponent:@"blob.bin"]];
  145. uint8_t digest[CC_SHA1_DIGEST_LENGTH];
  146. CC_SHA1(data.bytes, data.length, digest);
  147. NSMutableString *hash = [NSMutableString stringWithCapacity:CC_SHA1_DIGEST_LENGTH * 2];
  148. for (unsigned int u = 0; u < CC_SHA1_DIGEST_LENGTH; u++)
  149. [hash appendFormat:@"%02x", digest[u]];
  150. if ([hash isEqualToString:kBlobHash])
  151. return YES;
  152. else
  153. return NO;
  154. }
  155. - (void)viewWillAppear:(BOOL)animated
  156. {
  157. _mediaPlayer = [[VLCMediaPlayer alloc] init];
  158. [_mediaPlayer setDelegate:self];
  159. [_mediaPlayer setDrawable:self.movieView];
  160. [self.navigationController setNavigationBarHidden:YES animated:YES];
  161. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone)
  162. [UIApplication sharedApplication].statusBarStyle = UIStatusBarStyleBlackTranslucent;
  163. if (!self.mediaItem && !self.url)
  164. return;
  165. VLCMedia *media;
  166. if (self.mediaItem) {
  167. self.title = [self.mediaItem title];
  168. media = [VLCMedia mediaWithURL:[NSURL URLWithString:self.mediaItem.url]];
  169. self.mediaItem.unread = @(NO);
  170. } else {
  171. media = [VLCMedia mediaWithURL:self.url];
  172. self.title = @"Network Stream";
  173. }
  174. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  175. [media addOptions:
  176. @{kVLCSettingStretchAudio :
  177. [[defaults objectForKey:kVLCSettingStretchAudio] boolValue] ? kVLCSettingStretchAudioOnValue : kVLCSettingStretchAudioOffValue, kVLCSettingTextEncoding : [defaults objectForKey:kVLCSettingTextEncoding], kVLCSettingSkipLoopFilter : [defaults objectForKey:kVLCSettingSkipLoopFilter]}];
  178. [NSTimeZone resetSystemTimeZone];
  179. NSString *tzName = [[NSTimeZone systemTimeZone] name];
  180. 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"];
  181. if ([tzNames containsObject:tzName] || [[tzName stringByDeletingLastPathComponent] isEqualToString:@"US"]) {
  182. NSArray *tracksInfo = media.tracksInformation;
  183. for (NSUInteger x = 0; x < tracksInfo.count; x++) {
  184. if ([[tracksInfo[x] objectForKey:VLCMediaTracksInformationType] isEqualToString:VLCMediaTracksInformationTypeAudio])
  185. {
  186. NSInteger fourcc = [[tracksInfo[x] objectForKey:VLCMediaTracksInformationCodec] integerValue];
  187. switch (fourcc) {
  188. case 540161377:
  189. case 1647457633:
  190. case 858612577:
  191. case 862151027:
  192. case 2126701:
  193. case 544437348:
  194. case 542331972:
  195. case 1651733604:
  196. case 1668510820:
  197. case 1702065252:
  198. case 1752396900:
  199. case 1819505764:
  200. case 18903917:
  201. case 862151013:
  202. {
  203. if (![self _blobCheck]) {
  204. [media addOptions:@{@"no-audio" : [NSNull null]}];
  205. APLog(@"audio playback disabled because an unsupported codec was found");
  206. }
  207. break;
  208. }
  209. default:
  210. break;
  211. }
  212. }
  213. }
  214. }
  215. [_mediaPlayer setMedia:media];
  216. self.positionSlider.value = 0.;
  217. [self.timeDisplay setTitle:@"" forState:UIControlStateNormal];
  218. [super viewWillAppear:animated];
  219. if (![self _isMediaSuitableForDevice]) {
  220. UIAlertView * alert = [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"DEVICE_TOOSLOW_TITLE", @"") message:[NSString stringWithFormat:NSLocalizedString(@"DEVICE_TOOSLOW", @""), [[UIDevice currentDevice] model], self.mediaItem.title] delegate:self cancelButtonTitle:NSLocalizedString(@"BUTTON_CANCEL", @"") otherButtonTitles:NSLocalizedString(@"BUTTON_OPEN", @""), nil];
  221. [alert show];
  222. } else
  223. [self _playNewMedia];
  224. if (![self hasExternalDisplay])
  225. self.brightnessSlider.value = [UIScreen mainScreen].brightness * 2.;
  226. [self setControlsHidden:NO animated:YES];
  227. _viewAppeared = YES;
  228. }
  229. - (BOOL)_isMediaSuitableForDevice
  230. {
  231. if (!self.mediaItem)
  232. return YES;
  233. NSUInteger totalNumberOfPixels = [[[self.mediaItem videoTrack] valueForKey:@"width"] doubleValue] * [[[self.mediaItem videoTrack] valueForKey:@"height"] doubleValue];
  234. NSInteger speedCategory = [[UIDevice currentDevice] speedCategory];
  235. if (speedCategory == 1) {
  236. // iPhone 3GS, iPhone 4, first gen. iPad, 3rd and 4th generation iPod touch
  237. return (totalNumberOfPixels < 600000); // between 480p and 720p
  238. } else if (speedCategory == 2) {
  239. // iPhone 4S, iPad 2 and 3, iPod 4 and 5
  240. return (totalNumberOfPixels < 922000); // 720p
  241. } else if (speedCategory == 3) {
  242. // iPhone 5, iPad 4
  243. return (totalNumberOfPixels < 2074000); // 1080p
  244. }
  245. return YES;
  246. }
  247. - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
  248. {
  249. if (buttonIndex == 1)
  250. [self _playNewMedia];
  251. else
  252. [self closePlayback:nil];
  253. }
  254. - (void)_playNewMedia
  255. {
  256. NSNumber *playbackPositionInTime = @(0);
  257. if (self.mediaItem.lastPosition && [self.mediaItem.lastPosition floatValue] < .95) {
  258. if (self.mediaItem.duration.intValue != 0)
  259. playbackPositionInTime = @(self.mediaItem.lastPosition.floatValue * (self.mediaItem.duration.intValue / 1000.));
  260. }
  261. [_mediaPlayer.media addOptions:@{@"start-time": playbackPositionInTime}];
  262. APLog(@"set starttime to %i", playbackPositionInTime.intValue);
  263. [_mediaPlayer play];
  264. if (self.mediaItem.lastAudioTrack.intValue > 0)
  265. _mediaPlayer.currentAudioTrackIndex = self.mediaItem.lastAudioTrack.intValue;
  266. if (self.mediaItem.lastSubtitleTrack.intValue > 0)
  267. _mediaPlayer.currentVideoSubTitleIndex = self.mediaItem.lastSubtitleTrack.intValue;
  268. self.playbackSpeedSlider.value = [self _playbackSpeed];
  269. [self _updatePlaybackSpeedIndicator];
  270. [self performSelectorInBackground:@selector(_updateExportedPlaybackInformation) withObject:nil];
  271. _currentAspectRatioMask = 0;
  272. _mediaPlayer.videoAspectRatio = NULL;
  273. [self _resetIdleTimer];
  274. }
  275. - (void)viewWillDisappear:(BOOL)animated
  276. {
  277. _viewAppeared = NO;
  278. if (_idleTimer) {
  279. [_idleTimer invalidate];
  280. _idleTimer = nil;
  281. }
  282. [self.navigationController setNavigationBarHidden:NO animated:YES];
  283. [UIApplication sharedApplication].statusBarStyle = UIStatusBarStyleBlackOpaque;
  284. [[UIApplication sharedApplication] setStatusBarHidden:NO withAnimation:UIStatusBarAnimationFade];
  285. [_mediaPlayer pause];
  286. [super viewWillDisappear:animated];
  287. [self _saveCurrentState];
  288. [_mediaPlayer stop];
  289. _mediaPlayer = nil; // save memory and some CPU time
  290. // hide filter UI for next run
  291. if (!_videoFiltersHidden)
  292. _videoFiltersHidden = YES;
  293. if (!_playbackSpeedViewHidden)
  294. _playbackSpeedViewHidden = YES;
  295. }
  296. - (void)_saveCurrentState
  297. {
  298. if (self.mediaItem) {
  299. self.mediaItem.lastPosition = @([_mediaPlayer position]);
  300. self.mediaItem.lastAudioTrack = @(_mediaPlayer.currentAudioTrackIndex);
  301. self.mediaItem.lastSubtitleTrack = @(_mediaPlayer.currentVideoSubTitleIndex);
  302. }
  303. }
  304. #pragma mark - remote events
  305. - (void)viewDidAppear:(BOOL)animated
  306. {
  307. [super viewDidAppear:animated];
  308. [[UIApplication sharedApplication] beginReceivingRemoteControlEvents];
  309. [self becomeFirstResponder];
  310. }
  311. - (void)viewDidDisappear:(BOOL)animated
  312. {
  313. [super viewDidDisappear:animated];
  314. [[UIApplication sharedApplication] endReceivingRemoteControlEvents];
  315. [self resignFirstResponder];
  316. [[NSUserDefaults standardUserDefaults] setBool:_displayRemainingTime forKey:kVLCShowRemainingTime];
  317. }
  318. - (BOOL)canBecomeFirstResponder
  319. {
  320. return YES;
  321. }
  322. - (void)motionEnded:(UIEventSubtype)motion withEvent:(UIEvent *)event
  323. {
  324. if (motion == UIEventSubtypeMotionShake)
  325. [[VLCBugreporter sharedInstance] handleBugreportRequest];
  326. }
  327. - (void)remoteControlReceivedWithEvent:(UIEvent *)event
  328. {
  329. switch (event.subtype) {
  330. case UIEventSubtypeRemoteControlPlay:
  331. [_mediaPlayer play];
  332. break;
  333. case UIEventSubtypeRemoteControlPause:
  334. [_mediaPlayer pause];
  335. break;
  336. case UIEventSubtypeRemoteControlTogglePlayPause:
  337. [self playPause];
  338. break;
  339. default:
  340. break;
  341. }
  342. }
  343. #pragma mark - controls visibility
  344. - (void)handlePinchGesture:(UIPinchGestureRecognizer *)recognizer
  345. {
  346. if (recognizer.velocity < 0.)
  347. [self closePlayback:nil];
  348. }
  349. - (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch
  350. {
  351. if (touch.view != self.view)
  352. return NO;
  353. return YES;
  354. }
  355. - (void)setControlsHidden:(BOOL)hidden animated:(BOOL)animated
  356. {
  357. _controlsHidden = hidden;
  358. CGFloat alpha = _controlsHidden? 0.0f: 1.0f;
  359. if (!_controlsHidden) {
  360. _controllerPanel.alpha = 0.0f;
  361. _controllerPanel.hidden = !_videoFiltersHidden;
  362. _toolbar.alpha = 0.0f;
  363. _toolbar.hidden = NO;
  364. _videoFilterView.alpha = 0.0f;
  365. _videoFilterView.hidden = _videoFiltersHidden;
  366. _playbackSpeedView.alpha = 0.0f;
  367. _playbackSpeedView.hidden = _playbackSpeedViewHidden;
  368. }
  369. void (^animationBlock)() = ^() {
  370. _controllerPanel.alpha = alpha;
  371. _toolbar.alpha = alpha;
  372. _videoFilterView.alpha = alpha;
  373. _playbackSpeedView.alpha = alpha;
  374. };
  375. void (^completionBlock)(BOOL finished) = ^(BOOL finished) {
  376. if (_videoFiltersHidden)
  377. _controllerPanel.hidden = _controlsHidden;
  378. else
  379. _controllerPanel.hidden = NO;
  380. _toolbar.hidden = _controlsHidden;
  381. _videoFilterView.hidden = _videoFiltersHidden;
  382. _playbackSpeedView.hidden = _playbackSpeedViewHidden;
  383. };
  384. UIStatusBarAnimation animationType = animated? UIStatusBarAnimationFade: UIStatusBarAnimationNone;
  385. NSTimeInterval animationDuration = animated? 0.3: 0.0;
  386. [[UIApplication sharedApplication] setStatusBarHidden:_viewAppeared ? _controlsHidden : NO withAnimation:animationType];
  387. [UIView animateWithDuration:animationDuration animations:animationBlock completion:completionBlock];
  388. _volumeView.hidden = _controllerPanel.hidden;
  389. }
  390. - (void)toggleControlsVisible
  391. {
  392. if (_controlsHidden && !_videoFiltersHidden)
  393. _videoFiltersHidden = YES;
  394. [self setControlsHidden:!_controlsHidden animated:YES];
  395. }
  396. - (void)_resetIdleTimer
  397. {
  398. if (!_idleTimer)
  399. _idleTimer = [NSTimer scheduledTimerWithTimeInterval:4.
  400. target:self
  401. selector:@selector(idleTimerExceeded)
  402. userInfo:nil
  403. repeats:NO];
  404. else {
  405. if (fabs([_idleTimer.fireDate timeIntervalSinceNow]) < 4.)
  406. [_idleTimer setFireDate:[NSDate dateWithTimeIntervalSinceNow:4.]];
  407. }
  408. }
  409. - (void)idleTimerExceeded
  410. {
  411. _idleTimer = nil;
  412. if (!_controlsHidden)
  413. [self toggleControlsVisible];
  414. if (!_videoFiltersHidden)
  415. _videoFiltersHidden = YES;
  416. if (!_playbackSpeedViewHidden)
  417. _playbackSpeedViewHidden = YES;
  418. if (self.scrubIndicatorView.hidden == NO)
  419. self.scrubIndicatorView.hidden = YES;
  420. }
  421. - (UIResponder *)nextResponder
  422. {
  423. [self _resetIdleTimer];
  424. return [super nextResponder];
  425. }
  426. #pragma mark - controls
  427. - (IBAction)closePlayback:(id)sender
  428. {
  429. [self setControlsHidden:NO animated:NO];
  430. [self.navigationController popViewControllerAnimated:YES];
  431. }
  432. - (IBAction)positionSliderAction:(UISlider *)sender
  433. {
  434. /* we need to limit the number of events sent by the slider, since otherwise, the user
  435. * wouldn't see the I-frames when seeking on current mobile devices. This isn't a problem
  436. * within the Simulator, but especially on older ARMv7 devices, it's clearly noticeable. */
  437. [self performSelector:@selector(_setPositionForReal) withObject:nil afterDelay:0.3];
  438. VLCTime *newPosition = [VLCTime timeWithInt:(int)(_positionSlider.value * self.mediaItem.duration.intValue)];
  439. [self.timeDisplay setTitle:newPosition.stringValue forState:UIControlStateNormal];
  440. _positionSet = NO;
  441. [self _resetIdleTimer];
  442. }
  443. - (void)_setPositionForReal
  444. {
  445. if (!_positionSet) {
  446. _mediaPlayer.position = _positionSlider.value;
  447. _positionSet = YES;
  448. }
  449. }
  450. - (IBAction)positionSliderTouchDown:(id)sender
  451. {
  452. [self _updateScrubLabel];
  453. self.scrubIndicatorView.hidden = NO;
  454. }
  455. - (IBAction)positionSliderTouchUp:(id)sender
  456. {
  457. self.scrubIndicatorView.hidden = YES;
  458. }
  459. - (void)_updateScrubLabel
  460. {
  461. float speed = self.positionSlider.scrubbingSpeed;
  462. if (speed == 1.)
  463. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HIGH", @"");
  464. else if (speed == .5)
  465. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_HALF", @"");
  466. else if (speed == .25)
  467. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_QUARTER", @"");
  468. else
  469. self.currentScrubSpeedLabel.text = NSLocalizedString(@"PLAYBACK_SCRUB_FINE", @"");
  470. [self _resetIdleTimer];
  471. }
  472. - (IBAction)positionSliderDrag:(id)sender
  473. {
  474. [self _updateScrubLabel];
  475. }
  476. - (IBAction)volumeSliderAction:(id)sender
  477. {
  478. [self _resetIdleTimer];
  479. }
  480. - (void)mediaPlayerTimeChanged:(NSNotification *)aNotification {
  481. self.positionSlider.value = [_mediaPlayer position];
  482. if (_displayRemainingTime)
  483. [self.timeDisplay setTitle:[[_mediaPlayer remainingTime] stringValue] forState:UIControlStateNormal];
  484. else
  485. [self.timeDisplay setTitle:[[_mediaPlayer time] stringValue] forState:UIControlStateNormal];
  486. }
  487. - (void)mediaPlayerStateChanged:(NSNotification *)aNotification
  488. {
  489. VLCMediaPlayerState currentState = _mediaPlayer.state;
  490. if (currentState == VLCMediaPlayerStateError) {
  491. [self.statusLabel showStatusMessage:NSLocalizedString(@"PLAYBACK_FAILED", @"")];
  492. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  493. }
  494. if (currentState == VLCMediaPlayerStateEnded || currentState == VLCMediaPlayerStateStopped)
  495. [self performSelector:@selector(closePlayback:) withObject:nil afterDelay:2.];
  496. UIImage *playPauseImage = [_mediaPlayer isPlaying]? [UIImage imageNamed:@"pauseIcon"] : [UIImage imageNamed:@"playIcon"];
  497. [_playPauseButton setImage:playPauseImage forState:UIControlStateNormal];
  498. if ([[_mediaPlayer audioTrackIndexes] count] > 2)
  499. self.audioSwitcherButton.hidden = NO;
  500. else
  501. self.audioSwitcherButton.hidden = YES;
  502. if ([[_mediaPlayer videoSubTitlesIndexes] count] > 1)
  503. self.subtitleSwitcherButton.hidden = NO;
  504. else
  505. self.subtitleSwitcherButton.hidden = YES;
  506. }
  507. - (IBAction)playPause
  508. {
  509. if ([_mediaPlayer isPlaying])
  510. [_mediaPlayer pause];
  511. else
  512. [_mediaPlayer play];
  513. }
  514. - (IBAction)forward:(id)sender
  515. {
  516. [_mediaPlayer mediumJumpForward];
  517. }
  518. - (IBAction)backward:(id)sender
  519. {
  520. [_mediaPlayer mediumJumpBackward];
  521. }
  522. - (IBAction)switchAudioTrack:(id)sender
  523. {
  524. _audiotrackActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_AUDIO_TRACK", @"audio track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  525. NSArray *audioTracks = [_mediaPlayer audioTrackNames];
  526. NSArray *audioTrackIndexes = [_mediaPlayer audioTrackIndexes];
  527. NSUInteger count = [audioTracks count];
  528. for (NSUInteger i = 0; i < count; i++) {
  529. NSString *indexIndicator = ([audioTrackIndexes[i] intValue] == [_mediaPlayer currentAudioTrackIndex])? @"\u2713": @"";
  530. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, audioTracks[i]];
  531. [_audiotrackActionSheet addButtonWithTitle:buttonTitle];
  532. }
  533. [_audiotrackActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  534. [_audiotrackActionSheet setCancelButtonIndex:[_audiotrackActionSheet numberOfButtons] - 1];
  535. [_audiotrackActionSheet showInView:self.audioSwitcherButton];
  536. }
  537. - (IBAction)switchSubtitleTrack:(id)sender
  538. {
  539. NSArray *spuTracks = [_mediaPlayer videoSubTitlesNames];
  540. NSArray *spuTrackIndexes = [_mediaPlayer videoSubTitlesIndexes];
  541. NSUInteger count = [spuTracks count];
  542. if (count <= 1)
  543. return;
  544. _subtitleActionSheet = [[UIActionSheet alloc] initWithTitle:NSLocalizedString(@"CHOOSE_SUBTITLE_TRACK", @"subtitle track selector") delegate:self cancelButtonTitle:nil destructiveButtonTitle:nil otherButtonTitles: nil];
  545. for (NSUInteger i = 0; i < count; i++) {
  546. NSString *indexIndicator = ([spuTrackIndexes[i] intValue] == [_mediaPlayer currentVideoSubTitleIndex])? @"\u2713": @"";
  547. NSString *buttonTitle = [NSString stringWithFormat:@"%@ %@", indexIndicator, spuTracks[i]];
  548. [_subtitleActionSheet addButtonWithTitle:buttonTitle];
  549. }
  550. [_subtitleActionSheet addButtonWithTitle:NSLocalizedString(@"BUTTON_CANCEL", @"cancel button")];
  551. [_subtitleActionSheet setCancelButtonIndex:[_subtitleActionSheet numberOfButtons] - 1];
  552. [_subtitleActionSheet showInView: self.subtitleSwitcherButton];
  553. }
  554. - (void)actionSheet:(UIActionSheet *)actionSheet clickedButtonAtIndex:(NSInteger)buttonIndex {
  555. if (buttonIndex == [actionSheet cancelButtonIndex])
  556. return;
  557. NSArray *indexArray;
  558. if (actionSheet == _subtitleActionSheet) {
  559. indexArray = _mediaPlayer.videoSubTitlesIndexes;
  560. if (buttonIndex <= indexArray.count) {
  561. _mediaPlayer.currentVideoSubTitleIndex = [indexArray[buttonIndex] intValue];
  562. }
  563. } else if (actionSheet == _audiotrackActionSheet) {
  564. indexArray = _mediaPlayer.audioTrackIndexes;
  565. if (buttonIndex <= indexArray.count) {
  566. _mediaPlayer.currentAudioTrackIndex = [indexArray[buttonIndex] intValue];
  567. }
  568. }
  569. }
  570. - (IBAction)toggleTimeDisplay:(id)sender
  571. {
  572. _displayRemainingTime = !_displayRemainingTime;
  573. [self _resetIdleTimer];
  574. }
  575. #pragma mark - swipe gestures
  576. - (void)horizontalSwipePercentage:(CGFloat)percentage inView:(UIView *)view
  577. {
  578. if (percentage != 0.) {
  579. _mediaPlayer.position = _mediaPlayer.position + percentage;
  580. }
  581. }
  582. - (void)verticalSwipePercentage:(CGFloat)percentage inView:(UIView *)view half:(NSUInteger)half
  583. {
  584. if (percentage != 0.) {
  585. if (half > 0) {
  586. CGFloat currentValue = self.brightnessSlider.value;
  587. currentValue = currentValue + percentage;
  588. self.brightnessSlider.value = currentValue;
  589. if ([self hasExternalDisplay])
  590. _mediaPlayer.brightness = currentValue;
  591. else
  592. [[UIScreen mainScreen] setBrightness:currentValue / 2];
  593. } else
  594. NSLog(@"volume setting through swipe not implemented");//_mediaPlayer.audio.volume = percentage * 200;
  595. }
  596. }
  597. #pragma mark - Video Filter UI
  598. - (IBAction)videoFilterToggle:(id)sender
  599. {
  600. if (!_playbackSpeedViewHidden)
  601. self.playbackSpeedView.hidden = _playbackSpeedViewHidden = YES;
  602. if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
  603. if (!_controlsHidden)
  604. self.controllerPanel.hidden = _controlsHidden = YES;
  605. }
  606. self.videoFilterView.hidden = !_videoFiltersHidden;
  607. _videoFiltersHidden = self.videoFilterView.hidden;
  608. }
  609. - (IBAction)videoFilterSliderAction:(id)sender
  610. {
  611. if (sender == self.hueSlider)
  612. _mediaPlayer.hue = (int)self.hueSlider.value;
  613. else if (sender == self.contrastSlider)
  614. _mediaPlayer.contrast = self.contrastSlider.value;
  615. else if (sender == self.brightnessSlider) {
  616. if ([self hasExternalDisplay])
  617. _mediaPlayer.brightness = self.brightnessSlider.value;
  618. else
  619. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  620. } else if (sender == self.saturationSlider)
  621. _mediaPlayer.saturation = self.saturationSlider.value;
  622. else if (sender == self.gammaSlider)
  623. _mediaPlayer.gamma = self.gammaSlider.value;
  624. else if (sender == self.resetVideoFilterButton) {
  625. _mediaPlayer.hue = self.hueSlider.value = 0.;
  626. _mediaPlayer.contrast = self.contrastSlider.value = 1.;
  627. _mediaPlayer.brightness = self.brightnessSlider.value = 1.;
  628. [[UIScreen mainScreen] setBrightness:(self.brightnessSlider.value / 2.)];
  629. _mediaPlayer.saturation = self.saturationSlider.value = 1.;
  630. _mediaPlayer.gamma = self.gammaSlider.value = 1.;
  631. } else
  632. APLog(@"unknown sender for videoFilterSliderAction");
  633. [self _resetIdleTimer];
  634. }
  635. #pragma mark - playback view
  636. - (IBAction)playbackSpeedSliderAction:(UISlider *)sender
  637. {
  638. double speed = pow(2, sender.value / 17.);
  639. float rate = INPUT_RATE_DEFAULT / speed;
  640. if (_currentPlaybackRate != rate)
  641. [_mediaPlayer setRate:INPUT_RATE_DEFAULT / rate];
  642. _currentPlaybackRate = rate;
  643. [self _updatePlaybackSpeedIndicator];
  644. [self _resetIdleTimer];
  645. }
  646. - (void)_updatePlaybackSpeedIndicator
  647. {
  648. float f_value = self.playbackSpeedSlider.value;
  649. double speed = pow(2, f_value / 17.);
  650. self.playbackSpeedIndicator.text = [NSString stringWithFormat:@"%.2fx", speed];
  651. /* rate changed, so update the exported info */
  652. [self performSelectorInBackground:@selector(_updateExportedPlaybackInformation) withObject:nil];
  653. }
  654. - (float)_playbackSpeed
  655. {
  656. float f_rate = _mediaPlayer.rate;
  657. double value = 17 * log(f_rate) / log(2.);
  658. float returnValue = (int) ((value > 0) ? value + .5 : value - .5);
  659. if (returnValue < -34.)
  660. returnValue = -34.;
  661. else if (returnValue > 34.)
  662. returnValue = 34.;
  663. _currentPlaybackRate = returnValue;
  664. return returnValue;
  665. }
  666. - (IBAction)videoDimensionAction:(id)sender
  667. {
  668. if (sender == self.playbackSpeedButton) {
  669. if (!_videoFiltersHidden)
  670. self.videoFilterView.hidden = _videoFiltersHidden = YES;
  671. self.playbackSpeedView.hidden = !_playbackSpeedViewHidden;
  672. _playbackSpeedViewHidden = self.playbackSpeedView.hidden;
  673. [self _resetIdleTimer];
  674. } else if (sender == self.aspectRatioButton) {
  675. NSUInteger count = [_aspectRatios count];
  676. if (_currentAspectRatioMask + 1 > count - 1) {
  677. _mediaPlayer.videoAspectRatio = NULL;
  678. _mediaPlayer.videoCropGeometry = NULL;
  679. _currentAspectRatioMask = 0;
  680. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", @""), NSLocalizedString(@"DEFAULT", @"")]];
  681. } else {
  682. _currentAspectRatioMask++;
  683. if ([_aspectRatios[_currentAspectRatioMask] isEqualToString:@"FILL_TO_SCREEN"]) {
  684. UIScreen *screen;
  685. if (![self hasExternalDisplay])
  686. screen = [UIScreen mainScreen];
  687. else
  688. screen = [UIScreen screens][1];
  689. float f_ar = screen.bounds.size.width / screen.bounds.size.height;
  690. if (f_ar == (float)(640./1136.)) // iPhone 5 aka 16:9.01
  691. _mediaPlayer.videoCropGeometry = "16:9";
  692. else if (f_ar == (float)(2./3.)) // all other iPhones
  693. _mediaPlayer.videoCropGeometry = "16:10"; // libvlc doesn't support 2:3 crop
  694. else if (f_ar == .75) // all iPads
  695. _mediaPlayer.videoCropGeometry = "4:3";
  696. else if (f_ar == .5625) // AirPlay
  697. _mediaPlayer.videoCropGeometry = "16:9";
  698. else
  699. APLog(@"unknown screen format %f, can't crop", f_ar);
  700. [self.statusLabel showStatusMessage:NSLocalizedString(@"FILL_TO_SCREEN", @"")];
  701. return;
  702. }
  703. _mediaPlayer.videoCropGeometry = NULL;
  704. _mediaPlayer.videoAspectRatio = (char *)[_aspectRatios[_currentAspectRatioMask] UTF8String];
  705. [self.statusLabel showStatusMessage:[NSString stringWithFormat:NSLocalizedString(@"AR_CHANGED", @""), _aspectRatios[_currentAspectRatioMask]]];
  706. }
  707. }
  708. }
  709. #pragma mark - background interaction
  710. - (void)applicationWillResignActive:(NSNotification *)aNotification
  711. {
  712. [self _saveCurrentState];
  713. if (![[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingContinueAudioInBackgroundKey] boolValue]) {
  714. [_mediaPlayer pause];
  715. _shouldResumePlaying = YES;
  716. } else
  717. _mediaPlayer.currentVideoTrackIndex = 0;
  718. }
  719. - (void)applicationDidEnterBackground:(NSNotification *)notification
  720. {
  721. _shouldResumePlaying = NO;
  722. }
  723. - (void)applicationDidBecomeActive:(NSNotification *)notification
  724. {
  725. if (_shouldResumePlaying) {
  726. _shouldResumePlaying = NO;
  727. [_mediaPlayer play];
  728. } else
  729. _mediaPlayer.currentVideoTrackIndex = 1;
  730. }
  731. - (void)_updateExportedPlaybackInformation
  732. {
  733. if (!_mediaItem) {
  734. [MPNowPlayingInfoCenter defaultCenter].nowPlayingInfo = nil;
  735. return;
  736. }
  737. MLFile * currentFile = _mediaItem;
  738. /* we omit artwork for now since we had to read it from storage as we can't access
  739. * the artwork cache at the moment - FIXME? */
  740. NSDictionary *currentlyPlayingTrackInfo = @{ MPMediaItemPropertyTitle : currentFile.title,
  741. MPMediaItemPropertyPlaybackDuration : @(currentFile.duration.intValue / 1000.),
  742. MPNowPlayingInfoPropertyElapsedPlaybackTime : @(_mediaPlayer.time.intValue / 1000.),
  743. MPNowPlayingInfoPropertyPlaybackRate : @(_mediaPlayer.rate) };
  744. [MPNowPlayingInfoCenter defaultCenter].nowPlayingInfo = currentlyPlayingTrackInfo;
  745. }
  746. #pragma mark - autorotation
  747. - (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation {
  748. return UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad
  749. || toInterfaceOrientation != UIInterfaceOrientationPortraitUpsideDown;
  750. }
  751. #pragma mark - AVSession delegate
  752. - (void)beginInterruption
  753. {
  754. if ([[[NSUserDefaults standardUserDefaults] objectForKey:kVLCSettingContinueAudioInBackgroundKey] boolValue])
  755. _shouldResumePlaying = YES;
  756. [_mediaPlayer pause];
  757. }
  758. - (void)endInterruption
  759. {
  760. if (_shouldResumePlaying) {
  761. [_mediaPlayer play];
  762. _shouldResumePlaying = NO;
  763. }
  764. }
  765. #pragma mark - External Display
  766. - (BOOL)hasExternalDisplay
  767. {
  768. return ([[UIScreen screens] count] > 1);
  769. }
  770. - (void)showOnExternalDisplay
  771. {
  772. UIScreen *screen = [UIScreen screens][1];
  773. screen.overscanCompensation = UIScreenOverscanCompensationInsetApplicationFrame;
  774. self.externalWindow = [[UIWindow alloc] initWithFrame:screen.bounds];
  775. UIViewController *controller = [[VLCExternalDisplayController alloc] init];
  776. self.externalWindow.rootViewController = controller;
  777. [controller.view addSubview:_movieView];
  778. controller.view.frame = screen.bounds;
  779. _movieView.frame = screen.bounds;
  780. self.playingExternallyView.hidden = NO;
  781. self.externalWindow.screen = screen;
  782. self.externalWindow.hidden = NO;
  783. }
  784. - (void)hideFromExternalDisplay
  785. {
  786. [self.view addSubview:_movieView];
  787. [self.view sendSubviewToBack:_movieView];
  788. _movieView.frame = self.view.frame;
  789. self.playingExternallyView.hidden = YES;
  790. self.externalWindow.hidden = YES;
  791. self.externalWindow = nil;
  792. }
  793. - (void)handleExternalScreenDidConnect:(NSNotification *)notification
  794. {
  795. [self showOnExternalDisplay];
  796. }
  797. - (void)handleExternalScreenDidDisconnect:(NSNotification *)notification
  798. {
  799. [self hideFromExternalDisplay];
  800. }
  801. @end