NDDD

ドメイン駆動開発_フォルダー構成編_#19_PropertyChangedの方法を変更する

前回はGetLatestメソッドを呼んだ後に,ViewModelBaseのOnPropertyChangedを呼ぶようにすることで,View側に変更通知をする方法を解説しました。

今回は,ViewModelBaseのSetPropertyを使うことで,ViewModelのプロパティの値が変更された時点でView側に変更通知を行うスマートなやり方に変更したいと思います。

ViewModelBaseのSetProperty

ViewModelBaseには次のようなSetPropertyというメソッドがあります。これを使うと,プロパティの値が変更された時だけ,View側に変更通知が行われるようになります。

        protected bool SetProperty(ref T field,
            T value, [CallerMemberName]string propertyName = null)
        {
            if (Equals(field, value))
            {
                return false;
            }

            field = value;
            var h = this.PropertyChanged;
            if (h != null)
            {
                h(this, new PropertyChangedEventArgs(propertyName));
            }

            return true;
        }

SetPropertyの使い方

SetPropertyは次のように記述します。

using NDDD.Domain.Entities;
using NDDD.Domain.Ripositories;
using NDDD.Infrastructure;
using System;
using System.ComponentModel;

namespace NDDD.WinForm.ViewModels
{
    public class LatestViewModel : ViewModelBase
    {
        private IMeasureRepository _measureRepository;

        private string _areaIdText = string.Empty;
        private string _measureDateText = string.Empty;
        private string _measureValueText = string.Empty;

        public LatestViewModel() :
            this(Factories.CreateMeasure())
        {
        }

        public LatestViewModel(IMeasureRepository measureRepository)
        {
            _measureRepository = measureRepository;
        }

       
        public string AreaIdText
        {
            get { return _areaIdText; }
            set
            {
                SetProperty(ref _areaIdText, value);
            }
        }
        public string MeasureDateText
        {
            get { return _measureDateText; }
            set
            {
                SetProperty(ref _measureDateText, value);
            }
        }
        public string MeasureValueText
        {
            get { return _measureValueText; }
            set
            {
                SetProperty(ref _measureValueText, value);
            }
        }

        public void Search()
        {
            var measure = _measureRepository.GetLatest();
            AreaIdText = measure.AreaId.ToString().PadLeft(4, '0'); 
            MeasureDateText = measure.MeasureDate.ToString("yyyy/MM/dd HH:mm:ss"); 
            MeasureValueText = Math.Round(measure.MeasureValue, 2) + "℃";
        }
    }
}

書き方はエリアID,計測日時,計測値ともに同じ要領です。これまでAreaIdTextプロパティなどは,getだけがあるプロパティになっていましたが,getとsetを書き,setするときに,ViewModelBaseのSetPropertyを呼び出します。これにより,変更されている場合のみView側に変更通知するという処理が行われます。そのプロパティの中で値を編集したり,値を返却したりするために値を保持しておくまずprivateな変数を宣言します。

        private string _areaIdText = string.Empty;

プロパティは次のように変更します。

        public string AreaIdText
        {
            get { return _areaIdText; }
            set
            {
                SetProperty(ref _areaIdText, value);
            }
        }

getでは_areaIdTextを返却するのみです。setではSetPropertyを呼び出しています。 計測日時と計測値は同じ要領で記述します。 最後にSearchメソッドのGetLatestを呼んだあとの処理を変更します。

            var measure = _measureRepository.GetLatest();
            AreaIdText = measure.AreaId.ToString().PadLeft(4, '0'); 
            MeasureDateText = measure.MeasureDate.ToString("yyyy/MM/dd HH:mm:ss"); 
            MeasureValueText = Math.Round(measure.MeasureValue, 2) + "℃";

これまでは,GetLatestで取得した値を_measureに入れていましたが,値取得後に,AreaIdTextなどのそれぞれのプロパティに値を設定するようにしています。このAreaIdTextなどに値を設定するときにSetPropertyが呼び出され,View側に値の変更通知が行われます。

このような記述にすることで,privateで宣言していた_measureが不要になるので,削除していてください。

プログラムの実行

この状態で一度実行してみましょう。

これでOnPropertyChangedを呼び出していた時と同じ結果が得られていることがわかると思います。

NDDD

#01_プロジェクトの作成

#02_プロジェクトの追加
#03_依存関係
#04_ドメイン駆動開発でApplication層は必要?
#05_Domainのフォルダー構成
#06_Infrastructureのフォルダー構成
#07_WinFormのフォルダー構成
#08_Testsのフォルダー構成
#09_テスト駆動で実装するための事前準備
#10_テストコードとViewModelの追加
#11_テストコードを追加する
#12_ Repositoriesフォルダーの作成
#13_ Entitiesフォルダーの作成
#14_ Mockの作成
#15_フォーム画面の作成
#16_画面のコントロールデータバインドする
#17_Fakeを使ってタミーデータを画面に表示させる
#18_Fakeデータを画面に通知する
#19_PropertyChangedの方法を変更する
#20_Fakeとデータベースの値を切り替える方法
#21_Sharedクラスを作成する
#22_クラスを生成するファクトリークラスを作る
#23_#if DEBUGでFakeデータがリリースされないようにする
#24_DEBUGモードであることをわかりやすくしておく
#25_Factories以外から生成できないようにしておく
#26_Factoriesの呼び出しはViewModelで行う
#27_外部の設定ファイルの値で判断する
#28_Fakeデータを切り替える方法
#29_FakePathを設定ファイルとSharedに移す
#30_Fakeデータのバリエーション
#31_Shareクラスの活用方法
#32_ベースフォームを作る
#33_SharedにログインIDを記憶する
#34_BaseFormでログインユーザーを表示する
#35_ValueObject
#36_ValueObjectを作成する
#37_抽象クラスValueObjectを使用してイコールの問題の解消
#38_AreaIdにビジネスロジックを入れる
#39_AreaIdクラスをEntityに乗せる
#40_MeasureDateの作成
#41_MeasureValueの作成
#42_オブジェクト指向の自動化
#43_Repositoryの具象クラス
#44_例外処理
#45_例外の作成
#46_インナーエクセプション
#47_例外の欠点
#48_メッセージの区分
#49_エラー処理の共通化
#50_ログの出力
#51_タイマー処理はどこに置く?
#52_タイマークラスの作成
#53_StaticValues
#54_Logics
#55_Helpers
#56_Module
#57_トランザクションはどこでかける?
#58_特徴を見極める
#59_さいごに