程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> ASP編程 >> ASP技巧 >> ASP支持嵌套模板和循環標簽的模板類

ASP支持嵌套模板和循環標簽的模板類

編輯:ASP技巧

ASP Template 類說明

時間:17:05 2008-12-10
++功能簡介

--支持單層循環標簽,並可在一個頁面類多次使用.
--支持引入模板文件,在裝載的時候,將進行模板的合並
--可指定模板文件路徑,路徑為相對路徑,默認為當前文件路徑
--對於空白行最終輸出的時候,進行刪除

++標簽定義

{$tag$} 普通標簽
{$include:filename$} 模板文件標簽
<loop name="tagname">...</loop> 循環標簽,name屬性為標簽名稱
{$tag/subtag$} 循環標簽中的子標簽

++標簽說明:

采用正則表達式進行標簽的匹配和過濾,loop標簽中的name屬性之前可以有多個空格,之前之後可以存在其他屬性,name屬性可以帶引號也可以不帶,識別單引號和雙引號,設定只匹配第一個

++函數說明

LoadTPL函數 讀取模板文件,讀取的時候,檢查模板文件裡的嵌套模板文件標簽,首先替換嵌套的模板文件標簽的內容,合並模板文件,存入變量

Assign函數 分析模板標簽,對於普通標簽將其加入數據對象,如果為循環標簽,則存入循環數據對象,如果循環標簽對象更換,則將循環累加的數據加入數據對象

Flush函數 模板類很重要的一個函數,用於處理循環標簽,對於單次的循環,執行循環塊內部替換,對循環數據進行累加保存,每個單次循環完後必須調用

Bulid函數 將沒有來的及保存的循環數據加入到數據對象,然後按照模板定義輸出數據對象中的所有數據,普通標簽的替換在這一步完成

特別說明一下,assign函數有一個便捷的賦值方法,就是調用默認屬性來賦值,效果是一致的,例如:

 程序代碼
tp.assign("title","新聞")

可以采取這樣更簡潔的賦值方式

 程序代碼
tp("title")="新聞"

tp是實例化的模板對象

整個模板了代碼如下(template.ASP):


 程序代碼
<%
Class Template

PRivate m_content,m_looptmp,tagData,loopdata,m_loop_content,m_Looptag,m_TplPath,m_SetTplPath
Private m_ClassName,m_Version,m_Copyright

Private Sub Class_Initialize()
  m_content="" : m_looptmp="" : m_loop_content="" : m_looptag=""
  m_ClassName="Shaoyun ASP Template類" : m_Version="1.0" : m_Copyright="DevJS.com"
  m_TplPath="./" : m_SetTplPath=false
  Set tagData = Server.CreateObject("Scripting.Dictionary")
  Set loopData = Server.CreateObject("Scripting.Dictionary")
End Sub

Private Sub Class_Terminate()
  m_content="" : m_looptmp="" : m_loop_content="" : m_looptag=""
  m_TplPath="./" : m_SetTplPath=false
  Set tagData = Nothing : Set loopData = Nothing
End Sub

Public Property Get ClassName
  ClassName = m_ClassName
End Property

Public Property Get Version
  Version = m_Version
End Property

Public Property Get Copyright
  Copyright = m_Copyright
End Property

Rem 模板類的默認屬性,判斷模板中是否含有這個標簽
Public Default Property Get Tag(tagname)
  Tag = InStr(m_content,"{$" & tagname & "$")>0
End Property

Rem 調用定義好的賦值函數,這個屬性用來簡化賦值操作
Public Property Let Tag(tagname,replaceString)
  Call Assign(tagname,replaceString)
End Property

Public Property Get TplPath
  TplPath = m_TplPath
End Property

Rem 設定模板文件的路徑
Public Property Let TplPath(sTplPath)
  If sTplPath<>"" Then m_TplPath = sTplPath
  If Right(m_TplPath,1)<>"/" Then m_TplPath = m_TplPath & "/"
End Property

Private Function LoadFromFile(sFilePath,sCharset)
  LoadFromFile=false
  Dim oStream
  Set oStream=Server.CreateObject("ADODB.Stream")
  oStream.Type=2
  oStream.Mode=3
  oStream.Open
  oStream.Charset=sCharset
  oStream.Position=oStream.Size
  oStream.LoadFromFile sFilePath
  LoadFromFile=oStream.ReadText
  oStream.Close
  Set oStream=Nothing
End Function

Private Function FileExist(filespec)
  On Error Resume Next
  FileExist=False
  Dim ofso : Set oFSO = Server.CreateObject("Scripting.FileSystemObject")
  FileExist=oFSO.FileExists(filespec)
  Set oFSO=Nothing
End Function

Rem 獲取循環塊
Private Function GetTmpStr(tplstr,tagname,attname)
  Dim regEx,Matches,Match
  Set regEx = New RegExp
  regEx.Pattern = "<" & tagname & ".*?\s+name=[\""|\']?" & attname & "[\""|\']?.*?>([\s\S.]*?)<\/" & tagname & ">"
  regEx.Global = False
  regEx.IgnoreCase = True
  Set Matches = regEx.Execute(tplstr)
  For Each Match in Matches
  GetTmpStr=Match.Value
  Next
  Set regEx = Nothing
End Function

Rem 移除Html標記
Private Function RemoveTag(tagString,tagname)
  Dim regex
  Set regex=New RegExp
  regEx.Pattern = "<[\/]?" & tagname & ".*?>"
  regEx.Global = True
  regEx.IgnoreCase = True
  RemoveTag = regEx.Replace(tagString,"")
  Set regex=nothing
End Function

Rem 移除空白行
Private Function RemoveSpace(tagString)
  Dim regex
  Set regex=New RegExp
  regEx.Pattern = "\n\s*\r"
  regEx.Global = True
  regEx.IgnoreCase = True
  RemoveSpace = regEx.Replace(tagString,"")
  Set regex=nothing
End Function

Rem 讀取模板文件,同時處理嵌套模板,進行模板的合並
Public Function LoadTpl(tplfile)
  tplfile=Server.MapPath(tplfile)
  If Not FileExist(tplfile) Then
    Response.Write "Load template file faild!"
    Response.End
    Exit Function
  End If
  m_content=LoadFromFile(tplfile,"GB2312")
  Dim regEx,Matches,Match,fname,sContent
  Set regEx = New RegExp
  regEx.Pattern = "\{\$include\:(.*?)\$\}"
  regEx.Global = True
  regEx.IgnoreCase = True
  Set Matches = regEx.Execute(m_content)
  For Each Match in Matches
    fname=Match.SubMatches(0)
    fname=Server.MapPath(m_TplPath & fname)
    If FileExist(fname) Then
      sContent=LoadFromFile(fname,"GB2312")
      m_content=replace(m_content,Match.value,sContent)
    End If
  Next
  Set regEx = Nothing
End Function

Rem 賦值替換函數
Public Function Assign(tagname,replaceString)
  If tagname="" Then Exit Function
  Rem 如果是循環標簽
  If InStr(tagname,"/")>0 and InStr(tagname,"/")<Len(tagname) Then
    Rem 獲取循環標簽名稱
    m_curLooptag=Left(tagname,InStrRev(tagname,"/")-1)
    If m_Looptag="" Then
      Rem 如果是第一次檢測到循環標簽,設置循環所需變量初始值
      m_looptag=m_curLooptag : m_loop_content=""
      m_looptmp=GetTmpStr(m_content,"loop",m_Looptag)
    Else
      If m_LoopTag<>m_curLooptag Then
        Rem 如果循環標簽改變,初始循環變量
        m_content=replace(m_content,m_looptmp,m_loop_content)
        m_looptag=m_curLooptag : m_loop_content=""
        m_looptmp=GetTmpStr(m_content,"loop",m_Looptag)
      End If
    End If
    If Not(loopData.Exists(tagname)) Then loopData.Add tagname,replaceString
  Else
    Rem 普通標簽
    tagData.Add tagname,replaceString
  End If
End Function

Rem 執行塊內替換
Public Function Flush()
  If loopdata.count>0 then
    Dim i
    chgtmp=RemoveTag(m_looptmp,"loop")
    arrtag=loopData.keys
    arrval=loopData.items
    For i=0 To loopData.count-1
      chgtmp=replace(chgtmp,"{$" & arrtag(i) & "$}",arrval(i))
    Next
    Rem 將塊內數據保存到變量中
    m_loop_content=m_loop_content & chgtmp
    loopdata.RemoveAll
  End if
End Function

Rem 構建,完成模板的最後替換
Public Function Bulid()
  m_content=replace(m_content,m_looptmp,m_loop_content)
  arrtag=tagData.keys
  arrval=tagData.items
  For i=0 To tagData.count-1
    m_content=replace(m_content,"{$" & arrtag(i) & "$}",arrval(i))
  Next
  m_Content=RemoveSpace(m_Content)
  Response.Write m_Content
End Function

End Class
%>


父模板模板代碼(default.tpl):


 程序代碼
{$include:head.tpl$}
<h1 align=center>{$doc_title$}</h1>
<h3>{$news_title$}</h3>
<ul>
<loop name="news">
  <Li >新聞標題:{$news/title$}--作者:{$news/author$}</Li>
</loop>
</ul>
<h3>{$lastest_news$}</h3>
<ul>
<!-- 這裡loop中的bing和count只用作測試,不是必須的,實際使用的時候請刪除 -->
<loop bind="id"  name=arts count="15">
  <Li>文章標題:{$arts/title$}--作者:{$arts/author$}</Li>
</loop>
</ul>
{$include:foot.tpl$}


嵌套的子模板(head.tpl):


 程序代碼
<title>{$doc_title$}</title>


嵌套的子模板(foot.tpl):


 程序代碼
<p align=center>Copyright By DevJS.Com</p>


調用代碼(default.ASP):


 程序代碼
<!--#include file="function/template.ASP"-->
<%
Rem 模板類的使用方法事例

Set tp = new Template
tp.tplpath="tpl"
tp.LoadTpl(tp.tplpath & "default.tpl")
tp.assign "doc_title","模板機制的例子"
tp.assign "news_title","國內新聞"
for i=0 to 2
  call tp.assign("arts/title","金融危機導致大批失業人員")
  call tp.assign("arts/author","網易")
  tp.flush
next
tp.assign "lastest_news","最新文章"
Rem 這裡改用另一種賦值方式
for i=0 to 2
  tp("news/title")="政府利好消息將有助拉高股市"
  tp("news/author")="SOHU"
  tp.flush
next
tp.bulid
Set tp = nothing
%>

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved