c#中WinForm中拖拽窗体实现移动功能(无边框模态窗体)

在WindowsForm 应用程序中,如果将窗体的FormBorderStyle属性设置为none,这时,用鼠标拖拽窗体时就无法实现移动的功能了,下面就是解决方案.

在FormBordeStyle属性设置为none的窗体的后台代码中添加以下代码:

 [DllImport("user32.dll")]
        public static extern bool ReleaseCapture();
        [DllImport("user32.dll")]
        public static extern bool SendMessage(IntPtr hwnd, int wMsg, int wParam, int IParam);
        public const int WM_SYSCOMMAND = 0x0112;
        public const int SC_MOVE = 0xF010;
        public const int HTCAPTION = 0x0002;

该处为补充内容(用于解释上述代码中用到的API函数及要用到的参数常量):

(1)上述代码中用到了两个常用的API函数SendMessage和ReleaseCapture,介绍一下这两个函数。

ReleaseCapture函数的作用是为当前的应用程序释放鼠标的捕获

SendMessage函数的作用是调用一个窗口的窗口函数,将一条消息命令发给那个窗口。需要注意的是,这里所说的窗口不是.Net中的窗体(Form),而是相当于.Net中的控件。当然,System.Windows.Forms.Form也是一个Control.

SendMessage函数的声明及参数作用

SendMessage(IntPtr hwnd, int wMsg, int wParam, int lParam);

hwnd,要接收消息的那个窗口的句柄

wMsg,消息的标识符

wParam,具体取决于发送的消息

lParam,具体取决于发送的消息

(2)参数常量

WM_SYSCOMMAND常量代表要向窗口发送消息,SC_MOVE常代表要向窗口发送移动的消息,HTCAPTION 指的是 标题栏
##以下程序主要是将鼠标点击标题栏的消息发送给窗口,来实现移动窗口的功能##

当然你需要引入 DllImport  相对应的命名空间 :

using System.Runtime.InteropServices;

这还不算完,前提有了,然后我们就需要调用了;既然是拖拽,那么我们就在窗体的 MouseDown 事件中调用相应的函数,代码如下:

    private void Form1_MouseDown(object sender, MouseEventArgs e) {
            ReleaseCapture();
            SendMessage(this.Handle, WM_SYSCOMMAND, SC_MOVE + HTCAPTION, 0);
        }
SendMessage(this.Handle, WM_SYSCOMMAND, SC_MOVE + HTCAPTION, 0);//发送的消息里并没有移动位置等参数,可能是消息过程收到此消息后自己去系统里找

以上这样就可以实现我们想要的功能了!

这时,有的人就会问了,如果我设置了窗体的FormBordeStyle属性设置为none,但是在这个窗体的最上边有一个Panel挡住了。我怎么点击是移动呢?

这好办,在Panel 的 MouseDown 事件中调用如上所示的函数代码,代码如下:

   private void panel1_MouseDown(object sender, MouseEventArgs e) {
            ReleaseCapture();
            SendMessage(this.Handle, WM_SYSCOMMAND, SC_MOVE + HTCAPTION, 0);
        }

这样就可以实现了!其它控件上如果也希望实现这种功能,与上面同理!在此不多讲!

/*********C# 自定义窗体移动窗体的方法*******/

自定义窗体一般会隐藏任务栏
一般在C# 编写窗体时,都需要自己编写窗体让软件更美观一点。一般窗体在隐藏任务栏后都不会移动了,所以要做一些移动处理。
下面演示代码:
private const int WM_NCHITTEST = 0x84;
        private const int HTCLIENT = 0x1;
        private const int HTCAPTION = 0x2;
        //在Form1中改写鼠标消息
        protected override void WndProc(ref Message m)
        {
            switch (m.Msg)
            {
                case WM_NCHITTEST:
                    base.WndProc(ref m);
                    if ((int)m.Result == HTCLIENT)
                        m.Result = (IntPtr)HTCAPTION;
                    return;
                    break;
            }
            base.WndProc(ref m);
        }

/***********源码实例**************/

 #region 无边框拖动效果
        [DllImport("user32.dll")]//拖动无窗体的控件
        public static extern bool ReleaseCapture();
        [DllImport("user32.dll")]
        public static extern bool SendMessage(IntPtr hwnd, int wMsg, int wParam, int lParam);
        public const int WM_SYSCOMMAND = 0x0112;
        public const int SC_MOVE = 0xF010;
        public const int HTCAPTION = 0x0002;
        
        private void Start_MouseDown(object sender, MouseEventArgs e)
        {
            //拖动窗体
            ReleaseCapture();
            SendMessage(this.Handle, WM_SYSCOMMAND, SC_MOVE + HTCAPTION, 0);
        }
        #endregion

/***************Win32无边框窗体拖动、改变大小(WM_SYSCOMMAND方式)************/

拖动
对于win32无边框窗体的拖动功能,网上一般找到的是通过:MoveWindow
函数实现。当然,也能搜出另一种更好的实现,
通过:

SendMessage(hwnd,WM_SYSCOMMAND,SC_MOVE|HTCAPTION,0);
实现。相比于第一种,第二种更加友好,因为它能响应windows的手势操作(比如,晃动几下窗口,桌面所有窗口最小化)。

改变大小
事实上,改变大小也可以通过MoveWindow实现,但是并不友好,而且也麻烦,既然WM_SYSCOMMAND具有系统亲和性,那能不能通过这个消息来改变窗口大小呢?经过查阅,发现该消息的wParam参数可以是SC_SIZE参数。表面意思就是改变大小,但是第一次测试的时候并没有起作用,后来才发现与SC_MOVE类似,需要与另一个子参数进行OR运算,这一系列子参数以WMSZ_打头。比如,改变最左边的为WMSZ_LEFT,其他7个方向类似。所以更好的改变窗口大小的消息为:

SendMessage(hwnd,WM_SYSCOMMAND,SC_SIZE|WMSZ_方向,0);
例如,改变左边的消息为:

SendMessage(hwnd,WM_SYSCOMMAND,SC_SIZE|WMSZ_LEFT,0);
所有8个方向的名称为:

WMSZ_LEFT
WMSZ_RIGHT
WMSZ_TOP
WMSZ_TOPLEFT
WMSZ_TOPRIGHT
WMSZ_BOTTOM
WMSZ_BOTTOMLEFT
WMSZ_BOTTOMRIGHT
(提示:WM_SYSCOMMAND的wParam的低4位为系统内部使用)

/*******************通过控件位置对控件拖动*********************/

1 //在picturebox鼠标移动
2 private void picBox_MouseMove(object sender, MouseEventArgs e)
3 {
4     if (MoveFlag)
5     {
6         picBox.Left += Convert.ToInt16(e.X - xPos);//设置x坐标.
7         picBox.Top += Convert.ToInt16(e.Y - yPos);//设置y坐标.
8     }
9 }

复制代码

在需要移动的控件的MouseMove事件添加以上代码

1 //在picturebox的鼠标按下事件里.
2 private void picBox_MouseUp(object sender, MouseEventArgs e)
3 {
4     MoveFlag = false;
5 }

在需要移动的控件的MouseUp事件添加以上代码

1 //在picturebox的鼠标按下事件里,记录三个变量.
2 private void picBox_MouseDown(object sender, MouseEventArgs e)
3 {
4     MoveFlag = true;//已经按下.
5     xPos = e.X;//当前x坐标.
6     yPos = e.Y;//当前y坐标.
7 }

在需要移动的控件的MouseDown事件添加以上代码

声明需要的变量

记录x坐标: int xPos;
记录y坐标: int yPos;
记录是否按下鼠标: bool MoveFlag;

 

  • 5
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值