在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 {4if (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 {4MoveFlag = false;5 }
在需要移动的控件的MouseUp事件添加以上代码
1 //在picturebox的鼠标按下事件里,记录三个变量.2 private void picBox_MouseDown(object sender, MouseEventArgs e)3 {4MoveFlag = true;//已经按下.5xPos = e.X;//当前x坐标.6yPos = e.Y;//当前y坐标.7 }
在需要移动的控件的MouseDown事件添加以上代码
声明需要的变量
记录x坐标: int xPos;
记录y坐标: int yPos;
记录是否按下鼠标: bool MoveFlag;