ASP.NET DataPager не сохраняет порядок параметров QueryString

У меня есть ListView с DataPager пейджинг в кнопках (1,2,3,...) не сохраняет порядок parameters в querystring что плохо в SEO контекст (бот думает, что это дублированный контент).

IE, если моя страница:

Test.aspx?r=8&type=10
the pager buttons show:
Test.aspx?type=8&r=10&page=1
Test.aspx?type=8&r=10&page=2

и т.д... Я думаю, что управление упорядочивает их в алфавитном порядке?

Я пытался переопределить эти URL на элементе управления PreRender() событие, но оно по-прежнему оказывает URL как описано выше...

Кто-нибудь сталкивался с этой проблемой??

Благодарю.

ASPX:

  <asp:ListView ID="ListViewSearchResults" runat="server"  GroupItemCount="3" 
                    OnItemDataBound="ListViewSearchResults_ItemDataBound"        
   OnPagePropertiesChanging="ListViewSearchResults_PagePropertiesChanging" >
   ......
                </asp:ListView>

.....

    <asp:DataPager ID="ListViewPager" runat="server"  QueryStringField="page"          
    OnDataBinding="ListViewPager_DataBinding" OnPreRender="ListViewPager_PreRender"
    PagedControlID="ListViewSearchResults" PageSize="21"  >
    <Fields>
    <asp:NextPreviousPagerField  ShowFirstPageButton="true"          
    ShowPreviousPageButton="false" ShowNextPageButton="false"       
     ShowLastPageButton="false" ButtonCssClass="pagingright1"   ButtonType="Link"     
     FirstPageText=""  />
     <asp:NextPreviousPagerField ShowFirstPageButton="false"      
     ShowPreviousPageButton="true"   ShowNextPageButton="false" 
     ShowLastPageButton="false" ButtonCssClass="pagingright2"   ButtonType="Link" 
     PreviousPageText="" />                      
                    <asp:NumericPagerField  ButtonType="Link" CurrentPageLabelCssClass="pagingcurrent"  RenderNonBreakingSpacesBetweenControls="false" ButtonCount="10" />                       
                <asp:NextPreviousPagerField  ShowFirstPageButton="false" ShowPreviousPageButton="false" ShowNextPageButton="true" ShowLastPageButton="false" ButtonCssClass="pagingleft2"   ButtonType="Link"  NextPageText="" />
                <asp:NextPreviousPagerField ShowFirstPageButton="false" ShowPreviousPageButton="false"   ShowNextPageButton="false" ShowLastPageButton="true" ButtonCssClass="pagingleft1"   ButtonType="Link" LastPageText="" />

            </Fields>
        </asp:DataPager>

CS: пытается исправить ссылки программно

    private void FixAllPagerLinks(string FixedUrl)
    {

    foreach (Control c in ListViewPager.Controls)
    {
        if(c.HasControls()) 
        {
            if(c.Controls[0] is HyperLink)
            {
                string PageParam = string.Empty; 
                string CurrentUrl = ((HyperLink)c.Controls[0]).NavigateUrl;
                char[] pageparam = new char[CurrentUrl.Length - CurrentUrl.IndexOf("page")];                    
                CurrentUrl.CopyTo(CurrentUrl.IndexOf("page"), pageparam, 0, CurrentUrl.Length - CurrentUrl.IndexOf("page"));
                PageParam = new string(pageparam);
                ((HyperLink)c.Controls[0]).NavigateUrl = string.Format("/{0}&{1}", FixedUrl , PageParam);
            }
        }
    }
}

    // in the DataPager PreRender Event nothing happen
    //but on page prerender event  it show the correct links but after 
    //clicking on next page in the pager it's messed up again
    protected override void OnPreRender(EventArgs e)
    {
    //getting the URL
    string NextPageurl = ((System.Web.UI.WebControls.HyperLink)        
    (ListViewPager.Controls[0].Controls[0])).NavigateUrl;
    //removing the page=1 to get the Basic url of the link
        NextPageurl = NextPageurl.Replace("&page=1", string.Empty).Replace("?page=1", string.Empty);
        string FullFixedUrl = FixQueryStringParameters(NextPageurl );
    FixAllPagerLinks(FullFixedUrl);
    }



private string FixQueryStringParameters(string QueryWithParams)
{
    string returnQueryStringParams = string.Empty;

    string p1 = string.Empty;
    string p2 = string.Empty;
    string p3 = string.Empty;
    Dictionary<string, string> QSKeyValuePairs = new Dictionary<string, string>();

    if (QueryWithParams.Split('?').Length > 1)
    {
        string parameters = QueryWithParams.Split('?')[1];

        if (parameters.Split('&').Length > 1)
        {
            p1 = parameters.Split('&')[0];
            QSKeyValuePairs.Add(p1.Split('=')[0], p1.Split('=')[1]);
            p2 = parameters.Split('&')[1];
            QSKeyValuePairs.Add(p2.Split('=')[0], p2.Split('=')[1]);

            if (parameters.Split('&').Length > 2)
            {
                p3 = parameters.Split('&')[2];
                QSKeyValuePairs.Add(p3.Split('=')[0], p3.Split('=')[1]);
            }
        }
    }

1 ответ

Решение состоит в том, чтобы поместить код, который изменяет URL-адрес, в событие DataBound соответствующего ListView. Вот мой код, чтобы сделать это

ASPX:

<asp:DataPager PagedControlID="_lvCategoryProduct" PageSize="3" ID="dpProducts" runat="server" QueryStringField="p">
    <Fields>
        <asp:NextPreviousPagerField ButtonCssClass="lnkPager bold" PreviousPageText="Previous" ButtonType="Link" ShowFirstPageButton="False" ShowNextPageButton="False" ShowPreviousPageButton="true" />
        <asp:NumericPagerField CurrentPageLabelCssClass="selectedPager" NumericButtonCssClass="lnkPager pagerNumber" />
        <asp:NextPreviousPagerField ButtonCssClass="lnkPager bold" NextPageText="Next" ButtonType="Link" ShowLastPageButton="False" ShowNextPageButton="True" ShowPreviousPageButton="False" />
    </Fields>
</asp:DataPager>
<asp:ListView ID="_lvCategoryProduct" runat="server" GroupItemCount="3" ItemPlaceholderID="_phItems" GroupPlaceholderID="_phGroups">
    ...
</asp:ListView>

Код страницы позади:

protected void Page_Load(object sender, EventArgs e)
{
    _lvCategoryProduct.DataBound += _lvCategoryProduct_DataBound;
}
protected void _lvCategoryProduct_DataBound(object sender, EventArgs e)
{
    if (!String.IsNullOrEmpty(_dpProducts.QueryStringField))
    {
        Common.SetDataPagerUrls(_dpProducts);
    }
}

Методы в "общем" объекте

/// <summary>
/// Method to change DataPager URLs so they keep the URL Rewriting pattern
/// Code adapted from http://weblogs.asp.net/anasghanem/archive/2009/10/08/programmatically-modifying-the-hyperlinks-in-the-datapager-control.aspx
/// </summary>
/// <param name="pager">The DataPager for which we modify the URLS</param>    
static public void SetDataPagerUrls(DataPager pager)
{
    string queryStringField = pager.QueryStringField;

    //Rewrited URL is in Page.Request.RawUrl
    string rawUrl = pager.Page.Request.RawUrl;
    //Get the value of the current page =>  abc.aspx?[queryStringField]=[X]
    string currentQueryStringValue = pager.Page.Request[queryStringField];

    //Remove the current page querystring from the URL
    rawUrl = rawUrl.Replace(queryStringField + "=" + currentQueryStringValue,"");        
    //Remove the trailing "?" or "&"        
    if (rawUrl.EndsWith("?") || rawUrl.EndsWith("&"))
    {
        rawUrl = rawUrl.Substring(0, rawUrl.Length - 1);
    }

    string queryStringValue = "";    
    //Loop through the DataPager HyperLinks to modify the NavigateUrl   
    foreach (DataPagerFieldItem Pitem in pager.Controls)
    {
        foreach (Control c in Pitem.Controls)
        {
            if (c is HyperLink)
            {
                HyperLink link = c as HyperLink;
                //Get the page value for the link NavigateUrl
                queryStringValue = GetQueryStringNumericValue(queryStringField, link.NavigateUrl);

                //Set the new URL
                link.NavigateUrl = rawUrl;
                if (link.NavigateUrl.Contains("?"))
                {
                     link.NavigateUrl += "&";
                }
                else
                {
                     link.NavigateUrl += "?";
                }
                link.NavigateUrl += queryStringField + "=" + queryStringValue;                    
            }
        }
    } 
}

/// <summary>
/// Gets a numeric value of a querystring parameter in a URL
/// </summary>
/// <param name="queryStringField">The name of the querystring field</param>
/// <param name="url">The URL in which the querystring value must be found</param>
/// <returns></returns>
static protected string GetQueryStringNumericValue(string queryStringField, string url)
{
    string returnValue = "";
    //Regular expression will match value like "p=9" in a url like "abc.aspx?p=9" or "abc.aspx?a=1&p=9&..."
    string pat = queryStringField + @"=[0-9]+";
    Regex r = new Regex(pat, RegexOptions.IgnoreCase);
    MatchCollection m = r.Matches(url);
    if (m.Count > 0)
    {
        //Remove the "[parameter]=" to keep only the value
        returnValue = m[0].Value.Replace(queryStringField + "=", "");
    }
    return returnValue;
}
Другие вопросы по тегам