如何在使用自动布局时使用动画折叠NSSplitView窗格?

问题描述 投票:7回答:7

我已经尝试了我能想到的所有内容,包括我在SO和其他邮件列表中找到的所有建议,但我无法弄清楚如何在启用“自动布局”时以编程方式折叠带有动画的NSSplitView窗格。

这就是我现在所拥有的(用Swift编写的乐趣),但它有多种方式:

@IBAction func toggleSourceList(sender: AnyObject?) {
    let isOpen = !splitView.isSubviewCollapsed(sourceList.view.superview!)
    let position = (isOpen ? 0 : self.lastWidth)

    if isOpen {
        self.lastWidth = sourceList.view.frame.size.width
    }

    NSAnimationContext.runAnimationGroup({ context in
        context.allowsImplicitAnimation = true
        context.timingFunction = CAMediaTimingFunction(name: kCAMediaTimingFunctionEaseIn)
        context.duration = self.duration

        self.splitView.setPosition(position, ofDividerAtIndex: 0)
    }, completionHandler: { () -> Void in
    })
}

期望的行为和外观是Mail.app,它的动画效果非常好。

我在https://github.com/mdiep/NSSplitViewTest上有一个完整的示例应用程序。

cocoa appkit nssplitview
7个回答
18
投票

Objective-C的:

[[splitViewItem animator] setCollapse:YES]

迅速:

splitViewItem.animator().collapsed = true

来自Apple的帮助:

是否在SplitViewController中折叠了与SplitViewItem对应的子ViewController。默认值为NO。可以使用动画制作代理设置此操作以设置折叠或解除折叠的动画。使用的精确动画可以通过使用“折叠”键在-animations字典中设置来自定义。如果在将它添加到SplitViewController之前将其设置为YES,则它最初将被折叠,并且SplitViewController将不会导致视图加载,直到它被解除。这符合KVC / KVO标准,如果值因用户交互而变化,则会更新。


1
投票

我最终能够在一些帮助下解决这个问题。我已将我的测试项目转换为可重用的NSSplitView子类:https://github.com/mdiep/MDPSplitView


0
投票

如果您正在使用自动布局并且想要为视图的尺寸/位置的某些方面设置动画,那么您可能会更有运气为约束本身设置动画。我已经快速使用了NSSplitView,但到目前为止只取得了有限的成功。按下按钮后,我可以进行拆分以扩展和折叠,但我最终不得不试图破解由于干扰约束而导致的其他大量问题。如果你不熟悉它,这里是一个简单的约束动画:

- (IBAction)animate:(NSButton *)sender {
    /* Shrink view to invisible */

    NSLayoutConstraint *constraint = self.viewWidthConstraint;
    [NSAnimationContext runAnimationGroup:^(NSAnimationContext *context) {

        [[NSAnimationContext currentContext] setDuration:0.33];
        [[NSAnimationContext currentContext] setTimingFunction:[CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionDefault]];
        [[constraint animator] setConstant:0];

    } completionHandler:^{
        /* Do Some clean-up, if required */

    }];

请记住,你只能动画约束constant,你不能动画它的priority


0
投票

由于某种原因,没有任何动画帧的方法适用于我的scrollview。我没有尝试动画约束。

我最终创建了一个自定义动画来设置分隔符位置的动画。如果有人有兴趣,这是我的解决方案:

动画.h:

@interface MySplitViewAnimation : NSAnimation <NSAnimationDelegate>

@property (nonatomic, strong) NSSplitView* splitView;
@property (nonatomic) NSInteger dividerIndex;
@property (nonatomic) float startPosition;
@property (nonatomic) float endPosition;
@property (nonatomic, strong) void (^completionBlock)();

- (instancetype)initWithSplitView:(NSSplitView*)splitView
                   dividerAtIndex:(NSInteger)dividerIndex
                             from:(float)startPosition
                               to:(float)endPosition
                  completionBlock:(void (^)())completionBlock;
@end

动画.m

@implementation MySplitViewAnimation

- (instancetype)initWithSplitView:(NSSplitView*)splitView
                   dividerAtIndex:(NSInteger)dividerIndex
                             from:(float)startPosition
                               to:(float)endPosition
                  completionBlock:(void (^)())completionBlock;
{
    if (self = [super init]) {
        self.splitView = splitView;
        self.dividerIndex = dividerIndex;
        self.startPosition = startPosition;
        self.endPosition = endPosition;
        self.completionBlock = completionBlock;

        [self setDuration:0.333333];
        [self setAnimationBlockingMode:NSAnimationNonblocking];
        [self setAnimationCurve:NSAnimationEaseIn];
        [self setFrameRate:30.0];
        [self setDelegate:self];
    }
    return self;
}

- (void)setCurrentProgress:(NSAnimationProgress)progress
{
    [super setCurrentProgress:progress];

    float newPosition = self.startPosition + ((self.endPosition - self.startPosition) * progress);

    [self.splitView setPosition:newPosition
               ofDividerAtIndex:self.dividerIndex];

    if (progress == 1.0) {
        self.completionBlock();
    }
}

@end

我正在使用它 - 我有一个3窗格分割器视图,并且正在以固定量(235)移动/移出右窗格。

- (IBAction)togglePropertiesPane:(id)sender
{
    if (self.rightPane.isHidden) {

        self.rightPane.hidden = NO;

        [[[MySplitViewAnimation alloc] initWithSplitView:_splitView
                                          dividerAtIndex:1
                                                  from:_splitView.frame.size.width  
                                                   to:_splitView.frame.size.width - 235                                                                                                             
                         completionBlock:^{
              ;
                                 }] startAnimation];
}
else {
    [[[MySplitViewAnimation alloc] initWithSplitView:_splitView
                                      dividerAtIndex:1                                                          
                                               from:_splitView.frame.size.width - 235
                                                 to:_splitView.frame.size.width
                                     completionBlock:^{          
        self.rightPane.hidden = YES;
                                     }] startAnimation];
    } 
}

0
投票

NSSplitViewItem(即排列的NSSplitView子视图)可以完全折叠,如果它可以达到Zero尺寸(宽度或高度)。因此,我们只需要在动画之前停用适当的约束,并允许视图达到Zero维度。动画后,我们可以再次激活所需的约束。

请参阅我对SO问题How to expand and collapse NSSplitView subviews with animation?的评论。


0
投票

这是一个不需要任何子类或类别的解决方案,没有NSSplitViewController(需要macOS 10.10+),支持自动布局,动画视图,并在macOS 10.8+上工作。

正如其他人所建议的,解决方案是使用NSAnimationContext,但诀窍是设置context.allowsImplicitAnimation = YESApple docs)。然后只需将分配器位置设置为正常位置。

#import <Quartz/Quartz.h>
#import <QuartzCore/QuartzCore.h>

- (IBAction)toggleLeftPane:(id)sender
{
    [NSAnimationContext runAnimationGroup:^(NSAnimationContext * _Nonnull context) {
        context.allowsImplicitAnimation = YES;
        context.duration = 0.25; // seconds
        context.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseOut];

        if ([self.splitView isSubviewCollapsed:self.leftPane]) {
            // -> expand
            [self.splitView setPosition:self.leftPane.frame.size.width ofDividerAtIndex:0];
        } else {
            // <- collapse
            _lastLeftPaneWidth = self.leftPane.frame.size.width;
            // optional: remember current width to restore to same size
            [self.splitView setPosition:0 ofDividerAtIndex:0];
        }

        [self.splitView layoutSubtreeIfNeeded];
    }];
}

使用自动布局来约束子视图(宽度,最小/最大尺寸等)。确保在分割视图和所有子视图中检查Interface Builder中的“核心动画层”(即为视图设置视图) - 这是动画过渡所必需的。 (它仍然有效,但没有动画。)

这里有一个完整的工作项目:https://github.com/demitri/SplitViewAutoLayout


0
投票
/// Collapse the sidebar
    func collapsePanel(_ number: Int = 0){
        guard number < self.splitViewItems.count else {
            return
        }
        let panel = self.splitViewItems[number]

        if panel.isCollapsed {
            panel.animator().isCollapsed = false
        } else {
            panel.animator().isCollapsed = true
        }

    }
© www.soinside.com 2019 - 2024. All rights reserved.