picupload.asp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536
  1. <!--#include file="Conn.asp"--><%Checklogin("")
  2. ' upload demo for asp
  3. ' @requires xhEditor
  4. '
  5. ' @author Yanis.Wang<yanis.wang@gmail.com>
  6. ' @site http://xheditor.com/
  7. ' @licence LGPL(http://www.opensource.org/licenses/lgpl-license.php)
  8. '
  9. ' @Version: 0.9.3 (build 100504)
  10. '
  11. ' 注1:本程序仅为演示用,请您根据自己需求进行相应修改,或者重开发
  12. ' 注2:本程序调用的无惧上传类 V2.2为xhEditor特别针对HTML5上传而修改过的版本
  13. 'option explicit
  14. response.charset="UTF-8"
  15. dim inputname,immediate,attachdir,dirtype,maxattachsize,upext,msgtype
  16. inputname="filedata"'表单文件域name
  17. attachdir="u"'上传文件保存路径,结尾不要带/
  18. dirtype=2'1:按天存入目录 2:按月存入目录 3:按扩展名存目录 建议使用按天存
  19. maxattachsize=2097152'最大上传大小,默认是2M
  20. upext="txt,rar,zip,jpg,jpeg,gif,png,swf,wmv,avi,wma,mp3,mid"'上传扩展名
  21. msgtype=2'返回上传参数的格式:1,只返回url,2,返回参数数组
  22. immediate=Request.QueryString("immediate")'立即上传模式,仅为演示用
  23. dim err,msg,upfile
  24. err = ""
  25. msg = "''"
  26. set upfile=new upfile_class
  27. upfile.AllowExt=replace(upext,",",";")+";"
  28. upfile.GetData(maxattachsize)
  29. if upfile.isErr then
  30. select case upfile.isErr
  31. case 1
  32. err="无数据提交"
  33. case 2
  34. err="文件大小超过 "+cstr(maxattachsize)+"字节"
  35. case else
  36. err=upfile.ErrMessage
  37. end select
  38. else
  39. dim attach_dir,attach_subdir,filename,extension,target,tmpfile
  40. extension=upfile.file(inputname).FileExt
  41. select case dirtype
  42. case 1
  43. attach_subdir="day_"+DateFormat(now,"yymmdd")
  44. case 2
  45. attach_subdir="m"+DateFormat(now,"yymm")
  46. case 3
  47. attach_subdir="ext_"+extension
  48. end select
  49. attach_dir=attachdir+"/"+attach_subdir+"/"
  50. '建文件夹
  51. CreateFolder attach_dir
  52. tmpfile=upfile.AutoSave(inputname,Server.mappath(attach_dir)+"\")
  53. if upfile.isErr then
  54. if upfile.isErr=3 then
  55. err="上传文件扩展名必需为:"+upext
  56. else
  57. err=upfile.ErrMessage
  58. end if
  59. else
  60. '生成随机文件名并改名
  61. Randomize timer
  62. filename=DateFormat(now,"ddhhnnss")+cstr(cint(99*Rnd))+"."+extension
  63. target=attach_dir+filename
  64. moveFile attach_dir+tmpfile,target
  65. if immediate="1" then target="!"+target
  66. imgurl = target
  67. target=jsonString(target)
  68. if msgtype=1 then
  69. msg="'"+target+"'"
  70. else
  71. msg="{'url':'"+target+"','localname':'"+jsonString(upfile.file(inputname).FileName)+"','id':'1'}"
  72. end if
  73. end if
  74. end if
  75. set upfile=nothing
  76. Set Jpeg=Server.CreateObject("Persits.Jpeg")
  77. Set Jpeg2=Server.CreateObject("Persits.Jpeg")
  78. Jpeg.Open Server.MapPath(Trim("/System/"&imgurl))
  79. img_w=520/Jpeg.OriginalWidth
  80. img_h=520/Jpeg.OriginalHeight
  81. IF img_w<1 Or img_h<1 Then
  82. If img_w<img_h Then
  83. Jpeg.Height=520
  84. Jpeg.Width=Round(Jpeg.OriginalWidth * img_h)
  85. Else
  86. Jpeg.Width=520
  87. Jpeg.Height=Round(Jpeg.OriginalHeight * img_w)
  88. End IF
  89. Jpeg.Crop 0, 0, 520, 520
  90. 'Jpeg2.Open Server.MapPath("/logo/logo.gif")
  91. 'Jpeg.DrawImage 258,231,Jpeg2,0.5,100
  92. Jpeg.Quality=100
  93. Jpeg.Save Server.MapPath("/System/"&imgurl)
  94. Else
  95. 'Jpeg2.Open Server.MapPath("/logo/logo.gif")
  96. 'Jpeg.DrawImage Jpeg.OriginalWidth-110,Jpeg.OriginalHeight-34,Jpeg2,0.5,100
  97. Jpeg.Quality=100
  98. Jpeg.Save Server.MapPath("/System/"&imgurl)
  99. End If
  100. Set Jpeg=Nothing
  101. Set Jpeg2=Nothing
  102. cpid=Request.QueryString("cpid")
  103. Keys=Server.UrlEnCode(Request.QueryString("Keys"))
  104. Ord=Server.UrlEnCode(Request.QueryString("Ord"))
  105. Set Rs8=Server.CreateObject("ADODB.RecordSet")
  106. Rs8.Open "pic",conn,1,3
  107. Rs8.addnew
  108. Rs8("cpid")=cpid
  109. Rs8("picurl")="/System/"&imgurl
  110. Rs8.Update
  111. Rs8.Close
  112. set Rs8=Nothing
  113. Response.Redirect "pic.asp?cpid="&cpid&"&Page="&Page&"&Keys="&Keys&"&Ord="&Ord
  114. function jsonString(str)
  115. str=replace(str,"\","\\")
  116. str=replace(str,"/","\/")
  117. str=replace(str,"'","\'")
  118. jsonString=str
  119. end function
  120. Function Iif(expression,returntrue,returnfalse)
  121. If expression=true Then
  122. iif=returntrue
  123. Else
  124. iif=returnfalse
  125. End If
  126. End Function
  127. function DateFormat(strDate,fstr)
  128. if isdate(strDate) then
  129. dim i,temp
  130. temp=replace(fstr,"yyyy",DatePart("yyyy",strDate))
  131. temp=replace(temp,"yy",mid(DatePart("yyyy",strDate),3))
  132. temp=replace(temp,"y",DatePart("y",strDate))
  133. temp=replace(temp,"w",DatePart("w",strDate))
  134. temp=replace(temp,"ww",DatePart("ww",strDate))
  135. temp=replace(temp,"q",DatePart("q",strDate))
  136. temp=replace(temp,"mm",iif(len(DatePart("m",strDate))>1,DatePart("m",strDate),"0"&DatePart("m",strDate)))
  137. temp=replace(temp,"dd",iif(len(DatePart("d",strDate))>1,DatePart("d",strDate),"0"&DatePart("d",strDate)))
  138. temp=replace(temp,"hh",iif(len(DatePart("h",strDate))>1,DatePart("h",strDate),"0"&DatePart("h",strDate)))
  139. temp=replace(temp,"nn",iif(len(DatePart("n",strDate))>1,DatePart("n",strDate),"0"&DatePart("n",strDate)))
  140. temp=replace(temp,"ss",iif(len(DatePart("s",strDate))>1,DatePart("s",strDate),"0"&DatePart("s",strDate)))
  141. DateFormat=temp
  142. else
  143. DateFormat=false
  144. end if
  145. end function
  146. Function CreateFolder(FolderPath)
  147. dim lpath,fs,f
  148. lpath=Server.MapPath(FolderPath)
  149. Set fs=Server.CreateObject("Scri"&"pting.File"&"Sys"&"temObject")
  150. If not fs.FolderExists(lpath) then
  151. Set f=fs.CreateFolder(lpath)
  152. CreateFolder=F.Path
  153. end if
  154. Set F=Nothing
  155. Set fs=Nothing
  156. End Function
  157. Function moveFile(oldfile,newfile)
  158. dim fs
  159. Set fs=Server.CreateObject("Scri"&"pting.File"&"Sys"&"temObject")
  160. fs.movefile Server.MapPath(oldfile),Server.MapPath(newfile)
  161. Set fs=Nothing
  162. End Function
  163. '----------------------------------------------------------------------
  164. '转发时请保留此声明信息,这段声明不并会影响你的速度!
  165. '******************* 无惧上传类 V2.2 xheditor特别修改版 ************************************
  166. '作者:梁无惧
  167. '网站:http://www.25cn.com
  168. '电子邮件:yjlrb@21cn.com
  169. '版权声明:版权所有,源代码公开,各种用途均可免费使用,但是修改后必须把修改后的文件
  170. '发送一份给作者.并且保留作者此版权信息
  171. '**********************************************************************
  172. '----------------------------------------------------------------------
  173. '----------------------------------------------------------------------
  174. '文件上传类
  175. Class UpFile_Class
  176. Dim Form,File
  177. Dim AllowExt_ '允许上传类型(白名单)
  178. Dim NoAllowExt_ '不允许上传类型(黑名单)
  179. Dim IsDebug_ '是否显示出错信息
  180. Private oUpFileStream '上传的数据流
  181. Private isErr_ '错误的代码,0或true表示无错
  182. Private ErrMessage_ '错误的字符串信息
  183. Private isGetData_ '指示是否已执行过GETDATA过程
  184. '------------------------------------------------------------------
  185. '类的属性
  186. Public Property Get Version
  187. Version="无惧上传类 Version V2.0"
  188. End Property
  189. Public Property Get isErr '错误的代码,0或true表示无错
  190. isErr=isErr_
  191. End Property
  192. Public Property Get ErrMessage '错误的字符串信息
  193. ErrMessage=ErrMessage_
  194. End Property
  195. Public Property Get AllowExt '允许上传类型(白名单)
  196. AllowExt=AllowExt_
  197. End Property
  198. Public Property Let AllowExt(Value) '允许上传类型(白名单)
  199. AllowExt_=LCase(Value)
  200. End Property
  201. Public Property Get NoAllowExt '不允许上传类型(黑名单)
  202. NoAllowExt=NoAllowExt_
  203. End Property
  204. Public Property Let NoAllowExt(Value) '不允许上传类型(黑名单)
  205. NoAllowExt_=LCase(Value)
  206. End Property
  207. Public Property Let IsDebug(Value) '是否设置为调试模式
  208. IsDebug_=Value
  209. End Property
  210. '----------------------------------------------------------------
  211. '类实现代码
  212. '初始化类
  213. Private Sub Class_Initialize
  214. isErr_ = 0
  215. NoAllowExt="" '黑名单,可以在这里预设不可上传的文件类型,以文件的后缀名来判断,不分大小写,每个每缀名用;号分开,如果黑名单为空,则判断白名单
  216. NoAllowExt=LCase(NoAllowExt)
  217. AllowExt="" '白名单,可以在这里预设可上传的文件类型,以文件的后缀名来判断,不分大小写,每个后缀名用;号分开
  218. AllowExt=LCase(AllowExt)
  219. isGetData_=false
  220. End Sub
  221. '类结束
  222. Private Sub Class_Terminate
  223. on error Resume Next
  224. '清除变量及对像
  225. Form.RemoveAll
  226. Set Form = Nothing
  227. File.RemoveAll
  228. Set File = Nothing
  229. oUpFileStream.Close
  230. Set oUpFileStream = Nothing
  231. if Err.number<>0 then OutErr("清除类时发生错误!")
  232. End Sub
  233. '分析上传的数据
  234. Public Sub GetData (MaxSize)
  235. '定义变量
  236. on error Resume Next
  237. if isGetData_=false then
  238. Dim RequestBinData,sSpace,bCrLf,sInfo,iInfoStart,iInfoEnd,tStream,iStart,oFileInfo
  239. Dim sFormValue,sFileName
  240. Dim iFindStart,iFindEnd
  241. Dim iFormStart,iFormEnd,sFormName
  242. '代码开始
  243. If Request.TotalBytes < 1 Then '如果没有数据上传
  244. isErr_ = 1
  245. ErrMessage_="没有数据上传,这是因为直接提交网址所产生的错误!"
  246. OutErr("没有数据上传,这是因为直接提交网址所产生的错误!!")
  247. Exit Sub
  248. End If
  249. If MaxSize > 0 Then '如果限制大小
  250. If Request.TotalBytes > MaxSize Then
  251. isErr_ = 2 '如果上传的数据超出限制大小
  252. ErrMessage_="上传的数据超出限制大小!"
  253. OutErr("上传的数据超出限制大小!")
  254. Exit Sub
  255. End If
  256. End If
  257. Set Form = Server.CreateObject ("Scripting.Dictionary")
  258. Form.CompareMode = 1
  259. Set File = Server.CreateObject ("Scripting.Dictionary")
  260. File.CompareMode = 1
  261. Set tStream = Server.CreateObject ("ADODB.Stream")
  262. Set oUpFileStream = Server.CreateObject ("ADODB.Stream")
  263. if Err.number<>0 then OutErr("创建流对象(ADODB.STREAM)时出错,可能系统不支持或没有开通该组件")
  264. oUpFileStream.Type = 1
  265. oUpFileStream.Mode = 3
  266. oUpFileStream.Open
  267. oUpFileStream.Write Request.BinaryRead (Request.TotalBytes)
  268. oUpFileStream.Position = 0
  269. RequestBinData = oUpFileStream.Read
  270. Dim sHtml5FileInfo
  271. sHtml5FileInfo=Request.ServerVariables("HTTP_CONTENT_DISPOSITION")
  272. If sHtml5FileInfo<>"" Then'针对Html5上传特别修正
  273. iFindStart = InStr (1,sHtml5FileInfo,"name=""",1)+6
  274. iFindEnd = InStr (iFindStart,sHtml5FileInfo,"""",1)
  275. sFormName=Trim(Mid(sHtml5FileInfo,iFindStart,iFindEnd-iFindStart))
  276. iFindStart = InStr (iFindStart,sHtml5FileInfo,"filename=""",1)+10
  277. iFindEnd = InStr (iFindStart,sHtml5FileInfo,"""",1)
  278. sFileName = Trim(Mid(sHtml5FileInfo,iFindStart,iFindEnd-iFindStart))
  279. Set oFileInfo = new FileInfo_Class
  280. oFileInfo.FileName = GetFileName(sFileName)
  281. oFileInfo.FilePath = GetFilePath(sFileName)
  282. oFileInfo.FileExt = GetFileExt(sFileName)
  283. oFileInfo.FileStart = 0
  284. oFileInfo.FileSize = Request.TotalBytes
  285. oFileInfo.FormName = sFormName
  286. file.add sFormName,oFileInfo
  287. Else
  288. iFormEnd = oUpFileStream.Size
  289. bCrLf = ChrB (13) & ChrB (10)
  290. '取得每个项目之间的分隔符
  291. sSpace = MidB (RequestBinData,1, InStrB (1,RequestBinData,bCrLf)-1)
  292. iStart = LenB(sSpace)
  293. iFormStart = iStart+2
  294. '分解项目
  295. Do
  296. iInfoEnd = InStrB (iFormStart,RequestBinData,bCrLf & bCrLf)+3
  297. tStream.Type = 1
  298. tStream.Mode = 3
  299. tStream.Open
  300. oUpFileStream.Position = iFormStart
  301. oUpFileStream.CopyTo tStream,iInfoEnd-iFormStart
  302. tStream.Position = 0
  303. tStream.Type = 2
  304. tStream.CharSet = "utf-8"
  305. sInfo = tStream.ReadText
  306. '取得表单项目名称
  307. iFormStart = InStrB (iInfoEnd,RequestBinData,sSpace)-1
  308. iFindStart = InStr (22,sInfo,"name=""",1)+6
  309. iFindEnd = InStr (iFindStart,sInfo,"""",1)
  310. sFormName = Mid(sinfo,iFindStart,iFindEnd-iFindStart)
  311. '如果是文件
  312. If InStr (45,sInfo,"filename=""",1) > 0 Then
  313. Set oFileInfo = new FileInfo_Class
  314. '取得文件属性
  315. iFindStart = InStr (iFindEnd,sInfo,"filename=""",1)+10
  316. iFindEnd = InStr (iFindStart,sInfo,""""&vbCrLf,1)
  317. sFileName = Trim(Mid(sinfo,iFindStart,iFindEnd-iFindStart))
  318. oFileInfo.FileName = GetFileName(sFileName)
  319. oFileInfo.FilePath = GetFilePath(sFileName)
  320. oFileInfo.FileExt = GetFileExt(sFileName)
  321. iFindStart = InStr (iFindEnd,sInfo,"Content-Type: ",1)+14
  322. iFindEnd = InStr (iFindStart,sInfo,vbCr)
  323. oFileInfo.FileMIME = Mid(sinfo,iFindStart,iFindEnd-iFindStart)
  324. oFileInfo.FileStart = iInfoEnd
  325. oFileInfo.FileSize = iFormStart -iInfoEnd -2
  326. oFileInfo.FormName = sFormName
  327. file.add sFormName,oFileInfo
  328. else
  329. '如果是表单项目
  330. tStream.Close
  331. tStream.Type = 1
  332. tStream.Mode = 3
  333. tStream.Open
  334. oUpFileStream.Position = iInfoEnd
  335. oUpFileStream.CopyTo tStream,iFormStart-iInfoEnd-2
  336. tStream.Position = 0
  337. tStream.Type = 2
  338. tStream.CharSet = "utf-8"
  339. sFormValue = tStream.ReadText
  340. If Form.Exists (sFormName) Then
  341. Form (sFormName) = Form (sFormName) & ", " & sFormValue
  342. else
  343. Form.Add sFormName,sFormValue
  344. End If
  345. End If
  346. tStream.Close
  347. iFormStart = iFormStart+iStart+2
  348. '如果到文件尾了就退出
  349. Loop Until (iFormStart+2) >= iFormEnd
  350. if Err.number<>0 then OutErr("分解上传数据时发生错误,可能客户端的上传数据不正确或不符合上传数据规则")
  351. End if
  352. RequestBinData = ""
  353. Set tStream = Nothing
  354. isGetData_=true
  355. end if
  356. End Sub
  357. '保存到文件,自动覆盖已存在的同名文件
  358. Public Function SaveToFile(Item,Path)
  359. SaveToFile=SaveToFileEx(Item,Path,True)
  360. End Function
  361. '保存到文件,自动设置文件名
  362. Public Function AutoSave(Item,Path)
  363. AutoSave=SaveToFileEx(Item,Path,false)
  364. End Function
  365. '保存到文件,OVER为真时,自动覆盖已存在的同名文件,否则自动把文件改名保存
  366. Private Function SaveToFileEx(Item,Path,Over)
  367. On Error Resume Next
  368. Dim FileExt
  369. if file.Exists(Item) then
  370. Dim oFileStream
  371. Dim tmpPath
  372. isErr_=0
  373. Set oFileStream = CreateObject ("ADODB.Stream")
  374. oFileStream.Type = 1
  375. oFileStream.Mode = 3
  376. oFileStream.Open
  377. oUpFileStream.Position = File(Item).FileStart
  378. oUpFileStream.CopyTo oFileStream,File(Item).FileSize
  379. tmpPath=Split(Path,".")(0)
  380. FileExt=GetFileExt(Path)
  381. if Over then
  382. if isAllowExt(FileExt) then
  383. oFileStream.SaveToFile tmpPath & "." & FileExt,2
  384. if Err.number<>0 then OutErr("保存文件时出错,请检查路径,是否存在该上传目录!该文件保存路径为" & tmpPath & "." & FileExt)
  385. Else
  386. isErr_=3
  387. ErrMessage_="该后缀名的文件不允许上传!"
  388. OutErr("该后缀名的文件不允许上传")
  389. End if
  390. Else
  391. Path=GetFilePath(Path)
  392. dim fori
  393. fori=1
  394. if isAllowExt(File(Item).FileExt) then
  395. do
  396. fori=fori+1
  397. Err.Clear()
  398. tmpPath=Path&GetNewFileName()&"."&File(Item).FileExt
  399. oFileStream.SaveToFile tmpPath
  400. loop Until ((Err.number=0) or (fori>50))
  401. if Err.number<>0 then OutErr("自动保存文件出错,已经测试50次不同的文件名来保存,请检查目录是否存在!该文件最后一次保存时全路径为"&Path&GetNewFileName()&"."&File(Item).FileExt)
  402. Else
  403. isErr_=3
  404. ErrMessage_="该后缀名的文件不允许上传!"
  405. OutErr("该后缀名的文件不允许上传")
  406. End if
  407. End if
  408. oFileStream.Close
  409. Set oFileStream = Nothing
  410. else
  411. ErrMessage_="不存在该对象(如该文件没有上传,文件为空)!"
  412. OutErr("不存在该对象(如该文件没有上传,文件为空)")
  413. end if
  414. if isErr_=3 then SaveToFileEx="" else SaveToFileEx=GetFileName(tmpPath)
  415. End Function
  416. '取得文件数据
  417. Public Function FileData(Item)
  418. isErr_=0
  419. if file.Exists(Item) then
  420. if isAllowExt(File(Item).FileExt) then
  421. oUpFileStream.Position = File(Item).FileStart
  422. FileData = oUpFileStream.Read (File(Item).FileSize)
  423. Else
  424. isErr_=3
  425. ErrMessage_="该后缀名的文件不允许上传"
  426. OutErr("该后缀名的文件不允许上传")
  427. FileData=""
  428. End if
  429. else
  430. ErrMessage_="不存在该对象(如该文件没有上传,文件为空)!"
  431. OutErr("不存在该对象(如该文件没有上传,文件为空)")
  432. end if
  433. End Function
  434. '取得文件路径
  435. Public function GetFilePath(FullPath)
  436. If FullPath <> "" Then
  437. GetFilePath = Left(FullPath,InStrRev(FullPath, "\"))
  438. Else
  439. GetFilePath = ""
  440. End If
  441. End function
  442. '取得文件名
  443. Public Function GetFileName(FullPath)
  444. If FullPath <> "" Then
  445. GetFileName = mid(FullPath,InStrRev(FullPath, "\")+1)
  446. Else
  447. GetFileName = ""
  448. End If
  449. End function
  450. '取得文件的后缀名
  451. Public Function GetFileExt(FullPath)
  452. If FullPath <> "" Then
  453. GetFileExt = LCase(Mid(FullPath,InStrRev(FullPath, ".")+1))
  454. Else
  455. GetFileExt = ""
  456. End If
  457. End function
  458. '取得一个不重复的序号
  459. Public Function GetNewFileName()
  460. dim ranNum
  461. dim dtNow
  462. dtNow=Now()
  463. randomize
  464. ranNum=int(90000*rnd)+10000
  465. '以下这段由webboy提供
  466. GetNewFileName=year(dtNow) & right("0" & month(dtNow),2) & right("0" & day(dtNow),2) & right("0" & hour(dtNow),2) & right("0" & minute(dtNow),2) & right("0" & second(dtNow),2) & ranNum
  467. End Function
  468. Public Function isAllowExt(Ext)
  469. if NoAllowExt="" then
  470. isAllowExt=cbool(InStr(1,";"&AllowExt&";",LCase(";"&Ext&";")))
  471. else
  472. isAllowExt=not CBool(InStr(1,";"&NoAllowExt&";",LCase(";"&Ext&";")))
  473. end if
  474. End Function
  475. End Class
  476. Public Sub OutErr(ErrMsg)
  477. if IsDebug_=true then
  478. Response.Write ErrMsg
  479. Response.End
  480. End if
  481. End Sub
  482. '----------------------------------------------------------------------------------------------------
  483. '文件属性类
  484. Class FileInfo_Class
  485. Dim FormName,FileName,FilePath,FileSize,FileMIME,FileStart,FileExt
  486. End Class
  487. %>