cocoamacosanimationnssplitview

How to expand and collapse NSSplitView subviews with animation?


Is it possible to animate the collapsing and expanding of NSSplitView subviews? (I am aware of the availability of alternative classes, but would prefer using NSSplitView over having animations.)

I am using the method - (void)setPosition:(CGFloat)position ofDividerAtIndex:(NSInteger)dividerIndex to perform the collapsing and expanding.


Solution

  • After some more trying, I found the answer: yes, it's possible.

    The code below shows how it can be done. The splitView is the NSSplitView which is vertically divided into mainView (on the left) and the inspectorView (on the right). The inspectorView is the one that collapses.

    - (IBAction)toggleInspector:(id)sender {
       if ([self.splitView isSubviewCollapsed:self.inspectorView]) {
            // NSSplitView hides the collapsed subview
            self.inspectorView.hidden = NO;
    
            NSMutableDictionary *expandMainAnimationDict = [NSMutableDictionary dictionaryWithCapacity:2];
            [expandMainAnimationDict setObject:self.mainView forKey:NSViewAnimationTargetKey];
            NSRect newMainFrame = self.mainView.frame;
            newMainFrame.size.width =  self.splitView.frame.size.width-lastInspectorWidth;
            [expandMainAnimationDict setObject:[NSValue valueWithRect:newMainFrame] forKey:NSViewAnimationEndFrameKey];
    
            NSMutableDictionary *expandInspectorAnimationDict = [NSMutableDictionary dictionaryWithCapacity:2];
            [expandInspectorAnimationDict setObject:self.inspectorView forKey:NSViewAnimationTargetKey];
            NSRect newInspectorFrame = self.inspectorView.frame;
            newInspectorFrame.size.width = lastInspectorWidth;
            newInspectorFrame.origin.x = self.splitView.frame.size.width-lastInspectorWidth;
            [expandInspectorAnimationDict setObject:[NSValue valueWithRect:newInspectorFrame] forKey:NSViewAnimationEndFrameKey];
    
            NSViewAnimation *expandAnimation = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObjects:expandMainAnimationDict, expandInspectorAnimationDict, nil]];
            [expandAnimation setDuration:0.25f];
            [expandAnimation startAnimation];
        } else {
            // Store last width so we can jump back
            lastInspectorWidth = self.inspectorView.frame.size.width;
    
            NSMutableDictionary *collapseMainAnimationDict = [NSMutableDictionary dictionaryWithCapacity:2];
            [collapseMainAnimationDict setObject:self.mainView forKey:NSViewAnimationTargetKey];
            NSRect newMainFrame = self.mainView.frame;
            newMainFrame.size.width =  self.splitView.frame.size.width;
            [collapseMainAnimationDict setObject:[NSValue valueWithRect:newMainFrame] forKey:NSViewAnimationEndFrameKey];
    
            NSMutableDictionary *collapseInspectorAnimationDict = [NSMutableDictionary dictionaryWithCapacity:2];
            [collapseInspectorAnimationDict setObject:self.inspectorView forKey:NSViewAnimationTargetKey];
            NSRect newInspectorFrame = self.inspectorView.frame;
            newInspectorFrame.size.width = 0.0f;
            newInspectorFrame.origin.x = self.splitView.frame.size.width;
            [collapseInspectorAnimationDict setObject:[NSValue valueWithRect:newInspectorFrame] forKey:NSViewAnimationEndFrameKey];
    
            NSViewAnimation *collapseAnimation = [[NSViewAnimation alloc] initWithViewAnimations:[NSArray arrayWithObjects:collapseMainAnimationDict, collapseInspectorAnimationDict, nil]];
            [collapseAnimation setDuration:0.25f];
            [collapseAnimation startAnimation];
        }
    }
    
    - (BOOL)splitView:(NSSplitView *)splitView canCollapseSubview:(NSView *)subview {
        BOOL result = NO;
        if (splitView == self.splitView && subview == self.inspectorView) {
            result = YES;
        }
        return result;
    }
    
    - (BOOL)splitView:(NSSplitView *)splitView shouldCollapseSubview:(NSView *)subview forDoubleClickOnDividerAtIndex:(NSInteger)dividerIndex {
        BOOL result = NO;
        if (splitView == self.splitView && subview == self.inspectorView) {
            result = YES;
        }
        return result;
    }