//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ namespace System.Web.UI.WebControls.WebParts { using System; using System.Collections; using System.Collections.Specialized; using System.ComponentModel; using System.Drawing; using System.Globalization; using System.Web; using System.Web.UI; using System.Web.UI.WebControls; using System.Web.Util; public abstract class CatalogZoneBase : ToolZone, IPostBackDataHandler { private CatalogPartCollection _catalogParts; private string[] _selectedCheckBoxValues; private string _selectedZoneID; private string _selectedCatalogPartID; private const int baseIndex = 0; private const int addVerbIndex = 1; private const int closeVerbIndex = 2; private const int partLinkStyleIndex = 3; private const int selectedPartLinkStyleIndex = 4; private const int viewStateArrayLength = 5; // Use same baseIndex as above private const int selectedCatalogPartIDIndex = 1; private const int controlStateArrayLength = 2; private WebPartVerb _addVerb; private WebPartVerb _closeVerb; private Style _partLinkStyle; private Style _selectedPartLinkStyle; private CatalogPartChrome _catalogPartChrome; private const string addEventArgument = "add"; private const string closeEventArgument = "close"; private const string selectEventArgument = "select"; protected CatalogZoneBase() : base(WebPartManager.CatalogDisplayMode) { } [ DefaultValue(null), DesignerSerializationVisibility(DesignerSerializationVisibility.Content), NotifyParentProperty(true), PersistenceMode(PersistenceMode.InnerProperty), WebCategory("Verbs"), WebSysDescription(SR.CatalogZoneBase_AddVerb), ] public virtual WebPartVerb AddVerb { get { if (_addVerb == null) { _addVerb = new WebPartCatalogAddVerb(); _addVerb.EventArgument = addEventArgument; if (IsTrackingViewState) { ((IStateManager)_addVerb).TrackViewState(); } } return _addVerb; } } internal string CheckBoxName { get { return UniqueID + ID_SEPARATOR + "_checkbox"; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public CatalogPartChrome CatalogPartChrome { get { if (_catalogPartChrome == null) { _catalogPartChrome = CreateCatalogPartChrome(); } return _catalogPartChrome; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public CatalogPartCollection CatalogParts { get { if (_catalogParts == null) { CatalogPartCollection catalogParts = CreateCatalogParts(); // Verify that each CatalogPart has a nonempty ID. Don't throw an exception in the designer, // since we want only the offending control to render as an error block, not the whole CatalogZone. if (!DesignMode) { foreach (CatalogPart catalogPart in catalogParts) { if (String.IsNullOrEmpty(catalogPart.ID)) { throw new InvalidOperationException(SR.GetString(SR.CatalogZoneBase_NoCatalogPartID)); } } } _catalogParts = catalogParts; // Call EnsureChildControls to parent the CatalogParts and set the WebPartManager, and Zone EnsureChildControls(); } return _catalogParts; } } [ DefaultValue(null), DesignerSerializationVisibility(DesignerSerializationVisibility.Content), NotifyParentProperty(true), PersistenceMode(PersistenceMode.InnerProperty), WebCategory("Verbs"), WebSysDescription(SR.CatalogZoneBase_CloseVerb), ] public virtual WebPartVerb CloseVerb { get { if (_closeVerb == null) { _closeVerb = new WebPartCatalogCloseVerb(); _closeVerb.EventArgument = closeEventArgument; if (IsTrackingViewState) { ((IStateManager)_closeVerb).TrackViewState(); } } return _closeVerb; } } [ WebSysDefaultValue(SR.CatalogZoneBase_DefaultEmptyZoneText) ] public override string EmptyZoneText { // Must look at viewstate directly instead of the property in the base class, // so we can distinguish between an unset property and a property set to String.Empty. get { string s = (string)ViewState["EmptyZoneText"]; return((s == null) ? SR.GetString(SR.CatalogZoneBase_DefaultEmptyZoneText) : s); } set { ViewState["EmptyZoneText"] = value; } } [ WebSysDefaultValue(SR.CatalogZoneBase_HeaderText), ] public override string HeaderText { get { string s = (string)ViewState["HeaderText"]; return((s == null) ? SR.GetString(SR.CatalogZoneBase_HeaderText) : s); } set { ViewState["HeaderText"] = value; } } [ WebSysDefaultValue(SR.CatalogZoneBase_InstructionText), ] public override string InstructionText { get { string s = (string)ViewState["InstructionText"]; return((s == null) ? SR.GetString(SR.CatalogZoneBase_InstructionText) : s); } set { ViewState["InstructionText"] = value; } } [ DefaultValue(null), NotifyParentProperty(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Content), PersistenceMode(PersistenceMode.InnerProperty), WebCategory("Styles"), WebSysDescription(SR.CatalogZoneBase_PartLinkStyle), ] public Style PartLinkStyle { get { if (_partLinkStyle == null) { _partLinkStyle = new Style(); if (IsTrackingViewState) { ((IStateManager)_partLinkStyle).TrackViewState(); } } return _partLinkStyle; } } [ DefaultValue(""), Themeable(false), WebCategory("Behavior"), WebSysDescription(SR.CatalogZoneBase_SelectedCatalogPartID), ] public string SelectedCatalogPartID { get { if (String.IsNullOrEmpty(_selectedCatalogPartID)) { if (DesignMode) { return String.Empty; } else { CatalogPartCollection catalogParts = CatalogParts; if (catalogParts != null && catalogParts.Count > 0) { return catalogParts[0].ID; } else { return String.Empty; } } } else { return _selectedCatalogPartID; } } set { _selectedCatalogPartID = value; } } // private CatalogPart SelectedCatalogPart { get { CatalogPartCollection catalogParts = CatalogParts; if (catalogParts != null && catalogParts.Count > 0) { if (String.IsNullOrEmpty(_selectedCatalogPartID)) { return catalogParts[0]; } else { return catalogParts[_selectedCatalogPartID]; } } else { // If there are no catalog parts, return null return null; } } } [ Localizable(true), WebSysDefaultValue(SR.CatalogZoneBase_DefaultSelectTargetZoneText), WebCategory("Behavior"), WebSysDescription(SR.CatalogZoneBase_SelectTargetZoneText), ] public virtual string SelectTargetZoneText { get { string s = (string)ViewState["SelectTargetZoneText"]; return((s == null) ? SR.GetString(SR.CatalogZoneBase_DefaultSelectTargetZoneText) : s); } set { ViewState["SelectTargetZoneText"] = value; } } [ DefaultValue(null), NotifyParentProperty(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Content), PersistenceMode(PersistenceMode.InnerProperty), WebCategory("Styles"), WebSysDescription(SR.CatalogZoneBase_SelectedPartLinkStyle), ] public Style SelectedPartLinkStyle { get { if (_selectedPartLinkStyle == null) { _selectedPartLinkStyle = new Style(); if (IsTrackingViewState) { ((IStateManager)_selectedPartLinkStyle).TrackViewState(); } } return _selectedPartLinkStyle; } } [ DefaultValue(true), WebCategory("Behavior"), WebSysDescription(SR.CatalogZoneBase_ShowCatalogIcons), ] public virtual bool ShowCatalogIcons { get { object b = ViewState["ShowCatalogIcons"]; return (b != null) ? (bool)b : true; } set { ViewState["ShowCatalogIcons"] = value; } } private string ZonesID { get { return UniqueID + ID_SEPARATOR + "_zones"; } } private void AddSelectedWebParts() { WebPartZoneBase selectedZone = null; if (WebPartManager != null) { selectedZone = WebPartManager.Zones[_selectedZoneID]; } CatalogPart selectedCatalogPart = SelectedCatalogPart; WebPartDescriptionCollection availableWebParts = null; if (selectedCatalogPart != null) { availableWebParts = selectedCatalogPart.GetAvailableWebPartDescriptions(); } if (selectedZone != null && selectedZone.AllowLayoutChange && _selectedCheckBoxValues != null && availableWebParts != null) { ArrayList selectedWebParts = new ArrayList(); // Fetch all of the WebParts before calling AddWebPart() on any of them. // This is necessary if the CatalogPart would refresh its list of // AvailableWebPartDescriptions in response to adding a WebPart. // PageCatalogPart is an example of this. (VSWhidbey 337539) for (int i = 0; i < _selectedCheckBoxValues.Length; i++) { string value = _selectedCheckBoxValues[i]; WebPartDescription webPartDescription = availableWebParts[value]; if (webPartDescription != null) { WebPart part = selectedCatalogPart.GetWebPart(webPartDescription); if (part != null) { selectedWebParts.Add(part); } } } AddWebParts(selectedWebParts, selectedZone); } } private void AddWebParts(ArrayList webParts, WebPartZoneBase zone) { // Add web parts from the list in reverse order, so they appear in the zone in the same // order they were returned from the catalog part. (VSWhidbey 77750) webParts.Reverse(); foreach (WebPart part in webParts) { WebPartZoneBase targetZone = zone; if (part.AllowZoneChange == false && part.Zone != null) { targetZone = part.Zone; } // WebPartManager is checked for null in AddWebParts() Debug.Assert(WebPartManager != null); // Add new parts to the top of the Zone, so the user will see them without scrolling the page WebPartManager.AddWebPart(part, targetZone, 0); } } protected override void Close() { if (WebPartManager != null) { WebPartManager.DisplayMode = WebPartManager.BrowseDisplayMode; } } protected virtual CatalogPartChrome CreateCatalogPartChrome() { return new CatalogPartChrome(this); } protected abstract CatalogPartCollection CreateCatalogParts(); /// protected internal override void CreateChildControls() { Controls.Clear(); // foreach (CatalogPart catalogPart in CatalogParts) { catalogPart.SetWebPartManager(WebPartManager); catalogPart.SetZone(this); Controls.Add(catalogPart); } } internal string GetCheckBoxID(string value) { return ClientID + ClientIDSeparator + "_checkbox" + ClientIDSeparator + value; } // Called by a derived class if the list of CatalogParts changes, and they want CreateCatalogParts() // to be called again. protected void InvalidateCatalogParts() { _catalogParts = null; ChildControlsCreated = false; } /// /// Loads the control state for those properties that should persist across postbacks /// even when EnableViewState=false. /// protected internal override void LoadControlState(object savedState) { if (savedState == null) { base.LoadControlState(null); } else { object[] myState = (object[])savedState; if (myState.Length != controlStateArrayLength) { throw new ArgumentException(SR.GetString(SR.Invalid_ControlState)); } base.LoadControlState(myState[baseIndex]); if (myState[selectedCatalogPartIDIndex] != null) { _selectedCatalogPartID = (string)myState[selectedCatalogPartIDIndex]; } } } protected virtual bool LoadPostData(string postDataKey, NameValueCollection postCollection) { string selectedCheckBoxValues = postCollection[CheckBoxName]; if (!String.IsNullOrEmpty(selectedCheckBoxValues)) { //Validate postback reference if exists in the postdata. ValidateEvent(CheckBoxName); _selectedCheckBoxValues = selectedCheckBoxValues.Split(','); } _selectedZoneID = postCollection[ZonesID]; // Do not raise a changed event return false; } protected override void LoadViewState(object savedState) { if (savedState == null) { base.LoadViewState(null); } else { object[] myState = (object[]) savedState; if (myState.Length != viewStateArrayLength) { throw new ArgumentException(SR.GetString(SR.ViewState_InvalidViewState)); } base.LoadViewState(myState[baseIndex]); if (myState[addVerbIndex] != null) { ((IStateManager) AddVerb).LoadViewState(myState[addVerbIndex]); } if (myState[closeVerbIndex] != null) { ((IStateManager) CloseVerb).LoadViewState(myState[closeVerbIndex]); } if (myState[partLinkStyleIndex] != null) { ((IStateManager) PartLinkStyle).LoadViewState(myState[partLinkStyleIndex]); } if (myState[selectedPartLinkStyleIndex] != null) { ((IStateManager) SelectedPartLinkStyle).LoadViewState(myState[selectedPartLinkStyleIndex]); } } } protected internal override void OnInit(EventArgs e) { base.OnInit(e); Page page = Page; Debug.Assert(page != null); if (page != null) { page.RegisterRequiresControlState(this); } } // We don't need to handle WebPartManager.DisplayModeChanged in this class. // We need it in EditorZoneBase since the available editor parts changes when the // WebPartToEdit changes, but the list of catalog parts never changes // when the DisplayMode changes. protected internal override void OnPreRender(EventArgs e) { base.OnPreRender(e); CatalogPartChrome.PerformPreRender(); Page.RegisterRequiresPostBack(this); } protected override void RaisePostBackEvent(string eventArgument) { string[] eventArguments = eventArgument.Split(ID_SEPARATOR); if ((eventArguments.Length == 2) && (eventArguments[0] == selectEventArgument)) { SelectedCatalogPartID = eventArguments[1]; } else if (String.Equals(eventArgument, addEventArgument, StringComparison.OrdinalIgnoreCase)) { if (AddVerb.Visible && AddVerb.Enabled) { AddSelectedWebParts(); } } else if (String.Equals(eventArgument, closeEventArgument, StringComparison.OrdinalIgnoreCase)) { if (CloseVerb.Visible && CloseVerb.Enabled) { Close(); } } else { base.RaisePostBackEvent(eventArgument); } } protected internal override void Render(HtmlTextWriter writer) { if (Page != null) { Page.VerifyRenderingInServerForm(this); } base.Render(writer); } protected override void RenderBody(HtmlTextWriter writer) { RenderBodyTableBeginTag(writer); if (DesignMode) { RenderDesignerRegionBeginTag(writer, Orientation.Vertical); } CatalogPartCollection catalogParts = CatalogParts; if (catalogParts != null && catalogParts.Count > 0) { bool firstCell = true; // Only render links if there is more than 1 catalog part (VSWhidbey 77672) if (catalogParts.Count > 1) { writer.RenderBeginTag(HtmlTextWriterTag.Tr); writer.RenderBeginTag(HtmlTextWriterTag.Td); firstCell = false; RenderCatalogPartLinks(writer); writer.RenderEndTag(); // Td writer.RenderEndTag(); // Tr } CatalogPartChrome chrome = CatalogPartChrome; if (DesignMode) { foreach (CatalogPart catalogPart in catalogParts) { RenderCatalogPart(writer, catalogPart, chrome, ref firstCell); } } else { CatalogPart selectedCatalogPart = SelectedCatalogPart; if (selectedCatalogPart != null) { RenderCatalogPart(writer, selectedCatalogPart, chrome, ref firstCell); } } writer.RenderBeginTag(HtmlTextWriterTag.Tr); // Mozilla renders padding on an empty TD without this attribute writer.AddStyleAttribute(HtmlTextWriterStyle.Padding, "0"); // Add an extra row with height of 100%, to [....] up any extra space // if the height of the zone is larger than its contents // Mac IE needs height=100% set on instead of writer.AddStyleAttribute(HtmlTextWriterStyle.Height, "100%"); writer.RenderBeginTag(HtmlTextWriterTag.Td); writer.RenderEndTag(); // Td writer.RenderEndTag(); // Tr } else { RenderEmptyZoneText(writer); } if (DesignMode) { RenderDesignerRegionEndTag(writer); } RenderBodyTableEndTag(writer); } private void RenderCatalogPart(HtmlTextWriter writer, CatalogPart catalogPart, CatalogPartChrome chrome, ref bool firstCell) { writer.RenderBeginTag(HtmlTextWriterTag.Tr); if (!firstCell) { writer.AddStyleAttribute(HtmlTextWriterStyle.PaddingTop, "0"); } writer.RenderBeginTag(HtmlTextWriterTag.Td); firstCell = false; chrome.RenderCatalogPart(writer, catalogPart); writer.RenderEndTag(); // Td writer.RenderEndTag(); // Tr } protected virtual void RenderCatalogPartLinks(HtmlTextWriter writer) { RenderInstructionText(writer); CatalogPart selectedCatalogPart = SelectedCatalogPart; foreach (CatalogPart catalogPart in CatalogParts) { WebPartDescriptionCollection availableWebParts = catalogPart.GetAvailableWebPartDescriptions(); int count = ((availableWebParts != null) ? availableWebParts.Count : 0); string displayTitle = catalogPart.DisplayTitle; // string text = displayTitle + " (" + count.ToString(CultureInfo.CurrentCulture) + ")"; if (catalogPart == selectedCatalogPart) { Label label = new Label(); label.Text = text; label.Page = Page; label.ApplyStyle(SelectedPartLinkStyle); label.RenderControl(writer); } else { Debug.Assert(!String.IsNullOrEmpty(catalogPart.ID)); string eventArgument = selectEventArgument + ID_SEPARATOR + catalogPart.ID; ZoneLinkButton linkButton = new ZoneLinkButton(this, eventArgument); linkButton.Text = text; linkButton.ToolTip = SR.GetString(SR.CatalogZoneBase_SelectCatalogPart, displayTitle); linkButton.Page = Page; linkButton.ApplyStyle(PartLinkStyle); linkButton.RenderControl(writer); } writer.WriteBreak(); } writer.WriteBreak(); } private void RenderEmptyZoneText(HtmlTextWriter writer) { string emptyZoneText = EmptyZoneText; if (!String.IsNullOrEmpty(emptyZoneText)) { writer.RenderBeginTag(HtmlTextWriterTag.Tr); writer.AddAttribute(HtmlTextWriterAttribute.Valign, "top"); Style emptyZoneTextStyle = EmptyZoneTextStyle; if (!emptyZoneTextStyle.IsEmpty) { emptyZoneTextStyle.AddAttributesToRender(writer, this); } writer.RenderBeginTag(HtmlTextWriterTag.Td); writer.Write(emptyZoneText); writer.RenderEndTag(); // Td writer.RenderEndTag(); // Tr } } protected override void RenderFooter(HtmlTextWriter writer) { writer.AddStyleAttribute(HtmlTextWriterStyle.Margin, "4px"); writer.RenderBeginTag(HtmlTextWriterTag.Div); DropDownList zonesDropDownList = new DropDownList(); zonesDropDownList.ClientIDMode = ClientIDMode.AutoID; zonesDropDownList.ID = ZonesID; // Populate the DropDownList if (DesignMode) { // Add sample zone to dropdown zonesDropDownList.Items.Add(SR.GetString(SR.Zone_SampleHeaderText)); } else { if (WebPartManager != null && WebPartManager.Zones != null) { foreach (WebPartZoneBase zone in WebPartManager.Zones) { if (zone.AllowLayoutChange) { Debug.Assert(!String.IsNullOrEmpty(zone.ID)); ListItem item = new ListItem(zone.DisplayTitle, zone.ID); if (String.Equals(zone.ID, _selectedZoneID, StringComparison.OrdinalIgnoreCase)) { item.Selected = true; } zonesDropDownList.Items.Add(item); } } } } LabelStyle.AddAttributesToRender(writer, this); // Only render the "for" attribute if we are going to render the associated DropDownList (VSWhidbey 541458) if (zonesDropDownList.Items.Count > 0) { writer.AddAttribute(HtmlTextWriterAttribute.For, zonesDropDownList.ClientID); } writer.RenderBeginTag(HtmlTextWriterTag.Label); writer.Write(SelectTargetZoneText); writer.RenderEndTag(); // Render   before the DropDownList (VSWhidbey 77709) writer.Write(" "); zonesDropDownList.ApplyStyle(EditUIStyle); // Do not render empty DropDownList (VSWhidbey 534498) if (zonesDropDownList.Items.Count > 0) { zonesDropDownList.RenderControl(writer); } writer.Write(" "); RenderVerbs(writer); writer.RenderEndTag(); // Div } private void RenderInstructionText(HtmlTextWriter writer) { string instructionText = InstructionText; if (!String.IsNullOrEmpty(instructionText)) { Label label = new Label(); label.Text = instructionText; label.Page = Page; label.ApplyStyle(InstructionTextStyle); label.RenderControl(writer); writer.WriteBreak(); writer.WriteBreak(); } } protected override void RenderVerbs(HtmlTextWriter writer) { int count = 0; bool originalAddVerbEnabled = false; CatalogPart selectedCatalogPart = SelectedCatalogPart; if (selectedCatalogPart != null) { WebPartDescriptionCollection availableWebParts = selectedCatalogPart.GetAvailableWebPartDescriptions(); count = ((availableWebParts != null) ? availableWebParts.Count : 0); } // If the current CatalogPart has no WebPartDescriptions, disable the AddVerb if (count == 0) { originalAddVerbEnabled = AddVerb.Enabled; AddVerb.Enabled = false; } try { RenderVerbsInternal(writer, new WebPartVerb[] {AddVerb, CloseVerb}); } finally { if (count == 0) { AddVerb.Enabled = originalAddVerbEnabled; } } } /// /// Saves the control state for those properties that should persist across postbacks /// even when EnableViewState=false. /// protected internal override object SaveControlState() { object[] myState = new object[controlStateArrayLength]; myState[baseIndex] = base.SaveControlState(); if (!String.IsNullOrEmpty(_selectedCatalogPartID)) { myState[selectedCatalogPartIDIndex] = _selectedCatalogPartID; } for (int i=0; i < controlStateArrayLength; i++) { if (myState[i] != null) { return myState; } } // More performant to return null than an array of null values return null; } protected override object SaveViewState() { object[] myState = new object[viewStateArrayLength]; myState[baseIndex] = base.SaveViewState(); myState[addVerbIndex] = (_addVerb != null) ? ((IStateManager)_addVerb).SaveViewState() : null; myState[closeVerbIndex] = (_closeVerb != null) ? ((IStateManager)_closeVerb).SaveViewState() : null; myState[partLinkStyleIndex] = (_partLinkStyle != null) ? ((IStateManager)_partLinkStyle).SaveViewState() : null; myState[selectedPartLinkStyleIndex] = (_selectedPartLinkStyle != null) ? ((IStateManager)_selectedPartLinkStyle).SaveViewState() : null; for (int i=0; i < viewStateArrayLength; i++) { if (myState[i] != null) { return myState; } } // More performant to return null than an array of null values return null; } protected override void TrackViewState() { base.TrackViewState(); if (_addVerb != null) { ((IStateManager) _addVerb).TrackViewState(); } if (_closeVerb != null) { ((IStateManager) _closeVerb).TrackViewState(); } if (_partLinkStyle != null) { ((IStateManager) _partLinkStyle).TrackViewState(); } if (_selectedPartLinkStyle != null) { ((IStateManager) _selectedPartLinkStyle).TrackViewState(); } } #region Implementation of IPostBackDataHandler bool IPostBackDataHandler.LoadPostData(string postDataKey, NameValueCollection postCollection) { return LoadPostData(postDataKey, postCollection); } void IPostBackDataHandler.RaisePostDataChangedEvent() { } #endregion } }