8

カスタム を作成しようとしているNSWindowので、適切なボーダーレス ウィンドウ マスクを使用して作成しましたが、それは機能します。私は独自のコンテンツビューを提供していますが、これは問題ありません。しかし、私がやろうとしているのは、サブビューをそれらの角にクリップする丸みを帯びた角で描画することです。これは可能ですか?

私のコンテンツ ビューでは、drawRect:角の丸いパスをオーバーライドして描画できますが、これにサブビューを追加すると、それらはクリップされません。

代わりに、コンテンツ ビューをレイヤーでサポートし、角の半径を ( にmasksToBounds設定してYES) 与えることができますが、サブビューを追加しても、丸い角によってまだクリップされません。

これを行う方法はありますか?または、タイトルバーなしで NSWindow を描画し、描画を完全に制御しながら、丸みを帯びたクリッピングコーナーを維持する方法はありますか?

4

3 に答える 3

13

私ができたことは、NSWindow のカスタム サブクラスを提供することです。

@implementation ELGRoundWindow

- (id)initWithContentRect:(NSRect)contentRect styleMask:(NSUInteger)aStyle backing:(NSBackingStoreType)bufferingType defer:(BOOL)flag
{
    self = [super initWithContentRect:contentRect styleMask:NSBorderlessWindowMask backing:bufferingType defer:flag];

    if ( self )
    {
        [self setStyleMask:NSBorderlessWindowMask];
        [self setOpaque:NO];
        [self setBackgroundColor:[NSColor clearColor]];
    }

    return self;
}


- (void) setContentView:(NSView *)aView
{
    aView.wantsLayer            = YES;
    aView.layer.frame           = aView.frame;
    aView.layer.cornerRadius    = 20.0;
    aView.layer.masksToBounds   = YES;


    [super setContentView:aView];

}

@end

そして IB で、コンテンツ ビューのクラスを ELGRoundView に変更しました。

@implementation ELGRoundView

- (void)drawRect:(NSRect)dirtyRect
{
    [[NSColor colorWithCalibratedRed:0.0 green:0.5 blue:1 alpha:1] set];
    NSRectFill(dirtyRect);
}

@end

次のように、コンテンツ ビューに別の正方形のサブビューを配置しました。

@implementation ELGSquareView

- (void)drawRect:(NSRect)dirtyRect
{
    [[NSColor colorWithCalibratedRed:0.0 green:0 blue:1 alpha:1] set];
    NSRectFill(dirtyRect);
}

@end

私は結局:

サブビューが切り取られた丸みを帯びたウィンドウ

于 2013-03-02T21:51:42.710 に答える
5

@ericgorrが示唆していることは正しいです。さらに、ウィンドウを移動およびサイズ変更可能にしたい場合は、NSWindow の init を次のように変更します。

- (id)initWithContentRect:(NSRect)contentRect
                 styleMask:(NSUInteger)aStyle
                   backing:(NSBackingStoreType)bufferingType
                     defer:(BOOL)flag
{
    self = [super initWithContentRect:contentRect
                            styleMask:aStyle
                              backing:bufferingType
                                defer:flag];
    if (self) {            
        [self setOpaque:NO];
        [self setBackgroundColor:[NSColor clearColor]];
        [self setMovableByWindowBackground:YES];
        [self setStyleMask:NSResizableWindowMask];
    }
    return self;
}

さらにカスタマイズするには、Apple のサンプル コード http://developer.apple.com/library/mac/#samplecode/RoundTransparentWindow/Introduction/Intro.htmlを参照してください。

于 2013-06-21T06:44:41.860 に答える
1

サブクラス化は最も柔軟な方法です。サブクラス化したくない場合は、このコードを使用してください。

// unfortunately the window can't be moved
// but this is just an alert type window, so i don't care
//
[window setOpaque:NO];
[window setBackgroundColor:[NSColor clearColor]];

NSView*  contentView = window.contentView;

contentView.wantsLayer = YES;
contentView.layer.backgroundColor = [NSColor windowBackgroundColor].CGColor;
contentView.layer.masksToBounds = YES;
contentView.layer.cornerRadius = 16.0;

[window makeKeyAndOrderFront:self];
[window center];
于 2016-04-25T17:11:36.443 に答える