<!-- // (c) Copyright Microsoft Corporation. // This source is subject to the Microsoft Public License (Ms-PL). // Please see http://go.microsoft.com/fwlink/?LinkID=131993 for details. // All other rights reserved. --> <ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:controls="clr-namespace:System.Windows.Controls" xmlns:vsm="clr-namespace:System.Windows;assembly=System.Windows"> <Style TargetType="controls:ChildWindow"> <Setter Property="IsTabStop" Value="false"/> <Setter Property="TabNavigation" Value="Cycle"/> <Setter Property="HorizontalAlignment" Value="Center"/> <Setter Property="VerticalAlignment" Value="Center"/> <Setter Property="HorizontalContentAlignment" Value="Stretch"/> <Setter Property="VerticalContentAlignment" Value="Stretch"/> <Setter Property="BorderThickness" Value="1"/> <Setter Property="BorderBrush"> <Setter.Value> <LinearGradientBrush StartPoint="0.5,0" EndPoint="0.5,1"> <GradientStop Color="#FFA3AEB9" Offset="0"/> <GradientStop Color="#FF8399A9" Offset="0.375"/> <GradientStop Color="#FF718597" Offset="0.375"/> <GradientStop Color="#FF617584" Offset="1"/> </LinearGradientBrush> </Setter.Value> </Setter> <Setter Property="OverlayBrush"> <Setter.Value> <SolidColorBrush Color="#7F000000"/> </Setter.Value> </Setter> <Setter Property="OverlayOpacity" Value="1"/> <Setter Property="Template"> <Setter.Value> <ControlTemplate TargetType="controls:ChildWindow"> <Grid x:Name="Root"> <Grid.Resources> <Style x:Key="ButtonStyle" TargetType="Button"> <Setter Property="Background" Value="#FF1F3B53"/> <Setter Property="Foreground" Value="#FF000000"/> <Setter Property="Padding" Value="3"/> <Setter Property="BorderThickness" Value="1"/> <Setter Property="BorderBrush"> <Setter.Value> <LinearGradientBrush EndPoint="0.5,1" StartPoint="0.5,0"> <GradientStop Color="#FFA3AEB9" Offset="0"/> <GradientStop Color="#FF8399A9" Offset="0.375"/> <GradientStop Color="#FF718597" Offset="0.375"/> <GradientStop Color="#FF617584" Offset="1"/> </LinearGradientBrush> </Setter.Value> </Setter> <Setter Property="Template"> <Setter.Value> <ControlTemplate TargetType="Button"> <Grid HorizontalAlignment="Center" VerticalAlignment="Center" Width="15" Height="14" Background="#02FFFFFF" x:Name="grid"> <vsm:VisualStateManager.VisualStateGroups> <vsm:VisualStateGroup x:Name="CommonStates"> <vsm:VisualState x:Name="Normal"/> <vsm:VisualState x:Name="MouseOver"> <Storyboard> <ObjectAnimationUsingKeyFrames Storyboard.TargetName="X_Fuzz2" Storyboard.TargetProperty="Visibility"> <DiscreteObjectKeyFrame KeyTime="0" Value="Visible" /> </ObjectAnimationUsingKeyFrames> <ObjectAnimationUsingKeyFrames Storyboard.TargetName="X_Fuzz1" Storyboard.TargetProperty="Visibility"> <DiscreteObjectKeyFrame KeyTime="0" Value="Visible" /> </ObjectAnimationUsingKeyFrames> <ObjectAnimationUsingKeyFrames Storyboard.TargetName="X_Fuzz0" Storyboard.TargetProperty="Visibility"> <DiscreteObjectKeyFrame KeyTime="0" Value="Visible" /> </ObjectAnimationUsingKeyFrames> <DoubleAnimation Duration="0" Storyboard.TargetName="X" Storyboard.TargetProperty="Opacity" To="0.95"/> </Storyboard> </vsm:VisualState> <vsm:VisualState x:Name="Pressed"> <Storyboard> <DoubleAnimation Duration="0" Storyboard.TargetName="X" Storyboard.TargetProperty="Opacity" To="0.85"/> <ObjectAnimationUsingKeyFrames Storyboard.TargetName="X_Fuzz2" Storyboard.TargetProperty="Visibility"> <DiscreteObjectKeyFrame KeyTime="0" Value="Visible" /> </ObjectAnimationUsingKeyFrames> <ObjectAnimationUsingKeyFrames Storyboard.TargetName="X_Fuzz1" Storyboard.TargetProperty="Visibility"> <DiscreteObjectKeyFrame KeyTime="0" Value="Visible" /> </ObjectAnimationUsingKeyFrames> <ObjectAnimationUsingKeyFrames Storyboard.TargetName="X_Fuzz0" Storyboard.TargetProperty="Visibility"> <DiscreteObjectKeyFrame KeyTime="0" Value="Visible" /> </ObjectAnimationUsingKeyFrames> </Storyboard> </vsm:VisualState> <vsm:VisualState x:Name="Disabled"> <Storyboard> <DoubleAnimation Duration="0" Storyboard.TargetName="X" Storyboard.TargetProperty="Opacity" To="0.5"/> </Storyboard> </vsm:VisualState> </vsm:VisualStateGroup> </vsm:VisualStateManager.VisualStateGroups> <Path HorizontalAlignment="Center" Margin="0,-1,0,0" Width="9" Fill="#14C51900" Stretch="Fill" Data="F1 M 6.742676,3.852539 L 9.110840,1.559570 L 8.910645,0.500000 L 6.838379,0.500000 L 4.902832,2.435547 L 2.967285,0.500000 L 0.895020,0.500000 L 0.694824,1.559570 L 3.062988,3.852539 L 0.527832,6.351563 L 0.689941,7.600586 L 2.967285,7.600586 L 4.897949,5.575195 L 6.854004,7.600586 L 9.115723,7.600586 L 9.277832,6.351563 L 6.742676,3.852539 Z" x:Name="X_Fuzz2" Stroke="#14C51900" Height="8" VerticalAlignment="Center" Opacity="1" RenderTransformOrigin="0.5,0.5" Visibility="Collapsed"> <Path.RenderTransform> <TransformGroup> <ScaleTransform ScaleX="1.3" ScaleY="1.3"/> </TransformGroup> </Path.RenderTransform> </Path> <Path HorizontalAlignment="Center" Margin="0,-1,0,0" Width="9" Fill="#1EC51900" Stretch="Fill" Data="F1 M 6.742676,3.852539 L 9.110840,1.559570 L 8.910645,0.500000 L 6.838379,0.500000 L 4.902832,2.435547 L 2.967285,0.500000 L 0.895020,0.500000 L 0.694824,1.559570 L 3.062988,3.852539 L 0.527832,6.351563 L 0.689941,7.600586 L 2.967285,7.600586 L 4.897949,5.575195 L 6.854004,7.600586 L 9.115723,7.600586 L 9.277832,6.351563 L 6.742676,3.852539 Z" x:Name="X_Fuzz1" Stroke="#1EC51900" Height="8" VerticalAlignment="Center" Opacity="1" RenderTransformOrigin="0.5,0.5" Visibility="Collapsed"> <Path.RenderTransform> <TransformGroup> <ScaleTransform ScaleX="1.1" ScaleY="1.1"/> </TransformGroup> </Path.RenderTransform> </Path> <Path HorizontalAlignment="Center" Margin="0,-1,0,0" Width="9" Fill="#FFC51900" Stretch="Fill" Data="F1 M 6.742676,3.852539 L 9.110840,1.559570 L 8.910645,0.500000 L 6.838379,0.500000 L 4.902832,2.435547 L 2.967285,0.500000 L 0.895020,0.500000 L 0.694824,1.559570 L 3.062988,3.852539 L 0.527832,6.351563 L 0.689941,7.600586 L 2.967285,7.600586 L 4.897949,5.575195 L 6.854004,7.600586 L 9.115723,7.600586 L 9.277832,6.351563 L 6.742676,3.852539 Z" x:Name="X_Fuzz0" Stroke="#FFC51900" Height="8" VerticalAlignment="Center" Opacity="1" Visibility="Collapsed"/> <Path HorizontalAlignment="Center" Margin="0,-1,0,0" Width="9" Fill="#FFFFFFFF" Stretch="Fill" Data="F1 M 6.742676,3.852539 L 9.110840,1.559570 L 8.910645,0.500000 L 6.838379,0.500000 L 4.902832,2.435547 L 2.967285,0.500000 L 0.895020,0.500000 L 0.694824,1.559570 L 3.062988,3.852539 L 0.527832,6.351563 L 0.689941,7.600586 L 2.967285,7.600586 L 4.897949,5.575195 L 6.854004,7.600586 L 9.115723,7.600586 L 9.277832,6.351563 L 6.742676,3.852539 Z" x:Name="X" Height="8" VerticalAlignment="Center" Opacity="0.7"> <Path.Stroke> <LinearGradientBrush EndPoint="0.5,1" StartPoint="0.5,0"> <GradientStop Color="#FF313131" Offset="1"/> <GradientStop Color="#FF8E9092" Offset="0"/> </LinearGradientBrush> </Path.Stroke> </Path> </Grid> </ControlTemplate> </Setter.Value> </Setter> </Style> </Grid.Resources> <vsm:VisualStateManager.VisualStateGroups> <vsm:VisualStateGroup x:Name="WindowStates"> <vsm:VisualState x:Name="Open"> <Storyboard> <DoubleAnimationUsingKeyFrames BeginTime="0" Storyboard.TargetName="Overlay" Storyboard.TargetProperty="Opacity"> <EasingDoubleKeyFrame KeyTime="0" Value="0"/> <EasingDoubleKeyFrame KeyTime="00:00:00.3" Value="1"/> </DoubleAnimationUsingKeyFrames> <DoubleAnimationUsingKeyFrames BeginTime="0" Storyboard.TargetName="ContentRoot" Storyboard.TargetProperty="(RenderTransform).(Children)[0].ScaleX"> <SplineDoubleKeyFrame KeyTime="0" Value="0"/> <SplineDoubleKeyFrame KeyTime="00:00:00.25" Value="0"/> <SplineDoubleKeyFrame KeyTime="00:00:00.4" Value="1"/> <SplineDoubleKeyFrame KeySpline="0,0,0.5,1" KeyTime="00:00:00.45" Value="1.05"/> <SplineDoubleKeyFrame KeyTime="00:00:00.55" Value="1"/> </DoubleAnimationUsingKeyFrames> <DoubleAnimationUsingKeyFrames BeginTime="0" Storyboard.TargetName="ContentRoot" Storyboard.TargetProperty="(RenderTransform).(Children)[0].ScaleY"> <SplineDoubleKeyFrame KeyTime="0" Value="0"/> <SplineDoubleKeyFrame KeyTime="00:00:00.25" Value="0"/> <SplineDoubleKeyFrame KeyTime="00:00:00.4" Value="1"/> <SplineDoubleKeyFrame KeySpline="0,0,0.5,1" KeyTime="00:00:00.45" Value="1.05"/> <SplineDoubleKeyFrame KeyTime="00:00:00.55" Value="1"/> </DoubleAnimationUsingKeyFrames> </Storyboard> </vsm:VisualState> <vsm:VisualState x:Name="Closed"> <Storyboard> <DoubleAnimationUsingKeyFrames BeginTime="0" Storyboard.TargetName="Overlay" Storyboard.TargetProperty="Opacity"> <EasingDoubleKeyFrame KeyTime="0" Value="1"/> <EasingDoubleKeyFrame KeyTime="00:00:00.3" Value="0"/> </DoubleAnimationUsingKeyFrames> <DoubleAnimationUsingKeyFrames BeginTime="0" Storyboard.TargetName="ContentRoot" Storyboard.TargetProperty="(RenderTransform).(Children)[0].ScaleX"> <SplineDoubleKeyFrame KeyTime="00:00:00.2" Value="1"/> <SplineDoubleKeyFrame KeyTime="00:00:00.25" Value="1.05"/> <SplineDoubleKeyFrame KeyTime="00:00:00.45" Value="0"/> </DoubleAnimationUsingKeyFrames> <DoubleAnimationUsingKeyFrames BeginTime="0" Storyboard.TargetName="ContentRoot" Storyboard.TargetProperty="(RenderTransform).(Children)[0].ScaleY"> <SplineDoubleKeyFrame KeyTime="00:00:00.2" Value="1"/> <SplineDoubleKeyFrame KeyTime="00:00:00.25" Value="1.05"/> <SplineDoubleKeyFrame KeyTime="00:00:00.45" Value="0"/> </DoubleAnimationUsingKeyFrames> </Storyboard> </vsm:VisualState> </vsm:VisualStateGroup> </vsm:VisualStateManager.VisualStateGroups> <Grid x:Name="Overlay" HorizontalAlignment="Stretch" VerticalAlignment="Top" Margin="0" Background="{TemplateBinding OverlayBrush}" Opacity="{TemplateBinding OverlayOpacity}"/> <Grid x:Name="ContentRoot" HorizontalAlignment="{TemplateBinding HorizontalAlignment}" VerticalAlignment="{TemplateBinding VerticalAlignment}" RenderTransformOrigin="0.5,0.5" Height="{TemplateBinding Height}" Width="{TemplateBinding Width}"> <Grid.RenderTransform> <TransformGroup> <ScaleTransform /> <SkewTransform /> <RotateTransform /> <TranslateTransform /> </TransformGroup> </Grid.RenderTransform> <Border BorderThickness="1" CornerRadius="2" BorderBrush="#14000000" Background="#14000000" HorizontalAlignment="Stretch" VerticalAlignment="Stretch" Margin="-1"/> <Border BorderThickness="1" CornerRadius="2.25" BorderBrush="#0F000000" Background="#0F000000" HorizontalAlignment="Stretch" Margin="-2" VerticalAlignment="Stretch"/> <Border BorderThickness="1" CornerRadius="2.5" BorderBrush="#0C000000" Background="#0C000000" HorizontalAlignment="Stretch" Margin="-3" VerticalAlignment="Stretch"/> <Border BorderThickness="1" CornerRadius="2.75" BorderBrush="#0A000000" Background="#0A000000" HorizontalAlignment="Stretch" Margin="-4" VerticalAlignment="Stretch"/> <Border Background="#FFFFFFFF" BorderThickness="{TemplateBinding BorderThickness}" BorderBrush="{TemplateBinding BorderBrush}" CornerRadius="2"> <Border CornerRadius="1.5" Margin="1"> <Border.Background> <LinearGradientBrush EndPoint="0.5,1" StartPoint="0.5,0"> <GradientStop Color="#FFE5E8EB" Offset="1"/> <GradientStop Color="#FFF6F8F9" Offset="0"/> </LinearGradientBrush> </Border.Background> <Grid> <Grid.RowDefinitions> <RowDefinition Height="Auto"/> <RowDefinition/> </Grid.RowDefinitions> <Border x:Name="Chrome" Width="Auto" BorderBrush="#FFFFFFFF" BorderThickness="0,0,0,1"> <Border.Background> <LinearGradientBrush EndPoint="0.5,0.528" StartPoint="0.5,0"> <GradientStop Color="#FFE5E8EB" Offset="1"/> <GradientStop Color="#FFFEFEFE" Offset="0"/> </LinearGradientBrush> </Border.Background> <Grid Height="Auto" Width="Auto"> <Grid.ColumnDefinitions> <ColumnDefinition/> <ColumnDefinition Width="30"/> </Grid.ColumnDefinitions> <ContentControl Content="{TemplateBinding Title}" IsTabStop="False" FontWeight="Bold" HorizontalAlignment="Stretch" VerticalAlignment="Center" Margin="6,0,6,0"/> <Button x:Name="CloseButton" Grid.Column="1" IsTabStop="False" HorizontalAlignment="Center" VerticalAlignment="Center" Width="15" Height="14" Style="{StaticResource ButtonStyle}"/> </Grid> </Border> <Border Background="{TemplateBinding Background}" Margin="7" Grid.Row="1"> <ContentPresenter x:Name="ContentPresenter" Content="{TemplateBinding Content}" ContentTemplate="{TemplateBinding ContentTemplate}" HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}" VerticalAlignment="{TemplateBinding VerticalContentAlignment}"/> </Border> </Grid> </Border> </Border> </Grid> </Grid> </ControlTemplate> </Setter.Value> </Setter> </Style> </ResourceDictionary>
?
?
代码文件:
// (c) Copyright Microsoft Corporation. // This source is subject to the Microsoft Public License (Ms-PL). // Please see http://go.microsoft.com/fwlink/?LinkID=131993 for details. // All other rights reserved. using System.Collections; using System.ComponentModel; using System.Diagnostics; using System.Windows.Automation; using System.Windows.Automation.Peers; using System.Windows.Controls.Primitives; using System.Windows.Input; using System.Windows.Media; using System.Windows.Media.Animation; namespace System.Windows.Controls { /// <summary> /// Provides a window that can be displayed over a parent window and blocks /// interaction with the parent window. /// </summary> /// <QualityBand>Preview</QualityBand> [TemplatePart(Name = PART_Chrome, Type = typeof(FrameworkElement))] [TemplatePart(Name = PART_CloseButton, Type = typeof(ButtonBase))] [TemplatePart(Name = PART_ContentPresenter, Type = typeof(FrameworkElement))] [TemplatePart(Name = PART_ContentRoot, Type = typeof(FrameworkElement))] [TemplatePart(Name = PART_Overlay, Type = typeof(Panel))] [TemplatePart(Name = PART_Root, Type = typeof(FrameworkElement))] [TemplateVisualState(Name = VSMSTATE_StateClosed, GroupName = VSMGROUP_Window)] [TemplateVisualState(Name = VSMSTATE_StateOpen, GroupName = VSMGROUP_Window)] public class ChildWindow : ContentControl { #region Static Fields and Constants /// <summary> /// The name of the Chrome template part. /// </summary> private const string PART_Chrome = "Chrome"; /// <summary> /// The name of the CloseButton template part. /// </summary> private const string PART_CloseButton = "CloseButton"; /// <summary> /// The name of the ContentPresenter template part. /// </summary> private const string PART_ContentPresenter = "ContentPresenter"; /// <summary> /// The name of the ContentRoot template part. /// </summary> private const string PART_ContentRoot = "ContentRoot"; /// <summary> /// The name of the Overlay template part. /// </summary> private const string PART_Overlay = "Overlay"; /// <summary> /// The name of the Root template part. /// </summary> private const string PART_Root = "Root"; /// <summary> /// The name of the WindowStates VSM group. /// </summary> private const string VSMGROUP_Window = "WindowStates"; /// <summary> /// The name of the Closing VSM state. /// </summary> private const string VSMSTATE_StateClosed = "Closed"; /// <summary> /// The name of the Opening VSM state. /// </summary> private const string VSMSTATE_StateOpen = "Open"; /// <summary> /// Stores the previous value of RootVisual.IsEnabled. /// </summary> private static bool RootVisual_PrevEnabledState = true; /// <summary> /// Stores a count of the number of open ChildWindow instances. /// </summary> private static int OpenChildWindowCount = 0; #region public bool HasCloseButton /// <summary> /// Gets or sets a value indicating whether the /// <see cref="T:System.Windows.Controls.ChildWindow" /> has a close /// button. /// </summary> /// <value> /// True if the child window has a close button; otherwise, false. The /// default is true. /// </value> public bool HasCloseButton { get { return (bool)GetValue(HasCloseButtonProperty); } set { SetValue(HasCloseButtonProperty, value); } } /// <summary> /// Identifies the /// <see cref="P:System.Windows.Controls.ChildWindow.HasCloseButton" /> /// dependency property. /// </summary> /// <value> /// The identifier for the /// <see cref="P:System.Windows.Controls.ChildWindow.HasCloseButton" /> /// dependency property. /// </value> public static readonly DependencyProperty HasCloseButtonProperty = DependencyProperty.Register( "HasCloseButton", typeof(bool), typeof(ChildWindow), new PropertyMetadata(true, OnHasCloseButtonPropertyChanged)); /// <summary> /// HasCloseButtonProperty PropertyChangedCallback call back static function. /// </summary> /// <param name="d">ChildWindow object whose HasCloseButton property is changed.</param> /// <param name="e">DependencyPropertyChangedEventArgs which contains the old and new values.</param> private static void OnHasCloseButtonPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) { ChildWindow cw = (ChildWindow)d; if (cw.CloseButton != null) { if ((bool)e.NewValue) { cw.CloseButton.Visibility = Visibility.Visible; } else { cw.CloseButton.Visibility = Visibility.Collapsed; } } } #endregion public bool HasCloseButton #region public Brush OverlayBrush /// <summary> /// Gets or sets the visual brush that is used to cover the parent /// window when the child window is open. /// </summary> /// <value> /// The visual brush that is used to cover the parent window when the /// <see cref="T:System.Windows.Controls.ChildWindow" /> is open. The /// default is null. /// </value> public Brush OverlayBrush { get { return (Brush)GetValue(OverlayBrushProperty); } set { SetValue(OverlayBrushProperty, value); } } /// <summary> /// Identifies the /// <see cref="P:System.Windows.Controls.ChildWindow.OverlayBrush" /> /// dependency property. /// </summary> /// <value> /// The identifier for the /// <see cref="P:System.Windows.Controls.ChildWindow.OverlayBrush" /> /// dependency property. /// </value> public static readonly DependencyProperty OverlayBrushProperty = DependencyProperty.Register( "OverlayBrush", typeof(Brush), typeof(ChildWindow), new PropertyMetadata(OnOverlayBrushPropertyChanged)); /// <summary> /// OverlayBrushProperty PropertyChangedCallback call back static function. /// </summary> /// <param name="d">ChildWindow object whose OverlayBrush property is changed.</param> /// <param name="e">DependencyPropertyChangedEventArgs which contains the old and new values.</param> private static void OnOverlayBrushPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) { ChildWindow cw = (ChildWindow)d; if (cw.Overlay != null) { cw.Overlay.Background = (Brush)e.NewValue; } } #endregion public Brush OverlayBrush #region public double OverlayOpacity /// <summary> /// Gets or sets the opacity of the overlay brush that is used to cover /// the parent window when the child window is open. /// </summary> /// <value> /// The opacity of the overlay brush that is used to cover the parent /// window when the <see cref="T:System.Windows.Controls.ChildWindow" /> /// is open. The default is 1.0. /// </value> public double OverlayOpacity { get { return (double)GetValue(OverlayOpacityProperty); } set { SetValue(OverlayOpacityProperty, value); } } /// <summary> /// Identifies the /// <see cref="P:System.Windows.Controls.ChildWindow.OverlayOpacity" /> /// dependency property. /// </summary> /// <value> /// The identifier for the /// <see cref="P:System.Windows.Controls.ChildWindow.OverlayOpacity" /> /// dependency property. /// </value> public static readonly DependencyProperty OverlayOpacityProperty = DependencyProperty.Register( "OverlayOpacity", typeof(double), typeof(ChildWindow), new PropertyMetadata(OnOverlayOpacityPropertyChanged)); /// <summary> /// OverlayOpacityProperty PropertyChangedCallback call back static function. /// </summary> /// <param name="d">ChildWindow object whose OverlayOpacity property is changed.</param> /// <param name="e">DependencyPropertyChangedEventArgs which contains the old and new values.</param> private static void OnOverlayOpacityPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) { ChildWindow cw = (ChildWindow)d; if (cw.Overlay != null) { cw.Overlay.Opacity = (double)e.NewValue; } } #endregion public double OverlayOpacity #region private static Control RootVisual /// <summary> /// Gets the root visual element. /// </summary> private static Control RootVisual { get { return Application.Current == null ? null : (Application.Current.RootVisual as Control); } } #endregion private static Control RootVisual #region public object Title /// <summary> /// Gets or sets the title that is displayed in the frame of the /// <see cref="T:System.Windows.Controls.ChildWindow" />. /// </summary> /// <value> /// The title displayed at the top of the window. The default is null. /// </value> public object Title { get { return GetValue(TitleProperty); } set { SetValue(TitleProperty, value); } } /// <summary> /// Identifies the /// <see cref="P:System.Windows.Controls.ChildWindow.Title" /> /// dependency property. /// </summary> /// <value> /// The identifier for the /// <see cref="P:System.Windows.Controls.ChildWindow.Title" /> /// dependency property. /// </value> public static readonly DependencyProperty TitleProperty = DependencyProperty.Register( "Title", typeof(object), typeof(ChildWindow), null); #endregion public object Title #endregion Static Fields and Constants #region Member Fields /// <summary> /// Private accessor for the Chrome. /// </summary> private FrameworkElement _chrome; /// <summary> /// Private accessor for the click point on the chrome. /// </summary> private Point _clickPoint; /// <summary> /// Private accessor for the Closing storyboard. /// </summary> private Storyboard _closed; /// <summary> /// Private accessor for the ContentPresenter. /// </summary> private FrameworkElement _contentPresenter; /// <summary> /// Private accessor for the translate transform that needs to be applied on to the ContentRoot. /// </summary> private TranslateTransform _contentRootTransform; /// <summary> /// Content area desired width. /// </summary> private double _desiredContentWidth; /// <summary> /// Content area desired height. /// </summary> private double _desiredContentHeight; /// <summary> /// Desired margin for the window. /// </summary> private Thickness _desiredMargin; /// <summary> /// Private accessor for the Dialog Result property. /// </summary> private bool? _dialogresult; /// <summary> /// Private accessor for the ChildWindow InteractionState. /// </summary> private WindowInteractionState _interactionState; /// <summary> /// Boolean value that specifies whether the application is exit or not. /// </summary> private bool _isAppExit; /// <summary> /// Boolean value that specifies whether the window is in closing state or not. /// </summary> private bool _isClosing; /// <summary> /// Boolean value that specifies whether the window is opened. /// </summary> private bool _isOpen; /// <summary> /// Private accessor for the Opening storyboard. /// </summary> private Storyboard _opened; /// <summary> /// Boolean value that specifies whether the mouse is captured or not. /// </summary> private bool _isMouseCaptured; /// <summary> /// Boolean value that specifies whether we are listening to RootVisual.GotFocus. /// </summary> private bool _attachedRootVisualListener; /// <summary> /// Private accessor for the Root of the window. /// </summary> private FrameworkElement _root; /// <summary> /// Private accessor for the position of the window with respect to RootVisual. /// </summary> private Point _windowPosition; #endregion Member Fields #region Constructors /// <summary> /// Initializes a new instance of the /// <see cref="T:System.Windows.Controls.ChildWindow" /> class. /// </summary> public ChildWindow() { this.DefaultStyleKey = typeof(ChildWindow); this.InteractionState = WindowInteractionState.NotResponding; } #endregion Constructors #region Events /// <summary> /// Occurs when the <see cref="T:System.Windows.Controls.ChildWindow" /> /// is closed. /// </summary> public event EventHandler Closed; /// <summary> /// Occurs when the <see cref="T:System.Windows.Controls.ChildWindow" /> /// is closing. /// </summary> public event EventHandler<CancelEventArgs> Closing; #endregion Events #region Properties /// <summary> /// Gets the internal accessor for the ContentRoot of the window. /// </summary> internal FrameworkElement ContentRoot { get; private set; } /// <summary> /// Gets or sets a value indicating whether the /// <see cref="T:System.Windows.Controls.ChildWindow" /> was accepted or /// canceled. /// </summary> /// <value> /// True if the child window was accepted; false if the child window was /// canceled. The default is null. /// </value> [TypeConverter(typeof(NullableBoolConverter))] public bool? DialogResult { get { return this._dialogresult; } set { if (this._dialogresult != value) { this._dialogresult = value; this.Close(); } } } /// <summary> /// Gets the internal accessor for the PopUp of the window. /// </summary> internal Popup ChildWindowPopup { get; private set; } /// <summary> /// Gets the internal accessor for the close button of the window. /// </summary> internal ButtonBase CloseButton { get; private set; } /// <summary> /// Gets the InteractionState for the ChildWindow. /// </summary> internal WindowInteractionState InteractionState { get { return this._interactionState; } private set { if (this._interactionState != value) { WindowInteractionState oldValue = this._interactionState; this._interactionState = value; ChildWindowAutomationPeer peer = ChildWindowAutomationPeer.FromElement(this) as ChildWindowAutomationPeer; if (peer != null) { peer.RaiseInteractionStatePropertyChangedEvent(oldValue, this._interactionState); } } } } /// <summary> /// Gets a value indicating whether the PopUp is open or not. /// </summary> private bool IsOpen { get { return (this.ChildWindowPopup != null && this.ChildWindowPopup.IsOpen); } } /// <summary> /// Gets the internal accessor for the overlay of the window. /// </summary> internal Panel Overlay { get; private set; } #endregion Properties #region Static Methods /// <summary> /// Inverts the input matrix. /// </summary> /// <param name="matrix">The matrix values that is to be inverted.</param> /// <returns>Returns a value indicating whether the inversion was successful or not.</returns> private static bool InvertMatrix(ref Matrix matrix) { double determinant = (matrix.M11 * matrix.M22) - (matrix.M12 * matrix.M21); if (determinant == 0.0) { return false; } Matrix matCopy = matrix; matrix.M11 = matCopy.M22 / determinant; matrix.M12 = -1 * matCopy.M12 / determinant; matrix.M21 = -1 * matCopy.M21 / determinant; matrix.M22 = matCopy.M11 / determinant; matrix.OffsetX = ((matCopy.OffsetY * matCopy.M21) - (matCopy.OffsetX * matCopy.M22)) / determinant; matrix.OffsetY = ((matCopy.OffsetX * matCopy.M12) - (matCopy.OffsetY * matCopy.M11)) / determinant; return true; } #endregion Static Methods #region Methods /// <summary> /// Executed when the application is exited. /// </summary> /// <param name="sender">The sender.</param> /// <param name="e">Event args.</param> internal void Application_Exit(object sender, EventArgs e) { if (this.IsOpen) { this._isAppExit = true; try { this.Close(); } finally { this._isAppExit = false; } } } /// <summary> /// Changes the visual state of the ChildWindow. /// </summary> private void ChangeVisualState() { if (this._isClosing) { VisualStateManager.GoToState(this, VSMSTATE_StateClosed, true); } else { VisualStateManager.GoToState(this, VSMSTATE_StateOpen, true); } } /// <summary> /// Executed when ChildWindow size is changed. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Size changed event args.</param> private void ChildWindow_SizeChanged(object sender, SizeChangedEventArgs e) { if (this.Overlay != null) { if (e.NewSize.Height != this.Overlay.ActualHeight) { this._desiredContentHeight = e.NewSize.Height; } if (e.NewSize.Width != this.Overlay.ActualWidth) { this._desiredContentWidth = e.NewSize.Width; } } if (this.IsOpen) { this.UpdateOverlaySize(); } } /// <summary> /// Closes a <see cref="T:System.Windows.Controls.ChildWindow" />. /// </summary> public void Close() { // AutomationPeer returns "Closing" when Close() is called // but the window is not closed completely: this.InteractionState = WindowInteractionState.Closing; CancelEventArgs e = new CancelEventArgs(); this.OnClosing(e); // On ApplicationExit, close() cannot be cancelled if (!e.Cancel || this._isAppExit) { if (RootVisual != null && this.IsOpen) { --OpenChildWindowCount; if (OpenChildWindowCount == 0) { // Restore the value saved when the first window was opened RootVisual.IsEnabled = RootVisual_PrevEnabledState; } } // Close Popup if (this.IsOpen) { if (this._closed != null) { // Popup will be closed when the storyboard ends this._isClosing = true; try { this.ChangeVisualState(); } finally { this._isClosing = false; } } else { // If no closing storyboard is defined, close the Popup this.ChildWindowPopup.IsOpen = false; } if (!this._dialogresult.HasValue) { // If close action is not happening because of DialogResult property change action, // Dialogresult is always false: this._dialogresult = false; } this.OnClosed(EventArgs.Empty); this.UnSubscribeFromEvents(); this.UnsubscribeFromTemplatePartEvents(); if (Application.Current.RootVisual != null) { Application.Current.RootVisual.GotFocus -= new RoutedEventHandler(this.RootVisual_GotFocus); _attachedRootVisualListener = false; } } } else { // If the Close is cancelled, DialogResult should always be NULL: this._dialogresult = null; this.InteractionState = WindowInteractionState.Running; } } /// <summary> /// Executed when the CloseButton is clicked. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Routed event args.</param> internal void CloseButton_Click(object sender, RoutedEventArgs e) { this.Close(); } /// <summary> /// Executed when the Closing storyboard ends. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Event args.</param> private void Closing_Completed(object sender, EventArgs e) { if (this.ChildWindowPopup != null) { this.ChildWindowPopup.IsOpen = false; } // AutomationPeer returns "NotResponding" when the ChildWindow is closed: this.InteractionState = WindowInteractionState.NotResponding; if (this._closed != null) { this._closed.Completed -= new EventHandler(this.Closing_Completed); } } /// <summary> /// Executed when the a key is presses when the window is open. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Key event args.</param> private void ChildWindow_KeyDown(object sender, KeyEventArgs e) { ChildWindow ew = sender as ChildWindow; Debug.Assert(ew != null, "ChildWindow instance is null."); // Ctrl+Shift+F4 closes the ChildWindow if (e != null && !e.Handled && e.Key == Key.F4 && ((Keyboard.Modifiers & ModifierKeys.Control) == ModifierKeys.Control) && ((Keyboard.Modifiers & ModifierKeys.Shift) == ModifierKeys.Shift)) { ew.Close(); e.Handled = true; } } /// <summary> /// Executed when the window loses focus. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Routed event args.</param> private void ChildWindow_LostFocus(object sender, RoutedEventArgs e) { // If the ChildWindow loses focus but the popup is still open, // it means another popup is opened. To get the focus back when the // popup is closed, we handle GotFocus on the RootVisual // TODO: Something else could get focus and handle the GotFocus event right. // Try listening to routed events that were Handled (new SL 3 feature) // Blocked by Jolt bug #29419 if (this.IsOpen && Application.Current != null && Application.Current.RootVisual != null) { this.InteractionState = WindowInteractionState.BlockedByModalWindow; if (!_attachedRootVisualListener) { Application.Current.RootVisual.GotFocus += new RoutedEventHandler(this.RootVisual_GotFocus); _attachedRootVisualListener = true; } } } /// <summary> /// Executed when mouse left button is down on the chrome. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Mouse button event args.</param> private void Chrome_MouseLeftButtonDown(object sender, MouseButtonEventArgs e) { if (this._chrome != null) { e.Handled = true; if (this.CloseButton != null && !this.CloseButton.IsTabStop) { this.CloseButton.IsTabStop = true; try { this.Focus(); } finally { this.CloseButton.IsTabStop = false; } } else { this.Focus(); } this._chrome.CaptureMouse(); this._isMouseCaptured = true; this._clickPoint = e.GetPosition(sender as UIElement); } } /// <summary> /// Executed when mouse left button is up on the chrome. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Mouse button event args.</param> private void Chrome_MouseLeftButtonUp(object sender, MouseButtonEventArgs e) { if (this._chrome != null) { e.Handled = true; this._chrome.ReleaseMouseCapture(); this._isMouseCaptured = false; } } /// <summary> /// Executed when mouse moves on the chrome. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Mouse event args.</param> private void Chrome_MouseMove(object sender, MouseEventArgs e) { if (this._isMouseCaptured && this.ContentRoot != null && Application.Current != null && Application.Current.RootVisual != null) { Point position = e.GetPosition(Application.Current.RootVisual); GeneralTransform gt = this.ContentRoot.TransformToVisual(Application.Current.RootVisual); if (gt != null) { Point p = gt.Transform(this._clickPoint); this._windowPosition = gt.Transform(new Point(0, 0)); if (position.X < 0) { double Y = FindPositionY(p, position, 0); position = new Point(0, Y); } if (position.X > this.Width) { double Y = FindPositionY(p, position, this.Width); position = new Point(this.Width, Y); } if (position.Y < 0) { double X = FindPositionX(p, position, 0); position = new Point(X, 0); } if (position.Y > this.Height) { double X = FindPositionX(p, position, this.Height); position = new Point(X, this.Height); } double x = position.X - p.X; double y = position.Y - p.Y; // Take potential RightToLeft layout into account FrameworkElement fe = Application.Current.RootVisual as FrameworkElement; if (fe != null && fe.FlowDirection == FlowDirection.RightToLeft) { x = -x; } UpdateContentRootTransform(x, y); } } } /// <summary> /// Executed when the ContentPresenter size changes. /// </summary> /// <param name="sender">Content Presenter object.</param> /// <param name="e">SizeChanged event args.</param> private void ContentPresenter_SizeChanged(object sender, SizeChangedEventArgs e) { if (this.ContentRoot != null && Application.Current != null && Application.Current.RootVisual != null && _isOpen) { GeneralTransform gt = this.ContentRoot.TransformToVisual(Application.Current.RootVisual); if (gt != null) { Point p = gt.Transform(new Point(0, 0)); double x = this._windowPosition.X - p.X; double y = this._windowPosition.Y - p.Y; UpdateContentRootTransform(x, y); } } RectangleGeometry rg = new RectangleGeometry(); rg.Rect = new Rect(0, 0, this._contentPresenter.ActualWidth, this._contentPresenter.ActualHeight); this._contentPresenter.Clip = rg; this.UpdatePosition(); } /// <summary> /// Finds the X coordinate of a point that is defined by a line. /// </summary> /// <param name="p1">Starting point of the line.</param> /// <param name="p2">Ending point of the line.</param> /// <param name="y">Y coordinate of the point.</param> /// <returns>X coordinate of the point.</returns> private static double FindPositionX(Point p1, Point p2, double y) { if (y == p1.Y || p1.X == p2.X) { return p2.X; } Debug.Assert(p1.Y != p2.Y, "Unexpected equal Y coordinates"); return (((y - p1.Y) * (p1.X - p2.X)) / (p1.Y - p2.Y)) + p1.X; } /// <summary> /// Finds the Y coordinate of a point that is defined by a line. /// </summary> /// <param name="p1">Starting point of the line.</param> /// <param name="p2">Ending point of the line.</param> /// <param name="x">X coordinate of the point.</param> /// <returns>Y coordinate of the point.</returns> private static double FindPositionY(Point p1, Point p2, double x) { if (p1.Y == p2.Y || x == p1.X) { return p2.Y; } Debug.Assert(p1.X != p2.X, "Unexpected equal X coordinates"); return (((p1.Y - p2.Y) * (x - p1.X)) / (p1.X - p2.X)) + p1.Y; } /// <summary> /// Builds the visual tree for the /// <see cref="T:System.Windows.Controls.ChildWindow" /> control when a /// new template is applied. /// </summary> [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Maintainability", "CA1502:AvoidExcessiveComplexity", Justification = "No need to split the code into two parts.")] public override void OnApplyTemplate() { this.UnsubscribeFromTemplatePartEvents(); base.OnApplyTemplate(); this.CloseButton = GetTemplateChild(PART_CloseButton) as ButtonBase; if (this.CloseButton != null) { if (this.HasCloseButton) { this.CloseButton.Visibility = Visibility.Visible; } else { this.CloseButton.Visibility = Visibility.Collapsed; } } if (this._closed != null) { this._closed.Completed -= new EventHandler(this.Closing_Completed); } if (this._opened != null) { this._opened.Completed -= new EventHandler(this.Opening_Completed); } this._root = GetTemplateChild(PART_Root) as FrameworkElement; if (this._root != null) { IList groups = VisualStateManager.GetVisualStateGroups(this._root); if (groups != null) { IList states = null; foreach (VisualStateGroup vsg in groups) { if (vsg.Name == ChildWindow.VSMGROUP_Window) { states = vsg.States; break; } } if (states != null) { foreach (VisualState state in states) { if (state.Name == ChildWindow.VSMSTATE_StateClosed) { this._closed = state.Storyboard; } if (state.Name == ChildWindow.VSMSTATE_StateOpen) { this._opened = state.Storyboard; } } } } } this.ContentRoot = GetTemplateChild(PART_ContentRoot) as FrameworkElement; this._chrome = GetTemplateChild(PART_Chrome) as FrameworkElement; this.Overlay = GetTemplateChild(PART_Overlay) as Panel; this._contentPresenter = GetTemplateChild(PART_ContentPresenter) as FrameworkElement; this.SubscribeToTemplatePartEvents(); this.SubscribeToStoryBoardEvents(); this._desiredMargin = this.Margin; this.Margin = new Thickness(0); // Update overlay size if (this.IsOpen) { this._desiredContentHeight = this.Height; this._desiredContentWidth = this.Width; this.UpdateOverlaySize(); this.UpdateRenderTransform(); this.ChangeVisualState(); } } /// <summary> /// Raises the /// <see cref="E:System.Windows.Controls.ChildWindow.Closed" /> event. /// </summary> /// <param name="e">The event data.</param> protected virtual void OnClosed(EventArgs e) { EventHandler handler = this.Closed; if (null != handler) { handler(this, e); } this._isOpen = false; } /// <summary> /// Raises the /// <see cref="E:System.Windows.Controls.ChildWindow.Closing" /> event. /// </summary> /// <param name="e">The event data.</param> protected virtual void OnClosing(CancelEventArgs e) { EventHandler<CancelEventArgs> handler = this.Closing; if (null != handler) { handler(this, e); } } /// <summary> /// Returns a /// <see cref="T:System.Windows.Automation.Peers.ChildWindowAutomationPeer" /> /// for use by the Silverlight automation infrastructure. /// </summary> /// <returns> /// <see cref="T:System.Windows.Automation.Peers.ChildWindowAutomationPeer" /> /// for the <see cref="T:System.Windows.Controls.ChildWindow" /> object. /// </returns> protected override AutomationPeer OnCreateAutomationPeer() { return new ChildWindowAutomationPeer(this); } /// <summary> /// This method is called every time a /// <see cref="T:System.Windows.Controls.ChildWindow" /> is displayed. /// </summary> protected virtual void OnOpened() { this.UpdatePosition(); this._isOpen = true; if (this.Overlay != null) { this.Overlay.Opacity = this.OverlayOpacity; this.Overlay.Background = this.OverlayBrush; } if (!this.Focus()) { // If the Focus() fails it means there is no focusable element in the // ChildWindow. In this case we set IsTabStop to true to have the keyboard functionality this.IsTabStop = true; this.Focus(); } } /// <summary> /// Executed when the opening storyboard finishes. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Event args.</param> private void Opening_Completed(object sender, EventArgs e) { if (this._opened != null) { this._opened.Completed -= new EventHandler(this.Opening_Completed); } // AutomationPeer returns "ReadyForUserInteraction" when the ChildWindow // is open and all animations have been completed. this.InteractionState = WindowInteractionState.ReadyForUserInteraction; this.OnOpened(); } /// <summary> /// Executed when the page resizes. /// </summary> /// <param name="sender">Sender object.</param> /// <param name="e">Event args.</param> private void Page_Resized(object sender, EventArgs e) {