我们在DataGrid
中的SelectedItem
属性周围看到了一些奇怪的行为。 一些背景资料:
DataGrid
显示对数据库的查询结果。 有一个按钮允许用户手动刷新DataGrid
中的结果。 有一个自动刷新机制,结果将每30秒自动刷新一次。
我们看到的是,当自动刷新发生时,SelectedItem
属性将始终成为Datagrid的ItemsSource的索引0。 但我们希望当前选定的行在刷新后保持为选定行。 但是,如果用户手动单击refresh,则所选行在刷新后保持不变,这很奇怪,因为刷新逻辑运行的是相同的代码。 是的,我们有代码来记住当前选择的项目,然后在刷新完成后重新设置。
下面是一些相关代码:
<UserControl.Resources>
<CollectionViewSource Source="{Binding DataGridResults}" x:Key="ReferralItemsSource"/>
</UserControl.Resources>
<customControls:CustomDataGrid x:Name="GridControl"
ItemsSource="{Binding Source={StaticResource ReferralItemsSource}}"
SelectedItem="{Binding DataContext.SelectedReferral, RelativeSource={RelativeSource AncestorType={x:Type UserControl}}}"
IsReadOnly="False"
IsSynchronizedWithCurrentItem="True"
SelectionMode="Single">
private async void RefreshWorklist(bool invokedByAutoRefresh = false)
{
try
{
if (Initialising || ShowSpinner || IsProcessing || ShowRefreshSpinner || IsCurrentWorklistDeleted || !_sessionData.IsActive()) return;
IsProcessing = true;
RefreshWorklistCommand.RaiseCanExecuteChanged();
if (CurrentWorklistId != null)
{
var selectedReferralId = SelectedReferral.pk_Referral_ID;
if (invokedByAutoRefresh)
{
// Refresh has been invoked by _timer, so show spinner on the results page only
ShowRefreshSpinner = true;
}
else
{
// User has manually clicked refresh button so show app wide spinner
ShowSpinner = true;
if (_timer != null)
{
SetupWorklistRefreshTimer(); // Setup _timer again so that it will refresh again at an appropriate time
}
}
Referrals = await _referralRepository.GetReferralsFromWorklistAsync(CurrentWorklistId.Value, invokedByAutoRefresh);
if (Filters.Count > 0)
{
var listOfReferralPks = ReferralFiltering.GetFilteredResults(Referrals, Filters.Where(f => f.HasBeenApplied).ToList());
var filteredResults = Referrals.Where(r => listOfReferralPks.Contains(r.pk_Referral_ID)).ToList();
DataGridResults = MapReferralLookupItemsToReferralLookupItemViewModels(filteredResults);
}
else
{
DataGridResults = MapReferralLookupItemsToReferralLookupItemViewModels(Referrals);
}
SelectedReferral = DataGridResults.FirstOrDefault(r => r.pk_Referral_ID == selectedReferralId);
}
}
catch (Exception e)
{
_errorHandler.DisplayError(e);
}
}
如前所述,RefreshWorkList()
由通过命令
调用的手动刷新调用:
private void Execute_RefreshWorklist()
{
RefreshWorklist();
}
或通过使用计时器
自动:
private void SetupWorklistRefreshTimer()
{
_timer?.Dispose();
var refreshInterval = _userSettingsRepository.GetIntegerSystemSetting("ReferralsWorklistRefreshInterval");
if (refreshInterval <= 0) return; // If this is 0 or below then the refresh should be disabled
if (refreshInterval < 10) // If it is less than 10 then set it to 10 to avoid too many MT calls
{
refreshInterval = 10;
}
var timeUntilFirstTick = refreshInterval * 1000;
_timer = new Timer((s) => RefreshWorklist(true), null, timeUntilFirstTick, refreshInterval * 1000);
}
最后是SelectedItem
属性视图模型绑定属性:
public ReferralLookupItemViewModel SelectedReferral
{
get { return _selectedReferral; }
set
{
if (_selectedReferral != value)
{
_selectedReferral = value;
OnPropertyChanged();
}
}
}
有人知道为什么会发生这种行为吗? 是否与计时器
有关? 我知道这不是一个简单的问题,所以请询问更多的信息。
您需要在binding
中与UI线程上的UI分配属性。
调用RefreshWorkList
时,将Timer
替换为DispatcherTimer
,或在现有Timer
回调中使用Dispatcher.Invoke
或Dispatcher.BeginInvoke
。
通过按按钮
,您已经在UI线程上,但是计时器
有它自己的线程,与UI线程不同。 DispatcherTimer
回调是在UI线程上调用的,而不是https://docs.microsoft.com/en-us/dotnet/api/system.windows.threading.DispatcherTimer?view=NetFramework-4.0