VLCMovieViewController.m 36 KB

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