How do I use a DataPager with Server Side Paging?

前端 未结 3 1444
灰色年华
灰色年华 2020-12-30 05:46

I\'m trying to use a DataPager to do Server Side paging. Here is my code



        
相关标签:
3条回答
  • 2020-12-30 06:23

    It seems the only way to get server-side paging to work with the DataPager is to use a LinqDataSource in your markup (update: this is not true, see below), and set the DataSourceID of your ListView (as in ScottGu's sample - step 6), not via the ListView DataSource property. However I did discover that there is a trick to make this more workable so you can define your LinqDataSource query via code-behind rather than in the markup (note, the article says this will work with any DataSource control but I don't think it is the case).

    This will probably not be any help for your situation as I noticed that your calling some sort of service that probably won't (and shouldn't) be returning an IQueryable result which is necessary for this to work. Here it is anyway in-case you're interested ...

    aspx markup:

    <asp:ListView ID="lvFooBars" DataSourceID="dsLinq" ....>
       ......
    </asp:ListView>
    
    <asp:DataPager ID="pgrFooBars" PagedControlID="lvFooBars" 
        QueryStringField="page" runat="server" >
    <Fields>
        <asp:NumericPagerField />
    </Fields>
    </asp:DataPager>
    
    <asp:LinqDataSource id="dsLinq" runat="server" OnSelecting="dsLinq_Selecting" />
    

    code-behind:

    protected void dsLinq_Selecting(object sender, LinqDataSourceSelectEventArgs e)
    {
        //notice this method on FooService requires no paging variables
        e.Result = FooService.GetIQueryableFooBars(); 
    }
    

    Note, MSDN states in regards to the QueryStringField attribute:

    Setting this property is useful if you want to have all the pages of data indexed by a search engine. This occurs because the control produces a different URL for each page of data.

    Update: infact you can get this to working using an ObjectDataSource control instead of the LinqDataSource control - see this article and download the sample. While using an ObjectDataSource isn't nearly as simple as using a LinqDataSource control, it's uses less magic linq stuff (instead it uses heaps of magic strings that map to business/data layer methods) and allows the exposure of IEnumerable for your data access method instead of IQueryable.

    Still, I have never really a fan of embedding any sort of DataSource control in my UI markup (I believe this is necessary), so I would probably steer clear of the DataPager control except for small applications as you have suggested in your first update.

    John, one other thing I noticed was that you are trying to marry standard Asp.Net server controls (that rely on ViewState) with the PagedList class which was developed as a helper class for Asp.Net Mvc applications. While this can potentially work, there may be simpler routes to take.

    0 讨论(0)
  • 2020-12-30 06:23

    Exactly needed what you did - SQL server paging utilizing listview, datapager and linq data source. As you say TotalRowCount is readonly. There is also SetPageProperties method that can be used to set the total row count but that also didn't work for me.

    However I managed to trick it this way. I have a dummy hidden listview with empty item template. The pager will be pointing at this dummy listview rather than the original listview. Then we need to bind the dummy listview to a dummy collection with the same number of items as in the original data source. This will ensure rendering the pager correctly. Here is the markup.

    <asp:DataPager ID="pdPagerBottom" runat="server" PagedControlID="lvDummy" PageSize="5" QueryStringField="page">
    <Fields>
        <asp:NumericPagerField ButtonType="Link" RenderNonBreakingSpacesBetweenControls="true" NextPageText="Next" PreviousPageText="Previous" ButtonCount="40" />
    </Fields>
    </asp:DataPager>
    
    <asp:ListView ID="lvDummy" runat="server" Visible="false">
    <ItemTemplate></ItemTemplate>
    </asp:ListView>
    
    <asp:ListView ID="lvPropertyList" runat="server">...</asp:ListView>
    

    and code behind

            var datasourceQuery = context.Properties.OrderBy(x => x.PropertyID).Skip(pdPagerBottom.StartRowIndex).Take(pdPagerBottom.PageSize);
    
            this.lvPropertyList.DataSource = datasourceQuery;
            this.lvPropertyList.DataBind();
    
            this.lvDummy.DataSource = new List<int>(Enumerable.Range(0, context.Properties.Count()));
            this.lvDummy.DataBind();
    

    Tested it with 100k records. Works a treat.

    0 讨论(0)
  • 2020-12-30 06:41

    John, I did a little proof of concept here so that you can see the code. I made this as basic as possible so there's no fluff. Please let me know if there is some attribute to this that you need that is missing and I will revise.

    Things that stood out:

    1. The data needs to be bound on the OnPreRender event of the pager when you're not using a DataSource object (xml,Sql,Access...). I am pretty sure this is your issue because this is where I had the most trouble.

    2. Viewstate must be enabled on the ListView and the pager (it is by default)

    Code-Behind:

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Web;
    using System.Web.UI;
    using System.Web.UI.WebControls;
    
    public partial class _Default : System.Web.UI.Page 
    {
        protected void Page_Load(object sender, EventArgs e)
        {
            if (!Page.IsPostBack)
            {
                this.BindListData();
            }
        }
    
        protected void dp_PreRender(object sender, EventArgs e)
        {
            this.BindListData();
        }
    
        protected void BindListData()
        {
            List<Nums> n = new List<Nums>();
            for (int i = 0; i <= 100; i++)
            {
                n.Add(new Nums { Num1 = i, Num2 = i * 3 });
            }
    
            this.lvMyGrid.DataSource = n;
            this.lvMyGrid.DataBind();
        }
    }
    
    public class Nums
    {
        public int Num1 { get; set; }
        public int Num2 { get; set; }
    }
    

    ASPX (left out all the other fluff):

    <asp:DataPager OnPreRender="dp_PreRender" runat="server" ID="dpMyPager" QueryStringField="page" PagedControlID="lvMyGrid" PageSize="15">
        <Fields>
            <asp:NumericPagerField />
        </Fields>
    </asp:DataPager>
    
    <asp:ListView runat="server" ID="lvMyGrid" DataKeyNames="Num1">
        <LayoutTemplate>
            <asp:Literal runat="server" ID="itemPlaceholder" />
        </LayoutTemplate>
        <ItemTemplate>
            <div style="border: 1px solid red; padding: 3px; margin: 5px; float: left; clear: both;">
            <%# Eval("Num1") %> : <%# Eval("Num2") %>
            </div>
        </ItemTemplate>
    </asp:ListView>
    

    Enjoy, let me know if there is something else in this that you need.

    mike

    0 讨论(0)
提交回复
热议问题