Как запустить код внутри конвертера в отдельном потоке, чтобы пользовательский интерфейс не зависал?
У меня есть конвертер WPF, который медленный (вычисления, онлайн-выборка и т.д.). Как я могу конвертировать асинхронно, чтобы мой пользовательский интерфейс не зависал? Я нашел это, но решение состоит в том, чтобы поместить код конвертера в свойство - http://social.msdn.microsoft.com/Forums/pl-PL/wpf/thread/50d288a2-eadc-4ed6-a9d3-6e249036cb71 - который я бы предпочел не делать.
Ниже приведен пример, демонстрирующий проблему. Здесь выпадающее меню замерзает до тех пор, пока сон не истечет.
namespace testAsync
{
using System;
using System.Collections.Generic;
using System.Threading;
using System.Windows;
using System.Windows.Data;
using System.Windows.Threading;
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
MyNumbers = new Dictionary<string, int> { { "Uno", 1 }, { "Dos", 2 }, { "Tres", 3 } };
this.DataContext = this;
}
public Dictionary<string, int> MyNumbers
{
get { return (Dictionary<string, int>)GetValue(MyNumbersProperty); }
set { SetValue(MyNumbersProperty, value); }
}
public static readonly DependencyProperty MyNumbersProperty =
DependencyProperty.Register("MyNumbers", typeof(Dictionary<string, int>), typeof(MainWindow), new UIPropertyMetadata(null));
public string MyNumber
{
get { return (string)GetValue(MyNumberProperty); }
set { SetValue(MyNumberProperty, value); }
}
public static readonly DependencyProperty MyNumberProperty = DependencyProperty.Register(
"MyNumber", typeof(string), typeof(MainWindow), new UIPropertyMetadata("Uno"));
}
public class AsyncConverter : IMultiValueConverter
{
public object Convert(object[] values, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
object result = null;
if (values[0] is string && values[1] is IDictionary<string, int>)
{
DoAsync(
() =>
{
Thread.Sleep(2000); // Simulate long task
var number = (string)(values[0]);
var numbers = (IDictionary<string, int>)(values[1]);
result = numbers[number];
result = result.ToString();
});
}
return result;
}
private void DoAsync(Action action)
{
var frame = new DispatcherFrame();
new Thread((ThreadStart)(() =>
{
action();
frame.Continue = false;
})).Start();
Dispatcher.PushFrame(frame);
}
public object[] ConvertBack(object value, Type[] targetType, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotSupportedException();
}
}
и XAML:
<Window x:Class="testAsync.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:testAsync"
Title="MainWindow" Height="200" Width="200">
<Window.Resources>
<local:AsyncConverter x:Key="asyncConverter"/>
</Window.Resources>
<DockPanel>
<ComboBox DockPanel.Dock="Top" SelectedItem="{Binding MyNumber, IsAsync=True}"
ItemsSource="{Binding MyNumbers.Keys, IsAsync=True}"/>
<TextBlock DataContext="{Binding IsAsync=True}"
FontSize="50" FontWeight="Bold" HorizontalAlignment="Center" VerticalAlignment="Center">
<TextBlock.Text>
<MultiBinding Converter="{StaticResource asyncConverter}">
<Binding Path="MyNumber" IsAsync="True"/>
<Binding Path="MyNumbers" IsAsync="True"/>
</MultiBinding>
</TextBlock.Text>
</TextBlock>
</DockPanel>
</Window>
Обратите внимание, что все привязки теперь IsAsync = "True", но это не помогает.
![enter image description here]()
Комбобокс будет застревать в течение 2000 мс.
Ответы
Ответ 1
Я знаю, что вы сказали, что не хотите ссылаться на перевод с устройства настройки свойств, но я утверждаю, что это более чистый подход, чем IValueConverter
/IMultiValueConverter
.
В конечном счете вы хотите установить значение выбранного номера из выпадающего списка и немедленно вернуться к нему. Вы хотите отложить обновление отображаемого/переведенного значения до завершения процесса перевода.
Я думаю, что яснее моделировать данные так, что переведенное значение само является свойством, которое просто обновляется асинхронным процессом.
<ComboBox SelectedItem="{Binding SelectedNumber, Mode=OneWayToSource}"
ItemsSource="{Binding MyNumbers.Keys}"/>
<TextBlock Text="{Binding MyNumberValue}" />
public partial class MainWindow : Window, INotifyPropertyChanged
{
public MainWindow()
{
InitializeComponent();
MyNumbers = new Dictionary<string, int> { { "Uno", 1 }, { "Dos", 2 }, { "Tres", 3 } };
DataContext = this;
}
public IDictionary<string, int> MyNumbers { get; set; }
string _selectedNumber;
public string SelectedNumber
{
get { return _selectedNumber; }
set
{
_selectedNumber = value;
Notify("SelectedNumber");
UpdateMyNumberValue();
}
}
int _myNumberValue;
public int MyNumberValue
{
get { return _myNumberValue; }
set
{
_myNumberValue = value;
Notify("MyNumberValue");
}
}
void UpdateMyNumberValue()
{
var key = SelectedNumber;
if (key == null || !MyNumbers.ContainsKey(key)) return;
new Thread(() =>
{
Thread.Sleep(3000);
MyNumberValue = MyNumbers[key];
}).Start();
}
public event PropertyChangedEventHandler PropertyChanged;
void Notify(string property)
{
var handler = PropertyChanged;
if(handler != null) handler(this, new PropertyChangedEventArgs(property));
}
}
Ответ 2
Для этого вы можете использовать DispatcherFrame
, здесь пример конвертера:
public class AsyncConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
object result = null;
DoAsync(() =>
{
Thread.Sleep(2000); // Simulate long task
result = (int)value * 2; // Some sample conversion
});
return result;
}
private void DoAsync(Action action)
{
var frame = new DispatcherFrame();
new Thread((ThreadStart)(() =>
{
action();
frame.Continue = false;
})).Start();
Dispatcher.PushFrame(frame);
}
public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotSupportedException();
}
}
Ответ 3
Не очень хороший дизайн для тяжелых вычислений в конвертере - особенно если вы используете функциональность, которую другие должны использовать в качестве хорошего примера.
Я бы переписал и использовал MVVM с вашим ViewModel как конвертер на стероидах, где вы можете делать все это прозрачным способом - проще программировать, более понятный программный процесс, проще понимать код.
И тогда вы можете использовать Prioritybindings:
http://msdn.microsoft.com/en-us/library/system.windows.data.prioritybinding.aspx
Для вашей исходной проблемы я бы посмотрел, когда вызывается конвертер - если это когда привязка вернула его значение, вы, вероятно, не сможете заставить Async делать то, что он делает. Я подозреваю, что wpf ждет возвращения свойства, а затем вызывает конвертер - в этом случае может оказаться невозможным заставить ваш конвертер не замораживать gui.
Подход, который вы можете предпринять:
- В конвертере вы должны начать получать ваши данные и возвращаться, например, с помощью backgroundworker - иначе ui замерзнет.
- В многосвязном проходе ссылается на что-то, поэтому, когда ваши данные поступают, вы можете запустить свойство propertychanged
Ответ 4
Предлагаю посмотреть BackgroundWorker
. Он может выполнять перевод в фоновом потоке, а затем вызывает завершенное событие в потоке пользовательского интерфейса.
См. http://www.dotnetperls.com/backgroundworker