0

/我正在使用并在使用以下设备构建的计算机上进行测试:{1 GB RAM(现为 1.5 GB)、1.7 GHz Intel Pentium 处理器、ATI Mobility Radeon X600 GFX}

我需要缩放/转换控件并使其流畅。目前,我每 24-33 毫秒(30 fps)±3 像素处理一次控件的大小和位置。当我向图像添加“淡入淡出”效果时,它会平滑地淡入淡出,但它的大小只有 25x25 像素。控件的大小为 450x75 像素到 450x250 像素。在宝石迷阵 3 等 2D 游戏中,精灵的动画没有断断续续的动画。

所以正如标题所暗示的那样:在处理器上哪个更容易/更快:动画位图(在动画期间将其渲染到父控件)或动画控件它自己?

编辑: 嘿,我认为这是一个很有帮助的社区,而不是一个对似乎没有挑战性的问题进行评分的社区!(而且我在这里看到了更荒谬的问题,评分也更高!)请先给我留言,然后再对我的问题进行负面评价!

4

1 回答 1

1

我设法在我的日程安排中找到了一些空闲时间,以快速启动一个新项目。我敢肯定我的时间本可以花在其他地方更好,但希望我的鞋子里的其他人可能会发现这个有用...

答案是:图片优于控件。将位图渲染到画布上时,几乎不会触发任何事件(如果有的话)。至于控件,它充满了事件——一些链接,一些循环,以及递归的添加,所以一个简单的“LocationChanged”事件甚至不会覆盖引擎盖下实际发生的一半。

对于在运行时应用了很多动态动画的控件,我会做的是开发一个两部分集:控件 [渲染] 模板或活动界面(用于当控件处于静止状态或播放之前动画的),以及具有基本定义属性的动画结构,例如显示图像[渲染的控件]、矩形边界以及可能在以后应用的任何动画算法。

编辑:根据要求,这里是之前和之后的代码示例:

// This is the triggering event of the translating animation
private void object_Click(object sender, EventArgs e)
{
      // the starting point is at (75,75)
      element.Transform(new Point(500, 250));
}

前:

public class ControlElement : UserControl
{
      private Timer tick;
      private Point pT0;
      public ControlElement() : base()
      {
            tick = new Timer();
            tick.Interval = 30; // about 30fps
            tick.Tick += new EventHandler(tick_Tick);
      }
      void tick_Tick(object sender, EventArgs e)
      {
            // get the new point from distance and current location/destination
            this.Location = Utils.Transform(this.Location, pT0, 3);
            if ((pT0.X - this.Location.X)+(pT0.Y - this.Location.Y) <= 0)
            {
                this.Location = pT0;
                tick.Stop();
                //this.Visible = true;
            }
      }
      public void Transform(Point destination)
      {
            pT0 = destination;
            //this.Visible = false;
            tick.Start();
      }
}

DrawToBitmap之后:我创建了一个类,其中包含使用该功能的控件外观的图片。它仍然包含与上面相同的动画方法。我不得不添加LocationandLocationChanged元素,因为这个类不再是一个控件。如果并且当需要访问实际控件时,我将停止渲染并显示它自身的控件实例。
这是渲染调用:

void element_LocationChanged(object sender, EventArgs e)
{
      canvas.Invalidate();
}
void canvas_Paint(object sender, PaintEventArgs e)
{
      if (element != null)
      {
            Bitmap bmp = new Bitmap(element.Display);
            Pen p = new Pen(Color.FromArgb(128, 128, 128), 1);
            e.Graphics.DrawImage(bmp, element.Location);
            e.Graphics.DrawRectangle(p, 
                 element.Location.X, element.Location.Y, 
                 bmp.Width, bmp.Height);
      }
}
于 2012-10-18T00:58:29.980 回答