我有一个应用程序,在一个窗口中,有一个 NSImageView。用户应该能够将任何文件/文件夹(不仅仅是图像)拖放到图像视图中,因此我对 NSImageView 类进行了子类化以添加对这些类型的支持。
我选择 NSImageView 而不是普通视图的原因是,当用户将鼠标悬停在准备放置的文件上时,我还想显示动画(例如指向下并向上和向下的箭头)。我的问题是:最好的方法是什么(最有效、最快、最少的 CPU 使用率等)?
事实上,我已经做到了,但让我问这个问题的是,当我将图像设置为以低于 0.02 秒的速度变化时,它开始滞后。我是这样做的:
在 NSImageView 子类中:
- 有一个 ivar:NSTimer* animTimer;
- 覆盖 awakeFromNib,调用 [super awakeFromNib] 并使用 NSImage 将图像加载到数组中(大约 45 个图像)
- 每当用户输入文件时,以频率 = 0.025(越少则滞后)启动 animTimer,以及设置数组中下一个图像的选择器(称为 drawNextImage)
- 每当用户退出或结束拖放时,调用 [animTimer invalidate] 停止更新图像
这是我在子类中设置图像的方法:
- (void)drawNextImage
{
currentImageIndex++; // ivar / kNumberDNDImages is a constant defined as 46
if (currentImageIndex >= kNumberDNDImages) { currentImageIndex = 0;}
[super setImage: [imagesArray objectAtIndex: currentImageIndex]]; // imagesArray is ivar
}
那么,我怎样才能足够快地做到这一点呢?我希望频率约为 0.01 秒,但滞后小于 0.025,所以这就是我目前的设置。哦,我的图像大小正确(+或-一个像素或其他),并且它们是.png格式(我需要透明度 - 例如,jpeg就不能做到这一点)。
EDIT:
我尝试遵循 NSResponder 的建议,并将我的方法更新为:
- (void)drawNextImage
{
currentImageIndex++;
if (currentImageIndex >= kNumberDNDImages) { currentImageIndex = 0;}
NSRect smallImgRect;
smallImgRect.origin = NSMakePoint(kSmallImageWidth * currentImageIndex, [self.bigDNDImage size].height); // Up left corner - ??
smallImgRect.size = NSMakeSize(kSmallImageWidth, [self.bigDNDImage size].height);
// Bottom left corner - ??
NSPoint imgPoint = NSMakePoint(([self bounds].size.width - kSmallImageWidth) / 2, 0);
[bigDNDImage drawAtPoint: imgPoint fromRect: smallImgRect operation: NSCompositeCopy fraction: 1];
}
我还将此方法和其他拖放方法从 NSImageView 子类移至我已有的 NSView 子类。除了超类和此方法之外,一切都完全相同。我还修改了一些常量。
在我对此的早期测试中,我收到了一些错误/警告消息,这些消息并没有停止执行谈论 NSGraphicsContext 或其他内容。这些现在已经消失了,但只是让你知道。我完全不知道他们为什么出现以及他们的意思。如果他们再次出现,我会担心他们,而不是现在:)
EDIT 2:
这就是我现在正在做的事情:
- (void)drawNextImage
{
currentImageIndex++;
if (currentImageIndex >= kNumberDNDImages) { currentImageIndex = 0;}
[self drawCurrentImage];
}
- (void)drawCurrentImage
{
NSRect smallImgRect;
smallImgRect.origin = NSMakePoint(kSmallImageWidth * currentImageIndex, 0); // Bottom left, for sure
smallImgRect.size = NSMakeSize(kSmallImageWidth, [self.bigDNDImage size].height);
// Bottom left as well
NSPoint imgPoint = NSMakePoint(([self bounds].size.width - kSmallImageWidth) / 2, 0);
[bigDNDImage drawAtPoint: imgPoint fromRect: smallImgRect operation: NSCompositeCopy fraction: 1];
}
这里的问题是在调用drawRect时调用drawCurrentImage(看,它实际上比我想象的更容易解决)。
现在,我必须说我还没有对合成图像进行过尝试,因为我找不到一种好的、快速的方法来按照我想要的方式合并 40 多个图像(一个挨着另一个)。但对于那些不感兴趣的人,我修改了它以执行与 NSImageView 子类相同的操作(从数组中读取 40 多个图像并显示它们),并且我发现没有减速:NSView 与 NSImageView 一样滞后于 0.025 以下。另外,我在使用核心动画时发现了一些问题(图像绘制在奇怪的地方,而不是我告诉她的地方)和一些关于 NSGraphicsContext 的警告,我根本不知道如何解决(我是一个完整的人)当涉及到使用 Objective-C 工具进行绘图等方面时,我是菜鸟)。所以目前我正在使用 NSImageView,除非我找到一种方法来合并所有这些图像并尝试使用 NSView。