[C#]使用Label标签控件模拟窗体标题的移动及窗体颜色不断变换

本文为原创文章、源代码为原创代码,如转载/复制,请在网页/代码处明显位置标明原文名称、作者及网址,谢谢!

开发工具:VS2017

语言:C#

DotNet版本:.Net FrameWork 4.0及以上

一、使用的WIN32 API有两个,一个为ReleaseCapture,另外一个为SendMessage,这两个函数说明如下:

ReleaseCapture函数:为了说明ReleaseCapture的用法,需要先知道SetCapture的用法,MSDN是这样描述SetCapture函数:

该函数在属于当前线程的指定窗口里设置鼠标捕获。一旦窗口捕获了鼠标,所有鼠标输入都针对该窗口,无论光标是否在窗口的边界内。同一时刻只能有一个窗口捕获鼠标。如果鼠标光标在另一个线程创建的窗口上,只有当鼠标键按下时系统才将鼠标输入指向指定的窗口。

而ReleaseCapture悄悄相反,函数的功能就是释放对鼠标的捕捉。

为什么要使用这个ReleaseCapture函数,原因在于移动窗体标题时,需要释放对鼠标的捕捉,否则,就不能移动窗体标题。

SendMessage函数:该函数是用来给窗体发送Windows消息,

在本文中,该函数是模拟给非窗体客户区域(如窗体标题、最大化、最小化及关闭按钮区域)发送Windows消息,使特定区域能收到拖动窗体标题的消息。

两者在C#的定义如下:

[DllImport("user32.dll")]
static extern void ReleaseCapture();
[DllImport("user32.dll")]
static extern void SendMessage(IntPtr hwnd, int msg, int wParam, int lParam);

关于hwnd、msg、wParam,lParam的具体说明,可以自行百度。

二、构建模拟移动窗体标题的应用程序,在这里我们使用了一个label(左边,label1),用来将鼠标移到该控件并拖动时,可以移动窗体,

另外一个label(右边,label2)则用来关闭窗体,如下图所示:

我们在label1的MouseDown事件写下如下代码:

 private void label1_MouseDown(object sender, MouseEventArgs e)
{
    ReleaseCapture();
     SendMessage(Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0);
}

其中,WM_NCLBUTTONDOWN(0x00A1),用来给非客户端发送左键按下消息,

HT_CAPTION(0x0002),为所需要作用的区域,此次为窗体标题。

我们在label2的MouseEnter、MouserLeave、Click事件写下如下代码:

private void label2_MouseEnter(object sender, EventArgs e)
{
    label2.BackColor = Color.FromArgb(255, 192, 191);
    toolTip1.SetToolTip(label2, "关闭");
}

private void label2_MouseLeave(object sender, EventArgs e)
{
    label2.BackColor = Color.Silver;
    toolTip1.SetToolTip(label2, "");
}

private void label2_Click(object sender, EventArgs e)
{
    this.Close();
}

其中,需要为关闭按钮填写显示“关闭”按钮的提示,因此需要使用ToolTip控件。

三、窗体颜色变换

这里主要用到了一个变换彩虹颜色的算法,参考如下:

public static Color Rainbow(float progress)
{
    var div = (Math.Abs(progress % 1) * 6);
    var ascending = (int)((div % 1) * 255);
    var descending = 255 - ascending;

    switch ((int)div)
    {
        case 0:
            return Color.FromArgb(255, 255, ascending, 0);
        case 1:
            return Color.FromArgb(255, descending, 255, 0);
        case 2:
            return Color.FromArgb(255, 0, 255, ascending);
        case 3:
            return Color.FromArgb(255, 0, descending, 255);
        case 4:
            return Color.FromArgb(255, ascending, 0, 255);
        default: 
            return Color.FromArgb(255, 255, 0, descending);
    }
}

可以借用一个while循环,内部再嵌套一个for循环,达到不断变换颜色的目的,如下代码所示:

private void ChangeColor()
{
    Task.Factory.StartNew(() => 
    {
        Color color;
        while (true)
        {
            try
            {
                for (float i = 0; i < 1f; i += 0.01f)
                {
                    color = Rainbow(i);
                    this.Invoke((Action)(() =>
                    {
                        BackColor = color;
                    }));
                    Thread.Sleep(150);
                }
            }
            catch { }
        }
    });
}

四、构建完的程序界面如下:

五、源代码及EXE如下:

源代码:

https://pan.baidu.com/s/1dEOhx8P

EXE:

https://pan.baidu.com/s/1o87QncY

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏移动端开发

常用开发技巧系列(四)

一:友盟的错误日志怎么看? 先说说友盟崩溃日志怎么查看的问题, 友盟统计我自己用的是比较多的,因为这个第三方的分享也是有的,就直接把友盟集成进去,统计和第三方...

2029
来自专栏程序员的诗和远方

Tips-移动端滑动固顶效果(position: sticky)

先放个图看看滑动固顶是啥效果: image.png 中间那个 tab 条,平常的时候是固定的,等到页面滑上去的时候,又像 fixed 一样贴在顶部。 pos...

2836
来自专栏施炯的IoT开发专栏

创建分辨率自适应的Windows Phone 8应用程序

1. 引言     Windows Phone 7平台只支持WVGA分辨率(480*800)的设备,这对于应用程序的UI设计来说是有利的,因为设计人员不用考虑多...

1897
来自专栏抠抠空间

jQuery实例

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <script s...

3038
来自专栏张高兴的博客

张高兴的 Xamarin.Forms 开发笔记:TapGestureRecognizer 的简单介绍与应用

3387
来自专栏小白鼠

Ionic3 高德Web定位

高德提供了Web平台定位的JS API,同样需要用到 APP_Key,并且需要注意是使用Web端的Key,如下图所示。必须是Web端的,其它平台的无效。 ht...

732
来自专栏谈补锅

web页面和小程序页面实现瀑布流效果

  2、小程序实现瀑布流,大致流程差不多。只不过小程序的图片的宽高度的获取没有web页面那么方便。

1883
来自专栏Thinks

【译】W3C WAI-ARIA最佳实践 -- 布局

面包屑包含当前页面的父页面的链接列表,该列表是层级顺序的。它可以帮助用户在网站或网络应用程序中找到自己的位置。面包屑通常水平放置在页面的主要内容之前。

9165
来自专栏林德熙的博客

win10 uwp win2d 入门 看这一篇就够了

本文主要翻译,可能带有一定的主观性和局限性,说的东西可能不对或者不符合每个人的预期。如果觉得我有讲的不对的,就多多包含,或者直接关掉这篇文章,但是请勿生气或者发...

642
来自专栏前端知识分享

第122天:移动端开发常见事件和流式布局

可以看到,在京东各个模块的主容器中,都设置了最大最小宽度和宽度100%,而在导航区块中,由于一行有5个小区块,所以设置了宽度为20%,使得小区块也能达到自适应的...

854

扫码关注云+社区