使用Reactive Extensions更改不透明度

时间:2015-08-19 17:50:34

标签: c# winforms system.reactive opacity

  

< rant>是的,我知道这在WPF中更容易实现。我听到了很多。可悲的是,这是不可能的。   < /咆哮>

我正在编写一个WinForms应用程序,我需要“淡化”一个控件&出。在WinForms中,透明度几乎是不可能的,所以我试图使用不透明度:想法是在一段时间内更改每个子控件的ForeColor属性的Alpha通道。这似乎是我的Reactive Extensions工作的最佳时机!

我可能的解决方案是:

private void FadeOut()
{
   // our list of values for the alpha channel (255 to 0)
   var range = Enumerable.Range(0,256).Reverse().ToList();

   // how long between each setting of the alpha (huge value for example only)
   var delay = Timespan.FromSeconds(0.5);

   // our "trigger" sequence, every half second
   Observable.Interval(delay)
        // paired with the values from the range - we just keep the range
          .Zip(range, (lhs, rhs) => rhs)
        // make OnNext changes on the UI thread
          .ObserveOn(SynchronizationContext.Current)
        // do this every time a value is rec'd from the sequence
          .Subscribe(
              // set the alpha value
              onNext:ChangeAlphaValues, 
              // when we're done, really hide the control
              onCompleted: () => Visible = false, 
              // good citizenry
              onError: FailGracefully);
}

// no need to iterate the controls more than once - store them here
private IEnumerable<Control> _controls;

private void ChangeAlphaValues(int alpha)
{
    // get all the controls (via an extension method)
    var controls = _controls ?? this.GetAllChildControls(typeof (Control));

    // iterate all controls and change the alpha
    foreach (var control in controls)
       control.ForeColor = Color.FromArgb(alpha, control.ForeColor);
}

...看起来令人印象深刻,但它不起作用。真正令人印象深刻的是它不能以两种方式工作!如果我坚持下去,我的下一次绩效评估会得到“远远超过”。 : - )

  1. (不是问题的Rx部分)alpha值实际上似乎没有任何区别。尽管设置了值,但显示看起来都一样。
  2. 如果在序列完成之前关闭窗口,则会出现以下错误:
  3.   

    System.InvalidOperationException未处理   消息:System.Reactive.Core.dll中发生未处理的“System.InvalidOperationException”类型异常   附加信息:在创建窗口句柄之前,无法在控件上调用Invoke或BeginInvoke。

    我认为这是取消令牌会派上用场的地方 - 但我不知道如何实现它。

    我需要指导:

    如果序列仍在运行,如何优雅地关闭窗口(即没有抛出错误),以及如何在颜色更改后实际显示的颜色的alpha值。< / p>

    ...或者这可能是错误的做法。

    我愿意接受其他建议。

1 个答案:

答案 0 :(得分:3)

我无法帮助您使用WinForms透明度部分。也许你应该将问题分成两部分。

但是对于Rx部分,您只需要在窗口关闭时取消订阅。 Subscribe会返回IDisposable。您应该在Closed事件中处理它。

而且,由于我假设在关闭窗口之前可能会多次调用此淡入淡出动画,因此我们可以使用Rx助手SerialDisposable

最后,Interval实际上返回一个计数。我们可以通过计算所需的alpha来使用它来简化你的观察。

private SerialDisposable _fadeAnimation = new SerialDisposable();

private void FadeOut()
{
    // how long between each setting of the alpha (huge value for example only)
    var delay = Timespan.FromSeconds(0.5);

    _fadeAnimation.Disposable = Observable
        .Interval(delay)
        // 256 animation steps
        .Take(256)
        // calculate alpha
        .Select(i => 255 - i)
        .ObserveOn(SynchronizationContext.Current)
        .Subscribe(
            // set the alpha value
            onNext:ChangeAlphaValues, 
            // when we're done, really hide the control
            onCompleted: () => Visible = false, 
            // good citizenry
            onError: FailGracefully);
}

private void Form1_Closed()
{
    _fadeAnimation.Dispose();
}