2009-06-18 112 views
35

如何讓控件淡入/淡出時變爲可見。WPF Fade動畫

下面是我的失敗嘗試:

<Window x:Class="WadFileTester.Form1" 
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" 
    Name="MyWindow" Title="WAD File SI Checker" Height="386" Width="563" WindowStyle="SingleBorderWindow" DragEnter="Window_DragEnter" DragLeave="Window_DragLeave" DragOver="Window_DragOver" Drop="Window_Drop" AllowDrop="True"> 
    <Window.Resources> 
     <Style TargetType="ListView" x:Key="animatedList"> 
      <Style.Triggers> 
       <DataTrigger Binding="{Binding Visibility}" Value="Visible"> 
        <DataTrigger.EnterActions> 
         <BeginStoryboard> 
          <Storyboard> 
           <DoubleAnimation 
            Storyboard.TargetProperty="Opacity" 
            From="0.0" To="1.0" Duration="0:0:5" 
            /> 
          </Storyboard> 
         </BeginStoryboard> 
        </DataTrigger.EnterActions> 
        <DataTrigger.ExitActions> 
         <BeginStoryboard> 
          <Storyboard> 
           <DoubleAnimation 
            Storyboard.TargetProperty="Opacity" 
            From="1.0" To="0.0" Duration="0:0:5" 
            /> 
          </Storyboard> 
         </BeginStoryboard> 
        </DataTrigger.ExitActions> 
       </DataTrigger> 
      </Style.Triggers> 
     </Style> 
    </Window.Resources> 
    <Grid> 
     <ListView Name="listView1" Style="{StaticResource animatedList}" TabIndex="1" Margin="12,41,12,12" Visibility="Hidden"> 
     </ListView> 
    </Grid> 
</Window> 

回答

53

我不知道如何在純XAML一舉兩得動畫(淡入和淡出)。但簡單的淡出可以實現得比較簡單。用觸發器替換DataTriggers,並刪除ExitActions,因爲它們在淡出方案中沒有意義。這是你將有:

<Style TargetType="FrameworkElement" x:Key="animatedList"> 
    <Setter Property="Visibility" Value="Hidden"/> 
    <Style.Triggers> 
    <Trigger Property="Visibility" Value="Visible"> 
     <Trigger.EnterActions> 
     <BeginStoryboard> 
      <Storyboard> 
      <DoubleAnimation Storyboard.TargetProperty="Opacity" 
          From="0.0" To="1.0" Duration="0:0:0.2"/> 
      </Storyboard> 
     </BeginStoryboard> 
     </Trigger.EnterActions> 
    </Trigger> 
    </Style.Triggers> 
</Style> 

但是,嘿,不要放棄。如果你想支持這兩個動畫,我可以在XAML後面提供小編碼。我們做了一招後,我們會得到你想要的東西通過添加一行代碼在XAML:

<Button Content="Fading button" 
     x:Name="btn" 
     loc:VisibilityAnimation.IsActive="True"/> 

每次我們改變btn.Visibility從可見光到隱藏/摺疊按鈕時會淡出。每次我們改變Visibility時,按鈕都會淡入。這個技巧可以用於任何FrameworkElement(包括ListView :))。

這裏是VisibilityAnimation.IsActive附加屬性的代碼:

public class VisibilityAnimation : DependencyObject 
    { 
    private const int DURATION_MS = 200; 

    private static readonly Hashtable _hookedElements = new Hashtable(); 

    public static readonly DependencyProperty IsActiveProperty = 
     DependencyProperty.RegisterAttached("IsActive", 
     typeof(bool), 
     typeof(VisibilityAnimation), 
     new FrameworkPropertyMetadata(false, new PropertyChangedCallback(OnIsActivePropertyChanged))); 

    public static bool GetIsActive(UIElement element) 
    { 
     if (element == null) 
     { 
     throw new ArgumentNullException("element"); 
     } 

     return (bool)element.GetValue(IsActiveProperty); 
    } 

    public static void SetIsActive(UIElement element, bool value) 
    { 
     if (element == null) 
     { 
     throw new ArgumentNullException("element"); 
     } 
     element.SetValue(IsActiveProperty, value); 
    } 

    static VisibilityAnimation() 
    { 
     UIElement.VisibilityProperty.AddOwner(typeof(FrameworkElement), 
              new FrameworkPropertyMetadata(Visibility.Visible, new PropertyChangedCallback(VisibilityChanged), CoerceVisibility)); 
    } 

    private static void VisibilityChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) 
    { 
     // So what? Ignore. 
    } 

    private static void OnIsActivePropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) 
    { 
     var fe = d as FrameworkElement; 
     if (fe == null) 
     { 
     return; 
     } 
     if (GetIsActive(fe)) 
     { 
     HookVisibilityChanges(fe); 
     } 
     else 
     { 
     UnHookVisibilityChanges(fe); 
     } 
    } 

    private static void UnHookVisibilityChanges(FrameworkElement fe) 
    { 
     if (_hookedElements.Contains(fe)) 
     { 
     _hookedElements.Remove(fe); 
     } 
    } 

    private static void HookVisibilityChanges(FrameworkElement fe) 
    { 
     _hookedElements.Add(fe, false); 
    } 

    private static object CoerceVisibility(DependencyObject d, object baseValue) 
    { 
     var fe = d as FrameworkElement; 
     if (fe == null) 
     { 
     return baseValue; 
     } 

     if (CheckAndUpdateAnimationStartedFlag(fe)) 
     { 
     return baseValue; 
     } 
     // If we get here, it means we have to start fade in or fade out 
     // animation. In any case return value of this method will be 
     // Visibility.Visible. 

     var visibility = (Visibility)baseValue; 

     var da = new DoubleAnimation 
     { 
     Duration = new Duration(TimeSpan.FromMilliseconds(DURATION_MS)) 
     }; 

     da.Completed += (o, e) => 
         { 
          // This will trigger value coercion again 
          // but CheckAndUpdateAnimationStartedFlag() function will reture true 
          // this time, and animation will not be triggered. 
          fe.Visibility = visibility; 
          // NB: Small problem here. This may and probably will brake 
          // binding to visibility property. 
         }; 

     if (visibility == Visibility.Collapsed || visibility == Visibility.Hidden) 
     { 
     da.From = 1.0; 
     da.To = 0.0; 
     } 
     else 
     { 
     da.From = 0.0; 
     da.To = 1.0; 
     } 

     fe.BeginAnimation(UIElement.OpacityProperty, da); 
     return Visibility.Visible; 
    } 

    private static bool CheckAndUpdateAnimationStartedFlag(FrameworkElement fe) 
    { 
     var hookedElement = _hookedElements.Contains(fe); 
     if (!hookedElement) 
     { 
     return true; // don't need to animate unhooked elements. 
     } 

     var animationStarted = (bool) _hookedElements[fe]; 
     _hookedElements[fe] = !animationStarted; 

     return animationStarted; 
    } 
    } 

這裏最重要的是CoerceVisibility()方法。正如你所看到的,我們不允許改變這個屬性,直到淡入淡出的動畫完成。

此代碼既不是線程安全的也沒有缺陷。它唯一的意圖是顯示方向:)。所以請隨時改進,編輯並獲得聲望;)。

+0

可以通過以下方式實現淡入和淡出重複效果:a)將DoubleAnimation上的AutoReverse屬性設置爲True,b)將R epeatBehaviour物業在故事板永遠 – 2010-06-17 02:36:08

+0

你注意到*// /注:小問題在這裏。這可能會或可能會制動 //綁定到知名度屬性。* ...有誰知道如何解決這個問題?我真的希望能夠將可見性綁定到模型值。 – Akku 2012-01-10 15:59:24

2

現在已經很久了,但是你能不能連鎖DoubleAnimations?

<DataTrigger.EnterActions> 
    <BeginStoryboard> 
     <Storyboard> 
      <DoubleAnimation 
       Storyboard.TargetProperty="Opacity" 
       From="0.0" To="1.0" Duration="0:0:5" 
       /> 
      <DoubleAnimation 
       Storyboard.TargetProperty="Opacity" 
       From="1.0" To="0.0" Duration="0:0:5" 
       /> 
     </Storyboard> 
    </BeginStoryboard> 
</DataTrigger.EnterActions> 
2

您可能想嘗試AutoReverse屬性......但我不確定它是否按照您希望的方式工作。 這是我在MSDN上找到的:

當時間軸的AutoReverse屬性設置爲true並且其RepeatBehavior屬性導致它重複時,每個前向迭代後面跟着一個後向迭代。這使得重複一次。例如,一個帶有AutoReverse值爲true並且迭代計數爲2的時間線將向前播放一次,然後向後播放,然後再向前播放,然後再向後播放。

6

我意識到這個問題有點舊,但我現在只讀過它,並且我調整了Anvaka給出的代碼。它支持綁定到可見性(僅當綁定模式設置爲雙向時)。它還支持FadeIn和FadeOut的2個不同的持續時間值。

這裏是類:

public class VisibilityAnimation : DependencyObject 
    { 
    #region Private Variables 

    private static HashSet<UIElement> HookedElements = new HashSet<UIElement>(); 
    private static DoubleAnimation FadeAnimation = new DoubleAnimation(); 
    private static bool SurpressEvent; 
    private static bool Running; 

    #endregion 

    #region Attached Dependencies 

    public static readonly DependencyProperty IsActiveProperty = DependencyProperty.RegisterAttached("IsActive", typeof(bool), typeof(VisibilityAnimation), new FrameworkPropertyMetadata(false, new PropertyChangedCallback(OnIsActivePropertyChanged))); 
    public static bool GetIsActive(UIElement element) 
    { 
     if (element == null) throw new ArgumentNullException("element"); 
     return (bool)element.GetValue(IsActiveProperty); 
    } 
    public static void SetIsActive(UIElement element, bool value) 
    { 
     if (element == null) throw new ArgumentNullException("element"); 
     element.SetValue(IsActiveProperty, value); 
    } 

    public static readonly DependencyProperty FadeInDurationProperty = DependencyProperty.RegisterAttached("FadeInDuration", typeof(double), typeof(VisibilityAnimation), new PropertyMetadata(0.5)); 
    public static double GetFadeInDuration(UIElement e) 
    { 
     if (e == null) throw new ArgumentNullException("element"); 
     return (double)e.GetValue(FadeInDurationProperty); 
    } 
    public static void SetFadeInDuration(UIElement e, double value) 
    { 
     if (e == null) throw new ArgumentNullException("element"); 
     e.SetValue(FadeInDurationProperty, value); 
    } 

    public static readonly DependencyProperty FadeOutDurationProperty = DependencyProperty.RegisterAttached("FadeOutDuration", typeof(double), typeof(VisibilityAnimation), new PropertyMetadata(1.0)); 
    public static double GetFadeOutDuration(UIElement e) 
    { 
     if (e == null) throw new ArgumentNullException("element"); 
     return (double)e.GetValue(FadeOutDurationProperty); 
    } 
    public static void SetFadeOutDuration(UIElement e, double value) 
    { 
     if (e == null) throw new ArgumentNullException("element"); 
     e.SetValue(FadeOutDurationProperty, value); 
    } 

    #endregion 

    #region Callbacks 

    private static void VisibilityChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) 
    { 
     // So what? Ignore. 
     // We only specified a property changed call-back to be able to set a coercion call-back 
    } 

    private static void OnIsActivePropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) 
    { 
     // Get the framework element and leave if it is null 
     var fe = d as FrameworkElement; 
     if (fe == null) return; 

     // Hook the element if IsActive is true and unhook the element if it is false 
     if (GetIsActive(fe)) HookedElements.Add(fe); 
     else HookedElements.Remove(fe); 
    } 

    private static object CoerceVisibility(DependencyObject d, object baseValue) 
    { 
     if (SurpressEvent) return baseValue; // Ignore coercion if we set the SurpressEvent flag 

     var FE = d as FrameworkElement; 
     if (FE == null || !HookedElements.Contains(FE)) return baseValue; // Leave if the element is null or does not belong to our list of hooked elements 

     Running = true; // Set the running flag so that an animation does not change the visibility if another animation was started (Changing Visibility before the 1st animation completed) 

     // If we get here, it means we have to start fade in or fade out animation 
     // In any case return value of this method will be Visibility.Visible 

     Visibility NewValue = (Visibility)baseValue; // Get the new value 

     if (NewValue == Visibility.Visible) FadeAnimation.Duration = new Duration(TimeSpan.FromSeconds((double)d.GetValue(FadeInDurationProperty))); // Get the duration that was set for fade in 
     else FadeAnimation.Duration = new Duration(TimeSpan.FromSeconds((double)d.GetValue(FadeOutDurationProperty))); // Get the duration that was set for fade out 

     // Use an anonymous method to set the Visibility to the new value after the animation completed 
     FadeAnimation.Completed += (obj, args) => 
     { 
     if (FE.Visibility != NewValue && !Running) 
     { 
      SurpressEvent = true; // SuppressEvent flag to skip coercion 
      FE.Visibility = NewValue; 
      SurpressEvent = false; 
      Running = false; // Animation and Visibility change is now complete 
     } 
     }; 

     FadeAnimation.To = (NewValue == Visibility.Collapsed || NewValue == Visibility.Hidden) ? 0 : 1; // Set the to value based on Visibility 

     FE.BeginAnimation(UIElement.OpacityProperty, FadeAnimation); // Start the animation (it will only start after we leave the coercion method) 

     return Visibility.Visible; // We need to return Visible in order to see the fading take place, otherwise it just sets it to Collapsed/Hidden without showing the animation 
    } 

    #endregion 

    static VisibilityAnimation() 
    { 
     // Listen for visibility changes on all elements 
     UIElement.VisibilityProperty.AddOwner(typeof(FrameworkElement), new FrameworkPropertyMetadata(Visibility.Visible, new PropertyChangedCallback(VisibilityChanged), CoerceVisibility)); 
    }  
    } 
18

不能直接使用Visibility屬性的淡出,因爲設置一個觸發上會先隱藏/關閉的控制,那麼它的動畫。所以基本上你會得到一個動畫崩潰的控制=>什麼都沒有。

一個「可靠」的辦法是引入新的依賴項屬性(連接或沒有),說IsOpen,並在其上設置屬性觸發IsOpen=True

EnterAction:
  • 確保可見性設置至可見
  • 淡入不透明度從0到1
ExitAction:
  • 在能見度關鍵幀0中設置爲可見和摺疊/在最後的關鍵幀
  • 淡出隱藏了不透明度從1到0

下面是一個例子:

<Style TargetType="{x:Type local:TCMenu}"> 
    <Style.Resources> 
     <Storyboard x:Key="FadeInMenu"> 
      <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty="(UIElement.Opacity)" Storyboard.TargetName="{x:Null}"> 
       <EasingDoubleKeyFrame KeyTime="0" Value="0"/> 
       <EasingDoubleKeyFrame KeyTime="0:0:0.2" Value="1"/> 
      </DoubleAnimationUsingKeyFrames> 
       <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty="(UIElement.Visibility)" Storyboard.TargetName="{x:Null}"> 
        <DiscreteObjectKeyFrame KeyTime="0:0:0.0" Value="{x:Static Visibility.Visible}"/> 
       </ObjectAnimationUsingKeyFrames> 
      </Storyboard> 
     <Storyboard x:Key="FadeOutMenu"> 
      <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty="(UIElement.Opacity)" Storyboard.TargetName="{x:Null}"> 
       <EasingDoubleKeyFrame KeyTime="0" Value="1"/> 
       <EasingDoubleKeyFrame KeyTime="0:0:0.2" Value="0"/> 
      </DoubleAnimationUsingKeyFrames> 
      <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty="(UIElement.Visibility)" Storyboard.TargetName="{x:Null}"> 
        <DiscreteObjectKeyFrame KeyTime="0:0:0.0" Value="{x:Static Visibility.Visible}"/> 
        <DiscreteObjectKeyFrame KeyTime="0:0:0.2" Value="{x:Static Visibility.Collapsed}"/> 
      </ObjectAnimationUsingKeyFrames> 
     </Storyboard> 
    </Style.Resources> 
    <Style.Triggers> 
     <Trigger Property="IsOpen" Value="true"> 
      <Trigger.EnterActions> 
       <BeginStoryboard Storyboard="{StaticResource FadeInMenu}"/> 
      </Trigger.EnterActions> 
       <Trigger.ExitActions> 
        <BeginStoryboard Storyboard="{StaticResource FadeOutMenu}"/> 
       </Trigger.ExitActions> 
      </Trigger> 
     </Style.Triggers> 
     <Setter Property="Visibility" Value="Collapsed" /> 
</Style> 
+0

偉大的解決方案。因爲我正在使用ViewModels,所以我可以使用DataContext爲null/not null作爲觸發器,在這種情況下,我甚至不需要附加屬性:) – flq 2012-07-27 15:19:20

+1

1+這對我的需求是完美的,謝謝:) – 2013-05-15 12:13:11

3

我一直以一種稍微不同的方式 - 我有一個擴展版本的Ray的this question的答案,它將FadeIn()和FadeOut()擴展方法添加到適當摺疊或顯示元素的所有內容,然後代替使對象可見可以在它們上調用FadeIn()和FadeOut() - 一個它可以處理任何沒有任何特定動畫代碼的元素。

public static T FadeFromTo(this UIElement uiElement, 
            double fromOpacity, double toOpacity, 
            int durationInMilliseconds, bool loopAnimation, bool showOnStart, bool collapseOnFinish) 
    { 
     var timeSpan = TimeSpan.FromMilliseconds(durationInMilliseconds); 
     var doubleAnimation = 
       new DoubleAnimation(fromOpacity, toOpacity, 
            new Duration(timeSpan)); 
      if (loopAnimation) 
       doubleAnimation.RepeatBehavior = RepeatBehavior.Forever; 
      uiElement.BeginAnimation(UIElement.OpacityProperty, doubleAnimation); 
      if (showOnStart) 
      { 
       uiElement.ApplyAnimationClock(UIElement.VisibilityProperty, null); 
       uiElement.Visibility = Visibility.Visible; 
      } 
      if (collapseOnFinish) 
      { 
       var keyAnimation = new ObjectAnimationUsingKeyFrames{Duration = new Duration(timeSpan) }; 
       keyAnimation.KeyFrames.Add(new DiscreteObjectKeyFrame(Visibility.Collapsed, KeyTime.FromTimeSpan(timeSpan))); 
       uiElement.BeginAnimation(UIElement.VisibilityProperty, keyAnimation); 
      } 
      return uiElement; 
    } 

    public static T FadeIn(this UIElement uiElement, int durationInMilliseconds) 
    { 
     return uiElement.FadeFromTo(0, 1, durationInMilliseconds, false, true, false); 
    } 

    public static T FadeOut(this UIElement uiElement, int durationInMilliseconds) 
    { 
     return uiElement.FadeFromTo(1, 0, durationInMilliseconds, false, false, true); 
    } 
3

這是通過使用行爲

class AnimatedVisibilityFadeBehavior : Behavior<Border> 
    { 
     public Duration AnimationDuration { get; set; } 
     public Visibility InitialState { get; set; } 

     DoubleAnimation m_animationOut; 
     DoubleAnimation m_animationIn; 

     protected override void OnAttached() 
     { 
     base.OnAttached(); 

     m_animationIn = new DoubleAnimation(1, AnimationDuration, FillBehavior.HoldEnd); 
     m_animationOut = new DoubleAnimation(0, AnimationDuration, FillBehavior.HoldEnd); 
     m_animationOut.Completed += (sender, args) => 
      { 
       AssociatedObject.SetCurrentValue(Border.VisibilityProperty, Visibility.Collapsed); 
      }; 

     AssociatedObject.SetCurrentValue(Border.VisibilityProperty, 
              InitialState == Visibility.Collapsed 
              ? Visibility.Collapsed 
              : Visibility.Visible); 

     Binding.AddTargetUpdatedHandler(AssociatedObject, Updated); 
     } 

     private void Updated(object sender, DataTransferEventArgs e) 
     { 
     var value = (Visibility)AssociatedObject.GetValue(Border.VisibilityProperty); 
     switch (value) 
     { 
      case Visibility.Collapsed: 
       AssociatedObject.SetCurrentValue(Border.VisibilityProperty, Visibility.Visible); 
       AssociatedObject.BeginAnimation(Border.OpacityProperty, m_animationOut); 
       break; 
      case Visibility.Visible: 
       AssociatedObject.BeginAnimation(Border.OpacityProperty, m_animationIn); 
       break; 
     } 
     } 
    } 

這最好做的是專門被應用到邊界 - 我還沒有嘗試過的用戶控件,但我希望這同樣適用。

要使用它,你需要交融互動命名空間:

xmlns:i="clr-namespace:System.Windows.Interactivity;assembly=System.Windows.Interactivity" 

,並使用你想要的行爲對邊境此標記:

<i:Interaction.Behaviors> 
       <Interactivity:AnimatedVisibilityFadeBehavior AnimationDuration="0:0:0.3" InitialState="Collapsed" /> 
</i:Interaction.Behaviors> 

你需要添加行爲類的命名空間也是如此。