Winforms - フォーム内の任意の場所をクリック/ドラッグして、フォームのキャプションをクリックしたかのように移動します。

StackOverflow https://stackoverflow.com/questions/30184

  •  09-06-2019
  •  | 
  •  

質問

Winforms アプリケーションで使用される小さなモーダル フォームを作成しています。これは基本的に一種の進行状況バーです。ただし、フォームが表示されている間、ユーザーがフォーム内の任意の場所をクリックしてドラッグしてデスクトップ上で移動できるようにしたいと考えています。

この動作を実装するにはどうすればよいですか?

役に立ちましたか?

解決

Microsoft KB 記事 320687 この質問に対する詳細な回答があります。

基本的に、テスト対象のポイントがフォームのクライアント領域にある場合、WndProc メソッドをオーバーライドして WM_NCHITTEST メッセージに HTCAPTION を返します。これは、事実上、クリックがクリックされたときとまったく同じように 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);
}

他のヒント

ここでは、P/Invoke を使用してこれを行う方法を示します。

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);
    }
}

次のコードは、ProgressBarForm フォームに ProgressBar コントロールがあることを前提としています。 ドック に設定されたプロパティ 埋める

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;
    }
}

受け入れられた答えは素晴らしいトリックですが、たとえば、フォームがパネル(または派生)のようなフィルドッキングされた子コントロールで覆われている場合、このコントロールはほとんどの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 の):

#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);
                    }
                }

    };
}
ライセンス: CC-BY-SA帰属
所属していません StackOverflow
scroll top