如何使用约定实现 Caliburn.Micro 操作的延迟

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 以某种方式为此编写一个约定吗?
或者也许是一种不同的、更具可扩展性的方法?

Il *_*Vic 6

正如您在文档页面中所读到的那样

操作功能利用 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)

我希望它能帮助你。