資料回傳時,維護ASP.NET Tree控制項位置

來源:互聯網
上載者:User
asp.net|控制項|資料

  ASP.NET2.0提供了一個功能強大的TreeView控制項,但是它看起來有一個缺陷:它好像不能夠跟蹤使用者最後選擇的一個節點。如果你滾動到第50個節點然後展開該節點,那麼當單擊連結頁面進行回傳後,你必須重新利用捲軸下拉到你想要的節點位置。

  在.NET較早的版本裡,您可能考慮使用SmartNavigation這個特性.SmartNavigation是Web頁面指令的一個屬性,它的取值為布爾值,一個設定為true的頁面指令看起來類似如下:

<%@ Page Language="VB" AutoEventWireup="false" CodeFile="Default.aspx.vb" Inherits="_Default" SmartNavigation="true" %>

  不過,正如好多人已經注意到的,SmartNavigation本身就有洗衣阿問題,事實上微軟也被這個問題所困擾以至於在ASP.NET2.0裡添加了MaintainScrollbackPositionOnPostback屬性而取代SmartNavigation 。遺憾的是,我在使用它們時,感覺它們都有一些問題,我稍後將進行解釋。

  本文我將介紹SmartNavigation和MaintainScrollbackPositionOnPostback 在維護頁面回傳位置方面的缺點,並提供如何利用javascript來解決這個問題,這個小技巧即使對複雜的Web頁面也同樣有效.

  再見了SmartNavigationeb,歡迎MaintainScrollbackPositionOnPostback

  SmartNavigation主要作用是減少頁面導航時的閃動,它主要利用適當的IFrames來進行這個工作並僅僅顯示改變的部分。SmartNavigation 同樣被設計為能夠維護頁面位置,元素焦點,回傳瀏覽器訪問記錄的作用。遺憾的是,即使微軟知道SmartNavigation已經去掉,但是檢查MSDN文檔,您仍然能夠看到SmartNavigation其實僅僅被定義為"過時"的 。利用GOOGLE的搜尋您可以搜到大家對SmartNavigation的討論.

  下一步

  ASP.NET2.0引進了MaintainScrollbackPositionOnPostback,和SmartNavigation類似,您可以在Page屬性裡設定它的值為true或者為false。

<%@ Page Language="VB" AutoEventWireup="false" CodeFile="Default.aspx.vb" Inherits="_Default" MaintainScrollPositionOnPostback="true" %>

  非常簡明,這個屬性/屬性值對是用來維護頁面位置的。遺憾的是,它只是維護頁面的位置,因為如果你在使用者控制項裡使用了TreeView控制項,然後在頁面裡使用該使用者控制項,那麼頁面在在回傳後您將返回到使用者控制項的位置而不是TreeView節點位置。

  簡單的說,MaintainScrollbackPositionOnPostback只是用來維護頁面的回傳位置。如果你的頁面固定--也就是一個應用程式那樣不需要進行利用捲軸進行上下滾動,那麼這個屬性對你可能無用。如果你的頁面很常需要滾動,那麼你就需要利用該屬性.

  在TreeView裡維護控制項的位置

  最近,我在開發一個Web應用程式Windowsy,也就是每一個頁面都會全螢幕顯示而不是滾動。頁面裡使用類TreeView來進行導航想列表一樣進行顯示,但是頁面本身不需要上下滾動。但是問題是這裡的資料列可能需要擴充使得頁面出現滾動.我準備使用如下的方式解決這個問題.
首先,利用TreeView控制項的SelectedNode屬性,可以知道哪個節點被選取,這個被選取的節點需要儲存起來,它最終會程式為HTML元素。如果我知道了被選擇HTML控制項的ID,那麼我就可以滾動到該控制項並設定該控制項為當前焦點。確實,如果您看以下使用TreeView控制項的頁面HTML代碼,你將發現產生的一個隱藏<input>元素,以及為textbox類型,它的ID可能類似TreeViewx_SelectedNode

<input type="hidden" name="TreeView1_SelectedNode" id="TreeView1_SelectedNode" value="TreeView1t54" />

  有了這些知識,你就知道該怎麼做了,基本方法是隱藏的Input是一個textbox,我們要做的就是知道將來呈獻的內容。一個TreeView最終呈現為HTML表格,節點的值被用來作為儲存格的值,<TD>元素呈現節點名稱,所以通過尋找儲存格ID並滾動到那裡.

  為了具體說明做法,我使用TreeView編寫了一些代碼,在Page_Load時間裡載入一段指令碼來找到需要的儲存格(參考下表),在<body>的onload時間裡調用該函數.

Imports System.Collections.Generic
Partial Class _Default
Inherits System.Web.UI.Page
Protected Sub Page_Load(ByVal sender As Object, _
ByVal e As System.EventArgs) Handles Me.Load
InjectLoadEvent()
If (IsPostBack) Then Return
TreeView1.Nodes.Clear()
Dim chicken As New TreeNode("Chicken")
 TreeView1.Nodes.Add(chicken)
 Dim beef As New TreeNode("Beef")
 TreeView1.Nodes.Add(beef)
 Dim pork As New TreeNode("Pork")
 TreeView1.Nodes.Add(pork)
 Dim fish As New TreeNode("Fish")
 TreeView1.Nodes.Add(fish)
 chicken.ChildNodes.Add(New TreeNode("Crepes Florentine with Buffalo Chicken"))
 fish.ChildNodes.Add(New TreeNode("Linguine with White Clam Sauce"))
 pork.ChildNodes.Add(New TreeNode("Pork Loin with Peanut and Madarin Orange Sauce"))
 beef.ChildNodes.Add(New TreeNode("Standing Rib Roast with Fennel and Blue Cheese Potatoes"))
 ' We need a bunch of stuff here so we will add some stubs
 Dim I As Integer
 For I = 1 To 50
  chicken.ChildNodes.Add(New TreeNode("Placeholder " + I.ToString()))
  fish.ChildNodes.Add(New TreeNode("Placeholder " + I.ToString()))
  pork.ChildNodes.Add(New TreeNode("Placeholder " + I.ToString()))
  beef.ChildNodes.Add(New TreeNode("Placeholder " + I.ToString()))
 Next
 TreeView1.CollapseAll()
End Sub
Public Sub InjectLoadEvent()
 Dim script As String = _
   "function LoadEvent()" + _
   "{{" + _
   " try" + _
   " {{" + _
   " var elem = document.getElementById('{0}_SelectedNode');" + _
   " if(elem != null )" + _
   " {{" + _
   " var node = document.getElementById(elem.value);" + _
   " if(node != null)" + _
   " {{" + _
   " node.scrollIntoView(true);" + _
   " {1}.scrollLeft = 0;" + _
   " }}" + _
  " }}" + _
 " }}" + _
 " catch(oException)" + _
 " {{}}" + _
 "}}"
Page.ClientScript.RegisterClientScriptBlock(Me.GetType(), "LoadEvent", _
String.Format(script, TreeView1.ClientID, Panel1.ClientID), True)
 End Sub
End Class

  下面的代碼顯示了頁面的布局:

<%@ Page Language="VB" AutoEventWireup="false" CodeFile="Default.aspx.vb" Inherits="_Default" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" >
<head runat="server">
 <title>Focus Tree Node on Postback</title>
</head>
<body >
 <form id="form1" runat="server">
  <div>
   <asp:Panel ID="Panel1" runat="server" Height="200px" Width="200px" ScrollBars="Both">
    <asp:TreeView ID="TreeView1" runat="server">
     <SelectedNodeStyle BackColor="#8080FF" />
    </asp:TreeView>
   </asp:Panel>
  </div>
 </form>
</body>
</html>

  下圖顯示了本例子啟動並執行結果

  最後,下面的代碼顯示了javascript的注入方式:

<script>
function LoadEvent()
{
 try
 {
  var elem = document.getElementById('TreeView1_SelectedNode');
  if(elem != null )
  {
   var node = document.getElementById(elem.value);
   if(node != null)
   {
    node.scrollIntoView(true);
    Panel1.scrollLeft = 0;
   }
  }
 }
 catch(oException)
 {}
}// -->
</script>

  用Javascript定義的LoadEvent函數將尋找隱藏欄位,我們利用TreeView控制項的ClientID 尋找所有元素,不過,在嵌套多個TreeView控制項後,名稱將變得非常長.找到儲存格的值後,我使用scrollIntoView方法.scrollLeft屬性將讓捲軸滾動當前位置.



相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.