我目前正在从头开始为 XNA 游戏创建一个窗口系统。我主要针对 Windows 进行开发,但谁知道我将来可能支持哪些平台。如果您知道本机 Direct3D 的这一点,请随意回答,因为性能语义应该类似。如果可能,请考虑如果目标平台是 X-Box 360 会发生什么变化。
我取得了很好的进展,但现在我不确定如何准确渲染窗口。我想出了四种方法:
只需将所有控件直接渲染到屏幕上即可。这就是我现在所做的。只要控件不是半透明的,就可以通过状态之间的混合来设置动画。我没有找到在任意数量的状态之间进行动画处理的好方法(假设一个按钮当前正在从按钮按下到按钮按下以及从鼠标移出到鼠标悬停进行动画处理,然后它被禁用。它应该从最后一个状态平滑地混合到新状态。使用这种方法,只有在最后一个动画完成后播放一个动画时才有效,否则动画会发生跳跃。
将每个顶级窗口和所有控件渲染到渲染目标中,然后使用它以半透明的方式将顶级窗口渲染到屏幕上。这使得顶层工作半透明并且易于管理,但不会改变动画的情况。
将每个控件渲染到渲染目标中,该目标仅在控件变脏时更新(即必须进行动画处理或文本已更改)。这样,每个控件的半透明就可以了。
与前面一样,但除了解决动画问题之外,每个控件还有第二个渲染目标。每当动画开始时,交换渲染目标,以便我们拥有动画开始时的状态,并将其与目标状态混合到另一个渲染目标中。与前一种方法相比,这不会增加开销,我们的渲染目标数量只是原来的两倍,在任何给定的帧中,只有一个目标会被渲染(最多)。但问题来了:为了让它工作,我需要让“旧”渲染目标保留其内容。这在 Windows 上应该具有良好的性能,但似乎对 X-Box 360 具有严重的性能影响。另一方面,只有当动画处于活动状态时才需要“保留”位。
真正的问题来了。任何澄清的内容都是受欢迎的。对于性能问题,请记住,这只是游戏的窗口系统 - 后面的游戏可能会使用许多渲染目标并也会降低性能,而且可能比窗口系统还要多。假设在绝对最坏的情况下,我们可能有 5 个顶级窗口,每个窗口在屏幕上有 20-40 个控件。
- 您会推荐其中哪种方法(如果有)?为什么?当然,请随意添加另一种方法。
- 如果只有 200 或 400 个可用渲染目标,并且每帧中可能只有 20 个渲染目标,那么是否会对性能产生影响?
- PreserveContents 对 X-Box 360 的性能影响真的那么严重吗? Windows 上的情况有多糟糕?
- 可以写入 RenderTarget2D.RenderTargetUsage 属性。在运行时切换此选项以仅根据需要启用 PreserveContents 是一个好主意吗?
- 您(作为玩家)是否介意控制动画在某些情况下会跳跃,例如将鼠标悬停在按钮上,将鼠标移出然后再次移入,因此“正常->悬停”动画从一开始就播放两次,因为它是比你慢?
如果您正在为 Xbox 360 进行开发,则必须小心渲染目标。 Xbox 360 有一个特殊的内存 (10MB) 来保存渲染目标,包括用于在屏幕上渲染的目标。
只要不超过 10MB,即使使用 PreserveContents,从一个渲染目标切换到另一个渲染目标也不会产生影响,因为所有渲染目标都存储在这一特殊内存中。
但是,当您使用 PreserveContents 拥有超过 10MB 的渲染目标时,必须通过不断地从正常内存切换渲染目标来模拟此属性。
因此,渲染目标的数量并不像这些目标的总大小那么重要。您可以使用以下公式了解渲染目标的大小:
size (bits) = width x height x color data size (bits)
Xbox 360 上的颜色数据大小为 32 位,因此,举个例子,如果您的库的用户正在开发他的高清游戏,除了主目标之外没有其他渲染目标,他将使用:
3,515625MB = 29491200 bits = 1280 x 760 x 32 bits
另外,您应该避免动态创建渲染目标。成本太高了。您应该创建一个在游戏开始时分配的静态渲染目标池,并让 GUI 组件请求它们。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)