TreeView.cs source code in C# .NET

Source code for the .NET framework in C#

                        

Code:

/ Dotnetfx_Win7_3.5.1 / Dotnetfx_Win7_3.5.1 / 3.5.1 / DEVDIV / depot / DevDiv / releases / Orcas / NetFXw7 / wpf / src / Framework / System / Windows / Controls / TreeView.cs / 1 / TreeView.cs

                            //---------------------------------------------------------------------------- 
//
// Copyright (C) Microsoft Corporation.  All rights reserved.
//
//--------------------------------------------------------------------------- 

using System; 
using System.Collections; 
using System.Collections.Generic;
using System.Collections.Specialized; 
using System.ComponentModel;
using System.Diagnostics;
using System.Windows;
using System.Windows.Automation.Peers; 
using System.Windows.Controls.Primitives;
using System.Windows.Data; 
using System.Windows.Input; 
using System.Windows.Media;
using MS.Internal; 
using MS.Internal.Data;
using MS.Internal.KnownBoxes;

namespace System.Windows.Controls 
{
    ///  
    ///     A control that presents items in a tree structure. 
    /// 
    [StyleTypedProperty(Property = "ItemContainerStyle", StyleTargetType = typeof(TreeViewItem))] 
    public class TreeView : ItemsControl
    {
        #region Constructors
 
        static TreeView()
        { 
            DefaultStyleKeyProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(typeof(TreeView))); 
            VirtualizingStackPanel.IsVirtualizingProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(BooleanBoxes.FalseBox));
            _dType = DependencyObjectType.FromSystemTypeInternal(typeof(TreeView)); 

            KeyboardNavigation.DirectionalNavigationProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(KeyboardNavigationMode.Contained));
            KeyboardNavigation.TabNavigationProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(KeyboardNavigationMode.None));
        } 

        ///  
        ///     Creates an instance of this control. 
        /// 
        public TreeView() 
        {
            _focusEnterMainFocusScopeEventHandler = new EventHandler(OnFocusEnterMainFocusScope);
            KeyboardNavigation.Current.FocusEnterMainFocusScope += _focusEnterMainFocusScopeEventHandler;
        } 

        #endregion 
 
        #region Public Properties
 
        private static readonly DependencyPropertyKey SelectedItemPropertyKey =
            DependencyProperty.RegisterReadOnly("SelectedItem", typeof(object), typeof(TreeView), new FrameworkPropertyMetadata((object)null));

        ///  
        ///     The DependencyProperty for the  property.
        ///     Default Value: null 
        ///  
        public static readonly DependencyProperty SelectedItemProperty = SelectedItemPropertyKey.DependencyProperty;
 
        /// 
        ///     Specifies the selected item.
        /// 
        [Bindable(true), Category("Appearance"), ReadOnly(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
        public object SelectedItem
        { 
            get 
            {
                return GetValue(SelectedItemProperty); 
            }
        }

        private void SetSelectedItem(object data) 
        {
            if (SelectedItem != data) 
            { 
                SetValue(SelectedItemPropertyKey, data);
            } 
        }

        private static readonly DependencyPropertyKey SelectedValuePropertyKey =
            DependencyProperty.RegisterReadOnly("SelectedValue", typeof(object), typeof(TreeView), new FrameworkPropertyMetadata((object)null)); 

        ///  
        ///     The DependencyProperty for the  property. 
        ///     Default Value: null
        ///  
        public static readonly DependencyProperty SelectedValueProperty = SelectedValuePropertyKey.DependencyProperty;

        /// 
        ///     Specifies the a value on the selected item as defined by . 
        /// 
        [Bindable(true), Category("Appearance"), ReadOnly(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
        public object SelectedValue 
        {
            get 
            {
                return GetValue(SelectedValueProperty);
            }
        } 

        private void SetSelectedValue(object data) 
        { 
            if (SelectedValue != data)
            { 
                SetValue(SelectedValuePropertyKey, data);
            }
        }
 
        /// 
        ///     The DependencyProperty for the  property. 
        ///     Default Value: String.Empty 
        /// 
        public static readonly DependencyProperty SelectedValuePathProperty = 
            DependencyProperty.Register(
                    "SelectedValuePath",
                    typeof(string),
                    typeof(TreeView), 
                    new FrameworkPropertyMetadata(
                            String.Empty, 
                            new PropertyChangedCallback(OnSelectedValuePathChanged))); 

        ///  
        ///     Specifies the path to query on  to calculate .
        /// 
        [Bindable(true), Category("Appearance")]
        public string SelectedValuePath 
        {
            get { return (string) GetValue(SelectedValuePathProperty); } 
            set { SetValue(SelectedValuePathProperty, value); } 
        }
 
        private static void OnSelectedValuePathChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            TreeView tree = (TreeView)d;
            SelectedValuePathBindingExpression.ClearValue(tree); 
            tree.UpdateSelectedValue(tree.SelectedItem);
        } 
 
        #endregion
 
        #region Public Events

        /// 
        ///     Event fired when  changes. 
        /// 
        public static readonly RoutedEvent SelectedItemChangedEvent = EventManager.RegisterRoutedEvent("SelectedItemChanged", RoutingStrategy.Bubble, typeof(RoutedPropertyChangedEventHandler), typeof(TreeView)); 
 
        /// 
        ///     Event fired when  changes. 
        /// 
        [Category("Behavior")]
        public event RoutedPropertyChangedEventHandler SelectedItemChanged
        { 
            add
            { 
                AddHandler(SelectedItemChangedEvent, value); 
            }
 
            remove
            {
                RemoveHandler(SelectedItemChangedEvent, value);
            } 
        }
 
        ///  
        ///     Called when  changes.
        ///     Default implementation fires the  event. 
        /// 
        /// Event arguments.
        protected virtual void OnSelectedItemChanged(RoutedPropertyChangedEventArgs e)
        { 
            //
            RaiseEvent(e); 
        } 

        #endregion 

        #region Implementation

        #region Selection 

        internal void ChangeSelection(object data, TreeViewItem container, bool selected) 
        { 
            if (IsSelectionChangeActive)
            { 
                return;
            }

            object oldValue = null; 
            object newValue = null;
            bool changed = false; 
            TreeViewItem oldContainer = _selectedContainer; // Saved for the automation event 

            IsSelectionChangeActive = true; 

            try
            {
                if (selected) 
                {
                    if (container != _selectedContainer) 
                    { 
                        oldValue = SelectedItem;
                        newValue = data; 

                        if (_selectedContainer != null)
                        {
                            _selectedContainer.IsSelected = false; 
                            _selectedContainer.UpdateContainsSelection(false);
                        } 
                        _selectedContainer = container; 
                        _selectedContainer.UpdateContainsSelection(true);
                        SetSelectedItem(data); 
                        UpdateSelectedValue(data);
                        changed = true;
                    }
                } 
                else
                { 
                    if (container == _selectedContainer) 
                    {
                        _selectedContainer.UpdateContainsSelection(false); 
                        _selectedContainer = null;
                        SetSelectedItem(null);

                        oldValue = data; 
                        changed = true;
                    } 
                } 

                if (container.IsSelected != selected) 
                {
                    container.IsSelected = selected;
                }
            } 
            finally
            { 
                IsSelectionChangeActive = false; 
            }
 
            if (changed)
            {
                if (    _selectedContainer != null
                    &&  AutomationPeer.ListenerExists(AutomationEvents.SelectionItemPatternOnElementSelected)   ) 
                {
                    AutomationPeer peer = UIElementAutomationPeer.CreatePeerForElement(_selectedContainer); 
                    if (peer != null) 
                        peer.RaiseAutomationEvent(AutomationEvents.SelectionItemPatternOnElementSelected);
                } 

                if (    oldContainer != null
                    &&  AutomationPeer.ListenerExists(AutomationEvents.SelectionItemPatternOnElementRemovedFromSelection)   )
                { 
                    AutomationPeer peer = UIElementAutomationPeer.CreatePeerForElement(oldContainer);
                    if (peer != null) 
                        peer.RaiseAutomationEvent(AutomationEvents.SelectionItemPatternOnElementRemovedFromSelection); 
                }
 
                RoutedPropertyChangedEventArgs e = new RoutedPropertyChangedEventArgs(oldValue, newValue, SelectedItemChangedEvent);
                OnSelectedItemChanged(e);
            }
        } 

        internal bool IsSelectionChangeActive 
        { 
            get { return _bits[(int)Bits.IsSelectionChangeActive]; }
            set { _bits[(int)Bits.IsSelectionChangeActive] = value; } 
        }

        private void UpdateSelectedValue(object selectedItem)
        { 
            BindingExpression expression = PrepareSelectedValuePathBindingExpression(selectedItem);
 
            if (expression != null) 
            {
                expression.Activate(selectedItem); 
                object selectedValue = expression.Value;
                expression.Deactivate();

                SetValue(SelectedValuePropertyKey, selectedValue); 
            }
            else 
            { 
                ClearValue(SelectedValuePropertyKey);
            } 
        }

        private BindingExpression PrepareSelectedValuePathBindingExpression(object item)
        { 
            if (item == null)
            { 
                return null; 
            }
 
            Binding binding;
            bool useXml = XmlHelper.IsXmlNode(item);

            BindingExpression bindingExpr = SelectedValuePathBindingExpression.GetValue(this); 

            // replace existing binding if it's the wrong kind 
            if (bindingExpr != null) 
            {
                binding = bindingExpr.ParentBinding; 
                bool usesXml = (binding.XPath != null);
                if (usesXml != useXml)
                {
                    bindingExpr = null; 
                }
            } 
 
            if (bindingExpr == null)
            { 
                // create the binding
                binding = new Binding();
                binding.Source = item;
 
                if (useXml)
                { 
                    binding.XPath = SelectedValuePath; 
                    binding.Path = new PropertyPath("/InnerText");
                } 
                else
                {
                    binding.Path = new PropertyPath(SelectedValuePath);
                } 

                bindingExpr = (BindingExpression)BindingExpression.CreateUntargetedBindingExpression(this, binding); 
                SelectedValuePathBindingExpression.SetValue(this, bindingExpr); 
            }
 
            return bindingExpr;
        }

        internal void HandleSelectionAndCollapsed(TreeViewItem collapsed) 
        {
            if ((_selectedContainer != null) && (_selectedContainer != collapsed)) 
            { 
                // Check if current selection is under the collapsed element
                TreeViewItem current = _selectedContainer; 
                do
                {
                    current = current.ParentTreeViewItem;
                    if (current == collapsed) 
                    {
                        TreeViewItem oldContainer = _selectedContainer; 
 
                        ChangeSelection(collapsed.ParentItemsControl.ItemContainerGenerator.ItemFromContainer(collapsed), collapsed, true);
 
                        if (oldContainer.IsKeyboardFocusWithin)
                        {
                            // If the oldContainer had focus then move focus to the newContainer instead
                            _selectedContainer.Focus(); 
                        }
 
                        break; 
                    }
                } 
                while (current != null);
            }
        }
 
        // This method is called when MouseButonDown on TreeViewItem and also listen for handled events too
        // The purpose is to restore focus on TreeView when mouse is clicked and focus was outside the TreeView 
        // Focus goes either to selected item (if any) or treeview itself 
        internal void HandleMouseButtonDown()
        { 
            if (!this.IsKeyboardFocusWithin)
            {
                if (_selectedContainer != null)
                { 
                    if (!_selectedContainer.IsKeyboardFocused)
                        _selectedContainer.Focus(); 
                } 
                else
                { 
                    // If we don't have a selection - just focus the treeview
                    this.Focus();
                }
            } 
        }
 
        #endregion 

        #region Containers 

        /// 
        ///     Returns true if the item is or should be its own container.
        ///  
        /// The item to test.
        /// true if its type matches the container type. 
        protected override bool IsItemItsOwnContainerOverride(object item) 
        {
            return item is TreeViewItem; 
        }

        /// 
        ///     Create or identify the element used to display the given item. 
        /// 
        /// The container. 
        protected override DependencyObject GetContainerForItemOverride() 
        {
            return new TreeViewItem(); 
        }

        /// 
        ///     This method is invoked when the Items property changes. 
        /// 
        protected override void OnItemsChanged(NotifyCollectionChangedEventArgs e) 
        { 
            switch (e.Action)
            { 
                case NotifyCollectionChangedAction.Remove:
                case NotifyCollectionChangedAction.Reset:
                    if ((SelectedItem != null) && !IsSelectedContainerHookedUp)
                    { 
                        SelectFirstItem();
                    } 
                    break; 

                case NotifyCollectionChangedAction.Replace: 
                    {
                        // If old item is selected - remove the selection
                        // Revisit the condition when we support duplicate items in Items collection: if e.OldItems[0] is the same as selected items we will unselect the selected item
                        object selectedItem = SelectedItem; 
                        if ((selectedItem != null) && selectedItem.Equals(e.OldItems[0]))
                        { 
                            ChangeSelection(selectedItem, _selectedContainer, false); 
                        }
                    } 
                    break;

                case NotifyCollectionChangedAction.Add:
                case NotifyCollectionChangedAction.Move: 
                    break;
 
                default: 
                    throw new NotSupportedException(SR.Get(SRID.UnexpectedCollectionChangeAction, e.Action));
            } 
        }

        /// 
        /// Send down the IsVirtualizing property if it's set on this element. 
        /// 
        ///  
        ///  
        protected override void PrepareContainerForItemOverride(DependencyObject element, object item)
        { 
            base.PrepareContainerForItemOverride(element, item);
            TreeViewItem.IsVirtualizingPropagationHelper(this, element);
        }
 
        private void SelectFirstItem()
        { 
            object item; 
            TreeViewItem container;
            bool selected = GetFirstItem(out item, out container); 
            if (!selected)
            {
                item = SelectedItem;
                container = _selectedContainer; 
            }
 
            ChangeSelection(item, container, selected); 
        }
 
        private bool GetFirstItem(out object item, out TreeViewItem container)
        {
            if (HasItems)
            { 
                item = Items[0];
                container = ItemContainerGenerator.ContainerFromIndex(0) as TreeViewItem; 
                return ((item != null) && (container != null)); 
            }
            else 
            {
                item = null;
                container = null;
                return false; 
            }
        } 
 
        internal bool IsSelectedContainerHookedUp
        { 
            get
            {
                return (_selectedContainer != null) && (_selectedContainer.ParentTreeView == this);
            } 
        }
 
        internal TreeViewItem SelectedContainer 
        {
            get 
            {
                return _selectedContainer;
            }
        } 

        #endregion 
 
        #region Input
 
        /// 
        ///     If control has a scrollviewer in its style and has a custom keyboard scrolling behavior when HandlesScrolling should return true.
        /// Then ScrollViewer will not handle keyboard input and leave it up to the control.
        ///  
        protected internal override bool HandlesScrolling
        { 
            get { return true; } 
        }
 
        /// 
        ///     Called when a keyboard key is pressed down.
        /// 
        /// Event Arguments 
        protected override void OnKeyDown(KeyEventArgs e)
        { 
            base.OnKeyDown(e); 
            if (!e.Handled)
            { 
                if (IsControlKeyDown)
                {
                    switch (e.Key)
                    { 
                        case Key.Up:
                        case Key.Down: 
                        case Key.Left: 
                        case Key.Right:
                        case Key.Home: 
                        case Key.End:
                        case Key.PageUp:
                        case Key.PageDown:
                            if (HandleScrollKeys(e.Key)) 
                            {
                                e.Handled = true; 
                            } 
                            break;
                    } 
                }
                else
                {
                    switch (e.Key) 
                    {
                        case Key.Up: 
                        case Key.Down: 
                            if ((_selectedContainer == null) && FocusFirstItem())
                            { 
                                e.Handled = true;
                            }
                            break;
 
                        case Key.Home:
                            if (FocusFirstItem()) 
                            { 
                                e.Handled = true;
                            } 
                            break;

                        case Key.End:
                            if (FocusLastItem()) 
                            {
                                e.Handled = true; 
                            } 
                            break;
 
                        case Key.PageUp:
                        case Key.PageDown:
                            if (_selectedContainer == null)
                            { 
                                if (FocusFirstItem())
                                { 
                                    e.Handled = true; 
                                }
                            } 
                            else if (HandleScrollByPage(e.Key == Key.PageUp))
                            {
                                e.Handled = true;
                            } 
                            break;
 
                        case Key.Tab: 
                            if (IsShiftKeyDown && IsKeyboardFocusWithin)
                            { 
                                // SHIFT-TAB behavior for KeyboardNavigation needs to happen at the TreeView level
                                if (MoveFocus(new TraversalRequest(FocusNavigationDirection.Previous)))
                                {
                                    e.Handled = true; 
                                }
                            } 
                            break; 
                    }
                } 
            }
        }

        private static bool IsControlKeyDown 
        {
            get 
            { 
                return ((Keyboard.Modifiers & ModifierKeys.Control) == (ModifierKeys.Control));
            } 
        }

        private static bool IsShiftKeyDown
        { 
            get
            { 
                return ((Keyboard.Modifiers & ModifierKeys.Shift) == (ModifierKeys.Shift)); 
            }
        } 

        private bool FocusFirstItem()
        {
            if (IsVirtualizing) 
            {
                ScrollToEdge(/*toEnd = */ false); 
            } 

            TreeViewItem item = ItemContainerGenerator.ContainerFromIndex(0) as TreeViewItem; 
            if (item != null)
            {
                if (item.IsEnabled && item.Focus())
                { 
                    return true;
                } 
                else 
                {
                    return item.FocusDown(); 
                }
            }

            return false; 
        }
 
 
        private bool FocusLastItem()
        { 
            //
            // If virtualizing first scroll to the end so that the last item will be generated.
            //
            if (IsVirtualizing) 
            {
                ScrollToEdge(/* toEnd = */ true); 
            } 

            int index = Items.Count - 1; 
            while (index >= 0)
            {
                TreeViewItem item = ItemContainerGenerator.ContainerFromIndex(index) as TreeViewItem;
                if ((item != null) && item.IsEnabled) 
                {
                    return TreeViewItem.FocusIntoItem(item); 
                } 
                index--;
            } 

            return false;
        }
 
        private bool HandleScrollKeys(Key key)
        { 
            ScrollViewer scroller = ScrollHost; 
            if (scroller != null)
            { 
                bool invert = (FlowDirection == FlowDirection.RightToLeft);
                switch (key)
                {
                    case Key.Up: 
                        scroller.LineUp();
                        return true; 
 
                    case Key.Down:
                        scroller.LineDown(); 
                        return true;

                    case Key.Left:
                        if (invert) 
                        {
                            scroller.LineRight(); 
                        } 
                        else
                        { 
                            scroller.LineLeft();
                        }
                        return true;
 
                    case Key.Right:
                        if (invert) 
                        { 
                            scroller.LineLeft();
                        } 
                        else
                        {
                            scroller.LineRight();
                        } 
                        return true;
 
                    case Key.Home: 
                        scroller.ScrollToTop();
                        return true; 

                    case Key.End:
                        scroller.ScrollToBottom();
                        return true; 

                    case Key.PageUp: 
                        //if vertically scrollable - go vertical, otherwise horizontal 
                        if(DoubleUtil.GreaterThan(scroller.ExtentHeight, scroller.ViewportHeight))
                        { 
                            scroller.PageUp();
                        }
                        else
                        { 
                            scroller.PageLeft();
                        } 
                        return true; 

                    case Key.PageDown: 
                        //if vertically scrollable - go vertical, otherwise horizontal
                        if(DoubleUtil.GreaterThan(scroller.ExtentHeight, scroller.ViewportHeight))
                        {
                            scroller.PageDown(); 
                        }
                        else 
                        { 
                            scroller.PageRight();
                        } 
                        return true;

                }
            } 

            return false; 
        } 

        // Note the two assumptions TreeView is making here 
        // 1.) that everything is laid out vertically and
        // 2.) that the headers of TreeViewItems always appear above the ItemsPresenter.
        private bool HandleScrollByPage(bool up)
        { 
            ScrollViewer scroller = ScrollHost;
            if (scroller != null) 
            { 
                double viewportHeight = scroller.ViewportHeight;
                double startTop, startBottom; 


                if (VirtualizingStackPanel.GetIsVirtualizing(this))
                { 
                    PreScrollByPage(scroller, up);
                } 
 
                _selectedContainer.GetTopAndBottom(scroller, out startTop, out startBottom);
 
                TreeViewItem select = null;
                TreeViewItem next = _selectedContainer;
                ItemsControl parent = _selectedContainer.ParentItemsControl;
 
                if (parent != null)
                { 
                    if (up) 
                    {
                        // When going up, we need to start at the first level of TreeViewItems. 
                        // When going down, we need to start at the selected container.

                        while (parent != this)
                        { 
                            ItemsControl nextParent = ItemsControl.ItemsControlFromItemContainer(parent);
                            if (nextParent == null) 
                            { 
                                break;
                            } 
                            else
                            {
                                next = (TreeViewItem)parent;
                                parent = nextParent; 
                            }
                        } 
                    } 

                    int index = parent.ItemContainerGenerator.IndexFromContainer(next); 
                    int count = parent.Items.Count;

                    while ((parent != null) && (next != null))
                    { 
                        if (next.IsEnabled)
                        { 
                            double delta; 
                            if (next.HandleScrollByPage(up, scroller, viewportHeight, startTop, startBottom, out delta))
                            { 
                                // This item or one of its children was focused
                                return true;
                            }
                            else if (DoubleUtil.GreaterThan(delta, viewportHeight)) 
                            {
                                // This item does not fit 
 
                                // If select target is already the same element as _selectedContainer - there is no point to select it again
                                // In this case we select the next item although it cannot completely fit into view 
                                if (select == _selectedContainer || select == null)
                                    return up ? _selectedContainer.HandleUpKey() : _selectedContainer.HandleDownKey();

                                break; 
                            }
                            else 
                            { 
                                // This item does fit, but we should continue searching
                                select = next; 
                            }
                        }

                        index = index + (up ? -1 : 1); 
                        if ((0 <= index) && (index < count))
                        { 
                            next = parent.ItemContainerGenerator.ContainerFromIndex(index) as TreeViewItem; 
                        }
                        else if (parent == this) 
                        {
                            // That was the last item in the TreeView
                            next = null;
                        } 
                        else
                        { 
                            // Go up the parent chain to a parent with another item 
                            while (parent != null)
                            { 
                                ItemsControl oldParent = parent;
                                parent = ItemsControl.ItemsControlFromItemContainer(parent);
                                if (parent != null)
                                { 
                                    count = parent.Items.Count;
                                    index = parent.ItemContainerGenerator.IndexFromContainer(oldParent) + (up ? -1 : 1); 
                                    if ((0 <= index) && (index < count)) 
                                    {
                                        next = parent.ItemContainerGenerator.ContainerFromIndex(index) as TreeViewItem; 
                                        break;
                                    }
                                    else if (parent == this)
                                    { 
                                        // That was the last item in the TreeView
                                        parent = next = null; 
                                    } 
                                }
                            } 
                        }
                    }

                    if (select != null) 
                    {
                        // Earlier we found an item that fit but didn't focus it at that time 
                        if (up) 
                        {
                            if (select != _selectedContainer) 
                            {
                                return select.Focus();
                            }
                        } 
                        else
                        { 
                            return TreeViewItem.FocusIntoItem(select); 
                        }
                    } 
                }
            }

            return false; 
        }
 
        ///  
        /// Used when virtualization is on.  Moves the viewport so that HandleScrollByPage will work.
        ///  
        private void PreScrollByPage(ScrollViewer scroller, bool up)
        {
            double startTop, startBottom;   // offset of the top and bottom of the selected element from the top of the viewport
            double adjustmentOffset = 0d;   // offset we'll have to move the viewport in order to do the 'classic' page up / down algorithm 
            double distance = 0d;           // distance from the furthest edge of either the selected item or the item we're about to select to the nearest edge of the viewport;
            double viewportHeight = scroller.ViewportHeight; 
            _selectedContainer.GetTopAndBottom(scroller, out startTop, out startBottom); 

            // 
            // What we're doing here:
            // The TreeView page down algorithm walks the tree looking for the item a page up / down from the currently selected item.
            // When virtualized that algorithm breaks down since some items aren't generated.
            // 
            // VSP maintains one page of containers above and below the viewport.
            // Here we move the viewport so that the selected item or newly selected item, whichever is further, 
            // is only 1 page away from the top or bottom of the viewport.  That allows us to use the existing 
            // non-virtualization algorithm.
            // 

            if (startBottom < 0)
            {
                // The selected item is above the viewport; if we page up the furthest distance will be the soon-to-be selected item 
                distance = startBottom;
 
                if (up) 
                {
                    distance -= viewportHeight; 
                }
            }
            else if (startTop > 0)
            { 
                // The selected item is below the viewport; if we page down the furthest distance will be the soon-to-be selected item
                distance = startTop - viewportHeight; 
 
                if (!up)
                { 
                    distance += viewportHeight;
                }
            }
 
            if (Math.Abs(distance) > viewportHeight)
            { 
                // The selected or soon-to-be selected item is more than one page away from the nearest edge of the viewport. 
                // Create the adjustment offset. Distance is delta between the viewport edge and the furthest item;
                // we have to adjust by the viewport height to have the furthest item 1 page from the viewport 
                adjustmentOffset = distance < 0 ? distance + viewportHeight : distance - viewportHeight;

                //
                // Set the offset and update layout to ensure that all containers on the path from the selected item to the 
                // item we want to select are generated.
                // 
                scroller.ScrollToVerticalOffset(scroller.VerticalOffset + adjustmentOffset); 
                ContextLayoutManager layoutManager = ContextLayoutManager.From(Dispatcher);
                layoutManager.UpdateLayout(); 
            }
        }

 
        /// 
        /// Scrolls to either the start or end of the viewport.  Similar to the code in ItemsControl.NavigateToItem 
        ///  
        /// 
        ///  
        /// 
        private void ScrollToEdge(bool toEnd)
        {
            double oldOffset; 
            double newOffset = 0.0;
            ScrollViewer scrollViewer = ScrollHost; 
            bool isHorizontal = IsLogicalHorizontal; 

            if (scrollViewer != null) 
            {
                if (toEnd)
                {
                    newOffset = isHorizontal ? scrollViewer.ExtentWidth : scrollViewer.ExtentHeight; 
                }
 
                // 
                // This loop ensures that if the extent changes (new items may be generated as we move the viewport)
                // that we'll try again until the viewport really reaches the offset. 
                //

                while (MakeVisible((int)newOffset, false, false))
                { 
                    oldOffset = isHorizontal ? scrollViewer.HorizontalOffset : scrollViewer.VerticalOffset;
 
                    scrollViewer.UpdateLayout(); 

                    // If offset does not change - exit the loop 
                    if (DoubleUtil.AreClose(oldOffset, isHorizontal ? scrollViewer.HorizontalOffset : scrollViewer.VerticalOffset))
                        break;

                    if (toEnd) 
                    {
                        newOffset = isHorizontal ? scrollViewer.ExtentWidth : scrollViewer.ExtentHeight; 
                    } 
                }
            } 
        }

        #endregion
 
        #region IsSelectionActive
 
        ///  
        ///     An event reporting that the IsKeyboardFocusWithin property changed.
        ///  
        protected override void OnIsKeyboardFocusWithinChanged(DependencyPropertyChangedEventArgs e)
        {
            base.OnIsKeyboardFocusWithinChanged(e);
 
            // When focus within changes we need to update the value of IsSelectionActive.
            bool isSelectionActive = false; 
            bool isKeyboardFocusWithin = IsKeyboardFocusWithin; 
            if (isKeyboardFocusWithin)
            { 
                // Keyboard focus is within the control, selection should appear active.
                isSelectionActive = true;
            }
            else 
            {
                DependencyObject currentFocus = Keyboard.FocusedElement as DependencyObject; 
                if (currentFocus != null) 
                {
                    UIElement root = KeyboardNavigation.GetVisualRoot(this) as UIElement; 
                    if (root != null && root.IsKeyboardFocusWithin)
                    {
                        if (FocusManager.GetFocusScope(currentFocus) != root)
                        { 
                            isSelectionActive = true;
                        } 
                    } 
                }
            } 

            if ((bool)GetValue(Selector.IsSelectionActiveProperty) != isSelectionActive)
            {
                // The value changed, set the new value. 
                SetValue(Selector.IsSelectionActivePropertyKey, BooleanBoxes.Box(isSelectionActive));
            } 
 
            if (isKeyboardFocusWithin && IsKeyboardFocused && (_selectedContainer != null) && !_selectedContainer.IsKeyboardFocusWithin)
            { 
                _selectedContainer.Focus();
            }
        }
 
        private void OnFocusEnterMainFocusScope(object sender, EventArgs e)
        { 
            // When KeyboardFocus comes back to the main focus scope and the TreeView does not have focus within- clear IsSelectionActivePrivateProperty 
            if (!IsKeyboardFocusWithin)
            { 
                ClearValue(Selector.IsSelectionActivePropertyKey);
            }
        }
 
        private static DependencyObject FindParent(DependencyObject o)
        { 
            Visual v = o as Visual; 
            ContentElement ce = (v == null) ? o as ContentElement : null;
 
            if (ce != null)
            {
                o = ContentOperations.GetParent(ce);
                if (o != null) 
                {
                    return o; 
                } 
                else
                { 
                    FrameworkContentElement fce = ce as FrameworkContentElement;
                    if (fce != null)
                    {
                        return fce.Parent; 
                    }
                } 
            } 
            else if (v != null)
            { 
                return VisualTreeHelper.GetParent(v);
            }

            return null; 
        }
 
        #endregion 

        #region Automation 

        /// 
        /// Creates AutomationPeer ()
        ///  
        protected override AutomationPeer OnCreateAutomationPeer()
        { 
            return new TreeViewAutomationPeer(this); 
        }
 
        #endregion

        #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 Container Size Estimate
 
        //
        // Helper methods for TreeViewItem to compute an estimated container size.  This is used by VSP when virtualizing.
        //
 
        internal Size CurrentContainerSizeEstimate
        { 
            get 
            {
                double estimate = ContainerSizeEstimateField.GetValue(this); 
                return new Size(estimate, estimate);
            }
        }
 
        /// 
        /// TreeViewItem calls into this to provide its size in the stacking direction.  We use this to compute 
        /// the most common size. 
        /// 
        ///  
        internal void RegisterContainerSize(double containerSize)
        {
            bool found = false;
            ContainerSize newSize = new ContainerSize(containerSize); 
            ContainerSize mostCommon = newSize;
 
            List sizes = EnsureContainerSizeCount(); 

            for (int i = 0; i < sizes.Count; i++) 
            {
                ContainerSize size = sizes[i];
                if (size.IsCloseTo(newSize) && size.NumContainers < uint.MaxValue)
                { 
                    size.NumContainers++;
                    found = true; 
                } 

                if (size.NumContainers > mostCommon.NumContainers) 
                {
                    mostCommon = size;
                }
            } 

            // Only track the first 5 sizes we see. Container size could have a long tail and that ought to be sufficient. 
            if (!found && sizes.Count < 5) 
            {
                sizes.Add(newSize); 
            }


            // Update the estimate 
            ContainerSizeEstimateField.SetValue(this, mostCommon.Size);
        } 
 

        private struct ContainerSize 
        {
            public ContainerSize(double size)
            {
                Size = size; 
                NumContainers = 1;
            } 
 
            /// 
            /// We'll say two container sizes are close if they're within a half pixel. 
            /// 
            /// 
            /// 
            public bool IsCloseTo(ContainerSize size) 
            {
                return Math.Abs(Size - size.Size) < 0.5; 
            } 

            public double Size; 
            public uint NumContainers;
        }

 
        private List EnsureContainerSizeCount()
        { 
            List sizes = ContainerSizeCountField.GetValue(this); 

            if (sizes == null) 
            {
                sizes = new List();
                ContainerSizeCountField.SetValue(this, sizes);
            } 

            return sizes; 
        } 

        #endregion 

        #endregion

        #region Data 

        private enum Bits 
        { 
            IsSelectionChangeActive     = 0x1,
        } 

        // Packed boolean information
        private BitVector32 _bits = new BitVector32(0);
 
        private TreeViewItem _selectedContainer;
 
        // Used to retrieve the value of an item, according to the SelectedValuePath 
        private static readonly BindingExpressionUncommonField SelectedValuePathBindingExpression = new BindingExpressionUncommonField();
        private EventHandler _focusEnterMainFocusScopeEventHandler; 

        // Used to estimate the most common container size
        private static UncommonField> ContainerSizeCountField = new UncommonField>();
        private static UncommonField ContainerSizeEstimateField = new UncommonField(); 

        #endregion 
    } 
}
 

// File provided for Reference Use Only by Microsoft Corporation (c) 2007.
// Copyright (c) Microsoft Corporation. All rights reserved.
//---------------------------------------------------------------------------- 
//
// Copyright (C) Microsoft Corporation.  All rights reserved.
//
//--------------------------------------------------------------------------- 

using System; 
using System.Collections; 
using System.Collections.Generic;
using System.Collections.Specialized; 
using System.ComponentModel;
using System.Diagnostics;
using System.Windows;
using System.Windows.Automation.Peers; 
using System.Windows.Controls.Primitives;
using System.Windows.Data; 
using System.Windows.Input; 
using System.Windows.Media;
using MS.Internal; 
using MS.Internal.Data;
using MS.Internal.KnownBoxes;

namespace System.Windows.Controls 
{
    ///  
    ///     A control that presents items in a tree structure. 
    /// 
    [StyleTypedProperty(Property = "ItemContainerStyle", StyleTargetType = typeof(TreeViewItem))] 
    public class TreeView : ItemsControl
    {
        #region Constructors
 
        static TreeView()
        { 
            DefaultStyleKeyProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(typeof(TreeView))); 
            VirtualizingStackPanel.IsVirtualizingProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(BooleanBoxes.FalseBox));
            _dType = DependencyObjectType.FromSystemTypeInternal(typeof(TreeView)); 

            KeyboardNavigation.DirectionalNavigationProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(KeyboardNavigationMode.Contained));
            KeyboardNavigation.TabNavigationProperty.OverrideMetadata(typeof(TreeView), new FrameworkPropertyMetadata(KeyboardNavigationMode.None));
        } 

        ///  
        ///     Creates an instance of this control. 
        /// 
        public TreeView() 
        {
            _focusEnterMainFocusScopeEventHandler = new EventHandler(OnFocusEnterMainFocusScope);
            KeyboardNavigation.Current.FocusEnterMainFocusScope += _focusEnterMainFocusScopeEventHandler;
        } 

        #endregion 
 
        #region Public Properties
 
        private static readonly DependencyPropertyKey SelectedItemPropertyKey =
            DependencyProperty.RegisterReadOnly("SelectedItem", typeof(object), typeof(TreeView), new FrameworkPropertyMetadata((object)null));

        ///  
        ///     The DependencyProperty for the  property.
        ///     Default Value: null 
        ///  
        public static readonly DependencyProperty SelectedItemProperty = SelectedItemPropertyKey.DependencyProperty;
 
        /// 
        ///     Specifies the selected item.
        /// 
        [Bindable(true), Category("Appearance"), ReadOnly(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
        public object SelectedItem
        { 
            get 
            {
                return GetValue(SelectedItemProperty); 
            }
        }

        private void SetSelectedItem(object data) 
        {
            if (SelectedItem != data) 
            { 
                SetValue(SelectedItemPropertyKey, data);
            } 
        }

        private static readonly DependencyPropertyKey SelectedValuePropertyKey =
            DependencyProperty.RegisterReadOnly("SelectedValue", typeof(object), typeof(TreeView), new FrameworkPropertyMetadata((object)null)); 

        ///  
        ///     The DependencyProperty for the  property. 
        ///     Default Value: null
        ///  
        public static readonly DependencyProperty SelectedValueProperty = SelectedValuePropertyKey.DependencyProperty;

        /// 
        ///     Specifies the a value on the selected item as defined by . 
        /// 
        [Bindable(true), Category("Appearance"), ReadOnly(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
        public object SelectedValue 
        {
            get 
            {
                return GetValue(SelectedValueProperty);
            }
        } 

        private void SetSelectedValue(object data) 
        { 
            if (SelectedValue != data)
            { 
                SetValue(SelectedValuePropertyKey, data);
            }
        }
 
        /// 
        ///     The DependencyProperty for the  property. 
        ///     Default Value: String.Empty 
        /// 
        public static readonly DependencyProperty SelectedValuePathProperty = 
            DependencyProperty.Register(
                    "SelectedValuePath",
                    typeof(string),
                    typeof(TreeView), 
                    new FrameworkPropertyMetadata(
                            String.Empty, 
                            new PropertyChangedCallback(OnSelectedValuePathChanged))); 

        ///  
        ///     Specifies the path to query on  to calculate .
        /// 
        [Bindable(true), Category("Appearance")]
        public string SelectedValuePath 
        {
            get { return (string) GetValue(SelectedValuePathProperty); } 
            set { SetValue(SelectedValuePathProperty, value); } 
        }
 
        private static void OnSelectedValuePathChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            TreeView tree = (TreeView)d;
            SelectedValuePathBindingExpression.ClearValue(tree); 
            tree.UpdateSelectedValue(tree.SelectedItem);
        } 
 
        #endregion
 
        #region Public Events

        /// 
        ///     Event fired when  changes. 
        /// 
        public static readonly RoutedEvent SelectedItemChangedEvent = EventManager.RegisterRoutedEvent("SelectedItemChanged", RoutingStrategy.Bubble, typeof(RoutedPropertyChangedEventHandler), typeof(TreeView)); 
 
        /// 
        ///     Event fired when  changes. 
        /// 
        [Category("Behavior")]
        public event RoutedPropertyChangedEventHandler SelectedItemChanged
        { 
            add
            { 
                AddHandler(SelectedItemChangedEvent, value); 
            }
 
            remove
            {
                RemoveHandler(SelectedItemChangedEvent, value);
            } 
        }
 
        ///  
        ///     Called when  changes.
        ///     Default implementation fires the  event. 
        /// 
        /// Event arguments.
        protected virtual void OnSelectedItemChanged(RoutedPropertyChangedEventArgs e)
        { 
            //
            RaiseEvent(e); 
        } 

        #endregion 

        #region Implementation

        #region Selection 

        internal void ChangeSelection(object data, TreeViewItem container, bool selected) 
        { 
            if (IsSelectionChangeActive)
            { 
                return;
            }

            object oldValue = null; 
            object newValue = null;
            bool changed = false; 
            TreeViewItem oldContainer = _selectedContainer; // Saved for the automation event 

            IsSelectionChangeActive = true; 

            try
            {
                if (selected) 
                {
                    if (container != _selectedContainer) 
                    { 
                        oldValue = SelectedItem;
                        newValue = data; 

                        if (_selectedContainer != null)
                        {
                            _selectedContainer.IsSelected = false; 
                            _selectedContainer.UpdateContainsSelection(false);
                        } 
                        _selectedContainer = container; 
                        _selectedContainer.UpdateContainsSelection(true);
                        SetSelectedItem(data); 
                        UpdateSelectedValue(data);
                        changed = true;
                    }
                } 
                else
                { 
                    if (container == _selectedContainer) 
                    {
                        _selectedContainer.UpdateContainsSelection(false); 
                        _selectedContainer = null;
                        SetSelectedItem(null);

                        oldValue = data; 
                        changed = true;
                    } 
                } 

                if (container.IsSelected != selected) 
                {
                    container.IsSelected = selected;
                }
            } 
            finally
            { 
                IsSelectionChangeActive = false; 
            }
 
            if (changed)
            {
                if (    _selectedContainer != null
                    &&  AutomationPeer.ListenerExists(AutomationEvents.SelectionItemPatternOnElementSelected)   ) 
                {
                    AutomationPeer peer = UIElementAutomationPeer.CreatePeerForElement(_selectedContainer); 
                    if (peer != null) 
                        peer.RaiseAutomationEvent(AutomationEvents.SelectionItemPatternOnElementSelected);
                } 

                if (    oldContainer != null
                    &&  AutomationPeer.ListenerExists(AutomationEvents.SelectionItemPatternOnElementRemovedFromSelection)   )
                { 
                    AutomationPeer peer = UIElementAutomationPeer.CreatePeerForElement(oldContainer);
                    if (peer != null) 
                        peer.RaiseAutomationEvent(AutomationEvents.SelectionItemPatternOnElementRemovedFromSelection); 
                }
 
                RoutedPropertyChangedEventArgs e = new RoutedPropertyChangedEventArgs(oldValue, newValue, SelectedItemChangedEvent);
                OnSelectedItemChanged(e);
            }
        } 

        internal bool IsSelectionChangeActive 
        { 
            get { return _bits[(int)Bits.IsSelectionChangeActive]; }
            set { _bits[(int)Bits.IsSelectionChangeActive] = value; } 
        }

        private void UpdateSelectedValue(object selectedItem)
        { 
            BindingExpression expression = PrepareSelectedValuePathBindingExpression(selectedItem);
 
            if (expression != null) 
            {
                expression.Activate(selectedItem); 
                object selectedValue = expression.Value;
                expression.Deactivate();

                SetValue(SelectedValuePropertyKey, selectedValue); 
            }
            else 
            { 
                ClearValue(SelectedValuePropertyKey);
            } 
        }

        private BindingExpression PrepareSelectedValuePathBindingExpression(object item)
        { 
            if (item == null)
            { 
                return null; 
            }
 
            Binding binding;
            bool useXml = XmlHelper.IsXmlNode(item);

            BindingExpression bindingExpr = SelectedValuePathBindingExpression.GetValue(this); 

            // replace existing binding if it's the wrong kind 
            if (bindingExpr != null) 
            {
                binding = bindingExpr.ParentBinding; 
                bool usesXml = (binding.XPath != null);
                if (usesXml != useXml)
                {
                    bindingExpr = null; 
                }
            } 
 
            if (bindingExpr == null)
            { 
                // create the binding
                binding = new Binding();
                binding.Source = item;
 
                if (useXml)
                { 
                    binding.XPath = SelectedValuePath; 
                    binding.Path = new PropertyPath("/InnerText");
                } 
                else
                {
                    binding.Path = new PropertyPath(SelectedValuePath);
                } 

                bindingExpr = (BindingExpression)BindingExpression.CreateUntargetedBindingExpression(this, binding); 
                SelectedValuePathBindingExpression.SetValue(this, bindingExpr); 
            }
 
            return bindingExpr;
        }

        internal void HandleSelectionAndCollapsed(TreeViewItem collapsed) 
        {
            if ((_selectedContainer != null) && (_selectedContainer != collapsed)) 
            { 
                // Check if current selection is under the collapsed element
                TreeViewItem current = _selectedContainer; 
                do
                {
                    current = current.ParentTreeViewItem;
                    if (current == collapsed) 
                    {
                        TreeViewItem oldContainer = _selectedContainer; 
 
                        ChangeSelection(collapsed.ParentItemsControl.ItemContainerGenerator.ItemFromContainer(collapsed), collapsed, true);
 
                        if (oldContainer.IsKeyboardFocusWithin)
                        {
                            // If the oldContainer had focus then move focus to the newContainer instead
                            _selectedContainer.Focus(); 
                        }
 
                        break; 
                    }
                } 
                while (current != null);
            }
        }
 
        // This method is called when MouseButonDown on TreeViewItem and also listen for handled events too
        // The purpose is to restore focus on TreeView when mouse is clicked and focus was outside the TreeView 
        // Focus goes either to selected item (if any) or treeview itself 
        internal void HandleMouseButtonDown()
        { 
            if (!this.IsKeyboardFocusWithin)
            {
                if (_selectedContainer != null)
                { 
                    if (!_selectedContainer.IsKeyboardFocused)
                        _selectedContainer.Focus(); 
                } 
                else
                { 
                    // If we don't have a selection - just focus the treeview
                    this.Focus();
                }
            } 
        }
 
        #endregion 

        #region Containers 

        /// 
        ///     Returns true if the item is or should be its own container.
        ///  
        /// The item to test.
        /// true if its type matches the container type. 
        protected override bool IsItemItsOwnContainerOverride(object item) 
        {
            return item is TreeViewItem; 
        }

        /// 
        ///     Create or identify the element used to display the given item. 
        /// 
        /// The container. 
        protected override DependencyObject GetContainerForItemOverride() 
        {
            return new TreeViewItem(); 
        }

        /// 
        ///     This method is invoked when the Items property changes. 
        /// 
        protected override void OnItemsChanged(NotifyCollectionChangedEventArgs e) 
        { 
            switch (e.Action)
            { 
                case NotifyCollectionChangedAction.Remove:
                case NotifyCollectionChangedAction.Reset:
                    if ((SelectedItem != null) && !IsSelectedContainerHookedUp)
                    { 
                        SelectFirstItem();
                    } 
                    break; 

                case NotifyCollectionChangedAction.Replace: 
                    {
                        // If old item is selected - remove the selection
                        // Revisit the condition when we support duplicate items in Items collection: if e.OldItems[0] is the same as selected items we will unselect the selected item
                        object selectedItem = SelectedItem; 
                        if ((selectedItem != null) && selectedItem.Equals(e.OldItems[0]))
                        { 
                            ChangeSelection(selectedItem, _selectedContainer, false); 
                        }
                    } 
                    break;

                case NotifyCollectionChangedAction.Add:
                case NotifyCollectionChangedAction.Move: 
                    break;
 
                default: 
                    throw new NotSupportedException(SR.Get(SRID.UnexpectedCollectionChangeAction, e.Action));
            } 
        }

        /// 
        /// Send down the IsVirtualizing property if it's set on this element. 
        /// 
        ///  
        ///  
        protected override void PrepareContainerForItemOverride(DependencyObject element, object item)
        { 
            base.PrepareContainerForItemOverride(element, item);
            TreeViewItem.IsVirtualizingPropagationHelper(this, element);
        }
 
        private void SelectFirstItem()
        { 
            object item; 
            TreeViewItem container;
            bool selected = GetFirstItem(out item, out container); 
            if (!selected)
            {
                item = SelectedItem;
                container = _selectedContainer; 
            }
 
            ChangeSelection(item, container, selected); 
        }
 
        private bool GetFirstItem(out object item, out TreeViewItem container)
        {
            if (HasItems)
            { 
                item = Items[0];
                container = ItemContainerGenerator.ContainerFromIndex(0) as TreeViewItem; 
                return ((item != null) && (container != null)); 
            }
            else 
            {
                item = null;
                container = null;
                return false; 
            }
        } 
 
        internal bool IsSelectedContainerHookedUp
        { 
            get
            {
                return (_selectedContainer != null) && (_selectedContainer.ParentTreeView == this);
            } 
        }
 
        internal TreeViewItem SelectedContainer 
        {
            get 
            {
                return _selectedContainer;
            }
        } 

        #endregion 
 
        #region Input
 
        /// 
        ///     If control has a scrollviewer in its style and has a custom keyboard scrolling behavior when HandlesScrolling should return true.
        /// Then ScrollViewer will not handle keyboard input and leave it up to the control.
        ///  
        protected internal override bool HandlesScrolling
        { 
            get { return true; } 
        }
 
        /// 
        ///     Called when a keyboard key is pressed down.
        /// 
        /// Event Arguments 
        protected override void OnKeyDown(KeyEventArgs e)
        { 
            base.OnKeyDown(e); 
            if (!e.Handled)
            { 
                if (IsControlKeyDown)
                {
                    switch (e.Key)
                    { 
                        case Key.Up:
                        case Key.Down: 
                        case Key.Left: 
                        case Key.Right:
                        case Key.Home: 
                        case Key.End:
                        case Key.PageUp:
                        case Key.PageDown:
                            if (HandleScrollKeys(e.Key)) 
                            {
                                e.Handled = true; 
                            } 
                            break;
                    } 
                }
                else
                {
                    switch (e.Key) 
                    {
                        case Key.Up: 
                        case Key.Down: 
                            if ((_selectedContainer == null) && FocusFirstItem())
                            { 
                                e.Handled = true;
                            }
                            break;
 
                        case Key.Home:
                            if (FocusFirstItem()) 
                            { 
                                e.Handled = true;
                            } 
                            break;

                        case Key.End:
                            if (FocusLastItem()) 
                            {
                                e.Handled = true; 
                            } 
                            break;
 
                        case Key.PageUp:
                        case Key.PageDown:
                            if (_selectedContainer == null)
                            { 
                                if (FocusFirstItem())
                                { 
                                    e.Handled = true; 
                                }
                            } 
                            else if (HandleScrollByPage(e.Key == Key.PageUp))
                            {
                                e.Handled = true;
                            } 
                            break;
 
                        case Key.Tab: 
                            if (IsShiftKeyDown && IsKeyboardFocusWithin)
                            { 
                                // SHIFT-TAB behavior for KeyboardNavigation needs to happen at the TreeView level
                                if (MoveFocus(new TraversalRequest(FocusNavigationDirection.Previous)))
                                {
                                    e.Handled = true; 
                                }
                            } 
                            break; 
                    }
                } 
            }
        }

        private static bool IsControlKeyDown 
        {
            get 
            { 
                return ((Keyboard.Modifiers & ModifierKeys.Control) == (ModifierKeys.Control));
            } 
        }

        private static bool IsShiftKeyDown
        { 
            get
            { 
                return ((Keyboard.Modifiers & ModifierKeys.Shift) == (ModifierKeys.Shift)); 
            }
        } 

        private bool FocusFirstItem()
        {
            if (IsVirtualizing) 
            {
                ScrollToEdge(/*toEnd = */ false); 
            } 

            TreeViewItem item = ItemContainerGenerator.ContainerFromIndex(0) as TreeViewItem; 
            if (item != null)
            {
                if (item.IsEnabled && item.Focus())
                { 
                    return true;
                } 
                else 
                {
                    return item.FocusDown(); 
                }
            }

            return false; 
        }
 
 
        private bool FocusLastItem()
        { 
            //
            // If virtualizing first scroll to the end so that the last item will be generated.
            //
            if (IsVirtualizing) 
            {
                ScrollToEdge(/* toEnd = */ true); 
            } 

            int index = Items.Count - 1; 
            while (index >= 0)
            {
                TreeViewItem item = ItemContainerGenerator.ContainerFromIndex(index) as TreeViewItem;
                if ((item != null) && item.IsEnabled) 
                {
                    return TreeViewItem.FocusIntoItem(item); 
                } 
                index--;
            } 

            return false;
        }
 
        private bool HandleScrollKeys(Key key)
        { 
            ScrollViewer scroller = ScrollHost; 
            if (scroller != null)
            { 
                bool invert = (FlowDirection == FlowDirection.RightToLeft);
                switch (key)
                {
                    case Key.Up: 
                        scroller.LineUp();
                        return true; 
 
                    case Key.Down:
                        scroller.LineDown(); 
                        return true;

                    case Key.Left:
                        if (invert) 
                        {
                            scroller.LineRight(); 
                        } 
                        else
                        { 
                            scroller.LineLeft();
                        }
                        return true;
 
                    case Key.Right:
                        if (invert) 
                        { 
                            scroller.LineLeft();
                        } 
                        else
                        {
                            scroller.LineRight();
                        } 
                        return true;
 
                    case Key.Home: 
                        scroller.ScrollToTop();
                        return true; 

                    case Key.End:
                        scroller.ScrollToBottom();
                        return true; 

                    case Key.PageUp: 
                        //if vertically scrollable - go vertical, otherwise horizontal 
                        if(DoubleUtil.GreaterThan(scroller.ExtentHeight, scroller.ViewportHeight))
                        { 
                            scroller.PageUp();
                        }
                        else
                        { 
                            scroller.PageLeft();
                        } 
                        return true; 

                    case Key.PageDown: 
                        //if vertically scrollable - go vertical, otherwise horizontal
                        if(DoubleUtil.GreaterThan(scroller.ExtentHeight, scroller.ViewportHeight))
                        {
                            scroller.PageDown(); 
                        }
                        else 
                        { 
                            scroller.PageRight();
                        } 
                        return true;

                }
            } 

            return false; 
        } 

        // Note the two assumptions TreeView is making here 
        // 1.) that everything is laid out vertically and
        // 2.) that the headers of TreeViewItems always appear above the ItemsPresenter.
        private bool HandleScrollByPage(bool up)
        { 
            ScrollViewer scroller = ScrollHost;
            if (scroller != null) 
            { 
                double viewportHeight = scroller.ViewportHeight;
                double startTop, startBottom; 


                if (VirtualizingStackPanel.GetIsVirtualizing(this))
                { 
                    PreScrollByPage(scroller, up);
                } 
 
                _selectedContainer.GetTopAndBottom(scroller, out startTop, out startBottom);
 
                TreeViewItem select = null;
                TreeViewItem next = _selectedContainer;
                ItemsControl parent = _selectedContainer.ParentItemsControl;
 
                if (parent != null)
                { 
                    if (up) 
                    {
                        // When going up, we need to start at the first level of TreeViewItems. 
                        // When going down, we need to start at the selected container.

                        while (parent != this)
                        { 
                            ItemsControl nextParent = ItemsControl.ItemsControlFromItemContainer(parent);
                            if (nextParent == null) 
                            { 
                                break;
                            } 
                            else
                            {
                                next = (TreeViewItem)parent;
                                parent = nextParent; 
                            }
                        } 
                    } 

                    int index = parent.ItemContainerGenerator.IndexFromContainer(next); 
                    int count = parent.Items.Count;

                    while ((parent != null) && (next != null))
                    { 
                        if (next.IsEnabled)
                        { 
                            double delta; 
                            if (next.HandleScrollByPage(up, scroller, viewportHeight, startTop, startBottom, out delta))
                            { 
                                // This item or one of its children was focused
                                return true;
                            }
                            else if (DoubleUtil.GreaterThan(delta, viewportHeight)) 
                            {
                                // This item does not fit 
 
                                // If select target is already the same element as _selectedContainer - there is no point to select it again
                                // In this case we select the next item although it cannot completely fit into view 
                                if (select == _selectedContainer || select == null)
                                    return up ? _selectedContainer.HandleUpKey() : _selectedContainer.HandleDownKey();

                                break; 
                            }
                            else 
                            { 
                                // This item does fit, but we should continue searching
                                select = next; 
                            }
                        }

                        index = index + (up ? -1 : 1); 
                        if ((0 <= index) && (index < count))
                        { 
                            next = parent.ItemContainerGenerator.ContainerFromIndex(index) as TreeViewItem; 
                        }
                        else if (parent == this) 
                        {
                            // That was the last item in the TreeView
                            next = null;
                        } 
                        else
                        { 
                            // Go up the parent chain to a parent with another item 
                            while (parent != null)
                            { 
                                ItemsControl oldParent = parent;
                                parent = ItemsControl.ItemsControlFromItemContainer(parent);
                                if (parent != null)
                                { 
                                    count = parent.Items.Count;
                                    index = parent.ItemContainerGenerator.IndexFromContainer(oldParent) + (up ? -1 : 1); 
                                    if ((0 <= index) && (index < count)) 
                                    {
                                        next = parent.ItemContainerGenerator.ContainerFromIndex(index) as TreeViewItem; 
                                        break;
                                    }
                                    else if (parent == this)
                                    { 
                                        // That was the last item in the TreeView
                                        parent = next = null; 
                                    } 
                                }
                            } 
                        }
                    }

                    if (select != null) 
                    {
                        // Earlier we found an item that fit but didn't focus it at that time 
                        if (up) 
                        {
                            if (select != _selectedContainer) 
                            {
                                return select.Focus();
                            }
                        } 
                        else
                        { 
                            return TreeViewItem.FocusIntoItem(select); 
                        }
                    } 
                }
            }

            return false; 
        }
 
        ///  
        /// Used when virtualization is on.  Moves the viewport so that HandleScrollByPage will work.
        ///  
        private void PreScrollByPage(ScrollViewer scroller, bool up)
        {
            double startTop, startBottom;   // offset of the top and bottom of the selected element from the top of the viewport
            double adjustmentOffset = 0d;   // offset we'll have to move the viewport in order to do the 'classic' page up / down algorithm 
            double distance = 0d;           // distance from the furthest edge of either the selected item or the item we're about to select to the nearest edge of the viewport;
            double viewportHeight = scroller.ViewportHeight; 
            _selectedContainer.GetTopAndBottom(scroller, out startTop, out startBottom); 

            // 
            // What we're doing here:
            // The TreeView page down algorithm walks the tree looking for the item a page up / down from the currently selected item.
            // When virtualized that algorithm breaks down since some items aren't generated.
            // 
            // VSP maintains one page of containers above and below the viewport.
            // Here we move the viewport so that the selected item or newly selected item, whichever is further, 
            // is only 1 page away from the top or bottom of the viewport.  That allows us to use the existing 
            // non-virtualization algorithm.
            // 

            if (startBottom < 0)
            {
                // The selected item is above the viewport; if we page up the furthest distance will be the soon-to-be selected item 
                distance = startBottom;
 
                if (up) 
                {
                    distance -= viewportHeight; 
                }
            }
            else if (startTop > 0)
            { 
                // The selected item is below the viewport; if we page down the furthest distance will be the soon-to-be selected item
                distance = startTop - viewportHeight; 
 
                if (!up)
                { 
                    distance += viewportHeight;
                }
            }
 
            if (Math.Abs(distance) > viewportHeight)
            { 
                // The selected or soon-to-be selected item is more than one page away from the nearest edge of the viewport. 
                // Create the adjustment offset. Distance is delta between the viewport edge and the furthest item;
                // we have to adjust by the viewport height to have the furthest item 1 page from the viewport 
                adjustmentOffset = distance < 0 ? distance + viewportHeight : distance - viewportHeight;

                //
                // Set the offset and update layout to ensure that all containers on the path from the selected item to the 
                // item we want to select are generated.
                // 
                scroller.ScrollToVerticalOffset(scroller.VerticalOffset + adjustmentOffset); 
                ContextLayoutManager layoutManager = ContextLayoutManager.From(Dispatcher);
                layoutManager.UpdateLayout(); 
            }
        }

 
        /// 
        /// Scrolls to either the start or end of the viewport.  Similar to the code in ItemsControl.NavigateToItem 
        ///  
        /// 
        ///  
        /// 
        private void ScrollToEdge(bool toEnd)
        {
            double oldOffset; 
            double newOffset = 0.0;
            ScrollViewer scrollViewer = ScrollHost; 
            bool isHorizontal = IsLogicalHorizontal; 

            if (scrollViewer != null) 
            {
                if (toEnd)
                {
                    newOffset = isHorizontal ? scrollViewer.ExtentWidth : scrollViewer.ExtentHeight; 
                }
 
                // 
                // This loop ensures that if the extent changes (new items may be generated as we move the viewport)
                // that we'll try again until the viewport really reaches the offset. 
                //

                while (MakeVisible((int)newOffset, false, false))
                { 
                    oldOffset = isHorizontal ? scrollViewer.HorizontalOffset : scrollViewer.VerticalOffset;
 
                    scrollViewer.UpdateLayout(); 

                    // If offset does not change - exit the loop 
                    if (DoubleUtil.AreClose(oldOffset, isHorizontal ? scrollViewer.HorizontalOffset : scrollViewer.VerticalOffset))
                        break;

                    if (toEnd) 
                    {
                        newOffset = isHorizontal ? scrollViewer.ExtentWidth : scrollViewer.ExtentHeight; 
                    } 
                }
            } 
        }

        #endregion
 
        #region IsSelectionActive
 
        ///  
        ///     An event reporting that the IsKeyboardFocusWithin property changed.
        ///  
        protected override void OnIsKeyboardFocusWithinChanged(DependencyPropertyChangedEventArgs e)
        {
            base.OnIsKeyboardFocusWithinChanged(e);
 
            // When focus within changes we need to update the value of IsSelectionActive.
            bool isSelectionActive = false; 
            bool isKeyboardFocusWithin = IsKeyboardFocusWithin; 
            if (isKeyboardFocusWithin)
            { 
                // Keyboard focus is within the control, selection should appear active.
                isSelectionActive = true;
            }
            else 
            {
                DependencyObject currentFocus = Keyboard.FocusedElement as DependencyObject; 
                if (currentFocus != null) 
                {
                    UIElement root = KeyboardNavigation.GetVisualRoot(this) as UIElement; 
                    if (root != null && root.IsKeyboardFocusWithin)
                    {
                        if (FocusManager.GetFocusScope(currentFocus) != root)
                        { 
                            isSelectionActive = true;
                        } 
                    } 
                }
            } 

            if ((bool)GetValue(Selector.IsSelectionActiveProperty) != isSelectionActive)
            {
                // The value changed, set the new value. 
                SetValue(Selector.IsSelectionActivePropertyKey, BooleanBoxes.Box(isSelectionActive));
            } 
 
            if (isKeyboardFocusWithin && IsKeyboardFocused && (_selectedContainer != null) && !_selectedContainer.IsKeyboardFocusWithin)
            { 
                _selectedContainer.Focus();
            }
        }
 
        private void OnFocusEnterMainFocusScope(object sender, EventArgs e)
        { 
            // When KeyboardFocus comes back to the main focus scope and the TreeView does not have focus within- clear IsSelectionActivePrivateProperty 
            if (!IsKeyboardFocusWithin)
            { 
                ClearValue(Selector.IsSelectionActivePropertyKey);
            }
        }
 
        private static DependencyObject FindParent(DependencyObject o)
        { 
            Visual v = o as Visual; 
            ContentElement ce = (v == null) ? o as ContentElement : null;
 
            if (ce != null)
            {
                o = ContentOperations.GetParent(ce);
                if (o != null) 
                {
                    return o; 
                } 
                else
                { 
                    FrameworkContentElement fce = ce as FrameworkContentElement;
                    if (fce != null)
                    {
                        return fce.Parent; 
                    }
                } 
            } 
            else if (v != null)
            { 
                return VisualTreeHelper.GetParent(v);
            }

            return null; 
        }
 
        #endregion 

        #region Automation 

        /// 
        /// Creates AutomationPeer ()
        ///  
        protected override AutomationPeer OnCreateAutomationPeer()
        { 
            return new TreeViewAutomationPeer(this); 
        }
 
        #endregion

        #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 Container Size Estimate
 
        //
        // Helper methods for TreeViewItem to compute an estimated container size.  This is used by VSP when virtualizing.
        //
 
        internal Size CurrentContainerSizeEstimate
        { 
            get 
            {
                double estimate = ContainerSizeEstimateField.GetValue(this); 
                return new Size(estimate, estimate);
            }
        }
 
        /// 
        /// TreeViewItem calls into this to provide its size in the stacking direction.  We use this to compute 
        /// the most common size. 
        /// 
        ///  
        internal void RegisterContainerSize(double containerSize)
        {
            bool found = false;
            ContainerSize newSize = new ContainerSize(containerSize); 
            ContainerSize mostCommon = newSize;
 
            List sizes = EnsureContainerSizeCount(); 

            for (int i = 0; i < sizes.Count; i++) 
            {
                ContainerSize size = sizes[i];
                if (size.IsCloseTo(newSize) && size.NumContainers < uint.MaxValue)
                { 
                    size.NumContainers++;
                    found = true; 
                } 

                if (size.NumContainers > mostCommon.NumContainers) 
                {
                    mostCommon = size;
                }
            } 

            // Only track the first 5 sizes we see. Container size could have a long tail and that ought to be sufficient. 
            if (!found && sizes.Count < 5) 
            {
                sizes.Add(newSize); 
            }


            // Update the estimate 
            ContainerSizeEstimateField.SetValue(this, mostCommon.Size);
        } 
 

        private struct ContainerSize 
        {
            public ContainerSize(double size)
            {
                Size = size; 
                NumContainers = 1;
            } 
 
            /// 
            /// We'll say two container sizes are close if they're within a half pixel. 
            /// 
            /// 
            /// 
            public bool IsCloseTo(ContainerSize size) 
            {
                return Math.Abs(Size - size.Size) < 0.5; 
            } 

            public double Size; 
            public uint NumContainers;
        }

 
        private List EnsureContainerSizeCount()
        { 
            List sizes = ContainerSizeCountField.GetValue(this); 

            if (sizes == null) 
            {
                sizes = new List();
                ContainerSizeCountField.SetValue(this, sizes);
            } 

            return sizes; 
        } 

        #endregion 

        #endregion

        #region Data 

        private enum Bits 
        { 
            IsSelectionChangeActive     = 0x1,
        } 

        // Packed boolean information
        private BitVector32 _bits = new BitVector32(0);
 
        private TreeViewItem _selectedContainer;
 
        // Used to retrieve the value of an item, according to the SelectedValuePath 
        private static readonly BindingExpressionUncommonField SelectedValuePathBindingExpression = new BindingExpressionUncommonField();
        private EventHandler _focusEnterMainFocusScopeEventHandler; 

        // Used to estimate the most common container size
        private static UncommonField> ContainerSizeCountField = new UncommonField>();
        private static UncommonField ContainerSizeEstimateField = new UncommonField(); 

        #endregion 
    } 
}
 

// File provided for Reference Use Only by Microsoft Corporation (c) 2007.
// Copyright (c) Microsoft Corporation. All rights reserved.

                        

                        

Link Menu

Network programming in C#, Network Programming in VB.NET, Network Programming in .NET
This book is available now!
Buy at Amazon US or
Buy at Amazon UK