当前位置: 首页 > 编程笔记 >

C# Winform中如何绘制动画示例详解

何向荣
2023-03-14
本文向大家介绍C# Winform中如何绘制动画示例详解,包括了C# Winform中如何绘制动画示例详解的使用技巧和注意事项,需要的朋友参考一下

前言

这里介绍一个.net自身携带的类ImageAnimator,这个类类似于控制动画的时间轴,使用ImageAnimator.CanAnimate可以判断一个图片是否为动画,调用ImageAnimator.Animate可以开始播放动画,即每经过一帧的时间触发一次OnFrameChanged委托,我们只要在该委托中将Image的活动帧选至下一帧再迫使界面重绘就可以实现动画效果了。

为了方便以后的使用,我将这些代码整合到了一起,形成一个AnimateImage类,该类提供了CanAnimate、FrameCount、CurrentFrame等属性,以及Play()、Stop()、Reset()等动画常用的方法,代码如下

using System;  
using System.Collections.Generic;  
using System.Text;  
using System.Drawing;  
using System.Drawing.Imaging;  
 
namespace GifTest  
{  
  /// <summary>  
  /// 表示一类带动画功能的图像。  
  /// </summary>  
  public class AnimateImage  
  {  
    Image image;  
    FrameDimension frameDimension;  
    /// <summary>  
    /// 动画当前帧发生改变时触发。  
    /// </summary>  
    public event EventHandler<EventArgs> OnFrameChanged;  
 
    /// <summary>  
    /// 实例化一个AnimateImage。  
    /// </summary>  
    /// <param name="img">动画图片。</param>  
    public AnimateImage(Image img)  
    {  
      image = img;  
 
      lock (image)  
      {  
        mCanAnimate = ImageAnimator.CanAnimate(image);  
        if (mCanAnimate)  
        {  
          Guid[] guid = image.FrameDimensionsList;  
          frameDimension = new FrameDimension(guid[0]);  
          mFrameCount = image.GetFrameCount(frameDimension);  
        }  
      }  
    }  
 
    bool mCanAnimate;  
    int mFrameCount = 1, mCurrentFrame = 0;  
 
    /// <summary>  
    /// 图片。  
    /// </summary>  
    public Image Image  
    {  
      get { return image; }  
    }  
 
    /// <summary>  
    /// 是否动画。  
    /// </summary>  
    public bool CanAnimate  
    {  
      get { return mCanAnimate; }  
    }  
 
    /// <summary>  
    /// 总帧数。  
    /// </summary>  
    public int FrameCount  
    {  
      get { return mFrameCount; }  
    }  
 
    /// <summary>  
    /// 播放的当前帧。  
    /// </summary>  
    public int CurrentFrame  
    {  
      get { return mCurrentFrame; }  
    }  
 
    /// <summary>  
    /// 播放这个动画。  
    /// </summary>  
    public void Play()  
    {  
      if (mCanAnimate)  
      {  
        lock (image)  
        {  
          ImageAnimator.Animate(image, new EventHandler(FrameChanged));  
        }  
      }  
    }  
 
    /// <summary>  
    /// 停止播放。  
    /// </summary>  
    public void Stop()  
    {  
      if (mCanAnimate)  
      {  
        lock (image)  
        {  
          ImageAnimator.StopAnimate(image, new EventHandler(FrameChanged));  
        }  
      }  
    }  
 
    /// <summary>  
    /// 重置动画,使之停止在第0帧位置上。  
    /// </summary>  
    public void Reset()  
    {  
      if (mCanAnimate)  
      {  
        ImageAnimator.StopAnimate(image, new EventHandler(FrameChanged));  
        lock (image)  
        {  
          image.SelectActiveFrame(frameDimension, 0);  
          mCurrentFrame = 0;  
        }  
      }  
    }  
 
    private void FrameChanged(object sender, EventArgs e)  
    {  
      mCurrentFrame = mCurrentFrame + 1 >= mFrameCount ? 0 : mCurrentFrame + 1;  
      lock (image)  
      {  
        image.SelectActiveFrame(frameDimension, mCurrentFrame);  
      }  
      if (OnFrameChanged != null)  
      {  
        OnFrameChanged(image, e);  
      }  
    }  
  }  
}

使用如下方法调用:

using System;  
using System.Collections.Generic;  
using System.ComponentModel;  
using System.Data;  
using System.Drawing;  
using System.Drawing.Imaging;  
using System.Text;  
using System.Windows.Forms;  
 
namespace GifTest  
{  
  public partial class Form1 : Form  
  {  
    AnimateImage image;  
 
    public Form1()  
    {  
      InitializeComponent();  
      image = new AnimateImage(Image.FromFile(@"C:\Documents and Settings\Administrator\My Documents\My Pictures\未命名.gif"));  
      image.OnFrameChanged += new EventHandler<EventArgs>(image_OnFrameChanged);  
      SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint | ControlStyles.UserPaint, true);  
    }  
 
    void image_OnFrameChanged(object sender, EventArgs e)  
    {  
      Invalidate();  
    }  
 
    private void Form1_Load(object sender, EventArgs e)  
    {  
      image.Play();  
    }  
 
    private void Form1_Paint(object sender, PaintEventArgs e)  
    {  
      lock (image.Image)  
      {  
        e.Graphics.DrawImage(image.Image, new Point(0, 0));  
      }  
    }  
 
    private void button1_Click(object sender, EventArgs e)  
    {  
      if (button1.Text.Equals("Stop"))  
      {  
        image.Stop();  
        button1.Text = "Play";  
      }  
      else 
      {  
        image.Play();  
        button1.Text = "Stop";  
      }  
      Invalidate();  
    }  
 
    private void button2_Click(object sender, EventArgs e)  
    {  
      image.Reset();  
      button1.Text = "Play";  
      Invalidate();  
    }  
  }  
}

总结

到此这篇关于C# Winform中如何绘制动画的文章就介绍到这了,更多相关C# Winform绘制动画内容请搜索小牛知识库以前的文章或继续浏览下面的相关文章希望大家以后多多支持小牛知识库!

 类似资料:
  • 虽然canvas的API并未直接提供支持动画的方法,但就其本身而言,在canvas中实现动画效果也很简单:只需要持续的更新并重绘画布就行了。这种持续的更新并重绘就叫做动画循环,它是所有动画的核心逻辑。 在canvas中实现动画,首先需要初始化画布上的对象。然后,启动一个动画循环来更新画布、清除画布、重绘画布,再请求下一个新的动画帧。Canvas动画的基本原理如图 4‑36 所示: 图4-36 ca

  • 我编写了这段代码,可以在JavaFX画布上绘制。它可以很好地工作,但我不知道如何重新绘制画布(比如在Swing中),以便在新画布上重新开始绘制。这是我的代码,非常感谢你的帮助!马里奥

  • 动画、时间轴事件监听 本文将向您介绍如何使用 BindingX 来实现动画。如果您还不了解 BindingX 的工作原理,强烈建议先阅读文档 《简介》以及 《核心概念》。 特性介绍 要在 BindingX 中使用动画,需要将eventType的值设置为 timing 。在timing模式下, BindingX 提供了一个预置变量叫 t ,可以参与表达式运算,它代表的含义是动画流逝的时间,从0开始,

  • 我是Android新手,需要建议。我有一个带有多个ImageView的GridLayout。每个ImageView都有一个可绘制的背景色。在单击按钮时,我想根据用户的输入设置两件事的动画:1)将整个视图移动到新位置(这一部分很清晰,不会造成问题),2)仅将图像的可绘制部分移动到新单元格,将背景色保留在原始位置。我完全被困在第二项任务上。如何使用动画移动可绘制的内容?谢谢你的帮助。

  • 问题内容: 这是我的代码,我无法跳出while(!done)函数 使用XFlush(d)可以显示显示窗体缓冲区,并且在XCloseDisplay(d)之前不会消失 我想像这样画画线 g ++ -o youname youcppname -lX11 问题答案: 我不太确定为什么XFlush不起作用,但这是事件循环和冲洗的混合体。它等待暴露事件,绘制字符串并退出循环。 5秒钟后,它将取消映射窗口,然后

  • 本文向大家介绍C# WinForm调用Shell_NotifyIcon的示例代码,包括了C# WinForm调用Shell_NotifyIcon的示例代码的使用技巧和注意事项,需要的朋友参考一下 用法示例: 以上就是C# WinForm调用Shell_NotifyIcon的示例代码的详细内容,更多关于C# WinForm调用Shell_NotifyIcon的资料请关注呐喊教程其它相关文章!