//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//------------------------------------------------------------------------------
/*
* SiteMapNode class definition
*
* Copyright (c) 2002 Microsoft Corporation
*/
namespace System.Web {
using System;
using System.Configuration.Provider;
using System.Collections;
using System.Collections.Specialized;
using System.ComponentModel;
using System.Resources;
using System.Security.Permissions;
using System.Web.Configuration;
using System.Web.Compilation;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.Util;
///
///
///
public class SiteMapNode : ICloneable, IHierarchyData, INavigateUIData {
private static readonly string _siteMapNodeType = typeof(SiteMapNode).Name;
private SiteMapProvider _provider;
private bool _readonly;
private bool _parentNodeSet;
private bool _childNodesSet;
private VirtualPath _virtualPath;
private string _title;
private string _description;
private string _url;
private string _key;
private string _resourceKey;
private IList _roles;
private NameValueCollection _attributes;
private NameValueCollection _resourceKeys;
private SiteMapNode _parentNode;
private SiteMapNodeCollection _childNodes;
public SiteMapNode(SiteMapProvider provider, string key) :
this(provider, key, null, null, null, null, null, null, null) {
}
public SiteMapNode(SiteMapProvider provider, string key, string url) :
this(provider, key, url, null, null, null, null, null, null) {
}
public SiteMapNode(SiteMapProvider provider, string key, string url, string title) :
this(provider, key, url, title, null, null, null, null, null) {
}
public SiteMapNode(SiteMapProvider provider, string key, string url, string title, string description) :
this(provider, key, url, title, description, null, null, null, null) {
}
public SiteMapNode(SiteMapProvider provider, string key, string url, string title, string description,
IList roles, NameValueCollection attributes, NameValueCollection explicitResourceKeys, string implicitResourceKey) {
_provider = provider;
_title = title;
_description = description;
_roles = roles;
_attributes = attributes;
_key = key;
_resourceKeys = explicitResourceKeys;
_resourceKey = implicitResourceKey;
if (url != null) {
_url = url.Trim();
}
_virtualPath = CreateVirtualPathFromUrl(_url);
if (_key == null) {
throw new ArgumentNullException("key");
}
if (_provider == null) {
throw new ArgumentNullException("provider");
}
}
protected NameValueCollection Attributes {
get {
return _attributes;
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "Attributes"));
}
_attributes = value;
}
}
// Access custom attributes.
public virtual string this[string key] {
get {
string text = null;
if (_attributes != null) {
text = _attributes[key];
}
if (_provider.EnableLocalization) {
// Try the implicit resource first
string localizedText = GetImplicitResourceString(key);
if (localizedText != null) {
return localizedText;
}
// If not found, try the explicit resource.
localizedText = GetExplicitResourceString(key, text, true);
if (localizedText != null) {
return localizedText;
}
}
return text;
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "Item"));
}
if (_attributes == null) {
_attributes = new NameValueCollection();
}
_attributes[key] = value;
}
}
public virtual SiteMapNodeCollection ChildNodes {
get {
if (_childNodesSet)
return _childNodes;
return _provider.GetChildNodes(this);
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "ChildNodes"));
}
_childNodes = value;
_childNodesSet = true;
}
}
[
Localizable(true)
]
public virtual string Description {
get {
if (_provider.EnableLocalization) {
string localizedText = GetImplicitResourceString("description");
if (localizedText != null) {
return localizedText;
}
localizedText = GetExplicitResourceString("description", _description, true);
if (localizedText != null) {
return localizedText;
}
}
return _description == null? String.Empty : _description;
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "Description"));
}
_description = value;
}
}
public string Key {
get {
return _key;
}
}
public virtual bool HasChildNodes {
get {
IList children = ChildNodes;
return children != null && children.Count > 0;
}
}
public virtual SiteMapNode NextSibling {
get {
IList siblings = SiblingNodes;
if (siblings == null) {
return null;
}
int index = siblings.IndexOf(this);
if (index >= 0 && index < siblings.Count - 1) {
return (SiteMapNode)siblings[index + 1];
}
return null;
}
}
// Get parent node. If not found in current provider, search recursively in parent providers.
public virtual SiteMapNode ParentNode {
get {
if (_parentNodeSet)
return _parentNode;
return _provider.GetParentNode(this);
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "ParentNode"));
}
_parentNode = value;
_parentNodeSet = true;
}
}
public virtual SiteMapNode PreviousSibling {
get {
IList siblings = SiblingNodes;
if (siblings == null) {
return null;
}
int index = siblings.IndexOf(this);
if (index > 0 && index <= siblings.Count - 1) {
return (SiteMapNode)siblings[index - 1];
}
return null;
}
}
public SiteMapProvider Provider {
get {
return _provider;
}
}
public bool ReadOnly {
get {
return _readonly;
}
set {
_readonly = value;
}
}
public String ResourceKey {
get {
return _resourceKey;
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "ResourceKey"));
}
_resourceKey = value;
}
}
public IList Roles {
get {
return _roles;
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "Roles"));
}
_roles = value;
}
}
public virtual SiteMapNode RootNode {
get {
SiteMapNode root = _provider.RootProvider.RootNode;
if (root == null) {
String name = ((ProviderBase)_provider.RootProvider).Name;
throw new InvalidOperationException(SR.GetString(SR.SiteMapProvider_Invalid_RootNode, name));
}
return root;
}
}
private SiteMapNodeCollection SiblingNodes {
get {
SiteMapNode parent = ParentNode;
return parent == null? null : parent.ChildNodes;
}
}
[
Localizable(true)
]
public virtual string Title {
get {
if (_provider.EnableLocalization) {
string localizedText = GetImplicitResourceString("title");
if (localizedText != null) {
return localizedText;
}
localizedText = GetExplicitResourceString("title", _title, true);
if (localizedText != null) {
return localizedText;
}
}
return _title == null? String.Empty : _title;
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "Title"));
}
_title = value;
}
}
public virtual string Url {
get {
return _url == null? String.Empty : _url;
}
set {
if (_readonly) {
throw new InvalidOperationException(SR.GetString(SR.SiteMapNode_readonly, "Url"));
}
if (value != null) {
_url = value.Trim();
}
_virtualPath = CreateVirtualPathFromUrl(_url);
}
}
internal VirtualPath VirtualPath {
get {
return _virtualPath;
}
}
private VirtualPath CreateVirtualPathFromUrl(string url) {
if (String.IsNullOrEmpty(url)) {
return null;
}
if (!UrlPath.IsValidVirtualPathWithoutProtocol(url)) {
return null;
}
if (UrlPath.IsAbsolutePhysicalPath(url)) {
return null;
}
// Do not generate the virtualPath class at designtime.
if (HttpRuntime.AppDomainAppVirtualPath == null) {
return null;
}
if (UrlPath.IsRelativeUrl(url) && !UrlPath.IsAppRelativePath(url)) {
url = UrlPath.Combine(HttpRuntime.AppDomainAppVirtualPathString, url);
}
// Remove the query string from url so the path can be validated by Authorization module.
int queryStringIndex = url.IndexOf('?');
if (queryStringIndex != -1) {
url = url.Substring(0, queryStringIndex);
}
return VirtualPath.Create(url,
VirtualPathOptions.AllowAbsolutePath | VirtualPathOptions.AllowAppRelativePath);
}
public virtual SiteMapNode Clone() {
ArrayList newRoles = null;
NameValueCollection newAttributes = null;
NameValueCollection newResourceKeys = null;
if (_roles != null) {
newRoles = new ArrayList(_roles);
}
if (_attributes != null) {
newAttributes = new NameValueCollection(_attributes);
}
if (_resourceKeys != null) {
newResourceKeys = new NameValueCollection(_resourceKeys);
}
SiteMapNode newNode = new SiteMapNode(_provider, Key, Url, Title, Description, newRoles, newAttributes, newResourceKeys, _resourceKey);
return newNode;
}
public virtual SiteMapNode Clone(bool cloneParentNodes) {
SiteMapNode current = Clone();
if (cloneParentNodes) {
SiteMapNode node = current;
SiteMapNode parent = ParentNode;
while (parent != null) {
SiteMapNode cloneParent = parent.Clone();
node.ParentNode = cloneParent;
cloneParent.ChildNodes = new SiteMapNodeCollection(node);
parent = parent.ParentNode;
node = cloneParent;
}
}
return current;
}
public override bool Equals(object obj) {
SiteMapNode node = obj as SiteMapNode;
return node != null && (_key == node.Key) &&
(String.Equals(_url, node._url, StringComparison.OrdinalIgnoreCase));
}
public SiteMapNodeCollection GetAllNodes() {
SiteMapNodeCollection collection = new SiteMapNodeCollection();
GetAllNodesRecursive(collection);
return SiteMapNodeCollection.ReadOnly(collection);
}
private void GetAllNodesRecursive(SiteMapNodeCollection collection) {
SiteMapNodeCollection childNodes = this.ChildNodes;
if (childNodes != null && childNodes.Count > 0) {
collection.AddRange(childNodes);
foreach(SiteMapNode node in childNodes)
node.GetAllNodesRecursive(collection);
}
}
public SiteMapDataSourceView GetDataSourceView(SiteMapDataSource owner, string viewName) {
return new SiteMapDataSourceView(owner, viewName, this);
}
public SiteMapHierarchicalDataSourceView GetHierarchicalDataSourceView() {
return new SiteMapHierarchicalDataSourceView(this);
}
// Helpe method to retrieve localized string based on attribute name
protected string GetExplicitResourceString(string attributeName, string defaultValue, bool throwIfNotFound) {
if (attributeName == null) {
throw new ArgumentNullException("attributeName");
}
string text = null;
if (_resourceKeys != null) {
string[] keys = _resourceKeys.GetValues(attributeName);
if (keys != null && keys.Length > 1) {
try {
text = ResourceExpressionBuilder.GetGlobalResourceObject(keys[0], keys[1]) as string;
}
catch (MissingManifestResourceException) {
if (defaultValue != null) {
return defaultValue;
}
}
if (text == null && throwIfNotFound) {
// throw if default value is not specified.
throw new InvalidOperationException(
SR.GetString(SR.Res_not_found_with_class_and_key, keys[0], keys[1])); ;
}
}
}
return text;
}
// Only use the key to get the hashcode since url can be changed and makes the objects mutable.
public override int GetHashCode() {
return _key.GetHashCode();
}
// Helper method to retrieve localized string based on attribute name
protected string GetImplicitResourceString(string attributeName) {
if (attributeName == null) {
throw new ArgumentNullException("attributeName");
}
string text = null;
if (!String.IsNullOrEmpty(_resourceKey)) {
try {
text = ResourceExpressionBuilder.GetGlobalResourceObject(Provider.ResourceKey, ResourceKey + "." + attributeName) as String;
}
catch { }
}
return text;
}
public virtual bool IsAccessibleToUser(HttpContext context) {
return _provider.IsAccessibleToUser(context, this);
}
public virtual bool IsDescendantOf(SiteMapNode node) {
SiteMapNode parent = ParentNode;
while (parent != null) {
if (parent.Equals(node)) {
return true;
}
parent = parent.ParentNode;
}
return false;
}
public override string ToString() {
return Title;
}
#region ICloneable implementation
///
object ICloneable.Clone() {
return Clone();
}
#endregion
#region IHierarchyData implementation
///
bool IHierarchyData.HasChildren {
get {
return HasChildNodes;
}
}
///
object IHierarchyData.Item {
get {
return this;
}
}
///
string IHierarchyData.Path {
get {
return Key;
}
}
///
string IHierarchyData.Type {
get {
return _siteMapNodeType;
}
}
///
IHierarchicalEnumerable IHierarchyData.GetChildren() {
return ChildNodes;
}
///
IHierarchyData IHierarchyData.GetParent() {
SiteMapNode parentNode = ParentNode;
if (parentNode == null)
return null;
return parentNode;
}
#endregion
#region INavigateUIData implementations
string INavigateUIData.Description {
get {
return Description;
}
}
///
string INavigateUIData.Name {
get {
return Title;
}
}
///
string INavigateUIData.NavigateUrl {
get {
return Url;
}
}
///
string INavigateUIData.Value {
get {
return Title;
}
}
#endregion
}
}