2011年03月27日

フォームの端をドラッグしてサイズ変更する機能を提供するクラス・吸着機能付き(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 FormDragResizer
    {
        // サイズ変更の対象となるフォーム
        Form resizeForm;

        /// <summary>
        /// サイズ変更の対象となる枠の位置
        /// </summary>
        public enum ResizeDirection
        {
            None = 0,
            Top = 1,
            Left = 2,
            Bottom = 4,
            Right = 8,
            All = 15
        }

        // サイズ変更が有効になる枠
        ResizeDirection resizeDirection;

        // サイズ変更中を表す状態
        ResizeDirection resizeStatus;

        // サイズ変更が有効になる範囲の幅
        int resizeAreaWidth;

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

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

        // マウスをクリックした時点のサイズ
        Size lastMouseDownSize;

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

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

        // ウィンドウの最小サイズ
        Size minimumSize;

        /// <summary>
        /// コンストラクタ
        /// </summary>
        /// <param name="resizeForm">サイズ変更の対象となるフォーム</param>
        /// <param name="resizeDirection">サイズ変更が有効になる枠</param>
        /// <param name="resizeAreaWidth">サイズ変更が有効になる範囲の幅</param>
        public FormDragResizer(Form resizeForm, ResizeDirection resizeDirection, int resizeAreaWidth)
        {
            this.resizeForm = resizeForm;
            this.resizeDirection = resizeDirection;
            this.resizeAreaWidth = resizeAreaWidth;
            this.isAdsorb = false;
            
            // 最小サイズの設定
            // サイズ変更が有効になる範囲を確実に残すため、上下左右のマージン+1ドットを最小サイズにする
            this.minimumSize = new Size(resizeAreaWidth * 2 + 1, resizeAreaWidth * 2 + 1);

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

            // イベントハンドラを追加
            resizeForm.MouseDown += new MouseEventHandler(resizeForm_MouseDown);
            resizeForm.MouseMove += new MouseEventHandler(resizeForm_MouseMove);
            resizeForm.MouseUp += new MouseEventHandler(resizeForm_MouseUp);
        }

        /// <summary>
        /// コンストラクタ
        /// </summary>
        /// <param name="resizeForm">サイズ変更の対象となるフォーム</param>
        /// <param name="resizeDirection">サイズ変更が有効になる枠</param>
        /// <param name="resizeAreaWidth">サイズ変更が有効になる範囲の幅</param>
        /// <param name="isAdsorb">吸着機能の有効/無効</param>
        /// <param name="adsorbGap">吸着させるマージン</param>
        public FormDragResizer(Form resizeForm, ResizeDirection resizeDirection, int resizeAreaWidth, bool isAdsorb, Size adsorbGap)
        {
            this.resizeForm = resizeForm;
            this.resizeDirection = resizeDirection;
            this.resizeAreaWidth = resizeAreaWidth;
            this.isAdsorb = isAdsorb;
            this.adsorbGap = adsorbGap;

            // 最小サイズの設定
            // サイズ変更が有効になる範囲を確実に残すため、上下左右のマージン+1ドットを最小サイズにする
            this.minimumSize = new Size(resizeAreaWidth * 2 + 1, resizeAreaWidth * 2 + 1);

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

            // イベントハンドラを追加
            resizeForm.MouseDown += new MouseEventHandler(resizeForm_MouseDown);
            resizeForm.MouseMove += new MouseEventHandler(resizeForm_MouseMove);
            resizeForm.MouseUp += new MouseEventHandler(resizeForm_MouseUp);
        }

        /// <summary>
        /// マウスボタン押下イベントハンドラ
        /// </summary>
        void resizeForm_MouseDown(object sender, MouseEventArgs e)
        {
            // クリックしたポイントを保存する
            lastMouseDownPoint = e.Location;

            // クリックした時点でのフォームのサイズを保存する
            lastMouseDownSize = resizeForm.Size;

            // クリックした位置から、サイズ変更する方向を決める
            resizeStatus = ResizeDirection.None;

            // 上の判定
            if ((resizeDirection & ResizeDirection.Top) == ResizeDirection.Top)
            {
                Rectangle topRect = new Rectangle(0, 0, resizeForm.Width, resizeAreaWidth);
                if (topRect.Contains(e.Location))
                {
                    resizeStatus |= ResizeDirection.Top;
                }
            }

            // 左の判定
            if ((resizeDirection & ResizeDirection.Left) == ResizeDirection.Left)
            {
                Rectangle leftRect = new Rectangle(0, 0, resizeAreaWidth, resizeForm.Height);
                if (leftRect.Contains(e.Location))
                {
                    resizeStatus |= ResizeDirection.Left;
                }
            }

            // 下の判定
            if ((resizeDirection & ResizeDirection.Bottom) == ResizeDirection.Bottom)
            {
                Rectangle bottomRect = new Rectangle(0, resizeForm.Height - resizeAreaWidth, resizeForm.Width, resizeAreaWidth);
                if (bottomRect.Contains(e.Location))
                {
                    resizeStatus |= ResizeDirection.Bottom;
                }
            }

            // 右の判定
            if ((resizeDirection & ResizeDirection.Right) == ResizeDirection.Right)
            {
                Rectangle rightRect = new Rectangle(resizeForm.Width - resizeAreaWidth, 0, resizeAreaWidth, resizeForm.Height);
                if (rightRect.Contains(e.Location))
                {
                    resizeStatus |= ResizeDirection.Right;
                }
            }

            // サイズ変更の対象だったら、マウスキャプチャー
            if (resizeStatus != ResizeDirection.None)
            {
                resizeForm.Capture = true;
            }
        }

        /// <summary>
        /// マウス移動イベントハンドラ
        /// </summary>
        void resizeForm_MouseMove(object sender, MouseEventArgs e)
        {
            // サイズ変更が有効になる枠の上にカーソルが乗ったら
            // マウスカーソルをサイズ変更用のものに変更する

            // どの枠の上にカーソルが乗っているか
            ResizeDirection cursorPos = ResizeDirection.None;

            // 上の判定
            if ((resizeDirection & ResizeDirection.Top) == ResizeDirection.Top)
            {
                Rectangle topRect = new Rectangle(0, 0, resizeForm.Width, resizeAreaWidth);
                if (topRect.Contains(e.Location))
                {
                    cursorPos |= ResizeDirection.Top;
                }
            }

            // 左の判定
            if ((resizeDirection & ResizeDirection.Left) == ResizeDirection.Left)
            {
                Rectangle leftRect = new Rectangle(0, 0, resizeAreaWidth, resizeForm.Height);
                if (leftRect.Contains(e.Location))
                {
                    cursorPos |= ResizeDirection.Left;
                }
            }

            // 下の判定
            if ((resizeDirection & ResizeDirection.Bottom) == ResizeDirection.Bottom)
            {
                Rectangle bottomRect = new Rectangle(0, resizeForm.Height - resizeAreaWidth, resizeForm.Width, resizeAreaWidth);
                if (bottomRect.Contains(e.Location))
                {
                    cursorPos |= ResizeDirection.Bottom;
                }
            }

            // 右の判定
            if ((resizeDirection & ResizeDirection.Right) == ResizeDirection.Right)
            {
                Rectangle rightRect = new Rectangle(resizeForm.Width - resizeAreaWidth, 0, resizeAreaWidth, resizeForm.Height);
                if (rightRect.Contains(e.Location))
                {
                    cursorPos |= ResizeDirection.Right;
                }
            }

            // カーソルを変更

            // 左上(左上から右下への斜め矢印)
            if (((cursorPos & ResizeDirection.Left) == ResizeDirection.Left)
                && ((cursorPos & ResizeDirection.Top) == ResizeDirection.Top))
            {
                resizeForm.Cursor = Cursors.SizeNWSE;
            }
            // 右下(左上から右下への斜め矢印)
            else if (((cursorPos & ResizeDirection.Right) == ResizeDirection.Right)
                && ((cursorPos & ResizeDirection.Bottom) == ResizeDirection.Bottom))
            {
                resizeForm.Cursor = Cursors.SizeNWSE;
            }
            // 右上(右上から左下への斜め矢印)
            else if (((cursorPos & ResizeDirection.Right) == ResizeDirection.Right)
                && ((cursorPos & ResizeDirection.Top) == ResizeDirection.Top))
            {
                resizeForm.Cursor = Cursors.SizeNESW;
            }
            // 左下(右上から左下への斜め矢印)
            else if (((cursorPos & ResizeDirection.Left) == ResizeDirection.Left)
                && ((cursorPos & ResizeDirection.Bottom) == ResizeDirection.Bottom))
            {
                resizeForm.Cursor = Cursors.SizeNESW;
            }
            // 上(上下矢印)
            else if ((cursorPos & ResizeDirection.Top) == ResizeDirection.Top)
            {
                resizeForm.Cursor = Cursors.SizeNS;
            }
            // 左(左右矢印)
            else if ((cursorPos & ResizeDirection.Left) == ResizeDirection.Left)
            {
                resizeForm.Cursor = Cursors.SizeWE;
            }
            // 下(上下矢印)
            else if ((cursorPos & ResizeDirection.Bottom) == ResizeDirection.Bottom)
            {
                resizeForm.Cursor = Cursors.SizeNS;
            }
            // 右(左右矢印)
            else if ((cursorPos & ResizeDirection.Right) == ResizeDirection.Right)
            {
                resizeForm.Cursor = Cursors.SizeWE;
            }
            // どこにも属していない(デフォルト)
            else
            {
                resizeForm.Cursor = defaultCursor;
            }

            // リサイズ中の場合のみ処理。リサイズ中でなければ何もせず終わる
            if (resizeStatus == ResizeDirection.None) return;

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

            // サイズ変更の処理

            // ドラッグにより移動した距離を計算
            int diffX = e.X - lastMouseDownPoint.X;
            int diffY = e.Y - lastMouseDownPoint.Y;

            // サイズ変更後のフォーム位置
            Rectangle newPosition = new Rectangle(resizeForm.Location, resizeForm.Size);

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

            // 作業領域の取得(吸着させる場合は、この作業領域の内側に吸着する)
            Size area = new Size(
                System.Windows.Forms.Screen.GetWorkingArea(resizeForm).Width,
                System.Windows.Forms.Screen.GetWorkingArea(resizeForm).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);

            // 上
            if ((resizeStatus & ResizeDirection.Top) == ResizeDirection.Top)
            {
                // まず、ドラッグした距離分だけサイズを変更する
                // その後、フォームの位置を調整する
                // (順番を逆にすると、ちょっとちらつく?)
                int h = resizeForm.Height;
                newPosition.Height -= diffY;
                newPosition.Y += h - newPosition.Height;
                // 小さくなりすぎるのを防ぐ
                if (newPosition.Height < minimumSize.Height)
                {
                    newPosition.Y = resizeForm.Top;
                    newPosition.Height = resizeForm.Height;
                }
                // 吸着の処理
                if (this.isAdsorb)
                {
                    // 上端衝突判定
                    newRect = newPosition;
                    newRect.Height = adsorbGap.Height;

                    if (newRect.IntersectsWith(rectTop))
                    {
                        // 上端に吸着させる
                        newPosition.Height = resizeForm.Bottom;
                        newPosition.Y = 0;
                    }
                }
            }
            // 左
            if ((resizeStatus & ResizeDirection.Left) == ResizeDirection.Left)
            {
                // 上と同じ
                int w = resizeForm.Width;
                newPosition.Width -= diffX;
                newPosition.X += w - newPosition.Width;
                // 小さくなりすぎるのを防ぐ
                if (newPosition.Width < minimumSize.Width)
                {
                    newPosition.X = resizeForm.Left;
                    newPosition.Width = resizeForm.Width;
                }
                // 吸着の処理
                if (this.isAdsorb)
                {
                    // 左端衝突判定
                    newRect = newPosition;
                    newRect.Width = adsorbGap.Width;

                    if (newRect.IntersectsWith(rectLeft))
                    {
                        // 左端に吸着させる
                        newPosition.Width = resizeForm.Right;
                        newPosition.X = 0;
                    }
                }
            }
            // 下
            if ((resizeStatus & ResizeDirection.Bottom) == ResizeDirection.Bottom)
            {
                // マウスクリックした時点のサイズを基点として、
                // ドラッグした距離分だけサイズを変更する
                newPosition.Height = lastMouseDownSize.Height + diffY;
                // 小さくなりすぎるのを防ぐ
                if (newPosition.Height < minimumSize.Height)
                {
                    newPosition.Y = resizeForm.Top;
                    newPosition.Height = resizeForm.Height;
                }
                if (this.isAdsorb)
                {
                    // 下端衝突判定
                    newRect = newPosition;
                    newRect.Y = newPosition.Bottom - adsorbGap.Height; // ウィンドウの下端
                    newRect.Height = adsorbGap.Height;

                    if (newRect.IntersectsWith(rectBottom))
                    {
                        // 下端に吸着させる
                        newPosition.Height = area.Height - resizeForm.Top;
                    }
                }
            }
            // 右
            if ((resizeStatus & ResizeDirection.Right) == ResizeDirection.Right)
            {
                // マウスクリックした時点のサイズを基点として、
                // ドラッグした距離分だけサイズを変更する
                newPosition.Width = lastMouseDownSize.Width + diffX;
                // 小さくなりすぎるのを防ぐ
                if (newPosition.Width < minimumSize.Width)
                {
                    newPosition.X = resizeForm.Left;
                    newPosition.Width = resizeForm.Width;
                }
                // 吸着の処理
                if (this.isAdsorb)
                {
                    // 右端衝突判定
                    newRect = newPosition;
                    newRect.X = newPosition.Right - adsorbGap.Width;  // ウィンドウの右隅
                    newRect.Width = adsorbGap.Width;

                    if (newRect.IntersectsWith(rectRight))
                    {
                        // 右端に吸着させる
                        newPosition.Width = area.Width - resizeForm.Left;
                    }
                }
            }
            // 実際にサイズ変更させる
            resizeForm.Left = newPosition.Left;
            resizeForm.Top = newPosition.Top;
            resizeForm.Width = newPosition.Width;
            resizeForm.Height = newPosition.Height;
        }

        /// <summary>
        /// マウスボタン押上イベントハンドラ
        /// </summary>
        void resizeForm_MouseUp(object sender, MouseEventArgs e)
        {
            // マウスキャプチャーを終了する
            resizeForm.Capture = false;
        }
    }
}

使い方

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

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

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

posted by among at 20:50 | Comment(0) | TrackBack(0) | C#
この記事へのコメント
コメントを書く
お名前:

メールアドレス:

ホームページアドレス:

コメント:

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

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