動的に作成されたLinkButtonはコマンドイベントには入りません
-
28-09-2019 - |
質問
私は前にこれを尋ねましたが、悲しいことに私はまだ問題を抱えており、問題は解決しませんでした。基本的に、生成しているテーブルの各行のリンクボタンを動的に作成しています。そのボタンには、データベースから対応するIDを使用して行を削除するタスクがあります。これを行うには、リンクボタンにコマンドを割り当てる必要があるように見えます。そうすれば、クリックするとイベントに入ります。問題は、ボタンがプログラムをクリックしたときです 一度もない コマンドに入ります - 私はそこにブレークポイントを入れました、そしてそれは彼らに決して行きません。これが私のコードです:
protected void Page_Init(object sender, EventArgs e)
{
if (Request.QueryString["id"] != null)
{
ColorConverter conv = new ColorConverter();
string connection = ConfigurationManager.ConnectionStrings["TPRTestConnectionString"].ConnectionString;
TPRDBDataContext dc = new TPRDBDataContext();
DataContext db = new DataContext(connection);
Table<SageAccount> SageAccount = db.GetTable<SageAccount>();
Table<InvoiceItem> InvoiceItem = db.GetTable<InvoiceItem>();
Table<Invoice> Invoice = db.GetTable<Invoice>();
Boolean alloweditting = (from s in dc.Invoices where s.id.ToString() == Request.QueryString["id"] select s.alloweditting).Single();
if (alloweditting == false)
{
dtlsInsert.Visible = false;
modalPanel.Visible = false;
}
int sagepk = (from s in dc.Invoices where s.id.ToString() == Request.QueryString["id"] select s.sageaccount).Single();
lblSageID.Text = (from s in dc.SageAccounts where s.ID == sagepk select s.SageID).Single();
lblDate.Text = DateTime.Now.ToShortDateString();
Table table = new Table();
table.Width = Unit.Percentage(100);
table.GridLines = (GridLines)3;
TableHeaderRow header = new TableHeaderRow();
header.BackColor = (System.Drawing.Color)conv.ConvertFromString("#EDEDED");
foreach (string header2 in new string[] { "", "Quantity", "Rate", "Description", "Nominal Code", "Subtotal" })
{
TableCell cell = new TableCell();
cell.Text = header2;
header.Cells.Add(cell);
}
table.Rows.Add(header);
var data = (from s in dc.InvoiceItems where s.invoiceid.ToString() == Request.QueryString["id"].ToString() select s);
foreach (var x in data)
{
TableRow row = new TableRow();
if (x.invoicetext == null)
{
decimal total;
try
{
total = (decimal)x.rate * (decimal)x.quantity;
}
catch
{
total = 0;
}
int i = 0;
foreach (string columnData in new string[] { x.id.ToString(), x.quantity.ToString(), x.rate.ToString(), x.description, x.nominalcode, total.ToString("N2") })
{
TableCell cell = new TableCell();
{
if (i == 0)
{
LinkButton lnkdel = new LinkButton();
lnkdel.Text = "Delete";
lnkdel.ID = "lnkDel" + Guid.NewGuid();
if (alloweditting == false)
{
lnkdel.Enabled = false;
}
lnkdel.Font.Bold = false;
lnkdel.CommandArgument = x.id.ToString();
//lnkdel.Command += lnkdel_Command;
//lnkdel.Command += new CommandEventHandler(this.lnkdel);
cell.Controls.Add(lnkdel);
i++;
}
else
{
cell.Text = columnData;
}
}
row.Cells.Add(cell);
}
runningtotal = runningtotal + total;
}
else
{
int i = 0;
foreach (string columnData in new string[] { x.id.ToString(), x.invoicetext })
{
TableCell cell = new TableCell();
if (i == 0)
{
LinkButton lnkdel = new LinkButton();
lnkdel.Text = "Delete";
lnkdel.ID = "lnkDel" + Guid.NewGuid();
if (alloweditting == false)
{
lnkdel.Enabled = false;
}
lnkdel.Font.Bold = false;
//lnkdel.Command += lnkdel_Command;
//lnkdel.Command += new CommandEventHandler(this.lnkdel);
lnkdel.CommandArgument = x.id.ToString();
cell.Controls.Add(lnkdel);
i++;
}
else
{
cell.Text = columnData;
cell.ColumnSpan = 5;
}
row.Cells.Add(cell);
}
}
switch (x.formatoptions)
{
case 1:
row.ForeColor = (System.Drawing.Color)conv.ConvertFromString("black");
row.Font.Bold = false;
break;
case 2:
row.ForeColor = (System.Drawing.Color)conv.ConvertFromString("black");
row.Font.Bold = true;
break;
case 3:
row.ForeColor = (System.Drawing.Color)conv.ConvertFromString("red");
row.Font.Bold = false;
break;
case 4:
row.ForeColor = (System.Drawing.Color)conv.ConvertFromString("red");
row.Font.Bold = true;
break;
}
table.Rows.Add(row);
}
TableFooterRow row2 = new TableFooterRow();
TableCell cell2 = new TableCell();
cell2.Text = "<span style\"text-align: right; width: 100%;\">Total = <b>" + runningtotal.ToString("N2") + "</b></span>";
cell2.ColumnSpan = 6;
row2.Cells.Add(cell2);
table.Rows.Add(row2);
var update = (from s in dc.Invoices where s.id.ToString() == Request.QueryString["id"] select s).Single();
update.total = runningtotal;
dc.SubmitChanges();
datatable.Controls.Clear();
datatable.Controls.Add(table);
}
else
{
Response.Redirect("Invoices.aspx");
}
}
protected void Page_Load(object sender, EventArgs e)
{
}
protected void lnkdel_Command(object sender, CommandEventArgs e)
{
string connection = ConfigurationManager.ConnectionStrings["TPRTestConnectionString"].ConnectionString;
using (SqlConnection conn = new SqlConnection(connection))
{
SqlCommand comm = new SqlCommand("DELETE FROM InvoiceItem WHERE id = @id", conn);
comm.Parameters.AddWithValue("@id", e.CommandArgument.ToString());
conn.Open();
try
{
comm.ExecuteNonQuery();
}
catch (Exception ex)
{
Response.Write(ex);
}
}
}
注私はここに投稿するための2つの重要な行をコメントしました。私はコメントした行の両方を試したことを指摘するために、どちらも機能しないことを指摘します:(
解決
コントロールを追加する必要があります 毎日 ポストバック。最初のGet(そのクエリ文字列チェック)でのみ作成しているように見えます。ポストバックでは、これらのコントロールが再現されることはないため、イベントは発生しません。
それは非常に直感的であることで有名ですが、ASP.NETは2つのHTTP要求の間でページクラスのインスタンスが同じであると思わせるために逆方向に曲がりますが、現実は同じではないということです。毎回新しいインスタンスが作成されます。動的に生成されたコントロールを複数回追加しないようにしようとしているように見えます - 重複したくないと考えています。現実はあなたがそうすることです 一度もない 次のようなライフサイクル方法で動的に生成されたコントロールを追加するときに複製を取得します OnInit()
常にページクラスの新しいインスタンスであり、したがって動的に生成されたコントロールがなくなっているためです。
これが通常開発者に透明である理由は、コードフロントのすべてのコントロールが、最初の要求とすべてのポストバックの両方で自動的に再生されるためです。動的に作成されたコントロールについては、このラインがあります。
if (Request.QueryString["id"] != null) { ... }
特別なことをしていない限り、その「ID」属性は、ポストバックのクエリ文字列には含まれません。これは、のコードのいずれもないことを意味します if
ブロックはポストバックで実行されます(イベントが実際に発砲するとき)これはあなたのことを意味します if
- 上部のチェックは完全に削除する必要があります。 全て そのコードはすべてのリクエストに対して実行する必要があります(取得します と 役職)。
他のヒント
回避策を作成したと言って - ページへの単純なリンクを作成し、削除する行のIDを含むクエリ文字列を作成するだけです