2605 lines
94 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <copyright file="Menu.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
namespace System.Web.UI.WebControls {
using System.Collections;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Drawing.Design;
using System.Globalization;
using System.IO;
using System.Reflection;
using System.Security.Permissions;
using System.Text;
using System.Web.UI.WebControls.Adapters;
using System.Web.Util;
/// <devdoc>
/// Provides a cascading pop-out hierarchical menu control
/// </devdoc>
[ControlValueProperty("SelectedValue")]
[DefaultEvent("MenuItemClick")]
[Designer("System.Web.UI.Design.WebControls.MenuDesigner, " + AssemblyRef.SystemDesign)]
[SupportsEventValidation]
public partial class Menu : HierarchicalDataBoundControl, IPostBackEventHandler, INamingContainer {
internal const int ScrollUpImageIndex = 0;
internal const int ScrollDownImageIndex = 1;
internal const int PopOutImageIndex = 2;
internal const int ImageUrlsCount = 3;
private const string _getDesignTimeStaticHtml = "GetDesignTimeStaticHtml";
private const string _getDesignTimeDynamicHtml = "GetDesignTimeDynamicHtml";
// static readonly instead of const to be able to change it in the future without breaking existing client code
public static readonly string MenuItemClickCommandName = "Click";
private static readonly object _menuItemClickedEvent = new object();
private static readonly object _menuItemDataBoundEvent = new object();
private MenuRenderingMode _renderingMode = MenuRenderingMode.Default;
private string[] _imageUrls;
private SubMenuStyle _staticMenuStyle;
private SubMenuStyle _dynamicMenuStyle;
private MenuItemStyle _staticItemStyle;
private MenuItemStyle _staticSelectedStyle;
private Style _staticHoverStyle;
private HyperLinkStyle _staticHoverHyperLinkStyle;
private MenuItemStyle _dynamicItemStyle;
private MenuItemStyle _dynamicSelectedStyle;
private Style _dynamicHoverStyle;
private HyperLinkStyle _dynamicHoverHyperLinkStyle;
private Style _rootMenuItemStyle;
private SubMenuStyleCollection _levelStyles;
private MenuItemStyleCollection _levelMenuItemStyles;
private MenuItemStyleCollection _levelSelectedStyles;
// Cached styles. In the current implementation, the styles are the same for all items
// and submenus at a given depth.
private List<MenuItemStyle> _cachedMenuItemStyles;
private List<SubMenuStyle> _cachedSubMenuStyles;
private List<string> _cachedMenuItemClassNames;
private List<string> _cachedMenuItemHyperLinkClassNames;
private List<string> _cachedSubMenuClassNames;
private Collection<int> _cachedLevelsContainingCssClass;
private MenuItem _rootItem;
private MenuItem _selectedItem;
private MenuItemBindingCollection _bindings;
private string _cachedScrollUpImageUrl;
private string _cachedScrollDownImageUrl;
private string _cachedPopOutImageUrl;
private ITemplate _dynamicTemplate;
private ITemplate _staticTemplate;
private int _maximumDepth;
private int _nodeIndex;
private string _currentSiteMapNodeUrl;
private bool _dataBound;
private bool _subControlsDataBound;
private bool _accessKeyRendered;
private PopOutPanel _panel;
private Style _panelStyle;
private bool _isNotIE;
private Type _designTimeTextWriterType;
private MenuRenderer _renderer;
/// <devdoc>
/// Creates a new instance of a Menu.
/// </devdoc>
public Menu() {
_nodeIndex = 0;
_maximumDepth = 0;
IncludeStyleBlock = true;
}
internal bool AccessKeyRendered {
get {
return _accessKeyRendered;
}
set {
_accessKeyRendered = value;
}
}
private Collection<int> CachedLevelsContainingCssClass {
get {
if (_cachedLevelsContainingCssClass == null) {
_cachedLevelsContainingCssClass = new Collection<int>();
}
return _cachedLevelsContainingCssClass;
}
}
private List<string> CachedMenuItemClassNames {
get {
if (_cachedMenuItemClassNames == null) {
_cachedMenuItemClassNames = new List<string>();
}
return _cachedMenuItemClassNames;
}
}
private List<string> CachedMenuItemHyperLinkClassNames {
get {
if (_cachedMenuItemHyperLinkClassNames == null) {
_cachedMenuItemHyperLinkClassNames = new List<string>();
}
return _cachedMenuItemHyperLinkClassNames;
}
}
private List<MenuItemStyle> CachedMenuItemStyles {
get {
if (_cachedMenuItemStyles == null) {
_cachedMenuItemStyles = new List<MenuItemStyle>();
}
return _cachedMenuItemStyles;
}
}
private List<string> CachedSubMenuClassNames {
get {
if (_cachedSubMenuClassNames == null) {
_cachedSubMenuClassNames = new List<string>();
}
return _cachedSubMenuClassNames;
}
}
private List<SubMenuStyle> CachedSubMenuStyles {
get {
if (_cachedSubMenuStyles == null) {
_cachedSubMenuStyles = new List<SubMenuStyle>();
}
return _cachedSubMenuStyles;
}
}
/// <devdoc>
/// Gets the hidden field ID for the expand state of this Menu
/// </devdoc>
internal string ClientDataObjectID {
get {
//
return ClientID + "_Data";
}
}
public override ControlCollection Controls {
get {
EnsureChildControls();
return base.Controls;
}
}
[
DefaultValue(null),
Editor("System.Web.UI.Design.WebControls.MenuBindingsEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)),
MergableProperty(false),
PersistenceMode(PersistenceMode.InnerProperty),
WebCategory("Data"),
WebSysDescription(SR.Menu_Bindings)
]
public MenuItemBindingCollection DataBindings {
get {
if (_bindings == null) {
_bindings = new MenuItemBindingCollection(this);
if (IsTrackingViewState) {
((IStateManager)_bindings).TrackViewState();
}
}
return _bindings;
}
}
[WebCategory("Behavior")]
[DefaultValue(500)]
[WebSysDescription(SR.Menu_DisappearAfter)]
[Themeable(false)]
public int DisappearAfter {
get {
object o = ViewState["DisappearAfter"];
if (o == null) {
return 500;
}
return (int)o;
}
set {
if (value < -1) {
throw new ArgumentOutOfRangeException("value");
}
ViewState["DisappearAfter"] = value;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[Themeable(true)]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_DynamicBottomSeparatorImageUrl)]
public string DynamicBottomSeparatorImageUrl {
get {
object s = ViewState["DynamicBottomSeparatorImageUrl"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["DynamicBottomSeparatorImageUrl"] = value;
}
}
[
DefaultValue(true),
WebCategory("Appearance"),
WebSysDescription(SR.Menu_DynamicDisplayPopOutImage)
]
public bool DynamicEnableDefaultPopOutImage {
get {
object o = ViewState["DynamicEnableDefaultPopOutImage"];
if (o == null) {
return true;
}
return (bool)o;
}
set {
ViewState["DynamicEnableDefaultPopOutImage"] = value;
}
}
[
DefaultValue(0),
WebCategory("Appearance"),
WebSysDescription(SR.Menu_DynamicHorizontalOffset)
]
public int DynamicHorizontalOffset {
get {
object o = ViewState["DynamicHorizontalOffset"];
if (o == null) {
return 0;
}
return (int)o;
}
set {
ViewState["DynamicHorizontalOffset"] = value;
}
}
[
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebCategory("Styles"),
WebSysDescription(SR.Menu_DynamicHoverStyle)
]
public Style DynamicHoverStyle {
get {
if (_dynamicHoverStyle == null) {
_dynamicHoverStyle = new Style();
if (IsTrackingViewState) {
((IStateManager)_dynamicHoverStyle).TrackViewState();
}
}
return _dynamicHoverStyle;
}
}
[DefaultValue("")]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_DynamicItemFormatString)]
public string DynamicItemFormatString {
get {
object s = ViewState["DynamicItemFormatString"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["DynamicItemFormatString"] = value;
}
}
[
WebCategory("Styles"),
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebSysDescription(SR.Menu_DynamicMenuItemStyle)
]
public MenuItemStyle DynamicMenuItemStyle {
get {
if (_dynamicItemStyle == null) {
_dynamicItemStyle = new MenuItemStyle();
if (IsTrackingViewState) {
((IStateManager)_dynamicItemStyle).TrackViewState();
}
}
return _dynamicItemStyle;
}
}
[
WebCategory("Styles"),
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebSysDescription(SR.Menu_DynamicMenuStyle)
]
public SubMenuStyle DynamicMenuStyle {
get {
if (_dynamicMenuStyle == null) {
_dynamicMenuStyle = new SubMenuStyle();
if (IsTrackingViewState) {
((IStateManager)_dynamicMenuStyle).TrackViewState();
}
}
return _dynamicMenuStyle;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_DynamicPopoutImageUrl)]
public string DynamicPopOutImageUrl {
get {
object s = ViewState["DynamicPopOutImageUrl"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["DynamicPopOutImageUrl"] = value;
}
}
[WebSysDefaultValue(SR.MenuAdapter_Expand)]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_DynamicPopoutImageText)]
public string DynamicPopOutImageTextFormatString {
get {
object s = ViewState["DynamicPopOutImageTextFormatString"];
if (s == null) {
return SR.GetString(SR.MenuAdapter_Expand);
}
return (string)s;
}
set {
ViewState["DynamicPopOutImageTextFormatString"] = value;
}
}
[
WebCategory("Styles"),
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebSysDescription(SR.Menu_DynamicSelectedStyle)
]
public MenuItemStyle DynamicSelectedStyle {
get {
if (_dynamicSelectedStyle == null) {
_dynamicSelectedStyle = new MenuItemStyle();
if (IsTrackingViewState) {
((IStateManager)_dynamicSelectedStyle).TrackViewState();
}
}
return _dynamicSelectedStyle;
}
}
[
Browsable(false),
DefaultValue(null),
PersistenceMode(PersistenceMode.InnerProperty),
TemplateContainer(typeof(MenuItemTemplateContainer)),
WebSysDescription(SR.Menu_DynamicTemplate)
]
public ITemplate DynamicItemTemplate {
get {
return _dynamicTemplate;
}
set {
_dynamicTemplate = value;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_DynamicTopSeparatorImageUrl)]
public string DynamicTopSeparatorImageUrl {
get {
object s = ViewState["DynamicTopSeparatorImageUrl"];
if (s == null) {
return string.Empty;
}
return (string)s;
}
set {
ViewState["DynamicTopSeparatorImageUrl"] = value;
}
}
[
DefaultValue(0),
WebCategory("Appearance"),
WebSysDescription(SR.Menu_DynamicVerticalOffset)
]
public int DynamicVerticalOffset {
get {
object o = ViewState["DynamicVerticalOffset"];
if (o == null) {
return 0;
}
return (int)o;
}
set {
ViewState["DynamicVerticalOffset"] = value;
}
}
/// <devdoc>
/// A cache of urls for the built-in images.
/// </devdoc>
private string[] ImageUrls {
get {
if (_imageUrls == null) {
_imageUrls = new string[ImageUrlsCount];
}
return _imageUrls;
}
}
[DefaultValue(true)]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_IncludeStyleBlock)]
public bool IncludeStyleBlock { get; set; }
internal bool IsNotIE {
get {
return _isNotIE;
}
}
/// <devdoc>
/// Gets the collection of top-level nodes.
/// </devdoc>
[
DefaultValue(null),
Editor("System.Web.UI.Design.WebControls.MenuItemCollectionEditor," + AssemblyRef.SystemDesign, typeof(UITypeEditor)),
PersistenceMode(PersistenceMode.InnerProperty),
MergableProperty(false),
WebSysDescription(SR.Menu_Items)
]
public MenuItemCollection Items {
get {
return RootItem.ChildItems;
}
}
/// <devdoc>
/// Gets and sets whether the text of the items should be wrapped
/// </devdoc>
[DefaultValue(false)]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_ItemWrap)]
public bool ItemWrap {
get {
object o = ViewState["ItemWrap"];
if (o == null) {
return false;
}
return (bool)o;
}
set {
ViewState["ItemWrap"] = value;
}
}
/// <devdoc>
/// Gets the collection of MenuItemStyles corresponding to each level
/// </devdoc>
[
DefaultValue(null),
Editor("System.Web.UI.Design.WebControls.MenuItemStyleCollectionEditor," + AssemblyRef.SystemDesign, typeof(UITypeEditor)),
PersistenceMode(PersistenceMode.InnerProperty),
WebCategory("Styles"),
WebSysDescription(SR.Menu_LevelMenuItemStyles),
]
public MenuItemStyleCollection LevelMenuItemStyles {
get {
if (_levelMenuItemStyles == null) {
_levelMenuItemStyles = new MenuItemStyleCollection();
if (IsTrackingViewState) {
((IStateManager)_levelMenuItemStyles).TrackViewState();
}
}
return _levelMenuItemStyles;
}
}
/// <devdoc>
/// Gets the collection of MenuItemStyles corresponding to the selected item on each level
/// </devdoc>
[
DefaultValue(null),
Editor("System.Web.UI.Design.WebControls.MenuItemStyleCollectionEditor," + AssemblyRef.SystemDesign, typeof(UITypeEditor)),
PersistenceMode(PersistenceMode.InnerProperty),
WebCategory("Styles"),
WebSysDescription(SR.Menu_LevelSelectedStyles),
]
public MenuItemStyleCollection LevelSelectedStyles {
get {
if (_levelSelectedStyles == null) {
_levelSelectedStyles = new MenuItemStyleCollection();
if (IsTrackingViewState) {
((IStateManager)_levelSelectedStyles).TrackViewState();
}
}
return _levelSelectedStyles;
}
}
[
DefaultValue(null),
Editor("System.Web.UI.Design.WebControls.SubMenuStyleCollectionEditor," + AssemblyRef.SystemDesign, typeof(UITypeEditor)),
PersistenceMode(PersistenceMode.InnerProperty),
WebCategory("Styles"),
WebSysDescription(SR.Menu_LevelSubMenuStyles),
]
public SubMenuStyleCollection LevelSubMenuStyles {
get {
if (_levelStyles == null) {
_levelStyles = new SubMenuStyleCollection();
if (IsTrackingViewState) {
((IStateManager)_levelStyles).TrackViewState();
}
}
return _levelStyles;
}
}
internal int MaximumDepth {
get {
if (_maximumDepth > 0) {
return _maximumDepth;
}
_maximumDepth = MaximumDynamicDisplayLevels + StaticDisplayLevels;
if (_maximumDepth < MaximumDynamicDisplayLevels || _maximumDepth < StaticDisplayLevels) {
_maximumDepth = int.MaxValue;
}
return _maximumDepth;
}
}
[WebCategory("Behavior")]
[DefaultValue(3)]
[Themeable(true)]
[WebSysDescription(SR.Menu_MaximumDynamicDisplayLevels)]
public int MaximumDynamicDisplayLevels {
get {
object o = ViewState["MaximumDynamicDisplayLevels"];
if (o == null) {
return 3;
}
return (int)o;
}
set {
if (value < 0) {
throw new ArgumentOutOfRangeException("MaximumDynamicDisplayLevels", SR.GetString(SR.Menu_MaximumDynamicDisplayLevelsInvalid));
}
// Note: we're not testing against the old value here because
// the setter is not the only thing that can change the underlying
// ViewState entry: LoadViewState modifies it directly.
ViewState["MaximumDynamicDisplayLevels"] = value;
// Reset max depth cache
_maximumDepth = 0;
// Rebind if necessary
if (_dataBound) {
_dataBound = false;
PerformDataBinding();
}
}
}
[WebCategory("Layout")]
[DefaultValue(Orientation.Vertical)]
[WebSysDescription(SR.Menu_Orientation)]
public Orientation Orientation {
get {
object o = ViewState["Orientation"];
if (o == null) {
return Orientation.Vertical;
}
return (Orientation)o;
}
set {
ViewState["Orientation"] = value;
}
}
internal PopOutPanel Panel {
get {
if (_panel == null) {
_panel = new PopOutPanel(this, _panelStyle);
if (!DesignMode) {
_panel.Page = Page;
}
}
return _panel;
}
}
[DefaultValue('/')]
[WebSysDescription(SR.Menu_PathSeparator)]
public char PathSeparator {
get {
object o = ViewState["PathSeparator"];
if (o == null) {
return '/';
}
return (char)o;
}
set {
if (value == '\0') {
ViewState["PathSeparator"] = null;
}
else {
ViewState["PathSeparator"] = value;
}
foreach (MenuItem item in Items) {
item.ResetValuePathRecursive();
}
}
}
internal string PopoutImageUrlInternal {
get {
if (_cachedPopOutImageUrl != null) {
return _cachedPopOutImageUrl;
}
_cachedPopOutImageUrl = Page.ClientScript.GetWebResourceUrl(typeof(Menu), ("Menu_Popout.gif"));
return _cachedPopOutImageUrl;
}
}
private MenuRenderer Renderer {
get {
if (_renderer == null) {
switch (RenderingMode) {
case MenuRenderingMode.Table:
_renderer = new MenuRendererClassic(this);
break;
case MenuRenderingMode.List:
_renderer = new MenuRendererStandards(this);
break;
case MenuRenderingMode.Default:
if (RenderingCompatibility < VersionUtil.Framework40) {
_renderer = new MenuRendererClassic(this);
}
else {
_renderer = new MenuRendererStandards(this);
}
break;
default:
Debug.Fail("Unknown MenuRenderingMode.");
break;
}
}
return _renderer;
}
}
[
WebCategory("Layout"),
DefaultValue(MenuRenderingMode.Default),
WebSysDescription(SR.Menu_RenderingMode)
]
public MenuRenderingMode RenderingMode {
get {
return _renderingMode;
}
set {
if (value < MenuRenderingMode.Default || value > MenuRenderingMode.List) {
throw new ArgumentOutOfRangeException("value");
}
if (_renderer != null) {
throw new InvalidOperationException(SR.GetString(SR.Menu_CannotChangeRenderingMode));
}
_renderingMode = value;
}
}
/// <devdoc>
/// The 'virtual' root node of the menu. This menu item is never shown.
/// It is the parent of the "real" root items.
/// </devdoc>
internal MenuItem RootItem {
get {
if (_rootItem == null) {
_rootItem = new MenuItem(this, true);
}
return _rootItem;
}
}
// RootMenuItemStyle is roughly equivalent to ControlStyle.HyperLinkStyle if it existed.
internal Style RootMenuItemStyle {
get {
EnsureRootMenuStyle();
return _rootMenuItemStyle;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_ScrollDownImageUrl)]
public string ScrollDownImageUrl {
get {
object s = ViewState["ScrollDownImageUrl"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["ScrollDownImageUrl"] = value;
}
}
internal string ScrollDownImageUrlInternal {
get {
if (_cachedScrollDownImageUrl != null) {
return _cachedScrollDownImageUrl;
}
_cachedScrollDownImageUrl = Page.ClientScript.GetWebResourceUrl(typeof(Menu), ("Menu_ScrollDown.gif"));
return _cachedScrollDownImageUrl;
}
}
[WebSysDefaultValue(SR.Menu_ScrollDown)]
[Localizable(true)]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_ScrollDownText)]
public string ScrollDownText {
get {
object s = ViewState["ScrollDownText"];
if (s == null) {
return SR.GetString(SR.Menu_ScrollDown);
}
return (string)s;
}
set {
ViewState["ScrollDownText"] = value;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_ScrollUpImageUrl)]
public string ScrollUpImageUrl {
get {
object s = ViewState["ScrollUpImageUrl"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["ScrollUpImageUrl"] = value;
}
}
internal string ScrollUpImageUrlInternal {
get {
if (_cachedScrollUpImageUrl != null) {
return _cachedScrollUpImageUrl;
}
_cachedScrollUpImageUrl = Page.ClientScript.GetWebResourceUrl(typeof(Menu), ("Menu_ScrollUp.gif"));
return _cachedScrollUpImageUrl;
}
}
[WebSysDefaultValue(SR.Menu_ScrollUp)]
[Localizable(true)]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_ScrollUpText)]
public string ScrollUpText {
get {
object s = ViewState["ScrollUpText"];
if (s == null) {
return SR.GetString(SR.Menu_ScrollUp);
}
return (string)s;
}
set {
ViewState["ScrollUpText"] = value;
}
}
/// <devdoc>
/// Gets and sets the Menu's selected node.
/// </devdoc>
[
Browsable(false),
DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden),
]
public MenuItem SelectedItem {
get {
return _selectedItem;
}
}
[Browsable(false)]
[DefaultValue("")]
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
public string SelectedValue {
get {
if (SelectedItem != null) {
return SelectedItem.Value;
}
return String.Empty;
}
}
[WebSysDefaultValue(SR.Menu_SkipLinkTextDefault)]
[Localizable(true)]
[WebCategory("Accessibility")]
[WebSysDescription(SR.WebControl_SkipLinkText)]
public string SkipLinkText {
get {
object s = ViewState["SkipLinkText"];
if (s == null) {
return SR.GetString(SR.Menu_SkipLinkTextDefault);
}
return (string)s;
}
set {
ViewState["SkipLinkText"] = value;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_StaticBottomSeparatorImageUrl)]
public string StaticBottomSeparatorImageUrl {
get {
object s = ViewState["StaticBottomSeparatorImageUrl"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["StaticBottomSeparatorImageUrl"] = value;
}
}
[WebCategory("Behavior")]
[DefaultValue(1)]
[Themeable(true)]
[WebSysDescription(SR.Menu_StaticDisplayLevels)]
public int StaticDisplayLevels {
get {
object o = ViewState["StaticDisplayLevels"];
if (o == null) {
return 1;
}
return (int)o;
}
set {
if (value < 1) {
throw new ArgumentOutOfRangeException("value");
}
// Note: we're not testing against the old value here because
// the setter is not the only thing that can change the underlying
// ViewState entry: LoadViewState modifies it directly.
ViewState["StaticDisplayLevels"] = value;
// Reset max depth cache
_maximumDepth = 0;
// Rebind if necessary
if (_dataBound && !DesignMode) {
_dataBound = false;
PerformDataBinding();
}
}
}
[
DefaultValue(true),
WebCategory("Appearance"),
WebSysDescription(SR.Menu_StaticDisplayPopOutImage)
]
public bool StaticEnableDefaultPopOutImage {
get {
object o = ViewState["StaticEnableDefaultPopOutImage"];
if (o == null) {
return true;
}
return (bool)o;
}
set {
ViewState["StaticEnableDefaultPopOutImage"] = value;
}
}
[
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebCategory("Styles"),
WebSysDescription(SR.Menu_StaticHoverStyle)
]
public Style StaticHoverStyle {
get {
if (_staticHoverStyle == null) {
_staticHoverStyle = new Style();
if (IsTrackingViewState) {
((IStateManager)_staticHoverStyle).TrackViewState();
}
}
return _staticHoverStyle;
}
}
[DefaultValue("")]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_StaticItemFormatString)]
public string StaticItemFormatString {
get {
object s = ViewState["StaticItemFormatString"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["StaticItemFormatString"] = value;
}
}
[
WebCategory("Styles"),
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebSysDescription(SR.Menu_StaticMenuItemStyle)
]
public MenuItemStyle StaticMenuItemStyle {
get {
if (_staticItemStyle == null) {
_staticItemStyle = new MenuItemStyle();
if (IsTrackingViewState) {
((IStateManager)_staticItemStyle).TrackViewState();
}
}
return _staticItemStyle;
}
}
[
WebCategory("Styles"),
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebSysDescription(SR.Menu_StaticMenuStyle)
]
public SubMenuStyle StaticMenuStyle {
get {
if (_staticMenuStyle == null) {
_staticMenuStyle = new SubMenuStyle();
if (IsTrackingViewState) {
((IStateManager)_staticMenuStyle).TrackViewState();
}
}
return _staticMenuStyle;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_StaticPopoutImageUrl)]
public string StaticPopOutImageUrl {
get {
object s = ViewState["StaticPopOutImageUrl"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["StaticPopOutImageUrl"] = value;
}
}
[WebSysDefaultValue(SR.MenuAdapter_Expand)]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_StaticPopoutImageText)]
public string StaticPopOutImageTextFormatString {
get {
object s = ViewState["StaticPopOutImageTextFormatString"];
if (s == null) {
return SR.GetString(SR.MenuAdapter_Expand);
}
return (string)s;
}
set {
ViewState["StaticPopOutImageTextFormatString"] = value;
}
}
[
WebCategory("Styles"),
DefaultValue(null),
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
NotifyParentProperty(true),
PersistenceMode(PersistenceMode.InnerProperty),
WebSysDescription(SR.Menu_StaticSelectedStyle)
]
public MenuItemStyle StaticSelectedStyle {
get {
if (_staticSelectedStyle == null) {
_staticSelectedStyle = new MenuItemStyle();
if (IsTrackingViewState) {
((IStateManager)_staticSelectedStyle).TrackViewState();
}
}
return _staticSelectedStyle;
}
}
[WebCategory("Appearance")]
[DefaultValue(typeof(Unit), "")]
[Themeable(true)]
[WebSysDescription(SR.Menu_StaticSubMenuIndent)]
public Unit StaticSubMenuIndent {
get {
object o = ViewState["StaticSubMenuIndent"];
if (o == null) {
return Unit.Empty;
}
return (Unit)o;
}
set {
if (value.Value < 0) {
throw new ArgumentOutOfRangeException("value");
}
ViewState["StaticSubMenuIndent"] = value;
}
}
[
Browsable(false),
DefaultValue(null),
PersistenceMode(PersistenceMode.InnerProperty),
TemplateContainer(typeof(MenuItemTemplateContainer)),
WebSysDescription(SR.Menu_StaticTemplate)
]
public ITemplate StaticItemTemplate {
get {
return _staticTemplate;
}
set {
_staticTemplate = value;
}
}
[DefaultValue("")]
[Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor))]
[UrlProperty()]
[WebCategory("Appearance")]
[WebSysDescription(SR.Menu_StaticTopSeparatorImageUrl)]
public string StaticTopSeparatorImageUrl {
get {
object s = ViewState["StaticTopSeparatorImageUrl"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["StaticTopSeparatorImageUrl"] = value;
}
}
/// <devdoc>
/// Gets and sets the target window that the MenuItems will browse to if selected
/// </devdoc>
[DefaultValue("")]
[WebSysDescription(SR.MenuItem_Target)]
public string Target {
get {
object s = ViewState["Target"];
if (s == null) {
return String.Empty;
}
return (string)s;
}
set {
ViewState["Target"] = value;
}
}
protected override HtmlTextWriterTag TagKey {
get {
return HtmlTextWriterTag.Table;
}
}
#region events
/// <devdoc>
/// Triggered when an item has been clicked.
/// </devdoc>
[WebCategory("Behavior")]
[WebSysDescription(SR.Menu_MenuItemClick)]
public event MenuEventHandler MenuItemClick {
add {
Events.AddHandler(_menuItemClickedEvent, value);
}
remove {
Events.RemoveHandler(_menuItemClickedEvent, value);
}
}
/// <devdoc>
/// Triggered when a MenuItem has been databound.
/// </devdoc>
[WebCategory("Behavior")]
[WebSysDescription(SR.Menu_MenuItemDataBound)]
public event MenuEventHandler MenuItemDataBound {
add {
Events.AddHandler(_menuItemDataBoundEvent, value);
}
remove {
Events.RemoveHandler(_menuItemDataBoundEvent, value);
}
}
#endregion
protected override void AddAttributesToRender(HtmlTextWriter writer) {
VerifyRenderingInServerForm();
string oldAccessKey = AccessKey;
try {
AccessKey = String.Empty;
base.AddAttributesToRender(writer);
}
finally {
AccessKey = oldAccessKey;
}
}
// returns true if the style contains a class name
private static bool AppendCssClassName(StringBuilder builder, MenuItemStyle style, bool hyperlink) {
bool containsClassName = false;
if (style != null) {
// We have to merge with any CssClass specified on the Style itself
if (style.CssClass.Length != 0) {
builder.Append(style.CssClass);
builder.Append(' ');
containsClassName = true;
}
string className = (hyperlink ?
style.HyperLinkStyle.RegisteredCssClass :
style.RegisteredCssClass);
if (className.Length > 0) {
builder.Append(className);
builder.Append(' ');
}
}
return containsClassName;
}
private static void AppendMenuCssClassName(StringBuilder builder, SubMenuStyle style) {
if (style != null) {
// We have to merge with any CssClass specified on the Style itself
if (style.CssClass.Length != 0) {
builder.Append(style.CssClass);
builder.Append(' ');
}
string className = style.RegisteredCssClass;
if (className.Length > 0) {
builder.Append(className);
builder.Append(' ');
}
}
}
private static T CacheGetItem<T>(List<T> cacheList, int index) where T : class {
Debug.Assert(cacheList != null);
if (index < cacheList.Count) return cacheList[index];
return null;
}
private static void CacheSetItem<T>(List<T> cacheList, int index, T item) where T : class {
if (cacheList.Count > index) {
cacheList[index] = item;
}
else {
for (int i = cacheList.Count; i < index; i++) {
cacheList.Add(null);
}
cacheList.Add(item);
}
}
protected internal override void CreateChildControls() {
Controls.Clear();
if ((StaticItemTemplate != null) || (DynamicItemTemplate != null)) {
if (RequiresDataBinding &&
(!String.IsNullOrEmpty(DataSourceID) || DataSource != null)) {
EnsureDataBound();
}
else {
// Creating child controls from the Items tree
CreateChildControlsFromItems(/* dataBinding */ false);
ClearChildViewState();
}
}
}
private void CreateChildControlsFromItems(bool dataBinding) {
if (StaticItemTemplate != null || DynamicItemTemplate != null) {
int childPosition = 0;
foreach (MenuItem child in Items) {
CreateTemplatedControls(StaticItemTemplate, child, childPosition++, 0, dataBinding);
}
}
}
/// <devdoc>
/// Creates a menu node index that is unique for this menu
/// </devdoc>
internal int CreateItemIndex() {
return _nodeIndex++;
}
private void CreateTemplatedControls(ITemplate template, MenuItem item, int position, int depth, bool dataBinding) {
if (template != null) {
MenuItemTemplateContainer container = new MenuItemTemplateContainer(position, item);
item.Container = (MenuItemTemplateContainer)container;
template.InstantiateIn(container);
Controls.Add(container);
if (dataBinding) {
container.DataBind();
}
}
int childPosition = 0;
foreach (MenuItem child in item.ChildItems) {
int nextDepth = depth + 1;
if (template == DynamicItemTemplate) {
CreateTemplatedControls(DynamicItemTemplate, child, childPosition++, nextDepth, dataBinding);
}
else {
if (nextDepth < StaticDisplayLevels) {
CreateTemplatedControls(template, child, childPosition++, nextDepth, dataBinding);
}
else if (DynamicItemTemplate != null) {
CreateTemplatedControls(DynamicItemTemplate, child, childPosition++, nextDepth, dataBinding);
}
}
}
}
/// Data bound controls should override PerformDataBinding instead
/// of DataBind. If DataBind if overridden, the OnDataBinding and OnDataBound events will
/// fire in the wrong order. However, for backwards compat on ListControl and AdRotator, we
/// can't seal this method. It is sealed on all new BaseDataBoundControl-derived controls.
public override sealed void DataBind() {
base.DataBind();
}
/// <devdoc>
/// Databinds the specified node to the datasource
/// </devdoc>
private void DataBindItem(MenuItem item) {
HierarchicalDataSourceView view = GetData(item.DataPath);
// Do nothing if no datasource was set
if (!IsBoundUsingDataSourceID && (DataSource == null)) {
return;
}
if (view == null) {
throw new InvalidOperationException(SR.GetString(SR.Menu_DataSourceReturnedNullView, ID));
}
IHierarchicalEnumerable enumerable = view.Select();
item.ChildItems.Clear();
if (enumerable != null) {
// If we're bound to a SiteMapDataSource, automatically select the node
if (IsBoundUsingDataSourceID) {
SiteMapDataSource siteMapDataSource = GetDataSource() as SiteMapDataSource;
if (siteMapDataSource != null) {
SiteMapNode currentNode = siteMapDataSource.Provider.CurrentNode;
if (currentNode != null) {
_currentSiteMapNodeUrl = currentNode.Url;
}
}
}
try {
DataBindRecursive(item, enumerable);
}
finally {
_currentSiteMapNodeUrl = null;
}
}
}
/// <devdoc>
/// Databinds recursively, using the Menu's Bindings collection, until there is no more data.
/// </devdoc>
private void DataBindRecursive(MenuItem node, IHierarchicalEnumerable enumerable) {
// Since we are binding children, get the level below the current node's depth
int depth = node.Depth + 1;
// Don't databind beyond the maximum specified depth
if ((MaximumDynamicDisplayLevels != -1) && (depth >= MaximumDepth)) {
return;
}
foreach (object item in enumerable) {
IHierarchyData data = enumerable.GetHierarchyData(item);
string text = null;
string value = null;
string navigateUrl = String.Empty;
string imageUrl = String.Empty;
string popOutImageUrl = String.Empty;
string separatorImageUrl = String.Empty;
string target = String.Empty;
bool enabled = true;
bool enabledSet = false;
bool selectable = true;
bool selectableSet = false;
string toolTip = String.Empty;
string dataMember = String.Empty;
dataMember = data.Type;
MenuItemBinding level = DataBindings.GetBinding(dataMember, depth);
if (level != null) {
PropertyDescriptorCollection props = TypeDescriptor.GetProperties(item);
// Bind Text, using the static value if necessary
string textField = level.TextField;
if (textField.Length > 0) {
PropertyDescriptor desc = props.Find(textField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
if (level.FormatString.Length > 0) {
text = string.Format(CultureInfo.CurrentCulture, level.FormatString, objData);
}
else {
text = objData.ToString();
}
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, textField, "TextField"));
}
}
if (String.IsNullOrEmpty(text) && !String.IsNullOrEmpty(level.Text)) {
text = level.Text;
}
// Bind Value, using the static value if necessary
string valueField = level.ValueField;
if (valueField.Length > 0) {
PropertyDescriptor desc = props.Find(valueField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
value = objData.ToString();
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, valueField, "ValueField"));
}
}
if (String.IsNullOrEmpty(value) && !String.IsNullOrEmpty(level.Value)) {
value = level.Value;
}
// Bind Target, using the static value if necessary
string targetField = level.TargetField;
if (targetField.Length > 0) {
PropertyDescriptor desc = props.Find(targetField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
target = objData.ToString();
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, targetField, "TargetField"));
}
}
if (String.IsNullOrEmpty(target)) {
target = level.Target;
}
// Bind ImageUrl, using the static value if necessary
string imageUrlField = level.ImageUrlField;
if (imageUrlField.Length > 0) {
PropertyDescriptor desc = props.Find(imageUrlField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
imageUrl = objData.ToString();
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, imageUrlField, "ImageUrlField"));
}
}
if (String.IsNullOrEmpty(imageUrl)) {
imageUrl = level.ImageUrl;
}
// Bind NavigateUrl, using the static value if necessary
string navigateUrlField = level.NavigateUrlField;
if (navigateUrlField.Length > 0) {
PropertyDescriptor desc = props.Find(navigateUrlField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
navigateUrl = objData.ToString();
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, navigateUrlField, "NavigateUrlField"));
}
}
if (String.IsNullOrEmpty(navigateUrl)) {
navigateUrl = level.NavigateUrl;
}
// Bind PopOutImageUrl, using the static value if necessary
string popOutImageUrlField = level.PopOutImageUrlField;
if (popOutImageUrlField.Length > 0) {
PropertyDescriptor desc = props.Find(popOutImageUrlField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
popOutImageUrl = objData.ToString();
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, popOutImageUrlField, "PopOutImageUrlField"));
}
}
if (String.IsNullOrEmpty(popOutImageUrl)) {
popOutImageUrl = level.PopOutImageUrl;
}
// Bind SeperatorImageUrl, using the static value if necessary
string separatorImageUrlField = level.SeparatorImageUrlField;
if (separatorImageUrlField.Length > 0) {
PropertyDescriptor desc = props.Find(separatorImageUrlField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
separatorImageUrl = objData.ToString();
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, separatorImageUrlField, "SeparatorImageUrlField"));
}
}
if (String.IsNullOrEmpty(separatorImageUrl)) {
separatorImageUrl = level.SeparatorImageUrl;
}
// Bind ToolTip, using the static value if necessary
string toolTipField = level.ToolTipField;
if (toolTipField.Length > 0) {
PropertyDescriptor desc = props.Find(toolTipField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
toolTip = objData.ToString();
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, toolTipField, "ToolTipField"));
}
}
if (String.IsNullOrEmpty(toolTip)) {
toolTip = level.ToolTip;
}
// Bind Enabled, using the static value if necessary
string enabledField = level.EnabledField;
if (enabledField.Length > 0) {
PropertyDescriptor desc = props.Find(enabledField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
if (objData is bool) {
enabled = (bool)objData;
enabledSet = true;
}
else if (bool.TryParse(objData.ToString(), out enabled)) {
enabledSet = true;
}
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, enabledField, "EnabledField"));
}
}
if (!enabledSet) {
enabled = level.Enabled;
}
// Bind Selectable, using the static value if necessary
string selectableField = level.SelectableField;
if (selectableField.Length > 0) {
PropertyDescriptor desc = props.Find(selectableField, true);
if (desc != null) {
object objData = desc.GetValue(item);
if (objData != null) {
if (objData is bool) {
selectable = (bool)objData;
selectableSet = true;
}
else if (bool.TryParse(objData.ToString(), out selectable)) {
selectableSet = true;
}
}
}
else {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDataBinding, selectableField, "SelectableField"));
}
}
if (!selectableSet) {
selectable = level.Selectable;
}
}
else if (item is INavigateUIData) {
INavigateUIData navigateUIData = (INavigateUIData)item;
text = navigateUIData.Name;
value = navigateUIData.Value;
navigateUrl = navigateUIData.NavigateUrl;
if (String.IsNullOrEmpty(navigateUrl)) {
selectable = false;
}
toolTip = navigateUIData.Description;
}
if (text == null) {
text = item.ToString();
}
MenuItem newItem = null;
// Allow String.Empty for the text, but not null
if ((text != null) || (value != null)) {
newItem = new MenuItem(text, value, imageUrl, navigateUrl, target);
}
if (newItem != null) {
if (toolTip.Length > 0) {
newItem.ToolTip = toolTip;
}
if (popOutImageUrl.Length > 0) {
newItem.PopOutImageUrl = popOutImageUrl;
}
if (separatorImageUrl.Length > 0) {
newItem.SeparatorImageUrl = separatorImageUrl;
}
newItem.Enabled = enabled;
newItem.Selectable = selectable;
newItem.SetDataPath(data.Path);
newItem.SetDataBound(true);
node.ChildItems.Add(newItem);
if (String.Equals(data.Path, _currentSiteMapNodeUrl, StringComparison.OrdinalIgnoreCase)) {
newItem.Selected = true;
}
// Make sure we call user code if they've hooked the populate event
newItem.SetDataItem(data.Item);
OnMenuItemDataBound(new MenuEventArgs(newItem));
newItem.SetDataItem(null);
if (data.HasChildren && (depth < MaximumDepth)) {
IHierarchicalEnumerable newEnumerable = data.GetChildren();
if (newEnumerable != null) {
DataBindRecursive(newItem, newEnumerable);
}
}
}
}
}
protected override void EnsureDataBound() {
base.EnsureDataBound();
if (!_subControlsDataBound) {
foreach (Control ctrl in Controls) {
ctrl.DataBind();
}
_subControlsDataBound = true;
}
}
public MenuItem FindItem(string valuePath) {
if (valuePath == null) {
return null;
}
return Items.FindItem(valuePath.Split(PathSeparator), 0);
}
internal string GetCssClassName(MenuItem item, bool hyperLink) {
bool discarded;
return GetCssClassName(item, hyperLink, out discarded);
}
internal string GetCssClassName(MenuItem item, bool hyperlink, out bool containsClassName) {
if (item == null) {
throw new ArgumentNullException("item");
}
containsClassName = false;
int depth = item.Depth;
string baseClassName = CacheGetItem<string>(
hyperlink ? CachedMenuItemHyperLinkClassNames : CachedMenuItemClassNames,
depth);
if (CachedLevelsContainingCssClass.Contains(depth)) {
containsClassName = true;
}
if (!item.Selected && (baseClassName != null)) {
return baseClassName;
}
StringBuilder builder = new StringBuilder();
if (baseClassName != null) {
if (!item.Selected) return baseClassName;
builder.Append(baseClassName);
builder.Append(' ');
}
else {
// No cached style, so build it
if (hyperlink) {
builder.Append(RootMenuItemStyle.RegisteredCssClass);
builder.Append(' ');
}
if (depth < StaticDisplayLevels) {
containsClassName |= AppendCssClassName(builder, _staticItemStyle, hyperlink);
}
else {
containsClassName |= AppendCssClassName(builder, _dynamicItemStyle, hyperlink);
}
if ((depth < LevelMenuItemStyles.Count) && (LevelMenuItemStyles[depth] != null)) {
containsClassName |= AppendCssClassName(builder, LevelMenuItemStyles[depth], hyperlink);
}
baseClassName = builder.ToString().Trim();
CacheSetItem<string>(
hyperlink ? CachedMenuItemHyperLinkClassNames : CachedMenuItemClassNames,
depth,
baseClassName);
if (containsClassName && !CachedLevelsContainingCssClass.Contains(depth)) {
CachedLevelsContainingCssClass.Add(depth);
}
}
if (item.Selected) {
if (depth < StaticDisplayLevels) {
containsClassName |= AppendCssClassName(builder, _staticSelectedStyle, hyperlink);
}
else {
containsClassName |= AppendCssClassName(builder, _dynamicSelectedStyle, hyperlink);
}
if ((depth < LevelSelectedStyles.Count) && (LevelSelectedStyles[depth] != null)) {
MenuItemStyle style = LevelSelectedStyles[depth];
containsClassName |= AppendCssClassName(builder, style, hyperlink);
}
return builder.ToString().Trim();
}
return baseClassName;
}
/// <devdoc>Used by GetDesignModeState to find the first dynamic submenu in Items</devdoc>
private MenuItem GetOneDynamicItem(MenuItem item) {
if (item.Depth >= StaticDisplayLevels) {
return item;
}
for (int i = 0; i < item.ChildItems.Count; i++) {
MenuItem result = GetOneDynamicItem(item.ChildItems[i]);
if (result != null) {
return result;
}
}
return null;
}
/// <internalonly/>
/// <devdoc>
/// The designer can get the static and the dynamic mode html by using the
/// _getDesignTimeStaticHtml and _getDesignTimeDynamicHtml defined string keys
/// of the dictionary.
/// </devdoc>
[SecurityPermission(SecurityAction.Demand, Unrestricted = true)]
protected override IDictionary GetDesignModeState() {
IDictionary dictionary = base.GetDesignModeState();
Debug.Assert(dictionary != null && DesignMode);
CreateChildControls();
foreach (Control c in Controls) {
c.DataBind();
}
// Create the html for the static part
using (StringWriter staticHtmlBuilder = new StringWriter(CultureInfo.CurrentCulture)) {
using (HtmlTextWriter htmlWriter = GetDesignTimeWriter(staticHtmlBuilder)) {
Renderer.RenderBeginTag(htmlWriter, true);
Renderer.RenderContents(htmlWriter, true);
Renderer.RenderEndTag(htmlWriter, true);
dictionary[_getDesignTimeStaticHtml] = staticHtmlBuilder.ToString();
}
}
// Remember the static depth so we can lower it if necessary to make it faster and avoid overflows
int oldStaticDisplayLevels = StaticDisplayLevels;
try {
// Find a dynamic sub-menu
MenuItem dynamicSubMenu = GetOneDynamicItem(RootItem);
if (dynamicSubMenu == null) {
// We need to forge a whole new dynamic submenu
// First lower the static display levels
_dataBound = false;
StaticDisplayLevels = 1;
dynamicSubMenu = new MenuItem();
dynamicSubMenu.SetDepth(0);
dynamicSubMenu.SetOwner(this);
// Create a single dynamic submenu, with one submenu
string dummyText = SR.GetString(SR.Menu_DesignTimeDummyItemText);
for (int i = 0; i < 5; i++) {
MenuItem newItem = new MenuItem(dummyText);
if (DynamicItemTemplate != null) {
MenuItemTemplateContainer container = new MenuItemTemplateContainer(i, newItem);
newItem.Container = container;
DynamicItemTemplate.InstantiateIn(container);
container.Site = this.Site;
container.DataBind();
}
dynamicSubMenu.ChildItems.Add(newItem);
}
dynamicSubMenu.ChildItems[1].ChildItems.Add(new MenuItem());
// Delete cached styles to ensure consistency
_cachedLevelsContainingCssClass = null;
_cachedMenuItemStyles = null;
_cachedSubMenuStyles = null;
_cachedMenuItemClassNames = null;
_cachedMenuItemHyperLinkClassNames = null;
_cachedSubMenuClassNames = null;
}
else {
dynamicSubMenu = dynamicSubMenu.Parent;
}
// Create the html for the dynamic part
using (StringWriter dynamicHtmlBuilder = new StringWriter(CultureInfo.CurrentCulture)) {
using (HtmlTextWriter htmlWriter = GetDesignTimeWriter(dynamicHtmlBuilder)) {
// Render the control's position on the outer table
Attributes.AddAttributes(htmlWriter);
// Rendering a table around the div so that the designer sees it as a block
htmlWriter.RenderBeginTag(HtmlTextWriterTag.Table);
htmlWriter.RenderBeginTag(HtmlTextWriterTag.Tr);
htmlWriter.RenderBeginTag(HtmlTextWriterTag.Td);
dynamicSubMenu.Render(htmlWriter, true, false, false);
htmlWriter.RenderEndTag();
htmlWriter.RenderEndTag();
htmlWriter.RenderEndTag();
dictionary[_getDesignTimeDynamicHtml] = dynamicHtmlBuilder.ToString();
}
}
}
finally {
if (StaticDisplayLevels != oldStaticDisplayLevels) {
StaticDisplayLevels = oldStaticDisplayLevels;
}
}
return dictionary;
}
private HtmlTextWriter GetDesignTimeWriter(StringWriter stringWriter) {
if (_designTimeTextWriterType == null) {
return new HtmlTextWriter(stringWriter);
}
else {
Debug.Assert(_designTimeTextWriterType.IsSubclassOf(typeof(HtmlTextWriter)));
ConstructorInfo constructor = _designTimeTextWriterType.GetConstructor(new Type[] { typeof(TextWriter) });
if (constructor == null) {
return new HtmlTextWriter(stringWriter);
}
return (HtmlTextWriter)(constructor.Invoke(new object[] { stringWriter }));
}
}
/// <devdoc>
/// Gets the URL for the specified image, properly pathing the image filename depending on which image it is
/// </devdoc>
internal string GetImageUrl(int index) {
if (ImageUrls[index] == null) {
switch (index) {
case ScrollUpImageIndex:
ImageUrls[index] = ScrollUpImageUrlInternal;
break;
case ScrollDownImageIndex:
ImageUrls[index] = ScrollDownImageUrlInternal;
break;
case PopOutImageIndex:
ImageUrls[index] = PopoutImageUrlInternal;
break;
}
ImageUrls[index] = ResolveClientUrl(ImageUrls[index]);
}
return ImageUrls[index];
}
internal MenuItemStyle GetMenuItemStyle(MenuItem item) {
if (item == null) {
throw new ArgumentNullException("item");
}
int depth = item.Depth;
MenuItemStyle typedStyle = CacheGetItem<MenuItemStyle>(CachedMenuItemStyles, depth);
if (!item.Selected && typedStyle != null) return typedStyle;
if (typedStyle == null) {
typedStyle = new MenuItemStyle();
typedStyle.CopyFrom(RootMenuItemStyle);
if (depth < StaticDisplayLevels) {
if (_staticItemStyle != null) {
TreeView.GetMergedStyle(typedStyle, _staticItemStyle);
}
}
else if (depth >= StaticDisplayLevels && _dynamicItemStyle != null) {
TreeView.GetMergedStyle(typedStyle, _dynamicItemStyle);
}
if ((depth < LevelMenuItemStyles.Count) && (LevelMenuItemStyles[depth] != null)) {
TreeView.GetMergedStyle(typedStyle, LevelMenuItemStyles[depth]);
}
CacheSetItem<MenuItemStyle>(CachedMenuItemStyles, depth, typedStyle);
}
if (item.Selected) {
MenuItemStyle selectedStyle = new MenuItemStyle();
selectedStyle.CopyFrom(typedStyle);
if (depth < StaticDisplayLevels) {
if (_staticSelectedStyle != null) {
TreeView.GetMergedStyle(selectedStyle, _staticSelectedStyle);
}
}
else if (depth >= StaticDisplayLevels && _dynamicSelectedStyle != null) {
TreeView.GetMergedStyle(selectedStyle, _dynamicSelectedStyle);
}
if (depth < LevelSelectedStyles.Count && LevelSelectedStyles[depth] != null) {
TreeView.GetMergedStyle(selectedStyle, LevelSelectedStyles[depth]);
}
return selectedStyle;
}
return typedStyle;
}
internal string GetSubMenuCssClassName(MenuItem item) {
if (item == null) {
throw new ArgumentNullException("item");
}
int nextDepth = item.Depth + 1;
string baseClassName = CacheGetItem<string>(CachedSubMenuClassNames, nextDepth);
if (baseClassName != null) return baseClassName;
StringBuilder builder = new StringBuilder();
if (nextDepth < StaticDisplayLevels) {
AppendMenuCssClassName(builder, _staticMenuStyle);
}
else {
SubMenuStyle subMenuStyle = _panelStyle as SubMenuStyle;
if (subMenuStyle != null) {
AppendMenuCssClassName(builder, subMenuStyle);
}
AppendMenuCssClassName(builder, _dynamicMenuStyle);
}
if ((nextDepth < LevelSubMenuStyles.Count) && (LevelSubMenuStyles[nextDepth] != null)) {
SubMenuStyle style = LevelSubMenuStyles[nextDepth] as SubMenuStyle;
AppendMenuCssClassName(builder, style);
}
baseClassName = builder.ToString().Trim();
CacheSetItem<string>(CachedSubMenuClassNames, nextDepth, baseClassName);
return baseClassName;
}
internal SubMenuStyle GetSubMenuStyle(MenuItem item) {
if (item == null) {
throw new ArgumentNullException("item");
}
int nextDepth = item.Depth + 1;
SubMenuStyle subMenuStyle = CacheGetItem<SubMenuStyle>(CachedSubMenuStyles, nextDepth);
if (subMenuStyle != null) return subMenuStyle;
int staticDisplayLevels = StaticDisplayLevels;
if (nextDepth >= staticDisplayLevels && !DesignMode) {
subMenuStyle = new PopOutPanel.PopOutPanelStyle(Panel);
}
else {
subMenuStyle = new SubMenuStyle();
}
if (nextDepth < staticDisplayLevels) {
if (_staticMenuStyle != null) {
subMenuStyle.CopyFrom(_staticMenuStyle);
}
}
else if (nextDepth >= staticDisplayLevels && _dynamicMenuStyle != null) {
subMenuStyle.CopyFrom(_dynamicMenuStyle);
}
if (_levelStyles != null &&
_levelStyles.Count > nextDepth &&
_levelStyles[nextDepth] != null) {
TreeView.GetMergedStyle(subMenuStyle, _levelStyles[nextDepth]);
}
CacheSetItem<SubMenuStyle>(CachedSubMenuStyles, nextDepth, subMenuStyle);
return subMenuStyle;
}
internal void EnsureRootMenuStyle() {
if (_rootMenuItemStyle == null) {
_rootMenuItemStyle = new Style();
_rootMenuItemStyle.Font.CopyFrom(Font);
if (!ForeColor.IsEmpty) {
_rootMenuItemStyle.ForeColor = ForeColor;
}
// Not defaulting to black anymore for not entirely satisfying but reasonable reasons (VSWhidbey 356729)
if (!ControlStyle.IsSet(System.Web.UI.WebControls.Style.PROP_FONT_UNDERLINE)) {
_rootMenuItemStyle.Font.Underline = false;
}
}
}
protected internal override void LoadControlState(object savedState) {
Pair state = savedState as Pair;
if (state == null) {
base.LoadControlState(savedState);
return;
}
base.LoadControlState(state.First);
_selectedItem = null;
if (state.Second != null) {
string path = state.Second as string;
if (path != null) {
_selectedItem = Items.FindItem(path.Split(TreeView.InternalPathSeparator), 0);
}
}
}
protected override void LoadViewState(object state) {
if (state != null) {
object[] savedState = (object[])state;
if (savedState[1] != null) {
((IStateManager)StaticMenuItemStyle).LoadViewState(savedState[1]);
}
if (savedState[2] != null) {
((IStateManager)StaticSelectedStyle).LoadViewState(savedState[2]);
}
if (savedState[3] != null) {
((IStateManager)StaticHoverStyle).LoadViewState(savedState[3]);
}
if (savedState[4] != null) {
((IStateManager)StaticMenuStyle).LoadViewState(savedState[4]);
}
if (savedState[5] != null) {
((IStateManager)DynamicMenuItemStyle).LoadViewState(savedState[5]);
}
if (savedState[6] != null) {
((IStateManager)DynamicSelectedStyle).LoadViewState(savedState[6]);
}
if (savedState[7] != null) {
((IStateManager)DynamicHoverStyle).LoadViewState(savedState[7]);
}
if (savedState[8] != null) {
((IStateManager)DynamicMenuStyle).LoadViewState(savedState[8]);
}
if (savedState[9] != null) {
((IStateManager)LevelMenuItemStyles).LoadViewState(savedState[9]);
}
if (savedState[10] != null) {
((IStateManager)LevelSelectedStyles).LoadViewState(savedState[10]);
}
if (savedState[11] != null) {
((IStateManager)LevelSubMenuStyles).LoadViewState(savedState[11]);
}
if (savedState[12] != null) {
((IStateManager)Items).LoadViewState(savedState[12]);
if (!String.IsNullOrEmpty(DataSourceID) || DataSource != null) {
_dataBound = true;
}
}
// Restore the core viewstate last because some property changes here could
// necessitate item rebinding (for example, MaximumDynamicDisplayLevels)
if (savedState[0] != null) {
base.LoadViewState(savedState[0]);
}
}
}
protected override bool OnBubbleEvent(object source, EventArgs e) {
MenuEventArgs me = e as MenuEventArgs;
if (me != null && StringUtil.EqualsIgnoreCase(me.CommandName, MenuItemClickCommandName)) {
// Do not take any postback into account if the menu is disabled.
if (!IsEnabled) return true;
OnMenuItemClick(me);
if (AdapterInternal != null) {
MenuAdapter menuAdapter = AdapterInternal as MenuAdapter;
if (menuAdapter != null) {
MenuItem mi = me.Item;
// Need to tell the adapter about bubbled click events
// for the templated case if the item has children
if (mi != null &&
mi.ChildItems.Count > 0 &&
mi.Depth + 1 >= StaticDisplayLevels) {
menuAdapter.SetPath(me.Item.InternalValuePath);
}
}
}
RaiseBubbleEvent(this, e);
return true;
}
if (e is CommandEventArgs) {
RaiseBubbleEvent(this, e);
return true;
}
return false;
}
protected override void OnDataBinding(EventArgs e) {
EnsureChildControls();
base.OnDataBinding(e);
}
protected internal override void OnInit(EventArgs e) {
base.OnInit(e);
Page.RegisterRequiresControlState(this);
}
protected virtual void OnMenuItemClick(MenuEventArgs e) {
SetSelectedItem(e.Item);
MenuEventHandler handler = (MenuEventHandler)Events[_menuItemClickedEvent];
if (handler != null) {
handler(this, e);
}
}
/// <devdoc>
/// Raises the MenuItemDataBound event
/// </devdoc>
protected virtual void OnMenuItemDataBound(MenuEventArgs e) {
MenuEventHandler handler = (MenuEventHandler)Events[_menuItemDataBoundEvent];
if (handler != null) {
handler(this, e);
}
}
/// <devdoc>
/// Overridden to register for postback, and if client script is enabled, renders out
/// the necessary script and hidden field to function.
/// </devdoc>
protected internal override void OnPreRender(EventArgs e) {
base.OnPreRender(e);
if (Items.Count > 0) {
Renderer.PreRender(IsEnabled);
}
}
/// <devdoc>
/// Overridden to register for postback, and if client script is enabled, renders out
/// the necessary script and hidden field to function.
/// </devdoc>
internal void OnPreRender(EventArgs e, bool registerScript) {
base.OnPreRender(e);
if (Items.Count > 0) {
Renderer.PreRender(registerScript);
}
}
/// <devdoc>
/// Overridden to create all the items based on the datasource provided
/// </devdoc>
protected internal override void PerformDataBinding() {
base.PerformDataBinding();
DataBindItem(RootItem);
if (!DesignMode && _dataBound &&
String.IsNullOrEmpty(DataSourceID) && DataSource == null) {
Items.Clear();
Controls.Clear();
ClearChildViewState();
TrackViewState();
ChildControlsCreated = true;
return;
}
if (!String.IsNullOrEmpty(DataSourceID) ||
DataSource != null) {
Controls.Clear();
ClearChildState();
TrackViewState();
CreateChildControlsFromItems(true);
ChildControlsCreated = true;
_dataBound = true;
}
else if (!_subControlsDataBound) {
foreach (Control ctrl in Controls) {
ctrl.DataBind();
}
}
_subControlsDataBound = true;
}
protected internal override void Render(HtmlTextWriter writer) {
VerifyRenderingInServerForm();
if (Items.Count > 0) {
Renderer.RenderBeginTag(writer, false);
Renderer.RenderContents(writer, false);
Renderer.RenderEndTag(writer, false);
}
}
public override void RenderBeginTag(HtmlTextWriter writer) {
Renderer.RenderBeginTag(writer, false);
}
protected internal override void RenderContents(HtmlTextWriter writer) {
Renderer.RenderContents(writer, false);
}
public override void RenderEndTag(HtmlTextWriter writer) {
Renderer.RenderEndTag(writer, false);
}
internal void ResetCachedStyles() {
// Reset all these cached values so things can pick up changes in the designer
if (_dynamicItemStyle != null) {
_dynamicItemStyle.ResetCachedStyles();
}
if (_staticItemStyle != null) {
_staticItemStyle.ResetCachedStyles();
}
if (_dynamicSelectedStyle != null) {
_dynamicSelectedStyle.ResetCachedStyles();
}
if (_staticSelectedStyle != null) {
_staticSelectedStyle.ResetCachedStyles();
}
if (_staticHoverStyle != null) {
_staticHoverHyperLinkStyle = new HyperLinkStyle(_staticHoverStyle);
}
if (_dynamicHoverStyle != null) {
_dynamicHoverHyperLinkStyle = new HyperLinkStyle(_dynamicHoverStyle);
}
foreach (MenuItemStyle style in LevelMenuItemStyles) {
style.ResetCachedStyles();
}
foreach (MenuItemStyle style in LevelSelectedStyles) {
style.ResetCachedStyles();
}
if (_imageUrls != null) {
for (int i = 0; i < _imageUrls.Length; i++) {
_imageUrls[i] = null;
}
}
_cachedPopOutImageUrl = null;
_cachedScrollDownImageUrl = null;
_cachedScrollUpImageUrl = null;
_cachedLevelsContainingCssClass = null;
_cachedMenuItemClassNames = null;
_cachedMenuItemHyperLinkClassNames = null;
_cachedMenuItemStyles = null;
_cachedSubMenuClassNames = null;
_cachedSubMenuStyles = null;
}
protected internal override object SaveControlState() {
object baseState = base.SaveControlState();
if (_selectedItem != null) {
return new Pair(baseState, _selectedItem.InternalValuePath);
}
else {
return baseState;
}
}
protected override object SaveViewState() {
object[] state = new object[13];
state[0] = base.SaveViewState();
bool hasViewState = (state[0] != null);
if (_staticItemStyle != null) {
state[1] = ((IStateManager)_staticItemStyle).SaveViewState();
hasViewState |= (state[1] != null);
}
if (_staticSelectedStyle != null) {
state[2] = ((IStateManager)_staticSelectedStyle).SaveViewState();
hasViewState |= (state[2] != null);
}
if (_staticHoverStyle != null) {
state[3] = ((IStateManager)_staticHoverStyle).SaveViewState();
hasViewState |= (state[3] != null);
}
if (_staticMenuStyle != null) {
state[4] = ((IStateManager)_staticMenuStyle).SaveViewState();
hasViewState |= (state[4] != null);
}
if (_dynamicItemStyle != null) {
state[5] = ((IStateManager)_dynamicItemStyle).SaveViewState();
hasViewState |= (state[5] != null);
}
if (_dynamicSelectedStyle != null) {
state[6] = ((IStateManager)_dynamicSelectedStyle).SaveViewState();
hasViewState |= (state[6] != null);
}
if (_dynamicHoverStyle != null) {
state[7] = ((IStateManager)_dynamicHoverStyle).SaveViewState();
hasViewState |= (state[7] != null);
}
if (_dynamicMenuStyle != null) {
state[8] = ((IStateManager)_dynamicMenuStyle).SaveViewState();
hasViewState |= (state[8] != null);
}
if (_levelMenuItemStyles != null) {
state[9] = ((IStateManager)_levelMenuItemStyles).SaveViewState();
hasViewState |= (state[9] != null);
}
if (_levelSelectedStyles != null) {
state[10] = ((IStateManager)_levelSelectedStyles).SaveViewState();
hasViewState |= (state[10] != null);
}
if (_levelStyles != null) {
state[11] = ((IStateManager)_levelStyles).SaveViewState();
hasViewState |= (state[11] != null);
}
state[12] = ((IStateManager)Items).SaveViewState();
hasViewState |= (state[12] != null);
if (hasViewState) {
return state;
}
else {
return null;
}
}
[SecurityPermission(SecurityAction.Demand, Unrestricted = true)]
protected override void SetDesignModeState(IDictionary data) {
if (data.Contains("DesignTimeTextWriterType")) {
Type writerType = data["DesignTimeTextWriterType"] as Type;
if (writerType != null && writerType.IsSubclassOf(typeof(HtmlTextWriter))) {
_designTimeTextWriterType = writerType;
}
}
base.SetDesignModeState(data);
}
/// <devdoc>
/// Allows a derived Menu to set the DataBound proprety on a node
/// </devdoc>
protected void SetItemDataBound(MenuItem node, bool dataBound) {
node.SetDataBound(dataBound);
}
/// <devdoc>
/// Allows a derived Menu to set the DataItem on a node
/// </devdoc>
protected void SetItemDataItem(MenuItem node, object dataItem) {
node.SetDataItem(dataItem);
}
/// <devdoc>
/// Allows a derived Menu to set the DataPath on a node
/// </devdoc>
protected void SetItemDataPath(MenuItem node, string dataPath) {
node.SetDataPath(dataPath);
}
internal void SetSelectedItem(MenuItem node) {
Debug.Assert(node == null || node.Owner == this);
if (_selectedItem != node) {
if (node != null) {
if (node.Depth >= MaximumDepth) {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDepth));
}
if (!(node.IsEnabledNoOwner && node.Selectable)) {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidSelection));
}
}
// Unselect the previously selected item
if ((_selectedItem != null) && (_selectedItem.Selected)) {
_selectedItem.SetSelected(false);
}
_selectedItem = node;
// Notify the new selected item that it's now selected
if ((_selectedItem != null) && !_selectedItem.Selected) {
_selectedItem.SetSelected(true);
}
}
}
/// <internalonly/>
/// <devdoc>
/// Marks the starting point to begin tracking and saving changes to the
/// control as part of the control viewstate.
/// </devdoc>
protected override void TrackViewState() {
base.TrackViewState();
if (_staticItemStyle != null) {
((IStateManager)_staticItemStyle).TrackViewState();
}
if (_staticSelectedStyle != null) {
((IStateManager)_staticSelectedStyle).TrackViewState();
}
if (_staticHoverStyle != null) {
((IStateManager)_staticHoverStyle).TrackViewState();
}
if (_staticMenuStyle != null) {
((IStateManager)_staticMenuStyle).TrackViewState();
}
if (_dynamicItemStyle != null) {
((IStateManager)_dynamicItemStyle).TrackViewState();
}
if (_dynamicSelectedStyle != null) {
((IStateManager)_dynamicSelectedStyle).TrackViewState();
}
if (_dynamicHoverStyle != null) {
((IStateManager)_dynamicHoverStyle).TrackViewState();
}
if (_dynamicMenuStyle != null) {
((IStateManager)_dynamicMenuStyle).TrackViewState();
}
if (_levelMenuItemStyles != null) {
((IStateManager)_levelMenuItemStyles).TrackViewState();
}
if (_levelSelectedStyles != null) {
((IStateManager)_levelSelectedStyles).TrackViewState();
}
if (_levelStyles != null) {
((IStateManager)_levelStyles).TrackViewState();
}
if (_bindings != null) {
((IStateManager)_bindings).TrackViewState();
}
((IStateManager)Items).TrackViewState();
}
internal void VerifyRenderingInServerForm() {
if (Page != null) {
Page.VerifyRenderingInServerForm(this);
}
}
#region IPostBackEventHandler implementation
/// <internalonly/>
void IPostBackEventHandler.RaisePostBackEvent(string eventArgument) {
RaisePostBackEvent(eventArgument);
}
protected internal virtual void RaisePostBackEvent(string eventArgument) {
ValidateEvent(UniqueID, eventArgument);
// Do not take any postback into account if the menu is disabled.
if (!IsEnabled) return;
EnsureChildControls();
if (AdapterInternal != null) {
IPostBackEventHandler pbeh = AdapterInternal as IPostBackEventHandler;
if (pbeh != null) {
pbeh.RaisePostBackEvent(eventArgument);
}
}
else {
InternalRaisePostBackEvent(eventArgument);
}
}
internal void InternalRaisePostBackEvent(string eventArgument) {
if (eventArgument.Length == 0) {
return;
}
// Get the path of the node specified in the eventArgument
string nodePath = HttpUtility.HtmlDecode(eventArgument);
// Check the number of separator characters in the argument (should not be more than the max depth)
int matches = 0;
for (int i = 0; i < nodePath.Length; i++) {
if (nodePath[i] == TreeView.InternalPathSeparator) {
if (++matches >= MaximumDepth) {
throw new InvalidOperationException(SR.GetString(SR.Menu_InvalidDepth));
}
}
}
// Find that node in the tree
MenuItem node = Items.FindItem(nodePath.Split(TreeView.InternalPathSeparator), 0);
if (node != null) {
OnMenuItemClick(new MenuEventArgs(node));
}
}
#endregion
}
}