gpt4 book ai didi

c# - 为什么我的自定义控件的组件没有启动?

转载 作者:太空狗 更新时间:2023-10-29 20:24:07 26 4
gpt4 key购买 nike

我正在制作一个 GenericTable 作为 GridView 的自定义实现,它将显示插入的任何对象列表的值。

要在 aspx 页面上使用该控件,它需要是一个 UserControl,因此 GridView 作为一个组件包含在 GenericTable 中:

<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="GenericTable.ascx.cs" Inherits="CASH.WebApplication.Controls.GenericTable" %>
<div style="width: 100%; overflow: scroll">
<asp:GridView ID="grid" runat="server"></asp:GridView>
</div>

第一次使用我的控件时效果很好,它已添加到 aspx 页面上。这样做似乎增加了某种魔法来启动控制组件。当用户单击具有其自身属性的项目时,GenericTable 应在当前行下方插入一行并生成一个将显示所述属性的新 GenericTabletable 是我用来设置 GridView 内容的 DataTable:

var data = table.NewRow();
var child = new GenericTable();

data[0] = child;

table.Rows.InsertAt(data, row);
grid.DataSource = table;
grid.DataBind(); // The extra row is displayed now, initialize components in the aspx code?
child.MakeTable(); // Throws exception because it's `grid` property is null.

当我尝试激活新制作的 GenericTable 时,在这段代码之后,它的 grid 为空。

有没有办法初始化当此控件位于 aspx 代码中时发生的相同魔术?

更新:也许问题在于表在回发之间的存储方式,目前我正在使用 session ,也许有更好的方法来记住用户输入?整个 GenericTable 代码:

using Project.DomainModel.Models;
using System;
using System.Collections.Generic;
using System.Data;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Web.UI;
using System.Web.UI.WebControls;

namespace CASH.WebApplication.Controls
{
public partial class GenericTable : UserControl
{
private PropertyInfo[] properties;
//private GridView gridView;
private DataTable table = new DataTable();
private Dictionary<int, int> ingedrukt = new Dictionary<int, int>();

protected void Page_Init(object sender, EventArgs e)
{
grid.RowCommand += WeergaveDossiers_RowCommand;
}

protected void Page_Load(object sender, EventArgs e)
{
if (!IsPostBack)
{
for (int i = 0; i < grid.Rows.Count; i++)
{
grid.Rows[i].Cells[0].ColumnSpan = 0;
}
}
else
{
properties = (PropertyInfo[])Session["properties"];
table = (DataTable)Session["table"];
ingedrukt = (Dictionary<int, int>)Session["ingedrukt"];

foreach (var knop in ingedrukt)
{
DetailRijToevoegen(knop.Key, knop.Value);
}
}

grid.DataBind();
}

protected void SaveInSession()
{
Session["properties"] = properties;
Session["table"] = table;
Session["ingedrukt"] = ingedrukt;
}

protected void WeergaveDossiers_RowCommand(object sender, GridViewCommandEventArgs e)
{
int row = int.Parse((string)e.CommandArgument) + 1;
int col = GetKolomIndex(e.CommandName) + 1;

if (ingedrukt.ContainsKey(row))
{
if (ingedrukt[row] != col)
{
//DetailRijVerwijderen(row);
//ingedrukt.Remove(row);
//ingedrukt[row] = col;
}
}
else
{
ingedrukt[row] = col;
}

//DetailRijToevoegen(row, col);
SaveInSession();
}

protected void DetailRijToevoegen(int row, int col)
{
var data = table.NewRow();
var child = new GenericTable();
child.grid = new GridView();

data[0] = child;

table.Rows.InsertAt(data, row);
grid.DataSource = table;
grid.DataBind();

var cells = grid.Rows[row].Cells;
// Only keep the first cell
while (cells.Count > 1)
{
cells.RemoveAt(1);
}

child.MaakTable(new List<object>() { table.Rows[row][col] });

grid.Columns[0].Visible = true;
grid.Rows[row].Cells[0].ColumnSpan = table.Columns.Count;
}

protected void DetailRijVerwijderen(int row)
{

}

protected int GetKolomIndex(string naam)
{
for (int i = 0; i < properties.Length; i++)
{
if (properties[i].Name == naam)
{
return i;
}
}

throw new InvalidDataException("Kolom naam " + naam + " niet bekend");
}

public void MaakTable(IEnumerable<object> data)
{
properties = data.First().GetType().GetProperties().Where(p => p.CanRead).ToArray();

grid.AutoGenerateColumns = false;
var details = new BoundField();
details.DataField = "Details";
grid.Columns.Add(details);

table.Columns.Add(new DataColumn("Details", typeof(object)));
foreach (var veld in properties)
{
table.Columns.Add(new DataColumn(veld.Name, (veld.Name == "Id" ? typeof(object) : veld.PropertyType)));
grid.Columns.Add(MaakKolom(veld));
}

foreach (var entry in data)
{
var row = table.NewRow();
int col = 0;

foreach (var veld in properties)
{
row[++col] = veld.GetValue(entry);
}

table.Rows.Add(row);
}

grid.DataSource = table;

SaveInSession();
}

protected DataControlField MaakKolom(PropertyInfo veld)
{
DataControlField field;

if (typeof(Entity).IsAssignableFrom(veld.PropertyType))
{
field = new ButtonField();
((ButtonField)field).DataTextField = veld.Name;
((ButtonField)field).ButtonType = ButtonType.Button;
((ButtonField)field).CommandName = veld.Name;
}
else if (veld.PropertyType == typeof(bool))
{
field = new CheckBoxField();
((CheckBoxField)field).DataField = veld.Name;
}
else if (veld.PropertyType.IsEnum)
{
field = new TemplateField();
//((TemplateField)field).ItemTemplate = (ITemplate)new Label()
//{
// Text = "#DataBinder.Eval(\"" + veld.Name + "\")",
//};
}
else if (veld.PropertyType == typeof(DateTime))
{
field = new TemplateField();
//field.DatePicker = true;
}
else
{
field = new BoundField();
((BoundField)field).DataField = veld.Name;
}

field.HeaderText = veld.Name;

return field;
}

protected void OnRowDataBound(object sender, GridViewRowEventArgs e)
{
if (e.Row.RowType == DataControlRowType.DataRow)
{

}
}
}
}

最佳答案

我不太确定你想要达到什么目的,我只知道你所做的从根本上是错误的(好吧,就 ASP.NET 的世界观而言......)。

  1. 您将控件添加到输入数据,而不是将其添加为网格的子控件
  2. 您没有在控件的构造函数中实例化 GridView 控件。
  3. Session 绝对是存储大量页面相关数据的错误位置。

我将从最后一点开始:如果这是需要在两次访问站点之间保留的数据,那么您必须将其放入数据库中。如果这是仅从某人登录时到他们注销时存在的数据,那么是的,Session 可能是合适的地方。否则,如果它特定于页面并且在用户访问另一个页面时应该被丢弃,那么您应该每次都从数据库中重新加载它或者也许将它存储在 ViewState 中。 .

接下来,是否所有的对象都是同一类型/它们是否具有相同的字段?如果是这样,那么默认行为(由 AutoGenerateColumns 显式控制)将为您完成这项工作,无需额外工作:

<asp:GridView runat="server"
ID="grid"
AutoGenerateColumns="true" />

如果有相同的列,那么它们应该在单独的网格中;一个GridView是一种创建 HTTP <table> 的方法元素。表格元素应该只包含相关数据;您不会使用一个表格来显示鱼的价格和汽车的颜色。由此得出结论,如果您有不同的表和不相关的数据,那么您应该有不同的数据源……一个更简单的解决方案,这意味着您不需要实现您的控制正在尝试实现。

最后,为了完整起见,当您定义一个控件时,您只是在创建一个类。如果您希望能够以您尝试的方式实例化控件,那么您需要确保其所有数据成员都在构造函数中实例化任何引用都由 null- 保护引用检查:

if (grid != null)
{
// Do stuff with grid
}

关于c# - 为什么我的自定义控件的组件没有启动?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21337312/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com