利用Visual C#实现任务栏通知窗口

visual

  想必大部分网友都使用过QQ、MSN等聊天程序,它们的界面都相当华丽,尤其是当网友上线以及消息提示时会有一个浮动的窗体从屏幕的右下方缓慢升起,既美观又人性化,作为程序员在享受的同时我们也不禁要问:这到底是怎么实现的呢?本文就利用Visual Studio .Net C# 2005以及.Net框架绘图技术来实现这种任务栏通知窗口。

  简介

  QQ和MSN的任务栏通知窗口很人性化,它可以在不丢失主窗体焦点的前提下显示一个具备皮肤Skin的通知窗体,当它显示一段时间后会自动消失,所以用户根本不用干预它。这样的通知窗体和一般的具备标题栏、系统图标和按钮的窗体没有太大的区别,窗体表面其实就是画上去的一张位图而已,而窗体的浮动则会复杂一点,我们会用到.Net框架的双重缓冲区绘图技术(参见作者编译文章"Windows 窗体的.Net框架绘图技术")来保证移动窗体时所显示的内容平滑且不闪烁,以及使用P/Invoke平台调用进行对Win32API函数的调用来完成不获得焦点的窗体显示和非标题栏窗体拖动。两种位图的皮肤运行时的界面如下:

  背景知识

  通知窗口就是将一般的窗体附加上一层皮肤,这里所谓的皮肤就是一张位图图片,该位图图片通过窗体的OnPaintbackground事件被绘制到窗体表面,在附加位图之前需要调整窗体的可视属性,由于绘制操作是针对于窗体客户区域的,所谓客户区域就是指窗体标题栏下方以及窗体边框以内的所有区域,所以需要将窗体的边框和外观属性 FormBorderStyle调整为:None,这样所绘制的图像就会填充整个窗体。

  首先,我们会用到Region对象,Region对象可以精确的描绘出任意形状的轮廓范围,通过一个位图图像创建Region对象后再将其传递给窗体的Region属性就可以使窗体按照Region所定义的轮廓显示出来。作为皮肤使用的位图文件可以通过任何图像编辑软件诸如:Photeshop来创建和编辑,只是注意一点,需要将图片的背景色调成特定颜色以便程序绘制时将其清除,我们在这里使用的背景色为粉红色。为了能够让Region对象按照图像中感兴趣的内容边框来创建窗体,我们还需要使用GraphicsPath类将图像轮廓按照一定路径标注下来,稍后便按照该路径创建Region对象。

  然后通过窗体的绘图事件将位图的内容显示在窗体表面,我们没有直接使用OnPaintbackground事件而是重载了该方法,这样做的好处就是一些低层的绘制操作还继续交由.Net框架运行时来处理,我们只考虑实际需要的绘制操作即可。在OnPaintbackground方法中我们启用了双重缓冲区绘图技术,所谓该技术就是指先在内存中的一块画布上把将要显示的图像显示出来或进行处理,等到操作完成再将该画布上所显示的图像放置到窗体表面,这样的机制可以非常有效的降低闪烁的出现,使图像显示更加平滑。通知窗体从屏幕的右下方进行升起停留一段时间后再慢慢回落,这里需要用到返回屏幕区域的大小范围的.Net框架方法 Screen.GetWorkingArea(WorkAreaRectangle),通过一定算法计算出通知窗体显示前的初始位置。最后,我们将要显示的文本按照一定格式和Rectangle对象所指定的区域范围绘制到窗体表面。通知窗体的关闭操作是通过设定一个区域,当用户用鼠标单击时检测单击坐标是否在该区域内,若在区域内就可以执行隐藏通知窗体的代码。

  我们注意了,当QQ和MSN的通知窗口显示时其主窗体的焦点没有丢失,也就是说程序没有将自身的焦点转移到显示的通知窗体上。经过测试,我们无论怎么样调用.Net框架提供的窗体显示例程譬如:Form.Show都无法保证主窗体的焦点不丢失,在VC环境下我们可以使用Win32API的ShowWindows函数来完成复杂的窗体显示操作,但是.Net框架根本没有提供类似的方法,那么我们能否通过.Net框架调用该API函数来显示窗体呢?幸好.Net框架提供了P/Invoke平台调用,利用平台调用这种服务,托管代码就可以调用在动态链接库中实现的非托管函数,并可以封送其参数,我们可以轻松的显示但不获得焦点的窗体。程序中用到的Windows API以及常量的定义都保存在WinUser.h头文件中,其对应的动态链接库文件就是user32.dll,使用.Net框架提供的DllImportAttribute类对导入的函数进行定义,然后就可以非常方便的在程序中调用该函数了。

  由于我们将通知窗体的标题栏隐藏了,所以对窗体拖动操作还需要我们自己动手进行处理。本文介绍了如何更加高效的进行拖动窗体操作,有些网友在对于非标题栏拖动窗体编程时偏向组合使用鼠标事件来进行,这样做的本质没有任何不妥,但是频繁的事件响应和处理反而使程序性能有所降低。我们将继续使用Win32API的底层处理方法来解决该问题,就是向窗体发送标题栏被单击的消息,模拟实际的拖动操作。

  我们会通过2个计时器来完成窗体的显示、停留和隐藏,通过设置速度变量可以改变窗口显示和隐藏的速度。

  程序实现

  启动Visual Studio .Net 2005,创建C# Windows 窗体应用程序,将解决方案命名为TaskbarForm,包含的项目名也为TaskbarForm,首先创建程序的主窗体Form1,在上面添加两个Button控件,一个用于显示通知窗体,另一个则终止程序。然后在解决方案管理器中右击项目,单击"添加 - Windows 窗体",我们把新创建的窗体命名为TaskbarForm。

  在类TaskbarForm定义的下方,我们创建用于显示的字符串和其颜色的变量,再定义几个Rectangle对象的变量用于放置标题、提示内容以及可以拖动窗体的区域和关闭按钮的区域。然后,我们需要保存窗体在浮动时的高度以便计算移动后的新高度,intervalValue变量用来确定窗体显示和隐藏的速度。进行平台调用时我们需要提前定义好常量的值用来传递给函数,WM_NCLBUTTONDOWN和HT_CAPTION常量用于拖动窗体,他们的值都保存在WinUser.h头文件中,所对应的动态链接库名为:user32.dll。我们用到的Win32API为:SendMessage、ReleaseCapture和ShowWindow,通过使用DllImportAttribute可以导入相应的函数并在程序中重新进行定义,如下:

[DllImportAttribute("user32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
//发送消息//winuser.h 中有函数原型定义
[DllImportAttribute("user32.dll")]
public static extern bool ReleaseCapture(); //释放鼠标捕捉winuser.h
[DllImportAttribute("user32.dll")] //winuser.h
private static extern Boolean ShowWindow(IntPtr hWnd, Int32 nCmdShow);
  SendMessage向消息循环发送标题栏被按下的消息来模拟窗体的拖动,ShowWindow用来将特定句柄的窗体显示出来,注意第二个参数nCmdShow,它表示窗体应该怎样显示出来,而我们需要窗体不获得焦点显示出来,SW_SHOWNOACTIVATE可以满足我们要求,继续在WinUser.h文件中搜索找到该常量对应的值为4,于是我们就可以这样调用来显示窗体了:

ShowWindow(this.Handle, 4);
  我们创建了一个自定义函数ShowForm用来封装上面的ShowWindow用来是显示窗体,同时传递了所用到的几个Rectangle矩形区域对象,最后调用ShowWindows函数将窗体显示出来,代码片段如下:

public void ShowForm(string ftitletext, string fcontenttext, Rectangle fRegionofFormTitle, Rectangle fRegionofFormTitlebar, Rectangle fRegionofFormContent, Rectangle fRegionofCloseBtn)
{
 titleText = ftitletext;
 contentText = fcontenttext;
 WorkAreaRectangle = Screen.GetWorkingArea(WorkAreaRectangle);
 this.Top = WorkAreaRectangle.Height + this.Height;
 FormBorderStyle = FormBorderStyle.None;
 WindowState = FormWindowState.Normal;
 this.SetBounds(WorkAreaRectangle.Width - this.Width, WorkAreaRectangle.Height - currentTop, this.Width,  this.Height);
 CurrentState = 1;
 timer1.Enabled = true;
 TitleRectangle = fRegionofFormTitle;
 TitlebarRectangle = fRegionofFormTitlebar;
 ContentRectangle = fRegionofFormContent;
 CloseBtnRectangle = fRegionofCloseBtn;
 ShowWindow(this.Handle, 4); //#define SW_SHOWNOACTIVATE 4
}
  CurrentState变量表示窗体的状态是显示中、停留中还是隐藏中,两个计时器根据窗体不同状态对窗体的位置进行更改,我们会使用SetBounds来执行该操作:

this.SetBounds(WorkAreaRectangle.Width - this.Width, WorkAreaRectangle.Height - currentTop, this.Width, this.Height);
  当窗体需要升起时将窗体的Top属性值不断减少,而窗体回落时将Top属性值增加并超过屏幕的高度窗体就消失了,虽然原理很简单但仍需精确控制。

  SetBackgroundBitmap函数首先将窗体背景图像保存到BackgroundBitmap变量中,然后根据该位图图像轮廓和透明色创建Region,BitmapToRegion就用于完成Bitmap到Region的转换,程序再将这个Region付值给窗体的Region属性以完成不规则窗体的创建。

public void SetBackgroundBitmap(Image image, Color transparencyColor)
{
 BackgroundBitmap = new Bitmap(image);
 Width = BackgroundBitmap.Width;
 Height = BackgroundBitmap.Height;
 Region = BitmapToRegion(BackgroundBitmap, transparencyColor);
}

public Region BitmapToRegion(Bitmap bitmap, Color transparencyColor)
{
 if (bitmap == null)
  throw new ArgumentNullException("Bitmap", "Bitmap cannot be null!");

 int height = bitmap.Height;
 int width = bitmap.Width;
 GraphicsPath path = new GraphicsPath();
 for (int j = 0; j < height; j++)
  for (int i = 0; i < width; i++)
  {
   if (bitmap.GetPixel(i, j) == transparencyColor)
    continue;
   int x0 = i;
   while ((i < width) && (bitmap.GetPixel(i, j) != transparencyColor))
    i++;
   path.AddRectangle(new Rectangle(x0, j, i - x0, 1));
  }
  Region region = new Region(path);
  path.Dispose();
  return region;
} 
  通知窗体背景以及文字的绘制在重载的OnPaintBackground方法中完成,而且利用了双重缓冲区技术来进行绘制操作,代码如下:

protected override void OnPaintBackground(PaintEventArgs e)
{
 Graphics grfx = e.Graphics;
 grfx.PageUnit = GraphicsUnit.Pixel;
 Graphics offScreenGraphics;
 Bitmap offscreenBitmap;
 offscreenBitmap = new Bitmap(BackgroundBitmap.Width, BackgroundBitmap.Height);
 offScreenGraphics = Graphics.FromImage(offscreenBitmap);
 if (BackgroundBitmap != null)
 {
  offScreenGraphics.DrawImage(BackgroundBitmap, 0, 0, BackgroundBitmap.Width, BackgroundBitmap.Height);
 }
 DrawText(offScreenGraphics);
 grfx.DrawImage(offscreenBitmap, 0, 0);
}
  上述代码首先返回窗体绘制表面的Graphics并保存在变量grfx中,然后创建一个内存Graphics对象offScreenGraphics和内存位图对象offscreenBitmap,将内存位图对象的引用付值给offScreenGraphics,这样所有对offScreenGraphics的绘制操作也都同时作用于offscreenBitmap,这时就将需要绘制到通知窗体表面的背景图像BackgroundBitmap绘制到内存的Graphics对象上,DrawText函数根据需要显示文字的大小和范围调用Graphics.DrawString将文字显示在窗体的特定区域。最后,调用Graphics.DrawImage将内存中已经绘制完成的图像显示到通知窗体表面。

  我们还需要捕获窗体的鼠标操作,有三个操作在这里进行,1、处理拖动窗体操作,2、处理通知窗体的关闭操作,3、内容区域的单击操作。三个操作都需要检测鼠标的当前位置与每个Rectangle区域的包含关系,只要单击落在特定区域我们就进行相应的处理,代码如下:

private void TaskbarForm_MouseDown(object sender, MouseEventArgs e)
{
 if (e.Button == MouseButtons.Left)
 {
  if (TitlebarRectangle.Contains(e.Location)) //单击标题栏时拖动
  {
   ReleaseCapture(); //释放鼠标捕捉
   SendMessage(Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0); //发送左键点击的消息至该窗体(标题栏)
  }
  if (CloseBtnRectangle.Contains(e.Location)) //单击Close按钮关闭
  {
   this.Hide();
   currentTop = 1;
  }
  if (ContentRectangle.Contains(e.Location )) //单击内容区域
  {
   System.Diagnostics.Process.Start("http://www.Rithia.com");
  }
 }
}
  结论

  该程序可以很好的进行通知窗体的显示、停留和隐藏操作,并且具备简单的换肤机制,在利用了双重缓冲区绘图技术后,可以保证窗体的绘制平滑且没有闪烁。

时间: 2024-11-08 17:28:56

利用Visual C#实现任务栏通知窗口的相关文章

利用Visual C#开发一个媒体播放器

visual|媒体 摘要:了解如何在 Microsoft Visual C# .NET 中使用 DirectShow 控件,如何开发一个媒体播放器.按照本文介绍的操作步骤,您可以创建一个简单 Visual C# 应用程序,用来播放数字音频和视频. 简介 Microsoft Visual C# 是世界上最流行的编程语言,利用 Visual C# 的最新版本 Visual C# .NET,您能够快速.有效地开发基于 Windows 窗体的应用程序,还可以为嵌入了 Microsoft Windows

Win7系统任务栏通知区域图标无法修改怎么办

  最近有用户反应在win7系统中无法修改任务栏通知区域图标,这给我们设置操作系统带来一定麻烦,之前也介绍过"利用360安全卫士来解决通知区域图标无法显示"的问题,但有些用户不想使用360安全卫士,其实我们只要对win7 64位旗舰版进行相关设置就可以了. 解决方法/步骤: 1.打开电脑之后,鼠标右键直接点击任务栏空白地方,选择里面的"属性"选项; 2.这时候就会弹出一个属性窗口,在"任务栏"选项卡下面,单击通知区域下面的"自定义&qu

如何利用网页弹出各种形式的窗口

如何利用网页弹出各种形式的窗口,我想大家大多都是知道些的,但那种多种多样的弹出式窗口是怎么搞出来的,我们今天就来学习一下:  1.弹启一个全屏窗口  <html>  <body onload="window.open('http://www.pconline.com.cn','example01','fullscreen');">;  <b>www.e3i5.com</b>  </body>  </html>  2

自定义Win7任务栏通知区域

某天,小白运行QQ时,系统提示他已经登陆了,可是看屏幕右下角,却并没有QQ图标,难道是在别的电脑上登陆了?自己的QQ号被盗了?后来小白发现,原来自己已经运行了QQ,只是QQ图标被隐藏起来了,要点击Win7任务栏通知区域的小三角,才能展开隐藏区域,看到QQ图标.知道QQ号没有被盗,小白放心了,不过图标被隐藏到Win7任务栏通知区域后,有好友发来消息时,就看不到头像闪烁了,每次都要点击小三角才能打开消息窗口,有点麻烦. 查看Win7任务栏通知区域隐藏的图标 其实Win7任务栏通知区域的图标,哪些隐藏

如何利用 Visual Studio 自带工具提高开发效率

原文:如何利用 Visual Studio 自带工具提高开发效率 Visual Stuido 是一款强大的Windows 平台集成开发工具,你是否好好地利用了它呢?   显示行号 有些时候(比如错误定位)的时候,显示行号将有利于我们进行快速定位.   如何显示 1. 工具 / 选项 / 文本编辑器 -> 选择对应的语言 2. 勾选 "行号"       使用书签 和平常意义的书签类似,当我们希望在日后某一时刻快速定位到一处代码时使用.比如在项目例会上,你需要演示本周你所做的一些改

视窗标题显示问题-visual 2012 单文档所窗口切换时如何显示同一个标题名称

问题描述 visual 2012 单文档所窗口切换时如何显示同一个标题名称 大家好!请教一个问题:单文档多视窗切换时,如果在不同的视窗中始终显示同一个标题名称呢?采用下述方式时m_pMainWnd->SetWindowText(_T(""系统2012"")); 当切换多其它窗口时,在app文件中的名称就变成其它了,而不再是"系统2012",谢谢! 解决方案 把SetWindowText()加在View::OnDraw(CDC* pDC)里试

《Visual C++ 开发从入门到精通》——1.3 利用Visual C++ 6.0编写C++程序

1.3 利用Visual C++ 6.0编写C++程序 知识点讲解:光盘视频PPT讲解(知识点)第1章利用Visual C++ 6.0编写C++程序.mp4 实例003 编写.调试和运行一个标准的C++程序源码路径 光盘daimapart 01 视频路径 光盘视频实例第1章003 本实例的功能是,使用Visual C++6.0编写.调试并运行一个标准的C++程序.本实例的具体实现流程如下. (1)选择File→New命令,在New对话框中选择Win32 Console Application项,

解决Win7系统任务栏通知区域图标无法修改问题

1.我们在电脑桌面然后右击桌面空白处,之后再弹出的菜单中点击"属性"选项,效果如下所示; 2.然后在任务栏和开始菜单属性中我们点击"任务栏"在切换面板再点击"自定义"按键,效果如下; 3.在系统弹出的窗口里面,把"始终在任务栏上显示所有图标和通知"选项取消勾选,这时候就能够自由设置图标的显示状态了. 好了以上就是小编给各位介绍的关于任务栏通知上的图标不能修改了,其实就是我们一个设置问题了,当然win7好像可以直接解锁的哦.

阻止MSN弹出通知窗口的设置方法

  当在win7系统中运行全屏程序时,可以把MSN Messenger的工作状态设置为"忙碌",并阻止通知窗口的弹出. 第1步:在Messenger的主窗口中执行"工具"-"选项"命令,弹出"选项"窗口,选择"个人信息"选项卡, 第2步:找到"我的状态",选中复选框"当我运行全屏程序时(如幻灯片放映),将我的状态显示为'忙碌'并阻止通知."即可,如图所示.