在Option條目中填充前導空格的方法

    在使用Web頁面上的下拉列表框(SELECT element)顯示數據列表時,有時我們會遇到有層次的數據條目。比如論壇中的子論壇和它的分類之間,以及一些具有包含關係的層次數據條目。使下拉列表框中,不同的level有一定的顯示縮進將是非常友好的一種排版方式。

    如果在HTML編寫狀態下,或在ASP等腳本語言中,製作這樣的下拉列表窗口非常容易。我們知道空格" "在Option標籤的前後是會被brower的顯示引擎自動忽略掉的,所以我們使用硬空格 就可以了,效果如下圖:
    Select01.png

<select>
    
<option value="0">Level 00</option>
    
<option value="1">&nbsp;Level 01</option>
    
<option value="2">&nbsp;&nbsp;Level 02</option>
    
<option value="3">&nbsp;&nbsp;&nbsp;Level 03</option>
    
<option value="4">&nbsp;&nbsp;&nbsp;&nbsp;Level 04</option>
</select>

    如此簡單就實現這個效果了,似乎沒有什麼可說的。可是當我們在ASP.NET中使用服務器控件ListBox或DropDownList的時候,要實現這個效果問題就來了。由於ListItem類的Text屬性在輸出爲HTML代碼的時候,會自動進行HtmlEncode轉換,我們上面的示例會被輸出爲
    Select02.png
<select>
    
<option value="0">Level 00</option>
    
<option value="1">&amp;nbsp;Level 01</option>
    
<option value="2">&amp;nbsp;&amp;nbsp;Level 02</option>
    
<option value="3">&amp;nbsp;&amp;nbsp;&amp;nbsp;Level 03</option>
    
<option value="4">&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;Level 04</option>
</select>

    真是讓人鬱悶,我第一次遇到這個問題時,使用了一個很ugly的方法來解決。就是重載控件,在Render的時候把"&amp;nbsp;"變回爲"&nbsp;"。大概就是:
class XxxDropDownList : DropDownList
{
    
protected override void Render(HtmlTextWriter writer)
    
{
        StringBuilder strb 
= new StringBuilder();
        StringWriter sw 
= new StringWriter(strb);
        HtmlTextWriter htw 
= new HtmlTextWriter(sw);
        
base.Render(htw);
        strb.Replace(
"&amp;nbsp;""&nbsp;");
        writer.Write(strb.ToString());
    }

}

    這個解決方案問題很多,效率是一回事,並且非常不完備。除了使用這種"野蠻"的修改Render結果的方法,還有一個也是ugly的方法是使用全角的Space,就是" "。不過在中文的系統中這種方法似乎也說的過去,可是在純英文(Not support East-Asian language)的環境下,這樣的Option條目就暈菜了,錯誤效果如下圖:
    Select03.png
    // 左圖是在中文系統中,右圖是在純英文系統中

    那麼怎麼辦呢?重載ListItem?! 不過此路不通,ListItem類是被sealed修飾的。後來我發現,可以使用&nbsp;的"原始形式"來讓DropListBox輸出正確的HTML代碼。什麼是&nbsp;的原始形式呢?&nbsp;這個東西本來是在RFC 1866中定義的named entity,全稱叫做:no-break space,CDATA格式爲:&#160;。我們把160作爲字符放入ListItem的Text中,就能得到正確的HTML輸出,效果和第一幅圖中的HTML示例效果相同。服務器端代碼爲:
char nbsp = (char)0xA0;
for ( int i=0 ; i < 5 ; ++i )
{
    ddl3.Items.Add(new ListItem("Level 0".PadLeft(i+7, nbsp) + i, i.ToString()));
}


    生成的客戶端代碼爲:
<select>
    
<option value="0">Level 00</option>
    
<option value="1">&#160;Level 01</option>
    
<option value="2">&#160;&#160;Level 02</option>
    
<option value="3">&#160;&#160;&#160;Level 03</option>
    
<option value="4">&#160;&#160;&#160;&#160;Level 04</option>
</select>

 
發佈了147 篇原創文章 · 獲贊 0 · 訪問量 12萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章