3

FormView内にascxコントロールがあります。私が欲しいのは、ascx内で構文Bindを使用することです。これが私のページです:

    <asp:ObjectDataSource runat="server" ID="ods" TypeName="MyDS" SelectMethod="Get" UpdateMethod="Update" DataObjectTypeName="Ent">
    </asp:ObjectDataSource>
    <asp:FormView runat="server" DefaultMode="Edit" ID="fv1" DataSourceID="ods">
        <EditItemTemplate>
            <uc1:WebUserControl ID="WebUserControl1" runat="server" />
            <asp:Button runat="server" CommandName="Update" Text="Update"/>
        </EditItemTemplate>
    </asp:FormView>

WebUserControl.ascxは次のとおりです。

<asp:TextBox ID="txt1" runat="server" Text='<%# Bind("Name") %>' />

TextBoxに値を選択すると、すべてが正常に機能します。バインドにより、テキストボックスに期待値が入力されます。ただし、[更新]ボタンを押すと、ObjectDataSourceのメソッドUpdateは、入力されたテキストが期待されている間に、Nameではなくnullを持つEntのインスタンスを取得します。テストのために、テキストボックスを.aspxに配置しましたが、すべて正常に機能します。

最後に、リフレクターFormViewで逆コンパイルしました。直接の子に対してのみ反復するため、ExtractRowValuesは失敗したように見えます。子バインディングを回避する方法を知っている人はいますか?

4

2 に答える 2

3

これが少し遅れている場合は申し訳ありません:-)

「バインド」は奇妙な獣です。Bind メソッドはどこにもありません。ASP.NETコンパイラにデータバインディングに「Eval」呼び出し(実際にはEvalメソッドがあります)を追加し、デュアルウェイバインディング用の特定の抽出コード隠しメソッドを生成するように指示するのは、単なる特定のASP.NETトークンです。

ただし、このメソッドは、 ITemplateプロパティ (FormView の ItemTemplate、EditItemTemplate など) を備え、双方向バインド可能と宣言されているコントロールに対してのみ追加 (および抽出時に呼び出) されます。

残念ながら、ユーザー コントロール (ascx) の場合、そのようなメソッドを簡単に生成する方法はありません。

ただし、ユーザー コントロールに実装できるIBindableControl インターフェイスがあります。これにより、次のように自動化や宣言性は低くなりますが、ある程度の統合が可能になります。

WebUserControl.ascx:

<asp:TextBox ID="txt1" runat="server" Text='<%# Eval("Name") %>' />

(Bind は役に立たないので、Eval に固執することができます)。

WebUserControl.ascx.cs:

public partial class WebUserControl : UserControl, IBindableControl
{
    public void ExtractValues(IOrderedDictionary dictionary)
    {
        dictionary["Name"] = txt1.Text;
    }
}
于 2012-12-04T13:36:15.390 に答える
2

<%# Bind("") %>構文をユーザー コントロール (ASCX) からそれを含むページ (ASPX) に移動し、ユーザー コントロールでプロパティを使用してフォーム値を取得および設定する必要があります。以下の私の例は、 http://oudinia.blogspot.co.uk/2007/12/aspnet-20-and-up-c-user-control-within.htmlで与えられた解決策から、さらにいくつかの詳細を試すように変更されています現実的にするために:

ASCX マークアップ:

<%@ Control Language="C#" AutoEventWireup="true" CodeFile="PaperForm.ascx.cs" Inherits="PaperForm" %>

<asp:ValidationSummary runat="server" HeaderText="<b>Please supply the missing information below.</b>" />

<p class="SectionHeading">Section A: Your Details</p>
<table border="1" width="100%">
  <tr>
    <td width="220px">Membership number</td>
    <td colspan="3"><asp:TextBox ID="txtMemNo" runat="server" MaxLength="9"></asp:TextBox>
        <asp:RegularExpressionValidator runat="server" ControlToValidate="txtMemNo" Display="Dynamic" Text="Please check your membership number" ValidationExpression="\d{9}"></asp:RegularExpressionValidator>
    </td>
  </tr>
  <tr>
    <td><asp:Label ID="lblFirstName" runat="server">First name</asp:Label></td>
    <td><asp:TextBox ID="txtForename" runat="server"></asp:TextBox>
        <asp:RequiredFieldValidator runat="server" ControlToValidate="txtForename" Text="Required"></asp:RequiredFieldValidator>
    </td>
    <td width="110px"><asp:Label ID="lblLastName" runat="server">Last name</asp:Label></td>
    <td><asp:TextBox ID="txtSurname" runat="server"></asp:TextBox>
        <asp:RequiredFieldValidator runat="server" ControlToValidate="txtSurname" Text="Required"></asp:RequiredFieldValidator>
    </td>
  </tr>
  ...
</table>

<script type="text/javascript" language="javascript">
  // perform additional client-side validation in JavaScript
  function ValidateForm()
  {
    // check membership number (via web-service)
    ...
  }
</script>

ASCX コード ビハインド:

public partial class PaperForm : UserControl
{
    // **public properties representing form data**
    public string MemNo { get { return txtMemNo.Text; } set { txtMemNo.Text = value; } }
    public string Forename { get { return txtForename.Text; } set { txtForename.Text = value; } }
    public string Surname { get { return txtSurname.Text; } set { txtSurname.Text = value; } }
    ...

    protected void Page_Load(object sender, EventArgs e)
    {
        // prevent browser caching
        ...
    }
}

ASPX マークアップ:

<%--register the user control here--%>
<%@ Register TagPrefix="UC" TagName="PaperForm" Src="~/Proposals/PaperForm.ascx" %>

<asp:FormView ID="fvPaper" runat="server" DataKeyNames="PprPropID" DataSourceID="tblPprProp" DefaultMode="Insert" Width="780px">    
  <InsertItemTemplate>

    <%--insert the user control here, **and bind the database fields to its properties**--%>
    <UC:PaperForm ID="pf" runat="server" MemNo='<%# Bind("MemNo") %>' Forename='<%# Bind("Forename") %>' Surname='<%# Bind("Surname") %>' ... />
    ...
    <%--can use the JavaScript ValidateForm() function defined in the ASCX file--%>
    <asp:Button ID="InsertButton" runat="server" CausesValidation="True" CommandName="Insert" Text="Submit" BackColor="#C2D9EC" OnClientClick="return ValidateForm();" />
  </InsertItemTemplate>
</asp:FormView>

<%--define the data-source here rather than in the ASCX file--%>
<asp:SqlDataSource ID="tblPprProp" runat="server" ConflictDetection="CompareAllValues" ConnectionString="<%$ ConnectionStrings:confConnectionString %>" OnInserted="AfterInsertion" 
  InsertCommand="INSERT INTO [tblPprProp] ([MemNo], [Surname], [Forename], ...) VALUES (@MemNo, @Surname, @Forename, ...); SELECT @RowID = SCOPE_IDENTITY()" 
  OldValuesParameterFormatString="original_{0}">
  <InsertParameters>
    <asp:Parameter Name="MemNo" Type="String" />
    <asp:Parameter Name="Surname" Type="String" />
    <asp:Parameter Name="Forename" Type="String" />
    ...
  </InsertParameters>
</asp:SqlDataSource>

ASPX コード ビハインド:

public partial class PaperProposal : Page
{
    protected void Page_Load(object sender, EventArgs e)
    {
    }

    protected void AfterInsertion(object sender, SqlDataSourceStatusEventArgs e)
    {
        // get Proposal ID
        string rowId = e.Command.Parameters["@RowID"].Value.ToString();
        // get e-mail address from ASCX property
        string address = ((PaperForm)this.fvPaper.FindControl("pf")).Email;
        // send acknowledgement e-mail
        ...
    }
}
于 2012-07-13T18:40:45.553 に答える