带有视频播放器的自定义 NSView 未显示 (MacOS)

custom NSView with video player is not showing (MacOS)

我创建了一个嵌入了 AVPlayer 的自定义 NSView,但它根本没有显示

这是我的 header:

@interface VideoPlayer : NSView
- (void)loadVideo;
@end

实施:

@implementation VideoPlayer

AVPlayer *avPlayer;
AVPlayerLayer* playerLayer;

- (instancetype)initWithFrame:(NSRect)rect {
    self = [super initWithFrame:rect];
    [self setWantsLayer:YES];
    [self setNeedsDisplay:YES];
    self.hidden = NO;
    return self;
}

-(void) loadVideo {
    [avPlayer pause];
    avPlayer = nil;
    playerLayer = nil;

    NSURL *videoURL = [NSURL fileURLWithPath:@"file://~/Documents/tests.mp4"];
    //same error with a valid url
    //NSURL *videoURL = [NSURL URLWithString:@"http://commondatastorage.googleapis.com/gtv-videos-bucket/sample/ForBiggerEscapes.mp4"];

    avPlayer = [AVPlayer playerWithURL:videoURL];
    playerLayer = [AVPlayerLayer playerLayerWithPlayer:avPlayer];
    playerLayer.frame = self.frame;//or your desired CGRect
    playerLayer.videoGravity = AVLayerVideoGravityResizeAspectFill;
    //playerLayer.masksToBounds = YES;
    [self.layer addSublayer:playerLayer];

    [avPlayer play];
}

@end

这就是我创建视图的方式:

view = [[VideoPlayer alloc] initWithFrame:NSMakeRect(0, 0, 500, 500)];
view.hidden = false;
[view loadVideo];
[parent_view addSubview:view];

我尝试了多种组合,但结果总是一样的...应用程序没有呈现任何内容。

另一方面,如果我像这样在我的视图中覆盖方法 drawRect:

- (void)drawRect:(NSRect)dirtyRect {
         [super drawRect:dirtyRect];
         [[NSColor whiteColor] setFill];
         NSRectFill(dirtyRect);
         [super drawRect:dirtyRect];
}

我得到一个白色矩形,它应该显示我的视频,这是预期的。所以我假设我遗漏了其他东西,但大多数示例都使用了 UIView,这对我来说目前不是一个选项(我正在处理一些特定的限制)

有什么想法吗?

谢谢

更新:

这终于对我有用了:

-(void) loadVideo {

    AVPlayer *avPlayer = [AVPlayer playerWithURL:[NSURL URLWithString:@"file://localhost/Users/xxxxxxxx/Documents/xxxxxxx.mp4"]];
    AVPlayerView *playerView = [[AVPlayerView alloc] initWithFrame:self.bounds];
    [playerView setPlayer:avPlayer];
    [self addSubview:playerView];
    [playerView.player play];
}
  1. 确保将 AVKit 框架添加到项目中。
  2. AVKit 框架中有一个 header 叫做 AVPlayerView.h
  3. 那么你需要做的就是调用它:
// **** AVPlayer ***** //
AVPlayerView *playerView = [[AVPlayerView alloc] initWithFrame:NSMakeRect( 0, 0, _wndW, _wndH - 100)];
[[window contentView] addSubview:playerView];

我有一个程序化演示,如果您无法使用它,我可以 post。

为了让事情更清楚一些,以下源代码将为音频文件创建一个基本的 AVPlayer。请注意,AVPlayerView 和 AVPlayer 分别需要添加 AVKit 和 AVFoundation 框架。该代码使用自动引用计数,可以在终端中 运行 或在 Xcode 中使用,方法是替换 main.m 和 AppDelegate 并删除 info.plist 中的主要笔尖。要在终端中 运行,将 copy/paste 编码到名为 avplayer.m 的文件中,然后使用如下所示的 cmdLine。 Apple 文档声明它仅在 MacOS 上 运行s。

/*
To run in Terminal: clang avplayer.m -fobjc-arc -framework Cocoa -framework AVKit -framework AVFoundation -o avplayer && ./avplayer
*/

#import <Cocoa/Cocoa.h>
#import <AVKit/AVKit.h>
#import <AVFoundation/AVFoundation.h>

@interface AppDelegate : NSObject <NSApplicationDelegate> {
 NSWindow *window;
 AVPlayerView *playerView;
}
- (void) openAction;
- (void) buildMenu;
- (void) buildWindow;
@end

@implementation AppDelegate

- (void) openAction {
 NSOpenPanel *op = [NSOpenPanel openPanel];
 if ( [op runModal] == NSModalResponseOK ) {
 AVPlayer *player = [AVPlayer playerWithURL:[op URL]];
 [playerView setPlayer:player];
 [window setTitleWithRepresentedFilename:[[op URL]lastPathComponent]];
 }
}

- (void) buildMenu {
 NSMenu *menubar = [NSMenu new];
 NSMenuItem *menuBarItem = [NSMenuItem new] ;
 [menubar addItem:menuBarItem];
 [NSApp setMainMenu:menubar];
 NSMenu *appMenu = [NSMenu new];
 NSMenuItem *quitMenuItem = [[NSMenuItem alloc] initWithTitle:@"Quit" action:@selector(terminate:) keyEquivalent:@"q"];
 [appMenu addItem:quitMenuItem];
 [menuBarItem setSubmenu:appMenu];
}

- (void) buildWindow {

#define _wndW  500
#define _wndH  400

window = [[NSWindow alloc] initWithContentRect: NSMakeRect( 0, 0, _wndW, _wndH ) styleMask: NSWindowStyleMaskTitled | NSWindowStyleMaskMiniaturizable | NSWindowStyleMaskClosable backing: NSBackingStoreBuffered defer: NO];

[window center];
[window setTitle: @"Test window"];
[window makeKeyAndOrderFront: nil];

// **** AVPlayerView **** //
playerView = [[AVPlayerView alloc] initWithFrame:NSMakeRect( 0, 60, _wndW, _wndH - 60)];
[[window contentView] addSubview:playerView];

// **** Open Button **** //
NSButton *myBtn =[[NSButton alloc]initWithFrame:NSMakeRect( 30, 20, 125, 30 )];
[myBtn setBezelStyle:NSBezelStyleRounded ];
[myBtn setTitle: @"Open audio file"];
[myBtn setAction: @selector (openAction)];
[[window contentView] addSubview: myBtn];

// **** Quit btn **** //
NSButton *quitBtn = [[NSButton alloc]initWithFrame:NSMakeRect( _wndW - 50, 5, 40, 40 )];
[quitBtn setBezelStyle:NSBezelStyleCircular ];
[quitBtn setTitle: @"Q" ];
[quitBtn setAutoresizingMask: NSViewMinXMargin];
[quitBtn setAction:@selector(terminate:)];
[[window contentView] addSubview: quitBtn];
}

- (void) applicationWillFinishLaunching: (NSNotification *)notification {
[self buildMenu];
[self buildWindow];
}

- (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)sender {
 return YES;
}

@end

int main () {
 NSApplication *application = [NSApplication sharedApplication];
 AppDelegate *appDelegate = [[AppDelegate alloc] init];
 [application setDelegate:appDelegate];
 [application run];
 return 0;
}