Jawaban:
Ini artikel tentang CodeProject Rincian teknik. Pada dasarnya intinya adalah:
public const int WM_NCLBUTTONDOWN = 0xA1;
public const int HT_CAPTION = 0x2;
[System.Runtime.InteropServices.DllImport("user32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
[System.Runtime.InteropServices.DllImport("user32.dll")]
public static extern bool ReleaseCapture();
private void Form1_MouseDown(object sender, System.Windows.Forms.MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
ReleaseCapture();
SendMessage(Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0);
}
}
Ini pada dasarnya melakukan hal yang sama persis dengan mengambil bilah judul jendela, dari sudut pandang pengelola jendela.
Form1_MouseDown
tidak ditetapkan ke MouseDown
acara sebenarnya Form1
.
this.MouseDown += ...
ke Main()
fungsi untuk formulir
Jangan membuat segalanya lebih sulit dari yang mereka butuhkan. Saya telah menemukan begitu banyak potongan kode yang memungkinkan Anda menyeret formulir (atau Kontrol lain). Dan banyak dari mereka memiliki kekurangan / efek sampingnya sendiri. Terutama yang saat mereka mengelabui Windows dengan berpikir bahwa Kontrol pada formulir adalah bentuk sebenarnya.
Karena itu, inilah cuplikan saya. Saya menggunakannya sepanjang waktu. Saya juga ingin mencatat bahwa Anda tidak boleh menggunakan this.Invalidate (); seperti yang suka dilakukan orang lain karena dalam beberapa kasus menyebabkan formulir berkedip. Dan dalam beberapa kasus, hal ini juga terjadi. Menggunakan this.Update, saya tidak mengalami masalah berkedip:
private bool mouseDown;
private Point lastLocation;
private void Form1_MouseDown(object sender, MouseEventArgs e)
{
mouseDown = true;
lastLocation = e.Location;
}
private void Form1_MouseMove(object sender, MouseEventArgs e)
{
if(mouseDown)
{
this.Location = new Point(
(this.Location.X - lastLocation.X) + e.X, (this.Location.Y - lastLocation.Y) + e.Y);
this.Update();
}
}
private void Form1_MouseUp(object sender, MouseEventArgs e)
{
mouseDown = false;
}
Cara lain yang lebih sederhana untuk melakukan hal yang sama.
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
// set this.FormBorderStyle to None here if needed
// if set to none, make sure you have a way to close the form!
}
protected override void WndProc(ref Message m)
{
base.WndProc(ref m);
if (m.Msg == WM_NCHITTEST)
m.Result = (IntPtr)(HT_CAPTION);
}
private const int WM_NCHITTEST = 0x84;
private const int HT_CLIENT = 0x1;
private const int HT_CAPTION = 0x2;
}
gunakan MouseDown, MouseMove dan MouseUp. Anda dapat mengatur bendera variabel untuk itu. Saya punya sampel, tapi menurut saya Anda perlu merevisi.
Saya mengkodekan tindakan mouse ke panel. Setelah Anda mengklik panel, formulir Anda akan ikut bergerak.
//Global variables;
private bool _dragging = false;
private Point _offset;
private Point _start_point=new Point(0,0);
private void panel1_MouseDown(object sender, MouseEventArgs e)
{
_dragging = true; // _dragging is your variable flag
_start_point = new Point(e.X, e.Y);
}
private void panel1_MouseUp(object sender, MouseEventArgs e)
{
_dragging = false;
}
private void panel1_MouseMove(object sender, MouseEventArgs e)
{
if(_dragging)
{
Point p = PointToScreen(e.Location);
Location = new Point(p.X - this._start_point.X,p.Y - this._start_point.Y);
}
}
WPF saja
tidak memiliki kode yang tepat untuk ditangani, tetapi dalam proyek baru-baru ini saya pikir saya menggunakan acara MouseDown dan sederhananya ini:
frmBorderless.DragMove();
Ini telah diuji dan mudah dimengerti.
protected override void WndProc(ref Message m)
{
switch (m.Msg)
{
case 0x84:
base.WndProc(ref m);
if((int)m.Result == 0x1)
m.Result = (IntPtr)0x2;
return;
}
base.WndProc(ref m);
}
WM_NCHITTEST
menyamar.
Tidak ada properti yang dapat Anda balik untuk mewujudkannya secara ajaib. Lihatlah kejadian untuk formulir dan itu menjadi cukup sepele untuk menerapkan ini dengan menetapkan this.Top
dan this.Left
. Secara khusus, Anda ingin melihat MouseDown
, MouseUp
dan MouseMove
.
public Point mouseLocation;
private void frmInstallDevice_MouseDown(object sender, MouseEventArgs e)
{
mouseLocation = new Point(-e.X, -e.Y);
}
private void frmInstallDevice_MouseMove(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
Point mousePos = Control.MousePosition;
mousePos.Offset(mouseLocation.X, mouseLocation.Y);
Location = mousePos;
}
}
ini bisa menyelesaikan masalahmu ....
Sedikit kode dari tautan di atas melakukan trik dalam kasus saya :)
protected override void OnMouseDown(MouseEventArgs e)
{
base.OnMouseDown(e);
if (e.Button == MouseButtons.Left)
{
this.Capture = false;
Message msg = Message.Create(this.Handle, 0XA1, new IntPtr(2), IntPtr.Zero);
this.WndProc(ref msg);
}
}
Cara terbaik yang pernah saya temukan (dimodifikasi tentu saja)
// This adds the event handler for the control
private void AddDrag(Control Control) { Control.MouseDown += new System.Windows.Forms.MouseEventHandler(this.DragForm_MouseDown); }
public const int WM_NCLBUTTONDOWN = 0xA1;
public const int HT_CAPTION = 0x2;
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern bool ReleaseCapture();
private void DragForm_MouseDown(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
ReleaseCapture();
SendMessage(Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0);
// Checks if Y = 0, if so maximize the form
if (this.Location.Y == 0) { this.WindowState = FormWindowState.Maximized; }
}
}
Untuk menerapkan seret ke kontrol cukup masukkan ini setelah InitializeComponent ()
AddDrag(NameOfControl);
Itu berhasil untuk Aku.
private void Form1_MouseDown(object sender, MouseEventArgs e)
{
_mouseLoc = e.Location;
}
private void Form1_MouseMove(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
int dx = e.Location.X - _mouseLoc.X;
int dy = e.Location.Y - _mouseLoc.Y;
this.Location = new Point(this.Location.X + dx, this.Location.Y + dy);
}
}
Untuk .NET Framework 4,
Anda bisa menggunakan this.DragMove()
untuk MouseDown
acara komponen (mainLayout dalam contoh ini) yang Anda gunakan untuk menyeret.
private void mainLayout_MouseDown(object sender, MouseButtonEventArgs e)
{
this.DragMove();
}
Cara termudah adalah:
Pertama buat label bernama label1. Buka acara label1> acara mouse> Label1_Mouse Pindah dan tulis ini:
if (e.Button == MouseButtons.Left){
Left += e.X;
Top += e.Y;`
}
Saya mencoba membuat bentuk jendela tanpa batas yang dapat dipindahkan yang berisi kontrol Host Elemen WPF dan kontrol Pengguna WPF.
Saya berakhir dengan panel tumpukan yang disebut StackPanel di kontrol pengguna WPF saya yang sepertinya hal logis untuk dicoba klik untuk bergerak. Mencoba kode junmats berhasil saat saya menggerakkan mouse perlahan, tetapi jika saya menggerakkan mouse lebih cepat, mouse akan keluar dari formulir dan formulir akan macet di suatu tempat di tengah gerakan.
Ini memperbaiki jawabannya untuk situasi saya menggunakan CaptureMouse dan ReleaseCaptureMouse dan sekarang mouse tidak bergerak dari formulir saat memindahkannya bahkan jika saya memindahkannya dengan cepat.
private void StackPanel_MouseDown(object sender, MouseButtonEventArgs e)
{
_start_point = e.GetPosition(this);
StackPanel.CaptureMouse();
}
private void StackPanel_MouseUp(object sender, MouseButtonEventArgs e)
{
StackPanel.ReleaseMouseCapture();
}
private void StackPanel_MouseMove(object sender, MouseEventArgs e)
{
if (StackPanel.IsMouseCaptured)
{
var p = _form.GetMousePositionWindowsForms();
_form.Location = new System.Drawing.Point((int)(p.X - this._start_point.X), (int)(p.Y - this._start_point.Y));
}
}
//Global variables;
private Point _start_point = new Point(0, 0);
Karena beberapa jawaban tidak memungkinkan kontrol anak dapat diseret, saya telah membuat kelas pembantu kecil. Ini harus melewati formulir tingkat atas. Dapat dibuat lebih umum jika diinginkan.
class MouseDragger
{
private readonly Form _form;
private Point _mouseDown;
protected void OnMouseDown(object sender, MouseEventArgs e)
{
_mouseDown = e.Location;
}
protected void OnMouseMove(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
int dx = e.Location.X - _mouseDown.X;
int dy = e.Location.Y - _mouseDown.Y;
_form.Location = new Point(_form.Location.X + dx, _form.Location.Y + dy);
}
}
public MouseDragger(Form form)
{
_form = form;
MakeDraggable(_form);
}
private void MakeDraggable(Control control)
{
var type = control.GetType();
if (typeof(Button).IsAssignableFrom(type))
{
return;
}
control.MouseDown += OnMouseDown;
control.MouseMove += OnMouseMove;
foreach (Control child in control.Controls)
{
MakeDraggable(child);
}
}
}
Selain itu, jika Anda perlu DoubleClick dan membuat Formulir Anda lebih besar / lebih kecil, Anda dapat menggunakan Jawaban pertama, buat variabel int global, tambahkan 1 setiap kali pengguna mengklik komponen yang Anda gunakan untuk menyeret. Jika variable == 2
kemudian buat formulir Anda lebih besar / lebih kecil. Juga gunakan timer untuk setiap setengah detik atau satu detik untuk membuat variable = 0
;
Menambahkan MouseLeftButtonDown
event handler ke MainWindow berhasil untuk saya.
Dalam fungsi acara yang dibuat secara otomatis, tambahkan kode di bawah ini:
base.OnMouseLeftButtonDown(e);
this.DragMove();
Saya memperluas solusi dari jay_t55 dengan satu metode lagi ToolStrip1_MouseLeave
yang menangani kejadian mouse bergerak cepat dan meninggalkan wilayah.
private bool mouseDown;
private Point lastLocation;
private void ToolStrip1_MouseDown(object sender, MouseEventArgs e) {
mouseDown = true;
lastLocation = e.Location;
}
private void ToolStrip1_MouseMove(object sender, MouseEventArgs e) {
if (mouseDown) {
this.Location = new Point(
(this.Location.X - lastLocation.X) + e.X, (this.Location.Y - lastLocation.Y) + e.Y);
this.Update();
}
}
private void ToolStrip1_MouseUp(object sender, MouseEventArgs e) {
mouseDown = false;
}
private void ToolStrip1_MouseLeave(object sender, EventArgs e) {
mouseDown = false;
}
Saya mencoba yang berikut dan presto changeo, jendela transparan saya tidak lagi membeku di tempatnya tetapi dapat dipindahkan !! (buang semua solusi kompleks lainnya di atas ...)
private void Window_MouseLeftButtonDown(object sender, MouseButtonEventArgs e)
{
base.OnMouseLeftButtonDown(e);
// Begin dragging the window
this.DragMove();
}
Formulir1 (): new Moveable(control1, control2, control3);
Kelas:
using System;
using System.Windows.Forms;
class Moveable
{
public const int WM_NCLBUTTONDOWN = 0xA1;
public const int HT_CAPTION = 0x2;
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern bool ReleaseCapture();
public Moveable(params Control[] controls)
{
foreach (var ctrl in controls)
{
ctrl.MouseDown += (s, e) =>
{
if (e.Button == MouseButtons.Left)
{
ReleaseCapture();
SendMessage(ctrl.FindForm().Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0);
// Checks if Y = 0, if so maximize the form
if (ctrl.FindForm().Location.Y == 0) { ctrl.FindForm().WindowState = FormWindowState.Maximized; }
}
};
}
}
}
[DllImport("user32.DLL", EntryPoint = "ReleaseCapture")]
private extern static void ReleaseCapture();
[DllImport("user32.DLL", EntryPoint = "SendMessage")]
private extern static void SendMessage(System.IntPtr hWnd, int Msg, int wParam, int lParam);
private void panelTitleBar_MouseDown(object sender, MouseEventArgs e)
{
ReleaseCapture();
SendMessage(this.Handle, 0x112, 0xf012, 0);
}