• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号
您的位置:首页 > 程序设计 >C#教程 > C#中如何使用Winform实现炫酷的透明动画界面

C#中如何使用Winform实现炫酷的透明动画界面

作者: 字体:[增加 减小] 来源:互联网 时间:2017-05-28

通过本文主要向大家介绍了c语言,欲情 c max,维生素c,奔驰c200,85度c等相关知识,希望对您有所帮助,也希望大家支持linkedu.com www.linkedu.com

做过.NET Winform窗体美化的人应该都很熟悉UpdateLayeredWindow吧,UpdateLayeredWindow可以实现窗体的任意透明,效果很好,不会有毛边。不过使用这个API之后,会有一个问题就是无法使用普通控件,而且没有Paint消息。为了解决这个问题,有两种方法。

一、使用双层窗体,底层窗体使用UpdateLayeredWindow作为背景,上层窗体用普通窗体,并且可以使用TransparencyKey或者Region来实现去除不需要的窗体内容,让上层窗体能看到底层的窗体。

二、直接单层窗体,使用控件的DrawToBitmap把控件图像绘制到UpdateLayeredWindow 的窗体上,这样就可以看到普通控件了。不过这个也有问题:1.控件内容不能自动更新 2.效率低,很多控件使用DrawToBitmap绘制出的图像不完整,甚至绘制不出图像。比如TextBox无法显示光标,WebBrowser无法 显示内容。

三、采用DirectUI技术,重写所有基础控件。效果最好,不过工作量巨大。

使用UpdateLayeredWindow时,一般是需要对Bitmap缓存起来,通过设置剪辑区域,局部重绘来提高效率。另外还可以异步重绘,模拟Winform的失效到重绘。

有些人会说为什么不直接用WPF啊,Wpf和Winform各有优缺点,适应不同的场合。Winform相对于使用更简单一些,系统要求更低。当然需要看人的习惯了和擅长的。

UpdateLayeredWindow 基本使用方法:

protected  override CreateParams CreateParams 
      { 
       get 
         { 
         CreateParams cp =  base .CreateParams; 
         cp.ExStyle |=  0x00080000 ; // WS_EX_LAYERED 扩展样式 
         return cp; 
       } 
     } 
</div>

重写窗体的 CreateParams 属性

API调用:

public  void SetBitmap(Bitmap bitmap, byte opacity) 
  { 
   if (bitmap.PixelFormat != PixelFormat.Format32bppArgb) 
     throw  new ApplicationException( "位图必须是32位包含alpha 通道" ); 
 
  IntPtr screenDc = Win32.GetDC(IntPtr.Zero); 
  IntPtr memDc = Win32.CreateCompatibleDC(screenDc); 
  IntPtr hBitmap = IntPtr.Zero; 
  IntPtr oldBitmap = IntPtr.Zero; 
 
   try 
     { 
    hBitmap = bitmap.GetHbitmap(Color.FromArgb( 0 ));  // 创建GDI位图句柄,效率较低 
    oldBitmap = Win32.SelectObject(memDc, hBitmap); 
 
    Win32.Size size =  new Win32.Size(bitmap.Width, bitmap.Height); 
    Win32.Point pointSource =  new Win32.Point( 0 , 0 ); 
    Win32.Point topPos =  new Win32.Point(Left, Top); 
    Win32.BLENDFUNCTION blend =  new Win32.BLENDFUNCTION(); 
    blend.BlendOp       = Win32.AC_SRC_OVER; 
    blend.BlendFlags      =  0 ; 
    blend.SourceConstantAlpha = opacity; 
    blend.AlphaFormat     = Win32.AC_SRC_ALPHA; 
 
    Win32.UpdateLayeredWindow(Handle, screenDc, ref topPos, ref size, memDc, ref pointSource, 0 , ref blend, Win32.ULW_ALPHA); 
  } 
   finally 
     { 
    Win32.ReleaseDC(IntPtr.Zero, screenDc); 
     if (hBitmap != IntPtr.Zero) 
       { 
      Win32.SelectObject(memDc, oldBitmap); 
       
      Win32.DeleteObject(hBitmap); 
    } 
    Win32.DeleteDC(memDc); 
  } 
} 
</div>

API声明:

class Win32 
  { 
   public  enum Bool 
    { 
    False =  0 , 
    True 
  } ; 
 
 
  [StructLayout(LayoutKind.Sequential)] 
   public  struct Point 
     { 
     public Int32 x; 
     public Int32 y; 
 
     public Point(Int32 x, Int32 y) 
     { this .x = x; this .y = y; } 
  } 
 
 
  [StructLayout(LayoutKind.Sequential)] 
   public  struct Size 
     { 
     public Int32 cx; 
     public Int32 cy; 
 
     public Size(Int32 cx, Int32 cy) 
      { this .cx = cx; this .cy = cy; } 
  } 
 
 
  [StructLayout(LayoutKind.Sequential, Pack =  1 )] 
   struct ARGB 
    { 
     public  byte Blue; 
     public  byte Green; 
     public  byte Red; 
     public  byte Alpha; 
  } 
 
 
  [StructLayout(LayoutKind.Sequential, Pack =  1 )] 
   public  struct BLENDFUNCTION 
     { 
     public  byte BlendOp; 
     public  byte BlendFlags; 
     public  byte SourceConstantAlpha; 
     public  byte AlphaFormat; 
  } 
 
 
   public  const Int32 ULW_COLORKEY =  0x00000001 ; 
   public  const Int32 ULW_ALPHA =  0x00000002 ; 
   public  const Int32 ULW_OPAQUE =  0x00000004 ; 
 
   public  const  byte AC_SRC_OVER =  0x00 ; 
   public  const  byte AC_SRC_ALPHA =  0x01 ; 
 
 
  [DllImport( " user32.dll " , ExactSpelling =  true , SetLastError =  true )] 
   public  static  extern Bool UpdateLayeredWindow(IntPtr hwnd, IntPtr hdcDst, ref Point pptDst, ref Size psize, IntPtr hdcSrc, ref Point pprSrc, Int32 crKey, ref BLENDFUNCTION pblend, Int32 dwFlags); 
 
  [DllImport( " user32.dll " , ExactSpelling =  true , SetLastError =  true )] 
   public  static  extern IntPtr GetDC(IntPtr hWnd); 
 
  [DllImport( " user32.dll " , ExactSpelling =  true )] 
   public  static  extern  int ReleaseDC(IntPtr hWnd, IntPtr hDC); 
 
  [DllImport( " gdi32.dll " , ExactSpelling =  true , SetLastError =  true )] 
   public  static  extern IntPtr CreateCompatibleDC(IntPtr hDC); 
 
  [DllImport( " gdi32.dll " , ExactSpelling =  true , SetLastError =  true )] 
   public  static  extern Bool DeleteDC(IntPtr hdc); 
 
  [DllImport( " gdi32.dll " , ExactSpelling =  true )] 
   public  static  extern IntPtr SelectObject(IntPtr hDC, IntPtr hObject); 
 
  [DllImport( " gdi32.dll " , ExactSpelling =  true , SetLastError =  true )] 
   public  static  extern Bool DeleteObject(IntPtr hObject); 
 
  [DllImport( " user32.dll " , EntryPoint =  " SendMessage " )] 
   public  static  extern  int SendMessage( int hWnd, int wMsg, int wParam, int lParam); 
  [DllImport( " user32.dll " , EntryPoint =  " ReleaseCapture " )] 
 
   public  static  extern  int ReleaseCapture(); 
   public  const  int WM_SysCommand =  0x0112 ; 
   public  const  int SC_MOVE =  0xF012 ; 
 
   public  const  int SC_MAXIMIZE =  61488 ; 
   public  const  int SC_MINIMIZE =  61472 ; 
} 
</div>

需要呈现图像的时候调用 SetBitmap 方法。只要优化好,呈现效率比普通的Paint重绘方式高很多,并且不卡不闪烁,支持任意透明。

下面是自己开发出来的效果:

这个是用OpenGL绘制的

效果是不是很酷呀,通过以上内容的介绍,希望对大家的学习有所帮助。

</div>
分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • 详解C#中三个关键字params,Ref,out
  • 浅谈C#中的for循环与foreach循环
  • 关于C#中ajax跨域访问问题
  • 使用C#编写15子游戏
  • 为IObservable实现自己的运算符(详解)
  • 使用C#正则表达式获取必应每日图片地址
  • C#七大经典排序算法系列(上)
  • C#无损压缩图片
  • C# BackgroundWorker用法详解
  • 详解三种C#实现数组反转方式

相关文章

  • 2017-05-28C#实现动态生成静态页面的类详解
  • 2017-05-28c#使用微信接口开发微信门户应用
  • 2017-05-28C#子线程更新UI控件的方法实例总结
  • 2017-05-28c#中文转unicode字符示例分享
  • 2017-05-28C# Dynamic关键字之:dynamic为什么比反射快的详解
  • 2017-05-28关于finalize机制和引用、引用队列的用法详解
  • 2017-05-28Silverlight将图片转换为byte的实现代码
  • 2017-05-28深入解析C#中的泛型类与泛型接口
  • 2017-05-28c#闭包使用方法示例
  • 2017-05-28c#几种数据库的大数据批量插入(SqlServer、Oracle、SQLite和MySql)

文章分类

  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号

最近更新的内容

    • C#求n个数中最大值和最小值的方法
    • DevExpress设置饼状图的Lable位置实例
    • 英雄联盟辅助lol挂机不被踢的方法(lol挂机脚本)
    • C# BackgroundWorker组件学习入门介绍
    • WinForm特效之桌面上的遮罩层实现方法
    • C#创建、读取和修改Excel的方法
    • c# 快速排序算法
    • C#操作Access通用类实例
    • C# 执行bat批处理文件的小例子
    • C#中动态数组用法实例

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有