|
// 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.
#region Using declarations
using System.Collections;
using System.ComponentModel;
using System.Diagnostics;
using System.Globalization;
using System.Windows.Automation;
using System.Windows.Automation.Peers;
using System.Windows.Controls.Primitives;
using System.Windows.Data;
using System.Windows.Input;
using System.Windows.Interop;
using System.Windows.Markup;
using System.Windows.Markup.Primitives;
using System.Windows.Media;
using System.Windows.Threading;
#if RIBBON_IN_FRAMEWORK
using System.Windows.Controls.Ribbon.Primitives;
using Microsoft.Windows.Controls;
#else
using Microsoft.Windows.Controls.Ribbon.Primitives;
#endif
using MS.Internal;
#if RIBBON_IN_FRAMEWORK
namespace System.Windows.Controls.Ribbon
#else
namespace Microsoft.Windows.Controls.Ribbon
#endif
{
#endregion
/// <summary>
/// A static class which defines various helper methods.
/// </summary>
internal static class RibbonHelper
{
#region ToolTip
/// <summary>
/// Helper method which serves as the coercion callback for
/// ToolTip property of ribbon controls. It creates and updates a RibbonToolTip
/// if needed and if possible and returns that as the coerced value.
/// </summary>
public static object CoerceRibbonToolTip(DependencyObject d, object value)
{
if (value == null)
{
string toolTipTitle = RibbonControlService.GetToolTipTitle(d);
string toolTipDescription = RibbonControlService.GetToolTipDescription(d);
ImageSource toolTipImageSource = RibbonControlService.GetToolTipImageSource(d);
string toolTipFooterTitle = RibbonControlService.GetToolTipFooterTitle(d);
string toolTipFooterDescription = RibbonControlService.GetToolTipFooterDescription(d);
ImageSource toolTipFooterImageSource = RibbonControlService.GetToolTipFooterImageSource(d);
if (!string.IsNullOrEmpty(toolTipTitle) ||
!string.IsNullOrEmpty(toolTipDescription) ||
toolTipImageSource != null ||
!string.IsNullOrEmpty(toolTipFooterTitle) ||
!string.IsNullOrEmpty(toolTipFooterDescription) ||
toolTipFooterImageSource != null)
{
RibbonToolTip ribbonToolTip = new RibbonToolTip
{
Title = toolTipTitle,
Description = toolTipDescription,
ImageSource = toolTipImageSource,
FooterTitle = toolTipFooterTitle,
FooterDescription = toolTipFooterDescription,
FooterImageSource = toolTipFooterImageSource
};
value = ribbonToolTip;
}
}
return value;
}
/// <summary>
/// Helper method which serves as the property changed callback for
/// properties which impact ToolTip (like ToolTipTitle etc.). It calls
/// the coercion on ToolTip property.
/// </summary>
public static void OnRibbonToolTipPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
d.CoerceValue(FrameworkElement.ToolTipProperty);
}
/// <summary>
/// Determines whether a ToolTip is available on the first Visual child of a FrameworkElement.
/// </summary>
/// <param name="visualChild">First visual child of control</param>
/// <param name="content">Content to be set as ToolTip</param>
/// <returns></returns>
public static bool GetIsContentTooltip(FrameworkElement visualChild, object content)
{
if (content == null || visualChild == null)
{
return false;
}
RibbonToolTip ribbonToolTip = visualChild.ToolTip as RibbonToolTip;
if (ribbonToolTip == null)
{
return false;
}
return content.Equals(ribbonToolTip.Title);
}
/// <summary>
/// Sets ToolTip on the first Visual child of a FrameworkElement.
/// </summary>
/// <param name="element"></param>
/// <param name="visualChild">First visual child of control</param>
/// <param name="content">content to be set as ToolTip</param>
/// <param name="value">Set or Unset ToolTip</param>
public static void SetContentAsToolTip(FrameworkElement element, FrameworkElement visualChild, object content, bool value)
{
if (visualChild != null)
{
// Checks if ToolTip is not already set on the element
if (value && element.ToolTip == null && content != null)
{
RibbonToolTip ribbonToolTip = visualChild.ToolTip as RibbonToolTip;
if (ribbonToolTip == null ||
ribbonToolTip.Title != content.ToString())
{
ribbonToolTip = new RibbonToolTip
{
Title = content.ToString()
};
visualChild.ToolTip = ribbonToolTip;
}
}
else
{
visualChild.ToolTip = null;
}
}
}
internal static void FindAndHookPopup(DependencyObject element, ref Popup popup)
{
// Fetch the Popup parent
if (popup == null)
{
DependencyObject rootVisual = TreeHelper.FindVisualRoot(element);
if (rootVisual != null)
{
popup = LogicalTreeHelper.GetParent(rootVisual) as Popup;
if (popup != null)
{
popup.Opened += OnPopupOpenedOrClosed;
popup.Closed += OnPopupOpenedOrClosed;
popup.PopupAnimation = PopupAnimation.None;
}
}
}
}
private static void OnPopupOpenedOrClosed(object sender, EventArgs e)
{
RibbonHelper.UpdatePopupAnimation((Popup)sender);
}
private static void UpdatePopupAnimation(Popup popup)
{
#if RIBBON_IN_FRAMEWORK
if (SystemParameters.HighContrast || !popup.IsOpen)
#else
if (Microsoft.Windows.Shell.SystemParameters2.Current.HighContrast || !popup.IsOpen)
#endif
{
popup.PopupAnimation = PopupAnimation.None;
}
else
{
popup.PopupAnimation = PopupAnimation.Fade;
}
}
#endregion ToolTip
#region Workaround for hetrogenous triad of ItemsControl
public class ValueAndValueSource
{
public object Value { get; set; }
public BaseValueSource ValueSource { get; set; }
}
public static ValueAndValueSource GetValueAndValueSource(DependencyObject d, DependencyProperty property)
{
if (d == null)
{
return null;
}
Debug.Assert(property != null);
return new ValueAndValueSource() { Value = d.GetValue(property), ValueSource = DependencyPropertyHelper.GetValueSource(d, property).BaseValueSource };
}
private static void RestoreValue(DependencyObject d, DependencyProperty property, ValueAndValueSource v)
{
Debug.Assert(d != null);
Debug.Assert(property != null);
Debug.Assert(v != null);
if (v.ValueSource == BaseValueSource.Local)
{
d.SetValue(property, v.Value);
}
else
{
d.ClearValue(property);
d.CoerceValue(property);
}
}
/// <summary>
/// This is an helper function to overcome the restriction on WPF framework to have
/// hetrogenious hierarchy ItemsControl. The problem is an ItemsControl in it's
/// internal logic fetches Item related properties from it's parent if that is an ItemsControl.
/// That approach can only be successful only in homogenous hierarchy of controls from first layer
/// as in case of TreeViewItem.
///
/// In Ribbon control it is not true at various places and most common example could RibbonGalleryCategory.
/// RibbonGalleryCategory's parent is RibbonGallery which is an ItemsControl and hence the above explained
/// fetch happens. But the property specified on RibbonGallery are applicable to RibbonGalleryCategory only.
/// Now, RibbonGalleryCategory's Item are not RibbonGalleryCategory but RibbonGalleryItem and hence a conflict.
///
/// This function gets called from PrepareContainerOverride after the base.PrepareContainerOverride is being called
/// and if the properties being fetched from the parent (changed) it sets them back to original values passed as arguments.
/// </summary>
/// <param name="itemsControl"> current ItemsControl</param>
/// <param name="itemTemplate"></param>
/// <param name="itemTemplateSelector"></param>
/// <param name="itemStringFormat"></param>
/// <param name="itemContainerStyle"></param>
/// <param name="itemContainerStyleSelector"></param>
/// <param name="alternationCount"></param>
/// <param name="itemBindingGroup"></param>
public static void IgnoreDPInheritedFromParentItemsControl(
ItemsControl itemsControl,
ItemsControl parentItemsControl,
ValueAndValueSource itemTemplate,
ValueAndValueSource itemTemplateSelector,
ValueAndValueSource itemStringFormat,
ValueAndValueSource itemContainerStyle,
ValueAndValueSource itemContainerStyleSelector,
ValueAndValueSource alternationCount,
ValueAndValueSource itemBindingGroup,
ValueAndValueSource headerTemplate,
ValueAndValueSource headerTemplateSelector,
ValueAndValueSource headerStringFormat)
{
// HeaderedItemsControl needs special consideration as some of the properties needs to be kept as is
// if HierarichalDataTemplate is used. The reason is internal method PrepareHeirarchy() in HeaderedItemsControl
// fetches appropriate values from the template and updates the one assigned using parentItemsControl and
// those values should be kept for master-detail scenarios to be working correctly.
HeaderedItemsControl hic = itemsControl as HeaderedItemsControl;
HierarchicalDataTemplate hTemplate = hic != null ? hic.HeaderTemplate as HierarchicalDataTemplate : null;
if (itemsControl.ItemTemplate == parentItemsControl.ItemTemplate)
{
if (hTemplate == null || (hTemplate.ItemTemplate == null && string.IsNullOrEmpty(hTemplate.ItemStringFormat) && hTemplate.ItemTemplateSelector == null))
{
RestoreValue(itemsControl, ItemsControl.ItemTemplateProperty, itemTemplate);
}
}
if (itemsControl.ItemTemplateSelector == parentItemsControl.ItemTemplateSelector)
{
if (hTemplate == null || (string.IsNullOrEmpty(hTemplate.ItemStringFormat) && hTemplate.ItemTemplateSelector == null))
{
RestoreValue(itemsControl, ItemsControl.ItemTemplateSelectorProperty, itemTemplateSelector);
}
}
if (itemsControl.ItemStringFormat == parentItemsControl.ItemStringFormat)
{
if (hTemplate == null || string.IsNullOrEmpty(hTemplate.ItemStringFormat))
{
RestoreValue(itemsControl, ItemsControl.ItemStringFormatProperty, itemStringFormat);
}
}
if (itemsControl.ItemContainerStyle == parentItemsControl.ItemContainerStyle)
{
if (hTemplate == null || (hTemplate.ItemContainerStyleSelector == null && hTemplate.ItemContainerStyle == null))
{
RestoreValue(itemsControl, ItemsControl.ItemContainerStyleProperty, itemContainerStyle);
}
}
if (itemsControl.ItemContainerStyleSelector == parentItemsControl.ItemContainerStyleSelector)
{
if (hTemplate == null || hTemplate.ItemContainerStyleSelector == null)
{
RestoreValue(itemsControl, ItemsControl.ItemContainerStyleSelectorProperty, itemContainerStyleSelector);
}
}
if (itemsControl.AlternationCount == parentItemsControl.AlternationCount)
{
// Potential issue if 0 is set intentionally, but one can argue if it is explicitly defined on
// ItemsControl itself then use that one.
if (hTemplate == null || hTemplate.AlternationCount == 0)
{
RestoreValue(itemsControl, ItemsControl.AlternationCountProperty, alternationCount);
}
}
if (itemsControl.ItemBindingGroup == parentItemsControl.ItemBindingGroup)
{
if (hTemplate == null || hTemplate.ItemBindingGroup == null)
{
RestoreValue(itemsControl, ItemsControl.ItemBindingGroupProperty, itemBindingGroup);
}
}
if (hic != null)
{
if (headerTemplate != null && hic.HeaderTemplate == parentItemsControl.ItemTemplate && hTemplate == null)
{
RestoreValue(hic, HeaderedItemsControl.HeaderTemplateProperty, headerTemplate);
}
if (headerTemplateSelector != null && hic.HeaderTemplateSelector == parentItemsControl.ItemTemplateSelector && hTemplate == null)
{
RestoreValue(hic, HeaderedItemsControl.HeaderTemplateSelectorProperty, headerTemplateSelector);
}
if (headerStringFormat != null && hic.HeaderStringFormat == parentItemsControl.ItemStringFormat && hTemplate == null)
{
RestoreValue(hic, HeaderedItemsControl.HeaderStringFormatProperty, headerStringFormat);
}
}
}
#endregion
#region Workaround of IsOffScreen Issue in UI Automation in 3.5
#if !RIBBON_IN_FRAMEWORK
/// <summary>
/// Calculates the real visible rectangle within parent chain, borrowed from 4.0 IsOffScreen fix for UIA.
/// </summary>
/// <param name="element"></param>
/// <returns></returns>
public static Rect CalculateVisibleBoundingRect(UIElement element)
{
Rect boundingRect = Rect.Empty;
boundingRect = new Rect(element.RenderSize);
// Compute visible portion of the rectangle.
Visual visual = VisualTreeHelper.GetParent(element) as Visual;
while (visual != null && boundingRect != Rect.Empty && boundingRect.Height != 0 && boundingRect.Width != 0)
{
Geometry clipGeometry = VisualTreeHelper.GetClip(visual);
if (clipGeometry != null)
{
GeneralTransform transform = element.TransformToAncestor(visual).Inverse;
// Safer version of transform to descendent (doing the inverse ourself and saves us changing the co-ordinate space of the owner's bounding rectangle),
// we want the rect inside of our space. (Which is always rectangular and much nicer to work with)
if (transform != null)
{
Rect clipBounds = clipGeometry.Bounds;
clipBounds = transform.TransformBounds(clipBounds);
boundingRect.Intersect(clipBounds);
}
else
{
// No visibility if non-invertable transform exists.
boundingRect = Rect.Empty;
}
}
visual = VisualTreeHelper.GetParent(visual) as Visual;
}
return boundingRect;
}
#endif
#endregion
#region Mneumonics
// Returns the index of _ marker.
// _ can be escaped by double _
public static int FindAccessKeyMarker(string text)
{
int length = text.Length;
int startIndex = 0;
while (startIndex < length)
{
int index = text.IndexOf('_', startIndex);
if (index == -1)
return -1;
// If next char exist and different from _
if (index + 1 < length && text[index + 1] != '_')
return index;
startIndex = index + 2;
}
return -1;
}
#endregion
#region PseudoInheritedProperties
internal static void TransferPseudoInheritedProperties(DependencyObject parent, DependencyObject child)
{
if (RibbonControlService.GetIsInQuickAccessToolBar(parent))
{
RibbonControlService.SetIsInQuickAccessToolBar(child, true);
RibbonControlSizeDefinition qatControlSizeDefinition = RibbonControlService.GetQuickAccessToolBarControlSizeDefinition(child);
RibbonControlService.SetControlSizeDefinition(child, qatControlSizeDefinition);
}
else
{
if (RibbonControlService.GetIsInControlGroup(parent))
{
RibbonControlService.SetIsInControlGroup(child, true);
}
RibbonControlSizeDefinition controlSizeDefinition = RibbonControlService.GetControlSizeDefinition(parent);
if (controlSizeDefinition != null)
{
RibbonControlService.SetControlSizeDefinition(child, controlSizeDefinition);
}
}
}
internal static void ClearPseudoInheritedProperties(DependencyObject child)
{
if (child != null)
{
child.ClearValue(RibbonControlService.IsInQuickAccessToolBarPropertyKey);
child.ClearValue(RibbonControlService.IsInControlGroupPropertyKey);
child.ClearValue(RibbonControlService.ControlSizeDefinitionProperty);
}
}
#endregion PseudoInheritedProperties
#region KeyboardNavigation
internal static void EnableFocusVisual(DependencyObject d)
{
if (IsKeyboardMostRecentInputDevice())
{
RibbonControlService.SetShowKeyboardCues(d, true);
}
}
internal static void DisableFocusVisual(DependencyObject d)
{
RibbonControlService.SetShowKeyboardCues(d, false);
}
internal static bool IsKeyboardMostRecentInputDevice()
{
return (InputManager.Current.MostRecentInputDevice is KeyboardDevice);
}
public static bool MoveFocus(FocusNavigationDirection direction)
{
UIElement uie = Keyboard.FocusedElement as UIElement;
if (uie != null)
{
return uie.MoveFocus(new TraversalRequest(direction));
}
ContentElement ce = Keyboard.FocusedElement as ContentElement;
if (ce != null)
{
return ce.MoveFocus(new TraversalRequest(direction));
}
return false;
}
public static DependencyObject PredictFocus(DependencyObject element, FocusNavigationDirection direction)
{
UIElement uie;
ContentElement ce;
UIElement3D uie3d;
if ((uie = element as UIElement) != null)
{
return uie.PredictFocus(direction);
}
else if ((ce = element as ContentElement) != null)
{
return ce.PredictFocus(direction);
}
else if ((uie3d = element as UIElement3D) != null)
{
return uie3d.PredictFocus(direction);
}
return null;
}
public static bool Focus(DependencyObject element)
{
UIElement uie;
ContentElement ce;
UIElement3D uie3d;
if ((uie = element as UIElement) != null)
{
return uie.Focus();
}
else if ((ce = element as ContentElement) != null)
{
return ce.Focus();
}
else if ((uie3d = element as UIElement3D) != null)
{
return uie3d.Focus();
}
return false;
}
#endregion KeyboardNavigation
#region ItemsControl Navigation
internal static bool NavigateToFirstItem(ItemsControl itemsControl, Action<int> bringIntoViewCallback, Func<FrameworkElement, bool> additionalCheck)
{
FrameworkElement firstItem = FindContainer(itemsControl, 0, 1, bringIntoViewCallback, additionalCheck);
if (firstItem != null)
{
firstItem.Focus();
return true;
}
return false;
}
internal static bool NavigateToLastItem(ItemsControl itemsControl, Action<int> bringIntoViewCallback, Func<FrameworkElement, bool> additionalCheck)
{
FrameworkElement lastItem = FindContainer(itemsControl, itemsControl.Items.Count - 1, -1, bringIntoViewCallback, additionalCheck);
if (lastItem != null)
{
lastItem.Focus();
return true;
}
return false;
}
internal static FrameworkElement FindContainer(ItemsControl itemsControl, int startIndex, int direction, Action<int> bringIntoViewCallback, Func<FrameworkElement, bool> additionalCheck)
{
if (itemsControl.HasItems)
{
int count = itemsControl.Items.Count;
for (; startIndex >= 0 && startIndex < count; startIndex += direction)
{
FrameworkElement container = itemsControl.ItemContainerGenerator.ContainerFromIndex(startIndex) as FrameworkElement;
// If container is virtualized, call BringIntoView.
if (container == null && bringIntoViewCallback != null)
{
bringIntoViewCallback(startIndex);
container = itemsControl.ItemContainerGenerator.ContainerFromIndex(startIndex) as FrameworkElement;
}
if (container != null && (additionalCheck == null || additionalCheck(container)))
{
return container;
}
}
}
return null;
}
internal static bool NavigateToItem(ItemsControl parent, int itemIndex, Action<int> bringIntoViewCallback)
{
if (itemIndex < parent.Items.Count)
{
FrameworkElement nextElement = RibbonHelper.FindContainer(parent, itemIndex, 1, bringIntoViewCallback, null);
if (nextElement != null)
{
nextElement.Focus();
return true;
}
}
return false;
}
internal static bool NavigateToNextMenuItemOrGallery(ItemsControl parent, int startIndex, Action<int> bringIntoViewCallback)
{
if (startIndex == parent.Items.Count - 1)
startIndex = -1;
if (startIndex < parent.Items.Count - 1)
{
FrameworkElement nextElement = RibbonHelper.FindContainer(parent, startIndex + 1, 1, bringIntoViewCallback, IsMenuItemFocusable);
if (nextElement != null)
{
RibbonGallery gallery = nextElement as RibbonGallery;
if (gallery == null)
{
nextElement.Focus();
return true;
}
else
{
// Move focus into the gallery if it does not have already has focus.
if (!nextElement.IsKeyboardFocusWithin)
{
return NavigateDownToGallery(gallery);
}
}
}
}
return false;
}
internal static bool NavigateToPreviousMenuItemOrGallery(ItemsControl parent, int startIndex, Action<int> bringIntoViewCallback)
{
if (startIndex <= 0)
startIndex = parent.Items.Count;
if (startIndex > 0)
{
FrameworkElement previousElement = RibbonHelper.FindContainer(parent, startIndex - 1, -1, bringIntoViewCallback, IsMenuItemFocusable);
if (previousElement != null)
{
RibbonGallery gallery = previousElement as RibbonGallery;
if (gallery == null)
{
previousElement.Focus();
return true;
}
else
{
return NavigateUpToGallery(gallery);
}
}
}
return false;
}
private static bool NavigateUpToGallery(RibbonGallery gallery)
{
if (gallery != null)
{
RibbonGalleryCategory lastCategory = RibbonHelper.FindContainer(gallery, gallery.Items.Count - 1, -1, null, IsContainerVisible) as RibbonGalleryCategory;
if (lastCategory != null)
{
return RibbonHelper.NavigateToLastItem(lastCategory, /* BringIntoView callback */ null, IsContainerFocusable);
}
}
return false;
}
private static bool NavigateDownToGallery(RibbonGallery gallery)
{
if (gallery != null)
{
if (gallery.CanUserFilter)
{
// Move focus to FilterContentPane of FilterMenuButton accordingly.
FrameworkElement focusObject = null;
ContentPresenter filterContentPane = gallery.FilterContentPane;
if (filterContentPane != null &&
filterContentPane.IsVisible)
{
focusObject = filterContentPane;
}
if (focusObject == null)
{
RibbonFilterMenuButton filterButton = gallery.FilterMenuButton;
if (filterButton != null &&
filterButton.IsVisible)
{
focusObject = filterButton;
}
}
if (focusObject != null)
{
focusObject.MoveFocus(new TraversalRequest(FocusNavigationDirection.First));
return true;
}
}
RibbonGalleryCategory firstCategory = RibbonHelper.FindContainer(gallery, 0, 1, null, IsContainerVisible) as RibbonGalleryCategory;
if (firstCategory != null)
{
return RibbonHelper.NavigateToFirstItem(firstCategory, /* BringIntoView callback */ null, IsContainerFocusable);
}
}
return false;
}
private static bool IsContainerFocusable(FrameworkElement container)
{
return container != null && container.Focusable;
}
private static bool IsContainerVisible(FrameworkElement container)
{
return container != null && container.Visibility == Visibility.Visible;
}
private static bool IsMenuItemFocusable(FrameworkElement container)
{
return container != null && (container is RibbonGallery || container.Focusable);
}
// This method is called when trying to navigate a single step up, down, left or
// right within a RibbonGallery.
internal static bool NavigateAndHighlightGalleryItem(RibbonGalleryItem focusedElement, FocusNavigationDirection direction)
{
if (focusedElement != null)
{
RibbonGalleryItem predictedFocus = focusedElement.PredictFocus(direction) as RibbonGalleryItem;
if (predictedFocus != null)
{
predictedFocus.IsHighlighted = true;
return true;
}
}
return false;
}
internal static bool NavigatePageAndHighlightRibbonGalleryItem(RibbonGallery gallery, RibbonGalleryItem galleryItem, FocusNavigationDirection direction)
{
RibbonGalleryItem highlightedGalleryItem;
return NavigatePageAndHighlightRibbonGalleryItem(gallery, galleryItem, direction, out highlightedGalleryItem);
}
// This method is called when trying to navigate pages within a RibbonGallery.
// We approximate the RibbonGalleryItem that is a page away from the currently
// focused item based upon the precomputed MaxColumnWidth and MaxRowHeight values.
internal static bool NavigatePageAndHighlightRibbonGalleryItem(
RibbonGallery gallery,
RibbonGalleryItem galleryItem,
FocusNavigationDirection direction,
out RibbonGalleryItem highlightedGalleryItem)
{
highlightedGalleryItem = null;
RibbonGalleryCategoriesPanel categoriesPanel = gallery.ItemsHostSite as RibbonGalleryCategoriesPanel;
if (categoriesPanel != null)
{
double viewportWidth = categoriesPanel.ViewportWidth;
double viewportHeight = categoriesPanel.ViewportHeight;
RibbonGalleryCategory category, prevCategory = null;
if (galleryItem != null)
{
category = galleryItem.RibbonGalleryCategory;
}
else
{
category = gallery.Items.Count > 0 ? gallery.ItemContainerGenerator.ContainerFromIndex(0) as RibbonGalleryCategory : null;
galleryItem = category != null && category.Items.Count > 0 ? category.ItemContainerGenerator.ContainerFromIndex(0) as RibbonGalleryItem : null;
}
if (category != null)
{
Debug.Assert(category.RibbonGallery == gallery, "The reference RibbongalleryItem and the RibbonGallery must be related.");
int startCatIndex = gallery.ItemContainerGenerator.IndexFromContainer(category);
int endCatIndex, incr;
if (direction == FocusNavigationDirection.Up)
{
endCatIndex = -1;
incr = -1;
}
else
{
endCatIndex = gallery.Items.Count;
incr = 1;
}
for (int catIndex = startCatIndex; catIndex != endCatIndex && highlightedGalleryItem == null; catIndex += incr)
{
category = gallery.ItemContainerGenerator.ContainerFromIndex(catIndex) as RibbonGalleryCategory;
RibbonGalleryItemsPanel galleryItemsPanel = category.ItemsHostSite as RibbonGalleryItemsPanel;
// We want to skip over filtered categories
if (category.Visibility != Visibility.Visible)
{
continue;
}
int startItemIndex, endItemIndex, startColumnIndex, endColumnIndex, columnCount;
columnCount = (int)(viewportWidth / galleryItemsPanel.MaxColumnWidth);
if (direction == FocusNavigationDirection.Up)
{
startItemIndex = galleryItem != null ? category.ItemContainerGenerator.IndexFromContainer(galleryItem) : category.Items.Count - 1;
endItemIndex = -1;
if (prevCategory != null)
{
viewportHeight -= prevCategory.HeaderPresenter.ActualHeight;
if (DoubleUtil.LessThanOrClose(viewportHeight, 0))
{
highlightedGalleryItem = category.ItemContainerGenerator.ContainerFromIndex(startItemIndex) as RibbonGalleryItem;
break;
}
}
// startColumnIndex is the last column in the last row or the column of the anchor item
if (columnCount == 1)
{
startColumnIndex = 0;
endColumnIndex = 0;
}
else
{
startColumnIndex = (galleryItem != null ? startItemIndex : category.Items.Count - 1) % columnCount;
endColumnIndex = 0;
}
}
else
{
startItemIndex = galleryItem != null ? category.ItemContainerGenerator.IndexFromContainer(galleryItem) : 0;
endItemIndex = category.Items.Count;
if (prevCategory != null)
{
viewportHeight -= category.HeaderPresenter.ActualHeight;
if (DoubleUtil.LessThanOrClose(viewportHeight, 0))
{
highlightedGalleryItem = category.ItemContainerGenerator.ContainerFromIndex(startItemIndex) as RibbonGalleryItem;
break;
}
}
// endColumnIndex is the last column in the first row
if (columnCount == 1)
{
startColumnIndex = 0;
endColumnIndex = 0;
}
else
{
int remainingItems = category.Items.Count;
bool isLastRow = remainingItems <= columnCount;
startColumnIndex = galleryItem != null ? (startItemIndex % columnCount) : 0;
endColumnIndex = isLastRow ? remainingItems - 1 : columnCount - 1;
}
}
galleryItem = null;
for (int itemIndex = startItemIndex, columnIndex = startColumnIndex; itemIndex != endItemIndex; itemIndex += incr)
{
if (columnIndex == endColumnIndex)
{
// We are at the end of a row
viewportHeight -= galleryItemsPanel.MaxRowHeight;
if (DoubleUtil.LessThanOrClose(viewportHeight, 0) ||
(itemIndex == endItemIndex - incr && catIndex == endCatIndex - incr))
{
// If we have scrolled a page or have reached the boundary
// of the gallery, highlight that item
highlightedGalleryItem = category.ItemContainerGenerator.ContainerFromIndex(itemIndex) as RibbonGalleryItem;
break;
}
if (direction == FocusNavigationDirection.Up)
{
if (columnCount > 1)
{
startColumnIndex = columnCount - 1;
endColumnIndex = 0;
}
}
else
{
if (columnCount > 1)
{
int remainingItems = category.Items.Count - itemIndex;
bool isLastRow = remainingItems <= columnCount;
startColumnIndex = 0;
endColumnIndex = isLastRow ? remainingItems - 1 : columnCount - 1;
}
}
columnIndex = startColumnIndex;
}
else
{
// We are interating through the cells in a row
columnIndex += incr;
}
}
prevCategory = category;
}
if (highlightedGalleryItem != null)
{
highlightedGalleryItem.IsHighlighted = true;
return true;
}
}
}
return false;
}
#endregion
#region DismissPopup
internal static void AddHandler(DependencyObject element, RoutedEvent routedEvent, Delegate handler)
{
Debug.Assert(element != null, "Element must not be null");
Debug.Assert(routedEvent != null, "RoutedEvent must not be null");
UIElement uiElement = element as UIElement;
if (uiElement != null)
{
uiElement.AddHandler(routedEvent, handler);
}
else
{
ContentElement contentElement = element as ContentElement;
if (contentElement != null)
{
contentElement.AddHandler(routedEvent, handler);
}
else
{
UIElement3D uiElement3D = element as UIElement3D;
if (uiElement3D != null)
{
uiElement3D.AddHandler(routedEvent, handler);
}
}
}
}
internal static void RemoveHandler(DependencyObject element, RoutedEvent routedEvent, Delegate handler)
{
Debug.Assert(element != null, "Element must not be null");
Debug.Assert(routedEvent != null, "RoutedEvent must not be null");
UIElement uiElement = element as UIElement;
if (uiElement != null)
{
uiElement.RemoveHandler(routedEvent, handler);
}
else
{
ContentElement contentElement = element as ContentElement;
if (contentElement != null)
{
contentElement.RemoveHandler(routedEvent, handler);
}
else
{
UIElement3D uiElement3D = element as UIElement3D;
if (uiElement3D != null)
{
uiElement3D.RemoveHandler(routedEvent, handler);
}
}
}
}
/// <summary>
/// Helper method which determines if given hwnd belongs
/// to the same Dispatcher as that of given element.
/// </summary>
public static bool IsOurWindow(IntPtr hwnd, DependencyObject element)
{
Debug.Assert(element != null);
if (hwnd != IntPtr.Zero)
{
HwndSource hwndSource;
hwndSource = HwndSource.FromHwnd(hwnd);
if (hwndSource != null &&
hwndSource.Dispatcher == element.Dispatcher)
{
// The window has the same dispatcher, must be ours.
return true;
}
}
return false;
}
public static void HandleLostMouseCapture(UIElement element,
MouseEventArgs e,
Func<bool> getter,
Action<bool> setter,
UIElement targetCapture,
UIElement targetFocus)
{
if (getter() && targetCapture != null)
{
IntPtr capturedHwnd = IntPtr.Zero;
bool isOurWindowCaptured = false;
if (Mouse.Captured == null)
{
// If we are losing capture to some other window
// then close all the popups.
capturedHwnd = NativeMethods.GetCapture();
if (capturedHwnd != IntPtr.Zero &&
!(isOurWindowCaptured = IsOurWindow(capturedHwnd, element)))
{
element.RaiseEvent(new RibbonDismissPopupEventArgs());
e.Handled = true;
return;
}
}
if (e.OriginalSource == targetCapture)
{
if (Mouse.Captured == null)
{
PresentationSource mouseSource = Mouse.PrimaryDevice.ActiveSource;
if (mouseSource == null &&
(capturedHwnd == IntPtr.Zero ||
isOurWindowCaptured))
{
// If the active source is null and current captured
// is null, this capture loss is bacause of Mouse
// deactivation (because mouse is not on the window
// anymore). Hence reacquire capture and focus.
// Note that we do it only if the capture is not lost to
// some other window, and genuine closing of
// popups when both active source and current captured is
// null due to clicking some where else should be handled by
// click through event handler.
if (!ReacquireCapture(targetCapture, targetFocus))
{
// call the setter if we couldn't reacquire capture
setter(false);
}
e.Handled = true;
}
else
{
setter(false);
}
}
else if (!RibbonHelper.IsAncestorOf(targetCapture, Mouse.Captured as DependencyObject))
{
setter(false);
}
}
else if (RibbonHelper.IsAncestorOf(targetCapture, e.OriginalSource as DependencyObject))
{
if (Mouse.Captured == null)
{
// If a descendant of targetCapture is losing capture
// then take capture on targetCapture
if (!ReacquireCapture(targetCapture, targetFocus))
{
// call the setter if we couldn't reacquire capture
setter(false);
}
e.Handled = true;
}
else if (!IsCaptureInSubtree(targetCapture))
{
// If a descendant of targetCapture is losing capture
// to an element outside targetCapture's subtree
// then call setter
setter(false);
}
}
}
}
private static bool ReacquireCapture(UIElement targetCapture, UIElement targetFocus)
{
bool success = Mouse.Capture(targetCapture, CaptureMode.SubTree);
if (success && targetFocus != null && !targetFocus.IsKeyboardFocusWithin)
{
targetFocus.Focus();
}
return success;
}
public static bool IsMousePhysicallyOver(UIElement element)
{
if (element == null)
{
return false;
}
Point position = Mouse.GetPosition(element);
if (DoubleUtil.GreaterThan(position.X, 0) &&
DoubleUtil.GreaterThan(position.Y, 0) &&
DoubleUtil.LessThanOrClose(position.X, element.RenderSize.Width) &&
DoubleUtil.LessThanOrClose(position.Y, element.RenderSize.Height))
{
return true;
}
return false;
}
internal static void HandleClickThrough(
object sender,
MouseButtonEventArgs e,
UIElement alternateCaptureHost)
{
if (e.ChangedButton == MouseButton.Left || e.ChangedButton == MouseButton.Right)
{
// If this is the element with the leaf popup,
// then raise DismissPopupEvent. If capture belongs
// to the visual subtree of popup or the element,
// then we decide that this is the leaf popup.
UIElement element = sender as UIElement;
if (IsCaptureInVisualSubtree(element) ||
IsCaptureInVisualSubtree(alternateCaptureHost))
{
UIElement source = Mouse.Captured as UIElement;
if (source == null)
{
source = element;
}
if (source != null)
{
source.RaiseEvent(new RibbonDismissPopupEventArgs(RibbonDismissPopupMode.MousePhysicallyNotOver));
}
}
}
}
internal static void HandleDismissPopup(
RibbonDismissPopupEventArgs e,
Action<bool> setter,
Predicate<DependencyObject> cancelPredicate,
UIElement mouseOverTarget,
UIElement alternateMouseOverTarget)
{
if (!cancelPredicate(e.OriginalSource as DependencyObject))
{
// Call setter if the dismiss mode is always or
// if the mouse is not directly over either of
// the targets.
if (e.DismissMode == RibbonDismissPopupMode.Always ||
(!IsMousePhysicallyOver(mouseOverTarget) &&
!IsMousePhysicallyOver(alternateMouseOverTarget)))
{
setter(false);
}
else
{
e.Handled = true;
}
}
else
{
e.Handled = true;
}
}
public static void AsyncSetFocusAndCapture(UIElement element,
Func<bool> getter,
UIElement targetCapture,
UIElement targetFocus)
{
element.Dispatcher.BeginInvoke(
(Action)delegate()
{
if (getter())
{
if (targetFocus != null && !targetFocus.IsKeyboardFocusWithin)
{
targetFocus.Focus();
}
if (targetCapture != null &&
(Mouse.Captured == null ||
!IsCaptureInSubtree(targetCapture)))
{
Mouse.Capture(targetCapture, CaptureMode.SubTree);
}
}
},
DispatcherPriority.Input);
}
public static void RestoreFocusAndCapture(UIElement targetCapture,
UIElement targetFocus)
{
if (targetFocus != null &&
targetFocus.IsKeyboardFocusWithin)
{
Keyboard.Focus(null);
}
if (targetCapture != null &&
Mouse.Captured != null &&
IsCaptureInSubtree(targetCapture))
{
Mouse.Capture(null);
}
}
public static void HandleIsDropDownChanged(UIElement element,
Func<bool> getter,
UIElement targetCapture,
UIElement targetFocus)
{
if (targetCapture == null &&
targetFocus == null)
{
return;
}
if (getter())
{
AsyncSetFocusAndCapture(element,
getter,
targetCapture,
targetFocus);
}
else
{
RestoreFocusAndCapture(targetCapture,
targetFocus);
}
}
internal static void HandleDropDownKeyDown(
object sender, KeyEventArgs e, Func<bool> gettor, Action<bool> settor, UIElement targetFocusOnFalse, UIElement targetFocusContainerOnTrue)
{
Key key = e.Key;
switch (key)
{
case Key.Escape:
{
if (gettor())
{
settor(false);
e.Handled = true;
if (targetFocusOnFalse != null)
{
targetFocusOnFalse.Focus();
}
}
}
break;
case Key.System:
if (KeyTipService.Current.State != KeyTipService.KeyTipState.Enabled && ((e.SystemKey == Key.LeftAlt) || (e.SystemKey == Key.RightAlt))
|| (e.SystemKey == Key.F10))
{
if (gettor())
{
// Raise DismissPopup event and hence the key down event.
UIElement uie = sender as UIElement;
if (uie != null)
{
RibbonDismissPopupEventArgs dismissArgs = new RibbonDismissPopupEventArgs();
uie.RaiseEvent(dismissArgs);
e.Handled = true;
}
}
}
break;
case Key.F4:
{
if (gettor())
{
settor(false);
e.Handled = true;
if (targetFocusOnFalse != null)
{
targetFocusOnFalse.Focus();
}
}
else
{
settor(true);
if (targetFocusContainerOnTrue != null)
{
targetFocusContainerOnTrue.Dispatcher.BeginInvoke(
(Action)delegate()
{
targetFocusContainerOnTrue.MoveFocus(new TraversalRequest(FocusNavigationDirection.First));
},
DispatcherPriority.Input,
null);
}
e.Handled = true;
}
// Technically one needs to change active key tip scope,
// but since we do not have public api to do that yet,
// we dismiss keytips
KeyTipService.DismissKeyTips();
}
break;
}
}
public static UIElement TryGetChild(this Popup popup)
{
return (popup == null ? null : popup.Child);
}
public static bool IsCaptureInSubtree(UIElement element)
{
return (Mouse.Captured == element ||
RibbonHelper.IsAncestorOf(element, Mouse.Captured as DependencyObject));
}
public static bool IsCaptureInVisualSubtree(UIElement element)
{
return (element != null && Mouse.Captured != null && (Mouse.Captured == element ||
TreeHelper.IsVisualAncestorOf(element, Mouse.Captured as DependencyObject)));
}
internal static bool IsAncestorOf(DependencyObject ancestor, DependencyObject element)
{
if (ancestor == null || element == null)
{
return false;
}
return TreeHelper.FindAncestor(element, delegate(DependencyObject d) { return d == ancestor; }) != null;
}
/// <summary>
/// Helper method to coerce the given property
/// of the element at input priority.
/// </summary>
public static void DelayCoerceProperty(DependencyObject element, DependencyProperty property)
{
element.Dispatcher.BeginInvoke(
(Action)delegate()
{
element.CoerceValue(property);
},
DispatcherPriority.Input,
null);
}
#endregion DismissPopup
#region StarLayoutHelper
public static bool IsISupportStarLayout(DependencyObject d)
{
return (d is ISupportStarLayout);
}
// Registers itself to StarLayout manager in the parent chain.
public static void InitializeStarLayoutManager(DependencyObject starLayoutProvider)
{
Debug.Assert(starLayoutProvider != null);
ISupportStarLayout starLayoutManager = TreeHelper.FindVisualAncestor(starLayoutProvider,
RibbonHelper.IsISupportStarLayout) as ISupportStarLayout;
IContainsStarLayoutManager iContainsStarLayoutManager = starLayoutProvider as IContainsStarLayoutManager;
if (iContainsStarLayoutManager != null)
{
IProvideStarLayoutInfoBase iProvideStarLayoutInfoBase = (IProvideStarLayoutInfoBase)starLayoutProvider;
if (starLayoutManager == null && iContainsStarLayoutManager.StarLayoutManager != null)
{
iContainsStarLayoutManager.StarLayoutManager.UnregisterStarLayoutProvider(iProvideStarLayoutInfoBase);
iContainsStarLayoutManager.StarLayoutManager = null;
}
else if (starLayoutManager != null)
{
if (starLayoutManager != iContainsStarLayoutManager.StarLayoutManager)
{
if (iContainsStarLayoutManager.StarLayoutManager != null)
{
iContainsStarLayoutManager.StarLayoutManager.UnregisterStarLayoutProvider(iProvideStarLayoutInfoBase);
}
starLayoutManager.RegisterStarLayoutProvider(iProvideStarLayoutInfoBase);
iContainsStarLayoutManager.StarLayoutManager = starLayoutManager;
}
// It isn't appropriate for the star layout element to be
// measure outside the context of the parent star layout
// manager. Layout for suh an element will only be correct
// if we do the two pass star layout drill. So just in case
// we got here out of turn, we should invalidate measure
// on the manager so that we can settle things. In normal
// course of action if we arrived here from the manager,
// the InvalidateMeasure would no-o because the manager
// will already be dirty for measure.
UIElement managerElement = starLayoutManager as UIElement;
if (managerElement != null)
{
managerElement.InvalidateMeasure();
}
}
}
}
#endregion StarLayoutHelper
#region RibbonContextMenu
internal static object OnCoerceContextMenu(DependencyObject d, object baseValue)
{
DependencyProperty dp = ContextMenuService.ContextMenuProperty;
if (PropertyHelper.IsPropertyTransferEnabled(d, dp))
{
var propertySource = DependencyPropertyHelper.GetValueSource(d, dp);
var baseValueSource = propertySource.BaseValueSource;
if (baseValueSource == BaseValueSource.Default)
{
RibbonContextMenu cm = RibbonContextMenu.ChooseContextMenu(d);
if (cm != null)
{
return cm;
}
}
}
return baseValue;
}
internal static void OnContextMenuChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
PropertyHelper.TransferProperty(d, ContextMenuService.ContextMenuProperty);
}
#endregion
#region OnCommandChanged
// Performs proper hooking/unhooking and coerces QAT ID.
//
// All Ribbon controls should hook this delegate to coerce the QAT ID whenever the Command changes.
internal static void OnCommandChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
d.CoerceValue(RibbonControlService.QuickAccessToolBarIdProperty);
}
#endregion OnCommandChanged
#region QAT
// For Ribbon controls that implement ICommandSource, coerce the QAT ID to be the Command property if QAT ID is unspecified.
internal static object OnCoerceQuickAccessToolBarId(DependencyObject d, object baseValue)
{
ICommandSource commandSource = d as ICommandSource;
if (baseValue == null &&
commandSource != null)
{
return commandSource.Command;
}
return baseValue;
}
internal static void OnIsInQATChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
PropertyHelper.TransferProperty(d, ContextMenuService.ContextMenuProperty);
}
internal static object OnCoerceCanAddToQuickAccessToolBarDirectly(DependencyObject d, object baseValue)
{
DependencyProperty dp = RibbonControlService.CanAddToQuickAccessToolBarDirectlyProperty;
if (PropertyHelper.IsPropertyTransferEnabled(d, dp))
{
var propertySource = DependencyPropertyHelper.GetValueSource(d, dp);
var baseValueSource = propertySource.BaseValueSource;
if (baseValueSource == BaseValueSource.Default)
{
FrameworkElement fe = d as FrameworkElement;
if (fe != null && fe.TemplatedParent != null && !(fe.TemplatedParent is ContentPresenter))
{
return false;
}
}
}
return baseValue;
}
internal static void OnCanAddToQuickAccessToolBarDirectlyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
PropertyHelper.TransferProperty(d, RibbonControlService.CanAddToQuickAccessToolBarDirectlyProperty);
PropertyHelper.TransferProperty(d, ContextMenuService.ContextMenuProperty);
}
/// <summary>
/// Determines whether or not an item exists in the QAT.
/// </summary>
internal static bool ExistsInQAT(DependencyObject element)
{
ArgumentNullException.ThrowIfNull(element);
Ribbon ribbon = (Ribbon)element.GetValue(RibbonControlService.RibbonProperty);
object qatID = RibbonControlService.GetQuickAccessToolBarId(element);
if (ribbon != null &&
ribbon.QuickAccessToolBar != null &&
qatID != null)
{
return ribbon.QuickAccessToolBar.ContainsId(qatID);
}
return false;
}
[Flags]
private enum TransferMode
{
AlwaysTransfer = 0x01, // Always performs the transfer except if it is the default value.
OnlyTransferIfTemplateBound = 0x02, // Requires BaseValueSource.ParentTemplate or higher to perform the transfer.
}
private static void TransferProperty(UIElement original, UIElement clone, DependencyProperty dp, TransferMode mode)
{
TransferProperty(original, clone, dp, dp, mode);
}
private static void TransferProperty(UIElement original, UIElement clone, DependencyProperty originalProperty, DependencyProperty cloneProperty, TransferMode mode)
{
bool performTransfer = false;
if ((mode & TransferMode.AlwaysTransfer) == TransferMode.AlwaysTransfer)
{
performTransfer |= DependencyPropertyHelper.GetValueSource(original, originalProperty).BaseValueSource > BaseValueSource.Default;
}
else if ((mode & TransferMode.OnlyTransferIfTemplateBound) == TransferMode.OnlyTransferIfTemplateBound)
{
performTransfer |= DependencyPropertyHelper.GetValueSource(original, originalProperty).BaseValueSource >= BaseValueSource.ParentTemplate;
}
if (performTransfer)
{
// Actually perform the transfer.
BindingBase binding = BindingOperations.GetBindingBase(original, originalProperty);
if (binding != null)
{
// Transfer Bindings
BindingOperations.SetBinding(clone, cloneProperty, binding);
}
else
{
Expression expr = original.ReadLocalValue(originalProperty) as Expression;
if (expr != null)
{
// Transfer DynamicResource
DynamicResourceExtension dynamicResource = _rreConverter.ConvertTo(expr, typeof(MarkupExtension)) as DynamicResourceExtension;
if (dynamicResource != null)
{
clone.SetValue(cloneProperty, dynamicResource.ProvideValue(null));
}
}
else
{
// Transfer other DPs
object originalValue = original.GetValue(originalProperty);
clone.SetValue(cloneProperty, CreateClone(originalValue));
}
}
}
}
private static ResourceReferenceExpressionConverter _rreConverter = new ResourceReferenceExpressionConverter();
private struct PropertyAndTransferMode
{
internal DependencyProperty Property;
internal TransferMode Mode;
}
private static PropertyAndTransferMode[] _automationProperties;
private static PropertyAndTransferMode[] _feProperties;
private static PropertyAndTransferMode[] _controlProperties;
private static PropertyAndTransferMode[] _contentControlProperties;
private static PropertyAndTransferMode[] _buttonProperties;
private static PropertyAndTransferMode[] _toggleButtonProperties;
private static PropertyAndTransferMode[] _itemsControlProperties;
private static PropertyAndTransferMode[] _headeredItemsControlProperties;
private static PropertyAndTransferMode[] _ribbonProperties;
private static PropertyAndTransferMode[] _ribbonBrushProperties;
private static PropertyAndTransferMode[] _ribbonMenuButtonProperties;
private static PropertyAndTransferMode[] _ribbonSplitButtonProperties;
private static PropertyAndTransferMode[] _ribbonMenuItemProperties;
private static PropertyAndTransferMode[] _ribbonSplitMenuItemProperties;
private static PropertyAndTransferMode[] _ribbonGalleryProperties;
private static PropertyAndTransferMode[] _ribbonGalleryCategoryProperties;
private static PropertyAndTransferMode[] _ribbonGalleryItemProperties;
private static PropertyAndTransferMode[] _ribbonGroupProperties;
private static PropertyAndTransferMode[] _textBoxProperties;
private static PropertyAndTransferMode[] _ribbonComboBoxProperties;
private static PropertyAndTransferMode[] _scrollProperties;
private static object _syncRoot = new object();
internal static void PopulatePropertyLists()
{
lock (_syncRoot)
{
if (_feProperties == null)
{
_automationProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode() { Property = AutomationProperties.AcceleratorKeyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.AccessKeyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.AutomationIdProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.HelpTextProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.IsColumnHeaderProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.IsRequiredForFormProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.IsRowHeaderProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.ItemStatusProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.ItemTypeProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.LabeledByProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode() { Property = AutomationProperties.NameProperty, Mode = TransferMode.AlwaysTransfer },
};
_feProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = FrameworkElement.DataContextProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = FrameworkElement.BindingGroupProperty, Mode = TransferMode.AlwaysTransfer },
};
_controlProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = Control.BackgroundProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = Control.BorderBrushProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = Control.BorderThicknessProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = Control.ForegroundProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = Control.FontFamilyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = Control.FontSizeProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = Control.FontStretchProperty, Mode = TransferMode.AlwaysTransfer},
new PropertyAndTransferMode () { Property = Control.FontStyleProperty, Mode = TransferMode.AlwaysTransfer},
new PropertyAndTransferMode () { Property = Control.FontWeightProperty, Mode = TransferMode.AlwaysTransfer},
};
_contentControlProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = ContentControl.ContentProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ContentControl.ContentStringFormatProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ContentControl.ContentTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ContentControl.ContentTemplateSelectorProperty, Mode = TransferMode.AlwaysTransfer },
};
_buttonProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = ButtonBase.CommandProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ButtonBase.CommandParameterProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ButtonBase.CommandTargetProperty, Mode = TransferMode.AlwaysTransfer },
};
_toggleButtonProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = ToggleButton.IsCheckedProperty, Mode = TransferMode.AlwaysTransfer },
};
_itemsControlProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = ItemsControl.ItemBindingGroupProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.ItemContainerStyleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.ItemContainerStyleSelectorProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.ItemsPanelProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.ItemsSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.ItemStringFormatProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.ItemTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.ItemTemplateSelectorProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.DisplayMemberPathProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ItemsControl.AlternationCountProperty, Mode = TransferMode.AlwaysTransfer },
};
_headeredItemsControlProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = HeaderedItemsControl.HeaderProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = HeaderedItemsControl.HeaderStringFormatProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = HeaderedItemsControl.HeaderTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = HeaderedItemsControl.HeaderTemplateSelectorProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonControlService.LabelProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.SmallImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.LargeImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.ToolTipTitleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.ToolTipDescriptionProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.ToolTipImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.ToolTipFooterTitleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.ToolTipFooterDescriptionProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.ToolTipFooterImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.QuickAccessToolBarIdProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.QuickAccessToolBarControlSizeDefinitionProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.CanAddToQuickAccessToolBarDirectlyProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonBrushProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonControlService.MouseOverBackgroundProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.MouseOverBorderBrushProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.PressedBackgroundProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.PressedBorderBrushProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.FocusedBackgroundProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.FocusedBorderBrushProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.CheckedBackgroundProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.CheckedBorderBrushProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonControlService.CornerRadiusProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonMenuButtonProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonMenuButton.DropDownHeightProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuButton.CanUserResizeHorizontallyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuButton.CanUserResizeVerticallyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuButton.ItemContainerTemplateSelectorProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuButton.UsesItemContainerTemplateProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonSplitButtonProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonSplitButton.IsCheckableProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.IsCheckedProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.DropDownToolTipTitleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.DropDownToolTipDescriptionProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.DropDownToolTipImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.DropDownToolTipFooterTitleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.DropDownToolTipFooterDescriptionProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.DropDownToolTipFooterImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.HeaderQuickAccessToolBarIdProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.CommandProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.CommandParameterProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.CommandTargetProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitButton.LabelPositionProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonMenuItemProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonMenuItem.CommandProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.CommandParameterProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.CommandTargetProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.ImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.QuickAccessToolBarImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.IsCheckableProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.IsCheckedProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.ItemContainerTemplateSelectorProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.UsesItemContainerTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.CanUserResizeHorizontallyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.CanUserResizeVerticallyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.DropDownHeightProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonMenuItem.StaysOpenOnClickProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonSplitMenuItemProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonSplitMenuItem.DropDownToolTipTitleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitMenuItem.DropDownToolTipDescriptionProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitMenuItem.DropDownToolTipImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitMenuItem.DropDownToolTipFooterTitleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitMenuItem.DropDownToolTipFooterDescriptionProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitMenuItem.DropDownToolTipFooterImageSourceProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonSplitMenuItem.HeaderQuickAccessToolBarIdProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonGalleryProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonGallery.CanUserFilterProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.CategoryStyleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.CategoryTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.CommandParameterProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.CommandProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.CommandTargetProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.FilterItemContainerStyleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.FilterItemContainerStyleSelectorProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.FilterItemTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.FilterItemTemplateSelectorProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.FilterMenuButtonStyleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.FilterPaneContentProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.FilterPaneContentTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.GalleryItemStyleProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.GalleryItemTemplateProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.IsSharedColumnSizeScopeProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.IsSynchronizedWithCurrentItemProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.MaxColumnCountProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.MinColumnCountProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.PreviewCommandParameterProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.SelectedItemProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.SelectedValuePathProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGallery.SelectedValueProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonGalleryCategoryProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonGalleryCategory.HeaderVisibilityProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGalleryCategory.IsSharedColumnSizeScopeProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGalleryCategory.MaxColumnCountProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonGalleryCategory.MinColumnCountProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonGalleryItemProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonGalleryItem.IsSelectedProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonGroupProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonGroup.GroupSizeDefinitionsProperty, Mode = TransferMode.AlwaysTransfer },
};
_textBoxProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = TextBox.TextProperty, Mode = TransferMode.AlwaysTransfer },
};
_ribbonComboBoxProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = RibbonComboBox.IsEditableProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonComboBox.IsReadOnlyProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonComboBox.StaysOpenOnEditProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = RibbonComboBox.TextProperty, Mode = TransferMode.AlwaysTransfer },
};
_scrollProperties = new PropertyAndTransferMode[]
{
new PropertyAndTransferMode () { Property = ScrollViewer.HorizontalScrollBarVisibilityProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ScrollViewer.VerticalScrollBarVisibilityProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ScrollViewer.CanContentScrollProperty, Mode = TransferMode.AlwaysTransfer },
new PropertyAndTransferMode () { Property = ScrollViewer.IsDeferredScrollingEnabledProperty, Mode = TransferMode.AlwaysTransfer },
};
}
}
}
private static object CreateClone(object original)
{
bool allowTransformations = false;
return CreateClone(original, allowTransformations);
}
internal static object CreateClone(object original, bool allowTransformations)
{
if (original is UIElement ||
original is ContentElement)
{
FrameworkElement feOriginal = original as FrameworkElement;
if (feOriginal != null)
{
FrameworkElement feClone = CreateInstance(feOriginal, allowTransformations);
if (feOriginal.TemplatedParent == null ||
feOriginal.TemplatedParent is ContentPresenter)
{
TransferProperties(feOriginal, feClone, /*cloningForTemplatePart*/ false);
}
else
{
TransferProperties(feOriginal.TemplatedParent as FrameworkElement, feClone, /*cloningForTemplatePart*/ true);
}
FrameworkElement feWrapper = WrapClone(feClone, allowTransformations);
if (feWrapper != feClone)
{
TransferProperties(feClone, feWrapper, /*cloningForTemplatePart*/ false);
}
return feWrapper;
}
else
{
object clone = Activator.CreateInstance(original.GetType());
TransferMarkupProperties(original, clone);
return clone;
}
}
else
{
Freezable freezable = original as Freezable;
if (freezable != null && !freezable.CanFreeze)
{
return freezable.Clone();
}
}
return original;
}
private static FrameworkElement CreateInstance(FrameworkElement original, bool allowTransformations)
{
if (allowTransformations)
{
RibbonMenuItem menuItem = original as RibbonMenuItem;
if (menuItem != null)
{
// Determine which control type the wrapper should be
// based on the Items and IsCheckable values.
if (!(original is RibbonSplitMenuItem))
{
if (menuItem.IsCheckable)
{
return new RibbonCheckBox();
}
else if (menuItem.Items.Count == 0)
{
return new RibbonButton();
}
else
{
return new RibbonMenuButton();
}
}
else
{
if (menuItem.Items.Count == 0)
{
if (menuItem.IsCheckable)
{
return new RibbonToggleButton();
}
else
{
return new RibbonButton();
}
}
else
{
return new RibbonSplitButton();
}
}
}
#if IN_RIBBON_GALLERY
InRibbonGallery inRibbonGallery = original as InRibbonGallery;
if (inRibbonGallery != null)
{
return new RibbonMenuButton();
}
#endif
}
return (FrameworkElement)Activator.CreateInstance(original.GetType());
}
// RibbonGallery cannot be added to the QAT directly. Instead, the
// RibbonGallery is wrapped inside a RibbonMenuButton when adding
// it to the QAT. Here we create that RibbonMenuButton host and
// transfer over interesting properties from the RibbonGallery to
// the RibbonMenuButton host
private static FrameworkElement WrapClone(FrameworkElement clone, bool allowTransformations)
{
if (allowTransformations && clone is RibbonGallery)
{
RibbonMenuButton wrapperButton = new RibbonMenuButton();
wrapperButton.Items.Add(clone);
return wrapperButton;
}
return clone;
}
// The following is a list of usual suspects that are template bound, data bound or locally set.
//
// These are the possible combinations
// - Original - Clone
// -------- -----
#if IN_RIBBON_GALLERY
// - InRibbonGallery - RibbonMenuButton
// - InRibbonGallery - InRibbonGallery
#endif
// - RibbonButton - RibbonButton
// - RibbonToggleButton - RibbonToggleButton
// - RibbonRadioButton - RibbonRadioButton
// - RibbonCheckBox - RibbonCheckBox
// - RibbonTextBox - RibbonTextBox
// - RibbonMenuButton - RibbonMenuButton
// - RibbonSplitButton - RibbonButton
// - RibbonSplitButton - RibbonToggleButton
// - RibbonSplitButton - RibbonSplitButton
// - RibbonMenuItem - RibbonButton
// - RibbonMenuItem - RibbonToggleButton
// - RibbonMenuItem - RibbonMenuButton
// - RibbonMenuItem - RibbonMenuItem
// - RibbonSplitMenuItem - RibbonButton
// - RibbonSplitMenuItem - RibbonToggleButton
// - RibbonSplitMenuItem - RibbonSplitButton
// - RibbonSplitMenuItem - RibbonSplitMenuItem
// - RibbonGallery - RibbonMenuButton
// - RibbonGallery - RibbonGallery
// - RibbonSeparator - RibbonSeparator
// - RibbonGroup - RibbonGroup
// - RibbonComboBox - RibbonComboBox
// - RibbonGalleryCategory - RibbonGalleryCategory
// - RibbonGalleryItem - RibbonGalleryItem
private static void TransferProperties(FrameworkElement original, FrameworkElement clone, bool cloningForTemplatePart)
{
#if IN_RIBBON_GALLERY
if (original is InRibbonGallery)
{
Debug.Assert(clone is RibbonMenuButton);
TransferProperties(original, clone, _automationProperties);
TransferProperties(original, clone, _feProperties);
TransferProperties(original, clone, _controlProperties);
TransferProperties(original, clone, _ribbonProperties);
TransferProperties(original, clone, _ribbonBrushProperties);
TransferProperties(original, clone, _itemsControlProperties);
TransferItems((ItemsControl)original, (ItemsControl)clone);
TransferProperties(original, clone, _ribbonMenuButtonProperties);
TransferProperties(original, clone, _scrollProperties);
}
else if (clone.GetType().IsInstanceOfType(original))
#else
if (clone.GetType().IsInstanceOfType(original))
#endif
{
TransferProperties(original, clone, _automationProperties);
TransferProperties(original, clone, _feProperties);
TransferProperty(original, clone, FrameworkElement.StyleProperty, TransferMode.AlwaysTransfer);
if (original is Control)
{
TransferProperties(original, clone, _controlProperties);
TransferProperties(original, clone, _ribbonProperties);
TransferProperties(original, clone, _ribbonBrushProperties);
if (original is ContentControl)
{
TransferProperties(original, clone, _contentControlProperties);
if (original is ButtonBase)
{
TransferProperties(original, clone, _buttonProperties);
if (original is ToggleButton)
{
TransferProperties(original, clone, _toggleButtonProperties);
}
}
else if (original is RibbonGalleryItem)
{
TransferProperties(original, clone, _ribbonGalleryItemProperties);
}
}
else if (original is ItemsControl)
{
TransferProperties(original, clone, _itemsControlProperties);
TransferProperties(original, clone, _scrollProperties);
TransferItems((ItemsControl)original, (ItemsControl)clone);
if (original is HeaderedItemsControl)
{
TransferProperties(original, clone, _headeredItemsControlProperties);
if (original is RibbonMenuItem)
{
TransferProperties(original, clone, _ribbonMenuItemProperties);
if (original is RibbonSplitMenuItem)
{
TransferProperties(original, clone, _ribbonSplitMenuItemProperties);
}
}
else if (original is RibbonGalleryCategory)
{
TransferProperties(original, clone, _ribbonGalleryCategoryProperties);
}
else if (original is RibbonGroup)
{
TransferProperties(original, clone, _ribbonGroupProperties);
}
}
else if (original is RibbonMenuButton)
{
TransferProperties(original, clone, _ribbonMenuButtonProperties);
if (original is RibbonSplitButton)
{
TransferProperties(original, clone, _ribbonSplitButtonProperties);
}
else if (original is RibbonComboBox)
{
TransferProperties(original, clone, _ribbonComboBoxProperties);
}
}
else if (original is RibbonGallery)
{
TransferProperties(original, clone, _ribbonGalleryProperties);
}
}
else if (original is TextBox)
{
TransferProperties(original, clone, _textBoxProperties);
}
}
TransferMarkupProperties(original, clone);
}
else
{
TransferProperties(original, clone, _automationProperties);
TransferProperties(original, clone, _feProperties);
TransferProperties(original, clone, _controlProperties);
if (original is RibbonSplitButton)
{
Debug.Assert(clone is ButtonBase,
"We should only be here if a SplitButton's header is being added to the QAT");
TransferProperties(original, clone, _ribbonProperties);
TransferProperties(original, clone, _ribbonBrushProperties);
TransferProperty(original, clone, RibbonSplitButton.IsCheckedProperty, ToggleButton.IsCheckedProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitButton.HeaderQuickAccessToolBarIdProperty, RibbonControlService.QuickAccessToolBarIdProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitButton.CommandProperty, ButtonBase.CommandProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitButton.CommandParameterProperty, ButtonBase.CommandParameterProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitButton.CommandTargetProperty, ButtonBase.CommandTargetProperty, TransferMode.AlwaysTransfer);
}
else if (original is RibbonMenuItem)
{
TransferProperties(original, clone, _ribbonBrushProperties);
if (original.GetValue(RibbonMenuItem.HeaderProperty) is String)
{
TransferProperty(original, clone, RibbonMenuItem.HeaderProperty, RibbonControlService.LabelProperty, TransferMode.AlwaysTransfer);
}
TransferProperty(original, clone, RibbonMenuItem.QuickAccessToolBarImageSourceProperty, RibbonControlService.SmallImageSourceProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonControlService.ToolTipTitleProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonControlService.ToolTipDescriptionProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonControlService.ToolTipImageSourceProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonControlService.ToolTipFooterTitleProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonControlService.ToolTipFooterDescriptionProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonControlService.ToolTipFooterImageSourceProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonControlService.QuickAccessToolBarIdProperty, TransferMode.AlwaysTransfer);
if (original is RibbonSplitMenuItem)
{
Debug.Assert(clone is ButtonBase || clone is RibbonMenuButton,
"We could be here if either the header of a SplitMenuItem is being added to the QAT as a Button or a Toggle Button or the entire SplitMenuItem is being added to the QAT as MenuButton or a SplitButton");
if (clone is ButtonBase)
{
TransferProperty(original, clone, RibbonMenuItem.IsCheckedProperty, ToggleButton.IsCheckedProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandProperty, ButtonBase.CommandProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandParameterProperty, ButtonBase.CommandParameterProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandTargetProperty, ButtonBase.CommandTargetProperty, TransferMode.AlwaysTransfer);
if (cloningForTemplatePart)
{
TransferProperty(original, clone, RibbonSplitMenuItem.HeaderQuickAccessToolBarIdProperty, RibbonControlService.QuickAccessToolBarIdProperty, TransferMode.AlwaysTransfer);
}
}
else if (clone is RibbonMenuButton)
{
TransferProperties(original, clone, _itemsControlProperties);
TransferItems((ItemsControl)original, (ItemsControl)clone);
TransferProperties(original, clone, _ribbonMenuButtonProperties);
if (clone is RibbonSplitButton)
{
TransferProperty(original, clone, RibbonMenuItem.IsCheckableProperty, RibbonSplitButton.IsCheckableProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.IsCheckedProperty, RibbonSplitButton.IsCheckedProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandProperty, RibbonSplitButton.CommandProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandParameterProperty, RibbonSplitButton.CommandParameterProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandTargetProperty, RibbonSplitButton.CommandTargetProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitMenuItem.DropDownToolTipTitleProperty, RibbonSplitButton.DropDownToolTipTitleProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitMenuItem.DropDownToolTipDescriptionProperty, RibbonSplitButton.DropDownToolTipDescriptionProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitMenuItem.DropDownToolTipImageSourceProperty, RibbonSplitButton.DropDownToolTipImageSourceProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitMenuItem.DropDownToolTipFooterTitleProperty, RibbonSplitButton.DropDownToolTipFooterTitleProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitMenuItem.DropDownToolTipFooterDescriptionProperty, RibbonSplitButton.DropDownToolTipFooterDescriptionProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonSplitMenuItem.DropDownToolTipFooterImageSourceProperty, RibbonSplitButton.DropDownToolTipFooterImageSourceProperty, TransferMode.AlwaysTransfer);
}
}
}
else
{
Debug.Assert(clone is ButtonBase || clone is RibbonMenuButton,
"We could be here if either a leaf level MenuItem is being added to the QAT as a Button or a Toggle Button or the entire MenuItem is being added to the QAT as MenuButton");
if (clone is ButtonBase)
{
TransferProperty(original, clone, RibbonMenuItem.IsCheckedProperty, ToggleButton.IsCheckedProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandProperty, ButtonBase.CommandProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandParameterProperty, ButtonBase.CommandParameterProperty, TransferMode.AlwaysTransfer);
TransferProperty(original, clone, RibbonMenuItem.CommandTargetProperty, ButtonBase.CommandTargetProperty, TransferMode.AlwaysTransfer);
}
else
{
TransferProperties(original, clone, _itemsControlProperties);
TransferItems((ItemsControl)original, (ItemsControl)clone);
TransferProperties(original, clone, _ribbonMenuButtonProperties);
}
}
}
else if (original is RibbonGallery)
{
Debug.Assert(clone is RibbonMenuButton,
"We should only be here if a RibbonGallery is being wrapped in a RibbonMenuButton");
TransferProperties(original, clone, _ribbonProperties);
TransferProperties(original, clone, _ribbonBrushProperties);
}
}
}
private static void TransferProperties(FrameworkElement original, FrameworkElement clone, PropertyAndTransferMode[] properties)
{
for (int i = 0; i < properties.Length; i++)
{
TransferProperty(original, clone, properties[i].Property, properties[i].Mode);
}
}
private static void TransferItems(ItemsControl originalItemsControl, ItemsControl cloneItemsControl)
{
if (originalItemsControl.ItemsSource == null)
{
for (int i = 0; i < originalItemsControl.Items.Count; i++)
{
object clonedItem = CreateClone(originalItemsControl.Items[i]);
cloneItemsControl.Items.Add(clonedItem);
}
}
}
private static void TransferMarkupProperties(object original, object clone)
{
MarkupObject markupObjOriginal = MarkupWriter.GetMarkupObjectFor(original);
foreach (MarkupProperty markupProp in markupObjOriginal.Properties)
{
if (markupProp.DependencyProperty != null)
{
DependencyObject cloneElement = (DependencyObject)clone;
BaseValueSource baseValueSource = DependencyPropertyHelper.GetValueSource(
cloneElement, markupProp.DependencyProperty).BaseValueSource;
if (baseValueSource >= BaseValueSource.ParentTemplate)
{
// If we have already transferred this property
// previously omit it now.
continue;
}
if (markupProp.DependencyProperty == KeyTipService.KeyTipProperty ||
markupProp.DependencyProperty == RibbonSplitButton.HeaderKeyTipProperty ||
markupProp.DependencyProperty == RibbonMenuButton.IsDropDownOpenProperty ||
markupProp.DependencyProperty == RibbonGroup.IsDropDownOpenProperty ||
markupProp.DependencyProperty == MenuItem.IsSubmenuOpenProperty)
{
// Do not copy the KeyTip properties. KeyTips for elements
// in the QAT will be generated separately (except HeaderKeyTip).
continue;
}
BindingBase binding = markupProp.Value as BindingBase;
if (binding != null)
{
// Transfer bindings.
BindingOperations.SetBinding(cloneElement, markupProp.DependencyProperty, binding);
}
else
{
MarkupExtension markupExtension = markupProp.Value as MarkupExtension;
if (markupExtension != null)
{
// Transfer dynamic resources and other markup extensions.
cloneElement.SetValue(markupProp.DependencyProperty, markupExtension.ProvideValue(null));
}
else
{
// Transfer other DependencyProperties.
object clonedPropertyValue = CreateClone(markupProp.Value);
cloneElement.SetValue(markupProp.DependencyProperty, clonedPropertyValue);
}
}
}
else if (markupProp.PropertyDescriptor != null)
{
// Transfer CLR properties.
if (markupProp.PropertyDescriptor.SerializationVisibility == DesignerSerializationVisibility.Content)
{
if (markupProp.Name == "Items" &&
markupProp.PropertyDescriptor.ComponentType == typeof(ItemsControl))
{
// Skip the ItemsControl.Items property
// since this will be copied automatically.
continue;
}
IList items = markupProp.PropertyDescriptor.GetValue(clone) as IList;
if (items != null)
{
foreach (MarkupObject subObj in markupProp.Items)
{
items.Add(CreateClone(subObj));
}
}
else
{
object clonedPropertyValue = CreateClone(markupProp.Value);
markupProp.PropertyDescriptor.SetValue(clone, clonedPropertyValue);
}
}
else
{
object clonedPropertyValue = CreateClone(markupProp.Value);
markupProp.PropertyDescriptor.SetValue(clone, clonedPropertyValue);
}
}
}
}
#endregion QAT
#region ApplicationMenu
internal static void SetApplicationMenuLevel(bool parentIsTopLevel, DependencyObject element)
{
RibbonApplicationMenuItem rami = element as RibbonApplicationMenuItem;
if (rami != null)
{
if (parentIsTopLevel)
{
rami.Level = RibbonApplicationMenuItemLevel.Middle;
}
else
{
rami.Level = RibbonApplicationMenuItemLevel.Sub;
}
}
else
{
RibbonApplicationSplitMenuItem rasmi = element as RibbonApplicationSplitMenuItem;
if (rasmi != null)
{
if (parentIsTopLevel)
{
rasmi.Level = RibbonApplicationMenuItemLevel.Middle;
}
else
{
rasmi.Level = RibbonApplicationMenuItemLevel.Sub;
}
}
}
}
internal static bool CoerceIsSubmenuOpenForTopLevelItem(RibbonMenuItem menuItem, ItemsControl parentItemsControl, bool baseValue)
{
bool isSubMenuOpen = (bool)baseValue;
if (!isSubMenuOpen && menuItem.CloseSubmenuTimer != null && menuItem.CloseSubmenuTimer.IsEnabled)
{
RibbonApplicationMenu ram = parentItemsControl as RibbonApplicationMenu;
if (ram != null)
{
RibbonMenuItem currentMenuItem = ram.RibbonCurrentSelection as RibbonMenuItem;
if (currentMenuItem != null && currentMenuItem.CanOpenSubMenu && currentMenuItem != menuItem)
{
return true;
}
}
}
return baseValue;
}
internal static void HookPopupForTopLevelMenuItem(RibbonMenuItem menuItem, ItemsControl parentItemsControl)
{
Popup popup = menuItem.Popup;
if (popup != null)
{
Binding binding = new Binding("SubmenuPlaceholder")
{
Source = parentItemsControl
};
BindingOperations.SetBinding(popup, Popup.PlacementTargetProperty, binding);
binding = new Binding("SubmenuPlaceholder.ActualWidth")
{
Source = parentItemsControl
};
BindingOperations.SetBinding(popup, Popup.WidthProperty, binding);
binding = new Binding("SubmenuPlaceholder.ActualHeight")
{
Source = parentItemsControl
};
BindingOperations.SetBinding(popup, Popup.HeightProperty, binding);
BindingOperations.SetBinding(menuItem, RibbonMenuItem.DropDownHeightProperty, binding);
}
}
internal static void UnhookPopupForTopLevelMenuItem(RibbonMenuItem menuItem)
{
Popup popup = menuItem.Popup;
if (popup != null)
{
popup.ClearValue(Popup.PlacementTargetProperty);
popup.ClearValue(Popup.WidthProperty);
popup.ClearValue(Popup.HeightProperty);
menuItem.CoerceValue(RibbonMenuItem.DropDownHeightProperty);
}
}
public static void OnApplicationMenuItemUpDownKeyDown(KeyEventArgs e, RibbonMenuItem menuItem)
{
if (e.Handled || menuItem.IsSubmenuOpen)
{
return;
}
if (e.Key == Key.Up ||
e.Key == Key.Down)
{
RibbonApplicationMenu applicationMenu = ItemsControl.ItemsControlFromItemContainer(menuItem) as RibbonApplicationMenu;
if (applicationMenu != null)
{
if (RibbonHelper.IsEndFocusableMenuItem(menuItem, e.Key == Key.Up /* isFirst */))
{
if (e.Key == Key.Down)
{
// If the focus is at the last focusable item,
// then try moving the focus to first element of
// auxiliary pane and then to first element of footer
// pane if needed.
if (applicationMenu.AuxiliaryPaneMoveFocus(FocusNavigationDirection.First) ||
applicationMenu.FooterPaneMoveFocus(FocusNavigationDirection.First))
{
e.Handled = true;
}
}
else
{
// If the focus is at the first focusable item,
// then try moving the focus to last element of
// footer pane and then to last element of auxiliary
// pane if needed.
if (applicationMenu.FooterPaneMoveFocus(FocusNavigationDirection.Last) ||
applicationMenu.AuxiliaryPaneMoveFocus(FocusNavigationDirection.Last))
{
e.Handled = true;
}
}
}
}
}
}
/// <summary>
/// Helper method to determine if the given menu item
/// is the first / last focusable item of its parent.
/// </summary>
private static bool IsEndFocusableMenuItem(RibbonMenuItem menuItem, bool isFirst)
{
ItemsControl parentItemsControl = ItemsControl.ItemsControlFromItemContainer(menuItem);
Debug.Assert(parentItemsControl != null);
int parentItemCount = parentItemsControl.Items.Count;
int itemIndex = parentItemsControl.ItemContainerGenerator.IndexFromContainer(menuItem);
int incr = 1;
if (isFirst)
{
incr = -1;
}
for (int i = itemIndex + incr; i < parentItemCount && i >= 0; i += incr)
{
UIElement container = parentItemsControl.ItemContainerGenerator.ContainerFromIndex(i) as UIElement;
if (container != null &&
container.IsVisible &&
container.IsEnabled &&
container.Focusable)
{
return false;
}
}
return true;
}
#endregion
#region KeyTips
#if RIBBON_IN_FRAMEWORK
#endif
public static PresentationSource GetPresentationSourceFromVisual(Visual visual)
{
if (visual == null)
{
return null;
}
#if RIBBON_IN_FRAMEWORK
return PresentationSource.CriticalFromVisual(visual);
#else
return PresentationSource.FromVisual(visual);
#endif
}
/// <summary>
/// Produces default system beep.
/// </summary>
public static void Beep()
{
// Ignore the results of Beep because it is of very low importance.
NativeMethods.MessageBeep(0);
}
// ------------------------------------------------------------------
// Retrieve CultureInfo property from specified element.
// ------------------------------------------------------------------
public static CultureInfo GetCultureInfo(DependencyObject element)
{
XmlLanguage language = (XmlLanguage)element.GetValue(FrameworkElement.LanguageProperty);
try
{
return language.GetSpecificCulture();
}
catch (InvalidOperationException)
{
// We default to en-US if no part of the language tag is recognized.
return InvariantEnglishUS;
}
}
private static CultureInfo invariantEnglishUS;
public static CultureInfo InvariantEnglishUS
{
get
{
if (invariantEnglishUS == null)
{
invariantEnglishUS = CultureInfo.ReadOnly(new CultureInfo("en-us", false));
}
return invariantEnglishUS;
}
}
/// <summary>
/// Returns the first UIElement in the ancestral chain
/// including the element itself.
/// </summary>
public static UIElement GetContainingUIElement(DependencyObject element)
{
UIElement uie = element as UIElement;
if (uie != null)
{
return uie;
}
else
{
ContentElement ce = element as ContentElement;
if (ce != null)
{
DependencyObject parent = ContentOperations.GetParent(ce);
if (parent == null)
{
parent = LogicalTreeHelper.GetParent(ce);
}
if (parent != null)
{
return GetContainingUIElement(parent);
}
}
}
return null;
}
public static void SetDefaultQatKeyTipPlacement(ActivatingKeyTipEventArgs e)
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipTopAtTargetCenter;
e.KeyTipHorizontalOffset = e.KeyTipVerticalOffset = 0;
}
/// <summary>
/// Helper method to determine the keytip position for
/// RibbonButton, RibbonToggleButton, RibbonMenuButton,
/// and RibbonRadioButton.
/// </summary>
public static void SetKeyTipPlacementForButton(DependencyObject element,
ActivatingKeyTipEventArgs e,
UIElement mediumPlacementTarget)
{
if (RibbonControlService.GetIsInQuickAccessToolBar(element))
{
SetDefaultQatKeyTipPlacement(e);
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipHorizontalOffset = e.KeyTipVerticalOffset = 0;
RibbonControlSizeDefinition controlSizeDefinition = RibbonControlService.GetControlSizeDefinition(element);
if (controlSizeDefinition != null)
{
if (controlSizeDefinition.IsLabelVisible)
{
if (controlSizeDefinition.ImageSize == RibbonImageSize.Large)
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetBottom;
}
else if (controlSizeDefinition.ImageSize == RibbonImageSize.Small)
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipLeftAtTargetCenter;
e.PlacementTarget = mediumPlacementTarget;
}
}
else
{
if (controlSizeDefinition.ImageSize == RibbonImageSize.Small)
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipLeftAtTargetCenter;
}
}
}
else
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetBottom;
}
}
}
/// <summary>
/// Helper method which determines the keytip position for
/// RibbonTextBox, RibbonCheckBox and RibbonComboBox.
/// </summary>
public static void SetKeyTipPlacementForTextBox(DependencyObject element,
ActivatingKeyTipEventArgs e,
UIElement nonLargePlacementTarget)
{
if (RibbonControlService.GetIsInQuickAccessToolBar(element))
{
SetDefaultQatKeyTipPlacement(e);
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipHorizontalOffset = e.KeyTipVerticalOffset = 0;
RibbonControlSizeDefinition controlSizeDefinition = RibbonControlService.GetControlSizeDefinition(element);
if (controlSizeDefinition != null)
{
if (controlSizeDefinition.ImageSize == RibbonImageSize.Large)
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetBottom;
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipLeftAtTargetCenter;
e.PlacementTarget = nonLargePlacementTarget;
}
}
else
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetBottom;
}
}
}
public static void SetKeyTipPlacementForSplitButtonHeader(RibbonSplitButton splitButton,
ActivatingKeyTipEventArgs e,
UIElement mediumPlacementTarget)
{
bool dropDownKeyTipSet = !string.IsNullOrEmpty(splitButton.KeyTip);
if (splitButton.IsInQuickAccessToolBar)
{
SetDefaultQatKeyTipPlacement(e);
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipHorizontalOffset = e.KeyTipVerticalOffset = 0;
RibbonControlSizeDefinition controlSizeDefinition = splitButton.ControlSizeDefinition;
if (controlSizeDefinition != null)
{
if (controlSizeDefinition.IsLabelVisible)
{
if (controlSizeDefinition.ImageSize == RibbonImageSize.Large)
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetTop;
}
else if (controlSizeDefinition.ImageSize == RibbonImageSize.Small)
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipLeftAtTargetCenter;
e.PlacementTarget = mediumPlacementTarget;
}
}
else
{
if (controlSizeDefinition.ImageSize == RibbonImageSize.Small)
{
if (dropDownKeyTipSet)
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipLeftAtTargetCenter;
}
}
}
}
else
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetTop;
}
}
}
public static void SetKeyTipPlacementForSplitButtonDropDown(RibbonSplitButton splitButton,
ActivatingKeyTipEventArgs e,
UIElement mediumPlacementTarget)
{
bool headerKeyTipSet = !(string.IsNullOrEmpty(splitButton.HeaderKeyTip));
if (splitButton.IsInQuickAccessToolBar)
{
SetDefaultQatKeyTipPlacement(e);
if (headerKeyTipSet)
{
e.PlacementTarget = mediumPlacementTarget;
}
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetCenter;
e.KeyTipHorizontalOffset = e.KeyTipVerticalOffset = 0;
RibbonControlSizeDefinition controlSizeDefinition = splitButton.ControlSizeDefinition;
if (controlSizeDefinition != null)
{
if (controlSizeDefinition.IsLabelVisible)
{
if (controlSizeDefinition.ImageSize == RibbonImageSize.Large)
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetBottom;
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
if (headerKeyTipSet)
{
e.PlacementTarget = mediumPlacementTarget;
}
}
}
else
{
if (controlSizeDefinition.ImageSize == RibbonImageSize.Small)
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipCenterAtTargetCenter;
if (headerKeyTipSet)
{
e.PlacementTarget = mediumPlacementTarget;
}
}
}
}
else
{
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipCenterAtTargetBottom;
}
}
}
#if IN_RIBBON_GALLERY
public static void SetKeyTipPlacementForInRibbonGallery(InRibbonGallery irg,
ActivatingKeyTipEventArgs e)
{
if (irg.IsInQuickAccessToolBar)
{
SetDefaultQatKeyTipPlacement(e);
}
else
{
e.KeyTipHorizontalPlacement = KeyTipHorizontalPlacement.KeyTipLeftAtTargetCenter;
e.KeyTipVerticalPlacement = KeyTipVerticalPlacement.KeyTipTopAtTargetBottom;
e.KeyTipHorizontalOffset = e.KeyTipVerticalOffset = 0;
if (irg.PartToggleButton != null)
{
e.PlacementTarget = irg.PartToggleButton;
}
}
}
#endif
public static void OpenParentRibbonGroupDropDownSync(FrameworkElement fe, bool templateApplied)
{
if (!templateApplied)
{
// Apply template if not yet applied.
fe.ApplyTemplate();
}
// Get the Parent RibbonGroup and open its dropdown if needed.
RibbonGroup ribbonGroup = TreeHelper.FindAncestor(fe, delegate(DependencyObject element) { return (element is RibbonGroup); }) as RibbonGroup;
if (ribbonGroup == null)
{
ribbonGroup = TreeHelper.FindLogicalAncestor<RibbonGroup>(fe);
}
if (ribbonGroup != null &&
ribbonGroup.IsCollapsed &&
!ribbonGroup.IsDropDownOpen)
{
ribbonGroup.IsDropDownOpen = true;
fe.UpdateLayout();
}
}
#endregion
#region UIA
internal static AutomationPeer CreatePeer(UIElement element)
{
AutomationPeer peer = UIElementAutomationPeer.CreatePeerForElement(element);
if (peer == null)
{
FrameworkElement elementFE = element as FrameworkElement;
if (elementFE != null)
peer = new FrameworkElementAutomationPeer(elementFE);
else
peer = new UIElementAutomationPeer(element);
}
return peer;
}
#endregion UIA
#region DropDown ItemsControls
public static void SetDropDownHeight(FrameworkElement itemsPresenter, bool hasGallery, double dropDownHeight)
{
if (itemsPresenter != null)
{
// First time the dropdown opens, HasGallery is always false
// because item container generation never happened yet and hence
// it ignores DropDownHeight. Hence reuse DropDownHeight when HasGallery
// value changes.
double oldHeight = itemsPresenter.Height;
double newHeight = double.NaN;
if (hasGallery)
{
newHeight = dropDownHeight;
}
if (!DoubleUtil.AreClose(oldHeight, newHeight) &&
!(double.IsNaN(oldHeight) && double.IsNaN(newHeight)))
{
itemsPresenter.Height = newHeight;
itemsPresenter.Dispatcher.BeginInvoke(
(Action)delegate()
{
TreeHelper.InvalidateMeasureForVisualAncestorPath<Popup>(itemsPresenter);
},
DispatcherPriority.Normal,
null);
}
}
}
internal static void InvalidateScrollBarVisibility(ScrollViewer submenuScrollViewer)
{
if (submenuScrollViewer != null)
{
// The scroll viewer needs to re-evaluate the visibility of the scrollbars
// and that happens in its MeasureOverride call. Also note that we need to
// make this invalidate call async because we may already be within a
// ScrollViewer measure pass, by which we would miss the boat.
submenuScrollViewer.Dispatcher.BeginInvoke((Action)delegate()
{
submenuScrollViewer.InvalidateMeasure();
},
DispatcherPriority.Render);
}
}
#endregion
#region Transforms
internal static Matrix GetTransformToDevice(Visual targetVisual)
{
HwndSource hwndSource = null;
if (targetVisual != null)
{
hwndSource = PresentationSource.FromVisual(targetVisual) as HwndSource;
}
if (hwndSource != null)
{
CompositionTarget ct = hwndSource.CompositionTarget;
if (ct != null)
{
return ct.TransformToDevice;
}
}
return Matrix.Identity;
}
#endregion Transforms
#region Resizing
/// <summary>
/// Recursively patches the measure invalidation across the
/// descendant paths
/// </summary>
public static bool FixMeasureInvalidationPaths(DependencyObject element)
{
int childCount = VisualTreeHelper.GetChildrenCount(element);
bool measureInvalid = false;
for (int i = 0; i < childCount; i++)
{
DependencyObject child = VisualTreeHelper.GetChild(element, i);
if (child != null)
{
// Determine if any of the child's subtree is
// dirty for measure
measureInvalid |= FixMeasureInvalidationPaths(child);
}
}
UIElement uie = element as UIElement;
if (uie != null)
{
if (!uie.IsMeasureValid)
{
measureInvalid = true;
}
else if (measureInvalid)
{
// Invalidate self for measure if any of the
// descendants in the subtree is invalid
uie.InvalidateMeasure();
}
}
return measureInvalid;
}
#endregion
}
}
|