//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ namespace System.Web.UI.WebControls.WebParts { using System; using System.Collections; using System.Collections.Generic; using System.Collections.Specialized; using System.ComponentModel; using System.Drawing; using System.Drawing.Design; using System.Globalization; using System.Web; using System.Web.UI; using System.Web.UI.WebControls; using System.Web.Util; // /// /// Adds several features to the Part class, including connections, personalization behavior, /// and additional UI properties. /// [ Designer("System.Web.UI.Design.WebControls.WebParts.WebPartDesigner, " + AssemblyRef.SystemDesign) ] public abstract class WebPart : Part, IWebPart, IWebActionable, IWebEditable { private WebPartManager _webPartManager; private string _zoneID; private int _zoneIndex; private WebPartZoneBase _zone; private bool _allowClose; private bool _allowConnect; private bool _allowEdit; private bool _allowHide; private bool _allowMinimize; private bool _allowZoneChange; private string _authorizationFilter; private string _catalogIconImageUrl; private PartChromeState _chromeState; private string _connectErrorMessage; private WebPartExportMode _exportMode; private WebPartHelpMode _helpMode; private string _helpUrl; private bool _hidden; private string _importErrorMessage; private string _titleIconImageUrl; private string _titleUrl; private bool _hasUserData; private bool _hasSharedData; private bool _isClosed; private bool _isShared; private bool _isStandalone; private bool _isStatic; // Counter to detect circular connections private Dictionary _trackerCounter; internal const string WholePartIDPrefix = "WebPart_"; private const string titleBarIDPrefix = "WebPartTitle_"; protected WebPart() { _allowClose = true; _allowConnect = true; _allowEdit = true; _allowHide = true; _allowMinimize = true; _allowZoneChange = true; _chromeState = PartChromeState.Normal; _exportMode = WebPartExportMode.None; _helpMode = WebPartHelpMode.Navigate; _isStatic = true; _isStandalone = true; } /// /// Whether the user is allowed to close the web part /// [ DefaultValue(true), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_AllowClose), ] public virtual bool AllowClose { get { return _allowClose; } set { _allowClose = value; } } [ DefaultValue(true), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_AllowConnect), ] public virtual bool AllowConnect { get { return _allowConnect; } set { _allowConnect = value; } } /// /// If false, then LayoutEditorPart is the only visible EditorPart. Custom EditorParts /// may choose to be visible as well. /// [ DefaultValue(true), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_AllowEdit), ] public virtual bool AllowEdit { get { return _allowEdit; } set { _allowEdit = value; } } /// /// [ DefaultValue(true), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_AllowHide), ] public virtual bool AllowHide { get { return _allowHide; } set { _allowHide = value; } } /// /// Whether the user is allowed to minimize the web part /// [ DefaultValue(true), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_AllowMinimize), ] public virtual bool AllowMinimize { get { return _allowMinimize; } set { _allowMinimize = value; } } /// /// Whether the user is allowed move the web part around the page /// [ DefaultValue(true), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_AllowZoneChange), ] public virtual bool AllowZoneChange { get { return _allowZoneChange; } set { _allowZoneChange = value; } } [ DefaultValue(""), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_AuthorizationFilter), ] public virtual string AuthorizationFilter { get { return (_authorizationFilter != null) ? _authorizationFilter : String.Empty; } set { _authorizationFilter = value; } } [ DefaultValue(""), Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), UrlProperty(), WebCategory("WebPartAppearance"), Personalizable(PersonalizationScope.Shared), WebSysDescription(SR.WebPart_CatalogIconImageUrl), ] public virtual string CatalogIconImageUrl { get { return (_catalogIconImageUrl != null) ? _catalogIconImageUrl : String.Empty; } set { if (CrossSiteScriptingValidation.IsDangerousUrl(value)) { throw new ArgumentException(SR.GetString(SR.WebPart_BadUrl, value), "value"); } _catalogIconImageUrl = value; } } /// /// /// Overriden to mark as personalizable /// // PERF: Use a field instead of calling base.ChromeState, since the base implementation uses // viewstate. [ Personalizable ] public override PartChromeState ChromeState { get { return _chromeState; } set { if ((value < PartChromeState.Normal) || (value > PartChromeState.Minimized)) { throw new ArgumentOutOfRangeException("value"); } _chromeState = value; } } /// /// /// Overriden to mark as personalizable /// [ Personalizable ] public override PartChromeType ChromeType { get { return base.ChromeType; } set { base.ChromeType = value; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string ConnectErrorMessage { get { return (_connectErrorMessage != null) ? _connectErrorMessage : String.Empty; } } /// /// /// Overriden to mark as personalizable /// [ Personalizable(PersonalizationScope.Shared), ] public override string Description { get { return base.Description; } set { base.Description = value; } } /// /// /// Overriden to mark as personalizable /// [ Personalizable ] public override ContentDirection Direction { get { return base.Direction; } set { base.Direction = value; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string DisplayTitle { get { if (_webPartManager != null) { return _webPartManager.GetDisplayTitle(this); } else { // Needed for a WebPart in a DeclarativeCatalogPart, or any case where WebPartManager // has not been set. string displayTitle = Title; if (String.IsNullOrEmpty(displayTitle)) { displayTitle = SR.GetString(SR.Part_Untitled); } return displayTitle; } } } [ DefaultValue(WebPartExportMode.None), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_ExportMode), ] public virtual WebPartExportMode ExportMode { get { return _exportMode; } set { if (ControlState >= ControlState.Loaded && (WebPartManager == null || (WebPartManager.Personalization.Scope == PersonalizationScope.User && IsShared))) { throw new InvalidOperationException(SR.GetString(SR.WebPart_CantSetExportMode)); } if (value < WebPartExportMode.None || value > WebPartExportMode.NonSensitiveData) { throw new ArgumentOutOfRangeException("value"); } _exportMode = value; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public bool HasUserData { get { return _hasUserData; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public bool HasSharedData { get { return _hasSharedData; } } /// /// /// Overriden to mark as personalizable /// [ Personalizable ] public override Unit Height { get { return base.Height; } set { base.Height = value; } } /// /// The type of help UI used to display the help topic /// [ DefaultValue(WebPartHelpMode.Navigate), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_HelpMode), ] public virtual WebPartHelpMode HelpMode { get { return _helpMode; } set { if ((value < WebPartHelpMode.Modal) || (value > WebPartHelpMode.Navigate)) { throw new ArgumentOutOfRangeException("value"); } _helpMode = value; } } /// /// The URL of the web part's associated help topic /// [ DefaultValue(""), Editor("System.Web.UI.Design.UrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), UrlProperty(), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_HelpUrl), ] public virtual string HelpUrl { get { return (_helpUrl != null) ? _helpUrl : String.Empty; } set { if (CrossSiteScriptingValidation.IsDangerousUrl(value)) { throw new ArgumentException(SR.GetString(SR.WebPart_BadUrl, value), "value"); } _helpUrl = value; } } /// /// Whether the web part is to be visually displayed or not. /// A web part with Hidden set to true still participates in various /// page lifecycle phases such as PreRender. /// [ DefaultValue(false), Personalizable, Themeable(false), WebCategory("WebPartAppearance"), WebSysDescription(SR.WebPart_Hidden), ] public virtual bool Hidden { get { return _hidden; } set { _hidden = value; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public bool IsClosed { get { return _isClosed; } } /// /// An "orphaned" part has no Zone, but has not been moved to the page catalog yet. /// internal bool IsOrphaned { get { return (Zone == null && !IsClosed); } } [ Localizable(true), WebCategory("WebPartAppearance"), WebSysDefaultValue(SR.WebPart_DefaultImportErrorMessage), Personalizable(PersonalizationScope.Shared), WebSysDescription(SR.WebPart_ImportErrorMessage), ] public virtual string ImportErrorMessage { get { return (_importErrorMessage != null) ? _importErrorMessage : SR.GetString(SR.WebPart_DefaultImportErrorMessage); } set { _importErrorMessage = value; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public bool IsShared { get { return _isShared; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public bool IsStandalone { get { return _isStandalone; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public bool IsStatic { get { return _isStatic; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), Localizable(true), ] public virtual string Subtitle { get { return String.Empty; } } /// /// /// Overriden to mark as personalizable /// [ Personalizable, ] public override string Title { get { return base.Title; } set { base.Title = value; } } // ID rendered on the title bar of the WebPart, so a mouse listener can be attached // for drag and drop. internal string TitleBarID { get { return titleBarIDPrefix + ID; } } [ DefaultValue(""), Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), UrlProperty(), WebCategory("WebPartAppearance"), Personalizable(PersonalizationScope.Shared), WebSysDescription(SR.WebPart_TitleIconImageUrl), ] public virtual string TitleIconImageUrl { get { return (_titleIconImageUrl != null) ? _titleIconImageUrl : String.Empty; } set { if (CrossSiteScriptingValidation.IsDangerousUrl(value)) { throw new ArgumentException(SR.GetString(SR.WebPart_BadUrl, value), "value"); } _titleIconImageUrl = value; } } [ DefaultValue(""), Editor("System.Web.UI.Design.UrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), UrlProperty(), Personalizable(PersonalizationScope.Shared), Themeable(false), WebCategory("WebPartBehavior"), WebSysDescription(SR.WebPart_TitleUrl), ] public virtual string TitleUrl { get { return (_titleUrl != null) ? _titleUrl : String.Empty; } set { if (CrossSiteScriptingValidation.IsDangerousUrl(value)) { throw new ArgumentException(SR.GetString(SR.WebPart_BadUrl, value), "value"); } _titleUrl = value; } } internal Dictionary TrackerCounter { get { if (_trackerCounter == null) { _trackerCounter = new Dictionary(); } return _trackerCounter; } } /// /// Overriden by subclasses to add Verbs for this WebPart. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public virtual WebPartVerbCollection Verbs { get { return WebPartVerbCollection.Empty; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public virtual object WebBrowsableObject { get { return this; } } protected WebPartManager WebPartManager { get { return _webPartManager; } } // ID rendered on the table containing the whole web part. We shouldn't render ClientID // on the table, since it will be rendered by the WebPart on the container for the part // contents. We shouldn't render ID either, since it may be the same as another control // on the page, and it should be different than ID since it is being rendered by the Zone, // not the WebPart. internal string WholePartID { get { return WholePartIDPrefix + ID; } } /// /// /// Overriden to mark as personalizable /// [ Personalizable ] public override Unit Width { get { return base.Width; } set { base.Width = value; } } /// /// The WebPartZone that this WebPart is currently rendered within. /// If the WebPart is closed, returns the WebPartZone that the WebPart /// was last rendered within. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public WebPartZoneBase Zone { get { if (_zone == null) { string zoneID = ZoneID; if (!String.IsNullOrEmpty(zoneID) && WebPartManager != null) { WebPartZoneCollection zones = WebPartManager.Zones; if (zones != null) { _zone = zones[zoneID]; } } } return _zone; } } /// /// The ID of the web part zone that this web part logically belongs to /// internal string ZoneID { get { return _zoneID; } set { if (ZoneID != value) { _zoneID = value; // Invalidate cache _zone = null; } } } /// /// The index of this web part within the web part zone it logically belongs to. /// An index of -1 means the part is not currently in a zone. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public int ZoneIndex { get { return _zoneIndex; } } public virtual EditorPartCollection CreateEditorParts() { return EditorPartCollection.Empty; } protected internal virtual void OnClosing(EventArgs e) { } protected internal virtual void OnConnectModeChanged(EventArgs e) { } protected internal virtual void OnDeleting(EventArgs e) { } protected internal virtual void OnEditModeChanged(EventArgs e) { } internal override void PreRenderRecursiveInternal() { if (IsStandalone) { if (Hidden) { throw new InvalidOperationException(SR.GetString(SR.WebPart_NotStandalone, "Hidden", ID)); } } else { if (!Visible) { throw new InvalidOperationException(SR.GetString(SR.WebPart_OnlyStandalone, "Visible", ID)); } } base.PreRenderRecursiveInternal(); } internal void SetConnectErrorMessage(string connectErrorMessage) { // Only set the error message if it has not been set already. The first error message // set should be displayed. if (String.IsNullOrEmpty(_connectErrorMessage)) { _connectErrorMessage = connectErrorMessage; } } internal void SetHasUserData(bool hasUserData) { _hasUserData = hasUserData; } internal void SetHasSharedData(bool hasSharedData) { _hasSharedData = hasSharedData; } internal void SetIsClosed(bool isClosed) { _isClosed = isClosed; } internal void SetIsShared(bool isShared) { _isShared = isShared; } internal void SetIsStandalone(bool isStandalone) { _isStandalone = isStandalone; } internal void SetIsStatic(bool isStatic) { _isStatic = isStatic; } protected void SetPersonalizationDirty() { if (WebPartManager == null) { throw new InvalidOperationException(SR.GetString(SR.WebPartManagerRequired)); } WebPartManager.Personalization.SetDirty(this); } /// /// This method allows a non-WebPart control to mark its personalization as dirty. /// public static void SetPersonalizationDirty(Control control) { if (control == null) { throw new ArgumentNullException("control"); } if (control.Page == null) { throw new ArgumentException(SR.GetString(SR.PropertyCannotBeNull, "Page"), "control"); } WebPartManager wpm = WebPartManager.GetCurrentWebPartManager(control.Page); if (wpm == null) { throw new InvalidOperationException(SR.GetString(SR.WebPartManagerRequired)); } WebPart webPart = wpm.GetGenericWebPart(control); if (webPart == null) { throw new ArgumentException(SR.GetString(SR.WebPart_NonWebPart), "control"); } webPart.SetPersonalizationDirty(); } internal void SetWebPartManager(WebPartManager webPartManager) { _webPartManager = webPartManager; } /// /// The index of this web part within the web part zone it logically belongs to. /// internal void SetZoneIndex(int zoneIndex) { if (zoneIndex < 0) { throw new ArgumentOutOfRangeException("zoneIndex"); } _zoneIndex = zoneIndex; } // If this is a GenericWebPart, returns the ChildControl. Else, just returns a pointer to itself. // Used when you need the Control to pass to methods on ConnectionPoint. internal Control ToControl() { GenericWebPart genericWebPart = this as GenericWebPart; if (genericWebPart != null) { Control control = genericWebPart.ChildControl; if (control != null) { return control; } else { throw new InvalidOperationException(SR.GetString(SR.GenericWebPart_ChildControlIsNull)); } } else { return this; } } protected override void TrackViewState() { if (WebPartManager != null) { WebPartManager.Personalization.ApplyPersonalizationState(this); } base.TrackViewState(); } /// /// We need this to return nonzero for any two distinct WebParts on the page, /// since we use SortedList in WebPartManager, and SortedList cannot contain /// two keys where the IComparer returns zero. Two WebParts can have the same /// ZoneIndex when we merge the Shared and User parts. We use the ID /// to order the parts if ZoneIndex is the same. /// internal sealed class ZoneIndexComparer : IComparer { public int Compare(object x, object y) { WebPart p1 = (WebPart)x; WebPart p2 = (WebPart)y; int c = p1.ZoneIndex - p2.ZoneIndex; if (c == 0) { c = String.Compare(p1.ID, p2.ID, StringComparison.CurrentCulture); } return c; } } } }