还原时自动调整大小的窗口位置

时间:2012-12-01 00:50:48

标签: cocoa nswindow

我有一个应用程序,可以使用以下方法调整自身以适应不同的视图:

NSSize currentSize = [[box contentView] frame].size;
NSSize newSize = [v frame].size;
float deltaWidth = newSize.width - currentSize.width;
float deltaHeight = newSize.height - currentSize.height;
NSRect windowFrame = [w frame];
windowFrame.size.height += deltaHeight;
windowFrame.origin.y -= deltaHeight;
windowFrame.size.width += deltaWidth;

[box setContentView: nil];
[w setFrame: windowFrame
    display: YES
    animate: YES];

[box setContentView: v];

更改视图时,窗口会根据应用程序的左上角增大/缩小。应用程序始终在第一个视图上启动,无论用户退出时哪个视图,因为这是一个摘要视图。我还希望它可以恢复,所以最后使用的文档在发布时是开放的。

问题:由于大多数视图都比摘要视图高,因此更改为一个视图会将窗口的左下角推到屏幕的下方。现在退出应用程序并重新启动,应用程序将窗口定位到之前左下角的位置。我知道这是因为这是Cocoa放置原点的地方,但是对于用户来说,使用相同的 top 左角重新启动窗口更有意义,否则应用程序每次都会向下移动屏幕打开。

我尝试观察NSWindowWillCloseNotification并再次调用上面的方法将应用程序重置到摘要视图中,但是即使代码正常工作,窗口仍然会在错误的位置启动 - 我猜Cocoa设置了应用程序的可恢复性发送通知前的默认值。

之前已经完成了:系统偏好设备完成了它 - 它会自动调整视图大小,指示您的起始视图是什么,但是您可以移动窗口,下次打开它时,它会处于左上角的新位置。任何人都知道如何模仿它?

编辑:Document.xib有一个带有弹出按钮和一个框的小窗口。下面的方法更改视图并调整窗口大小以适合这些视图。当应用程序打开时,一切正常,窗口转换只是关闭应用程序并使用活动文档重新打开时的问题(即使用Lion的恢复)。

     ============================
    | Jump to:___                |    The "===" is the top and bottom edge of the window
    |                            |    The "___" is the pop-up menu for selecting the view
    | -------------------------- |    The dashed box on the inside is "box", which holds the views
    ||                          ||
    ||                          ||
    ||                          ||
    ||                          ||
    ||                          ||
    ||                          ||
    | -------------------------- |
     ============================
- (void)windowControllerDidLoadNib:(NSWindowController *)aController
{
    [super windowControllerDidLoadNib:aController];

    //Note that init creates the array viewControllers
    NSMenu *menu = [viewMenu menu];
    NSUInteger i, itemCount;
    itemCount = [viewControllers count];

    for (i = 0; i < itemCount; i++) {
        NSViewController *vc = [viewControllers objectAtIndex:i];
        NSMenuItem *mi = [[NSMenuItem alloc] initWithTitle: [vc title] action: @selector(changeViewController:) keyEquivalent:@""];
        [mi setTag: i];
        [menu addItem: mi];
    }

    [self displayViewController: [viewControllers objectAtIndex: 0]];
    [viewMenu selectItemAtIndex: 0];
}

- (void) displayViewController: (ManagingViewController *) vc {
    //End editing
    NSWindow *w = [box window];
    BOOL ended = [w makeFirstResponder:w];
    if (!ended) {
        NSBeep();
        return;
    }

    //Put view in box
    NSView *v = [vc view];

    NSSize currentSize = [[box contentView] frame].size;
    NSSize newSize = [v frame].size;
    float deltaWidth = newSize.width - currentSize.width;
    float deltaHeight = newSize.height - currentSize.height;
    NSRect windowFrame = [w frame];
    windowFrame.size.height += deltaHeight;
    windowFrame.origin.y -= deltaHeight;
    windowFrame.size.width += deltaWidth;

    [box setContentView: nil];
    [w setFrame: windowFrame display: YES animate: YES];

    [box setContentView: v];

}

- (IBAction)changeViewController:(id)sender {
    NSUInteger i = [sender tag];
    ManagingViewController *vc = [viewControllers objectAtIndex: i];
    [self displayViewController: vc];
}

我尝试将@trudyscousin中的第一个建议代码块放在-windowControllerDidLoadNib:中for循环的正下方,并且正确地锚定了窗口,但是当代码到达-displayViewController:时,它设置currentSize =(6, 6)。我不确定是不是因为IB中的盒子是空的。然后,当应用程序最后退出时,它会将(6,6)和SummaryView的大小之间的差异添加到窗口的大小,使其变得非常大。

然后我尝试通过向-changeViewController:

添加两行来使用NSUserDefaults
NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
[userDefaults setInteger: i forKey: @"lastViewController"];

并在-windowControllerDidLoadNib:

NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
    if ([userDefaults integerForKey: @"lastViewController"]) {
        NSInteger i = [userDefaults integerForKey: @"lastViewController"];
        ManagingViewController *vc = [viewControllers objectAtIndex: i];
        NSView *v = [vc view];
        NSSize currentSize = [v frame].size;
        [box setFrameSize: currentSize];
        [self displayViewController: vc];
        [viewMenu selectItemAtIndex: i];
    } else {
        [self displayViewController: [viewControllers objectAtIndex: 0]];
        [viewMenu selectItemAtIndex: 0];
    }

这样可行,但强制应用程序使用上次使用的视图启动,除非在打开/创建新文档时它甚至会这样做。如果我摆脱了其他(即,总是运行最后两行),我回到我开始的地方 - 应用程序启动在屏幕上垂直移动。看起来你不能在启动时设置窗口的原点,或者它可能会在以后重新加载它自己的默认值。

我做的最后一件事是回到我的开始代码(不使用@trudyscousin提供的掩码),但在-windowControllerDidLoadNib:的底部我添加了:

[self performSelector: @selector(adjustOrigin) withObject: nil afterDelay:0.0f];

调用:

- (void) adjustOrigin {
    NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
    if ([userDefaults integerForKey: @"lastViewController"]) {
        NSInteger i = [userDefaults integerForKey: @"lastViewController"];
        ManagingViewController *vc = [viewControllers objectAtIndex: i];
        NSView *v = [vc view];
        NSSize previousSize = [v frame].size;
        NSSize currentSize = [[[viewControllers objectAtIndex: 0] view] frame].size;
        CGFloat delta = previousSize.height - currentSize.height;
        NSRect windowFrame = [[box window] frame];
        windowFrame.origin.y += delta;
        [[box window] setFrame: windowFrame display:YES];
    }
}

实际上这确实有效,但是如果有多个东西在运行,那么滞后就足以让你看到窗口“跳”到它的新原点,这有点不雅。

1 个答案:

答案 0 :(得分:1)

您的代码似乎是合理的,但真正的问题可能是您的文档xib。

如果您的文档xib已设置为自动布局,请将其关闭。您可以通过在Xcode中选择文件来实现。在文件检查器中,关闭“使用自动布局”复选框。

完成后,选择弹出按钮。在“大小”检查器中,将按钮锚定在顶部和左侧。选择你的盒子。在“尺寸”检查器中,将框固定在所有四个边上,并确保它水平和垂直展开。

从我之前的回答中回顾一下,这是你的-windowControllerDidLoadNib:方法,其中包含我之前建议的更改:

- (void)windowControllerDidLoadNib:(NSWindowController *)aController
{
    [super windowControllerDidLoadNib:aController];

    //Note that init creates the array viewControllers
    NSMenu *menu = [viewMenu menu];
    NSUInteger i, itemCount;
    itemCount = [viewControllers count];

    for (i = 0; i < itemCount; i++) {
        NSViewController *vc = [viewControllers objectAtIndex:i];
        NSMenuItem *mi = [[NSMenuItem alloc] initWithTitle: [vc title] action: @selector(changeViewController:) keyEquivalent:@""];
        [mi setTag: i];
        [menu addItem: mi];
    }

    // ---

    NSWindow *myWindow = [[[self windowControllers] objectAtIndex:0] window];

    NSUInteger styleMask = [myWindow styleMask];

    styleMask ^= NSResizableWindowMask;
    [myWindow setStyleMask:styleMask];

    if ([myWindow setFrameUsingName:@"windowFrameAutosaveName"] == NO)
    {
        [myWindow center];
    }

    (void) [myWindow setFrameAutosaveName:@"windowFrameAutosaveName"];

    styleMask &= ~NSResizableWindowMask;
    [myWindow setStyleMask:styleMask];

    // ---

    [self displayViewController:[viewControllers objectAtIndex:0]];
    [viewMenu selectItemAtIndex:0];
}

我删除了较旧的评论,但现在指出了其他代码的位置。有了这个,您不必费心维护自己的文档默认值,而且不必费心调整原点。

把这两件事放在一起,你(很大程度上)有你的修复。

是的,我需要在某种程度上取得资格。你的是一个基于文档的应用程序,所以你必须要么想出一个唯一的名称来保存你的文档的窗口位置在你的用户默认值,或者可能以某种方式存储该信息与文档本身。

一如既往,祝你工作顺利。