If you have used the MPMoviePlayerViewController to play videos in iOS, chances are you have gotten a little frustrated by its rigidness. A couple of issues I personally encountered were:
- When presented modally, the view controller did not respect the modal transition style I had chosen for it.
- When the video finished playing, the view controller dismissed itself automatically. I wanted it to remain visible until the user pressed the Done button.
A couple of hours of browsing for the solution to no avail, I decided to try out sort of a hack, which turned out to work great.
The trick is to remove the MPMoviePlayerViewController instance from the MPMoviePlayerPlaybackDidFinishNotification notification observers, and instead handle it yourself. Seems like this view controller listens to this notification and auto dismisses itself when it’s fired.
In my application, I present a thumbnail of the video. When touched, the video is played.
- (void)playVideo:(NSString*)aVideoUrl { // Initialize the movie player view controller with a video URL string MPMoviePlayerViewController *playerVC = [[[MPMoviePlayerViewController alloc] initWithContentURL:[NSURL URLWithString:aVideoUrl]] autorelease]; // Remove the movie player view controller from the "playback did finish" notification observers [[NSNotificationCenter defaultCenter] removeObserver:playerVC name:MPMoviePlayerPlaybackDidFinishNotification object:playerVC.moviePlayer]; // Register this class as an observer instead [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(movieFinishedCallback:) name:MPMoviePlayerPlaybackDidFinishNotification object:playerVC.moviePlayer]; // Set the modal transition style of your choice playerVC.modalTransitionStyle = UIModalTransitionStyleCrossDissolve; // Present the movie player view controller [self presentModalViewController:playerVC animated:YES]; // Start playback [playerVC.moviePlayer prepareToPlay]; [playerVC.moviePlayer play]; }
In the code above, an instance of MPMoviePlayerViewController is created, and then removed from the observers of the MPMoviePlayerDidFinishNotification immediately; in its place, the current class is added as an observer, with a selector named movieFinishedCallback: (described below).
In addition, the modalTransitionStyle property of the view controller is set. The documentation suggests you to use the presentMoviePlayerViewControllerAnimated method to present it, in order to use the standard movie player transition. But if you want to use different transition, you must use the regular presentModalViewController method.
Back to the observer notification. Now that the view controller is no longer an observer, it won’t dismiss itself, ever, even if the Done button is pressed. So you must implement the code to dismiss it.
- (void)movieFinishedCallback:(NSNotification*)aNotification { // Obtain the reason why the movie playback finished NSNumber *finishReason = [[aNotification userInfo] objectForKey:MPMoviePlayerPlaybackDidFinishReasonUserInfoKey]; // Dismiss the view controller ONLY when the reason is not "playback ended" if ([finishReason intValue] != MPMovieFinishReasonPlaybackEnded) { MPMoviePlayerController *moviePlayer = [aNotification object]; // Remove this class from the observers [[NSNotificationCenter defaultCenter] removeObserver:self name:MPMoviePlayerPlaybackDidFinishNotification object:moviePlayer]; // Dismiss the view controller [self dismissModalViewControllerAnimated:YES]; } }
In my case, my original problem was that I wanted the video to stay visible until the user deliberately pressed the Done button. In the code above, the notification’s “playback finished reason” is examined: if the video playback ended normally, then the notification is ignored; if the video playback ended for another reason (i.e. an error or the user exited), then the view controller is dismissed.
As an added bonus, by dismissing the view controller using the dismissModalViewControllerAnimated: method, the modal transition style set earlier is respected.
The code is kinda messy, but I hope you get the idea. Perhaps it would be desirable to encapsulate this code by extending the MPMoviePlayerViewController class.
Outstanding!
This was very helpful. Thank you.
Thanks, it did really work. But now it broke this:
// Register for the “Done” button notification.
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(moviePlayerDidExitFullscreenCallback:)
name:MPMoviePlayerDidExitFullscreenNotification
object:_moviePlayerViewController.moviePlayer];
You’re probably right. In my example, I’m not handling MPMoviePlayerDidExitFullscreenNotification. Perhaps you can do something to isolate this notification inside the movieFinishedCallback:(NSNotification*)aNotification callback method.
Thanks for this. i`ve been struggled this problem for 2 days. now i got where the problem was.
Very Thank You for your post. It helped me lot.
Thank you!!!!!!!!!!!!!!
thanks for the information . It helped me to close a pending task π
Fantastic. this post is exactly what I’m looking for. thank you for your posting π
Great post. Solved my problem exactly!
Genius
Thank you!
thank you very much for you solution!!!!
nice
u can use this one
self.movieC.controlStyle=MPMovieControlStyleFullscreen;
and when u click “done” . it will invoke MPMoviePlayerPlaybackDidFinishNotification
Thanks a lot for your great answer π .
Excellent Post.. Helped me instead of using BLOCK function call .. Great Work. Thanks (y)..