Mouseovers in NSCollectionView+NSScrollView

1.9k Views Asked by At

In a 10.7+ project, I am trying to enable UI elements in an NSCollectionViewItem when the mouse is within the bounds of the collection view item's view. For each collection view item which populates the NSCollectionView, I have a custom view which creates an individual NSTrackingArea using its bounds:

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

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

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

The tracking areas work great until the NSCollectionView content is scrolled. It is obvious that the tracking areas need to be reset during and after scrolling. I attempted to invalidate and recreate the NSTrackingAreas in several ways:

- (void) resetMouseTracking
{
    [self removeTrackingArea:self.trackingArea];
    [self initializeMouseTracking];
}


- (void) scrollWheel:(NSEvent *)theEvent
{
    [self resetMouseTracking];
    [super scrollWheel:theEvent];
}


- (void) viewDidMoveToWindow:(NSWindow *)newWindow
{   
    [self resetMouseTracking];
    [super viewWillMoveToWindow:newWindow];
}


- (void) updateTrackingAreas
{
    [self resetMouseTracking];
    [super updateTrackingAreas];
}

but not only have these attempts had incomplete and buggy results, the constant recalculation of the tracking areas during scrolling (ala scrollwheel:) seems unnecessary. Instead it would be useful to have an effective way to capture the onset and offset of scrolling events (easy to do in iOS) such that I can invalidate all tracking areas during scrolling. Using an NSViewBoundsDidChangeNotification on the contentView of the scrollview tells me that scrolling is happening, but it does not indicate when it starts or stops.

Would getting scrolling start and end notifications from NSScrollView require deep subclassing or is there something else I am overlooking? It there an entirely different approach that shows more promise?

2

There are 2 best solutions below

2
On

In Cocoa for the scroll view, unlike iOS everything is done on the documentView (as opposed to contentView). You probably need [scrollView documentVisibleRect].origin for the content offset.

Check out this answer for more similar comparisons for UIScrollView and NSScrollView

1
On

A different approach... an NSView inherits from NSResponder class. NSResponder has methods "mouseEntered:" and "mouseExited:". My suggestion is to use those in your NSView subclass if possible and skip the tracking areas.