Jon*_*mit 4 c# wpf caliburn.micro async-await
我有时需要延迟绑定执行(想想一个调用服务器的搜索框,你希望它只在用户暂停一瞬间而不是每次击键时执行)。
延迟 WPF 绑定没问题 - 您只需使用绑定指定延迟:
<TextBlock Text="{Binding Name, Delay=500}"/>
。
每当我在使用 Caliburn.Micro's 的情况下需要延迟执行时Message.Attach
,我通常以这种方式实现它(消息附加到TextChanged
带有DoSomething
操作的事件):
private int doingSomething;
public async void DoSomething()
{
int current = ++doingSomething;
await Task.Delay(500);
if (current != doingSomething) //method was reentered
return;
await DoWorkCallServerEtc();
}
Run Code Online (Sandbox Code Playgroud)
这很有效,但它不能很好地扩展并且打破了DRY原则(我需要在需要延迟的任何地方再次编写它)。
我的问题是,我可以使用 Caliburn.Micro 以某种方式为此编写一个约定吗?
或者也许是一种不同的、更具可扩展性的方法?
正如您在文档页面中所读到的那样
操作功能利用 System.Windows.Interactivity 作为其触发机制。
这意味着
<TextBox Name="textBox" Margin="5"
cal:Message.Attach="[Event TextChanged] = [Action DoAction(textBox.Text)]" />
Run Code Online (Sandbox Code Playgroud)
相当于:
<TextBox Margin="5" Name="textBox">
<i:Interaction.Triggers>
<i:EventTrigger EventName="TextChanged">
<cal:ActionMessage MethodName="DoAction">
<cal:Parameter Value="{Binding ElementName=textBox, Path=Text}" />
</cal:ActionMessage>
</i:EventTrigger>
</i:Interaction.Triggers>
</TextBox>
Run Code Online (Sandbox Code Playgroud)
负责执行该DoAction
方法的对象是EventTrigger。一个EventTrigger
触发immediatelly,没有延迟。所以我们需要创建你自己的DelayedEventTrigger
;类似的东西:
public class DelayedEventTrigger : System.Windows.Interactivity.EventTrigger
{
private EventArgs args;
private DispatcherTimer dispatcherTimer;
public static readonly DependencyProperty DelayProperty =
DependencyProperty.Register("Delay", typeof(int), typeof(DelayedEventTrigger), new PropertyMetadata(1000));
public int Delay
{
get { return (int)base.GetValue(DelayProperty); }
set { base.SetValue(DelayProperty, value); }
}
protected override void OnEvent(EventArgs eventArgs)
{
if (dispatcherTimer != null)
{
dispatcherTimer.Stop();
}
args = eventArgs;
dispatcherTimer = new DispatcherTimer();
dispatcherTimer.Interval = TimeSpan.FromMilliseconds(Delay);
dispatcherTimer.Tick += new EventHandler(OnDispatcherTimerTick);
dispatcherTimer.Start();
}
protected override void OnDetaching()
{
if (dispatcherTimer != null)
{
dispatcherTimer.Stop();
dispatcherTimer = null;
}
base.OnDetaching();
}
private void OnDispatcherTimerTick(object sender, EventArgs e)
{
dispatcherTimer.Stop();
InvokeActions(args);
}
}
Run Code Online (Sandbox Code Playgroud)
其默认延迟为 1 秒(1000 毫秒)。所以现在我们可以在我们的 XAML 中使用它:
<TextBox Margin="5" Name="textBox">
<i:Interaction.Triggers>
<local:DelayedEventTrigger Delay="800" EventName="TextChanged">
<cal:ActionMessage MethodName="DoDelayAction">
<cal:Parameter Value="{Binding ElementName=textBox, Path=Text}" />
</cal:ActionMessage>
</local:DelayedEventTrigger>
</i:Interaction.Triggers>
</TextBox>
Run Code Online (Sandbox Code Playgroud)
在我看来,没有必要使用约定(您可以DelayedEventTrigger
显式使用约定),但是如果您愿意,可以在Bootstrapper
类中配置 Caliburn Parser :
protected override void Configure()
{
base.Configure();
Parser.CreateTrigger = delegate(DependencyObject target, string triggerText)
{
System.Windows.Interactivity.EventTrigger eventTrigger;
if (triggerText == null)
{
ElementConvention elementConvention = ConventionManager.GetElementConvention(target.GetType());
return elementConvention.CreateTrigger();
}
string eventName = triggerText.Replace("[", String.Empty).Replace("]", String.Empty);
if (eventName.StartsWith("Delayed", StringComparison.OrdinalIgnoreCase))
{
eventName = eventName.Replace("DelayedEvent", String.Empty).Trim();
eventTrigger = new DelayedEventTrigger();
}
else
{
eventName = eventName.Replace("Event", String.Empty).Trim();
eventTrigger = new System.Windows.Interactivity.EventTrigger();
}
eventTrigger.EventName = eventName;
return eventTrigger;
};
}
Run Code Online (Sandbox Code Playgroud)
通过添加此代码,您可以使用此约定:
<TextBox Name="textBox" Margin="5"
cal:Message.Attach="[DelayedEvent TextChanged] = [Action DoDelayAction(textBox.Text)]" />
Run Code Online (Sandbox Code Playgroud)
我希望它能帮助你。