プログラムを書こう!

実務や自作アプリ開発で習得した役に立つソフトウェア技術情報を発信するブログ

WPFのDataGridでマウスのクリックとダブルクリックを併用する

この記事は2020年04月02日に投稿しました。

f:id:paveway:20190914064630j:plain

目次

  1. はじめに
  2. WPFのDataGridでマウスのクリックとダブルクリックを併用する
  3. おわりに

1. はじめに

こんにちは、iOSのエディタアプリPWEditorの開発者の二俣です。
今回は業務で使用しているWPFのDataGridでマウスのクリックとダブルクリックを併用する方法についてです。

目次へ

2. WPFのDataGridでマウスのクリックとダブルクリックを併用する

WPFのDataGridでマウスのクリックとダブルクリックを併用についてですが、ダブルクリックのイベントは必ずクリックイベントの後に発生するため、そのままでは処理できません。
そのためタイマーを使用し、以下のような方法とします。

  1. クリックイベントの際、タイマーを起動します。
    タイマーは2回のクリックがダブルクリックとみなされる時間でイベントを発生させるようにします。
  2. ダブルクリックイベントが発生した場合、タイマーを停止し、ダブルクリックイベントとして処理します。
  3. ダブルクリックイベントが発生しない場合、タイマーのイベントによりクリックの処理を行います。

実装例

using System;
using System.Windows;
using System.Windows.Input;
using System.Data;

namespace WPFMouseClick
{
    /// <summary>
    /// MainWindow.xaml の相互作用ロジック
    /// </summary>
    public partial class MainWindow : Window
    {
        //! マウスクリックタイマー
        private System.Windows.Forms.Timer MouseClickTimer = new System.Windows.Forms.Timer() { Enabled = false };

        /**
         * @brief コンストラクタ
         */
        public MainWindow()
        {
            InitializeComponent();

            // マウスクリックタイマーにイベントハンドラを追加します。
            MouseClickTimer.Tick += MouseClickTimerTickHandler;
        }

        /**
         * @brief コンテントが描画されたあと呼び出されます。
         * 
         * @param [in] sender ウィンドウ
         * @param [in] e イベント
         */
        private void Window_ContentRendered(object sender, EventArgs e)
        {
            // データグリッドの内容を設定します。
            var dataTable = new DataTable();
            dataTable.Columns.Add("列1");
            dataTable.Columns.Add("列2");
            dataTable.Columns.Add("列3");

            var columnNum = dataTable.Columns.Count;
            for (int i = 0; i < 5; ++i)
            {
                var row = dataTable.NewRow();
                for (int j = 0; j < columnNum; ++j)
                {
                    row[j] = $"[{(i + 1)},{(j + 1)}";
                }
                dataTable.Rows.Add(row);
            }
            dataGrid.DataContext = dataTable;
        }

        /**
         * @brief データグリッドでマウスクリックした時に呼び出されます。
         *        MouseDownイベントだとデータグリッドのセルを選択した後にしか呼び出されないため
         *        PreviewMouseDownイベントを使用します。
         * 
         * @param [in] sender 
         * @param [in] e イベント
         */
        private void DataGrid_PreviewMouseDown(object sender, MouseButtonEventArgs e)
        {
            // マウスクリックタイマーを開始します。
            StartMouseClickTimer();
        }

        /**
         * @brief マウスダブルクリックした時に呼び出されます。
         * 
         * @param [in] sender 
         * @param [in] e イベント
         */
        private void DataGrid_MouseDoubleClick(object sender, MouseButtonEventArgs e)
        {
            // マウスクリックタイマーが有効な場合
            if (MouseClickTimer.Enabled)
            {
                // マウスクリックタイマーを停止します。
                MouseClickTimer.Enabled = false;
                MouseClickTimer.Stop();
            }
            
            // イベントを処理済みにします。
            e.Handled = true;
            Console.WriteLine("MouseDoubleClick");
        }

        /**
         * @brief マウスクリックタイマーで指定時間過ぎた時に呼び出されます。
         * 
         * @param [in] sender マウスクリックタイマー
         * @param [in] e イベント
         */
        private void MouseClickTimerTickHandler(object sender, EventArgs e)
        {
            // マウスクリックタイマーを停止します。
            MouseClickTimer.Enabled = false;
            MouseClickTimer.Stop();

            // 以降でマウスクリックの処理を行います。
            Console.WriteLine("MouseClick");
        }

        /**
         * @brief マウスクリックタイマーを開始します。
         */
        private void StartMouseClickTimer()
        {
            // マウスクリックタイマーが有効な場合
            if (MouseClickTimer.Enabled)
            {
                // 何もせず終了します。
                return;
            }

            // マウスクリックタイマーを開始します。
            // Intervalはマウスダブルクリックとみなされる時間とします。
            MouseClickTimer.Enabled = true;
            MouseClickTimer.Interval = System.Windows.Forms.SystemInformation.DoubleClickTime;
            MouseClickTimer.Start();
        }
    }
}

目次へ

3. おわりに

クリックとダブルクリックを併用しようと思いその方法を調べてみました。
簡単にできると思ったのですが、簡単にはできないようです。
今回はDataGridで説明しましたが、他のコントロールでも同様に処理できると思います。

紹介している一部の記事のコードはGitlabで公開しています。
興味のある方は覗いてみてください。

目次へ


私が勤務しているニューラルでは、主に組み込み系ソフトの開発を行っております。
弊社製品のハイブリッドOS Bi-OSは高い技術力を評価されており、特に制御系や通信系を得意としています。
私自身はiOSモバイルアプリウィンドウズアプリを得意としております。
ソフトウェア開発に関して相談などございましたら、お気軽にご連絡ください。

また一緒に働きたい技術者の方も随時募集中です。
興味がありましたらご連絡ください。

EMAIL : info-nr@newral.co.jp / m-futamata@newral.co.jp
TEL : 042-523-3663
FAX : 042-540-1688

目次へ