我已经尝试了我能想到的所有内容,包括我在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上有一个完整的示例应用程序。
Objective-C的:
[[splitViewItem animator] setCollapse:YES]
迅速:
splitViewItem.animator().collapsed = true
来自Apple的帮助:
是否在SplitViewController中折叠了与SplitViewItem对应的子ViewController。默认值为NO。可以使用动画制作代理设置此操作以设置折叠或解除折叠的动画。使用的精确动画可以通过使用“折叠”键在-animations字典中设置来自定义。如果在将它添加到SplitViewController之前将其设置为YES,则它最初将被折叠,并且SplitViewController将不会导致视图加载,直到它被解除。这符合KVC / KVO标准,如果值因用户交互而变化,则会更新。
我最终能够在一些帮助下解决这个问题。我已将我的测试项目转换为可重用的NSSplitView
子类:https://github.com/mdiep/MDPSplitView
如果您正在使用自动布局并且想要为视图的尺寸/位置的某些方面设置动画,那么您可能会更有运气为约束本身设置动画。我已经快速使用了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
。
由于某种原因,没有任何动画帧的方法适用于我的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];
}
}
NSSplitViewItem
(即排列的NSSplitView
子视图)可以完全折叠,如果它可以达到Zero
尺寸(宽度或高度)。因此,我们只需要在动画之前停用适当的约束,并允许视图达到Zero
维度。动画后,我们可以再次激活所需的约束。
请参阅我对SO问题How to expand and collapse NSSplitView subviews with animation?的评论。
这是一个不需要任何子类或类别的解决方案,没有NSSplitViewController
(需要macOS 10.10+),支持自动布局,动画视图,并在macOS 10.8+上工作。
正如其他人所建议的,解决方案是使用NSAnimationContext,但诀窍是设置context.allowsImplicitAnimation = YES
(Apple 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。
/// 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
}
}