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
方法没有立即绘制(它以前这样做过),它有什么用?
当处理非常大的数据集(数百万项)时,我的 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
方法没有立即绘制(它以前这样做过),它有什么用?