共用方式為


將驗證控制項新增至 DataList 的編輯介面 (C#)

作者:Scott Mitchell

下載 PDF

在本教學課程中,我們將瞭解如何將驗證控件新增至 DataList 的 EditItemTemplate,以提供更萬無一失的編輯使用者介面。

簡介

到目前為止,在 DataList 編輯教學課程中,即使遺漏產品名稱或負價等無效的使用者輸入,DataLists 編輯介面也沒有包含任何主動式使用者輸入驗證。 在上一個教學課程中,我們已檢查如何將例外狀況處理程式代碼新增至 DataList 的UpdateCommand事件處理程式,以便擷取和正常顯示所引發之任何例外狀況的相關信息。 不過,在理想情況下,編輯介面會包含驗證控件,以防止使用者第一次輸入這類無效的數據。

在本教學課程中,我們將瞭解如何將驗證控件新增至 DataList s EditItemTemplate ,以提供更萬無一失的編輯使用者介面。 具體而言,本教學課程會採用在上述教學課程中建立的範例,並增強編輯介面以包含適當的驗證。

步驟 1:從處理 BLL 和 DAL 層級例外狀況複寫範例

處理 BLL 和 DAL 層級例外狀況 教學課程中,我們建立了一個頁面,其中列出兩欄可編輯 DataList 中產品的名稱和價格。 本教學課程的目標是要增強 DataList 的編輯介面,以包含驗證控件。 特別是,我們的驗證邏輯會:

  • 要求提供產品名稱
  • 確定針對價格輸入的值是有效的貨幣格式
  • 確定針對價格輸入的值大於或等於零,因為負 UnitPrice 值是非法的

我們必須先將範例從 ErrorHandling.aspx EditDeleteDataList 資料夾中的頁面複寫到本教學 UIValidation.aspx課程的頁面,才能查看增強先前的範例以包含驗證。 若要達成此目的,我們需要複製 ErrorHandling.aspx 頁面的宣告式標記和其原始程式碼。 執行下列步驟,先複製宣告式標記:

  1. ErrorHandling.aspx在 Visual Studio 中開啟頁面
  2. 移至頁面的宣告式標記(按下頁面底部的[來源] 按鈕)
  3. 複製和 </asp:Content> 標記內的<asp:Content>文字(第 3 到 32 行),如圖 1 所示。

複製 asp:Content> 控件內的<文字

圖 1:複製控制元件內的 <asp:Content> 文字 (按兩下以檢視完整大小的影像

  1. UIValidation.aspx開啟頁面
  2. 移至頁面的宣告式標記
  3. 在控件中貼上 <asp:Content> 文字。

若要複製原始程式碼,請開啟ErrorHandling.aspx.vb頁面,並只複製 類別內的EditDeleteDataList_ErrorHandling文字。 複製三個事件處理程式 (、 和 ) 與 DisplayExceptionDetails 方法,但請勿複製類別宣告或using語句。Products_UpdateCommandProducts_CancelCommandProducts_EditCommand 在類別UIValidation.aspx.vb貼上複製的EditDeleteDataList_UIValidation文字

將內容和程式代碼從 ErrorHandling.aspx UIValidation.aspx移至 之後,請花點時間在瀏覽器中測試頁面。 您應該會在這兩個頁面中看到相同的輸出,並體驗相同的功能(請參閱圖 2)。

UIValidation.aspx 頁面會模擬ErrorHandling.aspx中的功能

圖 2:頁面 UIValidation.aspx 模擬 中的 ErrorHandling.aspx 功能(按兩下以檢視完整大小的影像

步驟 2:將驗證控件新增至 DataList s EditItemTemplate

建構數據輸入表單時,用戶必須輸入任何必要欄位,而且其所有提供的輸入都是合法且格式正確的值。 為了協助確保使用者的輸入有效,ASP.NET 提供五個內建驗證控件,其設計目的是驗證單一輸入Web控件的值:

如需這五個控件的詳細資訊,請參閱將驗證控件新增至編輯和插入介面教學課程,或參閱 ASP.NET 快速入門教學課程驗證控件一節。

在本教學課程中,我們需要使用 RequiredFieldValidator 來確保已提供產品名稱的值,以及 CompareValidator,以確保輸入的價格的值大於或等於 0,並以有效的貨幣格式呈現。

注意

雖然 ASP.NET 1.x 具有相同的五個驗證控件,ASP.NET 2.0 新增了許多改進,但除了 Internet Explorer 之外,主要兩個是瀏覽器的用戶端腳本支援,以及將頁面上的驗證控件分割成驗證群組的功能。 如需 2.0 中新驗證控件功能的詳細資訊,請參閱剖析 ASP.NET 2.0 中的驗證控件。

讓我們從將必要的驗證控件新增至 DataList s EditItemTemplate開始。 您可以透過設計工具執行這項工作,方法是單擊 DataList 智慧標記中的 [編輯範本] 連結,或透過宣告式語法來執行。 讓我們從 [設計] 檢視使用 [編輯範本] 選項逐步執行程式。 選擇編輯 DataList s EditItemTemplate之後,將 RequiredFieldValidator 從 [工具箱] 拖曳至範本編輯介面,將它放在 TextBox 之後 ProductName

將 RequiredFieldValidator 新增至 ProductName TextBox 之後的 EditItemTemplate

圖 3:將 RequiredFieldValidator 新增至 EditItemTemplate After ProductName TextBox (按兩下以檢視完整大小的影像

所有驗證控件的運作方式是驗證單一 ASP.NET Web 控件的輸入。 因此,我們需要指出我們剛才新增的 RequiredFieldValidator 應該驗證 ProductName TextBox;這是藉由將驗證控件的 ControlToValidate 屬性 設定為 ID 適當 Web 控件的 ,來完成此動作(ProductName在此實例中為 )。 接下來,將 ErrorMessage 屬性 設定為 [您必須提供產品名稱],並將 Text 屬性 設定為 *。 如果提供,屬性值 Text 就是驗證控件在驗證失敗時所顯示的文字。 ErrorMessage ValidationSummary 控件會使用所需的屬性值;如果Text省略屬性值,ErrorMessage則無效輸入上的驗證控件會顯示屬性值。

設定 RequiredFieldValidator 的這三個屬性之後,您的畫面看起來應該類似圖 4。

設定 RequiredFieldValidator s ControlToValidate、ErrorMessage 和 Text 屬性

圖 4:設定 RequiredFieldValidator s ControlToValidateErrorMessageText 屬性 (按兩下以檢視完整大小的影像

在 RequiredFieldValidator 新增至 EditItemTemplate之後,剩下的一切都是新增產品價格 TextBox 的必要驗證。 UnitPrice由於 在編輯記錄時是選擇性的,因此我們不需要新增 RequiredFieldValidator。 不過,我們確實需要新增 CompareValidator,以確保 UnitPrice所提供的 格式正確設定為貨幣,且大於或等於 0。

將 CompareValidator 新增至 , EditItemTemplate 並將其 屬性設定 ControlToValidateUnitPrice,其 ErrorMessage 屬性必須大於或等於零,且不能包含貨幣符號,且其 Text 屬性為 *。 若要指出UnitPrice值必須大於或等於 0,請將 CompareValidator s Operator 屬性設定為 GreaterThanEqual、將其ValueToCompare 屬性設定為 0,並將 其 Type 屬性設定為 。Currency

新增這兩個驗證控件之後,DataList 的 EditItemTemplate 宣告式語法看起來應該如下所示:

<EditItemTemplate>
    Product name:
        <asp:TextBox ID="ProductName" runat="server"
            Text='<%# Eval("ProductName") %>'></asp:TextBox>
        <asp:RequiredFieldValidator ID="RequiredFieldValidator1"
            ControlToValidate="ProductName"
            ErrorMessage="You must provide the product's name"
            runat="server">*</asp:RequiredFieldValidator>
    <br />
    Price:
        <asp:TextBox ID="UnitPrice" runat="server"
            Text='<%# Eval("UnitPrice", "{0:C}") %>'></asp:TextBox>
        <asp:CompareValidator ID="CompareValidator1"
            ControlToValidate="UnitPrice"
            ErrorMessage="The price must be greater than or equal to zero
                          and cannot include the currency symbol"
            Operator="GreaterThanEqual" Type="Currency" ValueToCompare="0"
            runat="server">*</asp:CompareValidator><br />
    <br />
    <asp:Button ID="UpdateProduct" runat="server" CommandName="Update"
        Text="Update" /> 
    <asp:Button ID="CancelUpdate" runat="server" CommandName="Cancel"
        Text="Cancel" />
</EditItemTemplate>

進行這些變更之後,請在瀏覽器中開啟頁面。 如果您在編輯產品時嘗試省略名稱或輸入無效的價格值,則文本框旁邊會出現星號。 如圖 5 所示,包含貨幣符號的價格值,例如 $19.95 會被視為無效。 CompareValidator 允許CurrencyType數位分隔符(例如逗號或句點,視文化特性設定而定),以及前置加號或減號,但不允許貨幣符號。 此行為可能會讓使用者感到困惑,因為編輯介面目前會使用貨幣格式轉譯 UnitPrice

文字框旁出現星號,輸入無效

圖 5:星號出現在無效輸入的文字框旁邊(按兩下以檢視完整大小的影像

當驗證依現成運作時,使用者編輯記錄時必須手動移除貨幣符號,這是無法接受的。 此外,如果編輯介面中沒有無效的輸入,則按兩下 [更新] 或 [取消] 按鈕時,將會叫用回傳。 在理想情況下,不論使用者輸入的有效性為何,[取消] 按鈕都會將DataList傳回其預先編輯狀態。 此外,我們需要確保頁面的數據在更新 DataList UpdateCommand 事件處理程式中的產品資訊之前有效,因為驗證控件用戶端邏輯可由瀏覽器不支援 JavaScript 或停用其支援的使用者略過。

從 EditItemTemplate s UnitPrice TextBox 移除貨幣符號

使用 CompareValidator s Currency``Type時,正在驗證的輸入不得包含任何貨幣符號。 這類符號的存在會導致 CompareValidator 將輸入標示為無效。 不過,我們的編輯介面目前在 TextBox 中包含 UnitPrice 貨幣符號,這表示用戶必須先明確移除貨幣符號,才能儲存變更。 若要解決此問題,我們有三個選項:

  1. EditItemTemplate設定 ,讓 UnitPrice TextBox 值不會格式化為貨幣。
  2. 允許使用者藉由移除 CompareValidator 並取代為正則ExpressionValidator 來輸入貨幣符號,以檢查格式正確的貨幣值。 此處的挑戰是驗證貨幣值的正則表達式不像 CompareValidator 那麼簡單,而且如果我們想要納入文化特性設定,則需要撰寫程序代碼。
  3. 完全移除驗證控制項,並依賴 GridView RowUpdating 事件處理程式中的自訂伺服器端驗證邏輯。

讓我們使用本教學課程的選項 1。 目前 格式 UnitPrice 為貨幣值,因為 中的 EditItemTemplateTextBox 數據系結表示式: <%# Eval("UnitPrice", "{0:c}") %>。 將 Eval 語句變更為 Eval("UnitPrice", "{0:n2}"),其會將結果格式化為具有兩位數有效位數的數位。 這可以直接透過宣告式語法完成,或按兩下 DataList 中 EditItemTemplateTextBox 的 [編輯 DataBindings] 連結UnitPrice

有了這項變更,編輯介面中的格式化價格會包含逗號做為群組分隔符,而句點則為小數分隔符,但會離開貨幣符號。

注意

從可編輯的介面移除貨幣格式時,我覺得將貨幣符號放在 TextBox 外部會很有説明。 這可作為使用者不需要提供貨幣符號的提示。

修正取消按鈕

根據預設,驗證 Web 控制件會發出 JavaScript,以在用戶端上執行驗證。 按兩下Button、LinkButton或ImageButton時,頁面上的驗證控件會在回傳發生之前,於客戶端檢查。 如果有任何無效的數據,則會取消回傳。 不過,對於某些按鈕而言,數據的有效性可能並不重要;在這種情況下,由於無效的數據而取消回傳是一種麻煩。

[取消] 按鈕就是這樣的範例。 假設使用者輸入無效的數據,例如省略產品名稱,然後決定她不想全部儲存產品,並點擊 [取消] 按鈕。 目前,[取消] 按鈕會觸發頁面上的驗證控件,其會報告產品名稱遺失並防止回傳。 我們的用戶必須輸入一些文字到 ProductName TextBox 中,只是為了取消編輯程式。

幸運的是,Button、LinkButton 和 ImageButton 有一個 CausesValidation 屬性 ,可以指出是否按下按鈕應該起始驗證邏輯(預設值為 True)。 將 [取消按鈕] 屬性 CausesValidation 設定為 False

確保 UpdateCommand 事件處理程式中的輸入有效

由於驗證控件發出的用戶端腳本,如果使用者輸入無效的輸入,驗證控件會取消 Button、LinkButton 或 ImageButton 控件所起始的任何回傳,其 CausesValidation 屬性為 True (預設值)。 不過,如果使用者正在流覽已過時的瀏覽器或已停用 JavaScript 支援的瀏覽器,客戶端驗證檢查將不會執行。

所有 ASP.NET 驗證控件都會在回傳時立即重複其驗證邏輯,並透過 Page.IsValid 屬性報告頁面輸入的整體有效性。 不過,根據的值 Page.IsValid,頁面流程不會以任何方式中斷或停止。 身為開發人員,我們有責任確保 Page.IsValid 屬性具有 值 True ,再繼續進行假設有效輸入數據的程序代碼。

如果使用者已停用 JavaScript,請瀏覽我們的頁面、編輯產品、輸入價格值太貴,然後按兩下 [更新] 按鈕,將會略過客戶端驗證,且回傳將會隨之而來。 在回傳時,ASP.NET 頁面的 UpdateCommand 事件處理程式會執行,並在嘗試剖析太昂貴至 時引發例外狀況 Decimal。 由於我們有例外狀況處理,因此會正常處理這類例外狀況,但是我們可以避免無效的數據在第一個位置滑過,只要具有的值True,才能繼續進行UpdateCommand事件處理程式Page.IsValid

將下列程式代碼新增至事件處理程式的 UpdateCommand 開頭,緊接在 Try 區塊之前:

if (!Page.IsValid)
    return;

此外,只有在送出的數據有效時,才會嘗試更新產品。 大部分的使用者都因驗證控制用戶端腳本而無法回傳無效的數據,但是瀏覽器不支援 JavaScript 或已停用 JavaScript 的使用者,可以略過客戶端檢查並提交無效的數據。

注意

精明的讀取器會記得,使用 GridView 更新數據時,我們不需要在頁面的程式代碼後置類別中明確檢查 Page.IsValid 屬性。 這是因為 GridView 會查閱 Page.IsValid 我們的 屬性,而且只有在傳回 的值 True時,才會繼續進行更新。

步驟 3:摘要數據輸入問題

除了五個驗證控制項之外,ASP.NET 還包含 ValidationSummary 控制件,其中顯示 ErrorMessage 偵測到無效資料的驗證控制件的 。 此摘要數據可以顯示為網頁上的文字,或透過強制回應的用戶端消息框來顯示。 讓我們增強本教學課程,以包含摘要任何驗證問題的用戶端消息框。

若要達成此目的,請將 ValidationSummary 控件從工具箱拖曳至設計工具。 ValidationSummary 控件的位置並不重要,因為我們打算將它設定為只將摘要顯示為消息框。 新增 控制項之後,將其 屬性設定為 False ,並將其 ShowMessageBox 屬性設定ShowSummaryTrue。 此外,用戶端消息框中會摘要說明任何驗證錯誤(請參閱圖 6)。

驗證錯誤摘要於用戶端消息框中

圖 6:驗證錯誤摘要於用戶端訊息框中(按兩下以檢視完整大小的影像

摘要

在本教學課程中,我們瞭解如何使用驗證控件來主動確保使用者輸入有效,然後再嘗試在更新工作流程中使用它們,以減少例外狀況的可能性。 ASP.NET 提供五個驗證 Web 控制項,其設計目的是檢查特定 Web 控件的輸入,並回報輸入的有效性。 在本教學課程中,我們使用這五個控件的兩個控件 RequiredFieldValidator 和 CompareValidator 來確保已提供產品名稱,而且價格具有大於或等於零值的貨幣格式。

將驗證控件新增至 DataList 的編輯介面很簡單,只要從 [工具箱] 將控件拖曳到 EditItemTemplate ,並設定一些屬性即可。 根據預設,驗證控件會自動發出客戶端驗證腳本;它們也會在回傳時提供伺服器端驗證,並將累計結果儲存在屬性中 Page.IsValid 。 若要在單擊 Button、LinkButton 或 ImageButton 時略過客戶端驗證,請將 button s CausesValidation 屬性設定為 False。 此外,在使用回傳時送出的數據執行任何工作之前,請確定 Page.IsValid 屬性會傳 True回 。

到目前為止,我們檢查的所有 DataList 編輯教學課程都有非常簡單的編輯介面 TextBox 作為產品名稱,另一個用於價格。 不過,編輯介面可以包含不同 Web 控件的混合,例如 DropDownLists、Calendars、RadioButtons、CheckBoxes 等等。 在下一個教學課程中,我們將探討如何建置使用各種 Web 控件的介面。

祝您程式設計愉快!

關於作者

Scott Mitchell,七本 ASP/ASP.NET 書籍的作者和 4GuysFromRolla.com 創始人,自 1998 年以來便開始使用 Microsoft Web 技術。 Scott 擔任獨立顧問、講師和作家。 他的新書是 Sams Teach Yourself ASP.NET 2.0 in 24 Hours。 您可以透過 mitchell@4GuysFromRolla.com 或他的部落格 (可以在 http://ScottOnWriting.NET 找到) 與他聯繫。

特別感謝

本教學課程系列已經過許多熱心的檢閱者檢閱。 本教學課程的主要檢閱者是 Dennis Patterson、Ken Pespisa 和 Liz Shulok。 有興趣檢閱我即將推出的 MSDN 文章嗎? 如果有,請發信到 mitchell@4GuysFromRolla.com 。