0

ここに私のカスタム制御コードがあります:

namespace MyControlNameSpace  
{  
  public class MyControlItemCommandEventArgs : CommandEventArgs
  {
    private MyControlItem _item;
    private object _commandSource;

    public MyControlItemCommandEventArgs(MyControlItem item, object commandSource, CommandEventArgs cea) : base(cea)
     {
      _item = item;
      _commandSource = commandSource;   
     }

    public MyControlItem Item { get{ return _item; } }
    public object CommandSource { get{ return _commandSource; } }

  }

  public class MyControlItem : Control, IDataItemContainer
  {
    public MyControlItem(object dataItem, int index)
    {
        _dataItem = dataItem;
        _dataItemIndex = _displayIndex = index;
    }  

    private readonly object _dataItem;
    private readonly int _dataItemIndex;
    private readonly int _displayIndex;

    public object DataItem { get { return _dataItem; } }
    public int DataItemIndex { get { return _dataItemIndex; } }
    public int DisplayIndex { get { return _displayIndex; } }

    protected override bool OnBubbleEvent(object source, EventArgs args)
    {
      if (args is CommandEventArgs)
      {
       var e = new MyControlItemCommandEventArgs(this, source, (CommandEventArgs)args);
       base.RaiseBubbleEvent(this, e);
       return true;
      }

      return false;
    }
  }  

  public delegate void ItemEventHandler(object sender, MyControlItemCommandEventArgs e);

  [DefaultProperty("Text")]
  [ToolboxData("<{0}:MyControl runat=server></{0}:MyControl>")]
  public class MyControl : CompositeDataBoundControl, IPrivilage
  {
    public event ItemEventHandler ItemCommand;

    protected virtual void OnItemCommand(MyControlItemCommandEventArgs e)
    {
     if (ItemCommand != null)
        ItemCommand(this, e);
    }

    protected override bool OnBubbleEvent(object source, EventArgs args)
    {
        // only bother bubbling appropriate events
        if (args is MyControlItemCommandEventArgs)
        {
            OnItemCommand((MyControlItemCommandEventArgs)args);
            return true;
        }

        return false;
    }

    protected override int CreateChildControls(IEnumerable dataSource, bool dataBinding)
    {
       // here I'm adding some controls like buttons and other

       // Iterate for ItemTemplate
       foreach (object dataItem in dataSource)
       {
         if (ItemTemplate != null)
         {
           // create instance of MyControlItem control
           var item = new MyControlItem(dataItem, count++);
           // instantiate in new item object
           ItemTemplate.InstantiateIn(item);
           // add item to Controls collection
           Controls.Add(item);
           // need to support <%# %> expressions
           item.DataBind();
         }
       }

      // some code here
    }
  }

.aspx ページ:

protected void Page_Load(object sender, EventArgs e)
 {
   // if it's not post back then get the data and assign to MyControl DataSource property
   if(!IsPostBack)
     GetData();
 }    

protected void MyControl1_OnItemCommand(object sender, MyControlItemCommandEventArgs e)
 {
   var x = e.DataItem;  
   // x here is always null;
 }

ご覧のとおり、常に null であるため、OnItemCommand イベントのときに e.DataItem を取得できません。ページロードのたびにデータソースが割り当てられるように削除しようとしましたが、これは正しくありません。Asp.Net コントロールのようであれば、データソースを割り当てないようにする必要が
あります。 if(!IsPostBack)Page_LoadPostBackRepeater

それを行う方法はありますか?

4

1 に答える 1

0

イベントはOnItemCommandポストバックで発生します。その時点で、CompositeDataBoundControlViewState有効になっている場合は、データベースにクエリを実行せずに子コントロールを再作成します。メソッドはset toでCreateChildControls呼び出され、実際のデータのプレースホルダーとして機能する列挙可能なオブジェクトのシーケンスになります。dataBindingfalsedataSourcenull

どのデータ バインド コントロールでも同じ動作が得られます。で同じことを試すと、がイベントに含まれてRepeaterいることがわかります。e.Item.DataItemnullItemCommand

于 2012-10-31T15:38:08.767 に答える