File: System\Windows\Window.cs
Web Access
Project: src\src\Microsoft.DotNet.Wpf\src\PresentationFramework\PresentationFramework.csproj (PresentationFramework)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using System.Collections;
using System.ComponentModel;
using System.Globalization;
using System.Runtime.InteropServices;
using System.Windows.Appearance;
using System.Windows.Automation.Peers;
using System.Windows.Controls;
using System.Windows.Input;
using System.Windows.Interop;
using System.Windows.Media;
using System.Windows.Shell;
using System.Windows.Threading;
using MS.Internal;
using MS.Internal.AppModel;
using MS.Internal.Interop;
using MS.Internal.KnownBoxes;
using MS.Win32;
using System.Diagnostics.CodeAnalysis;
 
using BuildInfo = MS.Internal.PresentationFramework.BuildInfo;
using SNM = Standard.NativeMethods;
using HRESULT = MS.Internal.Interop.HRESULT;
using Win32Error = MS.Internal.Interop.Win32Error;
 
namespace System.Windows
{
    [Localizability(LocalizationCategory.Ignore)]
    public class Window : ContentControl, IWindowService
    {
        //---------------------------------------------------
        //
        // Constructors
        //
        //---------------------------------------------------
        #region Constructors
 
        /// <summary>
        ///     Initializes the dependency ids of this class
        /// </summary>
        static Window()
        {
            HeightProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(new PropertyChangedCallback(_OnHeightChanged)));
            MinHeightProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(new PropertyChangedCallback(_OnMinHeightChanged)));
            MaxHeightProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(new PropertyChangedCallback(_OnMaxHeightChanged)));
            WidthProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(new PropertyChangedCallback(_OnWidthChanged)));
            MinWidthProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(new PropertyChangedCallback(_OnMinWidthChanged)));
            MaxWidthProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(new PropertyChangedCallback(_OnMaxWidthChanged)));
 
            // override VisibilityProperty Metadata. For Window, Visibility.Visible means the Window is visible.
            // Visibility.Hidden and Visibility.Collapsed mean the Window is not visible.
            // Visibility.Hidden and Visibility.Collapsed are treated the same.
            // We default to Visibility.Collapsed since RenderSize returns (0,0) only for
            // collapsed elements and not for hidden. We want to return (0,0) when window is
            // never shown.
            VisibilityProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(Visibility.Collapsed, new PropertyChangedCallback(_OnVisibilityChanged), new CoerceValueCallback(CoerceVisibility)));
 
            IsTabStopProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(BooleanBoxes.FalseBox));
            KeyboardNavigation.DirectionalNavigationProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(KeyboardNavigationMode.Cycle));
            KeyboardNavigation.TabNavigationProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(KeyboardNavigationMode.Cycle));
            KeyboardNavigation.ControlTabNavigationProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(KeyboardNavigationMode.Cycle));
            FocusManager.IsFocusScopeProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(BooleanBoxes.TrueBox));
 
            DefaultStyleKeyProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(typeof(Window)));
            _dType = DependencyObjectType.FromSystemTypeInternal(typeof(Window));
 
            FlowDirectionProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(new PropertyChangedCallback(_OnFlowDirectionChanged)));
            // ideally this would just use a ValidateValueCallback
            // We don't support setting RenderTransform and ClipToBounds on Window. Exception will be thrown in Coerce callbacks
            RenderTransformProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(Transform.Identity, new PropertyChangedCallback(_OnRenderTransformChanged), new CoerceValueCallback(CoerceRenderTransform)));
            ClipToBoundsProperty.OverrideMetadata(typeof(Window), new FrameworkPropertyMetadata(BooleanBoxes.FalseBox, new PropertyChangedCallback(_OnClipToBoundsChanged), new CoerceValueCallback(CoerceClipToBounds)));
 
            // Note that this event only gets raised in Windows 7 and later.
            WM_TASKBARBUTTONCREATED = UnsafeNativeMethods.RegisterWindowMessage("TaskbarButtonCreated");
 
            WM_APPLYTASKBARITEMINFO = UnsafeNativeMethods.RegisterWindowMessage("WPF_ApplyTaskbarItemInfo");
 
            EventManager.RegisterClassHandler(typeof(Window),
                UIElement.ManipulationCompletedEvent,
                new EventHandler<ManipulationCompletedEventArgs>(OnStaticManipulationCompleted),
                /*handledEventsToo*/ true);
            EventManager.RegisterClassHandler(typeof(Window),
                UIElement.ManipulationInertiaStartingEvent,
                new EventHandler<ManipulationInertiaStartingEventArgs>(OnStaticManipulationInertiaStarting),
                /*handledEventsToo*/ true);
 
            Window.DpiChangedEvent = EventManager.RegisterRoutedEvent("DpiChanged", RoutingStrategy.Bubble,
                typeof (System.Windows.DpiChangedEventHandler), typeof (Window));
        }
 
        /// <summary>
        ///     Constructs a window object
        /// </summary>
        /// <remarks>
        ///     Automatic determination of current Dispatcher. Use alternative constructor
        ///     that accepts a Dispatcher for best performance.
        ///
        ///     Initializes the Width/Height, Top/Left properties to use windows
        ///     default. Updates Application object properties if inside app.
        ///
        ///     Also, window style is set to WS_CHILD inside CreateSourceWindow
        ///     for browser hosted case
        ///
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        /// </remarks>
        public Window()
        {
            _inTrustedSubWindow = false;
            Initialize();
        }
 
        /// <summary>
        ///     Constructs a window object
        /// </summary>
        /// <remarks>
        ///     Automatic determination of current Dispatcher. Use alternative constructor
        ///     that accepts a Dispatcher for best performance.
        ///
        ///     Initializes the Width/Height, Top/Left properties to use windows
        ///     default. Updates Application object properties if inside app.
        ///
        ///     Also, window style is set to WS_CHILD inside CreateSourceWindow
        ///     for browser hosted case
        ///
        ///     This method currently requires full trust to run.
        /// </remarks>
        internal Window(bool inRbw):base()
        {
            if (inRbw)
            {
                _inTrustedSubWindow = true;
            }
            else
            {
                _inTrustedSubWindow = false;
            }
            Initialize();
}
        #endregion Constructors
 
        //---------------------------------------------------
        //
        // Public Methods
        //
        //---------------------------------------------------
        #region Public Methods
 
        /// <summary>
        ///     Show the window
        /// </summary>
        /// <remarks>
        ///     Calling Show on window is the same as setting the
        ///     Visibility property to Visibility.Visible.
        /// </remarks>
        public void Show()
        {
            VerifyContextAndObjectState();
            VerifyCanShow();
            VerifyNotClosing();
            VerifyConsistencyWithAllowsTransparency();
 
            // Update the property value only.  Do not do anything further in
            // _OnVisibilityInvalidate since we will synchronously call ShowHelper
            // from here.
            UpdateVisibilityProperty(Visibility.Visible);
 
            ShowHelper(BooleanBoxes.TrueBox);
        }
 
        /// <summary>
        ///     Hide the window
        /// </summary>
        /// <remarks>
        ///     Calling Hide on window is the same as setting the
        ///     Visibility property to Visibility.Hidden
        ///     </remarks>
        public void Hide()
        {
            VerifyContextAndObjectState();
 
            if (_disposed == true)
            {
                return;
            }
 
            // set Visibility to Hidden even if _isVisible is false since
            // _isVisible can be false b/c of Visibility = Collapsed and Hide()
            // should change Visibility to Hidden.
            //
            // Update the property value only.  Do not do anything further in
            // _OnVisibilityInvalidate since we will synchronously call ShowHelper
            // from here.
            UpdateVisibilityProperty(Visibility.Hidden);
 
            ShowHelper(BooleanBoxes.FalseBox);
        }
 
        /// <summary>
        ///     Closes the Window
        /// </summary>
        /// <remarks>
        ///     Window fires the Closing event before it closes. If the
        ///     user cancels the closing event, the window is not closed.
        ///     Otherwise, the window is closed and the Closed event is
        ///     fired.
        ///
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        /// </remarks>
        public void Close()
        {
            // this call ends up throwing an exception if Close
            // is not allowed
            VerifyApiSupported();
            VerifyContextAndObjectState();
            InternalClose(false, false);
        }
 
        /// <summary>
        ///     Kick off the Window's MoveWindow loop
        /// </summary>
        /// <remarks>
        ///     To enable custom chrome on Windows. First check if this is the Left MouseButton.
        ///     Will throw exception if it's not, otherwise, will kick off the Windows's MoveWindow loop.
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        /// </remarks>
        public void DragMove()
        {
            // this call ends up throwing an exception if dragmove
            // is not allowed
            VerifyApiSupported();
            VerifyContextAndObjectState();
            VerifyHwndCreateShowState();
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return;
            }
 
            // Mouse.LeftButton actually reflects the primary button user is using.
            // So we don't need to check whether the button has been swapped here.
            if (Mouse.LeftButton == MouseButtonState.Pressed)
            {
                if (WindowState == WindowState.Normal)
                {
                    // SendMessage's return value is dependent on the message send.  WM_SYSCOMMAND
                    // and WM_LBUTTONUP return value just signify whether the WndProc handled the
                    // message or not, so they are not interesting
                    UnsafeNativeMethods.SendMessage( Handle, WindowMessage.WM_SYSCOMMAND, (IntPtr)NativeMethods.SC_MOUSEMOVE, IntPtr.Zero);
                    UnsafeNativeMethods.SendMessage( Handle, WindowMessage.WM_LBUTTONUP, IntPtr.Zero, IntPtr.Zero);
                }
            }
            else
            {
                throw new InvalidOperationException(SR.DragMoveFail);
            }
}
 
        /// <summary>
        ///     Shows the window as a modal window
        /// </summary>
        /// <returns>bool?</returns>
        /// <remarks>
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        /// </remarks>
        public Nullable<bool> ShowDialog()
        {
            // this call ends up throwing an exception if ShowDialog
            // is not allowed
            VerifyApiSupported();
            VerifyContextAndObjectState();
            VerifyCanShow();
            VerifyNotClosing();
            VerifyConsistencyWithAllowsTransparency();
 
            if ( _isVisible == true )
            {
                throw new InvalidOperationException(SR.ShowDialogOnVisible);
            }
            else if ( _showingAsDialog == true )
            {
                throw new InvalidOperationException(SR.ShowDialogOnModal);
            }
 
            _dialogOwnerHandle = _ownerHandle;
 
            // verify owner handle is window
            if (UnsafeNativeMethods.IsWindow( new HandleRef( null, _dialogOwnerHandle ) ) != true)
            {
                _dialogOwnerHandle = IntPtr.Zero;
            }
 
 
            // remember the current active window;
            // this is used when dialog creation fails or dialog closes, we set the active window back to this one.
            _dialogPreviousActiveHandle = UnsafeNativeMethods.GetActiveWindow();
 
            // if owner window is not specified, we get the current active window on this thread's
            // message queue as the owner.
            if (_dialogOwnerHandle == IntPtr.Zero)
            {
                _dialogOwnerHandle = _dialogPreviousActiveHandle;
            }
 
            // If hwndOwner == HWNDESKTOP, change it to NULL.  This way the desktop
            // (and all its children) won't be disabled if the dialog is modal.
            if ((_dialogOwnerHandle != IntPtr.Zero) &&
                (_dialogOwnerHandle == UnsafeNativeMethods.GetDesktopWindow()))
            {
                _dialogOwnerHandle = IntPtr.Zero;
            }
 
            // if dialog owner is not null, get the top level window (case where dialog owner is a
            // child window), and save it's state regarding enabled and active window
            if (_dialogOwnerHandle != IntPtr.Zero)
            {
                // get the top level window from the dialog owner handle
                int style = 0;
 
                while (_dialogOwnerHandle != IntPtr.Zero)
                {
                    style = UnsafeNativeMethods.GetWindowLong(new HandleRef(this, _dialogOwnerHandle), NativeMethods.GWL_STYLE);
                    if ((style & NativeMethods.WS_CHILD) == NativeMethods.WS_CHILD)
                    {
                        _dialogOwnerHandle = UnsafeNativeMethods.GetParent(new HandleRef(null, _dialogOwnerHandle));
                    }
                    else
                    {
                        break;
                    }
                }
            }
 
            Debug.Assert(_threadWindowHandles == null, "_threadWindowHandles must be null before enumerating the thread windows");
 
            // NOTE:
            // _threadWindowHandles is created here. This reference is nulled out in EnableThreadWindows
            // when it is called with a true parameter. Please do not null it out anywhere else.
            // EnableThreadWindow(true) is called when dialog is going away. Once dialog is closed and
            // thread windows have been enabled, then there no need to keep the list around.
            // Please see BUG 929740 before making any changes to how _threadWindowHandles works.
            _threadWindowHandles = new List<IntPtr>();
            //Get visible and enabled windows in the thread
            // If the callback function returns true for all windows in the thread, the return value is true.
            // If the callback function returns false on any enumerated window, or if there are no windows
            // found in the thread, the return value is false.
            // No need for use to actually check the return value.
            UnsafeNativeMethods.EnumThreadWindows(SafeNativeMethods.GetCurrentThreadId(),
                                                  new NativeMethods.EnumThreadWindowsCallback(ThreadWindowsCallback),
                                                  NativeMethods.NullHandleRef);
 
            // Disable those windows
            EnableThreadWindows(false);
 
            IntPtr hWndCapture = SafeNativeMethods.GetCapture();
            if (hWndCapture != IntPtr.Zero)
            {
                //
                // NOTE:
                // EnableWindow(false) (called from EnableThreadWindows(false)
                // sends WM_CANCELMODE to the window, so we don't need
                // to send it again.  However, if we change our impl
                // of dialog such that we don't disable all windows on the
                // thread, then we would need this call. Keeping this code here
                // until we finish the Dialog task # 18498
 
                // UnsafeNativeMethods.SendMessage(hWndCapture,
                //                                WindowMessage.WM_CANCELMODE,
                //                                IntPtr.Zero,
                //                                IntPtr.Zero);
 
                // hWndCapture = UnsafeNativeMethods.GetCapture();
                // if (hWndCapture != IntPtr.Zero)
                // {
                    // PS # 862892
                    // WCP: Investigate whether ReleaseCapture is needed in ShowDialog
                    SafeNativeMethods.ReleaseCapture();
                // }
            }
 
            // Ensure Dialog RoutedCommand is registered with CommandManager
            EnsureDialogCommand();
 
            try
            {
                _showingAsDialog = true;
                Show();
            }
            catch
            {
                // NOTE:
                // See BUG 929740.
                // _threadWindowHandles is created before calling ShowDialog and is deleted in
                // EnableThreadWindows (when it's called with true).
                //
                // Window dlg = new Window();
                // Button b = new button();
                // b.OnClick += new ClickHandler(OnClick);
                // dlg.ShowDialog();
                //
                //
                // void OnClick(...)
                // {
                //      dlg.Close();
                //      throw new Exception();
                // }
                //
                //
                // If above code is written, then we get inside this exception handler only after the dialog
                // is closed.  In that case all the windows that we disabled before showing the dialog have already
                // been enabled and _threadWindowHandles set to null in EnableThreadWindows.  Thus, we don't
                // need to do it again.
                //
                // In any other exception cases, we get in this handler before Dialog is closed and thus we do
                // need to enable all the disable windows.
                if (_threadWindowHandles != null)
                {
                    // Some exception case. Re-enable the windows that were disabled
                    EnableThreadWindows(true);
                }
 
                // Activate the previously active window.
                // This code/logic came from User.
                if ( (_dialogPreviousActiveHandle != IntPtr.Zero) &&
                    (UnsafeNativeMethods.IsWindow(new HandleRef(null, _dialogPreviousActiveHandle)) == true))
                {
                    // SetFocus fails if the input hwnd is not a Window or if the Window is not on the
                    // calling thread.
                    //
                    // Furthermore, this code path is executed when an exception occurs when we try to
                    // show the window.  Here we are doing the minimum possible to restore state of
                    // the avalon window object.  Hence, if for some reason, we are not able to
                    // SetFocus to the window that previously had focus, we don't care as that failure
                    // is not important enought to warrant throwing an exception.
                    UnsafeNativeMethods.TrySetFocus(new HandleRef(null, _dialogPreviousActiveHandle), ref _dialogPreviousActiveHandle);
                }
 
                // clears _showingAsDialog and accelerators related fields
                ClearShowKeyboardCueState();
                _showingAsDialog = false;
 
                // using catch and throw instead of catch(Exception e) throw e;  since the former
                // gives the complete call stack upto the offending method where the exception is thrown
                throw;
            }
            finally
            {
                // If the owner window belongs to another thread, the reactivation
                // of the owner may have failed within DestroyWindow().  Therefore,
                // if the current thread is in the foreground and the owner is not
                // in the foreground we can safely set the foreground back
                // to the owner.
#if FIGURE_OUT

                // WCP Dialog: Figure out what to do when reactivating the owner window
                // which is in another thread.
                if (_dialogOwnerHandle != IntPtr.Zero)
                {
                    if (IsCurrentThreadForeground() &&
                        !IsInForegroundQueue(hwndOwner))
                    {
                        NtUserSetForegroundWindow(hwndOwner);
                    }
                }
#endif //FIGURE_OUT
                _showingAsDialog = false;
            }
            return _dialogResult;
        }
 
 
        /// <summary>
        ///     This method tries to activate the Window.
        /// </summary>
        /// <remarks>
        ///     This method calls SetForegroundWindow on the hWnd, thus the rules for SetForegroundWindow
        ///     apply to this method.
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        /// </remarks>
        /// <returns>bool -- indicating whether the window was activated or not</returns>
        public bool Activate()
        {
            // this call ends up throwing an exception if Activate
            // is not allowed
            VerifyApiSupported();
            VerifyContextAndObjectState();
            VerifyHwndCreateShowState();
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return false;
            }
 
            return UnsafeNativeMethods.SetForegroundWindow(new HandleRef(null, Handle));
        }
        #region LogicalTree
        /// <summary>
        ///     Returns enumerator to logical children
        /// </summary>
        protected internal override IEnumerator LogicalChildren
        {
            get
            {
                // Don't use UIElementCollection because we don't have a reference to content's visual parent;
                // window has style and user can change it.
                return new SingleChildEnumerator(this.Content);
            }
        }
 
        #endregion LogicalTree
 
        #region static public method
 
        /// <summary>
        /// Gets Window in which the given DependecyObject is hosted in.
        /// </summary>
        /// <param name="dependencyObject">Returns the Window the given dependencyObject is hosted in.</param>
        /// <returns>Window</returns>
        public static Window GetWindow(DependencyObject dependencyObject)
        {
            ArgumentNullException.ThrowIfNull(dependencyObject);
 
            // Window.IWindowServiceProperty is an internal inheritable dependency property
            // Normally this value is set to the root Window element, all the element
            // inside the window view will get this value through property inheritance mechanism.
 
            return dependencyObject.GetValue(Window.IWindowServiceProperty) as Window;
        }
 
        #endregion static public method
 
        #endregion Public Methods
        //---------------------------------------------------
        //
        // Public Properties
        //
        //---------------------------------------------------
        #region Public Properties
 
        /// <summary>
        /// Gets or sets the Fluent theme mode of the window.
        /// </summary>
        /// <remarks>
        /// Setting this property controls if Fluent theme is loaded in Light, Dark or System mode.
        /// It also controls the application of backdrop and darkmode on window.
        /// The four values for the ThemeMode enum are :
        ///     <see cref="ThemeMode.None"/> - No Fluent theme is loaded. However, if <see cref="Application.ThemeMode"/> is not None,
        ///         then the window will appear as defined in <see cref="Application.ThemeMode"/>.
        ///     <see cref="ThemeMode.System"/> - Fluent theme is loaded based on the system theme.
        ///     <see cref="ThemeMode.Light"/> - Fluent theme is loaded in Light mode.
        ///     <see cref="ThemeMode.Dark"/> - Fluent theme is loaded in Dark mode.
        ///
        /// These values are predefined in <see cref="ThemeMode"/> struct.
        ///
        /// The default value is <see cref="ThemeMode.None"/>.
        ///
        /// <see cref="ThemeMode"/> and <see cref="Resources"/> are designed to be in sync with each other.
        ///     Syncing is done in order to avoid UI inconsistencies, for example, if the application is in dark mode
        ///     but the windows are in light mode or vice versa.
        ///
        ///     Setting this property loads the Fluent theme dictionaries in the window resources.
        ///     So, if you set this property, it is preferrable to not include Fluent theme dictionaries
        ///     in the window resources manually. If you do, the Fluent theme dictionaries added in the window
        ///     resources will take precedence over the ones added by setting this property.
        ///
        ///     This property is experimental and may be removed in future versions.
        /// </remarks>
        [Experimental("WPF0001")]
        [TypeConverter(typeof(ThemeModeConverter))]
        public ThemeMode ThemeMode
        {
            get
            {
                VerifyContextAndObjectState();
                return _themeMode;
            }
            set
            {
                VerifyContextAndObjectState();
 
                if(!ThemeManager.IsValidThemeMode(value))
                {
                    throw new ArgumentException(string.Format("ThemeMode value {0} is invalid. Use None, System, Light or Dark", value));
                }
 
                ThemeMode oldTheme = _themeMode;
                _themeMode = value;
 
                if(!AreResourcesInitialized)
                {
                    ThemeManager.OnWindowThemeChanged(this, oldTheme, value);
                    AreResourcesInitialized = false;
 
                    _reloadFluentDictionary = true;
                }
 
                if(IsSourceWindowNull)
                {
                    _deferThemeLoading = true;
                }
                else
                {
                    ThemeManager.OnWindowThemeChanged(this, oldTheme, value);
                }
            }
        }
 
        /// <summary>
        /// DependencyProperty for TaskbarItemInfo
        /// </summary>
        public static readonly DependencyProperty TaskbarItemInfoProperty = DependencyProperty.Register(
            "TaskbarItemInfo",
            typeof(TaskbarItemInfo),
            typeof(Window),
            new PropertyMetadata(
                null,
                (d, e) => ((Window)d).OnTaskbarItemInfoChanged(e),
                VerifyAccessCoercion));
 
        /// <summary>
        /// RoutedEvent for when DPI of the screen the Window is on, changes.
        /// </summary>
        public static readonly RoutedEvent DpiChangedEvent;
 
        /// <summary>
        /// Get or set the TaskbarItemInfo associated with this Window.
        /// </summary>
        public TaskbarItemInfo TaskbarItemInfo
        {
            get
            {
                VerifyContextAndObjectState();
                return (TaskbarItemInfo)GetValue(TaskbarItemInfoProperty);
            }
            set
            {
                VerifyContextAndObjectState();
                SetValue(TaskbarItemInfoProperty, value);
            }
        }
 
        private void OnTaskbarItemInfoChanged(DependencyPropertyChangedEventArgs e)
        {
            var oldBar = (TaskbarItemInfo)e.OldValue;
            var newBar = (TaskbarItemInfo)e.NewValue;
 
            // We don't propagate changes to this on anything earlier than Windows 7.
            if (!Utilities.IsOSWindows7OrNewer)
            {
                return;
            }
 
            if (!e.IsASubPropertyChange)
            {
                if (oldBar != null)
                {
                    oldBar.PropertyChanged -= OnTaskbarItemInfoSubPropertyChanged;
                }
                if (newBar != null)
                {
                    newBar.PropertyChanged += OnTaskbarItemInfoSubPropertyChanged;
                }
                ApplyTaskbarItemInfo();
            }
        }
 
        private void HandleTaskbarListError(HRESULT hr)
        {
            if (hr.Failed)
            {
                // Even if some of the taskbar methods get this error it doesn't mean that all of them will.
                // They aren't all implemented with SendMessageTimeout, and unfortunately the ITaskbarList3 API inconsistently
                // exposes that implementation detail.
                if (hr == (HRESULT)Win32Error.ERROR_TIMEOUT)
                {
                    // Explorer appears to be busy.  Post back to the Window to try again.
                    if (TraceShell.IsEnabled)
                    {
                        TraceShell.Trace(TraceEventType.Error, TraceShell.ExplorerTaskbarTimeout);
                        TraceShell.Trace(TraceEventType.Warning, TraceShell.ExplorerTaskbarRetrying);
                    }
 
                    // Explorer being non-responsive should be a transient issue.  Post back to apply the full TaskbarItemInfo.
                    _taskbarRetryTimer.Start();
                }
                else if (hr == (HRESULT)Win32Error.ERROR_INVALID_WINDOW_HANDLE || hr == HRESULT.E_NOTIMPL)
                {
                    // We'll get this when Explorer's not running.  This means there's no Shell to integrate with.
                    if (TraceShell.IsEnabled)
                    {
                        TraceShell.Trace(TraceEventType.Warning, TraceShell.ExplorerTaskbarNotRunning);
                    }
                    // If this is a transient condition then we'll get a WM_TASKBARBUTTONCREATED when Explorer comes
                    // back and the taskbar button gets created again, at which point we'll rehook.
                    // In the meantime, just stop trying since there's no reasonable expectation of recovery.
                    Utilities.SafeRelease(ref _taskbarList);
                }
                else
                {
                    // That covers the troublesome errors that we know how to handle.
                    // For anything else we'll ignore the error and count on a subsequent update to correct the state.
                    if (TraceShell.IsEnabled)
                    {
                        TraceShell.Trace(TraceEventType.Error, TraceShell.NativeTaskbarError(hr.ToString()));
                    }
                }
            }
        }
 
        private void OnTaskbarItemInfoSubPropertyChanged(object sender, DependencyPropertyChangedEventArgs e)
        {
            // Don't propagate changes from other TaskbarItemInfos.
            if (sender != this.TaskbarItemInfo)
            {
                // Since this and the TaskbarItemInfo should share affinity for the same thread
                // this really shouldn't happen...
                Debug.Assert(false);
                return;
            }
 
            // Defer any sub-property changes until the native ITaskbarList3 has been set up.
            if (_taskbarList == null)
            {
                return;
            }
 
            // If the taskbar has timed out in the last minute, don't try to do this again.
            if (_taskbarRetryTimer != null && _taskbarRetryTimer.IsEnabled)
            {
                return;
            }
 
            DependencyProperty dp = e.Property;
 
            HRESULT hr = HRESULT.S_OK;
 
            if (dp == TaskbarItemInfo.ProgressStateProperty)
            {
                hr = UpdateTaskbarProgressState();
            }
            else if (dp == TaskbarItemInfo.ProgressValueProperty)
            {
                hr = UpdateTaskbarProgressValue();
            }
            else if (dp == TaskbarItemInfo.OverlayProperty)
            {
                hr = UpdateTaskbarOverlay();
            }
            else if (dp == TaskbarItemInfo.DescriptionProperty)
            {
                hr = UpdateTaskbarDescription();
            }
            else if (dp == TaskbarItemInfo.ThumbnailClipMarginProperty)
            {
                hr = UpdateTaskbarThumbnailClipping();
            }
            else if (dp == TaskbarItemInfo.ThumbButtonInfosProperty)
            {
                hr = UpdateTaskbarThumbButtons();
            }
 
            HandleTaskbarListError(hr);
        }
 
        /// <summary>
        /// DependencyProperty for AllowsTransparency
        /// </summary>
        public static readonly DependencyProperty AllowsTransparencyProperty =
                DependencyProperty.Register(
                        "AllowsTransparency",
                        typeof(bool),
                        typeof(Window),
                        new FrameworkPropertyMetadata(
                                BooleanBoxes.FalseBox,
                                new PropertyChangedCallback(OnAllowsTransparencyChanged),
                                new CoerceValueCallback(CoerceAllowsTransparency)));
 
        /// <summary>
        /// Whether or not the Window uses per-pixel opacity
        /// </summary>
        public bool AllowsTransparency
        {
            get { return (bool)GetValue(AllowsTransparencyProperty); }
            set { SetValue(AllowsTransparencyProperty, BooleanBoxes.Box(value)); }
        }
 
        private static void OnAllowsTransparencyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
        }
 
        private static object CoerceAllowsTransparency(DependencyObject d, object value)
        {
            value = VerifyAccessCoercion(d, value);
 
            if (!((Window) d).IsSourceWindowNull)
            {
                throw new InvalidOperationException(SR.ChangeNotAllowedAfterShow);
            }
 
            return value;
        }
 
        /// <summary>
        ///     The DependencyProperty for TitleProperty.
        ///     Flags:              None
        ///     Default Value:      String.Empty
        /// </summary>
        public static readonly DependencyProperty TitleProperty =
                DependencyProperty.Register("Title", typeof(String), typeof(Window),
                        new FrameworkPropertyMetadata(String.Empty,
                                new PropertyChangedCallback(_OnTitleChanged)),
                        new ValidateValueCallback(_ValidateText));
        /// <summary>
        ///     The data that will be displayed as the title of the window.
        ///     Hosts are free to display the title in any manner that they
        ///     want.  For example, the browser may display the title set via
        ///     the Title property somewhere besides the caption bar
        /// </summary>
        [Localizability(LocalizationCategory.Title)]
        public string Title
        {
            get
            {
                VerifyContextAndObjectState();
 
                return (String)GetValue(TitleProperty);
            }
            set
            {
                VerifyContextAndObjectState();
 
                SetValue(TitleProperty, value);
            }
        }
 
        /// <summary>
        ///     The DependencyProperty for Icon
        ///     Flags:              None
        ///     Default Value:      None
        /// </summary>
        public static readonly DependencyProperty IconProperty =
                DependencyProperty.Register(
                        "Icon",
                        typeof(ImageSource),
                        typeof(Window),
                        new FrameworkPropertyMetadata(
                                new PropertyChangedCallback(_OnIconChanged),
                                new CoerceValueCallback(VerifyAccessCoercion)));
 
        /// <summary>
        ///     Sets the Icon of the Window
        /// </summary>
        /// <remarks>
        ///     Following is the precedence for displaying the icon:
        ///
        ///     1) Use ImageSource provided by the Icon property.  If Icon property is
        ///     null, see 2 below.
        ///     2) If Icon Property is not set, then use the Application icon
        ///     embedded in the exe.  Querying Icon property returns null.
        ///     3) If no icon is embedded in the exe, then we set IntPtr.Zero
        ///     as the icon and Win32 displays its default icon.  Querying Icon
        ///     property returns null.
        ///
        ///     If Icon property is set, Window does not dispose that object when it
        ///     is closed.
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        /// </remarks>
        public ImageSource Icon
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // Icon is not allowed
                VerifyApiSupported();
 
                return (ImageSource) GetValue(IconProperty);
            }
 
            set
            {
                // this call ends up throwing an exception if accessing
                // Icon is not allowed
                VerifyApiSupported();
                VerifyContextAndObjectState();
 
                SetValue(IconProperty, value);
            }
        }
 
        /// <summary>
        ///     The DependencyProperty for SizeToContentProperty.
        ///     Flags:              None
        ///     Default Value:      "SizeToContent.Manual"
        /// </summary>
        public static readonly DependencyProperty SizeToContentProperty =
                DependencyProperty.Register("SizeToContent",
                        typeof(SizeToContent),
                        typeof(Window),
                        new FrameworkPropertyMetadata(
                                SizeToContent.Manual,
                                new PropertyChangedCallback(_OnSizeToContentChanged)),
                        new ValidateValueCallback(_ValidateSizeToContentCallback));
 
        /// <summary>
        /// Auto size Window to its content's size
        /// </summary>
        /// <remarks>
        /// 1. SizeToContent can be applied to Width Height independently
        /// 2. After SizeToContent is set, setting Width/Height does not take affect if that
        ///    dimension is sizing to content.
        /// 3. SizeToContent is turned off (restored to SizeToContent.Manual) if user starts to
        ///    interact with window in terms of size
        /// </remarks>
        /// <value>
        /// Default value is SizeToContent.Manual
        /// </value>
        public SizeToContent SizeToContent
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // SizeToContent is not allowed
                VerifyApiSupported();
 
                return (SizeToContent) GetValue(SizeToContentProperty);
            }
            set
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // SizeToContent is not allowed
                VerifyApiSupported();
 
                SetValue(SizeToContentProperty, value);
            }
        }
 
        /// <summary>
        /// DependencyProperty for <see cref="Top" /> property.
        /// </summary>
 
        public static readonly DependencyProperty TopProperty =
                Canvas.TopProperty.AddOwner(typeof(Window),
                        new FrameworkPropertyMetadata(
                                Double.NaN,
                                new PropertyChangedCallback(_OnTopChanged),
                                new CoerceValueCallback(CoerceTop)));
 
        /// <summary>
        ///     Position for Top of the host window
        /// </summary>
        /// <remarks>
        ///     The following values are valid:
        ///     Positive Doubles: sets the top location to the specified value
        ///     NaN: indicates to use the system default value. This
        ///     is the default for Top property
        ///     PositiveInfinity, NegativeInfinity: These are invalid inputs.
        /// </remarks>
        /// <value></value>
        [TypeConverter($"System.Windows.LengthConverter, PresentationFramework, Version={BuildInfo.WCP_VERSION}, Culture=neutral, PublicKeyToken={BuildInfo.WCP_PUBLIC_KEY_TOKEN}, Custom=null")]
        public double Top
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing Top
                // is not allowed
                VerifyApiSupported();
                return (double)GetValue(TopProperty);
            }
            set
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing Top
                // is not allowed
                VerifyApiSupported();
 
                // we don't do an if check here to see if the new value is the same
                // as the current Top value b/c the current value maybe as a result
                // of user resizing which means the the local value of Top has not
                // been written to.  So, if window.Top is explicitly set now we want
                // to write to the local value.  We do make this if check in Top
                // property invalidation callback for optimization
                SetValue(TopProperty, value);
            }
        }
 
        /// <summary>
        /// DependencyProperty for <see cref="Left" /> property.
        /// </summary>
        public static readonly DependencyProperty LeftProperty =
                Canvas.LeftProperty.AddOwner(typeof(Window),
                        new FrameworkPropertyMetadata(
                                Double.NaN,
                                new PropertyChangedCallback(_OnLeftChanged),
                                new CoerceValueCallback(CoerceLeft)));
 
        /// <summary>
        ///     Position for Left edge of  coordinate of the host window
        /// </summary>
        /// <remarks>
        ///     The following values are valid:
        ///     Positive Doubles: sets the top location to the specified value
        ///     NaN: indicates to use the system default value. This
        ///     is the default for Top property
        ///     PositiveInfinity, NegativeInfinity: These are invalid inputs.
        /// </remarks>
        /// <value></value>
        [TypeConverter($"System.Windows.LengthConverter, PresentationFramework, Version={BuildInfo.WCP_VERSION}, Culture=neutral, PublicKeyToken={BuildInfo.WCP_PUBLIC_KEY_TOKEN}, Custom=null")]
        public double Left
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing left
                // is not allowed
                VerifyApiSupported();
                return (double)GetValue(LeftProperty);
            }
            set
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing left
                // is not allowed
                VerifyApiSupported();
 
                // we don't do an if check here to see if the new value is the same
                // as the current Left value b/c the current value maybe as a result
                // of user resizing which means the the local value of Left has not
                // been written to.  So, if window.Left is explicitly set now we want
                // to write to the local value.  We do make this if check in Left
                // property invalidation callback for optimization
                SetValue(LeftProperty, value);
            }
        }
 
        /// <summary>
        ///     This property returns the restoring rectangle of the window.  This information
        ///     can be used to track a users size and position preferences when the
        ///     Window is maximized or minimized.
        ///
        ///     If RestoreBounds is queried before the Window has been shown or after it has
        ///     been closed, it will return Rect.Empty.
        /// </summary>
        /// <remarks>
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        /// </remarks>
 
        public Rect RestoreBounds
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing RestoreBounds
                // is not allowed
                VerifyApiSupported();
 
                // either before calling show or after closing AND
                // Adding check for IsCompositionTargetInvalid
                if (IsSourceWindowNull || IsCompositionTargetInvalid)
                {
                    return Rect.Empty;
                }
 
                return GetNormalRectLogicalUnits(Handle);
            }
        }
 
        /// <summary>
        ///     This enum can have following values
        ///         Manual (default)
        ///         CenterScreen
        ///         CenterOwner
        ///
        ///     If the WindowStartupLocation is WindowStartupLocation.Manual then
        ///     Top and Left properites are used to position the window.
        ///     This property is used only before window creation. Once the window is
        ///     created hiding it and showing it will not take this property into account.
        /// </summary>
        /// <remarks>
        ///     WindowStartupLocation is used to position the window only it it is set to
        ///     WindowStartupLocation.CenterScreen or WindowStartupLocation.CenterOwner,
        ///     otherwise Top/Left is used.  Furthermore, if determining the location
        ///     of the window is not possible when WindowStartupLocation is set to
        ///     WindowStartupLocation.CenterScreen or WindowStartupLocation.Owner, then
        ///     Top/Left is used instead.
        /// </remarks>
        [DefaultValue(WindowStartupLocation.Manual)]
        public WindowStartupLocation WindowStartupLocation
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // WindowStartupLocation is not allowed
                VerifyApiSupported();
 
                return _windowStartupLocation;
            }
 
            set
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // WindowStartupLocation is not allowed
                VerifyApiSupported();
 
                //validate WindowStartupLocation enum
                if (!IsValidWindowStartupLocation(value))
                {
                    throw new InvalidEnumArgumentException("value", (int)value, typeof( WindowStartupLocation ));
                }
                _windowStartupLocation = value;
            }
        }
 
        /// <summary>
        ///     The DependencyProperty for ShowInTaskbarProperty.
        ///     Flags:              None
        ///     Default Value:      true
        /// </summary>
        public static readonly DependencyProperty ShowInTaskbarProperty =
                DependencyProperty.Register("ShowInTaskbar",
                        typeof(bool),
                        typeof(Window),
                        new FrameworkPropertyMetadata(BooleanBoxes.TrueBox,
                                new PropertyChangedCallback(_OnShowInTaskbarChanged),
                                new CoerceValueCallback(VerifyAccessCoercion)));
 
        ///<summary>
        ///     Determines if the window should show up in the system taskbar.
        ///     This also determines if the window appears in the Alt-Tab list.
        ///</summary>
        public bool ShowInTaskbar
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // ShowInTaskbar is not allowed
                VerifyApiSupported();
 
                return (bool) GetValue(ShowInTaskbarProperty);
            }
            set
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // ShowInTaskbar is not allowed
                VerifyApiSupported();
 
                SetValue(ShowInTaskbarProperty, BooleanBoxes.Box(value));
            }
        }
 
        /// <summary>
        ///     The key needed set a read-only property.
        /// </summary>
        private static readonly DependencyPropertyKey IsActivePropertyKey
            = DependencyProperty.RegisterReadOnly("IsActive", typeof(bool), typeof(Window),
                                          new FrameworkPropertyMetadata(BooleanBoxes.FalseBox));
 
        /// <summary>
        ///     The DependencyProperty for IsActive.
        ///     Flags:              None
        ///     Default Value:      True
        ///     Read-Only:          true
        /// </summary>
        public static readonly DependencyProperty IsActiveProperty
            = IsActivePropertyKey.DependencyProperty;
 
        /// <summary>
        /// IsActive property. It indicates whether the Window is active.
        /// The title bar will have the active theme. The active window will be
        /// the topmost of all top-level windows that don't explicitly set the TopMost property or style.
        /// If a window is active, focus is within the window.
        /// </summary>
        public bool IsActive
        {
            get
            {
                VerifyContextAndObjectState();
                return (bool)GetValue(IsActiveProperty);
            }
        }
 
        /// <summary>
        ///     This set the owner for the property of the current window.
        ///     If the window has owner and the owner is minimized then
        ///     owned window is also minimized. Owner window can never be
        ///     over owned window. Owned window is NOT modal. So user can
        ///     still interact with owner window. This property can not be
        ///     set of the top level window.
        /// </summary>
        ///<remarks>
        ///     Callers must have UIPermission(UIPermissionWindow.AllWindows) to call this API.
        ///</remarks>
        [DefaultValue(null)]
        public Window Owner
        {
            get
            {
                // this call ends up throwing an exception if accessing Owner
                // is not allowed
                VerifyApiSupported();
                VerifyContextAndObjectState();
                return _ownerWindow;
            }
            set
            {
                // this call ends up throwing an exception if accessing Owner
                // is not allowed
                VerifyApiSupported();
                VerifyContextAndObjectState();
                if (value == this)
                {
                    throw new ArgumentException(SR.CannotSetOwnerToItself);
                }
 
                if ( _showingAsDialog == true )
                {
                    throw new InvalidOperationException(SR.CantSetOwnerAfterDialogIsShown);
                }
 
                if (value != null && value.IsSourceWindowNull == true)
                {
                    // Try to be specific in the error message.
                    if (value._disposed)
                    {
                        throw new InvalidOperationException(SR.CantSetOwnerToClosedWindow);
                    }
                    throw new InvalidOperationException(SR.CantSetOwnerWhosHwndIsNotCreated);
                }
 
                if ( _ownerWindow == value )
                {
                    return;
                }
 
                if (!_disposed)
                {
                    // Check to see if value is already a child of this window.
                    // If yes, throw Exception
                    if (value != null)
                    {
                        WindowCollection ownedWindows = OwnedWindows;
                        for (int i = 0; i < ownedWindows.Count; i++)
                        {
                            if (ownedWindows[i] == value)
                            {
                                throw new ArgumentException(SR.Format(SR.CircularOwnerChild, value, this));
                            }
                        }
                    }
 
                    // Update OwnerWindows of the previous owner
                    if (_ownerWindow != null)
                    {
                        // using OwnedWindowsInternl b/c we want to modifying the
                        // underlying collection
                        _ownerWindow.OwnedWindowsInternal.Remove(this);
                    }
                }
 
                // Update parent handle. If value is null, then make parent
                // handle IntPtr.Zero
                _ownerWindow = value;
 
                // We should not do anything if the window is already closed and maybe throw exception.
                // In Dev10, it is unknown whether we can begin to throw exceptions, because it is a BC.
                // In Dev10, we still update _ownerWindow after window is closed just so that the Owner getter
                // returns the right value.
                if (_disposed)
                {
                    return;
                }
 
                SetOwnerHandle(_ownerWindow != null ? _ownerWindow.Handle: IntPtr.Zero);
 
                // Update OwnerWindows of the new owner
                if (_ownerWindow != null)
                {
                    // using OwnedWindowsInternl b/c we want to modifying the
                    // underlying collection
                    _ownerWindow.OwnedWindowsInternal.Add(this);
                }
            }
        }
 
        /// This code checks to see if the owner property is null
        /// True if the window is null , false other wise
        private bool IsOwnerNull
        {
            get
            {
                return (_ownerWindow == null);
            }
}
        /// <summary>
        ///     This is a collection of windows that are owned by current window.
        /// </summary>
        // This collection is a copy of the original one to avoid synchronizing issues.
        // DO-NOT USE THIS PROPERY IF YOU MEAN TO MODIFY THE UNDERLYING COLLECTION.  USE
        // OwnedWindowsInternal PROPERTY FOR MODIFYING THE UNDERLYING DATASTRUCTURE.
        public WindowCollection OwnedWindows
        {
            get
            {
                VerifyContextAndObjectState();
                return OwnedWindowsInternal.Clone();
            }
        }
 
        /// <summary>
        /// Gets Showing as dialog
        /// </summary>
        internal bool IsShowingAsDialog
        {
            get
            {
                return _showingAsDialog;
            }
        }
 
        /// <summary>
        /// Sets/gets DialogResult
        /// </summary>
        [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), TypeConverter(typeof(DialogResultConverter))]
        public Nullable<bool> DialogResult
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // DialogResult is not allowed
                VerifyApiSupported();
 
                return _dialogResult;
            }
            set
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // DialogResult is not allowed
                VerifyApiSupported();
 
                if (_showingAsDialog == true)
                {
                    // This value should be set only after the window is created and shown as dialog.
 
                    // When _showingAsDialog is set, _sourceWindow must be set too.
                    Debug.Assert( IsSourceWindowNull == false , "IsSourceWindowNull cannot be true when _showingAsDialog is true");
 
 
                    // According to the new design, setting DialogResult to its current value will not have any effect.
                    if (_dialogResult != value)
                    {
                        _dialogResult = value;
 
                        // if DialogResult is set from within a Closing event then
                        // the window is in the closing state.  Thus, if we call
                        // Close() again from here we go into an infinite loop.
                        //
                        // Note: Windows OS bug # 934500 Setting DialogResult
                        // on the Closing EventHandler of a Dialog causes StackOverFlowException
 
                        if(_isClosing == false)
                        {
                            Close();
                        }
                    }
                }
                else
                {
                    throw new InvalidOperationException(SR.DialogResultMustBeSetAfterShowDialog);
}
            }
        }
 
        /// <summary>
        ///     The DependencyProperty for WindowStyleProperty.
        ///     Flags:              None
        ///     Default Value:      WindowStyle.SingleBorderWindow
        /// </summary>
        public static readonly DependencyProperty WindowStyleProperty =
                DependencyProperty.Register("WindowStyle", typeof(WindowStyle), typeof(Window),
                        new FrameworkPropertyMetadata(
                                WindowStyle.SingleBorderWindow,
                                new PropertyChangedCallback(_OnWindowStyleChanged),
                                new CoerceValueCallback(CoerceWindowStyle)),
                        new ValidateValueCallback(_ValidateWindowStyleCallback));
 
        /// <summary>
        ///     Defines the visual style of the window (3DBorderWindow,
        ///     SingleBorderWindow, ToolWindow, none).
        /// </summary>
        /// <remarks>
        ///     Default will be SingleBorderWindow.
        /// </remarks>
        public WindowStyle WindowStyle
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // WindowStyle is not allowed
                VerifyApiSupported();
 
                return (WindowStyle) GetValue(WindowStyleProperty);
            }
            set
            {
                VerifyContextAndObjectState();
                // this call ends up throwing an exception if accessing
                // WindowStyle is not allowed
                VerifyApiSupported();
 
                SetValue(WindowStyleProperty, value);
            }
        }
 
        private static object CoerceWindowStyle(DependencyObject d, object value)
        {
            value = VerifyAccessCoercion(d, value);
 
            if (!((Window)d).IsSourceWindowNull)
            {
                // Since the new style hasn't actually been set yet, verify against the new value.
                ((Window)d).VerifyConsistencyWithAllowsTransparency((WindowStyle)value);
            }
 
            return value;
        }
 
        /// <summary>
        ///     The DependencyProperty for WindowStateProperty.
        ///     Flags:              None
        ///     Default Value:      WindowState.Normal
        /// </summary>
        public static readonly DependencyProperty WindowStateProperty =
                DependencyProperty.Register("WindowState", typeof(WindowState), typeof(Window),
                        new FrameworkPropertyMetadata(
                                WindowState.Normal,
                                FrameworkPropertyMetadataOptions.BindsTwoWayByDefault,
                                new PropertyChangedCallback(_OnWindowStateChanged),
                                new CoerceValueCallback(VerifyAccessCoercion)),
                        new ValidateValueCallback(_ValidateWindowStateCallback));
 
        /// <summary>
        ///     Current state of the window.  Valid options are Maximized, Minimized,
        ///     or Normal.  The host window may choose to ignore a request to change
        ///     the current window state.
        /// </summary>
        public WindowState WindowState
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // WindowState is not allowed
                VerifyApiSupported();
 
                return (WindowState) GetValue(WindowStateProperty);
            }
            set
            {
                VerifyContextAndObjectState();
                // this call ends up throwing an exception if accessing
                // WindowState is not allowed
                VerifyApiSupported();
 
                SetValue(WindowStateProperty, value);
            }
        }
 
        /// <summary>
        ///     The DependencyProperty for the ResizeMode property.
        ///     Flags:                  AffectsMeasure
        ///     Default Value:      false
        /// </summary>
        public static readonly DependencyProperty ResizeModeProperty =
                DependencyProperty.Register("ResizeMode", typeof(ResizeMode), typeof(Window),
                        new FrameworkPropertyMetadata(ResizeMode.CanResize,
                                FrameworkPropertyMetadataOptions.AffectsMeasure,
                                new PropertyChangedCallback(_OnResizeModeChanged),
                                new CoerceValueCallback(VerifyAccessCoercion)),
                        new ValidateValueCallback(_ValidateResizeModeCallback));
 
        /// <summary>
        ///     Current state of the window.  Valid options are Maximized, Minimized,
        ///     or Normal.  The host window may choose to ignore a request to change
        ///     the current window state.
        /// </summary>
        public ResizeMode ResizeMode
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // ResizeMode is not allowed
                VerifyApiSupported();
 
                return ((ResizeMode) GetValue(ResizeModeProperty));
            }
            set
            {
                VerifyContextAndObjectState();
                // this call ends up throwing an exception if accessing
                // ResizeMode is not allowed
                VerifyApiSupported();
 
                SetValue(ResizeModeProperty, value);
            }
        }
 
        /// <summary>
        ///     The DependencyProperty for TopmostProperty.
        ///     Flags:              None
        ///     Default Value:      false
        /// </summary>
        public static readonly DependencyProperty TopmostProperty =
                DependencyProperty.Register("Topmost",
                        typeof(bool),
                        typeof(Window),
                        new FrameworkPropertyMetadata(BooleanBoxes.FalseBox,
                                new PropertyChangedCallback(_OnTopmostChanged),
                                new CoerceValueCallback(VerifyAccessCoercion)));
 
        /// <summary>
        ///     Determines if this window is always on the top.
        /// </summary>
        public bool Topmost
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // Topmost is not allowed
                VerifyApiSupported();
 
                return (bool) GetValue(TopmostProperty);
            }
            set
            {
                VerifyContextAndObjectState();
                // this call ends up throwing an exception if accessing
                // Topmost is not allowed
                VerifyApiSupported();
 
                SetValue(TopmostProperty, BooleanBoxes.Box(value));
            }
        }
 
        public static readonly DependencyProperty ShowActivatedProperty =
                DependencyProperty.Register("ShowActivated",
                        typeof(bool),
                        typeof(Window),
                        new FrameworkPropertyMetadata(BooleanBoxes.TrueBox,
                                null,
                                new CoerceValueCallback(VerifyAccessCoercion)));
 
        /// <summary>
        ///     Determines if this window is activated when shown (default = true).
        /// </summary>
        /// <remarks>
        ///     Not supported for RBW.
        /// </remarks>
        public bool ShowActivated
        {
            get
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // ShowActivated is not allowed
                VerifyApiSupported();
 
                return (bool)GetValue(ShowActivatedProperty);
            }
            set
            {
                VerifyContextAndObjectState();
 
                // this call ends up throwing an exception if accessing
                // ShowActivated is not allowed
                VerifyApiSupported();
 
                SetValue(ShowActivatedProperty, BooleanBoxes.Box(value));
            }
        }
 
        #endregion Public Properties
 
        //---------------------------------------------------
        //
        // Public Events
        //
        //---------------------------------------------------
        #region Public Events
 
        /// <summary>
        ///     This event is raised after the window source is created before it is shown
        /// </summary>
        public event EventHandler SourceInitialized
        {
            add { Events.AddHandler(EVENT_SOURCEINITIALIZED, value); }
            remove { Events.RemoveHandler(EVENT_SOURCEINITIALIZED, value); }
        }
 
        /// <summary>
        ///     This event is raised after the DPI of the screen on which the Window is displayed, changes.
        /// </summary>
        public event DpiChangedEventHandler DpiChanged
        {
            add { AddHandler(Window.DpiChangedEvent, value); }
            remove { RemoveHandler(Window.DpiChangedEvent, value); }
        }
 
        /// <summary>
        ///     This event is raised when the window is activated
        /// </summary>
        public event EventHandler Activated
        {
            add { Events.AddHandler(EVENT_ACTIVATED, value); }
            remove { Events.RemoveHandler(EVENT_ACTIVATED, value); }
        }
 
        /// <summary>
        ///     This event is raised when the window is deactivated
        /// </summary>
        public event EventHandler Deactivated
        {
            add { Events.AddHandler(EVENT_DEACTIVATED, value); }
            remove { Events.RemoveHandler(EVENT_DEACTIVATED, value); }
        }
 
        /// <summary>
        ///     This event is raised when the window state is changed
        /// </summary>
        public event EventHandler StateChanged
        {
            add { Events.AddHandler(EVENT_STATECHANGED, value); }
            remove { Events.RemoveHandler(EVENT_STATECHANGED, value); }
        }
 
        /// <summary>
        ///     This event is raised when the window location is changed
        /// </summary>
        public event EventHandler LocationChanged
        {
            add { Events.AddHandler(EVENT_LOCATIONCHANGED, value); }
            remove { Events.RemoveHandler(EVENT_LOCATIONCHANGED, value); }
        }
 
        /// <summary>
        ///     This event is raised before the window is closed
        /// </summary>
        /// <remarks>
        ///     The user can set the CancelEventArg.Cancel property to true to prevent
        ///     the window from closing. However, if the Applicaiton is shutting down
        ///     the window closing cannot be cancelled
        /// </remarks>
        public event CancelEventHandler Closing
        {
            add { Events.AddHandler(EVENT_CLOSING, value); }
            remove { Events.RemoveHandler(EVENT_CLOSING, value); }
        }
 
        /// <summary>
        ///     This event is raised when the window is closed.
        /// </summary>
        public event EventHandler Closed
        {
            add { Events.AddHandler(EVENT_CLOSED, value); }
            remove { Events.RemoveHandler(EVENT_CLOSED, value); }
        }
 
        /// <summary>
        ///     This event is raised when the window and its content is rendered.
        /// </summary>
        public event EventHandler ContentRendered
        {
            add { Events.AddHandler(EVENT_CONTENTRENDERED, value); }
            remove { Events.RemoveHandler(EVENT_CONTENTRENDERED, value); }
        }
 
        #endregion Public Events
 
        //---------------------------------------------------
        //
        // Internal Events
        //
        //---------------------------------------------------
        #region Internal Events
 
        /// <summary>
        ///     This event is raised when the <see cref="System.Windows.Media.VisualCollection"/> of
        ///     this <see cref="Window"/> object is modified
        /// </summary>
        /// <remarks>
        ///     This corresponds to <see cref="OnVisualChildrenChanged(DependencyObject, DependencyObject)"/>.
        ///     The primary consumer of this event is <see cref="System.Windows.Shell.WindowChromeWorker"/>,
        ///     which only needs to be aware of the fact that Visual children have changed, but doesn't
        ///     need to know which children have changed. Therefore the <see cref="DependencyObject"/>
        ///     parameters of <see cref="OnVisualChildrenChanged(DependencyObject, DependencyObject)"/> are
        ///     not passed along to the <see cref="Shell.WindowChromeWorker"/> via this event.
        /// </remarks>
        internal event EventHandler<EventArgs> VisualChildrenChanged
        {
            add { Events.AddHandler(EVENT_VISUALCHILDRENCHANGED, value); }
            remove { Events.RemoveHandler(EVENT_VISUALCHILDRENCHANGED, value); }
        }
 
        #endregion
 
        //---------------------------------------------------
        //
        // Protected Methods
        //
        //---------------------------------------------------
        #region Protected Methods
 
        /// <summary>
        /// Creates AutomationPeer (<see cref="UIElement.OnCreateAutomationPeer"/>)
        /// </summary>
        protected override AutomationPeer OnCreateAutomationPeer()
        {
            return new WindowAutomationPeer(this);
        }
 
        /// <summary>
        /// OnDpiChanged is called when the DPI at which this Window is rendered, changes.
        /// </summary>
        protected override void OnDpiChanged(DpiScale oldDpi, DpiScale newDpi)
        {
            RaiseEvent(new DpiChangedEventArgs(oldDpi, newDpi, Window.DpiChangedEvent, this));
        }
 
        /// <summary>
        /// OnVisualParentChanged is called when the parent of the Visual is changed.
        /// </summary>
        /// <param name="oldParent">Old parent or null if the Visual did not have a parent before.</param>
        protected internal sealed override void OnVisualParentChanged(DependencyObject oldParent)
        {
            VerifyContextAndObjectState();
            base.OnVisualParentChanged(oldParent);
 
            // Checking for Visual parent here covers all the scenarios
            // including the following:
 
            // Window w1 = new Window();
            // Window w2 = new WIndow();
            // w1.Show();
            // w2.Show();
            // w1.VisualChildren.Add(w2);
 
 
            // Window w1 = new Window();
            // Window w2 = new WIndow();
            // w1.Show();
            // w1.VisualChildren.Add(w2);
            //  w2.Show();
 
            if ( VisualTreeHelper.GetParent(this) != null )
            {
                throw new InvalidOperationException(SR.WindowMustBeRoot);
            }
        }
 
        /// <summary>
        ///     Measurement override. Implements content sizing logic.
        /// </summary>
        /// <remarks>
        ///     Deducts the frame size from the constraint and then passes it on
        ///     to its child.  Only supports one Visual child (just like control)
        /// </remarks>
        protected override Size MeasureOverride(Size availableSize)
        {
            VerifyContextAndObjectState();
 
            // Window content should respect Window's Max/Min size
            // setting in a SizeToContent Window.
            //
            // Take Min/Max[Width/Height] into consideration.  The logic here is similar to
            // that used in FE.MeasureCore but is limited to Min/Max restriction.  Furthermore,
            // we have our own version of MinMax struct called WindowMinMax that takes
            // SizeToContent into account when calculating the min/max values for height/width.
            //
            // We don't do anything special in ArrangeOverride the Arrange size is guaranteed
            // to be the available hwnd size which should be atleast as big as the desired size.
 
            Size frameworkAvailableSize = new Size(availableSize.Width, availableSize.Height);
 
            WindowMinMax mm = GetWindowMinMax();
 
            frameworkAvailableSize.Width  = Math.Max(mm.minWidth,  Math.Min(frameworkAvailableSize.Width, mm.maxWidth));
            frameworkAvailableSize.Height = Math.Max(mm.minHeight, Math.Min(frameworkAvailableSize.Height, mm.maxHeight));
 
            //  call to specific layout to measure
            Size desiredSize = MeasureOverrideHelper(frameworkAvailableSize);
 
            //  maximize desiredSize with user provided min size
            desiredSize = new Size(
                Math.Max(desiredSize.Width, mm.minWidth),
                Math.Max(desiredSize.Height, mm.minHeight));
 
            return desiredSize;
        }
 
        /// <summary>
        ///     ArrangeOverride allows for the customization of the positioning of children.
        /// </summary>
        /// <remarks>
        ///     Deducts the frame size of the window from the constraint and then
        ///     arranges its child.  Supports only one child.
        /// </remarks>
        protected override Size ArrangeOverride(Size arrangeBounds)
        {
            VerifyContextAndObjectState();
 
            // Window content should respect Window's Max/Min size
            // setting in a SizeToContent Window.
 
            WindowMinMax mm = GetWindowMinMax();
 
            arrangeBounds.Width  = Math.Max(mm.minWidth,  Math.Min(arrangeBounds.Width, mm.maxWidth));
            arrangeBounds.Height = Math.Max(mm.minHeight, Math.Min(arrangeBounds.Height, mm.maxHeight));
 
            // Three primary cases
            //      1) hwnd does not exist  -- don't do anything
            //      1a) CompositionTarget is invalid -- don't do anything
            //      2) Child visual exists  -- arrange child at arrangeBounds - window frame size
            //      3) No Child visual      -- don't do anything
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return arrangeBounds;
            }
 
            if (this.VisualChildrenCount > 0)
            {
                UIElement child = this.GetVisualChild(0) as UIElement;
                if (child != null)
                {
                    // Find out the size of the window frame x.
                    // (constraint - x) is the size we pass onto
                    // our child
                    Size frameSize = GetHwndNonClientAreaSizeInMeasureUnits();
 
                    // In some instances (constraint size - frame size) can be negative. One instance
                    // is when window is set to minimized before layout has happened.  Apparently, Win32
                    // gives a rect(-32000, -32000, -31840, -31975) for GetWindowRect when hwnd is
                    // minimized.  However, when we calculate the frame size we get width = 8 and
                    // height = 28!!!  Here, we will take the max of zero and the difference b/w the
                    // hwnd size and the frame size
                    //
                    // PS Windows OS Bug: 955861
 
                    Size childArrangeBounds = new Size
                    {
                        Width = Math.Max(0.0, arrangeBounds.Width - frameSize.Width),
                        Height = Math.Max(0.0, arrangeBounds.Height - frameSize.Height)
                    };
 
                    child.Arrange(new Rect(childArrangeBounds));
 
                    // Windows OS bug # 928719, 953458
                    // The default impl of FlowDirection is that it adds a transform on the element
                    // on whom the FlowDirection property is RlTb.  However, transforms work only if
                    // there is a parent visual.  In the window case, we are the root and thus this
                    // does not work.  Thus, we add the same transform to our child, if our
                    // FlowDireciton = Rltb.
                    if (FlowDirection == FlowDirection.RightToLeft)
                    {
                        InternalSetLayoutTransform(child, new MatrixTransform(-1.0, 0.0, 0.0, 1.0, childArrangeBounds.Width, 0.0));
                    }
}
            }
            return arrangeBounds;
        }
 
        /// <summary>
        ///     This method is invoked when the Content property changes.
        /// </summary>
        /// <param name="oldContent">The old value of the Content property.</param>
        /// <param name="newContent">The new value of the Content property.</param>
        protected override void OnContentChanged(object oldContent, object newContent)
        {
            base.OnContentChanged(oldContent, newContent);
 
            // FxCop: ConstructorsShouldNotCallBaseClassVirtualMethods::
            // System.Windows (presentationframework.dll 2 violation)
            //
            // We can't set IWS property in the cctor since it results
            // in calling some virtual.  So, as an alternative we set it
            // when content is changed and when we set the root visual.
            // We set it here, b/c once window has logical children, they
            // can query for inherited IWS property.
            SetIWindowService();
 
            // We post a dispatcher work item to fire ContentRendered
            // only if this is Loaded in the tree.  If not, we will
            // post it from the LoadedHandler.  This guarantees that
            // we don't fire ContentRendered on a subtree that is not
            // connected to a PresentationSource
            if (IsLoaded == true)
            {
                PostContentRendered();
            }
            else
            {
                // _postContentRenderedFromLoadedHandler == true means
                // that we deferred to the Loaded event to PostConetentRendered
                // for the previous content change and Loaded has not fired yet.
                // Thus we don't want to hook up another event handler
                if (_postContentRenderedFromLoadedHandler == false)
                {
                    this.Loaded += new RoutedEventHandler(LoadedHandler);
                    _postContentRenderedFromLoadedHandler = true;
                }
            }
        }
 
        /// <summary>
        ///     This even fires after the window source is created before it is shown. This event is non cancelable and is
        ///     for user infromational purposes
        /// </summary>
        /// <remarks>
        ///     This method follows the .Net programming guideline of having a protected virtual
        ///     method that raises an event, to provide a convenience for developers that subclass
        ///     the event. If you override this method - you need to call base.OnSourceInitialized(...) for
        ///     the corresponding event to be raised.
        /// </remarks>
        /// <param name="e"></param>
        protected virtual void OnSourceInitialized(EventArgs e)
        {
            VerifyContextAndObjectState();
 
            // Setting WindowBackdrop
            WindowBackdropManager.SetBackdrop(this, WindowBackdropType);
 
            EventHandler handler = (EventHandler)Events[EVENT_SOURCEINITIALIZED];
            if (handler != null) handler(this, e);
        }
 
        /// <summary>
        ///     This event fires when window is activated. This event is non cancelable and is
        ///     for user infromational purposes
        /// </summary>
        /// <remarks>
        ///     This method follows the .Net programming guideline of having a protected virtual
        ///     method that raises an event, to provide a convenience for developers that subclass
        ///     the event. If you override this method - you need to call base.OnActivated(...) for
        ///     the corresponding event to be raised.
        /// </remarks>
        /// <param name="e"></param>
        protected virtual void OnActivated(EventArgs e)
        {
            VerifyContextAndObjectState();
            EventHandler handler = (EventHandler)Events[EVENT_ACTIVATED];
            if (handler != null) handler(this, e);
        }
 
        /// <summary>
        ///     This even fires when window is deactivated. This event is non cancelable and is
        ///     for user infromational purposes
        /// </summary>
        /// <remarks>
        ///     This method follows the .Net programming guideline of having a protected virtual
        ///     method that raises an event, to provide a convenience for developers that subclass
        ///     the event. If you override this method - you need to call base.OnDeactivated(...) for
        ///     the corresponding event to be raised.
        /// </remarks>
        protected virtual void OnDeactivated(EventArgs e)
        {
            VerifyContextAndObjectState();
            EventHandler handler = (EventHandler)Events[EVENT_DEACTIVATED];
            if (handler != null) handler(this, e);
        }
 
        /// <summary>
        ///     This even fires when window state is changed. This event is non
        ///     cancelable and is for user infromational purposes
        /// </summary>
        /// <remarks>
        ///     This method follows the .Net programming guideline of having a protected virtual
        ///     method that raises an event, to provide a convenience for developers
        ///     that subclass the event. If you override this method - you need to call
        ///     base.OnStateChanged(...) for the corresponding event to be raised.
        /// </remarks>
        protected virtual void OnStateChanged(EventArgs e)
        {
            VerifyContextAndObjectState();
            EventHandler handler = (EventHandler)Events[EVENT_STATECHANGED];
            if (handler != null) handler(this, e);
        }
 
        /// <summary>
        ///     This event fires when window location changes. This event is not
        ///     cancelable and is for user infromational purposes
        /// </summary>
        /// <remarks>
        ///     This method follows the .Net programming guideline of having a protected virtual
        ///     method that raises an event, to provide a convenience for developers that subclass
        ///     the event. If you override this method - you need to call base.OnLocationChanged(...) for
        ///     the corresponding event to be raised.
        /// </remarks>
        protected virtual void OnLocationChanged(EventArgs e)
        {
            VerifyContextAndObjectState();
            EventHandler handler = (EventHandler)Events[EVENT_LOCATIONCHANGED];
            if (handler != null) handler(this, e);
        }
 
        /// <summary>
        ///     This event fires when window is Closing. This event is cancelable and thus the
        ///     user can set the CancelEventArgs.Cancel property to true to dismiss window
        ///     closing
        /// </summary>
        /// <remarks>
        ///     This method follows the .Net programming guideline of having a protected virtual
        ///     method that raises an event, to provide a convenience for developers that subclass
        ///     the event. If you override this method - you need to call base.OnClosing(...) for
        ///     the corresponding event to be raised.
        /// </remarks>
        protected virtual void OnClosing(CancelEventArgs e)
        {
            VerifyContextAndObjectState();
            CancelEventHandler handler = (CancelEventHandler)Events[EVENT_CLOSING];
            if (handler != null) handler(this, e);
        }
 
        /// <summary>
        ///     This event fires when window is closed. This event is non cancelable and is
        ///     for user infromational purposes
        /// </summary>
        /// <remarks>
        ///     This method follows the .Net programming guideline of having a protected virtual
        ///     method that raises an event, to provide a convenience for developers that subclass
        ///     the event. If you override this method - you need to call base.OnClosed(...) for
        ///     the corresponding event to be raised.
        /// </remarks>
        protected virtual void OnClosed(EventArgs e)
        {
            VerifyContextAndObjectState();
            EventHandler handler = (EventHandler)Events[EVENT_CLOSED];
            if (handler != null) handler(this, e);
        }
 
        /// <summary>
        ///     This override fires the ContentRendered event.
        /// </summary>
        /// <param name="e"></param>
        protected virtual void OnContentRendered(EventArgs e)
        {
            VerifyContextAndObjectState();
 
            // After the content is rendered we want to check if there is an element that needs to be focused
            // If there is - set focus to it
            DependencyObject doContent = Content as DependencyObject;
            if (doContent != null)
            {
                IInputElement focusedElement = FocusManager.GetFocusedElement(doContent) as IInputElement;
                if (focusedElement != null)
                    focusedElement.Focus();
            }
 
            EventHandler handler = (EventHandler)Events[EVENT_CONTENTRENDERED];
            if (handler != null) handler(this, e);
        }
        #endregion Protected Methods
 
        //---------------------------------------------------
        //
        // Protected Internal Methods
        //
        //---------------------------------------------------
        #region Protected Internal Methods
 
        /// <summary>
        /// Called when the <see cref="System.Windows.Media.VisualCollection"/> of the visual object is modified
        /// </summary>
        /// <param name="visualAdded">The <see cref="Visual"/> that was added to the collection</param>
        /// <param name="visualRemoved">The <see cref="Visual"/> that was removed from the collection</param>
        protected internal override void OnVisualChildrenChanged(DependencyObject visualAdded, DependencyObject visualRemoved)
        {
            VerifyContextAndObjectState();
 
            var handler = Events[EVENT_VISUALCHILDRENCHANGED] as EventHandler<EventArgs>;
            handler?.Invoke(this, new EventArgs());
        }
 
        #endregion
 
        //---------------------------------------------------
        //
        // Internal Methods
        //
        //---------------------------------------------------
        #region Internal Methods
        internal Point DeviceToLogicalUnits(Point ptDeviceUnits)
        {
            Invariant.Assert(IsCompositionTargetInvalid == false, "IsCompositionTargetInvalid is supposed to be false here");
            Point ptLogicalUnits = _swh.CompositionTarget.TransformFromDevice.Transform(ptDeviceUnits);
            return ptLogicalUnits;
        }
 
        internal Point LogicalToDeviceUnits(Point ptLogicalUnits)
        {
            Invariant.Assert(IsCompositionTargetInvalid == false, "IsCompositionTargetInvalid is supposed to be false here");
            Point ptDeviceUnits = _swh.CompositionTarget.TransformToDevice.Transform(ptLogicalUnits);
            return ptDeviceUnits;
        }
 
        internal void AddFluentDictionary(ResourceDictionary value, out bool invalidateResources)
        {
            invalidateResources = false;
 
            if(_reloadFluentDictionary && !AreResourcesInitialized)
            {
                if(value != null && ThemeMode != ThemeMode.None)
                {
                    value.MergedDictionaries.Insert(0, ThemeManager.GetThemeDictionary(ThemeMode));
                    invalidateResources = true;
                }
 
                _reloadFluentDictionary = false;
            }
        }
 
        internal static bool VisibilityToBool(Visibility v)
        {
            switch (v)
            {
                case Visibility.Visible:
                    return true;
                case Visibility.Hidden:
                case Visibility.Collapsed:
                    return false;
                default:
                    return false;
            }
        }
 
        /// <summary>
        ///     Called by ResizeGrip control to set its reference in the Window object
        /// </summary>
        /// <remarks>
        ///     RBW doesn't need ResizeGrip and hence it doesn't do
        ///     anything in this virtual
        /// </remarks>
        internal virtual void SetResizeGripControl(Control ctrl)
        {
            _resizeGripControl = ctrl;
        }
 
        internal virtual void ClearResizeGripControl(Control oldCtrl)
        {
            if (oldCtrl == _resizeGripControl)
            {
                _resizeGripControl = null;
            }
        }
 
        internal virtual void TryClearingMainWindow()
        {
            if (IsInsideApp && this == App.MainWindow)
            {
                App.MainWindow = null;
            }
        }
 
        /// <summary>
        ///     Send a WM_CLOSE message to close the window. When the WM_CLOSE message is
        ///     processed by the WindowFilterMessage function, the Closing event is fired.
        ///     Closing event is cancelable and thus can dismiss window closing.
        /// </summary>
        /// <param name="shutdown">Specifies whether the app should shutdown or not</param>
        /// <param name="ignoreCancel">Specifies whether cancelling closing should be ignored </param>
        internal void InternalClose(bool shutdown, bool ignoreCancel)
        {
            VerifyNotClosing();
 
            if (_disposed == true)
            {
                return;
            }
 
            _appShuttingDown = shutdown;
            _ignoreCancel = ignoreCancel;
 
            if ( IsSourceWindowNull )
            {
                _isClosing = true;
 
                // Event handler exception continuality: if exception occurs in Closing event handler, the
                // cleanup action is to finish closing.
                CancelEventArgs e = new CancelEventArgs(false);
                try
                {
                    OnClosing(e);
                }
                catch
                {
                    CloseWindowBeforeShow();
                    throw;
                }
 
                if (ShouldCloseWindow(e.Cancel))
                {
                    CloseWindowBeforeShow();
                }
                else
                {
                    _isClosing = false;
                    // Dialog does not close with ESC key after it has been cancelled
                    //
                    // No need to reset DialogResult to null here since source window is null.  That means
                    // that ShowDialog has not been called and thus no need to worry about DialogResult.
                }
            }
            else
            {
                // close window synchronously
 
                // We demand for UIPermission AllWindows at the public API, Window.Close(), level.
                // It can be called when shutting down the app.
                // The public entry to that code path Application.Shutdown is
                // also protected with a demand for UIPermission with AllWindow access
 
                // SendMessage's return value is dependent on the message send.  WM_CLOSE
                // return value just signify whether the WndProc handled the
                // message or not, so it is not interesting
                UnsafeNativeMethods.UnsafeSendMessage(Handle, WindowMessage.WM_CLOSE, new IntPtr(), new IntPtr());
            }
        }
 
        // NOTE:
        // We fire Closing and Closed envent even if the hwnd is not
        // created yet i.e. window is not shown.
        private void CloseWindowBeforeShow()
        {
            InternalDispose();
 
            // raise Closed event
            OnClosed(EventArgs.Empty);
        }
 
        internal bool IsSourceWindowNull
        {
            get
            {
                if ( _swh != null )
                {
                    return _swh.IsSourceWindowNull;
                }
                return true;
            }
        }
 
        internal bool IsCompositionTargetInvalid
        {
            get
            {
                if (_swh != null)
                {
                    return _swh.IsCompositionTargetInvalid;
                }
                return true;
            }
        }
 
        internal NativeMethods.RECT WorkAreaBoundsForNearestMonitor
        {
            get
            {
                Debug.Assert( _swh != null );
                return _swh.WorkAreaBoundsForNearestMonitor;
            }
        }
 
        internal Size WindowSize
        {
            get
            {
                Debug.Assert( _swh != null );
                return _swh.WindowSize;
            }
        }
 
        // This is currently exposed just for DRTs.
        // PLEASE NOTE THAT IF YOU ARE CALLING THIS WITHIN AVALON CODE - YOU ARE CALLING A SECURITY CRITICAL METHOD
        // YOU WILL HAVE TO WORK WITH THE SECURITY TEAM TO REVIEW YOUR USAGE
        internal HwndSource HwndSourceWindow
        {
            get
            {
 
                if ( _swh != null )
                    return _swh.HwndSourceWindow;
                else
                    return null;
            }
        }
 
        // WCP Window:  Define the dispose behavior for Window
        // We need to define the Dispose behavior for the Window class.  We used to
        // dispose off the object when the Window was closed; however, this does not
        // work for EmbeddedDialogs since we need to get the DialogResult after the
        // EmbeddedDialog is closed.  So, we changed this functionality for M6.
        // We need to define if and when we should dispose the Window object.
 
        // NOTE: added on 10/03/02
        // We should set our _hwndSource reference to null here.
        // Else, if the hwndSource is not null, users can call
        // public APIs that use hwndSource and it will fail
 
        /// <summary>
        ///     can be used by internal derived class
        /// </summary>
        private void InternalDispose()
        {
            _disposed = true;
 
            // UpdateWindowLists here instead of in WM_CLOSE for 2 reasons.
            // 1. WM_CLOSE is not fired for child window. An example would be RootBrowserWindow (bug 1754467).
            // 2. It is not fired as a result of calling Dispose directly on HwndSource (HwndSource distroy the window).
            UpdateWindowListsOnClose();
 
#if DISPOSE
            // detach all events
            Utilities.SafeDispose(ref _events);
#endif
            // NOTE:
            // This InternalDispose method is called while
            // processing WM_DESTROY msg. Once we're done
            // processing this msg, HwndWrapper does its processing.
            // We don't need to dispose _swh here b/c
            // HwndWrapper fires the hwndDisposed event while
            // processing WM_DESTROY msg. HwndSource listens to
            // this event and disposes itself i.e. the CompositionTarget etc.
            // If we call dispose here, HwndWrapper.Dispose sends
            // a WM_DESTROY msg and it is a duplicate msg.
 
            // When the window is closing, stop any deferred operations.
            if (_taskbarRetryTimer != null)
            {
                _taskbarRetryTimer.Stop();
                _taskbarRetryTimer = null;
            }
 
            try
            {
                ClearSourceWindow();
 
                Utilities.SafeDispose(ref _hiddenWindow);
                Utilities.SafeDispose(ref _defaultLargeIconHandle);
                Utilities.SafeDispose(ref _defaultSmallIconHandle);
                Utilities.SafeDispose(ref _currentLargeIconHandle);
                Utilities.SafeDispose(ref _currentSmallIconHandle);
                Utilities.SafeRelease(ref _taskbarList);
 
                if(ThemeMode != ThemeMode.None)
                {
                    ThemeManager.FluentEnabledWindows.Remove(this);
                }
            }
            finally
            {
                _isClosing = false;
            }
        }
 
        /// <summary>
        ///     This is a callback called to set the window Visual. It is called after
        ///     the source window has been created.
        /// </summary>
        internal override void OnAncestorChanged()
        {
            base.OnAncestorChanged();
            if (Parent != null)
            {
                throw new InvalidOperationException(SR.WindowMustBeRoot);
            }
        }
 
        /// <summary>
        ///     Initializes the _style and _styleEx bits.
        /// </summary>
        internal virtual void CreateAllStyle()
        {
            // we don't need to set to WS_OVERLAPPEDWINDOW since all the styles are set
            // manually depending on window properties.
            //
            // We always have the sysmenu
            // If the Window has a Caption, then this also
            // shows the icon and the close box
 
            _Style = NativeMethods.WS_CLIPCHILDREN | NativeMethods.WS_SYSMENU;
            _StyleEx = 0;
 
            CreateWindowStyle();
            CreateWindowState();
 
            // do all the other checks and update style bits
 
            // Visibility bits
            if ( _isVisible )
            {
                _Style |= NativeMethods.WS_VISIBLE;
            }
 
            SetTaskbarStatus();
            CreateTopmost();
            CreateResizibility();
            CreateRtl();
        }
 
        /// <summary>
        ///     Create the window
        ///     Virtual so that subclasses ( currently only RootBrowserWindow) - may assert for HwndSource creation.
        /// </summary>
        internal virtual void CreateSourceWindowDuringShow()
        {
            CreateSourceWindow(true);
        }
 
        /// <remarks>
        ///     This method is called in the following two cases:
        ///     1. As a result of calling Show on Window, Window.Show (Visibility = Visible).
        ///     2. WindowInteropHelper.EnsureHandle is called. Hwnd is created but not shown.
        ///        We only create hwnd. RootVisual is not set until Show.
        ///
        ///     This method does the following:
        ///     CalculateLocation of the window:
        ///         Calculates location of window. If either Top/Left is CW_USEDEFAULT,
        ///         we have to make sure that window displays on the screen. Details in
        ///         the methods.
        ///
        ///     Calculate size:
        ///         Calculates size. If either one of Width/Height is set, then we
        ///         create a window with both set to default and then resizing it
        ///         to the set value
        ///
        ///     Update Properties:
        ///         After the hwnd is created, we updated our property values for various
        ///         properties
        ///
        ///     Modify the message filter:
        ///         Elevated processes don't normally receive messages from unelevated processes.
        ///         For taskbar integration in Windows 7 we need to explicitly allow explorer to post
        ///         messages to this window..
        /// </remarks>
        /// <param name="duringShow">Specifies whether this method is called from Window.Show
        /// or WindowInteropHelper.EnsureHandle</param>
        internal void CreateSourceWindow(bool duringShow)
        {
            VerifyContextAndObjectState();
            VerifyCanShow();
            VerifyNotClosing();
            VerifyConsistencyWithAllowsTransparency();
 
            // We do not support create hwnd before shown for RBW.
            if (duringShow == false)
            {
                VerifyApiSupported();
            }
 
            // we need to cache initial requested top and left as the very first thing
            // since updating the styles etc (as for borderless case below) fires
            // WM_MOVE and update Top/Left.  Also, for RBW case, Top/Left is returned
            // as 0,0 since they are inconsequential.
 
            double requestedTop = 0;
            double requestedLeft = 0;
            double requestedWidth = 0;
            double requestedHeight = 0;
 
            GetRequestedDimensions(ref requestedLeft, ref requestedTop, ref requestedWidth, ref requestedHeight);
 
            // Initially, an HWND created in a Per-Monitor Aware process may not
            // have a reliable DPI. Instead, it would be associated with the monitor in which the screen point (0,0)
            // happens to be, but the screen point corresponding to a WPF Window's (requestedLeft, requestedTop) may
            // not map into that monitor. Our initial DeviceToLogicalUnits and LogicalToDeviceUnits transforms
            // would normally have to rely on the DPI obtained from this HWND, which may lead us to
            // an incorrect result when calculating screen points. To avoid this, we use a special heuristic here
            // to identify the (screenLeft, screenTop) initially without relying on the HWND, and then
            // supply it to HwndSourceParameters at the time of HWND creation. This would in turn set up the HWND
            // with the correct DPI from the start, and all our transformations would also follow suit.
            var topLeftPointHelper = new WindowStartupTopLeftPointHelper(new Point(requestedLeft, requestedTop));
 
            using (HwndStyleManager sm = HwndStyleManager.StartManaging(this, StyleFromHwnd, StyleExFromHwnd))
            {
                // set window style and styleEx bits
                // CreateAllStyle is internal virtual. RBW overrides it to set style to WS_CHILD and
                // set parent handle
                CreateAllStyle();
 
                // create the Win32 Window
                HwndSourceParameters param = CreateHwndSourceParameters();
                if (topLeftPointHelper.ScreenTopLeft.HasValue)
                {
                    var screenTopLeft = topLeftPointHelper.ScreenTopLeft.Value;
                    param.SetPosition((int)screenTopLeft.X, (int)screenTopLeft.Y);
                }
 
                // HwndSource disposes itself when HwndWrapper process the WM_DESTROY message.
                // Window sends WM_CLOSE (which in turn sends WM_DESTROY) to the hwnd when
                // Window.Close() is called.  Thus, this HwndSource created by window is always
                // disposed by HwndSource itself
                HwndSource source = new HwndSource(param);
 
                _swh = new SourceWindowHelper(source);
                source.SizeToContentChanged += new EventHandler(OnSourceSizeToContentChanged);
 
                // since we created the window with the style, mark
                // sm as not dirty so that we don't unneccessarialy update
                // the Win32 style bits.
                sm.Dirty = false;
 
                // NOTE:
                // HwndSource COULD ADD STYLE/STYLEEX BITS TO THE ONES SUPPLIED. TODAY,
                // HwndSource ADDS WS_CLIPCHILDREN TO THE STYLE WHICH MAY NOT
                // BE REFLECTED IN OUR CACHED STYLE BITS.  IF WE WERE
                // TO UPDATE THE WIN32 SYTLE, THE STYLES ADDED BY HWNDSOURCE
                // WOULD BE LOST.  IF IN THE FUTURE, WE UPDATE STYLE
                // BITS IN THIS METHOD, WE WOULD NEED TO FIRST GET THE
                // CURRENT STYLE FROM THE HWND USING:
                //
                // _Style = StyleFromHwnd;
                //
 
                // Since RBW cannot access WindowStyle, hence it overrides
                // this virtual and does nothing it it.
                CorrectStyleForBorderlessWindowCase();
} // end using StyleManager
 
            // We don't do anything that uses the Window styles below so we might as
            // well close the using so that we update the new style to the hwnd.
            // This change was made to solve a problem where SizeToContent.WidthAndHeight borderless
            // windows where actually bigger than the content size.  This was b/c of
            // the fact that we didn't update the style of the hwnd before setting
            // the RootVisual which inturn calls MeasureOverride on Window from
            // where we calculating the non-client area size using the stale
            // style bits from the hwnd.
 
            // Add Disposed event handler
            _swh.AddDisposedHandler ( new EventHandler(OnSourceWindowDisposed) );
 
            _hwndCreatedButNotShown = !duringShow;
 
            // Since this is only for Win7 taskbar integration, don't bother with this call unless we're on an appropriate OS.
            if (Utilities.IsOSWindows7OrNewer)
            {
                // In case the application is run elevated, explicitly allow WM_TASKBARBUTTONCREATED and WM_COMMAND
                // through the message filter.  This method call will fail if the application was started with
                // SECURITY_MANDATORY_LOW_RID, so don't propagate failed error codes.
                // It's not the end of the world if this fails: Shell integration simply won't work.
                MSGFLTINFO info;
                UnsafeNativeMethods.ChangeWindowMessageFilterEx(_swh.Handle, WM_TASKBARBUTTONCREATED, MSGFLT.ALLOW, out info);
                UnsafeNativeMethods.ChangeWindowMessageFilterEx(_swh.Handle, WindowMessage.WM_COMMAND, MSGFLT.ALLOW, out info);
            }
 
            if (Standard.Utility.IsOSWindows10OrNewer)
            {
                if (ThemeManager.IsFluentThemeEnabled)
                {
                    ThemeManager.ApplyStyleOnWindow(this);
                }
 
                if (_deferThemeLoading)
                {
                    _deferThemeLoading = false;
                    ThemeManager.OnWindowThemeChanged(this, ThemeMode.None, ThemeMode);
                }
            }
 
            // Sub classes can have different intialization. RBW does very minimalistic
            // stuff in its override
            SetupInitialState(requestedTop, requestedLeft, requestedWidth, requestedHeight);
 
            // Fire SourceInitialized event
            OnSourceInitialized(EventArgs.Empty);
        }
 
        internal void SetImmersiveDarkMode(bool useDarkMode)
        {
            if(!Standard.Utility.IsOSWindows11OrNewer) return;
 
            if(_useDarkMode == useDarkMode) return;
 
            IntPtr handle = Handle;
            if (handle != IntPtr.Zero)
            {
                bool succeeded = SNM.DwmSetWindowAttributeUseImmersiveDarkMode(handle, useDarkMode);
                if(succeeded)
                {
                    _useDarkMode = useDarkMode;
                }
            }
        }
 
        internal virtual HwndSourceParameters CreateHwndSourceParameters()
        {
            HwndSourceParameters param = new HwndSourceParameters(Title, NativeMethods.CW_USEDEFAULT, NativeMethods.CW_USEDEFAULT)
            {
                UsesPerPixelOpacity = AllowsTransparency,
                WindowStyle = _Style,
                ExtendedWindowStyle = _StyleEx,
                ParentWindow = _ownerHandle,
                AdjustSizingForNonClientArea = true,
                HwndSourceHook = new HwndSourceHook(WindowFilterMessage) // hook to process window messages
            };
            return param;
        }
 
        private void OnSourceSizeToContentChanged(object sender, EventArgs args)
        {
            SizeToContent = HwndSourceSizeToContent;
        }
 
        internal virtual void CorrectStyleForBorderlessWindowCase()
        {
            // We create an OverLapped window for which user adds WS_CAPTION
            // to the passed in style.  If we were creating a borderless window,
            // remove WS_CAPTION from the hwnd.
            //
            // We should really be using WS_POPUP for borderless windows, but
            // there's a bug with default sizing where user creates the popup
            // window with 0,0 size.
            //
 
            using (HwndStyleManager sm = HwndStyleManager.StartManaging(this, StyleFromHwnd, StyleExFromHwnd))
            {
                if (WindowStyle == WindowStyle.None)
                {
                    _Style = _swh.StyleFromHwnd;
                    _Style &= ~NativeMethods.WS_CAPTION;
                }
            }
        }
 
        internal virtual void GetRequestedDimensions(ref double requestedLeft, ref double requestedTop, ref double requestedWidth, ref double requestedHeight)
        {
            requestedTop = this.Top;
            requestedLeft = this.Left;
            requestedWidth = this.Width;
            requestedHeight = this.Height;
        }
 
        internal virtual void SetupInitialState(double requestedTop, double requestedLeft, double requestedWidth, double requestedHeight)
        {
            // Push the current SizeToContent value to HwndSource after it's created. Initial sync up.
            HwndSourceSizeToContent = (SizeToContent) GetValue(SizeToContentProperty);
            UpdateIcon();
 
            NativeMethods.RECT rc = WindowBounds;
            Size sizeDeviceUnits = new Size(rc.right - rc.left, rc.bottom - rc.top);
            double xDeviceUnits = rc.left;
            double yDeviceUnits = rc.top;
 
            bool updateHwndPlacement = false;
 
            // This code is absolutely necessary here.  This is the initial sync up
            // for Left/Top.  We don't do this in WM_MOVE if sourceWindow is
            // null (it is null b/c the call to create HwndSource has not returned at the point)
            // and we are not listening to WM_CREATE either.  The reasons for not
            // doing so is b/c if sourceWindow is null, we can't get to the CompositionTarget
            // to convert b/w device and logical pixels
            Point currentLocationLogicalUnits = DeviceToLogicalUnits(new Point(xDeviceUnits, yDeviceUnits));
 
            // Update our current hwnd ActualTop/ActualLeft values
            // The _actualLeft and _actualTop need to be updated before we coerce Top and Left.
            // See CoerceTop for more info.
            _actualLeft = currentLocationLogicalUnits.X;
            _actualTop = currentLocationLogicalUnits.Y;
            // Coerce Top and Left value to be the intial value used in CreateWindowEx.
            try
            {
                _updateHwndLocation = false;
                CoerceValue(TopProperty);
                CoerceValue(LeftProperty);
            }
            finally
            {
                _updateHwndLocation = true;
            }
 
            Point requestedSizeDeviceUnits = LogicalToDeviceUnits(new Point(requestedWidth, requestedHeight));
            Point requestedLocationDeviceUnits = LogicalToDeviceUnits(new Point(requestedLeft, requestedTop));
 
            // if Width was specified and is not the same as the current width, then update it
            if ((!double.IsNaN(requestedWidth)) && (!DoubleUtil.AreClose(sizeDeviceUnits.Width, requestedSizeDeviceUnits.X)))
            {
                // at this stage, ActualWidth/Height is not set since
                // layout has not happened (it happens when we set the
                // RootVisual of the HwndSource)
 
                updateHwndPlacement = true;
                sizeDeviceUnits.Width = requestedSizeDeviceUnits.X;
 
                // SetWindowPlacement for Width/Height when Width/Height is set and WindowState is maximized/minimized.
                if (WindowState != WindowState.Normal)
                {
                    UpdateHwndRestoreBounds(requestedWidth, BoundsSpecified.Width);
                }
            }
 
            // if Height was specified and is not the same as the current height, then update it
            if (!double.IsNaN(requestedHeight) && (!DoubleUtil.AreClose(sizeDeviceUnits.Height, requestedSizeDeviceUnits.Y)))
            {
                // at this stage, ActualWidth/Height is not set since
                // layout has not happened (it happens when we set the
                // RootVisual of the HwndSource)
 
                updateHwndPlacement = true;
                sizeDeviceUnits.Height = requestedSizeDeviceUnits.Y;
 
                // SetWindowPlacement for Width/Height when Width/Height is set and WindowState is maximized/minimized.
                if (WindowState != WindowState.Normal)
                {
                    UpdateHwndRestoreBounds(requestedHeight, BoundsSpecified.Height);
                }
            }
 
            // if left was specified and is not the same as the current left, then update it
            if (!double.IsNaN(requestedLeft) && (!DoubleUtil.AreClose(xDeviceUnits, requestedLocationDeviceUnits.X)))
            {
                updateHwndPlacement = true;
                xDeviceUnits = requestedLocationDeviceUnits.X;
 
                // SetWindowPlacement for Top/Left when Top/Left is set and WindowState is maximized/minimized.
                if (WindowState != WindowState.Normal)
                {
                    UpdateHwndRestoreBounds(requestedLeft, BoundsSpecified.Left);
                }
            }
 
            // if top was specified and is not the same as the current top, then update it
            if (!double.IsNaN(requestedTop) && (!DoubleUtil.AreClose(yDeviceUnits, requestedLocationDeviceUnits.Y)))
            {
                updateHwndPlacement = true;
                yDeviceUnits = requestedLocationDeviceUnits.Y;
 
                // SetWindowPlacement for Top/Left when Top/Left is set and WindowState is maximized/minimized.
                if (WindowState != WindowState.Normal)
                {
                    UpdateHwndRestoreBounds(requestedTop, BoundsSpecified.Top);
                }
            }
 
            Point minSizeDeviceUnits = LogicalToDeviceUnits(new Point(MinWidth, MinHeight));
            Point maxSizeDeviceUnits = LogicalToDeviceUnits(new Point(MaxWidth, MaxHeight));
 
            // We need this here b/c when WM_GETMINMAXINFO is handled as a result of
            // creating the hwnd, _swh is null and thus we cannot get to the CompositionTarget
            // to convert b/w device and logical units.  Thus, once the hwnd is created, we
            // need to check to make sure that the hwnd is within the min/max boundaries.
            //
            // Here is idea it to detect if we are outside the min/max range and if yes, cause
            // a resize.  This resize will call WM_GETMINMAXINFO and that will further take
            // care of the bounds
 
            if (!Double.IsPositiveInfinity(maxSizeDeviceUnits.X) && (sizeDeviceUnits.Width > maxSizeDeviceUnits.X))
            {
                updateHwndPlacement = true;
                sizeDeviceUnits.Width = maxSizeDeviceUnits.X;
            }
 
            if (!Double.IsPositiveInfinity(minSizeDeviceUnits.Y) && (sizeDeviceUnits.Height > maxSizeDeviceUnits.Y))
            {
                updateHwndPlacement = true;
                sizeDeviceUnits.Height = maxSizeDeviceUnits.Y;
            }
 
            if (sizeDeviceUnits.Width < minSizeDeviceUnits.X)
            {
                updateHwndPlacement = true;
                sizeDeviceUnits.Width = minSizeDeviceUnits.X;
            }
 
            if (sizeDeviceUnits.Height < minSizeDeviceUnits.Y)
            {
                updateHwndPlacement = true;
                sizeDeviceUnits.Height = minSizeDeviceUnits.Y;
            }
 
            // Now that we know the window Width/Height/Left/Top, we want to calculate
            // the location based on WindowStartupLocation enum.  All inputs to
            // CalculateWindowLocation must be in DEVICE units.  It will return true,
            // if it updated theinput values of left, top.
            //
            updateHwndPlacement = (CalculateWindowLocation(ref xDeviceUnits, ref yDeviceUnits, sizeDeviceUnits)? true: updateHwndPlacement);
 
            // We need to update the hwnd size before we set RootVisual b/c setting RootVisual
            // results in a Measure/Arrange/Layout and we want to set the correct hwnd size
            // now so that layout happens at the correct size.
            //
 
            //
            // We are intentionally not merging this SetWindowPos with the one in SetRootVisualAndUpdateSTC. The reason is
            // that if Width or Height is set we initially create the hwnd with both as default win32 size
            // and later (above call to SetWindowPos) resizes it to reflect the specified Width or Height.
            // After this is done, we set RootVisual which results in a layout with the correct size of
            // the hwnd.
            //
            // If we merge the two SetWindowPos calls, then for the scenario where
            // Width or Height is set initially, we create the hwnd with win32 default size.  Setting RootVisual
            // would cause layout to happen at that default size.  Now, we resize the window to reflect the
            // specified size which will result in another layout.
            //
            // The third option is to set RootVisual after doing all resizes.  This is not possible to do for
            // the SizeToContent case since we don't know the size of hwnd until RootVisual is set and layout
            // has happened.
            //
 
            if (updateHwndPlacement == true)
            {
                if (WindowState == WindowState.Normal)
                {
                    UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle),
                        new HandleRef(null, IntPtr.Zero),
                        DoubleUtil.DoubleToInt(xDeviceUnits),
                        DoubleUtil.DoubleToInt(yDeviceUnits),
                        DoubleUtil.DoubleToInt(sizeDeviceUnits.Width),
                        DoubleUtil.DoubleToInt(sizeDeviceUnits.Height),
                        NativeMethods.SWP_NOZORDER | NativeMethods.SWP_NOACTIVATE
                        );
 
                    // The value of Top and Left is affected by WindowState and WindowStartupLocation.
                    // we need to coerce Top and Left whenever these deciding factors change.
                    // More info in CoerceTop.
                    try
                    {
                        _updateHwndLocation = false;
                        _updateStartupLocation = true;
                        CoerceValue(TopProperty);
                        CoerceValue(LeftProperty);
                    }
                    finally
                    {
                        _updateHwndLocation = true;
                        _updateStartupLocation = false;
                    }
                }
            }
 
            // RootVisual is not set until Show.
            // Only set RootVisual when we are going to show the window.
            if (!HwndCreatedButNotShown)
            {
                SetRootVisualAndUpdateSTC();
            }
        }
 
        internal void SetRootVisual()
        {
            //FxCop: ConstructorsShouldNotCallBaseClassVirtualMethods::
            // System.Windows (presentationframework.dll 2 violation)
            //
            // We can't set IWS property in the cctor since it results
            // in calling some virtual.  So, as an alternative we set it
            // when content is changed and when we set the root visual.
            // We set it here, b/c once RootVisual is set, the visual tree
            // can/is created and visual children can query for inherited
            // IWS property.
            SetIWindowService();
 
            // set root visual  synchronously, shell request
            if ( IsSourceWindowNull == false )
            {
                _swh.RootVisual = this;
            }
        }
 
        internal void SetRootVisualAndUpdateSTC()
        {
            SetRootVisual();
 
            // We update size/location based on the value of SizeToContent after the RootVisual is set.
            // The world (size/location) could have changed after SetRootVisual.  Verify assumptions before moving on.
            // Specifically SetRootVisual may have initialized a Layout, and if an event handler Closed the Window
            // as a result then _swh (used by LogicalToDeviceUnits) is already disposed.
            if (!IsSourceWindowNull)
            {
                // if SizeToContent, we would need to set the location again if
                // WSL is CenterOwner or CenterScreen b/c size may have changed
                // after we set the RootVisual on HwndSource.
                // And if the hwnd is created but is not shown yet, we need to re-calculate the location again because
                // the value of the WindowStartupLocation can change after EnsureHandle is called.
                // We only update location if there is any change needed.
                if ((SizeToContent != SizeToContent.Manual) || (HwndCreatedButNotShown == true))
                {
                    NativeMethods.RECT rc = WindowBounds;
                    double xDeviceUnits = rc.left;
                    double yDeviceUnits = rc.top;
 
                    // inputs to CalculateWindowLocation must be in DEVICE units
                    Point newSizeDeviceUnits = LogicalToDeviceUnits(new Point(this.ActualWidth, this.ActualHeight));
                    if (CalculateWindowLocation(ref xDeviceUnits, ref yDeviceUnits, new Size(newSizeDeviceUnits.X, newSizeDeviceUnits.Y)))
                    {
                        if (WindowState == WindowState.Normal)
                        {
                            UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle),
                                new HandleRef(null, IntPtr.Zero),
                                DoubleUtil.DoubleToInt(xDeviceUnits),
                                DoubleUtil.DoubleToInt(yDeviceUnits),
                                0,
                                0,
                                NativeMethods.SWP_NOSIZE | NativeMethods.SWP_NOZORDER | NativeMethods.SWP_NOACTIVATE
                                );
 
                            // The value of Top and Left is affected by WindowState and WindowStartupLocation.
                            // we need to coerce Top and Left whenever these deciding factors change.
                            // More info in CoerceTop.
                            try
                            {
                                _updateHwndLocation = false;
                                _updateStartupLocation = true;
                                CoerceValue(TopProperty);
                                CoerceValue(LeftProperty);
                            }
                            finally
                            {
                                _updateHwndLocation = true;
                                _updateStartupLocation = false;
                            }
                        }
                    }
                }
            }
        }
 
        /// <summary>
        ///     Create the style bits depending on the WindowStyle property
        /// </summary>
        /// <remarks>
        ///     This method does not clear the bits, so the style bits should be cleared
        ///     before calling this method
        /// </remarks>
        private void CreateWindowStyle()
        {
            // Clear the style bits related to WindowStyle
            _Style &= ~NativeMethods.WS_CAPTION;
            _StyleEx &= ~(NativeMethods.WS_EX_CLIENTEDGE | NativeMethods.WS_EX_TOOLWINDOW);
 
            // WS_CAPTION == WS_BORDER | WS_DLGFRAME (0x00C00000L)
            // Thus no need to set/clear WS_BORDER when we are
            // already seting/clearing WS_CAPTION
            switch (WindowStyle)
            {
                case WindowStyle.None:
                    _Style &= (~NativeMethods.WS_CAPTION);
                    break;
 
                case WindowStyle.SingleBorderWindow:
                    _Style |= NativeMethods.WS_CAPTION;
                    break;
 
                case WindowStyle.ThreeDBorderWindow:
                    _Style |= NativeMethods.WS_CAPTION;
                    _StyleEx |= NativeMethods.WS_EX_CLIENTEDGE;
                    break;
 
                case WindowStyle.ToolWindow:
                    _Style |= NativeMethods.WS_CAPTION;
                    _StyleEx |= NativeMethods.WS_EX_TOOLWINDOW;
                    break;
#if Never
                case WindowBorderStyle.Sizable:
                    _startSettings.Style |= NativeMethods.WS_BORDER | NativeMethods.WS_THICKFRAME;
                    break;
 
                case WindowBorderStyle.FixedDialog:
                    _startSettings.Style |= NativeMethods.WS_BORDER;
                    _startSettings.StyleEx |= NativeMethods.WS_EX_DLGMODALFRAME;
                    break;
                case WindowBorderStyle.SizableToolWindow:
                    _startSettings.Style |= NativeMethods.WS_BORDER | NativeMethods.WS_THICKFRAME;
                    _startSettings.StyleEx |= NativeMethods.WS_EX_TOOLWINDOW;
                    break;
#endif
            }
        }
 
        // called as a result of title property changing to propagate it to the hwnd
        internal virtual void UpdateTitle(string title)
        {
            // Adding check for IsCompositionTargetInvalid
            if ( IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                UnsafeNativeMethods.SetWindowText(new HandleRef(this, Handle), title);
            }
        }
 
        // called as a result of Height/MinHeight/MaxHeight and Width/MinWidth/MaxWidth property changing to update the hwnd size
        private void UpdateHwndSizeOnWidthHeightChange(double widthLogicalUnits, double heightLogicalUnits)
        {
            if (!_inTrustedSubWindow)
            {
            }
            Debug.Assert( IsSourceWindowNull == false , "IsSourceWindowNull cannot be true when calling this function");
 
            Point ptDeviceUnits = LogicalToDeviceUnits(new Point(widthLogicalUnits, heightLogicalUnits));
            UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle),
                        new HandleRef(null, IntPtr.Zero),
                        0,
                        0,
                        DoubleUtil.DoubleToInt(ptDeviceUnits.X),
                        DoubleUtil.DoubleToInt(ptDeviceUnits.Y),
                        NativeMethods.SWP_NOMOVE | NativeMethods.SWP_NOZORDER | NativeMethods.SWP_NOACTIVATE
                        );
        }
 
        // Activate or Deactivate window
        internal void HandleActivate(bool windowActivated)
        {
            //
            // This method is called by WM_ACTIVATE msg hander for the stand alone
            // window case.  WM_ACTIVATE is sent twice when activating a minimized
            // window by mouse click; once with window state minimized and then again
            // with window state normal.  Thus, we have the following if conditions
            // to fire Activated/Deactivated events only if we're activating/deactivating
            // the window and the window was previously deactivated/activated.
            //
            // Please look at (Activated event fires twice on window when
            // you minimize and restore a window) on this issue.
            //
 
            // Event handler exception continuality: if exception occurs in Activated/Deactivated event handlers, our state will not be
            // corrupted because the state related to Activated/Deactivated, IsActive is set before the event is fired.
            // Please check Event handler exception continuality if the logic changes.
            if ((windowActivated == true) && (IsActive == false))
            {
                SetValue(IsActivePropertyKey, BooleanBoxes.TrueBox);
                OnActivated(EventArgs.Empty);
            }
            else if ((windowActivated == false) && (IsActive == true))
            {
                SetValue(IsActivePropertyKey, BooleanBoxes.FalseBox);
                OnDeactivated(EventArgs.Empty);
            }
        }
 
        internal virtual void UpdateHeight(double newHeight)
        {
            if (WindowState == WindowState.Normal)
            {
                // We cannot save the current hwnd height in logical units in WmSizeChanged b/c the HwndSource
                // might not be completely created at that time (when we call new HwndSource from CreateSourceWindow)
                // and DeviceToLogicalUnits fails.  Thus we convert to logical units here before we call
                // UpdateHwndSizeonWidthHeightChange since that expects logical units.
 
                NativeMethods.RECT rc = WindowBounds;
                Point sizeLogicalUnits = DeviceToLogicalUnits(new Point(rc.Width, 0));
                UpdateHwndSizeOnWidthHeightChange(sizeLogicalUnits.X, newHeight);
            }
            else
            {
                UpdateHwndRestoreBounds(newHeight, BoundsSpecified.Height);
            }
        }
 
        internal virtual void UpdateWidth(double newWidth)
        {
            if (WindowState == WindowState.Normal)
            {
                // We cannot save the current hwnd width in logical units in WmSizeChanged b/c the HwndSource
                // might not be completely created at that time (when we call new HwndSource from CreateSourceWindow)
                // and DeviceToLogicalUnits fails.  Thus we convert to logical units here before we call
                // UpdateHwndSizeonWidthHeightChange since that expects logical units.
 
                NativeMethods.RECT rc = WindowBounds;
                Point sizeLogicalUnits = DeviceToLogicalUnits(new Point(0, rc.Height));
                UpdateHwndSizeOnWidthHeightChange(newWidth, sizeLogicalUnits.Y);
            }
            else
            {
                // set restore width
                UpdateHwndRestoreBounds(newWidth, BoundsSpecified.Width);
            }
        }
 
        internal virtual void VerifyApiSupported()
        {
            // don't do anything here since we allow this API in Window.
            // Subclasses can throw exception in their override if
            // they don't allow the api.
        }
        #endregion Internal Methods
 
 
        //----------------------------------------------
        //
        // Internal Properties
        //
        //----------------------------------------------
        #region Internal Properties
 
        /// <summary>
        /// Gets or sets a value determining preferred backdrop type for current <see cref="Window"/>.
        /// </summary>
        internal WindowBackdropType WindowBackdropType
        {
            get => (WindowBackdropType)GetValue(WindowBackdropTypeProperty);
            set => SetValue(WindowBackdropTypeProperty, value);
        }
 
        /// <summary>
        /// Property for <see cref="WindowBackdropType"/>.
        /// </summary>
        internal static readonly DependencyProperty WindowBackdropTypeProperty = DependencyProperty.Register(
            nameof(WindowBackdropType),
            typeof(WindowBackdropType),
            typeof(Window),
            new PropertyMetadata(
                WindowBackdropType.MainWindow,
                new PropertyChangedCallback(OnBackdropTypeChanged)));
 
        private static void OnBackdropTypeChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            if (d is not Window window)
            {
                return;
            }
 
            if (e.OldValue == e.NewValue)
            {
                return;
            }
 
            WindowBackdropManager.SetBackdrop(window, (WindowBackdropType)e.NewValue);
        }
 
 
        internal bool HwndCreatedButNotShown
        {
            get
            {
                return _hwndCreatedButNotShown;
            }
        }
 
        internal bool IsDisposed
        {
            get
            {
                return _disposed;
            }
        }
 
        /// <summary>
        /// This tells whether user has set Visible or not. It's currently used in Application
        /// where if Visibility has not been set when the Window is navigated to, we set it to
        /// Visible
        /// </summary>
        internal bool IsVisibilitySet
        {
            get
            {
                VerifyContextAndObjectState();
                return _isVisibilitySet;
            }
        }
 
        /// <summary>
        ///     Exposes the hwnd of the window. This property is used by the WindowInteropHandler
        ///     class
        /// </summary>
        internal IntPtr Handle
        {
            get
            {
                VerifyContextAndObjectState();
                if (_swh != null)
                {
                    return _swh.Handle;
                }
                else
                return IntPtr.Zero;
            }
        }
 
        /// <summary>
        ///     Enables to get/set the owner handle for this window. This property is used by
        ///     the WindowInteropHelper class
        /// </summary>
        ///<remarks>
        ///     This API is currently not available for use in Internet Zone.
        ///</remarks>
        internal IntPtr OwnerHandle
        {
            get
            {
 
                VerifyContextAndObjectState();
                return _ownerHandle;
            }
            set
            {
 
                VerifyContextAndObjectState();
 
                if ( _showingAsDialog == true )
                {
                    throw new InvalidOperationException(SR.CantSetOwnerAfterDialogIsShown);
                }
 
                SetOwnerHandle(value);
            }
        }
 
        /// <summary>
        ///     This is used by RBW to set the correct win32 style
        /// </summary>
        internal int Win32Style
        {
            get
            {
                VerifyContextAndObjectState();
                return _Style;
            }
            set
            {
                VerifyContextAndObjectState();
 
                // The reason this is not wrapped by a manager is that it should never
                // be invoked outside the scope on an already established manager.
                Debug.Assert(Manager != null, "HwndStyleManager must have a valid value here");
                _Style = value;
            }
        }
 
        internal int _Style
        {
            get
            {
                if (Manager != null)
                {
                    return _styleDoNotUse;
}
                else if ( IsSourceWindowNull )
                {
                    return _styleDoNotUse;
}
                else
                {
                    return _swh.StyleFromHwnd;
                }
            }
            set
            {
                _styleDoNotUse = value;
                Manager.Dirty = true;
            }
        }
 
        internal int _StyleEx
        {
            get
            {
                if (Manager != null)
                {
                    return _styleExDoNotUse;
}
                else if (IsSourceWindowNull == true  )
                {
                    return _styleExDoNotUse;
}
                else
                {
                    return _swh.StyleExFromHwnd;
                }
                }
            set
            {
                _styleExDoNotUse = value;
                Manager.Dirty = true;
            }
        }
 
        internal HwndStyleManager Manager
        {
            get { return _manager; }
            set { _manager = value; }
        }
 
        bool IWindowService.UserResized
        {
            get { return false; }
        }
 
        internal bool AreResourcesInitialized
        {
            get
            {
                return _resourcesInitialized;
            }
            set
            {
                _resourcesInitialized = value;
            }
        }
 
        #endregion Internal Properties
 
        //----------------------------------------------
        //
        // Internal Fields
        //
        //----------------------------------------------
        #region Internal Fields
 
        /// <summary>
        /// DialogCancel Command. It closes window if it's dialog and return false as the dialog value.
        /// </summary>
        /// <remarks>
        /// Right now this is only used by Cancel Button to close the dialog.
        /// </remarks>
        internal static readonly RoutedCommand DialogCancelCommand = new RoutedCommand("DialogCancel", typeof(Window));
 
        #endregion Internal Fields
 
 
        //----------------------------------------------
        //
        // Internal Types
        //
        //----------------------------------------------
        #region Internal Types
        // similar to the one in FE except that it takes care of SizeToContent
        // while determining the min/max values for height and width.
        internal struct WindowMinMax
        {
            internal double minWidth;
            internal double maxWidth;
            internal double minHeight;
            internal double maxHeight;
 
            internal WindowMinMax(double minSize, double maxSize)
            {
                minWidth = minSize;
                maxWidth = maxSize;
                minHeight = minSize;
                maxHeight = maxSize;
            }
        }
        #endregion Internal Types
 
        //----------------------------------------------
        //
        // Private Methods
        //
        //----------------------------------------------
        #region Private Methods
        private Size MeasureOverrideHelper(Size constraint)
        {
            // need to handle infinity
            // return the entire client area of the window
            // Measure children properly.
 
            // Three primary cases
            //      1)  hwnd does not exist  -- return 0,0
            //      1a) CompositionTarget is invalid -- return 0,0
            //      2)  Child visual exists  -- we return child.DesiredSize + frame size
            //      3)  No Child visual      -- return the hwnd size (this should be the same
            //                                  as the one passed into MeasureOverride for our framework)
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                // No need to use CompositionTarget.TransformFromDevice
                // since size is 0,0
                return new Size(0,0);
            }
 
            if (this.VisualChildrenCount > 0)
            {
                UIElement child = this.GetVisualChild(0) as UIElement;
                if (child != null) // UIElement children
                {
                    // Find out the size of the window frame x.
                    // (constraint - x) is the size we pass onto
                    // our child
 
                    Size frameSize = GetHwndNonClientAreaSizeInMeasureUnits();
 
                    // In some instances (constraint size - frame size) can be negative. One instance
                    // is when window is set to minimized before layout has happened.  Apparently, Win32
                    // gives a rect(-32000, -32000, -31840, -31975) for GetWindowRect when hwnd is
                    // minimized.  However, when we calculate the frame size we get width = 8 and
                    // height = 28!!!  Here, we will take the max of zero and the difference b/w the
                    // hwnd size and the frame size
                    //
                    // PS Windows OS Bug: 955861
                    Size childConstraint = new Size
                    {
                        Width = ((constraint.Width == Double.PositiveInfinity) ? Double.PositiveInfinity : Math.Max(0.0, (constraint.Width - frameSize.Width))),
                        Height = ((constraint.Height == Double.PositiveInfinity) ? Double.PositiveInfinity : Math.Max(0.0, (constraint.Height - frameSize.Height)))
                    };
 
                    child.Measure(childConstraint);
                    Size childDesiredSize = child.DesiredSize;
                    return new Size(childDesiredSize.Width + frameSize.Width, childDesiredSize.Height + frameSize.Height);
                }
            }
 
            // if we reach here, we return the input size
            return _swh.GetSizeFromHwndInMeasureUnits();
        }
 
        // Similar logic as in FE.MinMax and takes care of max/min size allowed by win32 for the hwnd.  However, we
        // don't take Height/Width into consideration.  This is because of the following reasons:
        //
        // 1) Window Height/Width info doesn't matter here since we just need to ensure that the child element is
        //    layed out within the Max/Min restrictions of the window.  Window is layed out at the size sent into MO/AO
        //    and Height and Width of the window does not play a part MO/AO stage.
        //
        // 2) We had (FlowDocumentReader: When the maximise button is clicked on the window with FDR the
        //    content don't reflow to fill in the entire window) and the fix for that is to simply not use H/W here.
        //    GetWindowMinMax fixes the following bugs:
        //
        //    (Window content should respect Window's Max/Min size)
        //    (Wrong window actual size returned if Autosize window content is smaller than the actual
        //    window (seems to return content size as opposed to window size))
        //
        //  This method is called by both MeasureOverride( ) and WmGetMinMaxInfo( ).
        //  It will calculate a final Min/Max size in logic units for this HWND based on Win32 restricted value and
        //  current Min/Max setting in this instance.
        //
        internal virtual WindowMinMax GetWindowMinMax()
        {
            WindowMinMax mm = new WindowMinMax( );
 
            Invariant.Assert(IsCompositionTargetInvalid == false, "IsCompositionTargetInvalid is supposed to be false here");
 
            // convert the max/min size (taken in to account the hwnd size restrictions by win32) into logical units
            double maxWidthDeviceUnits = _trackMaxWidthDeviceUnits;
            double maxHeightDeviceUnits = _trackMaxHeightDeviceUnits;
            if (WindowState == WindowState.Maximized)
            {
                // On some systems, the trackMax size is a few pixels smaller than
                // the windowMax size.   Use the larger size for maximized windows.
                maxWidthDeviceUnits = Math.Max(_trackMaxWidthDeviceUnits, _windowMaxWidthDeviceUnits);
                maxHeightDeviceUnits = Math.Max(_trackMaxHeightDeviceUnits, _windowMaxHeightDeviceUnits);
            }
 
            Point maxSizeLogicalUnits = DeviceToLogicalUnits(new Point(maxWidthDeviceUnits, maxHeightDeviceUnits));
            Point minSizeLogicalUnits = DeviceToLogicalUnits(new Point(_trackMinWidthDeviceUnits, _trackMinHeightDeviceUnits));
 
            //
            // Get the final Min/Max Width
            //
            mm.minWidth = Math.Max(this.MinWidth, minSizeLogicalUnits.X);
 
            // Min's precedence is higher than Max; If Min is greater than Max, use Min.
            if (MinWidth > MaxWidth)
            {
                mm.maxWidth = Math.Min(MinWidth, maxSizeLogicalUnits.X);
            }
            else
            {
                if (!Double.IsPositiveInfinity(MaxWidth))
                {
                    mm.maxWidth = Math.Min(MaxWidth, maxSizeLogicalUnits.X);
                }
                else
                {
                    mm.maxWidth = maxSizeLogicalUnits.X;
                }
            }
 
            //
            // Get the final Min/Max Height
            //
            mm.minHeight = Math.Max(this.MinHeight, minSizeLogicalUnits.Y);
 
            // Min's precedence is higher than Max; If Min is greater than Max, use Min.
            if (MinHeight > MaxHeight)
            {
                mm.maxHeight = Math.Min(this.MinHeight, maxSizeLogicalUnits.Y);
            }
            else
            {
                if (!Double.IsPositiveInfinity(MaxHeight))
                {
                    mm.maxHeight = Math.Min(MaxHeight, maxSizeLogicalUnits.Y);
                }
                else
                {
                    mm.maxHeight = maxSizeLogicalUnits.Y;
                }
            }
 
            return mm;
        }
 
        private void LoadedHandler(object sender, RoutedEventArgs e)
        {
            if (_postContentRenderedFromLoadedHandler == true)
            {
                PostContentRendered();
                _postContentRenderedFromLoadedHandler = false;
                this.Loaded -= new RoutedEventHandler(LoadedHandler);
            }
        }
 
        /// <remarks> Keep this method in sync with Frame.PostContentRendered(). </remarks>
        private void PostContentRendered()
        {
            // Post the firing of ContentRendered as Input priority work item so
            // that ContentRendered will be fired after render query empties.
            if (_contentRenderedCallback != null)
            {
                // Content was changed again before the previous rendering completed (or at least
                // before the Dispatcher got to Input priority callbacks).
                _contentRenderedCallback.Abort();
            }
            _contentRenderedCallback = Dispatcher.BeginInvoke(DispatcherPriority.Input,
                                   (DispatcherOperationCallback) delegate (object arg)
                                   {
                                       // Event handler exception continuality: there are no state related/depending on ContentRendered event.
                                       // If an exception occurs in event handler, our state will not be corrupted.
                                       // Please check event handler exception continuality if the logic changes.
                                       Window thisRef = (Window)arg;
                                       thisRef._contentRenderedCallback = null;
                                       thisRef.OnContentRendered(EventArgs.Empty);
                                       return null;
                                   },
                                   this);
        }
 
        /// <summary>
        /// Ensure Dialog command is registered with the CommandManager
        /// </summary>
        private void EnsureDialogCommand()
        {
            // _dialogCommandAdded is a static variable, however, we're not synchronizing
            // access to it.  The reason is that, CommandManager is thread safe and according
            // to KiranKu we don't want to take the overhead of locking here.  For multiple
            // threaded cases, we could end up calling the CommandManager more than once but
            // KiranKu is okay with that perf hit in the corner case.
            if (!_dialogCommandAdded)
            {
                // Right now we only have DialogCancel Command, which closes window if it's dialog and return false as the dialog's result.
                CommandBinding binding = new CommandBinding(DialogCancelCommand);
                binding.Executed += new ExecutedRoutedEventHandler(OnDialogCommand);
                CommandManager.RegisterClassCommandBinding(typeof(Window), binding);
 
                _dialogCommandAdded = true;
            }
        }
 
        /// <summary>
        /// Dialog Command Execute handler
        /// Right now we only have DialogCancel Command, which closes window if it's dialog and return false for DialogResult.
        /// </summary>
        /// <param name="target"></param>
        /// <param name="e"></param>
        private static void OnDialogCommand(object target, ExecutedRoutedEventArgs e)
        {
            //close dialog & return result
            Window w = target as Window;
 
            Debug.Assert(w != null, "Target must be of type Window.");
            w.OnDialogCancelCommand();
        }
 
        /// <summary>
        /// Close window if it's dialog and return false for DialogResult.
        /// </summary>
        private void OnDialogCancelCommand()
        {
            if (_showingAsDialog)
            {
                DialogResult = false;
            }
        }
 
        /// <summary>
        /// The callback function for EnumThreadWindows
        /// </summary>
        /// <param name="hWnd"></param>
        /// <param name="lparam"></param>
        /// <returns></returns>
        private bool ThreadWindowsCallback(IntPtr hWnd, IntPtr lparam)
        {
            Debug.Assert(_threadWindowHandles != null, "_threadWindowHandles must not be null at this point");
 
            // the dialog's hwnd has not been created yet when calling into this function.
            // so its hwnd won't be in the list.
 
            // We only do visible && enabled windows.
            // We need to check Visible because there might be hidden windows that do message looping,
            // We don't want to disable them.
            if (SafeNativeMethods.IsWindowVisible(new HandleRef(null, hWnd)) &&
                SafeNativeMethods.IsWindowEnabled(new HandleRef(null, hWnd)))
            {
                _threadWindowHandles.Add(hWnd);
            }
 
            return true;
        }
 
        /// <summary>
        /// Enables/disables all Windows on this thread
        /// </summary>
        /// <param name="state"></param>
        private void EnableThreadWindows(bool state)
        {
            Debug.Assert(_threadWindowHandles != null, "_threadWindowHandles must not be null at this point");
 
            for (int i = 0; i < _threadWindowHandles.Count; i++)
            {
                IntPtr hWnd = _threadWindowHandles[i];
 
                if (UnsafeNativeMethods.IsWindow(new HandleRef(null, hWnd)))
                {
                    // Calls EnableWindow which returns the previous Window state
                    // (enable/disable) and we don't care about that here
                    UnsafeNativeMethods.EnableWindowNoThrow(new HandleRef(null, hWnd), state);
                }
            }
 
            // EnableThreadWindows is called with true only when dialog is going away.  Now
            // we've enabled the windows that we had earlier disabled; thus, disposing
            // _threadWindowHandles.
            if (state == true)
            {
                _threadWindowHandles = null;
            }
        }
 
        /// <summary>
        /// Intialization when Window is constructed
        /// </summary>
        ///     Initializes the Width/Height, Top/Left properties to use windows
        ///     default. Updates Application object properties if inside app.
        ///
        ///     Also, window style is set to WS_CHILD inside CreateSourceWindow
        ///     for browser hosted case
        private void Initialize()
        {
 
            //  this makes MeasureCore / ArrangeCore to defer to direct MeasureOverride and ArrangeOverride calls
            //  without reading Width / Height properties and modifying input constraint size parameter...
            BypassLayoutPolicies = true;
 
            // check if within an app && on the same thread
            if (IsInsideApp == true)
            {
                if (Application.Current.Dispatcher.Thread == Dispatcher.CurrentDispatcher.Thread)
                {
                    // add to window collection
                    // use internal version since we want to update the underlying collection
                    App.WindowsInternal.Add(this);
                    if (App.MainWindow == null)
                    {
                        App.MainWindow = this;
                    }
                }
                else
                {
                    App.NonAppWindowsInternal.Add(this);
                }
            }
        }
 
 
        internal void VerifyContextAndObjectState()
        {
            // Verify that we are executing on the right context
            VerifyAccess();
 
        // WCP Window:  Define the dispose behavior for Window
#if DISPOSE
            if (_disposed)
            {
                throw new ObjectDisposedException(null, SR.WindowDisposed);
            }
#endif
        }
 
        private void VerifyCanShow()
        {
            if (_disposed == true)
            {
                throw new InvalidOperationException(SR.ReshowNotAllowed);
            }
        }
 
        private void VerifyNotClosing()
        {
            if (_isClosing == true)
            {
                throw new InvalidOperationException(SR.InvalidOperationDuringClosing);
            }
 
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == true)
            {
                throw new InvalidOperationException(SR.InvalidCompositionTarget);
            }
        }
 
        private void VerifyHwndCreateShowState()
        {
            if (HwndCreatedButNotShown)
            {
                throw new InvalidOperationException(SR.NotAllowedBeforeShow);
            }
        }
 
        /// <summary>
        ///     sets the IWindowService attached property
        /// </summary>
        private void SetIWindowService()
        {
            if (GetValue(IWindowServiceProperty) == null)
            {
                SetValue(IWindowServiceProperty, (IWindowService)this);
            }
        }
 
        IntPtr GetCurrentMonitorFromMousePosition()
        {
            // center on the screen on which the mouse is on
            NativeMethods.POINT pt = default;
 
            UnsafeNativeMethods.TryGetCursorPos(ref pt);
 
            return SafeNativeMethods.MonitorFromPoint(pt, NativeMethods.MONITOR_DEFAULTTONEAREST);
        }
 
        // <summary>
        //     Calculates the window location based on the WindowStartupLocation property
        //     If values for CenterScreen, CenterOwner cannot be determined, then we return
        //     unmodified values.
        // </summary>
        // <remarks>
        //     This method can be called before or after the hwnd is created.  So, we have
        //     to accout for both scenarios.
        // </remarks>
        // <param name="left"></param>
        // <param name="top"></param>
        // <param name="currentSize"></param>
        private bool CalculateWindowLocation(ref double leftDeviceUnits, ref double topDeviceUnits, Size currentSizeDeviceUnits)
        {
            Debug.Assert(IsSourceWindowNull == false, "_swh should not be null here");
            double inLeft = leftDeviceUnits;
            double inTop = topDeviceUnits;
 
            switch (_windowStartupLocation)
            {
                case WindowStartupLocation.Manual:
                    break;
                case WindowStartupLocation.CenterScreen:
                    // NOTE:
                    // Which screen to center the
                    // window on?
                    //
                    // If Window has a parent handle, then center the Window on the
                    // same monitor as the parent hwnd.  If theres no parent hwnd,
                    // center the Window on the monitor where the mouse is currently
                    // on.
                    //
                    // The exception to this rule is when ShowInTaskbar is set to false
                    // and we parent to window to achieve that.  That's the reason for
                    // having the extra condition in the if statement below
                    //
 
                    IntPtr hMonitor = IntPtr.Zero;
                    if ((_ownerHandle == IntPtr.Zero) ||
                        ((_hiddenWindow != null) && (_hiddenWindow.Handle == _ownerHandle)))
                    {
                        hMonitor = GetCurrentMonitorFromMousePosition();
                    }
                    else
                    {
                        // have a parent hwnd; center on the screen on
                        // which our parent hwnd is.
                        hMonitor = MonitorFromWindow(_ownerHandle);
                    }
 
                    if (hMonitor != IntPtr.Zero)
                    {
                        CalculateCenterScreenPosition(hMonitor, currentSizeDeviceUnits, ref leftDeviceUnits, ref topDeviceUnits);
                    }
                    break;
                case WindowStartupLocation.CenterOwner:
                    Rect ownerRectDeviceUnits = Rect.Empty;
 
                    // If the owner is WPF window.
                    // The owner can be non-WPF window. It can be set via WindowInteropHelper.Owner.
                    if (CanCenterOverWPFOwner == true)
                    {
                        // If the owner is in a non-normal state use the screen bounds for centering the window.
                        // Top/Left/Width/Height reflect the restore bounds, so they can't be used in this scenario.
                        if (Owner.WindowState == WindowState.Maximized || Owner.WindowState == WindowState.Minimized)
                        {
                            goto case WindowStartupLocation.CenterScreen;
                        }
 
                        Point ownerSizeDeviceUnits;
                        // if owner hwnd is created, we use WindowSize to get its size. Note: we cannot use ActualWidth/Height here,
                        // because it is possible that the hwnd is created (WIH.EnsureHandle) but it is not shown yet; layout has not
                        // happen; ActualWidth/Height is not calculated yet.
                        // If the owner hwnd is not yet created, we use Owner.Width/Height.
                        if (Owner.Handle == IntPtr.Zero)
                        {
                            ownerSizeDeviceUnits = Owner.LogicalToDeviceUnits(new Point(Owner.Width, Owner.Height));
                        }
                        else
                        {
                            Size size = Owner.WindowSize;
                            ownerSizeDeviceUnits = new Point(size.Width, size.Height);
                        }
 
                        // A minimized window doesn't have valid Top,Left; that's why RestoreBounds.TopLeft is used.
                        Point ownerLocationDeviceUnits = Owner.LogicalToDeviceUnits(new Point(Owner.Left, Owner.Top));
                        ownerRectDeviceUnits = new Rect(ownerLocationDeviceUnits.X, ownerLocationDeviceUnits.Y,
                            ownerSizeDeviceUnits.X, ownerSizeDeviceUnits.Y);
                    }
                    else
                    {
                        // non-WPF owner
                        if ((_ownerHandle != IntPtr.Zero) && UnsafeNativeMethods.IsWindow(new HandleRef(null, _ownerHandle)))
                        {
                            ownerRectDeviceUnits = GetNormalRectDeviceUnits(_ownerHandle);
                        }
                    }
 
                    if (! ownerRectDeviceUnits.IsEmpty)
                    {
                        leftDeviceUnits = ownerRectDeviceUnits.X + ((ownerRectDeviceUnits.Width - currentSizeDeviceUnits.Width) / 2);
                        topDeviceUnits = ownerRectDeviceUnits.Y + ((ownerRectDeviceUnits.Height - currentSizeDeviceUnits.Height) / 2);
 
                        // (Window.CenterOwner doesn't make sure the window fits entirely on screen)
                        // Check the screen rect to make sure the window is shown on screen.
                        // It is the same as WinForms' behavior.
                        NativeMethods.RECT workAreaRectDeviceUnits = WorkAreaBoundsForHwnd(_ownerHandle);
                        leftDeviceUnits = Math.Min(leftDeviceUnits, workAreaRectDeviceUnits.right - currentSizeDeviceUnits.Width);
                        leftDeviceUnits = Math.Max(leftDeviceUnits, workAreaRectDeviceUnits.left);
                        topDeviceUnits = Math.Min(topDeviceUnits, workAreaRectDeviceUnits.bottom - currentSizeDeviceUnits.Height);
                        topDeviceUnits = Math.Max(topDeviceUnits, workAreaRectDeviceUnits.top);
                    }
 
                    break;
                default:
                    break;
            }
            return (!DoubleUtil.AreClose(inLeft, leftDeviceUnits) || !DoubleUtil.AreClose(inTop, topDeviceUnits));
        }
 
        private static NativeMethods.RECT WorkAreaBoundsForHwnd(IntPtr hwnd)
        {
            IntPtr hMonitor = MonitorFromWindow(hwnd);
 
            return WorkAreaBoundsForMointor(hMonitor);
        }
 
        private static NativeMethods.RECT WorkAreaBoundsForMointor(IntPtr hMonitor)
        {
            NativeMethods.MONITORINFOEX monitorInfo = new NativeMethods.MONITORINFOEX();
 
            Debug.Assert(hMonitor != IntPtr.Zero);
            SafeNativeMethods.GetMonitorInfo(new HandleRef(null, hMonitor), monitorInfo);
 
            return monitorInfo.rcWork;
        }
 
        private static IntPtr MonitorFromWindow(IntPtr hwnd)
        {
            IntPtr hMonitor = SafeNativeMethods.MonitorFromWindow(new HandleRef(null, hwnd), NativeMethods.MONITOR_DEFAULTTONEAREST);
            if (hMonitor == IntPtr.Zero)
            {
                throw new Win32Exception();
            }
            return hMonitor;
        }
 
        /// <summary>
        /// Calculates the left and right coordinates of a window
        /// when centered on a given monitor.
        /// </summary>
        /// <param name="hMonitor">Handle to the monitor to center the window on</param>
        /// <param name="currentSizeDeviceUnits">Size of the window, in device units</param>
        /// <param name="leftDeviceUnits">Receives the new left location in device units</param>
        /// <param name="topDeviceUnits">Receives the new top location in device units</param>
        internal static void CalculateCenterScreenPosition(IntPtr hMonitor, Size currentSizeDeviceUnits, ref double leftDeviceUnits, ref double topDeviceUnits)
        {
            NativeMethods.RECT workAreaRectDeviceUnits = WorkAreaBoundsForMointor(hMonitor);
 
            // We're using Width/Height here as opposed to ActualWidth/Height
            // as layout hasn't happened yet.
            double workAreaWidthDeviceUnits = workAreaRectDeviceUnits.right - workAreaRectDeviceUnits.left;
            double workAreaHeightDeviceUnits = workAreaRectDeviceUnits.bottom - workAreaRectDeviceUnits.top;
 
            Debug.Assert(workAreaWidthDeviceUnits >= 0, string.Create(CultureInfo.InvariantCulture, $"workAreaWidth ({hMonitor})for monitor ({workAreaWidthDeviceUnits}) is negative"));
            Debug.Assert(workAreaHeightDeviceUnits >= 0, string.Create(CultureInfo.InvariantCulture, $"workAreaHeight ({hMonitor}) for monitor ({workAreaHeightDeviceUnits}) is negative"));
 
            leftDeviceUnits = (workAreaRectDeviceUnits.left + ((workAreaWidthDeviceUnits - currentSizeDeviceUnits.Width) / 2));
            topDeviceUnits = (workAreaRectDeviceUnits.top + ((workAreaHeightDeviceUnits - currentSizeDeviceUnits.Height) / 2));
        }
 
        private bool CanCenterOverWPFOwner
        {
            get
            {
                Debug.Assert(IsSourceWindowNull == false, "_swh should not be null here");
 
                // if Owner is null, we cannot CenterOwner
                if (Owner == null)
                {
                    return false;
                }
 
                // if Owner._sourceWindow is null, and if Owner's Width or Height is not specified
                // then we cannot CenterOwner
                if (Owner.IsSourceWindowNull)
                {
                    if ((double.IsNaN(Owner.Width)) ||
                        (double.IsNaN(Owner.Height)))
                    {
                        return false;
                    }
                }
 
                // if Owner's Top or Left is not specified, we cannot CenterOwner
                if ((double.IsNaN(Owner.Left)) ||
                    (double.IsNaN(Owner.Top)))
                {
                    return false;
                }
                return true;
            }
        }
 
        private Rect GetNormalRectDeviceUnits(IntPtr hwndHandle)
        {
            int styleEx = UnsafeNativeMethods.GetWindowLong(new HandleRef(this, hwndHandle), NativeMethods.GWL_EXSTYLE);
 
            NativeMethods.WINDOWPLACEMENT wp = new NativeMethods.WINDOWPLACEMENT
            {
                length = Marshal.SizeOf(typeof(NativeMethods.WINDOWPLACEMENT))
            };
            UnsafeNativeMethods.GetWindowPlacement(new HandleRef(this, hwndHandle), ref wp);
            Point locationDeviceUnits = new Point(wp.rcNormalPosition_left, wp.rcNormalPosition_top);
 
            // GetWindowPlacement returns workarea co-ods for a top level window whose
            // WS_EX_TOOLWINDOW bit is clear.  If this bit is set, then the co-ods
            // returned are relative to the screen co-ods of the monitor.
            // TransfromWorkAreaScreenArea can transform a point from work area co-ods
            // to screen area co-ods and vice versa depending on TransformType value passed.
            // So, in our case, if the window is not a ToolWindow we want to transform
            // the point from work area co-ods to screen co-ods.
            if ((styleEx & NativeMethods.WS_EX_TOOLWINDOW) == 0)
            {
                locationDeviceUnits = TransformWorkAreaScreenArea(locationDeviceUnits, TransformType.WorkAreaToScreenArea);
            }
 
            Point sizeDeviceUnits = new Point(wp.rcNormalPosition_right - wp.rcNormalPosition_left,
                                              wp.rcNormalPosition_bottom - wp.rcNormalPosition_top);
 
            return new Rect(locationDeviceUnits.X, locationDeviceUnits.Y, sizeDeviceUnits.X, sizeDeviceUnits.Y);
        }
 
        private Rect GetNormalRectLogicalUnits(IntPtr hwndHandle)
        {
            Rect rectDeviceUnits = GetNormalRectDeviceUnits(hwndHandle);
 
            Point sizeLogicalUnits = DeviceToLogicalUnits(new Point(rectDeviceUnits.Width, rectDeviceUnits.Height));
            Point locationLogicalUnits = DeviceToLogicalUnits(new Point(rectDeviceUnits.X, rectDeviceUnits.Y));
 
            return new Rect(locationLogicalUnits.X, locationLogicalUnits.Y, sizeLogicalUnits.X, sizeLogicalUnits.Y);
        }
 
        /// <summary>
        ///     Update style bits for window state
        /// </summary>
        private void CreateWindowState()
        {
            switch (WindowState)
            {
                case WindowState.Normal:
                    break;
                case WindowState.Maximized:
                    _Style |= NativeMethods.WS_MAXIMIZE;
                    break;
                case WindowState.Minimized:
                    _Style |= NativeMethods.WS_MINIMIZE;
                    break;
#if THEATRE_FULLSCREEN
                case WindowState.Theatre:
                    throw new NotImplementedException(SR.NotImplementedException);
 
                case WindowState.FullScreen:
                    throw new NotImplementedException(SR.NotImplementedException);
#endif //THEATRE_FULLSCREEN
            }
        }
 
        /// <summary>
        ///     creates topmost window
        /// </summary>
        private void CreateTopmost()
        {
            // check for topmost
            if ( Topmost )
            {
                _StyleEx |= NativeMethods.WS_EX_TOPMOST;
            }
            else
            {
                _StyleEx &= ~NativeMethods.WS_EX_TOPMOST;
            }
        }
 
        /// <summary>
        ///     set's resizibility for the window
        /// </summary>
        private void CreateResizibility()
        {
            _Style &= ~(NativeMethods.WS_THICKFRAME | NativeMethods.WS_MAXIMIZEBOX | NativeMethods.WS_MINIMIZEBOX);
 
 
            switch(ResizeMode)
            {
                case ResizeMode.NoResize:
                    break;
                case ResizeMode.CanMinimize:
                    _Style |= NativeMethods.WS_MINIMIZEBOX;
                    break;
                case ResizeMode.CanResize:
                case ResizeMode.CanResizeWithGrip:
                    _Style |= NativeMethods.WS_THICKFRAME | NativeMethods.WS_MAXIMIZEBOX | NativeMethods.WS_MINIMIZEBOX;
                    break;
                default:
                    Debug.Assert(false, "Invalid value for ResizeMode");
                    break;
            }
        }
 
        /// <summary>
        ///     Updates the window icon
        /// </summary>
        private void UpdateIcon()
        {
            // NOTE: Set Window.Icon = null causes NullReferenceException
 
            // if _icon is null, set _defaultLargeIconHandle and _defaultSmallIconHandle
            //  to the app icon (embedded in the exe).  _icon is used as window icon if it
            // is not null, else default icons from exe are used.  If both are null,
            // we set IntPtr.Zero as the icons for the window and Win32 defaults
            // come into play.
            //
 
            NativeMethods.IconHandle largeIconHandle;
            NativeMethods.IconHandle smallIconHandle;
 
            if (_icon != null)
            {
                IconHelper.GetIconHandlesFromImageSource(_icon, out largeIconHandle, out smallIconHandle);
            }
            else
            {
                // these both should be null before we've queried that exe for icons.
                // Once, we looked in the exe, these are no longer null and hence we
                // don't want to re-query in the exe anymore
                if (_defaultLargeIconHandle == null && _defaultSmallIconHandle == null)
                {
                     // sets the default small and large icon handles
                     IconHelper.GetDefaultIconHandles(out largeIconHandle, out smallIconHandle);
                    _defaultLargeIconHandle = largeIconHandle;
                    _defaultSmallIconHandle = smallIconHandle;
                }
                else
                {
                    largeIconHandle = _defaultLargeIconHandle;
                    smallIconHandle = _defaultSmallIconHandle;
                }
                // get default icons
            }
 
            // One of the steps necessary to hide a Window's taskbar button is to parent it off another HWND.
            // On XP when this is done the Window's alt-tab icon takes on the icon of the parent.  If we've created
            // a hidden parent window for the sake of ShowInTaskbar=false, then we need to keep its icon in sync
            // with the Window's.
            // On Vista this isn't necessary.
 
            // Make the array big enough to hold anything we need to update.
            // We'll keep track of the true count separately.
            var iconWindows = new HandleRef[]
            {
                new HandleRef(this, Handle),
                default(HandleRef)
            };
            int iconWindowsCount = 1;
 
            if (_hiddenWindow != null)
            {
                iconWindows[1] = new HandleRef(_hiddenWindow, _hiddenWindow.Handle);
                ++iconWindowsCount;
            }
 
            for (int i = 0; i < iconWindowsCount; ++i)
            {
                HandleRef hwnd = iconWindows[i];
 
                UnsafeNativeMethods.SendMessage(
                                        hwnd,
                                        WindowMessage.WM_SETICON,
                                        (IntPtr)NativeMethods.ICON_BIG,
                                        largeIconHandle);
 
                UnsafeNativeMethods.SendMessage(
                                        hwnd,
                                        WindowMessage.WM_SETICON,
                                        (IntPtr)NativeMethods.ICON_SMALL,
                                        smallIconHandle);
            }
 
            // dispose the previous icon handle if it's not the default handle
            if (_currentLargeIconHandle != null && _currentLargeIconHandle != _defaultLargeIconHandle)
            {
                _currentLargeIconHandle.Dispose();
            }
 
            if (_currentSmallIconHandle != null && _currentSmallIconHandle != _defaultSmallIconHandle)
            {
                _currentSmallIconHandle.Dispose();
            }
 
            _currentLargeIconHandle = largeIconHandle;
            _currentSmallIconHandle = smallIconHandle;
        }
 
        /// <summary>
        ///     Sets the parent hwnd (Owner) for this window. This could be called as a result of
        ///     setting one of the following
        ///
        ///         Window Owner {set;}
        ///         IntPtr OwnerHandle {set;}  -- this is used by WindowInteropHelper
        /// </summary>
        /// <param name="ownerHandle">IntPtr of the parent window</param>
        private void SetOwnerHandle(IntPtr ownerHandle)
        {
            // Note:
            // "SetWindowLong failed.  Error = 1400" appears in console when setting
            // Window.Owner to a Window hasn't been shown (chk build)
            //
            // SetWindowLong with GWL_HWNDPARENT fails if the new parent/owner is IntPtr.Zero
            // and the old owner was also IntPtr.Zero.  The if check below works around this
            // issue.
            if (_ownerHandle == ownerHandle && _ownerHandle == IntPtr.Zero)
            {
                return;
            }
 
            // If this call is removing the owner then we possibly need to reparent it with
            // the hidden window (ShowInTaskbar==false)
            // Once the hidden window is created we keep its icon in sync with the Window's.
            _ownerHandle = (IntPtr.Zero == ownerHandle && !ShowInTaskbar)
                ? EnsureHiddenWindow().Handle
                : ownerHandle;
 
            if (IsSourceWindowNull == false)
            {
                UnsafeNativeMethods.SetWindowLong(new HandleRef(null, Handle),
                    NativeMethods.GWL_HWNDPARENT,
                    _ownerHandle);
 
                // Update and reset the Owner Window if this is set through WindowInteropHelper.
                // We want to do this because once we are passed in the IntPtr for
                // the parent window, the Owner window is not the parent anymore.
 
                if ((_ownerWindow != null) && (_ownerWindow.Handle != _ownerHandle))
                {
                    _ownerWindow.OwnedWindowsInternal.Remove(this);
                    _ownerWindow = null;
                }
            }
        }
 
        /// <summary>
        ///     This is called back into when HwndSouce being used by this object is disposed, or if the
        ///     hwnd is destroyed.
        ///
        ///     When that happens on the same thread, we receive WM_CLOSE, WM_DESTROY and
        ///     everything works fine.  However, if that happened on another thread, we don't get
        ///     WM_CLOSE, WM_DESTROY until later, but we get this disposed callback and we
        ///     set dispose this object.  If we don't do this, there could be timing related issues if
        ///     we get called into after the HwndSource or hwnd has been disposed/destoyed but before
        ///     we receive WM_CLOSE, WM_DESTROY messages.
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        /// <returns></returns>
        private void OnSourceWindowDisposed(object sender, EventArgs e)
        {
            if ( _disposed == false)
            {
                InternalDispose();
            }
        }
 
        /// <summary>
        ///     This is the hook to HwndSource that is called when window messages related to
        ///     this window occur. Currently, we listen to the following messages
        ///
        ///         WM_CLOSE        : We listen to this message in order to fire the Closing event.
        ///                           If the user cancels window closing, we set handled to true so
        ///                           that the DefWindowProc does not handle this message. Otherwise,
        ///                           we set handled to false.
        ///         WM_DESTROY      : We listen to this message in order to fire the Closed event.
        ///                           Handled is always set to false.
        ///         WM_ACTIVATE     : Used for Activated and deactivated events
        ///         WM_SIZE         : Used for SizeChanged, StateChanged events. Also, helps us keep our
        ///                           size updated
        ///         WM_MOVE:        : Used for location changed event and to keep our cached top/left
        ///                           updated
        ///         WM_GETMINMAXINFO: Used to enforce Max/MinHeight and Max/MinWidth
        /// </summary>
        /// <param name="hwnd"></param>
        /// <param name="msg"></param>
        /// <param name="wParam"></param>
        /// <param name="lParam"></param>
        /// <param name="handled"></param>
        /// <returns></returns>
        private IntPtr WindowFilterMessage( IntPtr hwnd,
            int msg,
            IntPtr wParam,
            IntPtr lParam,
            ref bool handled)
        {
            IntPtr retInt = IntPtr.Zero ;
            WindowMessage message = (WindowMessage)msg;
 
            //
            // we need to process WM_GETMINMAXINFO before _swh is assigned to
            // b/c we want to store the max/min size allowed by win32 for the hwnd
            // which is later used in GetWindowMinMax.  WmGetMinMaxInfo can handle
            // _swh == null case.
            switch (message)
            {
                case WindowMessage.WM_GETMINMAXINFO:
                    handled = WmGetMinMaxInfo(lParam);
                    break;
                case WindowMessage.WM_SIZE:
                    handled = WmSizeChanged(wParam);
                    break;
            }
 
            if(_swh != null && _swh.CompositionTarget != null) // For extraneous messages during shutdown
            {
                // Can't case this in the switch statement since it's dynamically generated.
                if (message == WM_TASKBARBUTTONCREATED || message == WM_APPLYTASKBARITEMINFO)
                {
                    // Either Explorer's created a new button or it's time to try again.
                    // Stop deferring updates to the Taskbar.
                    if (_taskbarRetryTimer != null)
                    {
                        _taskbarRetryTimer.Stop();
                    }
 
                    // We'll receive WM_TASKBARBUTTONCREATED at times other than when the Window was created,
                    //    e.g. Explorer restarting, in response to ShowInTaskbar=true, etc.
                    // We'll receive WM_APPLYTASKBARITEMINFO when we'ved posted it to ourself after a failed ITaskbarList3 call.
                    ApplyTaskbarItemInfo();
                }
                else switch (message)
                {
                    case WindowMessage.WM_CLOSE:
                        handled = WmClose();
                        break;
                    case WindowMessage.WM_DESTROY:
                        handled = WmDestroy();
                        break;
                    case WindowMessage.WM_ACTIVATE:
                        handled = WmActivate(wParam);
                        break;
                    case WindowMessage.WM_MOVE:
                        handled = WmMoveChanged();
                        break;
                    case WindowMessage.WM_NCHITTEST:
                        handled = WmNcHitTest(lParam, ref retInt);
                        break;
                    case WindowMessage.WM_SHOWWINDOW:
                        handled = WmShowWindow(wParam, lParam);
                        break;
                    case WindowMessage.WM_COMMAND:
                        handled = WmCommand(wParam, lParam);
                        break;
                    default:
                        handled = false;
                        break;
                }
            }
 
            return retInt;
        }
 
        /// <summary>
        ///     Called on WM_COMMAND message.
        /// </summary>
        /// <returns>
        ///     True if we want to handle the command, false otherwise.
        /// </returns>
        private bool WmCommand(IntPtr wParam, IntPtr lParam)
        {
            if (NativeMethods.SignedHIWORD(wParam.ToInt32()) == THUMBBUTTON.THBN_CLICKED)
            {
                TaskbarItemInfo taskbar = TaskbarItemInfo;
                if (taskbar != null)
                {
                    int index = NativeMethods.SignedLOWORD(wParam.ToInt32());
                    if (index >= 0 && index < taskbar.ThumbButtonInfos.Count)
                    {
                        taskbar.ThumbButtonInfos[index].InvokeClick();
                    }
                }
                return true;
            }
            return false;
        }
 
        /// <summary>
        ///     Called on WM_CLOSE message. Fires the Closing event.
        /// </summary>
        /// <returns>
        ///     True if we want to stop the window from closing, else false
        /// </returns>
        private bool WmClose()
        {
            // For WS_CHILD window, WM_SIZE, WM_MOVE (and maybe others) are called
            // synchronously from CreateWindowEx call and we run into issues if
            // _sourceWindow in null.  We only care to listen to WM_CREATE &
            // WM_GETMINMAXINFO synchronously from CreateWindowEx thus we want
            // to explicitly add the null check below at all other places.
            //
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return false;
            }
 
            // if DialogResult is set from within a Closing event then
            // the window is in the closing state.  In such a case, we
            // should not call Close() from DialogResult.set and thus
            // we have this variable.
            //
            // Note: Windows OS bug # 934500 Setting DialogResult
            // on the Closing EventHandler of a Dialog causes StackOverFlowException
            _isClosing = true;
 
            // Event handler exception continuality: if exception occurs in Closing event handler, the
            // cleanup action is to finish closing.
            CancelEventArgs e = new CancelEventArgs(false);
            try
            {
                OnClosing(e);
            }
            catch
            {
                CloseWindowFromWmClose();
                throw;
            }
 
            if (ShouldCloseWindow(e.Cancel))
            {
                CloseWindowFromWmClose();
                return false;
            }
            else
            {
                // close cancelled
                _isClosing = false;
 
                // Dialog does not close with ESC key after it has been cancelled
                //
                // Since closing is cancelled, DialogResult should be reset to null b/c
                // 1) DialogResult = true/false means that dialog has been accepeted/rejected (since dialog didn't
                //                   close (closing cancelled), DialogResult should be null)
                _dialogResult = null;
 
                return true;
            }
        }
 
        private void CloseWindowFromWmClose()
        {
            if (_showingAsDialog)
            {
                DoDialogHide();
            }
 
            // We should ClearRootVisual here instead of in InternalDispose. InternalDispose is called either as a result of HwndSource
            // disposing itself or it will try to dispose HwndSource. HwndSource will clear the root visual when it is disposed.
            ClearRootVisual();
 
            // We should also ClearHiddenWindow here, because in InternalDispose the window handle could be null if the dispose happens as
            // a result of HwndSource dispose. Our InternalDispose has been changed to handle reantrance, so the issue (bug 953988) described in
            // ClearHiddenWindowIfAny should not happen any more.
            ClearHiddenWindowIfAny();
        }
 
        private bool ShouldCloseWindow(bool cancelled)
        {
            // if shutdown Closing cannot be cancelled
            // if parent window is closing, child window Closing cannot be cancelled.
            return ((!cancelled) || (_appShuttingDown) || (_ignoreCancel));
        }
 
        private void DoDialogHide()
        {
 
            Debug.Assert(_showingAsDialog == true, "_showingAsDialog must be true when DoDialogHide is called");
 
            bool wasActive = false;
 
            //It's possible that _dispatcherFrame could be null at this time.
            //The scenario is: When showing the window as a modal dialog, the window Activated event is fired
            //before _dispatcherFrame is instantiated. In the Activated handler, if user closes the
            //window (setting DialogResult fires the WM_CLOSE event), the _dispatcherFrame is still null.
            //Bug 874463 addressed this.
            if (_dispatcherFrame != null)
            {
                // un block the push frame call
                _dispatcherFrame.Continue = false;
                _dispatcherFrame = null;
            }
 
            // Fix for Close Dialog Window should not return null
            //
            // The consensus here is that DialogResult should never be null when ShowDialog returns
            // As such, we coerce it to be false.  Furthermore, we don't use the DialogResult property
            // to update _dialogResult here since that does more than just updating the underlying
            // variable
            if (_dialogResult == null)
            {
                _dialogResult = false;
            }
 
            // clears _showingAsDialog
            _showingAsDialog = false;
 
            // enable previous window stuff goes here...
            wasActive = _swh.IsActiveWindow;
 
            // We assert here b/c I think _threadWindowHandles should never be null when we get
            // called in here.
            //
            // However, if inside ShowDialog we hit an exception after showing the dialog and the
            // exception handler is run, then _threadWindowHandles will be null here.
            //
            // Keeping this as assert.  If this turn out to be an over active assert, we'll switch to
            // an if condition.
            Debug.Assert(_threadWindowHandles != null, "_threadWindowHandles must not be null at this point");
            // reenable windows in the thread that were disabled
            EnableThreadWindows(true);
 
            // if dialog that is closing was active window and there was a previously active window,
            // set the active window.  The owner window may not be the previously active window. See DevDiv bug 122467 for details.
            // Furthermore, verify that _dialogPreviousActiveHandle is still a window b/c it
            // could have been destroyed by now by some other thread/codepath etc.
            // (BVT BLOCKER: System.ComponentModel.Win32Exception thrown when
            // trying to shutdown app inside a Dialog Window)
            if ((wasActive == true) &&
                (_dialogPreviousActiveHandle != IntPtr.Zero) &&
                (UnsafeNativeMethods.IsWindow(new HandleRef(this, _dialogPreviousActiveHandle)) == true))
            {
                UnsafeNativeMethods.SetActiveWindow(new HandleRef(this, _dialogPreviousActiveHandle));
            }
            else
            {
                // WCP: Fix code for a rare scenario when dialog is going away
 
                // rare situation, figure this out later
                // talk to user team as to what we need to do here
            }
        }
 
        private void UpdateWindowListsOnClose()
        {
            // Close all owned windows
            // use internal version since we want to update the underlying collection
            WindowCollection ownedWindows = OwnedWindowsInternal;
 
            // need to discuss what the correct behavior is if one of the owned window throws exception
            // when closing.
            // Although we explicitly do this here, we don't really need to. (see PS # 857285)
            // We use a while loop like this because closing an owned window will modify the owned windows list.
            while (ownedWindows.Count > 0)
            {
                // if parent window is closing, child window Closing cannot be cancelled.
                ownedWindows[0].InternalClose(false, true /* Ignore cancel */);
            }
 
            Debug.Assert(ownedWindows.Count == 0, "All owned windows should now be gone");
 
            // Update OwnerWindows of our Owner
            if (IsOwnerNull == false)
            {
                // use internal version since we want to update the underlying collection
                Owner.OwnedWindowsInternal.Remove(this);
            }
 
            if (this.IsInsideApp)
            {
                if (Application.Current.Dispatcher.Thread == Dispatcher.CurrentDispatcher.Thread)
                {
                    // use internal version since we want to update the underlying collection
                    App.WindowsInternal.Remove(this);
 
                    // Check to see if app should shut down--this behavior really belongs in Application
                    if (_appShuttingDown == false)
                    {
                        // If this is the last window that's closing and shutdownmode is onlastwindowclose, or
                        // if this is the main window closing and shutdownmode is onmainwindowclose, shutdown
                        // the app
                        if (((App.Windows.Count == 0) && (App.ShutdownMode == ShutdownMode.OnLastWindowClose))
                         || ((App.MainWindow == this) && (App.ShutdownMode == ShutdownMode.OnMainWindowClose)))
                        {
                            App.CriticalShutdown(0);
                        }
                    }
 
                    TryClearingMainWindow();
                }
                else
                {
                    App.NonAppWindowsInternal.Remove(this);
                }
            }
}
        private bool  WmDestroy()
        {
            // For WS_CHILD window, WM_SIZE, WM_MOVE (and maybe others) are called
            // synchronously from CreateWindowEx call and we run into issues if
            // _sourceWindow in null.  We only care to listen to WM_CREATE &
            // WM_GETMINMAXINFO synchronously from CreateWindowEx thus we want
            // to explicitly add the null check below at all other places.
            if (IsSourceWindowNull)
            {
                return false;
            }
 
            if (_disposed == false)
            {
                InternalDispose();
            }
 
            // STRESS: System.NullReferenceException @ System.Windows.Window.DeviceToLogicalUnits
            //
            // We're intentionally not adding a check for IsCompositionTargetInvlaid here since we
            // feel that it is okay to fire the Closed event to notify the developer that window has
            // closed.
 
            // Event handler exception continuality: if exception occurs in Closed event handler, the
            // cleanup action is to finish closing.
            // raise Closed event
            OnClosed(EventArgs.Empty);
 
            return false;
        }
 
        private bool WmActivate( IntPtr wParam )
        {
            // For WS_CHILD window, WM_SIZE, WM_MOVE (and maybe others) are called
            // synchronously from CreateWindowEx call and we run into issues if
            // _sourceWindow in null.  We only care to listen to WM_CREATE &
            // WM_GETMINMAXINFO synchronously from CreateWindowEx thus we want
            // to explicitly add the null check below at all other places.
            //
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return false;
            }
 
            int loWord = NativeMethods.SignedLOWORD(wParam);
            bool windowActivated;
 
 
            if ( loWord == NativeMethods.WA_INACTIVE )
            {
                windowActivated = false;
            }
            else
            {
                windowActivated = true;
            }
 
            HandleActivate(windowActivated);
 
            return false;
        }
 
        // When the window is in a maximized or minimized state, we want the dimension properties
        // to reflect what it would be when it's restored.
        private void UpdateDimensionsToRestoreBounds()
        {
            Rect restoreRect = RestoreBounds;
            SetValue(LeftProperty, restoreRect.Left);
            SetValue(TopProperty, restoreRect.Top);
            SetValue(WidthProperty, restoreRect.Width);
            SetValue(HeightProperty, restoreRect.Height);
        }
 
        private bool WmSizeChanged(IntPtr wParam)
        {
            // For WS_CHILD window, WM_SIZE, WM_MOVE (and maybe others) are called
            // synchronously from CreateWindowEx call and we run into issues if
            // _sourceWindow in null.  We only care to listen to WM_CREATE &
            // WM_GETMINMAXINFO synchronously from CreateWindowEx thus we want
            // to explicitly add the null check below, at all other places.
            //
            // Adding IsCompositionTargetInvalid check here
            // Add this check here means that we won't fire WindowStateChanged event.
            // However, since the hwnd is going away anyways, not firing StateChanged
            // should not be a big deal.  The other side effect is that Width/Height DPs
            // won't be updated that should be fine too since window is going away.
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return false;
            }
 
            NativeMethods.RECT rc = WindowBounds;
            Point windowSize = new Point(rc.right - rc.left, rc.bottom - rc.top);
            Point ptLogicalUnits = DeviceToLogicalUnits(windowSize);
 
            try
            {
                _updateHwndSize = false;
                SetValue(FrameworkElement.WidthProperty, ptLogicalUnits.X);
                SetValue(FrameworkElement.HeightProperty, ptLogicalUnits.Y);
            }
            finally
            {
                _updateHwndSize = true;
            }
 
            // Update the Taskbar's data about this window.  This can intermittently fail if Explorer gets busy.
            // It's not worth handling failures here since if there is a clip we'll try again on the next resize
            // and it doesn't affect the window itself.
            HRESULT hr = UpdateTaskbarThumbnailClipping();
 
            switch ((int)wParam)
            {
                // We introduced _previousWindowState for the following scenario:
                //
                // win1.WindowState = WindowState.Maximized; // or stateA
                // ...
                // win1.WindowState = WindowState.Normal; // or stateB
                //
                // Developer sets WindowState to Maximized/Minimized then return to Normal.
                // OnStateChanged should be fired. However, We were comparing that to WindowState.Normal.
                // The value in property engine had been updated to Normal in the CLR setter, so OnStateChanged
                // was never fired.
                // This is addressed in bug 937458.
                //
                // Another reason for remembering the previous state is that
                // WM_SIZE is sent when the client area size of the hwnd changes.
                // Thus, if the hwnd is maximized, and the border style changes,
                // WM_SIZE is sent.  In such cases, we don't want to fire
                // StateChanged since the previous state was maximized too.
                // (NullReferenceException thrown when changing ResizeMode from CanMinimize
                // to CanResizeWithGrip inside StateChanged event handler).
                //
                // There are two places we update _previousWindowState.
 
                // WindowState can be changed as a result of the following two passes
                // 1. User interaction changes WindowState
                // 2. Developer programmatically changes WindowState
                // We update _previousWindowState at two places
                // 1. Before Hwnd is created, when developer programmatically changes WindowState, we update it when WindowState is invalidated.
                // 2. After Hwnd is created, we update it here because both passes comes here eventually.
 
                // Event handler exception continuality: if exception occurs in StateChanged event handler, our state will not be
                // corrupted because the state related to StateChanged, WindowStateProperty and _previousWindowState, are set before the event is fired.
                // Please check Event handler exception continuality if the logic changes.
                case NativeMethods.SIZE_MAXIMIZED:
                    if (_previousWindowState != WindowState.Maximized)
                    {
                        // Do not set local value unless it is from user interaction.
                        // User interaction is considered as the same pri as SetValue
                        // If WindowState is not the same as the current win32 value, it means the change is
                        // not from the DP system but from user interaction.
                        if (WindowState != WindowState.Maximized)
                        {
                            try
                            {
                                _updateHwndLocation = false;
                                _updateHwndSize = false;
                                UpdateDimensionsToRestoreBounds();
                            }
                            finally
                            {
                                _updateHwndSize = true;
                                _updateHwndLocation = true;
                            }
                            WindowState = WindowState.Maximized;
                        }
                        // The maximizing size we get from WM_GETMINMAXINFO is only valid for the primary monitor, if the primary monitor
                        // happens to be smaller than the secondary monitor, we may end up not maximizing correctly in the secondary monitor.
                        // Here we are sure that this size value is coming from the OS so it is safe to update our maximizing size.
                        _windowMaxWidthDeviceUnits = Math.Max(_windowMaxWidthDeviceUnits, windowSize.X);
                        _windowMaxHeightDeviceUnits = Math.Max(_windowMaxHeightDeviceUnits, windowSize.Y);
 
                        _previousWindowState = WindowState.Maximized;
                        OnStateChanged(EventArgs.Empty);
                    }
                    break;
                case NativeMethods.SIZE_MINIMIZED:
                    if (_previousWindowState != WindowState.Minimized)
                    {
                        if (WindowState != WindowState.Minimized)
                        {
                            try
                            {
                                _updateHwndSize = false;
                                _updateHwndLocation = false;
                                UpdateDimensionsToRestoreBounds();
                            }
                            finally
                            {
                                _updateHwndSize = true;
                                _updateHwndLocation = true;
                            }
                            WindowState = WindowState.Minimized;
                        }
                        _previousWindowState = WindowState.Minimized;
                        OnStateChanged(EventArgs.Empty);
                    }
                    break;
                case NativeMethods.SIZE_RESTORED:
                    if (_previousWindowState != WindowState.Normal)
                    {
                        if (WindowState != WindowState.Normal)
                        {
                            WindowState = WindowState.Normal;
                            WmMoveChangedHelper();
                        }
                        _previousWindowState = WindowState.Normal;
                        OnStateChanged(EventArgs.Empty);
                    }
                    break;
                default:
                    break;
            }
 
            // DON'T DO ANYTHIHG HERE SINCE WE FIRE STATECHANGED ABOVE.  USER CODE
            // RUNS IN STATE CHANGED AND WINDOW COULD HAVE BEEN CLOSED.  THUS, THE
            // STATE OF VARIABLES IS UNKNOWN.
 
            // HwndSource passes the win32 msgs to the public hookds first before
            // passing layout hook etc.  Thus we get the WM_SIZE msg before
            // layout filter has processed it and we were reporting stale
            // value for height/width.  Now, UIElement fires SizeChanged event.
 
            return false;
        }
 
 
        private bool WmMoveChanged()
        {
            // We want to listen to WM_MOVE synchronously since if a Window is
            // created minimized/maximized we get this message as a result of
            // calling CreateWindowEx.  Here, sourceWindow is null but we still
            // need to update _actual[Top/Left] to reflect the correct Top/Left
            // when Show() returns.  Thus we input hwnd and process top/left info
            //
            // We won't fire LocationChanged unless Show has returned meaning
            // IsSourceWindowNull is false.  Furthermore, LocationChanged is
            // fired only if the top/left values really changed
 
            // Adding IsCompositionTargetInvalid check here
            // Since hwnd is going away, not updating Top/Left DPs and not firing
            // LocationChanged should not matter.
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return false;
            }
 
            // the input lparam gives the client location,
            // so just call GetWindowRect for Left and Top.
            NativeMethods.RECT rc = WindowBounds;
 
            Point ptLogicalUnits = DeviceToLogicalUnits(new Point(rc.left, rc.top));
 
 
            if (!DoubleUtil.AreClose(_actualLeft, ptLogicalUnits.X) ||
                !DoubleUtil.AreClose(_actualTop, ptLogicalUnits.Y))
            {
                _actualLeft = ptLogicalUnits.X;
                _actualTop = ptLogicalUnits.Y;
 
                // In Window, WmMoveChangedHelper write the local value of Top/Left
                // (if necessary) or updates the property system values for
                // Top/Left by calling CoerceValue.  Furthermore, it fires the
                // LocationChanged event.  RBW overrides WmMoveChangedHelper to do
                // nothing as writing Top/Left is not supported for RBW and
                // LocationChanged is never fired for it either.
                WmMoveChangedHelper();
 
                //Invalidate AutomationPeer if it was created/used by Automation.
                //This will schedule a deferred update of bounding rectangle and
                //corresponding notification to the Automation layer.
                AutomationPeer peer = UIElementAutomationPeer.FromElement(this);
                if(peer != null)
                {
                    peer.InvalidatePeer();
                }
}
 
            return false;
        }
 
        // This method updates the Left/Top values and fires the location changed event.
        // It is virtual so that RBW can override it.
        internal virtual void WmMoveChangedHelper()
        {
            if (WindowState == WindowState.Normal)
            {
                try
                {
                    _updateHwndLocation = false;
                    SetValue(LeftProperty, _actualLeft);
                    SetValue(TopProperty, _actualTop);
                }
                finally
                {
                    _updateHwndLocation = true;
                }
 
                // Event handler exception continuality: if exception occurs in LocationChanged event handler, our state will not be
                // corrupted because the states related to LocationChanged, LeftProperty, TopProperty, Left and Top are set before the event is fired.
                // Please check event handler exception continuality if the logic changes.
                OnLocationChanged(EventArgs.Empty);
            }
        }
 
 
        private bool WmGetMinMaxInfo( IntPtr lParam )
        {
            NativeMethods.MINMAXINFO mmi;
            unsafe
            {
                mmi = *(NativeMethods.MINMAXINFO*)lParam;
            }
 
            //
            // For Bug 1380569: Window SizeToContent does not work after changing Max size properties
            //
            // When Min/Max size is changed in this Window instance, we want to make sure the correct
            // final Min/Max size is used to measure the window layout and notify the Win32 of the required
            // Min/Max size.
            //
            // This method is responsible to notify Win32 of the new Min/Max size.
            // MeasureOverride( ) is responisble to use the right Min/Max size to calculate the desired layout size.
            //
            // But only this method knows the Win32 restricted Min/Max value for the HWND when it responds to WM_GETMINMAXINFO message.
            //
            // To generate the right final Min/Max size value in both places ( here and MeasureOverride), we should
            // cache the Win32 restricted size here.
            //
 
 
 
            // We need to store the max/min size the hwnd can take.  This is used in GetWindowMinMax to determine
            // the size passed to children for their layout.  These are stored in device units here so later
            // we change them to logical units.  Fixes the following bugs:
            //
            // Wrong window actual size returned if Autosize window
            // content is smaller than the actual window (seems to return content
            // size as opposed to window size)
            //
            // Window content should respect Window's Max/Min size
 
            _trackMinWidthDeviceUnits = mmi.ptMinTrackSize.x;
            _trackMinHeightDeviceUnits = mmi.ptMinTrackSize.y;
            _trackMaxWidthDeviceUnits = mmi.ptMaxTrackSize.x;
            _trackMaxHeightDeviceUnits = mmi.ptMaxTrackSize.y;
            _windowMaxWidthDeviceUnits = mmi.ptMaxSize.x;
            _windowMaxHeightDeviceUnits = mmi.ptMaxSize.y;
 
 
            // if IsCompositionTargetInvalid is true, then it means that the CompositionTarget is not available.
            // This can happen at hwnd creation or destruction time.
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                //
                // Get the final MinMax size for this HWND based on Win32 track value and Min/Max setting
                // in this instance.
                //
                WindowMinMax finalMinMax = GetWindowMinMax( );
 
                // The finalMinMax struct keeps the desired Min/Max size for this hwnd in Logic Units.
                Point minSizeDeviceUnits = LogicalToDeviceUnits(new Point(finalMinMax.minWidth, finalMinMax.minHeight));
                Point maxSizeDeviceUnits = LogicalToDeviceUnits(new Point(finalMinMax.maxWidth, finalMinMax.maxHeight));
 
                // Put the new value in mmi
                mmi.ptMinTrackSize.x = DoubleUtil.DoubleToInt(minSizeDeviceUnits.X);
                mmi.ptMinTrackSize.y = DoubleUtil.DoubleToInt(minSizeDeviceUnits.Y);
 
                mmi.ptMaxTrackSize.x = DoubleUtil.DoubleToInt(maxSizeDeviceUnits.X);
                mmi.ptMaxTrackSize.y = DoubleUtil.DoubleToInt(maxSizeDeviceUnits.Y);
 
                // Notify Win32 of the new Min/Max value for this HWND.
 
                unsafe
                {
                    *(NativeMethods.MINMAXINFO*)lParam = mmi;
                }
            }
 
            return true;
        }
 
        private bool WmNcHitTest( IntPtr lParam, ref IntPtr refInt )
        {
            // For WS_CHILD window, WM_SIZE, WM_MOVE (and maybe others) are called
            // synchronously from CreateWindowEx call and we run into issues if
            // _sourceWindow in null.  We only care to listen to WM_CREATE &
            // WM_GETMINMAXINFO synchronously from CreateWindowEx thus we want
            // to explicitly add the null check below at all other places.
            //
            // Adding check for IsCompositionTargetInvalid
            // This can be true either at HwndSource creation time or when hwnd is going
            // away.
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return false;
            }
 
            // WmNcHitTest is necessary to enable ResizeGrip and it only
            // relevant for Window.  Doing this processing in a virtual so
            // that RBW can override it
            return HandleWmNcHitTestMsg(lParam, ref refInt);
        }
 
        internal virtual bool HandleWmNcHitTestMsg(IntPtr lParam, ref IntPtr refInt)
        {
            // We need to make sure that the calculation to find out
            // whether the mouse is over the ResizeGrip control works
            // for localized version of Windows (Arabic, Hebrew) which
            // uses RTL.  Also, we need to make sure it works for
            // multi-mon case
 
            if ((_resizeGripControl == null) || (ResizeMode != ResizeMode.CanResizeWithGrip))
            {
                return false;
            }
 
            // mouse position wrt to the left/top of the screen
            int x = NativeMethods.SignedLOWORD(lParam);
            int y = NativeMethods.SignedHIWORD(lParam);
 
            // Find the client area 0,0 of the Window wrt the screen
            // This will be used to transform the mouse position from screen co-od
            // to window's client area co-od.  We need this to be able to figure out
            // whether the mouse is currently over the resize grip control or not
 
 
            NativeMethods.POINT pt = GetPointRelativeToWindow(x, y);
            Point ptLogicalUnits = DeviceToLogicalUnits(new Point(pt.x, pt.y));
 
            // Now, (ptLogicalUnits.X, ptLogicalUnits.Y) is the mouse postion wrt to the
            // Window's client region.
            // The next step is to find out whether the mouse is on top of the
            // ResizeGrip control
            // For this we first need to find out mouse location wrt to the ResizeGrip
            // control and then check whether the mouse location is on the control
            // Conditions when mouse is on top of the control:
            //     x,y should be not be less than zero
            //     x,y should not be greater than RenderSize.Width and RenderSize.Height
 
            GeneralTransform transfromFromWindow = this.TransformToDescendant(_resizeGripControl);
            Point mousePositionWRTResizeGripControl = ptLogicalUnits;
            if (transfromFromWindow == null || transfromFromWindow.TryTransform(ptLogicalUnits, out mousePositionWRTResizeGripControl) == false)
            {
                return false;
            }
 
            // check if the mouse is outside the ResizeGripControl region
            if ((mousePositionWRTResizeGripControl.X < 0) ||
                (mousePositionWRTResizeGripControl.Y < 0 ) ||
                (mousePositionWRTResizeGripControl.X > _resizeGripControl.RenderSize.Width) ||
                (mousePositionWRTResizeGripControl.Y > _resizeGripControl.RenderSize.Height))
            {
                // mouse not over ResizeGripControl; just let the DefWndProc handle this
                return false;
            }
 
            if (FlowDirection == FlowDirection.RightToLeft)
            {
                refInt = new IntPtr(NativeMethods.HTBOTTOMLEFT);
            }
            else
            {
                refInt = new IntPtr(NativeMethods.HTBOTTOMRIGHT);
            }
            // we've handled the WM_NCHITTEST msg thus return true
            return true;
        }
 
        private bool WmShowWindow(IntPtr wParam, IntPtr lParam)
        {
            // For WS_CHILD window, WM_SIZE, WM_MOVE (and maybe others) are called
            // synchronously from CreateWindowEx call and we run into issues if
            // _sourceWindow in null.  We only care to listen to WM_CREATE &
            // WM_GETMINMAXINFO synchronously from CreateWindowEx thus we want
            // to explicitly add the null check below at all other places.
            //
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return false;
            }
 
            //lParam has five values: 0(programtically show/hide), 1(SW_PARENTCLOSING),
            //2(SW_OTHERMAXIMIZED), 3(SW_PARENTOPENING), 4(SW_OTHERRESTORED). We only care
            //about 1 and 3.
            switch (NativeMethods.IntPtrToInt32(lParam))
            {
                //The window's owner window is being minimized.
                //In Win32, when lParam is SW_PARENTCLOSING, wParam is false,
                //which means the window is being hidden.
                case NativeMethods.SW_PARENTCLOSING:
                    //This window will be hidden. Update _isVisible to reflect the
                    // new state.  Furthermore update visibility such that we
                    // do not call ShowHelper again and thus calling
                    // UpdateVisibilityProperty.
                    _isVisible = false;
                    UpdateVisibilityProperty(Visibility.Hidden);
                    break;
 
                //The window's owner window is being restored.
                //In Win32, when lParam is SW_PARENTOPENING, wParam is true,
                //which means the window is being shown.
                case NativeMethods.SW_PARENTOPENING:
                    //This window will be shown. Update _isVisible to reflect the
                    // new state.  Furthermore update visibility such that we
                    // do not call ShowHelper again and thus calling
                    // UpdateVisibilityProperty.
                    _isVisible = true;
                    UpdateVisibilityProperty(Visibility.Visible);
                    break;
 
                default:
                    break;
            }
 
            return false;
        }
 
        private static void _OnIconChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = (Window)d;
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
 
            // We'll support most kinds of Images.  If it's not a BitmapFrame we'll rasterize it.
            w.OnIconChanged(e.NewValue as ImageSource);
        }
 
 
        private void OnIconChanged(ImageSource newIcon)
        {
            // No need to dispose previous _icon.
            // _icon is a ref to the ImageSource object
            // set by the developer.  Since the dev created
            // the ImageSource object it is his responsibility to
            // dispose it.
            _icon = newIcon;
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                UpdateIcon();
            }
        }
 
        private static void _OnTitleChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = (Window)d;
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
 
            w.OnTitleChanged();
        }
 
        private static bool _ValidateText(object value)
        {
            return (value != null);
        }
 
        private void OnTitleChanged()
        {
            UpdateTitle(Title);
        }
 
        private static void _OnShowInTaskbarChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = (Window)d;
 
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
            w.OnShowInTaskbarChanged();
        }
 
        private void OnShowInTaskbarChanged()
        {
            // this call ends up throwing an exception if accessing
            // ShowInTaskbar is not allowed
            VerifyApiSupported();
 
            // There are 2 cases
            // Case 1 : being set before source window is created
            // Case 2 : being set after the source window is created
            // Case 3 : bet set when CompositionTarget is invalid meaning we're in a bad state
 
            // Adding check for IsCompositionTargetInvalid
            if ( IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                bool fHideWindow = false;
                // Win32 bug. For ShowInTaskbar to change dynamically, we need to hide then show the window.
                // It is recommended to hide the window, chnage the style bits and then show it again.
                if (_isVisible)
                {
                    UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle), NativeMethods.NullHandleRef, 0, 0, 0, 0,
                                       NativeMethods.SWP_NOMOVE |
                                       NativeMethods.SWP_NOSIZE |
                                       NativeMethods.SWP_NOZORDER |
                                       NativeMethods.SWP_NOACTIVATE |
                                       NativeMethods.SWP_NOSENDCHANGING |
                                       NativeMethods.SWP_HIDEWINDOW);
                    fHideWindow = true;
                }
                using (HwndStyleManager sm = HwndStyleManager.StartManaging(this, StyleFromHwnd, StyleExFromHwnd  ))
                {
                    SetTaskbarStatus();
                }
                // Use fHideWindow instead of _isVisible in case if we listen to HideWindow messages and update _isVisible value,
                // it won't break this code.
                if (fHideWindow)
                {
                    UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle), NativeMethods.NullHandleRef, 0, 0, 0, 0,
                                                    NativeMethods.SWP_NOMOVE |
                                                    NativeMethods.SWP_NOSIZE |
                                                    NativeMethods.SWP_NOZORDER |
                                                    NativeMethods.SWP_NOACTIVATE |
                                                    NativeMethods.SWP_NOSENDCHANGING |
                                                    NativeMethods.SWP_SHOWWINDOW);
                }
            }
        }
 
        private static bool _ValidateWindowStateCallback(object value)
        {
            return IsValidWindowState((WindowState)value);
        }
 
        private static void _OnWindowStateChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = (Window)d;
 
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
            w.OnWindowStateChanged((WindowState) e.NewValue);
        }
 
        private void OnWindowStateChanged(WindowState windowState)
        {
 
            //      WCP:  Window.Visible.Set : Make sure that window updates the styles
            //      when set while window is hidden
            //
            // We can only change the window state if window is currently
            // visible.  Win32 does not provide a way to change the window
            // state without showing the window.  Thus, for the case where
            // window is hidden, we defer the acutal state change till the
            // window is shown again.
            //
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                if (_isVisible == true)
                {
                    HandleRef hr = new HandleRef(this,  Handle);
 
                    int style = _Style;
 
                    // Only call ShowWindow if window is in a different state
                    switch (windowState)
                    {
                        case WindowState.Normal:
                            if ((style & NativeMethods.WS_MAXIMIZE) == NativeMethods.WS_MAXIMIZE)
                            {
                                //
                                // The old behavior of this case is to restore the window using SW_RESTORE.
                                // With the ShowActivated property set to false we want this restore operation
                                // to take the current activation state into account when restoring the window.
                                //
                                if (ShowActivated || IsActive)
                                    UnsafeNativeMethods.ShowWindow(hr, NativeMethods.SW_RESTORE);
                                else
                                    UnsafeNativeMethods.ShowWindow(hr, NativeMethods.SW_SHOWNOACTIVATE);
                            }
                            else if ((style & NativeMethods.WS_MINIMIZE) == NativeMethods.WS_MINIMIZE)
                            {
                                //
                                // We query to WINDOWPLACEMENT to get an indication about the state before the
                                // minimize operation happened. If we were coming from a maximized state and now we
                                // switch to normal, we want activation to happen since the maximized state is always
                                // activated and transitioning from activated to non-activated would be weird.
                                //
                                NativeMethods.WINDOWPLACEMENT placement = new NativeMethods.WINDOWPLACEMENT();
                                placement.length = Marshal.SizeOf(placement);
                                UnsafeNativeMethods.GetWindowPlacement(hr, ref placement);
 
                                if ((placement.flags & NativeMethods.WPF_RESTORETOMAXIMIZED) == NativeMethods.WPF_RESTORETOMAXIMIZED)
                                    UnsafeNativeMethods.ShowWindow(hr, NativeMethods.SW_RESTORE);
                                else
                                {
                                    if (ShowActivated)
                                        UnsafeNativeMethods.ShowWindow(hr, NativeMethods.SW_RESTORE);
                                    else
                                        UnsafeNativeMethods.ShowWindow(hr, NativeMethods.SW_SHOWNOACTIVATE);
                                }
                            }
                            break;
 
                        case WindowState.Maximized:
                            if ((style & NativeMethods.WS_MAXIMIZE) != NativeMethods.WS_MAXIMIZE)
                            {
                                //
                                // The OS doesn't provide support for non-activated maximized windows.
                                //
                                UnsafeNativeMethods.ShowWindow(hr, NativeMethods.SW_MAXIMIZE);
                            }
                            break;
 
                        case WindowState.Minimized:
                            if ((style & NativeMethods.WS_MINIMIZE) != NativeMethods.WS_MINIMIZE)
                            {
                                //
                                // Historically, we used SW_MINIMIZE in here which activates the next top-level
                                // window in the Z order. Therefore, our ShowActivated property can't affect the
                                // minimized state since this would incur a breaking change requiring us to use
                                // SW_SHOWMINIMIZED instead in case ShowActivated is set to true (bw compat case).
                                //
                                UnsafeNativeMethods.ShowWindow(hr, NativeMethods.SW_MINIMIZE);
                            }
                            break;
                        // WCP: Window.WindowState should implement FullScreen and Theatre
                        // modes
                    }
                }
            }
            else
            {
                // WindowState can be changed as a result of the following two passes
                // 1. User interaction changes WindowState
                // 2. Developer programmatically changes WindowState
                // We update _previousWindowState at two places
                // 1. Before Hwnd is created, when developer programmatically changes WindowState, we update it here.
                // 2. After Hwnd is created, we update it when we get to WM_SIZE because both passes eventally meet there.
                _previousWindowState = windowState;
            }
 
            // The value of Top and Left is affected by WindowState and WindowStartupLocation.
            // we need to coerce Top and Left whenever these deciding factors change.
            // More info in CoerceTop.
            try
            {
                _updateHwndLocation = false;
                CoerceValue(TopProperty);
                CoerceValue(LeftProperty);
            }
            finally
            {
                _updateHwndLocation = true;
            }
        }
 
        private static bool _ValidateWindowStyleCallback(object value)
        {
            return IsValidWindowStyle((WindowStyle)value);
        }
 
        private static void _OnWindowStyleChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = (Window)d;
 
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
            w.OnWindowStyleChanged((WindowStyle) e.NewValue);
        }
 
        private void OnWindowStyleChanged(WindowStyle windowStyle)
        {
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                using (HwndStyleManager sm = HwndStyleManager.StartManaging(this, StyleFromHwnd, StyleExFromHwnd ))
                {
                    CreateWindowStyle();
                }
            }
        }
 
        private static void _OnTopmostChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = (Window)d;
 
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
            w.OnTopmostChanged((bool) e.NewValue);
        }
 
        private void OnTopmostChanged(bool topmost)
        {
 
            // this call ends up throwing an exception if accessing
            // Topmost is not allowed
            VerifyApiSupported();
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false  && IsCompositionTargetInvalid == false)
            {
                HandleRef hWnd = topmost ? NativeMethods.HWND_TOPMOST : NativeMethods.HWND_NOTOPMOST;
                UnsafeNativeMethods.SetWindowPos(new HandleRef(null, Handle),
                       hWnd,
                       0, 0, 0, 0,
                       NativeMethods.SWP_NOMOVE | NativeMethods.SWP_NOSIZE | NativeMethods.SWP_NOACTIVATE);
            }
        }
 
        private static object CoerceVisibility(DependencyObject d, object value)
        {
            Window w = (Window)d;
 
            Visibility newValue = (Visibility)value;
            if (newValue == Visibility.Visible)
            {
                w.VerifyCanShow();
                w.VerifyConsistencyWithAllowsTransparency();
                w.VerifyNotClosing();
                w.VerifyConsistencyWithShowActivated();
            }
 
            return value;
        }
 
        /// <summary>
        /// Called when VisiblityProperty is invalidated
        /// The actual window is created when the Visibility property is set
        /// to Visibility.Visible for the first time or when Show is called.
        /// For Window, Visibility.Visible means the Window is visible.
        /// Visibility.Hidden and Visibility.Collapsed mean the Window is not visible.
        /// Visibility.Hidden and Visibility.Collapsed are treated the same.
        /// </summary>
        private static void _OnVisibilityChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = (Window)d;
 
            // Indicate Visibility has been set
            // This works fine because Window is always the root.  So Visibility property
            // would not be invalidated usless it is set to a value.  But if that changes,
            // we will get invalidation when Window it added to the tree and this would be broken.
            w._isVisibilitySet = true;
 
            // _visibilitySetInternally is used to identify a call from Show/Hide in
            // _OnVisibilityInvalidated callback.  If a call originates in Show/Hide,
            // we DO NOT want to do anything in the _OnVisibilityCallback since, we
            // synchronously call ShowHelper from Show/Hide
            if (w._visibilitySetInternally == true)
            {
                return;
            }
 
            bool visibilityValue = VisibilityToBool((Visibility) e.NewValue);
 
            w.Dispatcher.BeginInvoke(
                DispatcherPriority.Normal,
                new DispatcherOperationCallback(w.ShowHelper),
                visibilityValue ? BooleanBoxes.TrueBox : BooleanBoxes.FalseBox);
        }
 
        private void SafeCreateWindowDuringShow()
        {
            //this is true the first time the window is created
            if (IsSourceWindowNull == true)
            {
                // _isVisible is false at this moment.  Thus CreateAllStyle
                // called by CreateSourceWindow does not set WS_VISIBLE style
 
                CreateSourceWindowDuringShow();
            }
                // If the hwnd has been created via WindowsInteropHelper.EnsureHandle,
                // we just need to hook up the RootVisual and update the size according to STC before Show.
            else if (HwndCreatedButNotShown)
            {
                SetRootVisualAndUpdateSTC();
                _hwndCreatedButNotShown = false;
            }
        }
 
        // We set/clear ShowKeyboardCue when Show(ShowDialog)/Hide is called.
        // We do not clear the state of ShowKeyboardCue when Window is closed.
        private void SetShowKeyboardCueState()
        {
            // set property on AccessKey control indicating the
            // invocation device
            if (KeyboardNavigation.IsKeyboardMostRecentInputDevice())
            {
                _previousKeyboardCuesProperty = (bool)GetValue(KeyboardNavigation.ShowKeyboardCuesProperty);
                SetValue(KeyboardNavigation.ShowKeyboardCuesProperty, BooleanBoxes.TrueBox);
                _resetKeyboardCuesProperty = true;
            }
        }
 
        // We set/clear ShowKeyboardCue when Show(ShowDialog)/Hide is called.
        // We do not clear the state of ShowKeyboardCue when Window is closed.
        private void ClearShowKeyboardCueState()
        {
            // if we set KeyboradNavigation.ShowKeyboardCuesProperty in ShowDialog,
            // set it to false here.
            if (_resetKeyboardCuesProperty == true)
            {
                _resetKeyboardCuesProperty = false;
                SetValue(KeyboardNavigation.ShowKeyboardCuesProperty, BooleanBoxes.Box(_previousKeyboardCuesProperty));
            }
        }
 
        private void UpdateVisibilityProperty(Visibility value)
        {
            // _visibilitySetInternally is used to identify a call (in _OnVisibilityInvalidated
            // callback) for updating the property value only and not changing the actual
            // visibility state of the hwnd.
            try
            {
                _visibilitySetInternally = true;
                SetValue(VisibilityProperty, value);
            }
            finally
            {
                _visibilitySetInternally = false;
            }
        }
 
        /// <summary>
        /// update _isVisible and call CreateSourceWindow if
        /// it's the first time window is set to Visibile
        /// </summary>
        private object ShowHelper(object booleanBox)
        {
            // Setting Visiblilty is async. When this is called from the async callback,
            // check whether the window is already closed.
            // E.g. window.Visibility = true;
            //      ...
            //      window.Close();
            // We should not do anything if the window is already closed.
            if (_disposed == true)
            {
                return null;
            }
 
            bool value = (bool) booleanBox;
            _isClosing = false;
 
            // (BVT Blocker: Invariant Assert when calling
            // Window.Show after setting Visibility=Hidden)
 
            // We should optimize for when visibilityValue == _isVisible only in ShowHelper
            // since this is called from a sync and async call.  We cannot optimize it there
            // since _isVisible may not reflect the exact state requested by the OM call.
            if (_isVisible == value)
            {
                return null;
            }
 
            // _isVisible should always be set after calling SafeCreateWindow, because
            // if exception occurs in Loading event (fired as a result of setting Visibility to visible) handler,
            // we set Visibility back to Collapsed. Otherwise we could get into a loop.
            if (value == true)
            {
                if (Application.IsShuttingDown)
                    return null;
 
                SetShowKeyboardCueState();
 
                SafeCreateWindowDuringShow();
                _isVisible = true;
            }
            else
            {
 
                ClearShowKeyboardCueState();
 
                if (_showingAsDialog == true)
                {
                    DoDialogHide();
                }
                _isVisible = false;
            }
 
            // we need this check here again, b/c creating the window fires the
            // Activted event and if user closes the window from it, then by
            // the time we get to this point _sourceWindow is already disposed.
            if ( IsSourceWindowNull == false )
            {
 
                // Specifying an Avalon app to start
                // maximized from a shortcut does not work.
 
                // ShowWindow MSDN documentation says that the first time ShowWindow
                // is called, nCmd passed in STARTUPINFO is used instead of the one
                // passed in via ShowWindow call. However, that is not the case.
                // ShowWindow implementation in user32 uses nCmd of STARTUPINFO only
                // if we pass SW_SHOW, SW_SHOWNORMAL, SW_SHOWDEFAULT to ShowWindow.
                // If anything else is passed, it does not use nCmd of STARTUPINFO.
 
                int nCmd = 0;
                if (value == true)
                {
                    // nCmdForShow access WindowState which is inaccessible for RBW.
                    // Thus doing so in a virtual that RBW overrides
                    nCmd = nCmdForShow();
                }
                else
                {
                    nCmd = NativeMethods.SW_HIDE;
                }
 
                // If this is a Topmost window, and the option is enabled to use SetWindowPos() when possible
                // for topmost windows, and this is SW_SHOW or SW_SHOWNA, then use SetWindowPos().  (Ideally
                // all SW_SHOW* options would use SetWindowPos, but (1) it is difficult to handle the maximize
                // and minimize options without ShowWindow, and (2) the z-order issue happens as soon as the
                // topmost window gets activated, which happens for SW_SHOWMAXIMIZED and SW_SHOWMINIMIZED.  So
                // to minimize complexity and risk those cases will continue to use ShowWindow.)
                //
                // Determine whether this is a Topmost window without calling
                // the Topmost property.  (This isn't allowed for xbap windows.)
                bool isTopmost = (bool)GetValue(TopmostProperty);
                if (isTopmost &&
                    FrameworkCompatibilityPreferences.GetUseSetWindowPosForTopmostWindows() &&
                    (nCmd == NativeMethods.SW_SHOW || nCmd == NativeMethods.SW_SHOWNA))
                {
                    int flags = (nCmd == NativeMethods.SW_SHOWNA) ? NativeMethods.SWP_NOACTIVATE : 0;
                    UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle),
                        NativeMethods.HWND_TOPMOST,
                        0, 0, 0, 0,
                        flags | NativeMethods.SWP_NOMOVE | NativeMethods.SWP_NOSIZE | NativeMethods.SWP_NOOWNERZORDER | NativeMethods.SWP_SHOWWINDOW);
                }
                else
                {
                    UnsafeNativeMethods.ShowWindow(new HandleRef(this, Handle), nCmd);
                }
 
                // We already did a ShowWindow upabove and then because of the using, we will flush which
                // will cause us to set the visibility. Ideally I would like to simply not have the ShowWindow
                // call above, *but* there is this SHOWNA stuff which is tied to Focus/Activation cleanup
                // scheduled for M8.
                //set the style
                SafeStyleSetter();
            }
 
 
            // dialog functionality; start dispatcher loop to block the call
            if ((_showingAsDialog == true) && (_isVisible == true))
            {
                //
                // Since we exited the Context, we need to make sure
                // we enter it before returning even if there is an
                // exception
                //
                Debug.Assert(_dispatcherFrame == null, "_dispatcherFrame must be null here");
 
                try
                {
                    // tell users we're going modal
                    ComponentDispatcher.PushModal();
 
                    _dispatcherFrame = new DispatcherFrame();
                    Dispatcher.PushFrame(_dispatcherFrame);
                }
                finally
                {
                    // tell users we're going non-modal
                    ComponentDispatcher.PopModal();
                }
            }
 
            return null;
        }
 
        internal virtual int nCmdForShow()
        {
            int nCmd = 0;
            switch(WindowState)
            {
                case WindowState.Maximized:
                    nCmd = NativeMethods.SW_SHOWMAXIMIZED; // The OS doesn't provide support for non-activated maximized windows.
                    break;
                case WindowState.Minimized:
                    nCmd = ShowActivated ? NativeMethods.SW_SHOWMINIMIZED : NativeMethods.SW_SHOWMINNOACTIVE;
                    break;
                 default:
                    nCmd = ShowActivated ? NativeMethods.SW_SHOW : NativeMethods.SW_SHOWNA;
                    break;
            }
            return nCmd;
        }
 
        private void SafeStyleSetter()
        {
            using (HwndStyleManager sm = HwndStyleManager.StartManaging(this, StyleFromHwnd, StyleExFromHwnd))
            {
                _Style = _isVisible ? (_Style | NativeMethods.WS_VISIBLE) : _Style;
            }
        }
        private static bool _ValidateSizeToContentCallback(object value)
        {
            return IsValidSizeToContent((SizeToContent)value);
        }
 
        /// <summary>
        /// SizeToContent property GetValue override
        /// </summary>
        /// <param name="d"></param>
        /// <returns></returns>
        private static object _SizeToContentGetValueOverride(DependencyObject d)
        {
            Window w = d as Window;
 
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
            return w.SizeToContent;
        }
 
        /// <summary>
        /// SizeToContent property invalidation callback
        /// </summary>
        private static void _OnSizeToContentChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
 
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
            w.OnSizeToContentChanged((SizeToContent) e.NewValue);
        }
 
        private void OnSizeToContentChanged(SizeToContent sizeToContent)
        {
            // this call ends up throwing an exception if accessing
            // SizeToContent is not allowed
            VerifyApiSupported();
 
            // Update HwndSource's SizeToContent.
            // HwndSource will only update layout if the value has changed.
            //
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                HwndSourceSizeToContent = sizeToContent;
            }
        }
 
        /// <summary>
        /// Validate [Max/Min]Width/Height and Top/Left value.
        /// </summary>
        /// Length takes Double; Win32 handles Int.
        /// We throw exception when the value goes below Int32Min and Int32Max.
        /// WorkItem 26263: ValidateValueCallback needs to move to PropertyMetadata so Window can
        /// add its own validation and validate before invalid value is set. Right now, we can only
        /// validate this in PropertyInalidatinonCallback because of this. (We couldn't make it virtual on
        /// FrameworkELement because ValidateValueCallback doesn't provide context. Work item 25275).
        private static void ValidateLengthForHeightWidth(double l)
        {
            //basically, NaN and PositiveInfinity are ok, and then anything
            //that can be converted to Int32
            if (!Double.IsPositiveInfinity(l) && !double.IsNaN(l) &&
                ((l > Int32.MaxValue) || (l < Int32.MinValue)))
            {
                throw new ArgumentException(SR.Format(SR.ValueNotBetweenInt32MinMax, l));
            }
        }
 
        private static void ValidateTopLeft(double length)
        {
            // Values not allowed: PositiveInfinity, NegativeInfinity
            // and values that are beyond the range of Int32
            if (Double.IsPositiveInfinity(length) ||
                Double.IsNegativeInfinity(length))
            {
                throw new ArgumentException(SR.Format(SR.InvalidValueForTopLeft, length));
            }
 
            if ((length > Int32.MaxValue) ||
                (length < Int32.MinValue))
            {
                throw new ArgumentException(SR.Format(SR.ValueNotBetweenInt32MinMax, length));
            }
        }
 
        private static void _OnHeightChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert(w != null, "d must be typeof Window");
            if (w._updateHwndSize)
            {
                w.OnHeightChanged((double) e.NewValue);
            }
        }
 
        private void OnHeightChanged(double height)
        {
            //  Move ValidateLengthForHeightWidth calls from property
            // invalidation callback to PropertyMetadata
            ValidateLengthForHeightWidth(height);
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false && !double.IsNaN(height))
            {
                UpdateHeight(height);
            }
        }
 
        private static void _OnMinHeightChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert(w != null, "d must be typeof Window");
            w.OnMinHeightChanged((double) e.NewValue);
        }
 
        private void OnMinHeightChanged(double minHeight)
        {
            // this call ends up throwing an exception if accessing
            // MinHeight is not allowed
            VerifyApiSupported();
 
            ValidateLengthForHeightWidth(minHeight);
            // Only trigger immediate size update when hwnd has been created and MinHeight is not Auto and MinHeight is
            // greater then current HWND height.
            // If hwnd hasn't been created, size will be controlled when it is created.
            // If MinHeight is Auto or ActualHeight is greater than MinHeight, there is no need update size of the window.
            //
            // Adding check for IsCompositionTargetInvalid
            if ((IsSourceWindowNull == false ) && (IsCompositionTargetInvalid == false))
            {
                NativeMethods.RECT rcHwnd = WindowBounds;
                Point logicalSize = DeviceToLogicalUnits(new Point(rcHwnd.Width, rcHwnd.Height));
                if (minHeight > logicalSize.Y)
                {
                    if (WindowState == WindowState.Normal)
                    {
                        UpdateHwndSizeOnWidthHeightChange(logicalSize.X, minHeight);
                    }
                    else
                    {
                        // no need to do anything.  When window is restored, we get WM_GETMINMAXINFO where
                        // we restrict the max/min size of the window to [Max/Min][Height/Width]
                    }
                }
            }
        }
 
        private static void _OnMaxHeightChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert(w != null, "d must be typeof Window");
            w.OnMaxHeightChanged((double) e.NewValue);
        }
 
        private void OnMaxHeightChanged(double maxHeight)
        {
            // this call ends up throwing an exception if accessing
            // MaxHeight is not allowed
            VerifyApiSupported();
 
            ValidateLengthForHeightWidth(MaxHeight);
 
            // Only trigger immediate size update when hwnd has been created and MaxHeight is not Auto and
            // the HWND's height > MaxHeight
            //
            // Adding check for IsCompositionTargetInvalid
            if ((IsSourceWindowNull == false) && (IsCompositionTargetInvalid == false))
            {
                NativeMethods.RECT rcHwnd = WindowBounds;
                Point logicalSize = DeviceToLogicalUnits(new Point(rcHwnd.Width, rcHwnd.Height));
                if (maxHeight < logicalSize.Y)
                {
                    if (WindowState == WindowState.Normal)
                    {
                        UpdateHwndSizeOnWidthHeightChange(logicalSize.X, maxHeight);
                    }
                    else
                    {
                        // no need to do anything.  When window is restored, we get WM_GETMINMAXINFO where
                        // we restrict the max/min size of the window to [Max/Min][Height/Width]
                    }
                }
            }
        }
 
        private static void _OnWidthChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert(w != null, "d must be typeof Window");
            if (w._updateHwndSize)
            {
                w.OnWidthChanged((double) e.NewValue);
            }
        }
 
        private void OnWidthChanged(double width)
        {
            ValidateLengthForHeightWidth(width);
 
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false && !double.IsNaN(width))
            {
                UpdateWidth(width);
            }
        }
 
        private static void _OnMinWidthChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert(w != null, "d must be typeof Window");
            w.OnMinWidthChanged((double) e.NewValue);
        }
 
        private void OnMinWidthChanged(double minWidth)
        {
            // this call ends up throwing an exception if accessing
            // MinWidth is not allowed
            VerifyApiSupported();
 
            // Move ValidateLengthForHeightWidth calls from property
            // invalidation callback to PropertyMetadata
            ValidateLengthForHeightWidth(minWidth);
            // Only trigger immediate size update when hwnd has been created and MinWidth is not Auto and MinWidth is
            // greater then current ActualWidth.
            // If hwnd hasn't been created, size will be controlled when it is created.
            // If MinWidth is Auto or ActualWidth is greater than MinWidth, there is no need update size of the window.
            //
            // Adding check for IsCompositionTargetInvalid
            if ((IsSourceWindowNull == false) && (IsCompositionTargetInvalid == false))
            {
                NativeMethods.RECT rcHwnd = WindowBounds;
                Point logicalSize = DeviceToLogicalUnits(new Point(rcHwnd.Width, rcHwnd.Height));
                if (minWidth > logicalSize.X)
                {
                    if (WindowState == WindowState.Normal)
                    {
                        UpdateHwndSizeOnWidthHeightChange(minWidth, logicalSize.Y);
                    }
                    else
                    {
                        // no need to do anything.  When window is restored, we get WM_GETMINMAXINFO where
                        // we restrict the max/min size of the window to [Max/Min][Height/Width]
                    }
                }
            }
        }
 
        private static void _OnMaxWidthChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert(w != null, "d must be typeof Window");
            w.OnMaxWidthChanged((double) e.NewValue);
        }
 
        private void OnMaxWidthChanged(double maxWidth)
        {
            // this call ends up throwing an exception if accessing
            // MaxWidth is not allowed
            VerifyApiSupported();
 
            ValidateLengthForHeightWidth(maxWidth);
            // Only trigger immediate size update when hwnd has been created and MaxWidth is not Auto and
            // ActualWidth > MaxWidth
            //
            // Adding check for IsCompositionTargetInvalid
            if ((IsSourceWindowNull == false ) && (IsCompositionTargetInvalid == false))
            {
                NativeMethods.RECT rcHwnd = WindowBounds;
                Point logicalSize = DeviceToLogicalUnits(new Point(rcHwnd.Width, rcHwnd.Height));
                if (maxWidth < logicalSize.X)
                {
                    if (WindowState == WindowState.Normal)
                    {
                        UpdateHwndSizeOnWidthHeightChange(maxWidth, logicalSize.Y);
                    }
                    else
                    {
                        // no need to do anything.  When window is restored, we get WM_GETMINMAXINFO where
                        // we restrict the max/min size of the window to [Max/Min][Height/Width]
                    }
                }
            }
        }
 
        // Updates the restore bounds of the hwnd based on BoundsSpecified enum values
        // OR-ing of BoundsSpecified enum is not supported.
        private void UpdateHwndRestoreBounds(double newValue, BoundsSpecified specifiedRestoreBounds)
        {
            NativeMethods.WINDOWPLACEMENT wp = new NativeMethods.WINDOWPLACEMENT
            {
                length = Marshal.SizeOf(typeof(NativeMethods.WINDOWPLACEMENT))
            };
 
            UnsafeNativeMethods.GetWindowPlacement(new HandleRef(this, Handle), ref wp);
 
            double convertedValue = (LogicalToDeviceUnits(new Point(newValue, 0))).X;
            switch (specifiedRestoreBounds)
            {
                case BoundsSpecified.Height:
                    wp.rcNormalPosition_bottom = wp.rcNormalPosition_top + DoubleUtil.DoubleToInt(convertedValue);
                    break;
                case BoundsSpecified.Width:
                    wp.rcNormalPosition_right = wp.rcNormalPosition_left + DoubleUtil.DoubleToInt(convertedValue);
                    break;
                case BoundsSpecified.Top:
                    // convert input value into work-area co-ods
                    double newTop = newValue;
                    // [Get/Set]WindowPlacement work with workarea co-ods for a top level
                    // window whose WS_EX_TOOLWINDOW bit is clear.  If this bit is set,
                    // then the co-ods are expected to be in screen co-ods of the monitor.
                    // TransfromWorkAreaScreenArea can transform a point from work area co-ods
                    // to screen area co-ods and vice versa depending on TransformType value passed.
                    // So, in our case, if the window is not a ToolWindow we want to transform
                    // the input value from screen co-ods to work area co-ods.
                    if ((StyleExFromHwnd & NativeMethods.WS_EX_TOOLWINDOW) == 0)
                    {
                        newTop = TransformWorkAreaScreenArea(new Point(0, newTop), TransformType.ScreenAreaToWorkArea).Y;
                    }
                    newTop = (LogicalToDeviceUnits(new Point(0, newTop))).Y;
                    int currentHeight = wp.rcNormalPosition_bottom - wp.rcNormalPosition_top;
                    wp.rcNormalPosition_top = DoubleUtil.DoubleToInt(newTop);
                    wp.rcNormalPosition_bottom = wp.rcNormalPosition_top + currentHeight;
                    break;
                case BoundsSpecified.Left:
                    // convert input value into work-area co-ods
                    double newLeft = newValue;
                    // [Get/Set]WindowPlacement work with workarea co-ods for a top level
                    // window whose WS_EX_TOOLWINDOW bit is clear.  If this bit is set,
                    // then the co-ods are expected to be in screen co-ods of the monitor.
                    // TransfromWorkAreaScreenArea can transform a point from work area co-ods
                    // to screen area co-ods and vice versa depending on TransformType value passed.
 
                    // So, in our case, if the window is not a ToolWindow we want to transform
                    // the input value from screen co-ods to work area co-ods.
                    if ((StyleExFromHwnd & NativeMethods.WS_EX_TOOLWINDOW) == 0)
                    {
                        newLeft = TransformWorkAreaScreenArea(new Point(newLeft, 0), TransformType.ScreenAreaToWorkArea).X;
                    }
                    newLeft = (LogicalToDeviceUnits(new Point(newLeft, 0))).X;
                    int currentWidth = wp.rcNormalPosition_right - wp.rcNormalPosition_left;
                    wp.rcNormalPosition_left = DoubleUtil.DoubleToInt(newLeft);
                    wp.rcNormalPosition_right = wp.rcNormalPosition_left + currentWidth;
                    break;
                default:
                    Debug.Assert(false, $"specifiedRestoreBounds can't be {specifiedRestoreBounds}");
                    break;
            }
 
            // The showCmd flag retreived by GetWindowPlacement is SW_SHOWMAXIMIZED when the window is maximized.
            // If the window is minimized, showCmd is SW_SHOWMINIMIZED. Otherwise, it is SW_SHOWNORMAL, regardless
            // of the window's visibility.
            // SetWindowPlacement with SW_SHOWMAXIMIZED and SW_SHOWMINIMIZED will cause a hidden window to show.
            // To workaround this issue, we check whether the current window is hidden and set showCmd to SW_HIDE if it is.
            if (!this._isVisible)
            {
                wp.showCmd = NativeMethods.SW_HIDE;
            }
 
 
            UnsafeNativeMethods.SetWindowPlacement(new HandleRef(this, Handle), ref wp);
        }
 
        // deltaX = workAreaOriginValue - screenOriginValue (both in virtual co-ods)
        // X(screenAreaCood) = x(workAreaCood) + deltaX
        private Point TransformWorkAreaScreenArea(Point pt, TransformType transformType)
        {
            int deltaX = 0;
            int deltaY = 0;
            Point retPt;
 
            // First we get the monitor on which the window is on.  [Get/Set]WindowPlacement
            // co-ods are dependent on the monitor on which the window is on.
            IntPtr hMonitor = SafeNativeMethods.MonitorFromWindow(new HandleRef(this, Handle), NativeMethods.MONITOR_DEFAULTTONULL);
 
            if (hMonitor != IntPtr.Zero)
            {
                NativeMethods.MONITORINFOEX monitorInfo = new NativeMethods.MONITORINFOEX
                {
                    cbSize = Marshal.SizeOf(typeof(NativeMethods.MONITORINFOEX))
                };
 
                SafeNativeMethods.GetMonitorInfo(new HandleRef(this, hMonitor), monitorInfo);
                NativeMethods.RECT workAreaRect = monitorInfo.rcWork;
                NativeMethods.RECT screenAreaRect = monitorInfo.rcMonitor;
                deltaX = workAreaRect.left - screenAreaRect.left;
                deltaY = workAreaRect.top - screenAreaRect.top;
            }
 
            if (transformType == TransformType.WorkAreaToScreenArea)
            {
                retPt = new Point(pt.X + deltaX, pt.Y + deltaY);
            }
            else
            {
                retPt = new Point(pt.X - deltaX, pt.Y - deltaY);
            }
            return retPt;
        }
 
        // The logic for coerce Top & Left
        // 1.   Before Window is first shown (w.IsSourceWindowNull == false)
        //      The value can come from 3 parties
        //          a. default
        //          b. SetValue
        //          c. Style, trigger...
        //      In all those 3 cases, we would like to pass the value because we don't have a
        //      different value from hwnd before it is shown.
        //
        // To understarnd the below 2 cases better, you will need to know the Window position API precedence as following:
        //      WindowState > WindowStartupLocation (only works the first time shown) > Top/Left
        //
        // 2.   During show
        //      There are 3 places that can coerce value during show. Right now they are all in
        //      SetupInitialState.
        //          a. After CreateWindowEx.
        //                  i. We should always update with the win32 default position when it is Nan.
        //                  ii. If WindowState is maxmized, we should always return from the hwnd (_actualTop/Left),
        //                  but update the hwnd restorebounds. So setting Top/Left WindowState to to Maxmized before show would
        //                  work for restorebounds (details in bug 1217802).
        //          b. If Top/Left is set and/or WindowStartupLocation is effective.
        //                  WindowState must be normal here because it takes precedence over WindowStartupLocation and Top/Left.
        //                  Since WindowStartupLocation only works the first time shown, we have a flag (_updateStartupLocation)
        //                  to help indicating that.
        //                  If StartupLocation is effective, we should return from the hwnd (_actualTop/Left).
        //          c. If SizeToContent is set and WindowStartupLocation is effective.
        //                  Same as b.
        //
        // 3.   After show
        //      a. User moves the Window.
        //          If user resize, we set local value (SetValue).
        //      b. User maximizes or minimizes. Or WindowState is changed programmtically
        //          We coerce Top and Left's value when WindowState is changed no matter whether it is
        //          from user action or programmtically.
        //      c. SetValue
        //
        //      For b and c, when WindowState is max or min, the hwnd value should always be returned.
        //      The new value should be set as restorebounds. Otherwise update with the new value.
        //
        // Note: as we can see from above logic, _actualTop/Left should always be updated with the current hwnd position before we
        // coerce Top and Left after hwnd is created.
        private static object CoerceTop(DependencyObject d, object value)
        {
            Window w = d as Window;
 
            // this call ends up throwing an exception if accessing Top
            // is not allowed
            w.VerifyApiSupported();
 
            double top = (double)value;
 
            // Move ValidateTopLeft calls from property
            // invalidation callback to PropertyMetadata
            ValidateTopLeft(top);
 
            if (w.IsSourceWindowNull || w.IsCompositionTargetInvalid)
            {
                return value;
            }
 
            if (double.IsNaN(top))
            {
                return w._actualTop;
            }
 
            if (w.WindowState != WindowState.Normal)
            {
                return value;
            }
 
            if (w._updateStartupLocation && (w.WindowStartupLocation != WindowStartupLocation.Manual))
            {
                return w._actualTop;
            }
 
            return value;
        }
 
        private static void _OnTopChanged (DependencyObject d , DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert( w != null, "DependencyObject must be of type Window." );
 
            if (w._updateHwndLocation)
            {
                w.OnTopChanged((double) e.NewValue);
            }
        }
 
        private void OnTopChanged(double newTop)
        {
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                // NaN is special and indicates using Win32 default,
                // so we exclude that.
                if (double.IsNaN(newTop) == false)
                {
                    if (WindowState == WindowState.Normal)
                    {
                        Invariant.Assert(!Double.IsNaN(_actualLeft), "_actualLeft cannot be NaN after show");
                        UpdateHwndPositionOnTopLeftChange(Double.IsNaN(Left) ? _actualLeft : Left, newTop);
                    }
                    else
                    {
                        UpdateHwndRestoreBounds(newTop, BoundsSpecified.Top);
                    }
                }
            }
            else
            {
                // here the value is stored as measure units as newTop is in measure/logical units
                _actualTop = newTop;
            }
        }
 
        // Please see comments for CoerceTop.
        private static object CoerceLeft(DependencyObject d, object value)
        {
            Window w = d as Window;
 
            // this call ends up throwing an exception if setting property is not allowed
            w.VerifyApiSupported();
 
            double left = (double)value;
 
            // Move ValidateTopLeft calls from property
            // invalidation callback to PropertyMetadata
            ValidateTopLeft(left);
 
            if (w.IsSourceWindowNull || w.IsCompositionTargetInvalid)
            {
                return value;
            }
 
            if (double.IsNaN(left))
            {
                return w._actualLeft;
            }
 
            if (w.WindowState != WindowState.Normal)
            {
                return value;
            }
 
            if (w._updateStartupLocation && (w.WindowStartupLocation != WindowStartupLocation.Manual))
            {
                return w._actualLeft;
            }
 
            return value;
        }
 
        private static void _OnLeftChanged (DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert( w != null, "DependencyObject must be of type Window." );
 
            if (w._updateHwndLocation)
            {
                w.OnLeftChanged((double) e.NewValue);
            }
        }
 
        // _actualLeft is used to determine if LocationChanged should be fired in WMMoveChagnged.
        // We need it b/c we need to remember the last hwnd Left location to decide whether
        // we need to fire the event or not.  Why do we need to update here?  Well, for the following
        // scenario:
        //    Window w = new Window();
        //    w.Left = 100;
        //    w.WindowStyle = WindowStyle.None;
        //    w.Show();
        //
        //  In this case, we want to not fire LocationChanged from SetWindowPos called called
        //  from CorrectStyleForBorderlessWindowCase().
        //
        //  _actualLeft is update from the following places:
        //
        //
        // 1) In WM_MOVE handler
        //
        // 2) In OnLeftChanged for the case when the hwnd is not created yet.
        //
        // 3) SetupInitialState
        private void OnLeftChanged(double newLeft)
        {
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                // NaN is special and indicates using Win32 default,
                // so we exclude that here.
                if (double.IsNaN(newLeft) == false)
                {
                    if (WindowState == WindowState.Normal)
                    {
                        Invariant.Assert(!Double.IsNaN(_actualTop), "_actualTop cannot be NaN after show");
                        UpdateHwndPositionOnTopLeftChange(newLeft, Double.IsNaN(Top) ? _actualTop : Top);
                    }
                    else
                    {
                        UpdateHwndRestoreBounds(newLeft, BoundsSpecified.Left);
                    }
                }
            }
            else
            {
                // here the value is stored as measure units as newLeft is in measure/logical units
                _actualLeft = newLeft;
            }
        }
 
        private void UpdateHwndPositionOnTopLeftChange(double leftLogicalUnits, double topLogicalUnits)
        {
            Debug.Assert( IsSourceWindowNull == false , "IsSourceWindowNull cannot be true when calling this function");
 
            Point ptDeviceUnits = LogicalToDeviceUnits(new Point(leftLogicalUnits, topLogicalUnits));
 
            UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle),
                        new HandleRef(null, IntPtr.Zero),
                        DoubleUtil.DoubleToInt(ptDeviceUnits.X),
                        DoubleUtil.DoubleToInt(ptDeviceUnits.Y),
                        0,
                        0,
                        NativeMethods.SWP_NOSIZE | NativeMethods.SWP_NOZORDER | NativeMethods.SWP_NOACTIVATE
                        );
        }
 
        private static bool _ValidateResizeModeCallback(object value)
        {
            return IsValidResizeMode((ResizeMode)value);
        }
 
        private static void _OnResizeModeChanged (DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert( w != null, "DependencyObject must be of type Window." );
 
            w.OnResizeModeChanged();
        }
 
        private void OnResizeModeChanged()
        {
            // this call ends up throwing an exception if accessing
            // ResizeMode is not allowed
            VerifyApiSupported();
 
            // Adding check for IsCompositionTargetInvalid
            if ( IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                using (HwndStyleManager sm = HwndStyleManager.StartManaging(this, StyleFromHwnd, StyleExFromHwnd  ))
                {
                    CreateResizibility();
                }
            }
        }
 
        private static object VerifyAccessCoercion(DependencyObject d, object value)
        {
            // this call ends up throwing an exception if setting property is not allowed
            ((Window)d).VerifyApiSupported();
 
            return value;
        }
 
        private static void _OnFlowDirectionChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            Window w = d as Window;
            Debug.Assert(w != null, "DependencyObject must be of type Window.");
 
            w.OnFlowDirectionChanged();
        }
 
        /// <summary>
        ///     Right to Left
        /// </summary>
        private void OnFlowDirectionChanged()
        {
            // Adding check for IsCompositionTargetInvalid
            if (IsSourceWindowNull == false && IsCompositionTargetInvalid == false)
            {
                using (HwndStyleManager sm = HwndStyleManager.StartManaging(this, StyleFromHwnd, StyleExFromHwnd ))
                {
                    CreateRtl();
                }
            }
        }
 
        private static object CoerceRenderTransform(DependencyObject d, object value)
        {
            Transform renderTransformValue = (Transform)value;
 
            if ((value == null) ||
                (renderTransformValue != null && renderTransformValue.Value.IsIdentity == true))
            {
                // setting this value is allowed.
            }
            else
            {
                throw new InvalidOperationException(SR.TransformNotSupported);
            }
 
            return value;
        }
 
        private static void _OnRenderTransformChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
        }
 
        private static object CoerceClipToBounds(DependencyObject d, object value)
        {
            if ((bool)value != false)
            {
                throw new InvalidOperationException(SR.ClipToBoundsNotSupported);
            }
            return value;
        }
 
        private static void _OnClipToBoundsChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
        }
 
 
        /// <summary>
        ///     Get or create the hidden window used for parenting when ShowInTaskbar == false.
        /// </summary>
        private HwndWrapper EnsureHiddenWindow()
        {
            if (_hiddenWindow == null)
            {
                _hiddenWindow = new HwndWrapper(
                    0, // classStyle
                    NativeMethods.WS_OVERLAPPEDWINDOW, // style
                    0, // exStyle
                    NativeMethods.CW_USEDEFAULT, // x
                    NativeMethods.CW_USEDEFAULT, // y
                    NativeMethods.CW_USEDEFAULT, // width
                    NativeMethods.CW_USEDEFAULT, // height
                    "Hidden Window", // name
                    IntPtr.Zero,
                    null
                );
            }
 
            return _hiddenWindow;
        }
 
        /// <summary>
        ///     sets taskbar status
        /// </summary>
        private void SetTaskbarStatus()
        {
            if (ShowInTaskbar == false) // don't show in taskbar
            {
                // To remove the taskbar button for this window it needs to have a non-null parent
                // (we'll create a hidden window for this purpose) and not have WS_EX_APPWINDOW
 
                // Create this now, even if we're not currently going to parent it.
                // If the Owner changes, we'll need to switch to this.
                EnsureHiddenWindow();
 
                // when this window is unowned
                if (_ownerHandle == IntPtr.Zero)
                {
                    SetOwnerHandle(_hiddenWindow.Handle);
 
                    // When we do this parenting trick on XP the alt-tab icon for the window takes on
                    // the parent's icon.  To keep things working right we need to apply Icon to the
                    // hidden window.  On Vista this is redundant.
                    if (!(IsSourceWindowNull || IsCompositionTargetInvalid))
                    {
                        UpdateIcon();
                    }
                }
 
                _StyleEx &= ~NativeMethods.WS_EX_APPWINDOW;
            }
            else // (ShowInTaskbar == true) show in task bar
            {
                _StyleEx |= NativeMethods.WS_EX_APPWINDOW;
                if( ! IsSourceWindowNull )
                {
                    if ((_hiddenWindow != null) && (_ownerHandle == _hiddenWindow.Handle))
                    {
                        SetOwnerHandle(IntPtr.Zero);
                    }
                }
            }
        }
 
        private void OnTaskbarRetryTimerTick(object sender, EventArgs e)
        {
            UnsafeNativeMethods.PostMessage(new HandleRef(this, Handle), WM_APPLYTASKBARITEMINFO, IntPtr.Zero, IntPtr.Zero);
        }
 
        private void ApplyTaskbarItemInfo()
        {
            if (!Utilities.IsOSWindows7OrNewer)
            {
                if (TraceShell.IsEnabled)
                {
                    TraceShell.Trace(TraceEventType.Warning, TraceShell.NotOnWindows7);
                }
                return;
            }
 
            // If the Window hasn't yet been shown then these calls will fail.
            // We'll try this again when the WM_TASKBARBUTTONCREATED message gets sent.
            if (IsSourceWindowNull || IsCompositionTargetInvalid)
            {
                return;
            }
 
            // If the taskbar has timed out in the last minute, don't try to do this again.
            if (_taskbarRetryTimer != null && _taskbarRetryTimer.IsEnabled)
            {
                return;
            }
 
            HRESULT hr = HRESULT.S_OK;
            if (_taskbarList == null)
            {
                // If we don't have a handle and there isn't a TaskbarItemInfo, then we don't have anything to apply or remove.
                if (TaskbarItemInfo == null)
                {
                    return;
                }
 
                ITaskbarList taskbarList = null;
                try
                {
                    taskbarList = (ITaskbarList)Activator.CreateInstance(Type.GetTypeFromCLSID(new Guid(CLSID.TaskbarList)));
 
                    hr = taskbarList.HrInit();
                    if (hr != HRESULT.S_OK)
                    {
                        // Taskbar not available (no user logged in, running under terminal service, custom shell, etc.)
                        HandleTaskbarListError(hr);
                        return;
                    }
 
                    // This QI will only work on Win7.
                    _taskbarList = (ITaskbarList3)taskbarList;
                    taskbarList = null;
                }
                finally
                {
                    Utilities.SafeRelease(ref taskbarList);
                }
 
                // Don't use SystemParameters here.  We need pixels.
                _overlaySize = new Size(
                    UnsafeNativeMethods.GetSystemMetrics(SM.CXSMICON),
                    UnsafeNativeMethods.GetSystemMetrics(SM.CYSMICON));
                // IsEmpty is inclusive, in case either of these succeeded it would pass...
                Debug.Assert(0 != (int)_overlaySize.Width);
                Debug.Assert(0 != (int)_overlaySize.Height);
 
                // When we detect that Explorer is nonresponsive, use this to defer further attempts
                // at updating the TaskbarList until a specified amount of time (1 minute)
                if (_taskbarRetryTimer == null)
                {
                    _taskbarRetryTimer = new DispatcherTimer { Interval = new TimeSpan(0, 1, 0) };
                    // Explorer being non-responsive should be a transient issue.  Post back to apply the full TaskbarItemInfo.
                    _taskbarRetryTimer.Tick += OnTaskbarRetryTimerTick;
                }
            }
 
            // Apply (or clear) all aspects of the TaskbarItemInfo to this Window.
            hr = RegisterTaskbarThumbButtons();
 
            if (hr.Succeeded)
            {
                // Updating the state will also update the value, so don't call UpdateTaskbarProgressValue.
                hr = UpdateTaskbarProgressState();
            }
            if (hr.Succeeded)
            {
                hr = UpdateTaskbarOverlay();
            }
            if (hr.Succeeded)
            {
                hr = UpdateTaskbarDescription();
            }
            if (hr.Succeeded)
            {
                hr = UpdateTaskbarThumbnailClipping();
            }
            if (hr.Succeeded)
            {
                hr = UpdateTaskbarThumbButtons();
            }
 
            // We'll asynchronously retry if we failed initially setting up the TaskbarItemInfo.
            HandleTaskbarListError(hr);
        }
 
        private HRESULT UpdateTaskbarProgressState()
        {
            Debug.Assert(null != _taskbarList);
 
            TaskbarItemInfo taskbarInfo = TaskbarItemInfo;
            TBPF tbpf = TBPF.NOPROGRESS;
 
            if (taskbarInfo != null)
            {
                switch (taskbarInfo.ProgressState)
                {
                    case TaskbarItemProgressState.Error:
                        tbpf = TBPF.ERROR;
                        break;
                    case TaskbarItemProgressState.Indeterminate:
                        tbpf = TBPF.INDETERMINATE;
                        break;
                    case TaskbarItemProgressState.None:
                        tbpf = TBPF.NOPROGRESS;
                        break;
                    case TaskbarItemProgressState.Normal:
                        tbpf = TBPF.NORMAL;
                        break;
                    case TaskbarItemProgressState.Paused:
                        tbpf = TBPF.PAUSED;
                        break;
                    default:
                        // The coercion should have caught this.
                        Debug.Assert(false);
                        tbpf = TBPF.NOPROGRESS;
                        break;
                }
            }
 
            HRESULT hr = _taskbarList.SetProgressState(Handle, tbpf);
            if (hr.Succeeded)
            {
                // Explicitly update this in case this property being set
                // to None or Indeterminate before made the value not update.
                hr = UpdateTaskbarProgressValue();
            }
 
            return hr;
        }
 
        private HRESULT UpdateTaskbarProgressValue()
        {
            Debug.Assert(null != _taskbarList);
 
            TaskbarItemInfo taskbarInfo = TaskbarItemInfo;
 
            // If we're not attached then don't modify this.
            if (taskbarInfo == null
                || taskbarInfo.ProgressState == TaskbarItemProgressState.None
                || taskbarInfo.ProgressState == TaskbarItemProgressState.Indeterminate)
            {
                return HRESULT.S_OK;
            }
 
            const ulong precisionValue = 1000;
            // The coercion should enforce this.
            Debug.Assert(0 <= taskbarInfo.ProgressValue && taskbarInfo.ProgressValue <= 1);
 
            var intValue = (ulong)(taskbarInfo.ProgressValue * precisionValue);
            return _taskbarList.SetProgressValue(Handle, intValue, precisionValue);
        }
 
        private HRESULT UpdateTaskbarOverlay()
        {
            Debug.Assert(null != _taskbarList);
 
            TaskbarItemInfo taskbarInfo = TaskbarItemInfo;
            NativeMethods.IconHandle hicon = NativeMethods.IconHandle.GetInvalidIcon();
 
            // The additional string at the end of SetOverlayIcon sets the accDescription
            // for screen readers.  We don't currently have a property that utilizes this.
 
            try
            {
                if (null != taskbarInfo && null != taskbarInfo.Overlay)
                {
                    hicon = IconHelper.CreateIconHandleFromImageSource(taskbarInfo.Overlay, _overlaySize);
                }
 
                return _taskbarList.SetOverlayIcon(Handle, hicon, null);
            }
            finally
            {
                hicon.Dispose();
            }
        }
 
        private HRESULT UpdateTaskbarDescription()
        {
            Debug.Assert(null != _taskbarList);
 
            TaskbarItemInfo taskbarInfo = TaskbarItemInfo;
            string tooltip = "";
 
            if (taskbarInfo != null)
            {
                tooltip = taskbarInfo.Description ?? "";
            }
 
            return _taskbarList.SetThumbnailTooltip(Handle, tooltip);
        }
 
 
        private HRESULT UpdateTaskbarThumbnailClipping()
        {
            // If TaskbarItemInfo isn't attached and active then there's nothing to do here.
            if (_taskbarList == null)
            {
                return HRESULT.S_OK;
            }
 
            if (_taskbarRetryTimer != null && _taskbarRetryTimer.IsEnabled)
            {
                // Explorer appears to be non-responsive.  Don't try this.
                return HRESULT.S_FALSE;
            }
 
            // Don't count on Window properties being in sync at the time of this call.
            // Just use native methods to check
            if (UnsafeNativeMethods.IsIconic(Handle))
            {
                // If the window is minimized then don't try to update the clip.
                return HRESULT.S_FALSE;
            }
 
            TaskbarItemInfo taskbarInfo = TaskbarItemInfo;
            NativeMethods.RefRECT interopRc = null;
 
            // If the taskbarInfo isn't available then remove any clipping.
            if (taskbarInfo != null && !taskbarInfo.ThumbnailClipMargin.IsZero)
            {
                Thickness margin = taskbarInfo.ThumbnailClipMargin;
                // Use the native GetClientRect.  Window.ActualWidth and .ActualHeight include the non-client areas.
                NativeMethods.RECT physicalClientRc = default(NativeMethods.RECT);
                SafeNativeMethods.GetClientRect(new HandleRef(this, Handle), ref physicalClientRc);
                var logicalClientRc = new Rect(
                    DeviceToLogicalUnits(new Point(physicalClientRc.left, physicalClientRc.top)),
                    DeviceToLogicalUnits(new Point(physicalClientRc.right, physicalClientRc.bottom)));
 
                // Crop the clipping to ensure that the margin doesn't overlap itself.
                if (margin.Left + margin.Right >= logicalClientRc.Width
                    || margin.Top + margin.Bottom >= logicalClientRc.Height)
                {
                    // Empty client rectangle is different than no clipping applied.
                    interopRc = new NativeMethods.RefRECT(0, 0, 0, 0);
                }
                else
                {
                    var physicalClip = new Rect(
                        LogicalToDeviceUnits(new Point(margin.Left, margin.Top)),
                        LogicalToDeviceUnits(new Point(logicalClientRc.Width - margin.Right, logicalClientRc.Height - margin.Bottom)));
                    interopRc = new NativeMethods.RefRECT((int)physicalClip.Left, (int)physicalClip.Top, (int)physicalClip.Right, (int)physicalClip.Bottom);
                }
            }
 
            return _taskbarList.SetThumbnailClip(Handle, interopRc);
        }
 
        private HRESULT RegisterTaskbarThumbButtons()
        {
            Debug.Assert(null != _taskbarList);
 
            // The ITaskbarList3 API requires that the maximum number of buttons to ever be used
            // are registered at the beginning.  Modifications can be made to this list later.
            var nativeButtons = new THUMBBUTTON[c_MaximumThumbButtons];
 
            for (int i = 0; i < c_MaximumThumbButtons; ++i)
            {
                nativeButtons[i] = new THUMBBUTTON
                {
                    iId = (uint)i,
                    dwFlags = THBF.NOBACKGROUND | THBF.DISABLED | THBF.HIDDEN | THBF.NONINTERACTIVE,
                    dwMask = THB.FLAGS | THB.ICON | THB.TOOLTIP
                };
            }
 
            // If this gets called (successfully) more than once it usually returns E_INVALIDARG.  It's not really
            // a failure and we potentially want to retry this operation.
            HRESULT hr = _taskbarList.ThumbBarAddButtons(Handle, (uint)nativeButtons.Length, nativeButtons);
            if (hr == HRESULT.E_INVALIDARG)
            {
                hr = HRESULT.S_FALSE;
            }
            return hr;
        }
 
        private HRESULT UpdateTaskbarThumbButtons()
        {
            Debug.Assert(null != _taskbarList);
 
            var nativeButtons = new THUMBBUTTON[c_MaximumThumbButtons];
            TaskbarItemInfo taskbarInfo = TaskbarItemInfo;
            ThumbButtonInfoCollection thumbButtons = null;
 
            if (taskbarInfo != null)
            {
                thumbButtons = taskbarInfo.ThumbButtonInfos;
            }
 
            var nativeIcons = new List<NativeMethods.IconHandle>();
 
            try
            {
                uint currentButton = 0;
                if (null != thumbButtons)
                {
                    foreach (ThumbButtonInfo wrappedTB in thumbButtons)
                    {
                        var nativeTB = new THUMBBUTTON
                        {
                            iId = (uint)currentButton,
                            dwMask = THB.FLAGS | THB.TOOLTIP | THB.ICON,
                        };
 
                        switch (wrappedTB.Visibility)
                        {
                            case Visibility.Collapsed:
                                // HIDDEN removes the button from layout logic.
                                nativeTB.dwFlags = THBF.HIDDEN;
                                break;
 
                            case Visibility.Hidden:
                                // To match WPF's notion of hidden, we want this not HIDDEN
                                // but disabled, without background, and without icon.
                                nativeTB.dwFlags = THBF.DISABLED | THBF.NOBACKGROUND;
                                nativeTB.hIcon = IntPtr.Zero;
 
                                break;
                            default:
                            case Visibility.Visible:
 
                                nativeTB.szTip = wrappedTB.Description ?? "";
                                if (wrappedTB.ImageSource != null)
                                {
                                    NativeMethods.IconHandle nativeIcon = IconHelper.CreateIconHandleFromImageSource(wrappedTB.ImageSource, _overlaySize);
                                    nativeTB.hIcon = nativeIcon.CriticalGetHandle();
                                    nativeIcons.Add(nativeIcon);
                                }
 
                                if (!wrappedTB.IsBackgroundVisible)
                                {
                                    nativeTB.dwFlags |= THBF.NOBACKGROUND;
                                }
 
                                if (!wrappedTB.IsEnabled)
                                {
                                    nativeTB.dwFlags |= THBF.DISABLED;
                                }
                                else
                                {
                                    nativeTB.dwFlags |= THBF.ENABLED;
                                }
 
                                // This is separate from enabled/disabled
                                if (!wrappedTB.IsInteractive)
                                {
                                    nativeTB.dwFlags |= THBF.NONINTERACTIVE;
                                }
 
                                if (wrappedTB.DismissWhenClicked)
                                {
                                    nativeTB.dwFlags |= THBF.DISMISSONCLICK;
                                }
                                break;
                        }
 
                        nativeButtons[currentButton] = nativeTB;
 
                        ++currentButton;
                        if (currentButton == c_MaximumThumbButtons)
                        {
                            break;
                        }
                    }
                }
 
                // If we're not attached, or the list is less than the maximum number of buttons
                // then fill in the rest with collapsed, empty buttons.
                for (; currentButton < c_MaximumThumbButtons; ++currentButton)
                {
                    nativeButtons[currentButton] = new THUMBBUTTON
                    {
                        iId = (uint)currentButton,
                        dwFlags = THBF.NOBACKGROUND | THBF.DISABLED | THBF.HIDDEN,
                        dwMask = THB.FLAGS | THB.ICON | THB.TOOLTIP
                    };
                }
 
                // Finally, apply the update.
                return _taskbarList.ThumbBarUpdateButtons(Handle, (uint)nativeButtons.Length, nativeButtons);
            }
            finally
            {
                foreach (var icon in nativeIcons)
                {
                    icon.Dispose();
                }
            }
        }
 
        private void CreateRtl()
        {
            if ( this.FlowDirection == FlowDirection.LeftToRight )
            {
                _StyleEx &= ~NativeMethods.WS_EX_LAYOUTRTL;
            }
            else if ( this.FlowDirection == FlowDirection.RightToLeft )
            {
                _StyleEx |= NativeMethods.WS_EX_LAYOUTRTL;
            }
            else
            {
                throw new InvalidOperationException(SR.IncorrectFlowDirection);
            }
}
 
        /// <summary>
        ///     Updates both style and styleEx for the window
        ///
        ///     WCP: Figure out how to update window styles after calling SetWindowLong
        ///     Currently hides the window and then shows it to make it update. Have to
        ///     find a better way of doing this.
        /// </summary>
        internal void Flush()
        {
            // (A NullReferenceException occurs when animating
            // the WindowStyle enum via a custom animation).  This bug contains details of
            // why we were seeing the null ref.
            //
            // Sometimes, the SetWindowPos call below results in sending certain window messages
            // like (WM_SIZE) and their handling leads to setting some property on the Window leading
            // to a call to HwndStyleManager.StartManaging.  Thus, we end up calling
            // dispose on the "new" usage of the Manager before we complete this run of Flush method.
            // This resulted in null ref in setting the Dirty bit below since we were not using
            // a local copy and the window member copy was already set to null by the "new" usage
            // of Manager.  To fix this bug, we do the following two things:
            //
            // 1) Keep a local copy of HwndStyleManager in this method to make it re-entrant.
            // 2) null out _window.Manager in HwndStyleMangager.Dispose only if _window.Manager is
            //    this instance of the Manager.
            //
            HwndStyleManager manager = Manager;
            if (manager.Dirty && Handle != IntPtr.Zero)
            {
                UnsafeNativeMethods.CriticalSetWindowLong(new HandleRef(this,Handle), NativeMethods.GWL_STYLE, (IntPtr)_styleDoNotUse);
                UnsafeNativeMethods.CriticalSetWindowLong(new HandleRef(this,Handle), NativeMethods.GWL_EXSTYLE, (IntPtr)_styleExDoNotUse);
 
                UnsafeNativeMethods.SetWindowPos(new HandleRef(this, Handle), NativeMethods.NullHandleRef, 0, 0, 0, 0,
                                               NativeMethods.SWP_NOMOVE |
                                               NativeMethods.SWP_NOSIZE |
                                               NativeMethods.SWP_NOZORDER |
                                               NativeMethods.SWP_FRAMECHANGED |
                                               NativeMethods.SWP_DRAWFRAME |
                                               NativeMethods.SWP_NOACTIVATE);
                manager.Dirty = false;
            }
        }
        private void ClearRootVisual()
        {
            if (_swh != null)
            {
                _swh.ClearRootVisual();
            }
        }
 
 
        private NativeMethods.POINT GetPointRelativeToWindow( int x, int y )
        {
            return _swh.GetPointRelativeToWindow( x, y, this.FlowDirection);
        }
 
        //     If you're in the middle of changing the window's _style or _styleEx and call this function,
        //     you may get inconsistent results.
        private Size GetHwndNonClientAreaSizeInMeasureUnits()
        {
            // HwndSource expands the client area to cover the entire window when it is in UsesPerPixelOpacity mode,
            // So non client area is (0,0)
            return AllowsTransparency ? new Size(0, 0) : _swh.GetHwndNonClientAreaSizeInMeasureUnits();
        }
 
        private void ClearSourceWindow()
        {
            if (_swh != null)
            {
                try
                {
                    _swh.RemoveDisposedHandler(OnSourceWindowDisposed);
                }
                finally
                {
                    HwndSource source = _swh.HwndSourceWindow;
                    _swh = null;
 
                    if (source != null)
                    {
                        source.SizeToContentChanged -= new EventHandler(OnSourceSizeToContentChanged);
                    }
                }
            }
        }
 
        private void ClearHiddenWindowIfAny()
        {
            // If there is a hiddenWindow and it's the owner of the current one as the result of setting ShowInTaskbar,
            // we need to unparent it. Because when we dipose the hiddenWindow, if it's still the parent
            // of the current window, the current Window will get a second WM_DESTORY because its owner being distoried.
            // See detail in bug 953988.
            // Unparent it in WM_CLOSE because when we get to WM_DESTROY, _sourceWindow.Handle could be IntPtr.Zero;
            // InternalDispose() is where _hiddenWindow is disposed. It could be called from two places: 1. WmDestroy 2. OnSourceWindowDisposed.
            // When it's called from OnSourceWindowDisposed, _sourceWindow.Handle could have been set to IntPtr.Zero.
            if ((_hiddenWindow != null) && (_hiddenWindow.Handle == _ownerHandle))
            {
                SetOwnerHandle(IntPtr.Zero);
            }
        }
 
        private void VerifyConsistencyWithAllowsTransparency()
        {
            if (AllowsTransparency)
            {
                VerifyConsistencyWithAllowsTransparency(WindowStyle);
            }
        }
 
        private void VerifyConsistencyWithAllowsTransparency(WindowStyle style)
        {
            if (AllowsTransparency && style != WindowStyle.None)
            {
                throw new InvalidOperationException(SR.MustUseWindowStyleNone);
            }
        }
 
        private void VerifyConsistencyWithShowActivated()
        {
            //
            // We don't support to show a maximized non-activated window.
            // Don't check this consistency in a RBW (would break because Visibility is set when launching the RBW).
            //
            if (!_inTrustedSubWindow && WindowState == WindowState.Maximized && !ShowActivated)
                throw new InvalidOperationException(SR.ShowNonActivatedAndMaximized);
        }
 
        private static bool IsValidSizeToContent(SizeToContent value)
        {
            return value == SizeToContent.Manual ||
                   value == SizeToContent.Width ||
                   value == SizeToContent.Height ||
                   value == SizeToContent.WidthAndHeight;
        }
 
        private static bool IsValidResizeMode(ResizeMode value)
        {
            return value == ResizeMode.NoResize
                || value == ResizeMode.CanMinimize
                || value == ResizeMode.CanResize
                || value == ResizeMode.CanResizeWithGrip;
        }
 
        private static bool IsValidWindowStartupLocation(WindowStartupLocation value)
        {
            return value == WindowStartupLocation.CenterOwner
                || value == WindowStartupLocation.CenterScreen
                || value == WindowStartupLocation.Manual;
        }
 
        private static bool IsValidWindowState(WindowState value)
        {
            return value == WindowState.Maximized
                || value == WindowState.Minimized
                || value == WindowState.Normal;
        }
 
        private static bool IsValidWindowStyle(WindowStyle value)
        {
            return value == WindowStyle.None
                || value == WindowStyle.SingleBorderWindow
                || value == WindowStyle.ThreeDBorderWindow
                || value == WindowStyle.ToolWindow;
        }
 
        #endregion Private Methods
 
        #region Manipulation Boundary Feedback
 
        /// <summary>
        ///     Provides feedback when a manipulation has encountered a boundary by nudging the window.
        /// </summary>
        /// <remarks>
        ///     PanningFeedback is reported on Handle of the Window using Begin/Update/EndPanningFeedback
        ///     APIs. When the Handle reported to these APIs is not of a top-level window, these APIs
        ///     do a GetAncestor call to get the top-level window and apply the effect of it. For eg.
        ///     in the case of XBAPs we report the feedback to the Handle of RootBrowserWindow which is
        ///     a child window, but still the panning feedback effect gets applied to the browser window
        ///     itself. Security wise this also assumes that PanningFeedback will not move the Window
        ///     by an arbitrary distance.
        /// </remarks>
        protected override void OnManipulationBoundaryFeedback(ManipulationBoundaryFeedbackEventArgs e)
        {
            base.OnManipulationBoundaryFeedback(e);
 
            // If the original source is not from the same PresentationSource as of the Window,
            // then do not act on the PanningFeedback.
            if (!PresentationSource.IsUnderSamePresentationSource(e.OriginalSource as DependencyObject, this))
            {
                return;
            }
 
            if (!e.Handled)
            {
                if (_currentPanningTarget == null ||
                    !_currentPanningTarget.IsAlive ||
                    _currentPanningTarget.Target != e.OriginalSource)
                {
                    if (_swh != null)
                    {
                        // Cache location if starting the panning feedback.
                        // Using SourceWindowHelper.WindowBounds instead of Window.Left
                        // and Window.Top so that the implementation works with
                        // RootBrowserWindow.
                        NativeMethods.RECT rc = WindowBounds;
                        _prePanningLocation = DeviceToLogicalUnits(new Point(rc.left, rc.top));
                    }
                }
                ManipulationDelta manipulation = e.BoundaryFeedback;
                UpdatePanningFeedback(manipulation.Translation, e.OriginalSource);
                e.CompensateForBoundaryFeedback = CompensateForPanningFeedback;
            }
        }
 
        private static void OnStaticManipulationInertiaStarting(object sender, ManipulationInertiaStartingEventArgs e)
        {
            Window window = sender as Window;
            if (window != null)
            {
                // Transitioning from direct manipulation to inertia, animate the window
                // back to its original position.
                window.EndPanningFeedback(true);
            }
        }
 
        private static void OnStaticManipulationCompleted(object sender, ManipulationCompletedEventArgs e)
        {
            Window window = sender as Window;
            if (window != null)
            {
                // A complete was encountered. If this was a forced complete, snap the window
                // back to its original position.
                window.EndPanningFeedback(false);
            }
        }
 
        /// <summary>
        ///     Provides feedback by nudging the window.
        /// </summary>
        private void UpdatePanningFeedback(Vector totalOverpanOffset, object originalSource)
        {
            if ((_currentPanningTarget != null) && !_currentPanningTarget.IsAlive)
            {
                // The old source is gone, end any pending feedback
                _currentPanningTarget = null;
                EndPanningFeedback(false);
            }
 
            if (_swh != null)
            {
                if (_currentPanningTarget == null)
                {
                    // Provide feedback for this source's events
                    _currentPanningTarget = new WeakReference(originalSource);
                }
 
                // Only provide feedback for one source at a time
                if (originalSource == _currentPanningTarget.Target)
                {
                    // Update the window position
                    _swh.UpdatePanningFeedback(totalOverpanOffset, false);
                }
            }
        }
 
        /// <summary>
        ///     Returns the window to its original position.
        /// </summary>
        private void EndPanningFeedback(bool animateBack)
        {
            if (_swh != null)
            {
                // Restore the window to its original position
                _swh.EndPanningFeedback(animateBack);
            }
            _currentPanningTarget = null;
            _prePanningLocation = new Point(double.NaN, double.NaN);
        }
 
        /// <summary>
        ///     Method to compensate a point for PanningFeedback.
        /// </summary>
        Point CompensateForPanningFeedback(Point point)
        {
            if (!double.IsNaN(_prePanningLocation.X) && !double.IsNaN(_prePanningLocation.Y) && (_swh != null))
            {
                // transform the point to pre pan coordinate system
                NativeMethods.RECT rc = WindowBounds;
                Point windowLocation = DeviceToLogicalUnits(new Point(rc.left, rc.top));
                return new Point(point.X - (_prePanningLocation.X - windowLocation.X), point.Y - (_prePanningLocation.Y - windowLocation.Y));
            }
            return point;
        }
 
        #endregion
 
        //----------------------------------------------
        //
        // Private Properties
        //
        //----------------------------------------------
        #region Private Properties
 
        private SizeToContent HwndSourceSizeToContent
        {
            get
            {
                return _swh.HwndSourceSizeToContent;
            }
            set
            {
                _swh.HwndSourceSizeToContent = value;
            }
        }
 
        private NativeMethods.RECT WindowBounds
        {
            get
            {
                Debug.Assert( _swh != null );
                return _swh.WindowBounds;
            }
        }
 
        private int StyleFromHwnd
        {
            get
            {
                return _swh != null ? _swh.StyleFromHwnd : 0;
            }
        }
 
        private int StyleExFromHwnd
        {
            get
            {
                return _swh != null ? _swh.StyleExFromHwnd : 0;
            }
        }
 
        /// <summary>
        ///     Private helper for OwnedWindows property.
        ///     The public version returns a copy of the WindowCollection.
        ///     For internal Window usage, we use OwnedWindowInternal
        ///     so that we can modify the underlying collection.
        /// </summary>
        private WindowCollection OwnedWindowsInternal
        {
            get
            {
                if (_ownedWindows == null)
                {
                    _ownedWindows = new WindowCollection();
                }
                return _ownedWindows;
            }
        }
 
        /// <summary>
        ///     Application Instance
        /// </summary>
        private System.Windows.Application App
        {
            get {return System.Windows.Application.Current;}
        }
 
        /// <summary>
        ///     Tells whether the Application object exists or not
        /// </summary>
        private bool IsInsideApp
        {
            get
            {
                return (Application.Current != null);
            }
        }
 
        /// <summary>
        ///     List of events on this Window
        /// </summary>
        private EventHandlerList Events
        {
            get
            {
                if (_events == null)
                {
                    _events = new EventHandlerList();
                }
                return _events;
            }
        }
 
        #endregion Private Properties
 
 
        //----------------------------------------------
        //
        // Private Fields
        //
        //----------------------------------------------
        #region Private Fields
 
        private SourceWindowHelper  _swh;                               // object that will hold the window
        private Window              _ownerWindow;                       // owner window
        private bool _reloadFluentDictionary = false;
        private bool _resourcesInitialized = false;
 
        // keeps track of the owner hwnd
        // we need this one b/c a owner/parent
        // can be set through the WindowInteropHandler
        // which is different than the owner Window object
        private IntPtr              _ownerHandle = IntPtr.Zero;   // no need to dispose this
        private WindowCollection    _ownedWindows;
        private List<IntPtr>        _threadWindowHandles;
 
        private bool                _updateHwndSize     = true;
        private bool                _updateHwndLocation = true;
        private bool                _updateStartupLocation;
        private bool                _isVisible;
        private bool                _isVisibilitySet;           // use this to tell whether Visibility is set or not.
        private bool                _resetKeyboardCuesProperty; // true if we set ShowKeyboradCuesProperty in ShowDialog
        private bool                _previousKeyboardCuesProperty;
 
        private static bool         _dialogCommandAdded;
        private bool                _postContentRenderedFromLoadedHandler;
 
        // WCP Window:  Define the dispose behavior for Window
        private bool                _disposed;
 
        private bool                _appShuttingDown;
        private bool                _ignoreCancel;
        private bool                _showingAsDialog;
        private bool                _isClosing;
        private bool                _visibilitySetInternally;
 
        // The hwnd can be created before window is shown via WindowInteropHelper.EnsureHandle.
        private bool                _hwndCreatedButNotShown;
 
        private double              _trackMinWidthDeviceUnits = 0;
        private double              _trackMinHeightDeviceUnits = 0;
        private double              _trackMaxWidthDeviceUnits = Double.PositiveInfinity;
        private double              _trackMaxHeightDeviceUnits = Double.PositiveInfinity;
        private double              _windowMaxWidthDeviceUnits = Double.PositiveInfinity;
        private double              _windowMaxHeightDeviceUnits = Double.PositiveInfinity;
 
        private double              _actualTop = Double.NaN;
 
        private double              _actualLeft = Double.NaN;
 
 
        private ThemeMode           _themeMode = ThemeMode.None;
        internal bool               _deferThemeLoading = false;
        private bool                _useDarkMode = false;
 
        //Never expose this at any cost
        private bool                        _inTrustedSubWindow;
 
        private ImageSource _icon;
 
        private NativeMethods.IconHandle    _defaultLargeIconHandle;
        private NativeMethods.IconHandle    _defaultSmallIconHandle;
        private NativeMethods.IconHandle    _currentLargeIconHandle;
        private NativeMethods.IconHandle    _currentSmallIconHandle;
 
        private bool?                       _dialogResult = null;
        private IntPtr                      _dialogOwnerHandle = IntPtr.Zero;
        private IntPtr                      _dialogPreviousActiveHandle;
        private DispatcherFrame             _dispatcherFrame;
 
        private WindowStartupLocation       _windowStartupLocation = WindowStartupLocation.Manual;
 
        // The previous WindowState value before WindowState changes
        private WindowState                 _previousWindowState = WindowState.Normal;
        private HwndWrapper         _hiddenWindow;
        private EventHandlerList    _events;
 
        // These should never be used directly, access only through property accessors
 
        private int                 _styleDoNotUse;
        private int                 _styleExDoNotUse;
        private HwndStyleManager    _manager;
 
        // reference to Resize Grip control; this is used to find out whether
        // the mouse of over the resizegrip control
        private Control                 _resizeGripControl;
 
        Point _prePanningLocation = new Point(double.NaN, double.NaN);
 
        // static objects for Events
        private static readonly object EVENT_SOURCEINITIALIZED = new object();
        private static readonly object EVENT_CLOSING = new object();
        private static readonly object EVENT_CLOSED = new object();
        private static readonly object EVENT_ACTIVATED = new object();
        private static readonly object EVENT_DEACTIVATED = new object();
        private static readonly object EVENT_STATECHANGED = new object();
        private static readonly object EVENT_LOCATIONCHANGED = new object();
        private static readonly object EVENT_CONTENTRENDERED = new object();
        private static readonly object EVENT_VISUALCHILDRENCHANGED = new object();
 
        #region Windows 7 Taskbar related fields
 
        // Register Window Message used by Shell to notify that the corresponding taskbar button has been added to the taskbar.
        private static readonly WindowMessage WM_TASKBARBUTTONCREATED;
 
        // Register Window Message used by Window to signal that we need to apply the taskbar button information again.
        private static readonly WindowMessage WM_APPLYTASKBARITEMINFO;
 
        // Magic constant determined by Shell.
        private const int c_MaximumThumbButtons = 7;
 
        private ITaskbarList3 _taskbarList;
 
        // When a taskbarList update fails because Explorer is non-responsive, defer further changes for a little while
        // to avoid causing the app to be non-responsive as well.
        private DispatcherTimer _taskbarRetryTimer;
 
        private Size _overlaySize;
 
        #endregion
 
        internal static readonly DependencyProperty IWindowServiceProperty
            = DependencyProperty.RegisterAttached("IWindowService", typeof(IWindowService), typeof(Window),
                                          new FrameworkPropertyMetadata((IWindowService)null,
                                          FrameworkPropertyMetadataOptions.Inherits | FrameworkPropertyMetadataOptions.OverridesInheritanceBehavior));
 
        DispatcherOperation         _contentRenderedCallback;
 
        private WeakReference _currentPanningTarget;
 
        #endregion Private Fields
 
        #region Private Class
 
        internal class SourceWindowHelper
        {
                internal SourceWindowHelper( HwndSource sourceWindow )
                {
                    Debug.Assert( sourceWindow != null );
                    _sourceWindow = sourceWindow;
                }
 
                internal bool IsSourceWindowNull
                {
                    get
                    {
                        return ( _sourceWindow == null );
                    }
                }
 
                internal bool IsCompositionTargetInvalid
                {
                    get
                    {
                        return (CompositionTarget == null);
                    }
                }
 
                internal IntPtr Handle
                {
                    get
                    {
                        if (_sourceWindow != null)
                        {
                            return _sourceWindow.Handle;
                        }
                        else
                        {
                            return IntPtr.Zero;
                        }
                    }
                }
 
                ///<summary>
                /// Get the work area bounds for this window - taking multi-mon into account.
                ///</summary>
                internal NativeMethods.RECT WorkAreaBoundsForNearestMonitor
                {
                    get
                    {
                        IntPtr monitor;
                        NativeMethods.MONITORINFOEX monitorInfo = new NativeMethods.MONITORINFOEX
                        {
                            cbSize = Marshal.SizeOf(typeof(NativeMethods.MONITORINFOEX))
                        };
 
                        monitor = SafeNativeMethods.MonitorFromWindow( new HandleRef( this, Handle), NativeMethods.MONITOR_DEFAULTTONEAREST  );
                        if ( monitor != IntPtr.Zero )
                        {
                            SafeNativeMethods.GetMonitorInfo( new HandleRef ( this, monitor ) , monitorInfo);
                        }
 
                        return monitorInfo.rcWork;
                    }
                }
 
                private NativeMethods.RECT ClientBounds
                {
                    get
                    {
                        NativeMethods.RECT rc = new NativeMethods.RECT(0,0,0,0);
                        SafeNativeMethods.GetClientRect(new HandleRef(this, Handle), ref rc);
 
                        return rc;
                    }
                }
 
                internal NativeMethods.RECT WindowBounds
                {
                    get
                    {
                        NativeMethods.RECT rc = new NativeMethods.RECT(0,0,0,0);
                        SafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref rc);
 
                        return rc;
                    }
                }
 
 
                private NativeMethods.POINT GetWindowScreenLocation(FlowDirection flowDirection)
                {
                    Debug.Assert(IsSourceWindowNull != true, "IsSourceWindowNull cannot be true here");
                    NativeMethods.POINT pt = default;
                    if (flowDirection == FlowDirection.RightToLeft)
                    {
                        NativeMethods.RECT rc = new NativeMethods.RECT(0, 0, 0, 0);
 
                        // with RTL window, GetClientRect returns reversed coordinates
                        SafeNativeMethods.GetClientRect(new HandleRef(this, Handle), ref rc);
 
                        // note that we use rc.right here for the RTL case and client to screen that point
                        pt = new NativeMethods.POINT(rc.right, rc.top);
                    }
                    UnsafeNativeMethods.ClientToScreen(new HandleRef(this, _sourceWindow.Handle), ref pt);
 
                    return pt;
                }
 
                internal SizeToContent HwndSourceSizeToContent
                {
                    get
                    {
                        return _sourceWindow.SizeToContent;
                    }
 
                    set
                    {
                        _sourceWindow.SizeToContent = value;
                    }
                }
 
                internal Visual RootVisual
                {
                    set
                    {
                        _sourceWindow.RootVisual = value;
                    }
                }
 
                internal bool IsActiveWindow
                {
                    get
                    {
                        return (_sourceWindow.Handle == UnsafeNativeMethods.GetActiveWindow());
                    }
                }
 
                internal HwndSource HwndSourceWindow
                {
                    get
                    {
                        return _sourceWindow;
                    }
                }
 
                internal HwndTarget CompositionTarget
                {
                    get
                    {
                        if (_sourceWindow != null)
                        {
                            HwndTarget compositionTarget = _sourceWindow.CompositionTarget;
                            if (compositionTarget != null && compositionTarget.IsDisposed == false)
                            {
                                return compositionTarget;
                            }
                        }
                        return null;
                    }
                }
 
                ///<summary>
                /// Return the relative window width and height.
                ///</summary>
                internal Size WindowSize
                {
                    get
                    {
                        // Get the size of the avalon window and pass it to
                        // the base implementation.
 
                        NativeMethods.RECT rc = WindowBounds;
 
                        return new Size(rc.right - rc.left, rc.bottom - rc.top);
                    }
                }
 
                internal int StyleExFromHwnd
                {
                    get
                    {
                        // Should never be called when Handle is non-null
                        Debug.Assert( IsSourceWindowNull == false , "Should only be invoked when we know Handle is non-null" );
                        return UnsafeNativeMethods.GetWindowLong(new HandleRef(this,Handle), NativeMethods.GWL_EXSTYLE);
                    }
                }
 
                internal int StyleFromHwnd
                {
                    get
                    {
                        // Should never be called when Handle is non-null
                        Debug.Assert( IsSourceWindowNull == false , "Should only be invoked when we know Handle is non-null" );
                        return UnsafeNativeMethods.GetWindowLong(new HandleRef(this,Handle), NativeMethods.GWL_STYLE);
                    }
                }
 
 
                ///<summary>
                ///     Transform global coords of window location
                ///     to coords relative to top/left of the window.
                ///</summary>
                internal NativeMethods.POINT GetPointRelativeToWindow( int x, int y, FlowDirection flowDirection )
                {
                    NativeMethods.POINT ptWindow = GetWindowScreenLocation(flowDirection);
 
                    // At this point ptWindow contains the location of the client area's top/left wrt
                    // the screen
 
                    return new NativeMethods.POINT( x - ptWindow.x, y - ptWindow.y );
                }
 
                /// <summary>
                ///     Gets the size from the hwnd
                /// </summary>
                internal Size GetSizeFromHwndInMeasureUnits()
                {
                    Debug.Assert( IsSourceWindowNull == false , "IsSourceWindowNull can't be true here");
 
                    Point pt = new Point(0,0);
                    NativeMethods.RECT rect = WindowBounds;
                    pt.X = rect.right - rect.left;
                    pt.Y = rect.bottom - rect.top;
                    pt = _sourceWindow.CompositionTarget.TransformFromDevice.Transform(pt);
                    return new Size(pt.X,pt.Y);
                }
 
 
                /// <summary>
                ///     Gets the frame size of the hwnd.
                ///     Note that we use the current Hwnd's style information.
                ///     If you're in the middle of changing the window's _style or _styleEx and call this function,
                ///     you may get inconsistent results.
                /// </summary>
                internal Size GetHwndNonClientAreaSizeInMeasureUnits()
                {
                    Debug.Assert( IsSourceWindowNull == false , "IsSourceWindowNull can't be true here");
 
                    // Diff the Client and Window sizes to get the dimensions of the frame.
                    NativeMethods.RECT clientRect = ClientBounds;
                    NativeMethods.RECT windowRect = WindowBounds;
 
                    Point pt = new Point(
                        (windowRect.right - windowRect.left) - (clientRect.right - clientRect.left),
                        (windowRect.bottom - windowRect.top) - (clientRect.bottom - clientRect.top));
 
                    pt = _sourceWindow.CompositionTarget.TransformFromDevice.Transform(pt);
 
                    // it's possible that the client rect is actually larger than the window rect
                    // Presumably this happens because the rects are changing
                    // out from under us, and it's just a transient state.  The Max calls
                    // avoid a crash when this happens.
                    return new Size(Math.Max(0.0, pt.X), Math.Max(0.0, pt.Y));
                }
 
                internal void ClearRootVisual()
                {
                    if ( _sourceWindow.RootVisual != null )
                    {
                        _sourceWindow.RootVisual = null;
                    }
                }
 
                internal void AddDisposedHandler( EventHandler theHandler )
                {
                    if (_sourceWindow != null)
                    {
                        _sourceWindow.Disposed += theHandler;
                    }
                }
 
                internal void RemoveDisposedHandler( EventHandler theHandler )
                {
                    if (_sourceWindow != null)
                    {
                        _sourceWindow.Disposed -= theHandler;
                    }
                }
 
                /// <summary>
                ///     Updates panning feedback for this window based on the offset.
                /// </summary>
                /// <param name="totalOverpanOffset">The amount of over-panning being reported.</param>
                /// <param name="animate">Whether to animate to the new feedback position.</param>
                internal void UpdatePanningFeedback(Vector totalOverpanOffset, bool animate)
                {
                    if ((_panningFeedback == null) && (_sourceWindow != null))
                    {
                        _panningFeedback = new HwndPanningFeedback(_sourceWindow);
                    }
 
                    if (_panningFeedback != null)
                    {
                        // Update the window position
                        _panningFeedback.UpdatePanningFeedback(totalOverpanOffset, animate);
                    }
                }
 
                /// <summary>
                ///     Return the window back to its original position.
                /// </summary>
                /// <param name="animateBack">Whether to animate to the original position.</param>
                internal void EndPanningFeedback(bool animateBack)
                {
                    if (_panningFeedback != null)
                    {
                        // Restore the window to its original position
                        _panningFeedback.EndPanningFeedback(animateBack);
                        _panningFeedback = null;
                    }
                }
 
                private HwndSource _sourceWindow;
 
                private HwndPanningFeedback _panningFeedback;
        }
 
        internal class HwndStyleManager : IDisposable
        {
            static internal HwndStyleManager StartManaging(Window w, int Style, int StyleEx )
            {
                if (w.Manager == null)
                {
                    return new HwndStyleManager(w, Style, StyleEx);
                }
                else
                {
                    w.Manager._refCount++;
                    return w.Manager;
                }
            }
 
            private HwndStyleManager(Window w, int Style, int StyleEx  )
            {
                _window = w;
                _window.Manager = this;
 
                if ( w.IsSourceWindowNull == false )
                {
                    _window._Style    =  Style;
                    _window._StyleEx  = StyleEx;
 
                    // Dirty ==> _style and hwnd are out of sync. Since we just got
                    // the style from hwnd, it obviously is not Dirty.
                    Dirty = false;
                }
                _refCount = 1;
            }
 
            void IDisposable.Dispose()
            {
                _refCount--;
 
                // (A NullReferenceException occurs when animating
                // the WindowStyle enum via a custom animation).  This bug contains details of
                // why we were seeing the null ref.
                //
                // Sometimes, the Flush call below results in sending certain window messages
                // and their handling leads to setting some property on the Window leading
                // to a call to HwndStyleManager.StartManaging.  Thus, we end up calling
                // dispose on that before we complete this run of the Dispose method.  This
                // resulted in null ref in Flush.  To fix this bug, we do the following two things:
                //
                // 1) Keep a local copy of HwndStyleManager in Flush to make it re-entrant
                // 2) null out _window.Manager below only if _window.Manager is this instance
                //    of the Manager.
 
                if (_refCount == 0)
                {
                    _window.Flush();
 
                    if (_window.Manager == this)
                    {
                        _window.Manager = null;
                    }
                }
            }
 
            internal bool Dirty
            {
                get { return _fDirty; }
                set { _fDirty = value; }
            }
 
            private Window          _window;
            private int             _refCount;
            private bool            _fDirty;
        }
 
 
        /// <summary>
        /// A helper class to convert (Left, Top) <see cref="Point"/> expressed
        /// in WPF-space (1/96" units) to Screen units.
        /// </summary>
        /// <remarks>
        /// Initially, an HWND created in <see cref="Window"/> in a Per-Monitor Aware process may not
        /// have a relaible DPI. Instead, it would be associated with the monitor where the screen point (0,0)
        /// happens to be, but the screen point corresponding to a WPF Window's (requestedLeft, requestedTop) may not map
        /// to that monitor. Our initial DeviceToLogicalUnits and LogicalToDeviceUnits transforms
        /// would normally have to rely on the DPI obtained from this HWND, which may lead us to
        /// an incorrect result when calculating screen points. To avoid this, we use a special heuristic here
        /// to identify the (screenLeft, screenTop) initially without relying on the HWND, and then
        /// supply it to HwndSourceParameters at the time of HWND creation. This would in turn set up the HWND
        /// with the correct DPI from the start, and all our transformations would also follow suit.
        /// </remarks>
        private class WindowStartupTopLeftPointHelper
        {
            /// <summary>
            /// The (Left, Top) logical point, in WPF coordinate space,
            /// that must be translated into Screen space.
            /// </summary>
            internal Point LogicalTopLeft { get; }
 
            /// <summary>
            /// The Screen-space translation of <see cref="LogicalTopLeft"/>
            /// </summary>
            internal Point? ScreenTopLeft { get; private set; } = null;
 
            /// <summary>
            /// Creates a new instance of <see cref="WindowStartupTopLeftPointHelper"/>
            /// </summary>
            /// <param name="topLeft">(left, top) point in WPF-space to be converted to a Screen-space point</param>
            internal WindowStartupTopLeftPointHelper(Point topLeft)
            {
                LogicalTopLeft = topLeft;
 
                if (IsHelperNeeded)
                {
                    IdentifyScreenTopLeft();
                }
            }
 
            /// <summary>
            /// Decides whether this helper should be used.
            /// This helper is used when -
            ///     a. WPF supports DPI scaling (HwndTarget.IsPerMonitorDpiScalingEnabled), and
            ///     b. The process is PMA (HwndTarget.IsProcessPerMonitorDpiAware)
            /// </summary>
            /// <remarks>
            /// <see cref="CoreAppContextSwitches.DoNotUsePresentationDpiCapabilityTier2OrGreater"/>
            /// is tested to determine whether accurate window placement behavior should be enabled, or not. This is a
            /// compatibility measure put in place to ensure that our improvement/fix does not clash with workarounds
            /// that developers might have built into their applications to correct this problem.
            /// </remarks>
            private bool IsHelperNeeded
            {
                get
                {
                    if (CoreAppContextSwitches.DoNotUsePresentationDpiCapabilityTier2OrGreater)
                    {
                        return false;
                    }
 
                    if (!HwndTarget.IsPerMonitorDpiScalingEnabled)
                    {
                        return false;
                    }
 
                    if (HwndTarget.IsProcessPerMonitorDpiAware.HasValue)
                    {
                        return HwndTarget.IsProcessPerMonitorDpiAware.Value;
                    }
 
                    // WPF supports Per-Monitor scaling, but HwndTarget has not
                    // yet been initialized with the first HWND, and therefore
                    // HwndTarget.IsProcessPerMonitorDpiAware is not queryable.
                    // Let's use the current process' DPI awareness as a proxy.
                    return DpiUtil.GetProcessDpiAwareness(IntPtr.Zero) == NativeMethods.PROCESS_DPI_AWARENESS.PROCESS_PER_MONITOR_DPI_AWARE;
                }
            }
 
            /// <summary>
            /// Uses EnumDisplayDevices to iterate through each monitor, and looks to see if
            /// <see cref="LogicalTopLeft"/> exists within that monitor's rectangle. To do this,
            /// it scales that monitors coordinate space by its corresponding DPI scale factor (this
            /// happens in <see cref="MonitorEnumProc(IntPtr, IntPtr, ref Runtime.InteropServices.NativeMethods.RECT, IntPtr)"/>)
            /// and normalizes it to a 96 DPI space - which is the space in which WPF itself operates - and then sees whether
            /// <see cref="LogicalTopLeft"/> lies within that rectangle. If it does, then it updates <see cref="ScreenTopLeft"/>
            /// with the corresponding unscaled monitor-space point.
            /// </summary>
            private void IdentifyScreenTopLeft()
            {
                var nullHandle = new HandleRef(null, IntPtr.Zero);
                var hdc = UnsafeNativeMethods.GetDC(nullHandle);
                UnsafeNativeMethods.EnumDisplayMonitors(
                    hdc,
                    IntPtr.Zero,
                    MonitorEnumProc,
                    IntPtr.Zero);
                UnsafeNativeMethods.ReleaseDC(nullHandle, new HandleRef(null, hdc));
            }
 
            /// <summary>
            /// Helper for <see cref="IdentifyScreenTopLeft"/>
            /// </summary>
            /// <remarks>See summary and remarks for <see cref="IdentifyScreenTopLeft"/></remarks>
            /// <param name="hMonitor">A handle to the display monitor. This value will always be non-NULL.</param>
            /// <param name="hdcMonitor">A handle to the device context</param>
            /// <param name="lprcMonitor">
            /// A pointer to a RECT structure.
            ///
            /// If hdcMonitor is non-NULL, this rectangle is the intersection of the clipping area of the device
            /// context identified by hdcMonitor and the display monitor rectangle.The rectangle coordinates are
            /// device-context coordinates.
            ///
            /// If hdcMonitor is NULL, this rectangle is the display monitor rectangle. The rectangle coordinates
            /// are virtual-screen coordinates.
            /// </param>
            /// <param name="dwData">
            /// Application-defined data that EnumDisplayMonitors passes directly to
            /// the enumeration function.
            /// </param>
            /// <returns>
            /// To continue the enumeration, return true.
            /// To stop the enumeration, return false.
            /// </returns>
            private bool MonitorEnumProc(IntPtr hMonitor, IntPtr hdcMonitor, ref NativeMethods.RECT lprcMonitor, IntPtr dwData)
            {
                // The call the EnumDisplayMonitors set hdc to null
                // This means that hdcMonitor will be null, and
                // lprcMonitor will represent the RECT of the whole monitor
                // skip the checks and trust the API contract that this will
                // be so
 
                bool continueMonitorEnumeration = true;
 
                uint dpiX, dpiY;
                var hr =
                    UnsafeNativeMethods.GetDpiForMonitor(
                        new HandleRef(null, hMonitor),
                        NativeMethods.MONITOR_DPI_TYPE.MDT_EFFECTIVE_DPI,
                        out dpiX, out dpiY);
 
                if (hr == NativeMethods.S_OK)
                {
                    double dpiScaleX = dpiX * 1.0d / 96.0d;
                    double dpiScaleY = dpiY * 1.0d / 96.0d;
 
                    Rect monitorRectInWpfScale = new Rect
                    {
                        X = lprcMonitor.left / dpiScaleX,
                        Y = lprcMonitor.top / dpiScaleY,
                        Width = (lprcMonitor.right - lprcMonitor.left) / dpiScaleX,
                        Height = (lprcMonitor.bottom - lprcMonitor.top) / dpiScaleY
                    };
 
                    if (monitorRectInWpfScale.Contains(LogicalTopLeft))
                    {
                        ScreenTopLeft = new Point
                        {
                            X = LogicalTopLeft.X * dpiScaleX,
                            Y = LogicalTopLeft.Y * dpiScaleY
                        };
 
                        continueMonitorEnumeration = false;
                    }
                }
 
                return continueMonitorEnumeration;
            }
        }
 
        #endregion PrivateClass
 
        #region Private Enums
        private enum TransformType
        {
            WorkAreaToScreenArea = 0,
            ScreenAreaToWorkArea = 1
        }
 
        private enum BoundsSpecified
        {
            Height = 0,
            Width = 1,
            Top = 2,
            Left = 3
        }
        #endregion Private Enums
 
        #region DTypeThemeStyleKey
 
        // Returns the DependencyObjectType for the registered ThemeStyleKey's default
        // value. Controls will override this method to return approriate types.
        internal override DependencyObjectType DTypeThemeStyleKey
        {
            get { return _dType; }
        }
 
        private static DependencyObjectType _dType;
 
        #endregion DTypeThemeStyleKey
    }
 
    #region Enums
 
    /// <summary>
    /// WindowStyle
    /// </summary>
    public enum WindowStyle
    {
        /// <summary>
        /// no border at all  also implies no caption
        /// </summary>
        None = 0,                                               // no border at all  also implies no caption
 
        /// <summary>
        /// SingleBorderWindow
        /// </summary>
        SingleBorderWindow = 1,                                    // WS_BORDER
 
        /// <summary>
        /// 3DBorderWindow
        /// </summary>
        ThreeDBorderWindow = 2,                                    // WS_BORDER | WS_EX_CLIENTEDGE
 
        /// <summary>
        /// FixedToolWindow
        /// </summary>
        ToolWindow = 3,                                           // WS_BORDER | WS_EX_TOOLWINDOW
 
        // NOTE: if you add or remove any values in this enum, be sure to update Window.IsValidWindowStyle()
    }
 
 
    /// <summary>
    /// WindowState
    /// </summary>
    public enum WindowState
    {
        /// <summary>
        /// Default size
        /// </summary>
        Normal = 0,
 
        /// <summary>
        /// Minimized
        /// </summary>
        Minimized = 1,   // WS_MINIMIZE
 
        /// <summary>
        /// Maximized
        /// </summary>
        Maximized = 2   // WS_MAXIMIZE
 
        // NOTE: if you add or remove any values in this enum, be sure to update Window.IsValidWindowState()
 
#if THEATRE_FULLSCREEN
        // The following Two are not Implement yet
        /// <summary>
        /// Theatre
        /// </summary>
        Theatre = 3,
 
        /// <summary>
        /// FullScreen
        /// </summary>
        FullScreen = 4
#endif //THEATRE_FULLSCREEN
    }
 
    /// <summary>
    ///
    /// </summary>
    [Localizability(LocalizationCategory.None, Readability = Readability.Unreadable)]
    public enum WindowStartupLocation
    {
        /// <summary>
        /// Uses the values specified by Left and Top properties to position the Window
        /// </summary>
        Manual = 0,
 
        /// <summary>
        /// Centers the Window on the screen.  If there are more than one monitors, then
        /// the Window is centered on the monitor that has the mouse on it
        /// </summary>
        CenterScreen = 1,
 
        /// <summary>
        /// Centers the Window on its owner.  If there is no owner window defined or if
        /// it is not possible to center it on the owner, then defaults to Manual
        /// </summary>
        CenterOwner = 2,
 
        // NOTE: if you add or remove any values in this enum, be sure to update Window.IsValidWindowStartupLocation()
    }
 
    /// <summary>
    ///     ResizeMode
    /// </summary>
    [Localizability(LocalizationCategory.None, Readability = Readability.Unreadable)]
    public enum ResizeMode
    {
        /// <summary>
        ///     User cannot resize the Window. Maximize and Minimize boxes
        ///     do not show in the caption bar.
        /// </summary>
        NoResize = 0,
 
        /// <summary>
        ///     User can only minimize the Window.  Minimize box is shown and enabled
        ///     in the caption bar while the Maximize box is disabled.
        /// </summary>
        CanMinimize = 1,
 
        /// <summary>
        ///     User can fully resize the Window including minimize and maximize.
        ///     Both Maximize and Minimize boxes are shown and enabled in the caption
        ///     bar.
        /// </summary>
        CanResize = 2,
 
        /// <summary>
        ///     Same as CanResize and ResizeGrip will show
        /// </summary>
        CanResizeWithGrip = 3
 
        // NOTE: if you add or remove any values in this enum, be sure to update Window.IsValidResizeMode()
    }
    #endregion Enums
 
    internal class SingleChildEnumerator : IEnumerator
    {
        internal SingleChildEnumerator(object Child)
        {
            _child = Child;
            _count = Child == null ? 0 : 1;
        }
 
        object IEnumerator.Current
        {
            get { return (_index == 0) ? _child : null; }
        }
 
        bool IEnumerator.MoveNext()
        {
            _index++;
            return _index < _count;
        }
 
        void IEnumerator.Reset()
        {
            _index = -1;
        }
 
        private int _index = -1;
        private int _count = 0;
        private object _child;
    }
}