gpt4 book ai didi

objective-c - 使用 ImageView 快速连续显示多张图像

转载 作者:行者123 更新时间:2023-12-03 16:38:43 24 4
gpt4 key购买 nike

我有一个应用程序,在一个窗口中,有一个 NSImageView。用户应该能够将任何文件/文件夹(不仅仅是图像)拖放到 ImageView 中,因此我对 NSImageView 类进行了子类化以添加对这些类型的支持。

我选择 NSImageView 而不是普通 View 的原因是,当用户将鼠标悬停在准备放置的文件上时,我还想显示动画(例如指向下并向上和向下的箭头)。我的问题是:最好的方法是什么(最有效、最快、最少的 CPU 使用率等)?

事实上,我已经做到了,但让我问这个问题的是,当我将图像设置为以低于 0.02 秒的速度变化时,它开始滞后。我是这样做的:

在 NSImageView 子类中:

  • 有一个 ivar:NSTimer* animTimer;
  • 覆盖awakeFromNib,调用[super awakeFromNib]并使用NSImage将图像加载到数组中(大约45张图像)
  • 每当用户输入文件时,启动 animTimer,频率 = 0.025(越少则滞后),并使用一个选择器设置数组中的下一个图像(称为 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 就无法做到这一点)。

编辑:

我尝试遵循 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 子类。除了父类(super class)和此方法之外,一切都完全相同。我还修改了一些常量。

在我对此的早期测试中,我收到了一些错误/警告消息,这些消息并没有停止执行谈论 NSGraphicsContext 或其他内容。这些现在已经消失了,但只是让你知道。我完全不知道他们为什么出现以及他们的意思。如果它们再次出现,我会担心它们,而不是现在:)

编辑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。

最佳答案

核心动画可能是最快的,因为它会在 GPU 上完成所有操作。为每个图像创建一个图层,将每个图层的 contents 设置为您可以从每个图像制作的 CGImage,将它们全部添加为单个顶级图层的子图层,host the top-level layer in a plain NSView ,然后依次切换每个图像层的 hidden 属性。

关于objective-c - 使用 ImageView 快速连续显示多张图像,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6541764/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com