如何在C#游戏编程中保持动画不闪烁?

时间:2013-11-12 21:23:38

标签: c#

每当我尝试在此游戏中添加新线或移动球拍时,屏幕都会闪烁。

当我移动球拍或添加线条时,如何防止屏幕闪烁?

using System;

using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace WindowsFormsApplication2
{
public partial class GameTest : Form
{
    int dx=3, dy=3, i =500, o = 100;

   int rex = 400, rey = 450 ;
       double c =0;


    public GameTest()
    {

        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        Graphics g = this.CreateGraphics();
        Invalidate();
    }

    private void Form1_KeyDown(object sender, KeyEventArgs e)
    {
        Graphics g = this.CreateGraphics();
        if (e.KeyCode == Keys.Left)
        {

            //Graphics g = this.CreateGraphics();


            Invalidate();
            Brush black = new SolidBrush(Color.White);
            g.FillRectangle(black, rex, rey, 200, 20);

            rex -= 40;
            Brush red = new SolidBrush(Color.Green);
            g.FillRectangle(red, rex, rey, 200, 20);




        }
        if (e.KeyCode == Keys.Right)
        {


            //Graphics g = this.CreateGraphics();



            Brush white = new SolidBrush(Color.White);
            g.FillRectangle(white, rex, rey, 200, 20);

            rex += 40;
            Brush red = new SolidBrush(Color.Green);
            g.FillRectangle(red, rex, rey, 200, 20);

        }
    }



    private void Form1_Paint(object sender, PaintEventArgs e)
    {
        DoubleBuffered = false;

         Graphics g = this.CreateGraphics();
         //  Bitmap bmp1 = new Bitmap("G:\c#\Bouncing ball\Pic.jpg");
        //TextureBrush tb2 = new TextureBrush(bmp1);

        Brush green = new SolidBrush(Color.Green);
            g.FillRectangle(green, rex, rey, 200, 20);

            Graphics b= this.CreateGraphics();

            Brush red = new SolidBrush(Color.Red);

            b.FillEllipse(red, i, o, 20, 20);



       Pen p = new Pen(Color.Black, 10);
        //
           g.DrawLine(p, 1000, 480,0,480);
          g.DrawLine(p, 1000, 485, 1000, 0);

        g .DrawLine(p, 0,480, 0, 0);
    }


    private void timer1_Tick(object sender, EventArgs e)
    {
        DoubleBuffered = false;
        //  int dx=3, dy=3, i = 300, o = 50;

        i += dx;


        if (i < 0)
        {
            dx = -dx;
        }

        else if (i + 50 > 1000)
        {
            dx = -dx;
        }



        o += dy;

        if ((o +20>= rey) &&(i+20<=rex+200)&&(i+20>=rex))
        {
            //int rex = 400, rey = 450; RECTANGLE
            // int dx=3, dy=3, i(x) = 500, o(y) = 100;

            dy =-dy;
            //c++;
            //label1.Text = c.ToString();
        }


            // Misgeret\\
            if (o < 0)
            {
                dy = -dy;
            }
            // Misgeret\\
            else if (o + 50 > 600)
            {
                dy = -dy;
            }
           this.Invalidate();



    }

    private void label1_Click(object sender, EventArgs e)
    {
        label1.Text = c.ToString();
    }
}
}

1 个答案:

答案 0 :(得分:3)

我注意到您在程序的多个区域中将DoubleBuffered设置为false。这绝对没有用,因为使用双缓冲的全部原因是为了防止闪烁。

另一件事是您正在应用程序中的多个位置创建图形上下文和绘图。尝试重构代码以仅在表单的OnPaint()事件处理程序中绘制,并且不创建新的Graphics上下文。使用PaintEventArgs活动OnPaint()中提供的内容。

private void Form1_Paint(object sender, PaintEventArgs e)
{
    Graphics g = e.Graphics;

    using(Brush green = new SolidBrush(Color.Green))
    {
         g.FillRectangle(green, rex, rey, 200, 20);
    }

    // .. etc, etc..        
}

然后,当您需要重新绘制表单时,您只需要调用Invalidate()方法告诉GDI您的表单或其中的一部分需要重新绘制。这将导致Paint事件被触发并调用OnPaint()

作为旁注,正如@HighCore在评论中所建议的那样,WinForms绝对不是创建游戏的正确框架。对于游戏,请尝试使用XNA框架,或者网上提供的几种开源框架之一,例如{ {3}}

<强>更新

为防止闪烁,您可以为表单使用自动双缓冲。在调用InitializeComponent()后,可以使用对SetStyle的调用在表单构造函数中启用此功能:

 SetStyle( ControlStyles.AllPaintingInWmPaint
         | ControlStyles.UserPaint
         | ControlStyles.DoubleBuffer , true);