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

来源:互联网 发布:91助手 ios5软件 编辑:程序博客网 时间:2024/04/29 19:34

以前,作不规则窗体涉及到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,制作出异形控件。

原创粉丝点击