NSView 显示代码在 Yosemite 中有效但在 macOS Sierra 中无效

NSView display code worked in Yosemite but not working in macOS Sierra

当处理非常大的数据集(数百万项)时,我的 macOS 应用程序有时可能需要很长时间才能完成一个操作,在某些情况下甚至超过一分钟。为了让用户知道应用程序正在运行,我在工具栏的一个特殊视图中像温度计一样显示进度。这最初是几年前在 Yosemite 上开发的,并且在 10.9 和 10.10 上 运行 时仍然可以正常工作。但是,它不适用于 Sierra(不确定 10.11,没有系统来测试它)。

这是它运行时的样子,显示了 30% 的进度。效果很像 Xcode 中工具栏中显示的构建进度(和 Xcode 一样,其他信息显示在 banner 中,这也是我没有使用 NSProgressIndicator 的原因)。

使用名为 InfoBannerView 的 class 显示进度条,它是 NSButton 的子class。这是此 class 的 drawRect: 方法(为清晰起见,省略了绘制背景和文本的非进度代码)。 progressValue 是一个可以在 0 和 1 之间的 属性,该方法简单地绘制一个圆角矩形,填充视图宽度的百分比,如温度计。

- (void)drawRect:(NSRect)dirtyRect
{
    [super drawRect:dirtyRect];

    NSRect bannerRectangle = [self bounds];
    {
        NSColor* progressColor = [NSColor colorFromHexRGBString:@"85ade1"];

        ... draw outline and background of banner

        if (progressValue > 0.0f && progressValue < 1.0f) {
            [progressColor setFill];
            NSRect progressRect = NSInsetRect(bannerRectangle, 1.0, 1.0);
            CGFloat progressWidth = progressRect.size.width * progressValue;
            progressRect.size.width = progressWidth;
            NSBezierPath* roundedProgressPath = [NSBezierPath bezierPathWithRoundedRect:progressRect xRadius:4 yRadius:4];
            [roundedProgressPath fill];
        }

        ... draw text on banner
    }
}

需要很长时间才能运行的代码如下所示:

for {

    ... do stuff that takes a long time

    InfoBannerView * ibv = [windowController recordCountView];
    [ibv setProgress:value];
    [ibv display];
}

[ibv display] 行应该立即更新进度条显示,而无需等待 运行 循环。 (注意:出于性能原因,我并不是每次都在循环中调用 -display,但为了清楚起见,我删除了该代码。)

如果此代码在 Mac OS X 10.10 上 运行,则它可以完美运行。随着工作的进行,进度条逐渐填充横幅视图。

但是,如果此代码在 macOS 10.12 Sierra 上 运行,则什么也看不到。我已经确定 drawRect: 代码 被重复调用,这是应该的。正在为正确的对象调用它。所有尺寸和值都是正确的。唯一的问题是——没有画任何东西。 (但是,在正常视图更新过程中调用 drawRect: 时,绘图确实会正确发生。通过将 progressValue 强制设置为 0.3,我确定该代码在视图更新期间也能正确绘制,事实上,这就是我制作上面插图的方式。)

我尝试了 运行 与 10.10 SDK 链接的旧版本应用程序。通过 -display 调用时仍然没有绘图。它仅在 10.10 或更早版本 运行 时有效。

我很希望在写这篇文章时云层会散去,答案会变得显而易见,但没有这样的运气。我在 10.11 或 10.12 发行说明中没有看到任何相关内容。也许我正在做一些一直以来都是错误的愚蠢事情,我很幸运它曾经奏效过?在这一点上,我没主意了。

Daniel Jalkut 在 Slack 上的建议导致了一个解决方案。替换此行:

[ibv display];

有了这个:

[ibv setNeedsDisplay:YES];
[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:0]];

导致正确显示,如此处所示。

我认为最初的问题可能是 Sierra 中的一个错误——如果 -display 方法没有立即绘制(它以前这样做过),它有什么用?