2011年03月25日

フォームをドラッグして移動させる機能を提供するクラス・吸着機能付き(C#用)

以前、紹介した「フォームをドラッグして移動させる機能を提供するクラス(C#用)」に、スクリーンの端に吸着させる機能を追加しました。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Drawing;
using System.Windows.Forms;

namespace Com.Mirano.Forms
{
    /// <summary>
    /// フォームをドラッグして移動するクラス
    /// </summary>
    class FormDragMover
    {
        // 移動の対象となるフォーム
        Form moveForm;

        // 移動中を表す状態
        bool moveStatus;

        // ドラッグを無効とする幅(フォームの端をサイズ変更に使うときなど)
        int noDragAreaWidth;

        // 標準のカーソル
        Cursor defaultCursor;

        // マウスをクリックした位置
        Point lastMouseDownPoint;

        // ウィンドウをスクリーンに吸着させる機能の有効/無効
        bool isAdsorb;

        // ウィンドウをスクリーンに吸着させる場合の吸着させるマージン
        Size adsorbGap;

        /// <summary>
        /// コンストラクタ
        /// </summary>
        /// <param name="moveForm">移動の対象となるフォーム</param>
        /// <param name="noDragAreaWidth">ドラッグを無効とする幅</param>
        public FormDragMover(Form moveForm, int noDragAreaWidth)
        {
            this.moveForm = moveForm;
            this.noDragAreaWidth = noDragAreaWidth;
            this.isAdsorb = false;

            // 現時点でのカーソルを保存しておく
            defaultCursor = moveForm.Cursor;

            // イベントハンドラを追加
            moveForm.MouseDown += new MouseEventHandler(moveForm_MouseDown);
            moveForm.MouseMove += new MouseEventHandler(moveForm_MouseMove);
            moveForm.MouseUp += new MouseEventHandler(moveForm_MouseUp);
        }
        /// <summary>
        /// コンストラクタ(吸着機能を使用する場合)
        /// </summary>
        /// <param name="moveForm">移動の対象となるフォーム</param>
        /// <param name="noDragAreaWidth">ドラッグを無効とする幅</param>
        /// <param name="isAdsorb">吸着機能の有効/無効</param>
        /// <param name="adsorbGap">吸着させるマージン</param>
        public FormDragMover(Form moveForm, int noDragAreaWidth, bool isAdsorb, Size adsorbGap)
        {
            this.moveForm = moveForm;
            this.noDragAreaWidth = noDragAreaWidth;
            this.isAdsorb = isAdsorb;
            this.adsorbGap = adsorbGap;

            // 現時点でのカーソルを保存しておく
            defaultCursor = moveForm.Cursor;

            // イベントハンドラを追加
            moveForm.MouseDown += new MouseEventHandler(moveForm_MouseDown);
            moveForm.MouseMove += new MouseEventHandler(moveForm_MouseMove);
            moveForm.MouseUp += new MouseEventHandler(moveForm_MouseUp);
        }

        /// <summary>
        /// マウスボタン押下イベントハンドラ
        /// </summary>
        void moveForm_MouseDown(object sender, MouseEventArgs e)
        {
            // 左クリック時のみ処理する。左クリックでなければ何もしない
            if ((e.Button & MouseButtons.Left) != MouseButtons.Left) return;

            // 移動が有効になる範囲
            // 例えばフォームの端から何ドットかをサイズ変更用の領域として使用する場合、
            // そこを避けるために使う。
            Rectangle moveArea = new Rectangle(
                noDragAreaWidth, noDragAreaWidth,
                moveForm.Width - (noDragAreaWidth * 2), moveForm.Height - (noDragAreaWidth * 2));

            // クリックした位置が移動が有効になる範囲であれば、移動中にする
            if (moveArea.Contains(e.Location))
            {
                // 移動中にする
                moveStatus = true;

                // クリックしたポイントを保存する
                lastMouseDownPoint = e.Location;

                // マウスキャプチャー
                moveForm.Capture = true;
            }
            else
            {
                moveStatus = false;
            }
        }

        /// <summary>
        /// マウス移動イベントハンドラ
        /// </summary>
        void moveForm_MouseMove(object sender, MouseEventArgs e)
        {
            // 移動中の場合のみ処理。移動中でなければ何もせず終わる
            if (moveStatus == false) return;

            // 左ボタン押下中のみ処理する。押下中ではないときは何もしない。
            if ((e.Button & MouseButtons.Left) != MouseButtons.Left) return;

            // マウスカーソルの変更
            moveForm.Cursor = Cursors.SizeAll;

            // フォームの移動
            if (this.isAdsorb)
            {
                // ウィンドウを吸着させる場合

                // 移動先のフォームの位置を計算しておく
                Rectangle newPosition = new Rectangle(
                    moveForm.Left + e.X - lastMouseDownPoint.X,
                    moveForm.Top + e.Y - lastMouseDownPoint.Y,
                    moveForm.Width, moveForm.Height);

                // 判定用のRECT(一時領域)
                Rectangle newRect = new Rectangle();

                // 作業領域の取得(この作業領域の内側に吸着する)
                Size area = new Size(
                    System.Windows.Forms.Screen.GetWorkingArea(moveForm).Width,
                    System.Windows.Forms.Screen.GetWorkingArea(moveForm).Height);

                // 画面端の判定用(画面の端の位置に、吸着するサイズ分のRECTを定義する)
                Rectangle rectLeft = new Rectangle(0, 0, adsorbGap.Width, area.Width);
                Rectangle rectTop = new Rectangle(0, 0, area.Width, adsorbGap.Height);
                Rectangle rectRight = new Rectangle(area.Width - adsorbGap.Width, 0, adsorbGap.Width, area.Height);
                Rectangle rectBottom = new Rectangle(0, area.Height - adsorbGap.Height, area.Width, adsorbGap.Height);

                // 衝突判定
                // 判定用のRECTを自分のウィンドウの隅に重ねるように移動し、
                // 画面端の判定用のRECTと衝突しているかチェックする。
                // 衝突していた場合は、吸着させるように移動する

                // 左端衝突判定
                {
                    newRect = newPosition;
                    newRect.Width = adsorbGap.Width;

                    if (newRect.IntersectsWith(rectLeft))
                    {
                        // 左端に吸着させる
                        newPosition.X = 0;
                    }
                }
                // 右端衝突判定
                {
                    newRect = newPosition;
                    newRect.X = newPosition.Right - adsorbGap.Width;  // ウィンドウの右隅
                    newRect.Width = adsorbGap.Width;

                    if (newRect.IntersectsWith(rectRight))
                    {
                        // 右端に吸着させる
                        newPosition.X = area.Width - moveForm.Width;
                    }
                }
                // 上端衝突判定
                {
                    newRect = newPosition;
                    newRect.Height = adsorbGap.Height;

                    if (newRect.IntersectsWith(rectTop))
                    {
                        // 上端に吸着させる
                        newPosition.Y = 0;
                    }
                }
                // 下端衝突判定
                {
                    newRect = newPosition;
                    newRect.Y = newPosition.Bottom - adsorbGap.Height; // ウィンドウの下端
                    newRect.Height = adsorbGap.Height;

                    if (newRect.IntersectsWith(rectBottom))
                    {
                        // 下端に吸着させる
                        newPosition.Y = area.Height - moveForm.Height;
                    }
                }

                // 実際に移動させる
                moveForm.Left = newPosition.Left;
                moveForm.Top = newPosition.Top;
            }
            else
            {
                // 通常の移動の場合
                moveForm.Left += e.X - lastMouseDownPoint.X;
                moveForm.Top += e.Y - lastMouseDownPoint.Y;
            }
        }

        /// <summary>
        /// マウスボタン押上イベントハンドラ
        /// </summary>
        void moveForm_MouseUp(object sender, MouseEventArgs e)
        {
            // 左ボタンのみ処理する。左ボタンではないときは何もしない。
            if ((e.Button & MouseButtons.Left) != MouseButtons.Left) return;

            // 移動を終了する
            moveStatus = false;

            // マウスキャプチャーを終了する
            moveForm.Capture = false;

            // マウスカーソルを戻す
            moveForm.Cursor = defaultCursor;
        }
    }
}

使い方

// 変数を定義しておき 
FormDragMover formMover;
 
private void InitializeForm()
{
    // Formのイニシャル処理で生成する 
    formMover = new FormDragMover(this, 8, true, new Size(16, 16));
}

第3引数に、吸着させる機能を有効にするため、trueを指定。 第4引数に、吸着させる距離(マージン)を指定します。

[2011-04-10追記] moveForm_MouseUpにマウスカーソルを戻す処理を追加しました。

良かったらクリックしてください
にほんブログ村 IT技術ブログ プログラム・プログラマーへ  人気ブログランキングへ

posted by among at 22:07 | Comment(1) | TrackBack(0) | C#
この記事へのコメント
フォームをドラッグして移動させる機能を提供するクラス・吸着機能付き(C#用): amongの雑記 weight http://skola.skinnskatteberg.se/infot/
Posted by weight at 2013年07月23日 14:27
コメントを書く
お名前:

メールアドレス:

ホームページアドレス:

コメント:

この記事へのトラックバックURL
http://blog.sakura.ne.jp/tb/43991788

この記事へのトラックバック