503 lines
18 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <copyright file="BulletedList.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
namespace System.Web.UI.WebControls {
using System;
using System.Collections;
using System.Collections.Specialized;
using System.ComponentModel;
using System.Globalization;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Drawing.Design;
using System.Web.Util;
/// <devdoc>
/// <para>Generates a bulleted list.</para>
/// </devdoc>
[DefaultProperty("BulletStyle")]
[DefaultEvent("Click")]
[Designer("System.Web.UI.Design.WebControls.BulletedListDesigner, " + AssemblyRef.SystemDesign)]
[SupportsEventValidation]
public class BulletedList : ListControl, IPostBackEventHandler {
private static readonly object EventClick = new object();
private bool _cachedIsEnabled;
private int _firstItem;
private int _itemCount;
/// <devdoc></devdoc>
public BulletedList() {
_firstItem = 0;
_itemCount = -1;
}
/// <devdoc>
/// <para>Gets the value of the base classes AutoPostBack propert.
/// AutoPostBack is not applicable to the bulleted list control</para>
/// </devdoc>
[
Browsable(false),
EditorBrowsable(EditorBrowsableState.Never)
]
public override bool AutoPostBack {
get {
return base.AutoPostBack;
}
set {
throw new NotSupportedException(SR.GetString(SR.Property_Set_Not_Supported, "AutoPostBack", this.GetType().ToString()));
}
}
/// <devdoc>
/// <para>Gets or sets a value indicating the style of bullet to be
/// applied to the list.</para>
/// </devdoc>
[
WebCategory("Appearance"),
DefaultValue(BulletStyle.NotSet),
WebSysDescription(SR.BulletedList_BulletStyle)
]
public virtual BulletStyle BulletStyle {
get {
object o = ViewState["BulletStyle"];
return((o == null) ? BulletStyle.NotSet : (BulletStyle)o);
}
set {
if (value < BulletStyle.NotSet || value > BulletStyle.CustomImage) {
throw new ArgumentOutOfRangeException("value");
}
ViewState["BulletStyle"] = value;
}
}
/// <devdoc>
/// <para>Gets or sets the source of the image used for an
/// Image styled bulleted list.</para>
/// </devdoc>
[
WebCategory("Appearance"),
DefaultValue(""),
Editor("System.Web.UI.Design.ImageUrlEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)),
UrlProperty(),
WebSysDescription(SR.BulletedList_BulletImageUrl)
]
public virtual string BulletImageUrl {
get {
object o = ViewState["BulletImageUrl"];
return((o == null) ? string.Empty : (string)o);
}
set {
ViewState["BulletImageUrl"] = value;
}
}
/// <devdoc>
/// <para>Gets the EmptyControlCollection.</para>
/// </devdoc>
public override ControlCollection Controls {
get {
return new EmptyControlCollection(this);
}
}
/// <devdoc>
/// <para>Gets or sets the display mode of the bulleted list.</para>
/// </devdoc>
[
WebCategory("Behavior"),
DefaultValue(BulletedListDisplayMode.Text),
WebSysDescription(SR.BulletedList_BulletedListDisplayMode) //
]
public virtual BulletedListDisplayMode DisplayMode {
get {
object o = ViewState["DisplayMode"];
return ((o == null) ? BulletedListDisplayMode.Text : (BulletedListDisplayMode)o);
}
set {
if (value < BulletedListDisplayMode.Text || value > BulletedListDisplayMode.LinkButton) {
throw new ArgumentOutOfRangeException("value");
}
ViewState["DisplayMode"] = value;
}
}
/// <devdoc>
/// <para>Gets or sets the value at which an ordered list should
/// begin its numbering.</para>
/// </devdoc>
[
WebCategory("Appearance"),
DefaultValue(1),
WebSysDescription(SR.BulletedList_FirstBulletNumber)
]
public virtual int FirstBulletNumber {
get {
object o = ViewState["FirstBulletNumber"];
return((o == null) ? 1 : (int)o);
}
set {
ViewState["FirstBulletNumber"] = value;
}
}
/// <summary>
/// <para>Indicates whether the control will be rendered when the data source has no items.</para>
/// </summary>
[DefaultValue(false)]
[Themeable(true)]
[WebCategory("Behavior")]
[WebSysDescription(SR.ListControl_RenderWhenDataEmpty)]
public virtual bool RenderWhenDataEmpty {
get {
object o = ViewState["RenderWhenDataEmpty"];
return ((o == null) ? false : (bool)o);
}
set {
ViewState["RenderWhenDataEmpty"] = value;
}
}
/// <devdoc>
/// <para>Gets the value of selected index. Not applicable to the
/// bulleted list control.</para>
/// </devdoc>
[
Bindable(false),
EditorBrowsable(EditorBrowsableState.Never)
]
public override int SelectedIndex {
get {
return base.SelectedIndex;
}
set {
throw new NotSupportedException(SR.GetString(SR.BulletedList_SelectionNotSupported));
}
}
/// <devdoc>
/// <para>Gets the selected item. Not applicable to the
/// bulleted list control.</para>
/// </devdoc>
[
EditorBrowsable(EditorBrowsableState.Never)
]
public override ListItem SelectedItem {
get {
return base.SelectedItem;
}
}
[
Bindable(false),
EditorBrowsable(EditorBrowsableState.Never)
]
public override string SelectedValue {
get {
return base.SelectedValue;
}
set {
throw new NotSupportedException(SR.GetString(SR.BulletedList_SelectionNotSupported));
}
}
/// <devdoc>
/// <para>Gets the HtmlTextWriterTag value that corresponds
/// to the particular bulleted list.</para>
/// </devdoc>
protected override HtmlTextWriterTag TagKey {
get {
return TagKeyInternal;
}
}
internal HtmlTextWriterTag TagKeyInternal {
get {
switch (BulletStyle) {
// Ordered Lists
case BulletStyle.LowerAlpha:
case BulletStyle.UpperAlpha:
case BulletStyle.LowerRoman:
case BulletStyle.UpperRoman:
case BulletStyle.Numbered:
return HtmlTextWriterTag.Ol;
// Unordered Lists
case BulletStyle.Square:
case BulletStyle.Circle:
case BulletStyle.Disc:
return HtmlTextWriterTag.Ul;
// Image Lists
case BulletStyle.CustomImage:
return HtmlTextWriterTag.Ul;
// Not Set
case BulletStyle.NotSet:
// NotSet is specified as an unordered list.
return HtmlTextWriterTag.Ul;
default:
Debug.Assert(false, "Invalid BulletStyle");
return HtmlTextWriterTag.Ol;
}
}
}
/// <devdoc>
/// <para>Gets or sets the Target window when the
/// list is displayed as Hyperlinks.</para>
/// </devdoc>
[
WebCategory("Behavior"),
DefaultValue(""),
WebSysDescription(SR.BulletedList_Target),
TypeConverter(typeof(TargetConverter))
]
public virtual string Target {
get {
object o = ViewState["Target"];
return ((o == null) ? string.Empty : (string)o);
}
set {
ViewState["Target"] = value;
}
}
[
EditorBrowsable(EditorBrowsableState.Never)
]
public override string Text {
get {
return base.Text;
}
set {
throw new NotSupportedException(SR.GetString(SR.BulletedList_TextNotSupported));
}
}
/// <devdoc>
/// <para>Occurs when the a link button is clicked.</para>
/// </devdoc>
[
WebCategory("Action"),
WebSysDescription(SR.BulletedList_OnClick)
]
public event BulletedListEventHandler Click {
add {
Events.AddHandler(EventClick, value);
}
remove {
Events.RemoveHandler(EventClick, value);
}
}
/// <devdoc>
/// <para>Adds HTML attributes that need to be rendered.</para>
/// </devdoc>
protected override void AddAttributesToRender(HtmlTextWriter writer) {
bool addBulletNumber = false;
switch (BulletStyle) {
case BulletStyle.NotSet:
break;
case BulletStyle.Numbered:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "decimal");
addBulletNumber = true;
break;
case BulletStyle.LowerAlpha:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "lower-alpha");
addBulletNumber = true;
break;
case BulletStyle.UpperAlpha:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "upper-alpha");
addBulletNumber = true;
break;
case BulletStyle.LowerRoman:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "lower-roman");
addBulletNumber = true;
break;
case BulletStyle.UpperRoman:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "upper-roman");
addBulletNumber = true;
break;
case BulletStyle.Disc:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "disc");
break;
case BulletStyle.Circle:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "circle");
break;
case BulletStyle.Square:
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleType, "square");
break;
case BulletStyle.CustomImage:
String url = ResolveClientUrl(BulletImageUrl);
writer.AddStyleAttribute(HtmlTextWriterStyle.ListStyleImage, "url(" + HttpUtility.UrlPathEncode(url) + ")");
break;
default:
Debug.Assert(false, "Invalid BulletStyle");
break;
}
int firstBulletNumber = FirstBulletNumber;
if ((addBulletNumber == true) && (firstBulletNumber != 1)) {
writer.AddAttribute("start", firstBulletNumber.ToString(CultureInfo.InvariantCulture));
}
base.AddAttributesToRender(writer);
}
private string GetPostBackEventReference(string eventArgument) {
if (CausesValidation && Page.GetValidators(ValidationGroup).Count > 0) {
return ClientScriptManager.JscriptPrefix + Util.GetClientValidatedPostback(this, ValidationGroup, eventArgument);
} else {
return Page.ClientScript.GetPostBackClientHyperlink(this, eventArgument, true);
}
}
/// <devdoc>
/// <para>Raises the Click event.</para>
/// </devdoc>
protected virtual void OnClick(BulletedListEventArgs e) {
BulletedListEventHandler onClickHandler = (BulletedListEventHandler)Events[EventClick];
if (onClickHandler != null)
onClickHandler(this, e);
}
/// <devdoc>
/// <para>Writes the text of each bullet according to the list's display mode.</para>
/// </devdoc>
protected virtual void RenderBulletText(ListItem item, int index, HtmlTextWriter writer) {
switch (DisplayMode) {
case BulletedListDisplayMode.Text:
if (!item.Enabled) {
RenderDisabledAttributeHelper(writer, false);
writer.RenderBeginTag(HtmlTextWriterTag.Span);
}
HttpUtility.HtmlEncode(item.Text, writer);
if (!item.Enabled) {
writer.RenderEndTag();
}
break;
case BulletedListDisplayMode.HyperLink:
if (_cachedIsEnabled && item.Enabled) {
writer.AddAttribute(HtmlTextWriterAttribute.Href, ResolveClientUrl(item.Value));
string target = Target;
if (!String.IsNullOrEmpty(target)) {
writer.AddAttribute(HtmlTextWriterAttribute.Target, Target);
}
}
else {
RenderDisabledAttributeHelper(writer, item.Enabled);
}
RenderAccessKey(writer, AccessKey);
writer.RenderBeginTag(HtmlTextWriterTag.A);
HttpUtility.HtmlEncode(item.Text, writer);
writer.RenderEndTag();
break;
case BulletedListDisplayMode.LinkButton:
if (_cachedIsEnabled && item.Enabled) {
writer.AddAttribute(HtmlTextWriterAttribute.Href, GetPostBackEventReference(index.ToString(CultureInfo.InvariantCulture)));
}
else {
RenderDisabledAttributeHelper(writer, item.Enabled);
}
RenderAccessKey(writer, AccessKey);
writer.RenderBeginTag(HtmlTextWriterTag.A);
HttpUtility.HtmlEncode(item.Text, writer);
writer.RenderEndTag();
break;
default:
Debug.Assert(false, "Invalid BulletedListDisplayMode");
break;
}
}
private void RenderDisabledAttributeHelper(HtmlTextWriter writer, bool isItemEnabled) {
if (SupportsDisabledAttribute) {
writer.AddAttribute(HtmlTextWriterAttribute.Disabled, "disabled");
}
else if (!isItemEnabled && !String.IsNullOrEmpty(DisabledCssClass)) {
writer.AddAttribute(HtmlTextWriterAttribute.Class, DisabledCssClass);
}
}
internal void RenderAccessKey(HtmlTextWriter writer, string AccessKey) {
string s = AccessKey;
if (s.Length > 0) {
writer.AddAttribute(HtmlTextWriterAttribute.Accesskey, s);
}
}
protected internal override void Render(HtmlTextWriter writer) {
// Don't render anything if the control is empty (unless the developer opts in)
if (Items.Count == 0 && !RenderWhenDataEmpty) {
return;
}
base.Render(writer);
}
/// <devdoc>
/// <para>Renders the ListItems as bullets in the bulleted list.</para>
/// </devdoc>
protected internal override void RenderContents(HtmlTextWriter writer) {
_cachedIsEnabled = IsEnabled;
if (_itemCount == -1) {
for (int i = 0; i < Items.Count; i++) {
Items[i].RenderAttributes(writer);
writer.RenderBeginTag(HtmlTextWriterTag.Li);
RenderBulletText(Items[i], i, writer);
writer.RenderEndTag();
}
}
else {
for (int i = _firstItem; i < _firstItem + _itemCount; i++) {
Items[i].RenderAttributes(writer);
writer.RenderBeginTag(HtmlTextWriterTag.Li);
RenderBulletText(Items[i], i, writer);
writer.RenderEndTag();
}
}
}
protected virtual void RaisePostBackEvent(string eventArgument) {
ValidateEvent(this.UniqueID, eventArgument);
if (CausesValidation) {
Page.Validate(ValidationGroup);
}
OnClick(new BulletedListEventArgs(Int32.Parse(eventArgument, CultureInfo.InvariantCulture)));
}
void IPostBackEventHandler.RaisePostBackEvent(string eventArgument) {
RaisePostBackEvent(eventArgument);
}
}
}