//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//------------------------------------------------------------------------------
/*
* Classes related to templated control support
*
* Copyright (c) 1999 Microsoft Corporation
*/
namespace System.Web.UI {
using System;
using System.Collections;
using System.Collections.Specialized;
using System.ComponentModel;
using System.ComponentModel.Design;
using System.Globalization;
using System.Reflection;
using System.Security.Permissions;
using System.Web.Util;
///
/// [To be supplied.]
///
public sealed class BindableTemplateBuilder : TemplateBuilder, IBindableTemplate {
private ExtractTemplateValuesMethod _extractTemplateValuesMethod;
/*
* No-compile delegate handler for ExtractValues.
*/
private IOrderedDictionary ExtractTemplateValuesMethod(Control container) {
/*System.Web.UI.OrderedDictionary @__table;
System.Web.UI.WebControls.DropDownList ddl2;
@__table = new System.Web.UI.OrderedDictionary();
ddl2 = ((System.Web.UI.WebControls.DropDownList)(@__container.FindControl("ddl2")));
if ((ddl2 != null)) {
@__table["FavVegetable"] = ddl2.SelectedValue;
}
return @__table;*/
BindableTemplateBuilder bindableTemplateBuilder = this as BindableTemplateBuilder;
Debug.Assert(bindableTemplateBuilder != null, "ExtractTemplateValuesMethod called on non-BindableTemplateBuilder.");
OrderedDictionary table = new OrderedDictionary();
if (bindableTemplateBuilder != null) {
ExtractTemplateValuesRecursive(bindableTemplateBuilder.SubBuilders, table, container);
}
return table;
}
private void ExtractTemplateValuesRecursive(ArrayList subBuilders, OrderedDictionary table, Control container) {
foreach (object subBuilderObject in subBuilders) {
ControlBuilder subBuilderControlBuilder = subBuilderObject as ControlBuilder;
if (subBuilderControlBuilder != null) {
ICollection entries;
// filter out device filtered bound entries that don't apply to this device
if (!subBuilderControlBuilder.HasFilteredBoundEntries) {
entries = subBuilderControlBuilder.BoundPropertyEntries;
}
else {
Debug.Assert(subBuilderControlBuilder.ServiceProvider == null);
Debug.Assert(subBuilderControlBuilder.TemplateControl != null, "TemplateControl should not be null in no-compile pages. We need it for the FilterResolutionService.");
ServiceContainer serviceContainer = new ServiceContainer();
serviceContainer.AddService(typeof(IFilterResolutionService), subBuilderControlBuilder.TemplateControl);
try {
subBuilderControlBuilder.SetServiceProvider(serviceContainer);
entries = subBuilderControlBuilder.GetFilteredPropertyEntrySet(subBuilderControlBuilder.BoundPropertyEntries);
}
finally {
subBuilderControlBuilder.SetServiceProvider(null);
}
}
string previousControlName = null;
bool newControl = true;
Control control = null;
foreach (BoundPropertyEntry entry in entries) {
// Skip all entries that are not two-way
if (!entry.TwoWayBound)
continue;
// Reset the "previous" Property Entry if we're not looking at the same control.
// If we don't do this, Two controls that have conditionals on the same named property will have
// their conditionals incorrectly merged.
if (String.Compare(previousControlName, entry.ControlID, StringComparison.Ordinal) != 0) {
newControl = true;
}
else {
newControl = false;
}
previousControlName = entry.ControlID;
if (newControl) {
control = container.FindControl(entry.ControlID);
if (control == null || !entry.ControlType.IsInstanceOfType(control)) {
Debug.Assert(false, "BoundPropertyEntry is of wrong control type or couldn't be found. Expected " + entry.ControlType.Name);
continue;
}
}
string propertyName;
// map the property in case it's a complex property
object targetObject = PropertyMapper.LocatePropertyObject(control, entry.Name, out propertyName, InDesigner);
// FastPropertyAccessor uses ReflectEmit for lightning speed
table[entry.FieldName] = FastPropertyAccessor.GetProperty(targetObject, propertyName, InDesigner);
}
ExtractTemplateValuesRecursive(subBuilderControlBuilder.SubBuilders, table, container);
}
}
}
/*
* IBindableTemplate implementation
* This implementation of ITemplate is used in the designer and no-compile.
*/
public IOrderedDictionary ExtractValues(Control container) {
if (_extractTemplateValuesMethod != null && !InDesigner) {
return _extractTemplateValuesMethod(container);
}
return new OrderedDictionary();
}
public override void OnAppendToParentBuilder(ControlBuilder parentBuilder) {
base.OnAppendToParentBuilder(parentBuilder);
if (HasTwoWayBoundProperties) {
_extractTemplateValuesMethod = new ExtractTemplateValuesMethod(ExtractTemplateValuesMethod);
}
}
}
// Delegates used for the compiled template and no-compile Bind
///
///
/// [To be supplied.]
///
public delegate IOrderedDictionary ExtractTemplateValuesMethod(Control control);
/*
* This class is the ITemplate implementation that is called from the
* generated page class code. It just passes the Initialize call on to a
* delegate.
*/
///
///
/// [To be supplied.]
///
public sealed class CompiledBindableTemplateBuilder : IBindableTemplate {
private BuildTemplateMethod _buildTemplateMethod;
private ExtractTemplateValuesMethod _extractTemplateValuesMethod;
///
/// [To be supplied.]
///
public CompiledBindableTemplateBuilder(BuildTemplateMethod buildTemplateMethod, ExtractTemplateValuesMethod extractTemplateValuesMethod) {
_buildTemplateMethod = buildTemplateMethod;
_extractTemplateValuesMethod = extractTemplateValuesMethod;
}
// IBindableTemplate::ExtractValues
///
/// Calls the ExtractTemplateValuesMethod delegate, which will return a dictionary of values.
///
public IOrderedDictionary ExtractValues(Control container) {
if (_extractTemplateValuesMethod != null) {
return _extractTemplateValuesMethod(container);
}
return new OrderedDictionary();
}
// ITemplate::InstantiateIn
///
/// [To be supplied.]
///
public void InstantiateIn(Control container) {
_buildTemplateMethod(container);
}
}
}