0

asp.net グリッドビューを継承するカスタム グリッドビュー コントロールを作成しました。このグリッドビューでアイテム テンプレートを使用する必要があります。カスタム グリッドビューで、アイテム テンプレートを生成するメソッドを作成します。

public void addTemplateField(Control headerTemplateControl, Control itemTemplateControl, EventHandler bindHandler, EventHandler initHandler, string headerText, string sortExpression, bool isVisible, int? heightPx, int? widthPercent)
{
    TemplateField tField = new TemplateField();

    if (headerTemplateControl != null)
        tField.HeaderTemplate = new GridViewTemplate(ListItemType.Header, headerTemplateControl);

    if (bindHandler != null && initHandler != null)
        tField.ItemTemplate = new GridViewTemplate(ListItemType.Item, itemTemplateControl, bindHandler, initHandler);

    else if (bindHandler != null)
        tField.ItemTemplate = new GridViewTemplate(ListItemType.Item, itemTemplateControl, bindHandler, false);

    else if (initHandler != null)
        tField.ItemTemplate = new GridViewTemplate(ListItemType.Item, itemTemplateControl, initHandler, true);

    else
        tField.ItemTemplate = new GridViewTemplate(ListItemType.Item, itemTemplateControl);

    if (sortExpression != null)
        tField.SortExpression = sortExpression;

    tField.Visible = isVisible;

    if (headerText != null)
        tField.HeaderText = headerText;

    if (heightPx.HasValue)
        tField.HeaderStyle.Height = new Unit(heightPx.Value, UnitType.Pixel);

    if (widthPercent.HasValue)
        tField.HeaderStyle.Height = new Unit(widthPercent.Value, UnitType.Percentage);

    addColumnField(tField);
}

そして、これが私が ITemplate を実装した方法です

public class GridViewTemplate : ITemplate
{
    int _controlCount = 0;
    ListItemType _templateType;
    EventHandler _bindHandler;
    EventHandler _initHandler;
    Control _control;
    public GridViewTemplate(ListItemType type, Control control)
    {
        this._templateType = type;
        this._control = control;
    }
    public GridViewTemplate(ListItemType type, Control control, EventHandler Handler, bool isInitHandler)
    {
        this._templateType = type;
        this._control = control;
        if (isInitHandler)
            this._initHandler = Handler;
        else
            this._bindHandler = Handler;
    }
    public GridViewTemplate(ListItemType type, Control control, EventHandler bindHandler, EventHandler initHandler)
    {
        this._templateType = type;
        this._control = control;
        this._bindHandler = bindHandler;
        this._initHandler = initHandler;
    }
    public Control Copy(Control ctrlSource)
    {
        Type _type = ctrlSource.GetType();
        Control ctrlDest = (Control)Activator.CreateInstance(_type);
        foreach (PropertyInfo prop in _type.GetProperties())
        {
            if (prop.CanWrite)
            {
                if (prop.Name == "ID")
                {
                    ctrlDest.ID = ctrlSource.ID + "_copy_" + _controlCount;
                }
                else
                {
                    prop.SetValue(ctrlDest, prop.GetValue(ctrlSource, null), null);
                }
            }
        }
        _controlCount++;

        return ctrlDest;
    }
    public void InstantiateIn(Control container)
    {
        switch (_templateType)
        {
            case ListItemType.Header:
                container.Controls.Add(_control);
                break;
            case ListItemType.Item:
                Control temp = Copy(_control);
                if(_bindHandler != null)
                    temp.DataBinding += _bindHandler;
                if (_initHandler != null)
                    temp.Init += _initHandler;
                container.Controls.Add(temp);
                break;
        }
    }

}

Default.aspx.cs と言う必要があるページで、このグリッドビュー onPreInit を作成し、そのイベント ハンドラー onInit をアタッチします。

addTemplateField() を呼び出して、グリッドにチェックボックスを追加します。

cbl = new CheckBox();
cbl.AutoPostBack = true;
init = new EventHandler(cbl_Init);
grd.addTemplateField(null, cbl, null, init, "SERVER", null, true, 20, 20);

void cbl_Init(object sender, EventArgs e)
{
    CheckBox c = (CheckBox)sender;
    c.CheckedChanged +=new EventHandler(cbl_CheckedChanged);
}

void cbl_CheckedChanged(object sender, EventArgs e)
{
  // Modify datasource
  // databind();

// このデータバインドを削除すると、毎回 checkchanged が処理されます。データバインドを維持すると、イベントは交互にのみ処理されます。}

問題は、チェックボックスの checkchanged イベントが交互に発生することです。ページは 1 回おきにポスト バックしますが、checkchanged イベントは処理されません。解決どころか原因究明に迷う!?!?!

4

1 に答える 1