繁体   English   中英

从动态创建的TextBox ASP.NET中读取值(在表内)

[英]Reading values from dynamically created TextBox ASP.NET (Within table)

所以我有一个List<SettingsValue> ,我将使用它来填充<table runat="server">System.Web.UI.HtmlControls.HtmlTable )。

问题在于,提交表单时,将使用初始值重新创建字段。 然后,当我遍历所有元素时,它们都没有改变。 但是,如果我不重新创建它们,那么检查输入值的代码甚至找不到控件。

我不想通过查询字符串发布数据

我通过调用带有参数“ id”的GetSettingsFromDatabase方法来接收设置。 ID是具有设置的“应用程序”的键。 在页面上有一个组合框,用户可以在其中选择要为其更改设置的应用程序。 当在组合框中选择一个项目时,发生回发,然后我希望再次调用LoadSettings()。


HTML-行:

<tr align="left" valign="top" runat="server">
    <th runat="server">
        <asp:Label runat="server" Text="{field.Name}" />
    </th>
    <td runat="server">
        <asp:TextBox ID="txt{field.ID}_{field.Name}" runat="server" Text="{value}" />
    </td>
</tr>


C#-模型:

public class SettingsValue
{
    public long ID { get; set; }
    public long FieldID { get; set; }
    public string Value { get; set; }
    public string RelatedID { get; set; }

    public SettingsField Field { get; set; }
}

public class SettingsField
{
    public long ID { get; set; }
    public long CategoryID { get; set; }
    public string Name { get; set; }
    public string DataType { get; set; }
    public bool Required { get; set; }
    public bool Visible { get; set; }

    public SettingsCategory Category { get; set; }
}

public class SettingsCategory
{
    public SettingsCategory()
    {
        Fields = new List<SettingsField>();
    }

    public long ID { get; set; }
    public string Name { get; set; }
    public string Provider { get; set; }
    public SettingsCategoryType Type { get; set; }

    public List<SettingsField> Fields { get; private set; }
}


C#-行生成:

private void LoadSettings(long id)
{
    // The id represents some application that has settings

    List<SettingsValue> setting = new List<SettingsValue>();
    if (id > 0)
        settings = GetSettingsFromDatabase(id);

    RenderSettings(settings);
}

private void RenderSettings(List<SettingsValue> settings)
{
    table.Rows.Clear();

    foreach (var value in settings)
    {
        var field = value.Field;
        var fieldGUI_ID = string.Format("{0}_{1}", field.ID, field.Name);

        var row = new HtmlTableRow();
        row.Attributes.Add("runat", "server");
        row.Align = "left";
        row.VAlign = "top";

        var th = new HtmlTableCell("th");
        th.Attributes.Add("runat", "server");

        var td = new HtmlTableCell("td");
        td.Attributes.Add("runat", "server");

        var input = new TextBox();
        input.ID = "txt" + fieldGUI_ID;
        input.Text = value.Value;
        input.Width = new Unit("305px");
        input.Attributes.Add("runat", "server");

        var label = new Label();
        label.Text = field.Name;
        label.Attributes.Add("for", input.ClientID);


        th.Controls.Add(label);
        row.Cells.Add(th);

        td.Controls.Add(input);
        row.Cells.Add(td);

        table.Rows.Add(row);
    }
}


C#-页面加载

protected void Page_Load( object sender, EventArgs e )
{
    if (!Page.IsPostBack )
    {
        LoadSettings(-1);
    }
}


C#-提交时

我们将ComponentArt库用于回调<ComponentArt:Callback ID="callSettings" runat="server" OnCallback="callSettings_OnCallback" /> ,但这与常规ASP.NET没什么不同。

protected void callSettings_OnCallback( object sender, ComponentArt.Web.UI.CallBackEventArgs e )
{
    long id = 421;
    List<SettingsValue> settings = GetSettingsFromDatabase(id);

    List<SettingsValue> updatedValues = new List<SettingsValue>();
    foreach (SettingsField f in fields)
    {
        var val = GetSettingValueFromInput(f);
        if (val != null)
            updatedValues.Add(val);
    }

    UpdateSettingsToDatabase(updatedValues);
}

private SettingsValue GetSettingValueFromInput(SettingsField field)
{
    var fieldGUI_ID = "txt" + string.Format("{0}_{1}", field.ID, field.Name);
    var res = settings.Single(x => x.FieldID == field.ID);

    foreach (HtmlTableRow row in table.Rows)
    {
        foreach (HtmlTableCell cell in row.Cells)
        {
            var ctrl = cell.FindControl(fieldGUI_ID);
            var input = ctrl as TextBox;
            if (ctrl == null || input == null)
                continue;

            res.Value = input.Text;
            return res;
        }
    }
    return res;
}

我研究了其他几个问题,但还没有解决。

动态创建控件时,必须仅在第一次时初始化该值。 像这样

PageLoad() {
    var txtBox = new TextBox();
    this.Controls.Add(txtBox);
    if (!this.IsPostBack) {
        txtBox.Text = "this is initial value";
    }
}

现在您可以看到,仅在首次加载页面时设置初始值(无回发)。 下次将页面回发时,控件将重新创建,并且该值将由页面自动设置。

对于动态控件,每次加载页面时(无论是回发还是非回发),您都必须创建和添加控件。 但是您不必每次都设置初始值。 仅在首次加载Page时设置初始值。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM