您的位置:首页 > 编程语言 > C#

转:用C#.net轻松制作不规则窗体

2010-07-20 10:22 316 查看
以前,作不规则窗体涉及到API的调用和大量的编程,不是谁都能作的。很多程序
员都望而却步。

    现在我们可以使用C#.net轻松的创建不规则窗体,下面我就用一个简单的例子来讲述其制作过程。

    1.绘制不规则窗体位图

    2.设置窗体基本属性

    3.编写窗体相关代码 (要实现窗口的关闭,移动等操作)  

    1.绘制不规则窗体位图

       
可以使用任意一种你喜欢的作图工具,制作一个有形状的位图,背景使用一种其他的颜色。这个颜色在编程中用得着,所以最好使用一种容易记忆的颜色。

如图下图,本例中使用的背景色为黄色(#ffff00/yellow),文件名为bk.bmp

2.创建windows窗体并设置窗体基本属性

        1>新建windows应用程序

2>选中新建的窗体,设置其相应属性:

                (1)。将 FormBorderStyle 属性设置为 None。

                (2)。将窗体的 BackgroundImage
属性设置为先前创建的位图文件。不必将文件添加到项目系统中;这将在指定该文件作为背景图像时自动完成。

                (3)。将 TransparencyKey
属性设置为位图文件的背景色,本例中为黄色。(此属性告诉应用程序窗体中的哪些部分需要设置为透明。 )

        
这时你就可以按F5测试你的程序,可以看到如图所示的窗体。现在窗体还不能拖动,只能通过结束程序,或者alt+F4关闭。下面我们编写相应的代码来实现
标题栏的相应功能。

3.编写窗体相关代码

        (要实现窗口的关闭,移动等操作)

        

        (1)。实现窗口关闭

                从工具栏中拖进一个按钮,设置其按钮文字为“×”,设置其大小为合适大小。双击该按钮进入其触发时间函数。

                写入如下代码:

                       

                this.Close();        //关闭本窗体

        (2)。设置窗体的移动操作,我们要用到两个全局的变量

                private Point mouseOffset;        //记录鼠标指针的坐标

                private bool isMouseDown = false; //记录鼠标按键是否按下

                创建该窗体 MouseDown事件的相应处理程序。

                private void Form1_MouseDown(object sender,
System.Windows.Forms.MouseEventArgs e)

                {

                        int xOffset;

                        int yOffset;

                        if (e.Button == MouseButtons.Left)

                        {

                                xOffset = -e.X -
SystemInformation.FrameBorderSize.Width;

                                yOffset = -e.Y -
SystemInformation.CaptionHeight -

                                       
SystemInformation.FrameBorderSize.Height;

                                mouseOffset = new Point(xOffset,
yOffset);

                                isMouseDown = true;

                        }

                }

                创建该窗体的 MouseMove事件的相应处理程序

                private void Form1_MouseMove(object sender,
System.Windows.Forms.MouseEventArgs e)

                {

                        if (isMouseDown)

                        {

                                Point mousePos = Control.MousePosition;

                                mousePos.Offset(mouseOffset.X,
mouseOffset.Y);

                                Location = mousePos;

                        }

                }

               

                创建该窗体的MouseUp事件的相应处理程序

                private void Form1_MouseUp(object sender,
System.Windows.Forms.MouseEventArgs e)

                {

                        // 修改鼠标状态isMouseDown的值

                        // 确保只有鼠标左键按下并移动时,才移动窗体

                        if (e.Button == MouseButtons.Left)

                        {

                                isMouseDown = false;

                        }

                }

        (3)。加入相应的其他的控件

                   其他的就是看你自己的需要,来添加控件,实现自己想要实现的功能。

                   本例中添加了一文本框,设置其背景为黄色,所以显示时也成了透明的。

       

         现在,我们就可以生成程序,看一下最后的效果了。

注意:如果监视器的颜色深度设置大于 24 位,则不管 TransparencyKey
属性是如何设置的,窗体的非透明部分都会产生显示问题。若要避免出现这种问题,请确保“显示”控制面板中的监视器颜色深度的设置小于 24
位。当开发具有这种透明功能的应用程序时,请牢记应使您的用户意识到此问题

制作异形窗体或控件的思路一般都是想办法生成一个region,然后设置给指定的窗口或控件。生成region的方法有很多,最常用的就是从一幅图
片生成,把该图片中的透明色部分“抠”掉,剩下的部分作为一个region。设置窗口或控件的region可以用SetWindowRgn
API,不过.NET
framework封装了这个操作,在C#中只要对窗口或控件的Region属性赋值就可以了。下面我就把我在C#中实现异形窗体的核心代码贴出来给大家
看看,有什么意见尽管提,别客气哦J

 

首先,是一个根据Bitmap对象生成Region的方法:

/// <summary>

/// 取得一个图片中非透明色部分的区域。

/// </summary>

/// <param name="Picture">取其区域的图片。</param>

/// <param name="TransparentColor">透明色。</param>

/// <returns>图片中非透明色部分的区域</returns>

private Region BmpRgn(Bitmap Picture, Color TransparentColor)

{

     int nWidth = Picture.Width;

     int nHeight = Picture.Height;

     Region rgn = new Region();

     rgn.MakeEmpty();

     bool isTransRgn;//前一个点是否在透明区

     Color curColor;//当前点的颜色

     Rectangle curRect = new Rectangle();

     curRect.Height = 1;

     int x = 0, y = 0;

     //逐像素扫描这个图片,找出非透明色部分区域并合并起来。

     for(y = 0; y < nHeight; ++y)

     {

         isTransRgn = true;

         for (x = 0; x < nWidth; ++x)

         {

              curColor = Picture.GetPixel(x,y);

              if(curColor == TransparentColor || x == nWidth -
1)//如果遇到透明色或行尾

                   {

                       if(isTransRgn == false)//退出有效区

                       {

                            curRect.Width = x - curRect.X;

                            rgn.Union(curRect);

                       }

                   }

                   else//非透明色

                   {

                       if(isTransRgn == true)//进入有效区

                       {

                            curRect.X = x;

                            curRect.Y = y;

                       }

                   }//if curColor

                   isTransRgn = curColor == TransparentColor;    

              }//for x

         }//for y

         return rgn;

     }

 

原理很简单,就是对该图片逐行扫描,在每一行中把那些非透明色的矩形(只有一个像素高)合并(union)到一个Region对象中,当扫描完整个
图片,得到的也就是我们想要的Region了。这种算法在很多文章里都有介绍的。

 

有了region,下面就简单了:

this.Region = BmpRgn(new Bitmap("d://a.bmp"), Color.FromArgb(0, 0,
0));

上面的代码就是把d:/a.bmp的轮廓作为主窗口的region的,假设该图片的背景黑色(Color.FromArgb(0, 0, 0))。

其实不光是Form,任何控件都可以用这个方法设置Region,制作出异形控件。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息