iPhone SDK:Как воспроизводить видео внутри view?Вместо полноэкранного режима
-
13-09-2019 - |
Вопрос
Я пытаюсь воспроизвести видео внутри UIView
, итак , моим первым шагом было добавить класс для этого представления и начать проигрывание фильма в нем , используя этот код:
- (IBAction)movie:(id)sender{
NSBundle *bundle = [NSBundle mainBundle];
NSString *moviePath = [bundle pathForResource:@"Movie" ofType:@"m4v"];
NSURL *movieURL = [[NSURL fileURLWithPath:moviePath] retain];
MPMoviePlayerController *theMovie = [[MPMoviePlayerController alloc] initWithContentURL:movieURL];
theMovie.scalingMode = MPMovieScalingModeAspectFill;
[theMovie play];
}
Но это просто приводит к сбою приложения при использовании этого метода внутри его собственного класса, но прекрасно работает в других местах.Кто-нибудь знает, как воспроизводить видео внутри вида?и избежать того, чтобы это было на весь экран?
Решение
Начиная с SDK 3.2, вы можете получить доступ к свойству просмотра MPMoviePlayerController
, измените его фрейм и добавьте его в иерархию представлений.
MPMoviePlayerController *player = [[MPMoviePlayerController alloc] initWithContentURL:[NSURL fileURLWithPath:url]];
player.view.frame = CGRectMake(184, 200, 400, 300);
[self.view addSubview:player.view];
[player play];
Здесь есть пример: http://www.devx.com/wireless/Article/44642/1954
Другие советы
Лучший способ — использовать слои вместо представлений:
AVPlayer *player = [AVPlayer playerWithURL:[NSURL url...]]; //
AVPlayerLayer *layer = [AVPlayerLayer layer];
[layer setPlayer:player];
[layer setFrame:CGRectMake(10, 10, 300, 200)];
[layer setBackgroundColor:[UIColor redColor].CGColor];
[layer setVideoGravity:AVLayerVideoGravityResizeAspectFill];
[self.view.layer addSublayer:layer];
[player play];
Не забудьте добавить фреймворки:
#import <QuartzCore/QuartzCore.h>
#import "AVFoundation/AVFoundation.h"
Глядя на ваш код, вам необходимо установить рамку представления контроллера видеопроигрывателя, а также добавить представление контроллера видеопроигрывателя в свое представление.Также не забудьте добавить MediaPlayer.framework к вашей цели.
Вот пример кода:
#import <MediaPlayer/MediaPlayer.h>
@interface ViewController () {
MPMoviePlayerController *moviePlayerController;
}
@property (weak, nonatomic) IBOutlet UIView *movieView; // this should point to a view where the movie will play
@end
@implementation ViewController
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
// Instantiate a movie player controller and add it to your view
NSString *moviePath = [[NSBundle mainBundle] pathForResource:@"foo" ofType:@"mov"];
NSURL *movieURL = [NSURL fileURLWithPath:moviePath];
moviePlayerController = [[MPMoviePlayerController alloc] initWithContentURL:movieURL];
[moviePlayerController.view setFrame:self.movieView.bounds]; // player's frame must match parent's
[self.movieView addSubview:moviePlayerController.view];
// Configure the movie player controller
moviePlayerController.controlStyle = MPMovieControlStyleNone;
[moviePlayerController prepareToPlay];
}
- (void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
// Start the movie
[moviePlayerController play];
}
@end
Быстрый
Это автономный проект, так что вы можете видеть все в контексте.
Планировка
Создайте макет, подобный следующему, с помощью UIView
и еще UIButton
.Тот Самый UIView
это будет контейнер, в котором мы будем воспроизводить наше видео.
Добавьте видео в проект
Если вам нужен пример видео для практики, вы можете получить его на сайте sample-videos.com.В этом примере я использую видео в формате mp4.Перетащите видеофайл в свой проект.Мне также пришлось явно добавить его в ресурсы пакета (перейдите по ссылке Этапы сборки > Копирование ресурсов пакета, видеть этот ответ для получения дополнительной информации).
Код
Вот полный код для проекта.
import UIKit
import AVFoundation
class ViewController: UIViewController {
var player: AVPlayer?
@IBOutlet weak var videoViewContainer: UIView!
override func viewDidLoad() {
super.viewDidLoad()
initializeVideoPlayerWithVideo()
}
func initializeVideoPlayerWithVideo() {
// get the path string for the video from assets
let videoString:String? = Bundle.main.path(forResource: "SampleVideo_360x240_1mb", ofType: "mp4")
guard let unwrappedVideoPath = videoString else {return}
// convert the path string to a url
let videoUrl = URL(fileURLWithPath: unwrappedVideoPath)
// initialize the video player with the url
self.player = AVPlayer(url: videoUrl)
// create a video layer for the player
let layer: AVPlayerLayer = AVPlayerLayer(player: player)
// make the layer the same size as the container view
layer.frame = videoViewContainer.bounds
// make the video fill the layer as much as possible while keeping its aspect size
layer.videoGravity = AVLayerVideoGravity.resizeAspectFill
// add the layer to the container view
videoViewContainer.layer.addSublayer(layer)
}
@IBAction func playVideoButtonTapped(_ sender: UIButton) {
// play the video if the player is initialized
player?.play()
}
}
Примечания
- Если вы собираетесь включать и выключать разные видео, вы можете использовать
AVPlayerItem
. - Если вы используете только
AVFoundation
иAVPlayer
, затем вы должны создать все свои собственные элементы управления.Если вы хотите воспроизводить видео в полноэкранном режиме, вы можете использоватьAVPlayerViewController
.Вам нужно будет импортироватьAVKit
за это.Он поставляется с полным набором элементов управления для паузы, быстрой перемотки вперед, назад, остановки и т.д. Здесь и здесь вот несколько видеоуроков. MPMoviePlayerController
то, что вы, возможно, видели в других ответах, устарело.
Результат
Сейчас проект должен выглядеть примерно так.
NSString * pathv = [[NSBundle mainBundle] pathForResource:@"vfile" ofType:@"mov"];
playerv = [[MPMoviePlayerViewController alloc] initWithContentURL:[NSURL fileURLWithPath:pathv]];
[self presentMoviePlayerViewControllerAnimated:playerv];
NSURL *url = [NSURL URLWithString:[exreciesDescription objectForKey:@"exercise_url"]];
moviePlayer =[[MPMoviePlayerController alloc] initWithContentURL: url];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(doneButtonClicked) name:MPMoviePlayerWillExitFullscreenNotification object:nil];
[[moviePlayer view] setFrame: [self.view bounds]]; // frame must match parent view
[self.view addSubview: [moviePlayer view]];
[moviePlayer play];
-(void)playMediaFinished:(NSNotification*)theNotification
{
moviePlayer=[theNotification object];
[[NSNotificationCenter defaultCenter] removeObserver:self
name:MPMoviePlayerPlaybackDidFinishNotification
object:moviePlayer];
[moviePlayer.view removeFromSuperview];
}
-(void)doneButtonClicked
{
[moviePlayer stop];
[moviePlayer.view removeFromSuperview];
[self.navigationController popViewControllerAnimated:YES];//no need this if you are opening the player in same screen;
}
Свифт-версия:
import AVFoundation
func playVideo(url: URL) {
let player = AVPlayer(url: url)
let layer: AVPlayerLayer = AVPlayerLayer(player: player)
layer.backgroundColor = UIColor.white.cgColor
layer.frame = CGRect(x: 0, y: 0, width: 300, height: 300)
layer.videoGravity = .resizeAspectFill
self.view.layer.addSublayer(layer)
player.play()
}
Используйте следующий метод.
self.imageView_VedioContainer
это вид контейнера вашего AVPlayer
.
- (void)playMedia:(UITapGestureRecognizer *)tapGesture
{
playerViewController = [[AVPlayerViewController alloc] init];
playerViewController.player = [AVPlayer playerWithURL:[[NSBundle mainBundle]
URLForResource:@"VID"
withExtension:@"3gp"]];
[playerViewController.player play];
playerViewController.showsPlaybackControls =YES;
playerViewController.view.frame=self.imageView_VedioContainer.bounds;
[playerViewController.view setAutoresizingMask:UIViewAutoresizingNone];// you can comment this line
[self.imageView_VedioContainer addSubview: playerViewController.view];
}
Вы не можете воспроизводить видео внутри представления.Надо играть в полноэкранном режиме.