基于属性隐藏整个数据列表项

Posted

技术标签:

【中文标题】基于属性隐藏整个数据列表项【英文标题】:Hide entire datalist item based on a property 【发布时间】:2021-12-16 09:35:10 【问题描述】:

我有一个数据列表,其中填充了数据库中的值。当用户单击复选框时,我想遍历所有数据列表项并隐藏所有具有属性 isActive = false 的项(在文本标签中显示为“已禁用”)。项目模板由一个表格组成,该表格包含多个其他项目、按钮等,以下未包括在内。因此,我想隐藏在特定项目的 itemtemplate 下找到的所有元素。

我的想法是通过访问其 id 来隐藏整个表,然后将表的可见属性设置为 false。当我运行代码时,这目前没有做任何事情。任何帮助都会 感激不尽!

 <asp:CheckBox runat="server" Text="Filter by active postings" OnCheckedChanged="filterByActive"/>
  <asp:DataList ID="postingsDataList" runat="server" OnItemCommand="itemCommand" >   

      <ItemTemplate>  
          <div class="postingRow">
              
      <table class="postingTable" id="posting">    
          
        <tr>  
               <td>     
               <asp:Label ID="lblActive"  runat="server" Text=<%#Eval("isActive").ToString().Equals("True") ? "Active   &#9989" : "Disabled   &#10060"%>/>  
          </td>  
        </tr>        
      </table>  

        </div>
        </ItemTemplate>  
  
    </asp:DataList>  

代码背后:

 protected void filterByActive (object sender, EventArgs e)
        
            int count = postingsDataList.Items.Count;
            CheckBox check = (CheckBox)sender;
            if (check.Checked == true)
                          
                for (int i = 0; i < count; i++)
                
                    Label lblActive = postingsDataList.Items[i].FindControl("lblActive") as Label;
                    string isActive = lblActive.Text.ToString();
                    if (isActive.Equals("Disabled   &#10060"))
                    
                        Table tbl = postingsDataList.Items[i].FindControl("posting") as Table;
                        tbl.Visible = false;
                       
                
            
            else
            
                for (int i = 0; i < count; i++)
                
                    Label lblActive = postingsDataList.Items[i].FindControl("lblActive") as Label;
                    string isActive = lblActive.Text.ToString();
                    if (isActive.Equals("Active   &#9989"))
                    
                        Table tbl = postingsDataList.Items[i].FindControl("posting") as Table;
                        tbl.Visible = true;
                    
                
            
        
    

【问题讨论】:

为什么不删除项目并刷新视图? 我有一个特定的问题,我不想将数据列表绑定到具有该过滤器的新 SQL 语句 我明白了。您不应该直接使用 SQL 的数据列表。一件事是您的数据,另一件事是您的视图的数据。您应该删除/编辑视图的数据。 我知道这不是一个好习惯,但它确实可以解决我遇到的问题...... 不是一个好的做法。是关于你试图解决一个你一开始不应该遇到的问题。就像这个问题是你自己的。如果您使用 MVC 模式,您将不需要任何代码,也不会遇到此问题。可能如果你编辑那个观点,你会浪费比修复这个逻辑的每个错误更少的时间。 【参考方案1】:

好的,我建议你这样做:

保留您的表 - 因此您不必再次访问 sql server。 (但是,你可以——不是世界末日)。

我没有您的数据,但我有一个简单的酒店列表 - 并且酒店有一个活动列。因此,让我们根据选中复选框过滤数据,而不必再次访问数据库服务器。更好的是,如果我们取消选中该框,我们可以显示我们拥有的所有记录。

我们当然假设这不是很多数据。

好的,所以我们有这个作为我们的标记:

(真的无所谓)

  <asp:DataList ID="DataList1" runat="server" DataKeyField="ID" RepeatColumns="4" RepeatDirection="Horizontal" >
    <ItemTemplate>
      <div style="border-style:solid;color:black;width:300px;">
        <div style="padding:5px;text-align:right">
            <p>Hotel Name: <asp:TextBox ID="HotelName" runat="server" Text ='<%# Eval("HotelName") %>' /></p>
            <p>First Name: <asp:TextBox ID="FirstName" runat="server" Text ='<%# Eval("FirstName") %>' /></p>
            <p>Last Name: <asp:TextBox ID="LastName" runat="server" Text ='<%# Eval("LastName") %>'    /></p>
            <p>City: <asp:TextBox ID="City" runat="server" Text ='<%# Eval("City") %>'  /></p>
            <p>Province: <asp:TextBox ID="Province" runat="server" Text ='<%# Eval("Province") %>'  /></p>
            Active: <asp:CheckBox ID="Active" runat="server" Checked = '<%# Eval("Active") %>'/>
        </div>
    </div>
</ItemTemplate>
</asp:DataList>

现在,我们要加载的代码可能是这样的:

    DataTable rstData = new DataTable();
    protected void Page_Load(object sender, EventArgs e)
    
        if (!IsPostBack)
        
            LoadData();
            ViewState["MyData"] = rstData;
        
        else
            rstData = (DataTable)ViewState["MyData"];
    

    void LoadData()
    
        using (SqlConnection conn = new SqlConnection(Properties.Settings.Default.TEST4))
        
            string strSQL = "SELECT top 10 * from tblHotels ORDER BY HotelName";
            using (SqlCommand cmdSQL = new SqlCommand(strSQL, conn))
            
                conn.Open();
                rstData.Load(cmdSQL.ExecuteReader());
                DataList1.DataSource = rstData;
                DataList1.DataBind();
            
        
    

现在我们有了这个:

现在,我们的过滤器显示/隐藏变成了这样:

   protected void ckFilter_CheckedChanged(object sender, EventArgs e)
    
        // filter based on check box

        if (ckFilter.Checked)
            rstData.DefaultView.RowFilter = "Active = 1";
        else
            rstData.DefaultView.RowFilter = "";

        DataList1.DataSource = rstData;
        DataList1.DataBind();
    

更好的是,如果我取消选中该复选框,所有数据都会重新显示。而且我们不必重新访问数据库来执行此操作!!!

现在,我们也可以只隐藏/显示每一行,然后如果由于某种原因您不想像我在上面所做的那样保留数据表,则取消隐藏。

检查框过滤器:

如前所述,如果我取消选中该框,那么我会取回所有数据 - 无需再次访问数据库。

编辑:==========================================

请注意,如果您不使用上面的横向,而只是向下行?

然后您也可以设置格式以隐藏/显示行,而不必保留数据表。

您可以通过这样做来应用格式:

    protected void ckFilter_CheckedChanged(object sender, EventArgs e)
    
        foreach (DataListItem gRow in DataList1.Items)
        
            // get checkbox
            CheckBox ckActive = (CheckBox)gRow.FindControl("Active");
            // get div
            htmlGenericControl Myiv = (HtmlGenericControl)gRow.FindControl("myrow");       
            string MyFormat = "normal";
            if (ckFilter2.Checked)
            
                // only show active ones
                if (ckActive.Checked)
                    MyFormat = "normal";
                else
                    MyFormat = "none";
            
            Myiv.Style["display"] = MyFormat;
        
    

所以上面的工作正常。但是限制是您不能像我的原始屏幕截图那样拥有一组横向数据项 (它实际上仍然有效,但实际上用空格隐藏了每个面板)。

所以,如果您使用垂直布局(看起来就是这样)。

请注意,我在“div”中添加了一个简单的 ID,并像这样运行服务器:

<ItemTemplate>
 <div id="myrow" runat="server" 
   style="border-style:solid;color:black;width:300px;">

然后跳过我的第一个示例。您可以循环数据列表行,并隐藏或取消隐藏。更令人惊奇的是,如果您取消隐藏 - 那么所有行都会重新出现并保持正确。

我提取了一个工作示例 - 但我试图弄清楚为什么我必须保留数据 - 原因是跨设置的“水平尾”。

但是,如上所示,实际上不需要持久化数据源表 - 您可以处理数据列表行,并根据条件隐藏/显示每个行,并且无需重新绑定数据即可做到这一点。

【讨论】:

以上是关于基于属性隐藏整个数据列表项的主要内容,如果未能解决你的问题,请参考以下文章

隐藏列表项,您在 React 中单击的项除外

隐藏列表视图中的项目

使用 RecyclerView 隐藏列表模型中的重复项

基于按钮单击事件的数据表隐藏和显示行

Angular 6显示或隐藏列表项

如何根据每个无序列表中的列表项的数量在 Jquery 中隐藏按钮元素