C#实现打造气泡屏幕保护效果

 更新时间:2016年10月19日 10:52:14   作者:李sir  
本文是介给大家介绍一个很好玩的小程序:气泡屏幕保护!类似于windows的屏保功能,有需要的朋友可以参考一下。

本文主要是介绍C#实现打造气泡屏幕保护效果,首先说一下制作要点:1 窗口要全屏置顶 2 模拟气泡的滚动和粘滞效果 3 支持快捷键ESC退出

大致就是这3个要点了,其他还有一些细节我们在程序中根据需要再看,OK,开工!

首先是全屏置顶,因为是屏幕保护嘛,这个简单,在窗体的属性设置里把FormBorderStyle设置为none表示无边框,把ShowInTaskbar设置为false表示不在任务栏出现,最后一个把WindowState设置为Maximized表示最大化即可,当然可以设置TopMost为true让窗口置顶,不过这个不是绝对的,如果有其他窗口也使用TopMost的话会让我们失去焦点,所以我们要注册一个快捷键让程序可以退出!

模拟气泡我们可以用Graphics类中的DrawEllipse方法来画一个圆,当然这个圆我们可以指定不同的颜色和大小,这里重点讲一下怎么模拟粘滞效果!

所谓粘滞效果相信大家到知道,胶体大家都见过吧?就是类似胶体那种有弹性并且可以在改变形状后回复原型的那种效果,当然这里要想模拟这个效果只能说是稍微类似,DrawEllipse方法中最后两个参数表示圆的大小,我们可以在这里做文章,由于循环的速度很快,我们只要动态改变圆的大小就可以产生类似粘滞的效果,当然这个改变大小的参数不能太大,否则就无效了!

我们在onpaint事件中写入如下代码来绘制一些圆:

Random ra = new Random(); //初始化随机数 
   bmp = new Bitmap(ClientSize.Width,ClientSize.Height, e.Graphics); 
   Graphics bmpGraphics = Graphics.FromImage(bmp); 
   // 绘制圆形      
  for (int i=1;i<=13;i++)//这里绘制13个圆形 
   { 
     bmpGraphics.DrawEllipse(new Pen(Color.FromName(colours[i]),2),//根据事先定义好的颜色绘制不同颜色的圆 
      ballarray[i, 1], ballarray[i, 2], 70+ra.Next(1, 10), 70+ra.Next(1, 10)); 
      //注意上面的最后两个参数利用随机数产生粘滞效果 
   } 
   e.Graphics.DrawImageUnscaled(bmp, 0, 0); 
   bmpGraphics.Dispose(); 
   bmp.Dispose();//这里是非托管的垃圾回收机制,避免产生内存溢出

这样,通过以上代码就可以绘制出一些不同颜色的具有粘滞效果的圆来模拟气泡

下面是注册系统热键,有个API函数RegisterHotKey可以完成系统快捷键的注册,使用他之前我们要先引用一个系统的DLL文件:USER32.DLL,然后对这个RegisterHotKey函数进行一下声明:

[DllImport("user32.dll")]//引用USER32.DLL 
public static extern UInt32 RegisterHotKey(IntPtr hWnd, UInt32 id, UInt32 fsModifiers, UInt32 vk); //声明函数原型

由于引用了一个DLL文件,我们不要忘了在文件头加入DLLImport的类声明using System.Runtime.InteropServices;然后在Form1的构造函数中来注册一个系统热键,这里我们注册ESC:RegisterHotKey(this.Handle, 247696411, 0, (UInt32)Keys.Escape); 通过以上步骤,我们就可以注册一个或多个系统热键,但是,注册系统热键后我们还不能立即使用,因为我们在程序中还无法对这个消息进行响应,我们重载一下默认的WndProc过程来响应我们的热键消息:

protected override void WndProc(ref Message m)//注意是保护类型的过程 
 { 
     const int WM_HOTKEY = 0x0312; 
 } 
    if (m.Msg == WM_HOTKEY & & m.WParam.ToInt32() == 247696411) //判断热键消息是不是我们设置的 
       { 
        Application.Exit();//如果消息等于我们的热键消息,程序退出 
      } 
    base.WndProc(ref m);//其他消息返回做默认处理

好了,通过以上一些步骤,我们就基本完成了这个屏幕保护程序的要点设计,其他的详细过程可以参考源码,程序运行的时候背景是透明的,这个也不难实现

1.this.BackColor = System.Drawing.Color.FromArgb(((int)(((byte)(255)))), ((int)(((byte)(192)))), ((int)(((byte)(192)))));
2.this.TransparencyKey = System.Drawing.Color.FromArgb(((int)(((byte)(255)))), ((int)(((byte)(192)))), ((int)(((byte)(192)))));

屏幕保护程序代码如下:

using System;
using System.Drawing;
using System.Collections;
using System.ComponentModel;
using System.Windows.Forms;
using System.Runtime.InteropServices;
/* 
 屏幕保护程序
 使用技术:系统热键,随机数,Graphics类绘制圆形
 编译环境:VisualStudio 2005
 运行要求:安装.net framework 2.0 框架
 其他:使用ESC退出
 
 说明:由于使用了循环控制图形位移,CPU占用在20%-30%左右
 程序具有自动垃圾回收机制避免造成内存溢出
 
 2009年3月15日
 */
namespace AnimatBall
{
  /// <summary>
  /// Summary description for Form1.
  /// </summary> 
  
  public class Form1 : System.Windows.Forms.Form
  { 

    public int[,] ballarray = new int[20,20];
    public string[] colours = new string[16];
  
    public Bitmap bmp;

    private System.Windows.Forms.Timer timer1;
    private System.ComponentModel.IContainer components; 
    [DllImport("user32.dll")]
    public static extern UInt32 RegisterHotKey(IntPtr hWnd, UInt32 id, UInt32 fsModifiers, UInt32 vk); //API
     //重写消息循环
    
    protected override void WndProc(ref Message m)
     {
       const int WM_HOTKEY = 0x0312;
      
       if (m.Msg == WM_HOTKEY && m.WParam.ToInt32() == 247696411) //判断热键
       {
         Application.Exit();
       }

       base.WndProc(ref m);
     }
    public Form1()
    {
      //
      // Required for Windows Form Designer support
      //
      InitializeComponent();
      //colours[0]="Red";
      colours[1]="Red";
      colours[2]="Blue";
      colours[3]="Black";
      colours[4]="Yellow";
      colours[5]="Crimson";
      colours[6]="Gold";
      colours[7]="Green";
      colours[8]="Magenta";
      colours[9]="Aquamarine";
      colours[10]="Brown";
      colours[11]="Red";
      colours[12]="DarkBlue";
      colours[13]="Brown";
      colours[14]="Red";
      colours[15]="DarkBlue";
      InitializeComponent();
      RegisterHotKey(this.Handle, 247696411, 0, (UInt32)Keys.Escape); //注册热键
      //
      // TODO: Add any constructor code after InitializeComponent call
      //
    } 

    /// <summary>
    /// Clean up any resources being used.
    /// </summary> 

    protected override void Dispose( bool disposing )
    {
      if( disposing )
      {
        if (components != null) 
        {
          components.Dispose();
        }
      }
      base.Dispose( disposing );
    } 

      #region Windows Form Designer generated code 

    /// <summary>
    /// Required method for Designer support - do not modify
    /// the contents of this method with the code editor.
    /// </summary> 

    private void InitializeComponent()
    {
      this.components = new System.ComponentModel.Container();
      this.timer1 = new System.Windows.Forms.Timer(this.components);
      this.SuspendLayout();
      // 
      // timer1
      // 
      this.timer1.Interval = 25;
      this.timer1.Tick += new System.EventHandler(this.timer1_Tick);
      // 
      // Form1
      // 
      this.AutoScaleBaseSize = new System.Drawing.Size(6, 14);
      this.BackColor = System.Drawing.Color.FromArgb(((int)(((byte)(255)))), ((int)(((byte)(192)))), ((int)(((byte)(192)))));
      this.ClientSize = new System.Drawing.Size(373, 294);
      this.DoubleBuffered = true;
      this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.None;
      this.Name = "Form1";
      this.ShowInTaskbar = false;
      this.Text = "小焱屏幕保护";
      this.TopMost = true;
      this.TransparencyKey = System.Drawing.Color.FromArgb(((int)(((byte)(255)))), ((int)(((byte)(192)))), ((int)(((byte)(192)))));
      this.WindowState = System.Windows.Forms.FormWindowState.Maximized;
      this.Load += new System.EventHandler(this.Form1_Load);
      this.ResumeLayout(false);

    }
      #endregion 
    /// <summary>
    /// The main entry point for the application.
    /// </summary> 

    [STAThread]
    static void Main() 
    {
      Application.Run(new Form1());
    } 

    private void timer1_Tick(object sender, System.EventArgs e)
    {
      
      for (int i=1;i<=13;i++)
      {   
        //add direction vectors to coordinates
        ballarray[i,1] = ballarray[i,1] + ballarray[i,3]; 
        ballarray[i,2] = ballarray[i,2] + ballarray[i,4]; 
        //if ball goes of to right
        if ((ballarray[i,1]+50)>=ClientSize.Width)
        {      
          ballarray[i,1]=ballarray[i,1]-ballarray[i,3];
          ballarray[i,3]=-ballarray[i,3];
        }
          //if ball goes off bottom
        else if ((ballarray[i,2]+50)>=ClientSize.Height)
        {
          ballarray[i,2]=ballarray[i,2]-ballarray[i,4];
          ballarray[i,4]=-ballarray[i,4];
        }
          //if ball goes off to left
        else if (ballarray[i,1]<=1)
        {
          ballarray[i,1]=ballarray[i,1]-ballarray[i,3];
          ballarray[i,3]=-ballarray[i,3];
        }         
          //if ball goes over top
        else if (ballarray[i,2]<=1)
        {
          ballarray[i,2]=ballarray[i,2]-ballarray[i,4];
          ballarray[i,4]=-ballarray[i,4];
        }
      }
      this.Refresh(); //force repaint of window
    }
    //Called from timer event when window needs redrawing
    protected override void OnPaint(PaintEventArgs e)
    {
      Random ra = new Random(); 
      bmp = new Bitmap(ClientSize.Width,ClientSize.Height, e.Graphics);
      Graphics bmpGraphics = Graphics.FromImage(bmp);
      // draw here        
      for (int i=1;i<=13;i++)
      {
        bmpGraphics.DrawEllipse(new Pen(Color.FromName(colours[i]),2),
          ballarray[i, 1], ballarray[i, 2], 70+ra.Next(1, 10), 70+ra.Next(1, 10));//利用随机数产生粘滞效果
      }
      e.Graphics.DrawImageUnscaled(bmp, 0, 0);
      //Draw ellipse acording to mouse coords.
      
      bmpGraphics.Dispose();
      bmp.Dispose();
    }      
    private void Form1_Load(object sender, EventArgs e)
    {
      Random r = new Random();
      //set ball coords and vectors x,y,xv,yv
      for (int i = 1; i <= 13; i++)
      {
        ballarray[i, 1] = +r.Next(10) + 1; //+1 means i lose zero values
        ballarray[i, 2] = +r.Next(10) + 1;

        ballarray[i, 3] = +r.Next(10) + 1;
        ballarray[i, 4] = +r.Next(10) + 1;
      }
      timer1.Start();
    }
   
  }
} 

TransparencyKey可以让窗体的某个颜色透明显示,我们只要把窗体的颜色和TransparencyKey的颜色设置一致就可以了,这里我设置的是粉红,注意最好设置的颜色是窗体所没有的,否则一旦匹配将会以透明显示!

效果如下:

相关文章

  • C# 使用EntityFramework CodeFirst 创建PostgreSQL数据库的详细过程

    C# 使用EntityFramework CodeFirst 创建PostgreSQL数据库的详细过程

    这篇文章主要介绍了C#使用EntityFramework CodeFirst创建PostgreSQL数据库的过程,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-07-07
  • C# 遍历枚举类型的所有元素

    C# 遍历枚举类型的所有元素

    写个小东西,刚好用到枚举类型,需要显示在DropDownList控件中。尝试了下,用如下方法可以实现
    2013-03-03
  • .NET使用IResourceMonitor实现获取资源信息

    .NET使用IResourceMonitor实现获取资源信息

    在 Microsoft.Extensions.Diagnostics.ResourceMonitoring 包提供了一系列定制 API,专用于监视 .NET 应用程序的资源利用率,本文将利用IResourceMonitor来实现获取资源状态信息,感兴趣的可以了解下
    2024-01-01
  • C#判断三角形的类型

    C#判断三角形的类型

    这篇文章主要介绍了C#判断三角形的类型的方法,通过输入三角形的三条边长,判断是否能构成一个三角形,感兴趣的小伙伴们可以参考一下
    2015-11-11
  • WPF微信聊天和通讯录按钮样式代码分享

    WPF微信聊天和通讯录按钮样式代码分享

    这篇文章主要为大家分享了WPF微信聊天和通讯录按钮样式代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-10-10
  • 使用C#实现一个简单的绘图工具

    使用C#实现一个简单的绘图工具

    这篇文章主要为大家详细介绍了如何使用C#开发的简单绘图工具,可以将签名简单绘图后的效果以图片的形式导出,有需要的小伙伴可以跟随小编一起学习一下
    2024-02-02
  • C#删除UL LI中指定标签里文字的方法

    C#删除UL LI中指定标签里文字的方法

    这篇文章主要介绍了C#删除UL LI中指定标签里文字的方法,涉及C#针对页面HTML元素进行正则匹配与替换的相关操作技巧,需要的朋友可以参考下
    2017-05-05
  • C#调用微信接口的相关代码

    C#调用微信接口的相关代码

    这篇文章主要为大家详细介绍了C#调用微信接口的相关代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-07-07
  • C#合并BitMap图像生成超大bitmap

    C#合并BitMap图像生成超大bitmap

    当两个图像合并的时候,以简单的使用gdi+,当需要将许多bitmap合并时就会造成宽度过大,那么怎么实现C#合并BitMap图像,本文就详细的介绍一下
    2021-11-11
  • C#使用Twain协议实现扫描仪连续扫描功能

    C#使用Twain协议实现扫描仪连续扫描功能

    这篇文章主要介绍了C#使用Twain协议实现扫描仪连续扫描,只需一行代码,就可实现一次扫描多张,且不需要更改扫描仪的任何设置,需要的朋友可以参考下
    2022-01-01

最新评论