首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >CATiledLayer显示了以前的瓷砖

CATiledLayer显示了以前的瓷砖
EN

Stack Overflow用户
提问于 2016-10-06 08:43:24
回答 1查看 912关注 0票数 4

当使由CATiledLayer支持的视图无效时,以前的瓷砖仍然是“卡住”的,不能正确地失效。

这似乎发生在视图失效时(在主线程上),同时,平铺呈现线程仍在处理以前版本的瓦片。而不是缓存新版本的瓷砖,前一个版本是缓存。

由CATiledLayer支持的视图是UIScrollView的子视图,并且是可缩放的。砖块的渲染可能很昂贵,并且可以使用渲染线程10 of。

示例

演示此问题的示例代码:https://github.com/Q42/CATiledLayerBug

  1. 在一个CATiledLayer中,开始渲染所有的红色瓷砖(这大约需要3秒才能完成)
  2. 每个呈现步骤大约需要10。
  3. 在呈现期间(800 the之后),使完整视图无效:tiledView.setNeedsDisplay()
  4. 开始渲染所有灰色的瓷砖(再次花费大约3秒)
  5. 两个瓷砖(随机?)保持红色,而不是变成灰色。

参见此处的update函数:https://github.com/Q42/CATiledLayerBug/blob/master/TiledLayerTest/ViewController.swift#L45

解决办法?

这似乎是CATiledLayer实现中的一个缺陷。既然我不能解决这个问题,有没有人知道解决这个问题的好办法?

我已经准备好雷达了:http://www.openradar.me/28648050

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2016-10-17 17:41:59

基于我在示例项目中添加的一些进一步的日志记录,我认为问题是:

CATiledLayer有两个呈现线程,用于绘制每个平铺。如果在执行draw(_: CGRect)调用期间调用setNeedsDisplay,则完成draw调用的当前执行并缓存结果。缓存的值是基于以前的“数据源”(本例中只是瓷砖颜色),而不是更新的数据源。

一位苹果支持工程师为我提供了一个解决办法:

  • updateID字段添加到TiledView
  • 添加draw(_: CGRect)调用的开始保存当前的updateID
  • 当“数据源”更改时,更改updateID
  • 添加draw(_: CGRect)调用的末尾,将保存的updateID与当前的updateID进行比较。
  • 如果ids不同,则安排一个新的setNeedsDisplay调用。

摘录:

代码语言:javascript
复制
override func draw(_ rect: CGRect) {
  let originalID = updateID

  // all actual (slow) drawing code here...

  if originalID != updateID {

    // dispatch a redraw request, but wait a little while first
    DispatchQueue.main.asyncAfter(deadline: .now() + .milliseconds(17)) {
      self.layer.setNeedsDisplayIn(rect)
    }
  }
}
票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/39891468

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档