Winformsアプリケーションで使用される小さなモーダルフォームを作成しています。それは基本的には種類のプログレスバーです。しかし、私は、ユーザーがフォーム内のどこかをクリックしてドラッグして、まだ表示されている間にデスクトップ上を移動させたいと考えています。Winforms - フォームの任意の場所をクリック/ドラッグして、フォームのキャプションのように移動します。
この動作はどのように実装できますか?
Winformsアプリケーションで使用される小さなモーダルフォームを作成しています。それは基本的には種類のプログレスバーです。しかし、私は、ユーザーがフォーム内のどこかをクリックしてドラッグして、まだ表示されている間にデスクトップ上を移動させたいと考えています。Winforms - フォームの任意の場所をクリック/ドラッグして、フォームのキャプションのように移動します。
この動作はどのように実装できますか?
Microsoft KB Article 320687には、この質問に対する詳細な回答があります。
基本的には、WndProcメソッドをオーバーライドして、テスト対象のポイントがフォームのクライアント領域にある場合にHTCAPTIONをWM_NCHITTESTメッセージに戻します。これは、実際には、Windowsがクリックをまったく同じように扱うように指示しますそれはフォームのキャプションに発生していた。
private const int WM_NCHITTEST = 0x84;
private const int HTCLIENT = 0x1;
private const int HTCAPTION = 0x2;
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;
}
base.WndProc(ref m);
}
次のコードは、ProgressBarFormフォームはこちら
public partial class ProgressBarForm : Form
{
private bool mouseDown;
private Point lastPos;
public ProgressBarForm()
{
InitializeComponent();
}
private void progressBar1_MouseMove(object sender, MouseEventArgs e)
{
if (mouseDown)
{
int xoffset = MousePosition.X - lastPos.X;
int yoffset = MousePosition.Y - lastPos.Y;
Left += xoffset;
Top += yoffset;
lastPos = MousePosition;
}
}
private void progressBar1_MouseDown(object sender, MouseEventArgs e)
{
mouseDown = true;
lastPos = MousePosition;
}
private void progressBar1_MouseUp(object sender, MouseEventArgs e)
{
mouseDown = false;
}
}
は、P /呼び出しを使用して、それを行うための方法であるフィルに設定ドックプロパティとProgressBarコントロールを持っていることを前提としています。
public const int WM_NCLBUTTONDOWN = 0xA1;
public const int HTCAPTION = 0x2;
[DllImport("User32.dll")]
public static extern bool ReleaseCapture();
[DllImport("User32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
void Form_Load(object sender, EventArgs e)
{
this.MouseDown += new MouseEventHandler(Form_MouseDown);
}
void Form_MouseDown(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
ReleaseCapture();
SendMessage(Handle, WM_NCLBUTTONDOWN, HTCAPTION, 0);
}
}
必要があります:using System.Runtime.InteropServices; [DllImport()]が機能するために。 –
私のプロジェクトでは、Timothy Fries'sに似たコードを使っていましたが、同じようなテクニックを使用してラベルのようなコントロールでドラッグできるようにすることができます。他の手法ではフォームをドラッグ可能にしますが、ラベルをクリックするとドラッグしません。 – flamey
右クリックイベント(コンテキストメニュー用)を引き続き処理できるため、これはさらに便利です。上記のように左クリックを使い分けることも、左クリックの代わりに右/中クリックを使うこともできます。全体的にもっと柔軟になりました。 – coderforlife
受け入れ答えはクールなトリックですが、このコントロールはすべて食べるようになるので、フォームは、例えばパネル(または誘導体)のようなフィル・ドッキングされた子コントロールで覆われている場合、それは常に動作しません。ほとんどのWindowsメッセージここで
が、この場合にも動作しますシンプルなアプローチである。このようなハンドルのマウスメッセージ(代わりに、標準の一つで、このクラスを使用)問題の制御を導き出す:
private class MyTableLayoutPanel : Panel // or TableLayoutPanel, etc.
{
private Point _mouseDown;
private Point _formLocation;
private bool _capture;
// NOTE: we cannot use the WM_NCHITTEST/HTCAPTION trick because the table is in control, not the owning form...
protected override void OnMouseDown(MouseEventArgs e)
{
_capture = true;
_mouseDown = e.Location;
_formLocation = ((Form)TopLevelControl).Location;
}
protected override void OnMouseUp(MouseEventArgs e)
{
_capture = false;
}
protected override void OnMouseMove(MouseEventArgs e)
{
if (_capture)
{
int dx = e.Location.X - _mouseDown.X;
int dy = e.Location.Y - _mouseDown.Y;
Point newLocation = new Point(_formLocation.X + dx, _formLocation.Y + dy);
((Form)TopLevelControl).Location = newLocation;
_formLocation = newLocation;
}
}
}
VC++ 2010のバージョン(のFlySwat's):
#include <Windows.h>
namespace DragWithoutTitleBar {
using namespace System;
using namespace System::Windows::Forms;
using namespace System::ComponentModel;
using namespace System::Collections;
using namespace System::Data;
using namespace System::Drawing;
public ref class Form1 : public System::Windows::Forms::Form
{
public:
Form1(void) { InitializeComponent(); }
protected:
~Form1() { if (components) { delete components; } }
private:
System::ComponentModel::Container ^components;
HWND hWnd;
#pragma region Windows Form Designer generated code
void InitializeComponent(void)
{
this->SuspendLayout();
this->AutoScaleDimensions = System::Drawing::SizeF(6, 13);
this->AutoScaleMode = System::Windows::Forms::AutoScaleMode::Font;
this->ClientSize = System::Drawing::Size(640, 480);
this->FormBorderStyle = System::Windows::Forms::FormBorderStyle::None;
this->Name = L"Form1";
this->Text = L"Form1";
this->Load += gcnew EventHandler(this, &Form1::Form1_Load);
this->MouseDown += gcnew System::Windows::Forms::MouseEventHandler(this, &Form1::Form1_MouseDown);
this->ResumeLayout(false);
}
#pragma endregion
private: System::Void Form1_Load(Object^ sender, EventArgs^ e) {
hWnd = static_cast<HWND>(Handle.ToPointer());
}
private: System::Void Form1_MouseDown(Object^ sender, System::Windows::Forms::MouseEventArgs^ e) {
if (e->Button == System::Windows::Forms::MouseButtons::Left) {
::ReleaseCapture();
::SendMessage(hWnd, /*WM_NCLBUTTONDOWN*/ 0xA1, /*HT_CAPTION*/ 0x2, 0);
}
}
};
}
これにより、コンテキストメニューが使用できなくなります。 –
これは動作しますが、スプラッシュ画面にコントロール(ラベルなど)がある場合は、ラベルをクリックしても移動しません。スプラッシュにいくつかのコントロールがある場合、これを特別に管理する方法は? – supafly