37

为什么当鼠标通过滚动或做动画退出 NStrackingArea 时不调用 mouseExited/mouseEntered?

我创建这样的代码:

鼠标进入和退出:

-(void)mouseEntered:(NSEvent *)theEvent {
    NSLog(@"Mouse entered");
}

-(void)mouseExited:(NSEvent *)theEvent
{
    NSLog(@"Mouse exited");
}

追踪区域:

-(void)updateTrackingAreas
{ 
    if(trackingArea != nil) {
        [self removeTrackingArea:trackingArea];
        [trackingArea release];
    }

    int opts = (NSTrackingMouseEnteredAndExited | NSTrackingActiveAlways);
    trackingArea = [ [NSTrackingArea alloc] initWithRect:[self bounds]
                                             options:opts
                                               owner:self
                                            userInfo:nil];
    [self addTrackingArea:trackingArea];
}

更多细节:

我在 NSScrollView 的视图中添加了 NSViews 作为子视图。每个 NSView 都有自己的跟踪区域,当我滚动滚动视图并离开跟踪区域时,不会调用“mouseExited”,但不滚动一切正常。问题是当我滚动时调用“updateTrackingAreas”,我认为这会产生问题。

*仅 NSView 没有将其添加为子视图也有同样的问题,所以这不是问题。

4

2 回答 2

76

正如您在问题标题中指出的那样,仅在鼠标移动时才调用 mouseEntered 和 mouseExited 。要了解为什么会这样,我们先来看看第一次添加 NSTrackingAreas 的过程。

举个简单的例子,让我们创建一个通常绘制白色背景的视图,但如果用户将鼠标悬停在视图上,它会绘制红色背景。此示例使用 ARC。

@interface ExampleView

- (void) createTrackingArea

@property (nonatomic, retain) backgroundColor;
@property (nonatomic, retain) trackingArea;

@end

@implementation ExampleView

@synthesize backgroundColor;
@synthesize trackingArea

- (id) awakeFromNib
{
    [self setBackgroundColor: [NSColor whiteColor]];
    [self createTrackingArea];
}

- (void) createTrackingArea
{
    int opts = (NSTrackingMouseEnteredAndExited | NSTrackingActiveAlways);
    trackingArea = [ [NSTrackingArea alloc] initWithRect:[self bounds]
                                             options:opts
                                               owner:self
                                            userInfo:nil];
    [self addTrackingArea:trackingArea];
}

- (void) drawRect: (NSRect) rect
{
    [[self backgroundColor] set];
    NSRectFill(rect);
}

- (void) mouseEntered: (NSEvent*) theEvent
{
    [self setBackgroundColor: [NSColor redColor]];
}

- (void) mouseEntered: (NSEvent*) theEvent
{
    [self setBackgroundColor: [NSColor whiteColor]];
}

@end

这段代码有两个问题。首先,当调用 -awakeFromNib 时,如果鼠标已经在视图内,则不会调用 -mouseEntered。这意味着即使鼠标在视图上,背景仍然是白色的。这实际上在 NSView 文档中提到了 -addTrackingRect:owner:userData:assumeInside 的assumeInside 参数:

如果是,第一个事件将在光标离开 aRect 时生成,无论添加跟踪矩形时光标是否在 aRect 内。如果否,如果光标最初位于 aRect 内,则在光标离开 aRect 时生成第一个事件,或者如果光标最初位于 aRect 外部,则在光标进入 aRect 时生成第一个事件。

在这两种情况下,如果鼠标在跟踪区域内,则在鼠标离开跟踪区域之前不会生成任何事件。

所以为了解决这个问题,当我们添加跟踪区域时,我们需要确定光标是否在跟踪区域内。我们的 -createTrackingArea 方法因此变成

- (void) createTrackingArea
{
    int opts = (NSTrackingMouseEnteredAndExited | NSTrackingActiveAlways);
    trackingArea = [ [NSTrackingArea alloc] initWithRect:[self bounds]
                                             options:opts
                                               owner:self
                                            userInfo:nil];
    [self addTrackingArea:trackingArea];

    NSPoint mouseLocation = [[self window] mouseLocationOutsideOfEventStream];
    mouseLocation = [self convertPoint: mouseLocation
                              fromView: nil];

    if (NSPointInRect(mouseLocation, [self bounds]))
    {
        [self mouseEntered: nil];
    }
    else
    {
        [self mouseExited: nil];
    }
}

第二个问题是滚动。当滚动或移动视图时,我们需要重新计算该视图中的 NSTrackingAreas。这是通过删除跟踪区域然后将它们重新添加来完成的。如您所述,滚动视图时会调用 -updateTrackingAreas。这是删除和重新添加该区域的地方。

- (void) updateTrackingAreas
{
    [self removeTrackingArea:trackingArea];
    [self createTrackingArea];
    [super updateTrackingAreas]; // Needed, according to the NSView documentation
}

这应该可以解决您的问题。诚然,每次添加跟踪区域时都需要找到鼠标位置,然后将其转换为视图坐标,这很快就会过时,因此我建议在 NSView 上创建一个自动处理此问题的类别。您将无法始终调用 [self mouseEntered: nil] 或 [self mouseExited: nil],因此您可能希望让类别接受几个块。如果鼠标在 NSTrackingArea 中则运行一个,如果不在则运行一个。

于 2012-02-02T04:35:34.143 回答
4

@Michael 提供了一个很好的答案,并解决了我的问题。但有一件事,

if (CGRectContainsPoint([self bounds], mouseLocation))
{
    [self mouseEntered: nil];
}
else
{
    [self mouseExited: nil];
}

CGRectContainsPoint我在我的盒子里找到了作品,不是CGPointInRect

于 2012-10-03T14:05:33.420 回答