2

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

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 放置原点的地方,但是对于用户来说,让窗口以相同的左上角重新启动更有意义否则应用程序每次打开时都会向下移动屏幕。

我尝试观察 NSWindowWillCloseNotification 并再次调用上述方法以在关闭之前将应用程序重置为摘要视图,但即使代码有效,窗口仍然在错误的位置开始 - 我猜 Cocoa 在通知已发送。

以前做过:系统偏好设置 - 它会自动调整视图大小,指示您的起始视图是什么,但您可以移动窗口,下次打开它时,它将位于基于左上角的新位置角落。有人知道如何模仿吗?

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

     ==============================
    | 跳转到:___ | “===”是窗口的顶部和底部边缘
    | | “___”是选择视图的弹出菜单
    | -------------------------- | 里面的虚线框是“框”,它包含视图
    || ||
    || ||
    || ||
    || ||
    || ||
    || ||
    | -------------------------- |
     ==============================
- (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 的第一个建议代码块放在 for 循环的下方,-windowControllerDidLoadNib:并且正确地锚定了窗口,但是当代码到达它时,-displayViewController:它设置 currentSize = (6,6)。我不确定这是否是因为 IB 中的框是空的。然后它将 (6,6) 和 SummaryView 大小之间的差异添加到应用程序上次退出时的窗口大小,使其变得很大。

然后我尝试通过添加两行来使用 NSUserDefaults-changeViewController:

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];
    }

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

我做的最后一件事是回到我的起始代码(不使用@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];
    }
}

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

4

1 回答 1

2

您的代码看起来不错,但真正的问题可能是您的文档 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];
}

我删除了较旧的注释,但现在指出附加代码在哪里。有了这个,您不必费心维护自己的文档默认值,也不必费心调整原点。

把这两件事放在一起,你(很大程度上)就有了解决办法。

是的,我需要在某种程度上限定它。您的应用程序是基于文档的应用程序,因此您将不得不想出一个唯一的名称,在该名称下将文档的窗口位置保存在您的用户默认值中,或者以某种方式将该信息与文档一起存储本身。

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

于 2012-12-01T02:37:35.180 回答