2 回答

TA貢獻1860條經驗 獲得超9個贊
有沒有辦法IsDirty在所有設置器之后將其設置為 false?
setter 不是自己調用的,必須有人調用他們。您應該確定是誰在這樣做,并且要么阻止他在沒有充分理由的情況下設置內容(首選),要么讓他在完成后重置臟標志。
正如評論中所建議的,在設置器中添加斷點并查看堆棧跟蹤是查找設置來源的一個很好的起點......如果我不得不猜測,我會懷疑一些與導航相關的回調。
但是您應該嘗試確保視圖模型在構造函數之后初始化,這IsDirty實際上意味著“已通過視圖更改”而不是“可能由用戶更改,也可能只是延遲初始化的一部分”。
經過多次編輯后,我的編輯:
您應該修改架構以考慮視圖模型的異步初始化。僅僅并行運行所有事情并希望得到最好的結果很少會奏效。
例如,您可以將屬性設置為只讀,直到初始化完成,然后IsDirty在.falseInitializeLookupValues
偽代碼:
Constructor()
{
Task.Run( async () => await InitializeAsync() );
}
string Property
{
get => _backingField;
set
{
if (_isInitialized && SetProperty( ref _backingField, value ))
_isDirty = true;
}
}
private async Task InitializeAsync()
{
await SomeAsynchronousStuff();
_isInitialized = true;
}
private bool _isInitialized;
private bool _isDirty;
也許,您想將_isInitialized其作為屬性公開給視圖以顯示一些沙漏,并使用 aManualResetEvent而不是簡單的bool... 但您明白了。

TA貢獻1828條經驗 獲得超13個贊
由于這些SetProperty方法是可重寫的,因此您可以注入一些自定義邏輯。當您需要驗證對象是否已被更改時,這可能非常有用。
public class StatefulObject : Prism.Mvvm.BindableBase
{
private bool _isDirty;
public bool IsDirty
{
get => _isDirty;
private set => SetProperty(ref _isDirty, value);
}
protected override bool SetProperty<T>(ref T storage, T value, Action onChanged, [CallerMemberName] string propertyName = null)
{
var isDirty = base.SetProperty(ref storage, value, onChanged, propertyName);
if(isDirty && propertyName != nameof(isDirty))
{
IsDirty = true;
}
return isDirty;
}
public void Reset() => IsDirty = false;
}
請記住,當您初始化字段時,此IsDirty值為 true,因此在綁定之前,您需要調用該Reset方法將其設置IsDirty回 false,這樣您就可以可靠地知道字段何時已更改。
請注意,如何處理這個問題在某種程度上取決于您。例如,您可以使用 Linq 執行此操作...
var fooDTOs = someService.GetDTOs().Select(x => { x.Reset(); return x; });
您還可以強制執行如下模式:
public class FooDTO : StatefulObject
{
public FooDTO(string prop1, string prop2)
{
// Set the properties...
Prop1 = prop1;
// Ensure IsDirty is false;
Reset();
}
}
- 2 回答
- 0 關注
- 147 瀏覽
添加回答
舉報