共用方式為


Web Form 樣板化資料繫結控制項設計工具範例

下列範例實作樣板資料繫結控制項 TemplatedList 的自訂設計工具 (TemplatedListDesigner),樣板化的資料繫結控制項範例中將說明該設計工具。TemplatedListDesigner 將說明下列概念。

  • 轉譯設計階段 HTML。
  • 實作樣板編輯器。
  • 實作屬性篩選。
  • 實作設計階段資料繫結。
'-----------------------------------------------------------------------
' TemplatedListDesigner.vb.
'------------------------------------------------------------------------
Imports System
Imports System.Collections
Imports System.ComponentModel
Imports System.ComponentModel.Design
Imports System.Data
Imports System.Diagnostics
Imports System.Web.UI
Imports System.Web.UI.Design
Imports System.Web.UI.WebControls
Imports CustomControls

Namespace CustomControls.Design
   Public Class TemplatedListDesigner
      Inherits TemplatedControlDesigner
      Implements IDataSourceProvider
      
      Private dummyDataTable As DataTable
      Private designTimeDataTable As DataTable
      Private templateVerbs() As TemplateEditingVerb
      Private templateVerbsDirty As Boolean
      
      Public Sub New()
         templateVerbsDirty = True
      End Sub
      
      Public Overrides ReadOnly Property AllowResize() As Boolean
         Get
            ' When templates are not defined, render a read-only fixed-
            ' size block. Once templates are defined or are being edited, the control should allow
            ' resizing.
            Return TemplatesExist Or InTemplateMode
         End Get
      End Property
      
      Public Property DataSource() As String
         Get
            Dim binding As DataBinding = DataBindings("DataSource")
            
            If Not (binding Is Nothing) Then
               Return binding.Expression
            End If
            Return [String].Empty
         End Get
         Set
            If value Is Nothing Or value.Length = 0 Then
               DataBindings.Remove("DataSource")
            Else
               Dim binding As DataBinding = DataBindings("DataSource")
               
               If binding Is Nothing Then
                  binding = New DataBinding("DataSource", GetType(IEnumerable), value)
               Else
                  binding.Expression = value
               End If
               DataBindings.Add(binding)
            End If
            
            OnDataSourceChanged()
            OnBindingsCollectionChanged("DataSource")
         End Set
      End Property
      
      Public Overrides ReadOnly Property DesignTimeHtmlRequiresLoadComplete() As Boolean
         Get
            ' If there is a data source, look it up in the container
            ' and require the document to be loaded completely.
            Return DataSource.Length <> 0
         End Get
      End Property
      
      Protected ReadOnly Property TemplatesExist() As Boolean
         Get
            Return Not (CType(Component, TemplatedList).ItemTemplate Is Nothing)
         End Get
      End Property
      
      Protected Overrides Function CreateTemplateEditingFrame(verb As TemplateEditingVerb) As ITemplateEditingFrame
         Dim teService As ITemplateEditingService = CType(GetService(GetType(ITemplateEditingService)), ITemplateEditingService)
         Debug.Assert( Not (teService Is Nothing), "How did we get this far without an ITemplateEditingService")
         Debug.Assert((verb.Index = 0))
         
         Dim templateNames() As String = {"ItemTemplate"}
         Dim templateStyles() As Style = {CType(Component, TemplatedList).ItemStyle}
         
         Dim editingFrame As ITemplateEditingFrame = teService.CreateFrame(Me, verb.Text, templateNames, CType(Component, TemplatedList).ControlStyle, templateStyles)
         Return editingFrame
      End Function
      
       Protected Overloads Overrides Sub Dispose(disposing As Boolean)
        If disposing
         DisposeTemplateVerbs()
        End If
         MyBase.Dispose(disposing)
      End Sub

      Private Sub DisposeTemplateVerbs()
         If Not (templateVerbs Is Nothing) Then
            templateVerbs(0).Dispose()
            
            templateVerbs = Nothing
            templateVerbsDirty = True
         End If
      End Sub
      
      Protected Overrides Function GetCachedTemplateEditingVerbs() As TemplateEditingVerb()
         If templateVerbsDirty = True Then
            DisposeTemplateVerbs()
            
            templateVerbs = New TemplateEditingVerb(1) {}
            templateVerbs(0) = New TemplateEditingVerb("Item Template", 0, Me)
            
            templateVerbsDirty = False
         End If
         
         Return templateVerbs
      End Function
      
      Protected Function GetDesignTimeDataSource(minimumRows As Integer, ByRef dummyDataSource As Boolean) As IEnumerable
         dummyDataSource = False
         
         Dim selectedDataSource As IEnumerable = CType(Me, IDataSourceProvider).GetResolvedSelectedDataSource()
         Dim dataTable As DataTable = designTimeDataTable
         
         ' Use the data table corresponding to the selected data source 
         ' if possible.
         If dataTable Is Nothing Then
            If Not (selectedDataSource Is Nothing) Then
               designTimeDataTable = DesignTimeData.CreateSampleDataTable(selectedDataSource)
               dataTable = designTimeDataTable
            End If
            
            If dataTable Is Nothing Then
               ' Fall back on a dummy data source if a sample data-table cannot be created.
               If dummyDataTable Is Nothing Then
                  dummyDataTable = DesignTimeData.CreateDummyDataTable()
               End If
               
               dataTable = dummyDataTable
               dummyDataSource = True
            End If
         End If
         
         Dim liveDataSource As IEnumerable = DesignTimeData.GetDesignTimeDataSource(dataTable, minimumRows)
         Return liveDataSource
      End Function
      
      Public Overrides Function GetDesignTimeHtml() As String
         Dim control As TemplatedList = CType(Component, TemplatedList)
         Dim designTimeHTML As String = Nothing
         Dim hasATemplate As Boolean = Me.TemplatesExist
         
         If hasATemplate Then
            Dim dummyDataSource As Boolean
            Dim designTimeDataSource As IEnumerable = GetDesignTimeDataSource(5, dummyDataSource)
            
            Try
               control.DataSource = designTimeDataSource
               control.DataBind()
               
               designTimeHTML = MyBase.GetDesignTimeHtml()
            Finally
               control.DataSource = Nothing
            End Try
         Else
            designTimeHTML = GetEmptyDesignTimeHtml()
         End If
         
         Return designTimeHTML
      End Function
      
      Protected Overrides Function GetEmptyDesignTimeHtml() As String
         Dim _text As String
         
         If CanEnterTemplateMode Then
            _text = "Right-click and choose a set of templates to edit their content.<br>The ItemTemplate is required."
         Else
            _text = "Switch to HTML view to edit the control's templates.<br>The ItemTemplate is required."
         End If
         Return CreatePlaceHolderDesignTimeHtml(_text)
      End Function
      
      Protected Overrides Function GetErrorDesignTimeHtml(e As Exception) As String
         Debug.Fail(e.ToString())
         Return CreatePlaceHolderDesignTimeHtml("There was an error rendering the control.<br>Check to make sure all properties are valid.")
      End Function
      
      Public Overrides Function GetTemplateContainerDataSource(templateName As String) As IEnumerable
         Return CType(Me, IDataSourceProvider).GetResolvedSelectedDataSource()
      End Function
      
      Public Overrides Function GetTemplateContainerDataItemProperty(templateName As String) As String
         Return "DataItem"
      End Function
      
      Public Overrides Function GetTemplateContent(editingFrame As ITemplateEditingFrame, templateName As String, ByRef allowEditing As Boolean) As String
         Debug.Assert((editingFrame.Verb.Index = 0))
         Debug.Assert(templateName.Equals("ItemTemplate"))
         allowEditing = True
         
         Dim template As ITemplate = CType(Component, TemplatedList).ItemTemplate
         Dim templateContent As String = [String].Empty
         
         If Not (template Is Nothing) Then
            templateContent = GetTextFromTemplate(template)
         End If
         
         Return templateContent
      End Function
      
      Public Overrides Sub OnComponentChanged(sender As Object, e As ComponentChangedEventArgs)
         If Not (e.Member Is Nothing) Then
            Dim memberName As String = e.Member.Name
            If memberName.Equals("DataSource") Or memberName.Equals("DataMember") Then
               OnDataSourceChanged()
            Else
               If memberName.Equals("ItemStyle") Then
                  OnStylesChanged()
               End If
            End If
         End If 
         MyBase.OnComponentChanged(sender, e)
      End Sub
      
      Protected Overridable Sub OnDataSourceChanged()
         designTimeDataTable = Nothing
      End Sub
      
      Protected Sub OnStylesChanged()
         OnTemplateEditingVerbsChanged()
      End Sub
      
      Protected Sub OnTemplateEditingVerbsChanged()
         templateVerbsDirty = True
      End Sub
      
      Protected Overrides Sub PreFilterProperties(properties As IDictionary)
         MyBase.PreFilterProperties(properties)
         
         Dim prop As PropertyDescriptor
         
         prop = CType(properties("DataSource"), PropertyDescriptor)
         Debug.Assert(( Not (prop Is Nothing)))
         prop = TypeDescriptor.CreateProperty(Me.GetType(), prop, New Attribute() {New TypeConverterAttribute(GetType(DataSourceConverter))})
         properties("DataSource") = prop
      End Sub
      
      Public Overrides Sub SetTemplateContent(editingFrame As ITemplateEditingFrame, templateName As String, templateContent As String)
         Debug.Assert((editingFrame.Verb.Index = 0))
         Debug.Assert(templateName.Equals("ItemTemplate"))
         
         Dim template As ITemplate = Nothing
         
         If Not (templateContent Is Nothing) And templateContent.Length <> 0 Then
            template = GetTemplateFromText(templateContent)
         End If
         
         CType(Component, TemplatedList).ItemTemplate = template
      End Sub
      
      Function GetResolvedSelectedDataSource() As IEnumerable Implements IDataSourceProvider.GetResolvedSelectedDataSource
         Return CType(CType(Me, IDataSourceProvider).GetSelectedDataSource(), IEnumerable)
      End Function
      
      Function GetSelectedDataSource() As Object Implements IDataSourceProvider.GetSelectedDataSource
         Dim selectedDataSource As Object = Nothing
         Dim dataSource As String = Nothing
         
         Dim binding As DataBinding = DataBindings("DataSource")
         If Not (binding Is Nothing) Then
            dataSource = binding.Expression
         End If
         
         If Not (dataSource Is Nothing) Then
            Dim componentSite As ISite = Component.Site
            If Not (componentSite Is Nothing) Then
               Dim container As IContainer = CType(componentSite.GetService(GetType(IContainer)), IContainer)
               
               If Not (container Is Nothing) Then
                  Dim comp As IComponent = container.Components(dataSource)
                  If TypeOf comp Is IEnumerable Then
                     selectedDataSource = comp
                  End If
               End If
            End If
         End If
         
         Return selectedDataSource
      End Function
   End Class
End Namespace
[C#]
//-----------------------------------------------------------------------
// TemplatedListDesigner.cs.
//------------------------------------------------------------------------

namespace CustomControls.Design {

    using System;
    using System.Collections;
    using System.ComponentModel;
    using System.ComponentModel.Design;
    using System.Data;
    using System.Diagnostics;
    using System.Web.UI;
    using System.Web.UI.Design;
    using System.Web.UI.WebControls;

    using CustomControls;

    public class TemplatedListDesigner : TemplatedControlDesigner, IDataSourceProvider {

        private DataTable dummyDataTable;
        private DataTable designTimeDataTable;
        private TemplateEditingVerb[] templateVerbs;
        private bool templateVerbsDirty;

        public TemplatedListDesigner() {
            templateVerbsDirty = true;
        }

        public override bool AllowResize {
            get {
                // When templates are not defined, render a read-only fixed-
                // size block. Once templates are defined or are being edited, the control should allow
                // resizing.
                return TemplatesExist || InTemplateMode;
            }
        }

        public string DataSource {
            get {
                DataBinding binding = DataBindings["DataSource"];

                if (binding != null) {
                    return binding.Expression;
                }
                return String.Empty;
            }
            set {
                if ((value == null) || (value.Length == 0)) {
                    DataBindings.Remove("DataSource");
                }
                else {
                    DataBinding binding = DataBindings["DataSource"];

                    if (binding == null) {
                        binding = new DataBinding("DataSource", typeof(IEnumerable), value);
                    }
                    else {
                        binding.Expression = value;
                    }
                    DataBindings.Add(binding);
                }

                OnDataSourceChanged();
                OnBindingsCollectionChanged("DataSource");
            }
        }

        public override bool DesignTimeHtmlRequiresLoadComplete {
            get {
                // If there is a data source, look it up in the container
                // and require the document to be loaded completely.
                return (DataSource.Length != 0);
            }
        }

        protected bool TemplatesExist {
            get {
                return (((TemplatedList)Component).ItemTemplate != null);
            }
        }

        protected override ITemplateEditingFrame CreateTemplateEditingFrame(TemplateEditingVerb verb) {
            ITemplateEditingService teService = (ITemplateEditingService)GetService(typeof(ITemplateEditingService));
            Debug.Assert(teService != null, "How did we get this far without an ITemplateEditingService");
            Debug.Assert(verb.Index == 0);

            string[] templateNames = new string[] { "ItemTemplate" };
            Style[] templateStyles = new Style[] { ((TemplatedList)Component).ItemStyle };

            ITemplateEditingFrame editingFrame =
                teService.CreateFrame(this, verb.Text, templateNames, ((TemplatedList)Component).ControlStyle, templateStyles);
            return editingFrame;
        }

        protected override void Dispose(bool disposing) {
            if (disposing) {
               DisposeTemplateVerbs();
            }
            base.Dispose(disposing);
        }

        private void DisposeTemplateVerbs() {
            if (templateVerbs != null) {
                templateVerbs[0].Dispose();

                templateVerbs = null;
                templateVerbsDirty = true;
            }
        }

        protected override TemplateEditingVerb[] GetCachedTemplateEditingVerbs() {
            if (templateVerbsDirty == true) {
                DisposeTemplateVerbs();

                templateVerbs = new TemplateEditingVerb[1];
                templateVerbs[0] = new TemplateEditingVerb("Item Template", 0, this);

                templateVerbsDirty = false;
            }

            return templateVerbs;
        }

        protected IEnumerable GetDesignTimeDataSource(int minimumRows, out bool dummyDataSource) {
            dummyDataSource = false;

            IEnumerable selectedDataSource = ((IDataSourceProvider)this).GetResolvedSelectedDataSource();
            DataTable dataTable = designTimeDataTable;

            // Use the data table corresponding to the selected data source 
            // if possible.
            if (dataTable == null) {
                if (selectedDataSource != null) {
                    designTimeDataTable = DesignTimeData.CreateSampleDataTable(selectedDataSource);
                    dataTable = designTimeDataTable;
                }

                if (dataTable == null) {
                    // Fall back on a dummy data source if a sample data table cannot be created.
                    if (dummyDataTable == null) {
                        dummyDataTable = DesignTimeData.CreateDummyDataTable();
                    }

                    dataTable = dummyDataTable;
                    dummyDataSource = true;
                }
            }

            IEnumerable liveDataSource = DesignTimeData.GetDesignTimeDataSource(dataTable, minimumRows);
            return liveDataSource;
        }

        public override string GetDesignTimeHtml() {
            TemplatedList control = (TemplatedList)Component;
            string designTimeHTML = null;
            bool hasATemplate = this.TemplatesExist;

            if (hasATemplate) {
                bool dummyDataSource;
                IEnumerable designTimeDataSource = GetDesignTimeDataSource(5, out dummyDataSource);

                try {
                    control.DataSource = designTimeDataSource;
                    control.DataBind();

                    designTimeHTML = base.GetDesignTimeHtml();
                }
                finally {
                    control.DataSource = null;
                }
            }
            else {
                designTimeHTML = GetEmptyDesignTimeHtml();
            }

            return designTimeHTML;
        }

        protected override string GetEmptyDesignTimeHtml() {
            string text;

            if (CanEnterTemplateMode) {
                text = "Right-click and choose a set of templates to edit their content.<br>The ItemTemplate is required.";
            }
            else {
                text = "Switch to HTML view to edit the control's templates.<br>The ItemTemplate is required.";
            }
            return CreatePlaceHolderDesignTimeHtml(text);
        }

        protected override string GetErrorDesignTimeHtml(Exception e) {
            Debug.Fail(e.ToString());
            return CreatePlaceHolderDesignTimeHtml("There was an error rendering the control.<br>Check to make sure all properties are valid.");
        }

        public override IEnumerable GetTemplateContainerDataSource(string templateName) {
            return ((IDataSourceProvider)this).GetResolvedSelectedDataSource();
        }

        public override string GetTemplateContainerDataItemProperty(string templateName) {
            return "DataItem";
        }

        public override string GetTemplateContent(ITemplateEditingFrame editingFrame, string templateName, out bool allowEditing) {
            Debug.Assert(editingFrame.Verb.Index == 0);
            Debug.Assert(templateName.Equals("ItemTemplate"));
            allowEditing = true;

            ITemplate template = ((TemplatedList)Component).ItemTemplate;
            string templateContent = String.Empty;

            if (template != null) {
                templateContent = GetTextFromTemplate(template);
            }

            return templateContent;
        }

        public override void OnComponentChanged(object sender, ComponentChangedEventArgs e) {
            if (e.Member != null) {
                string memberName = e.Member.Name;
                if (memberName.Equals("DataSource") || memberName.Equals("DataMember")) {
                    OnDataSourceChanged();
                }
                else if (memberName.Equals("ItemStyle")) {
                    OnStylesChanged();
                }
            }

            base.OnComponentChanged(sender, e);
        }

        protected virtual void OnDataSourceChanged() {
            designTimeDataTable = null;
        }

        protected void OnStylesChanged() {
            OnTemplateEditingVerbsChanged();
        }

        protected void OnTemplateEditingVerbsChanged() {
            templateVerbsDirty = true;
        }

        protected override void PreFilterProperties(IDictionary properties) {
            base.PreFilterProperties(properties);

            PropertyDescriptor prop;

            prop = (PropertyDescriptor)properties["DataSource"];
            Debug.Assert(prop != null);
            prop = TypeDescriptor.CreateProperty(this.GetType(), prop,
                                                 new Attribute[] {
                                                     new TypeConverterAttribute(typeof(DataSourceConverter))
                                                 });
            properties["DataSource"] = prop;
        }

        public override void SetTemplateContent(ITemplateEditingFrame editingFrame, string templateName, string templateContent) {
            Debug.Assert(editingFrame.Verb.Index == 0);
            Debug.Assert(templateName.Equals("ItemTemplate"));

            ITemplate template = null;

            if ((templateContent != null) && (templateContent.Length != 0)) {
                template = GetTemplateFromText(templateContent);
            }

            ((TemplatedList)Component).ItemTemplate = template;
        }


        IEnumerable IDataSourceProvider.GetResolvedSelectedDataSource() {
            return (IEnumerable)((IDataSourceProvider)this).GetSelectedDataSource();
        }

        object IDataSourceProvider.GetSelectedDataSource() {
            object selectedDataSource = null;
            string dataSource = null;

            DataBinding binding = DataBindings["DataSource"];
            if (binding != null) {
                dataSource = binding.Expression;
            }

            if (dataSource != null) {
                ISite componentSite = Component.Site;
                if (componentSite != null) {
                    IContainer container = (IContainer)componentSite.GetService(typeof(IContainer));

                    if (container != null) {
                        IComponent comp = container.Components[dataSource];
                        if (comp is IEnumerable) {
                            selectedDataSource = comp;
                        }
                    }
                }
            }

            return selectedDataSource;
        }
    }
}

請參閱

實作 Web Form 樣板編輯器 | 實作 Web Form 資料繫結設計工具 | 自訂設計工具 | Web Form 的設計階段支援