fckxhtml.js
上传用户:dbstep
上传日期:2022-08-06
资源大小:2803k
文件大小:16k
源码类别:

WEB源码(ASP,PHP,...)

开发平台:

ASP/ASPX

  1. /*
  2.  * FCKeditor - The text editor for Internet - http://www.fckeditor.net
  3.  * Copyright (C) 2003-2009 Frederico Caldeira Knabben
  4.  *
  5.  * == BEGIN LICENSE ==
  6.  *
  7.  * Licensed under the terms of any of the following licenses at your
  8.  * choice:
  9.  *
  10.  *  - GNU General Public License Version 2 or later (the "GPL")
  11.  *    http://www.gnu.org/licenses/gpl.html
  12.  *
  13.  *  - GNU Lesser General Public License Version 2.1 or later (the "LGPL")
  14.  *    http://www.gnu.org/licenses/lgpl.html
  15.  *
  16.  *  - Mozilla Public License Version 1.1 or later (the "MPL")
  17.  *    http://www.mozilla.org/MPL/MPL-1.1.html
  18.  *
  19.  * == END LICENSE ==
  20.  *
  21.  * Defines the FCKXHtml object, responsible for the XHTML operations.
  22.  */
  23. var FCKXHtml = new Object() ;
  24. FCKXHtml.CurrentJobNum = 0 ;
  25. FCKXHtml.GetXHTML = function( node, includeNode, format )
  26. {
  27. FCKDomTools.CheckAndRemovePaddingNode( FCKTools.GetElementDocument( node ), FCKConfig.EnterMode ) ;
  28. FCKXHtmlEntities.Initialize() ;
  29. // Set the correct entity to use for empty blocks.
  30. this._NbspEntity = ( FCKConfig.ProcessHTMLEntities? 'nbsp' : '#160' ) ;
  31. // Save the current IsDirty state. The XHTML processor may change the
  32. // original HTML, dirtying it.
  33. var bIsDirty = FCK.IsDirty() ;
  34. // Special blocks are blocks of content that remain untouched during the
  35. // process. It is used for SCRIPTs and STYLEs.
  36. FCKXHtml.SpecialBlocks = new Array() ;
  37. // Create the XML DOMDocument object.
  38. this.XML = FCKTools.CreateXmlObject( 'DOMDocument' ) ;
  39. // Add a root element that holds all child nodes.
  40. this.MainNode = this.XML.appendChild( this.XML.createElement( 'xhtml' ) ) ;
  41. FCKXHtml.CurrentJobNum++ ;
  42. // var dTimer = new Date() ;
  43. if ( includeNode )
  44. this._AppendNode( this.MainNode, node ) ;
  45. else
  46. this._AppendChildNodes( this.MainNode, node, false ) ;
  47. // Get the resulting XHTML as a string.
  48. var sXHTML = this._GetMainXmlString() ;
  49. // alert( 'Time: ' + ( ( ( new Date() ) - dTimer ) ) + ' ms' ) ;
  50. this.XML = null ;
  51. // Safari adds xmlns="http://www.w3.org/1999/xhtml" to the root node (#963)
  52. if ( FCKBrowserInfo.IsSafari )
  53. sXHTML = sXHTML.replace( /^<xhtml.*?>/, '<xhtml>' ) ;
  54. // Strip the "XHTML" root node.
  55. sXHTML = sXHTML.substr( 7, sXHTML.length - 15 ).Trim() ;
  56. // According to the doctype set the proper end for self-closing tags
  57. // HTML: <br>
  58. // XHTML: Add a space, like <br/> -> <br />
  59. if (FCKConfig.DocType.length > 0 && FCKRegexLib.HtmlDocType.test( FCKConfig.DocType ) )
  60. sXHTML = sXHTML.replace( FCKRegexLib.SpaceNoClose, '>');
  61. else
  62. sXHTML = sXHTML.replace( FCKRegexLib.SpaceNoClose, ' />');
  63. if ( FCKConfig.ForceSimpleAmpersand )
  64. sXHTML = sXHTML.replace( FCKRegexLib.ForceSimpleAmpersand, '&' ) ;
  65. if ( format )
  66. sXHTML = FCKCodeFormatter.Format( sXHTML ) ;
  67. // Now we put back the SpecialBlocks contents.
  68. for ( var i = 0 ; i < FCKXHtml.SpecialBlocks.length ; i++ )
  69. {
  70. var oRegex = new RegExp( '___FCKsi___' + i ) ;
  71. sXHTML = sXHTML.replace( oRegex, FCKXHtml.SpecialBlocks[i] ) ;
  72. }
  73. // Replace entities marker with the ampersand.
  74. sXHTML = sXHTML.replace( FCKRegexLib.GeckoEntitiesMarker, '&' ) ;
  75. // Restore the IsDirty state if it was not dirty.
  76. if ( !bIsDirty )
  77. FCK.ResetIsDirty() ;
  78. FCKDomTools.EnforcePaddingNode( FCKTools.GetElementDocument( node ), FCKConfig.EnterMode ) ;
  79. return sXHTML ;
  80. }
  81. FCKXHtml._AppendAttribute = function( xmlNode, attributeName, attributeValue )
  82. {
  83. try
  84. {
  85. if ( attributeValue == undefined || attributeValue == null )
  86. attributeValue = '' ;
  87. else if ( attributeValue.replace )
  88. {
  89. if ( FCKConfig.ForceSimpleAmpersand )
  90. attributeValue = attributeValue.replace( /&/g, '___FCKAmp___' ) ;
  91. // Entities must be replaced in the attribute values.
  92. attributeValue = attributeValue.replace( FCKXHtmlEntities.EntitiesRegex, FCKXHtml_GetEntity ) ;
  93. }
  94. // Create the attribute.
  95. var oXmlAtt = this.XML.createAttribute( attributeName ) ;
  96. oXmlAtt.value = attributeValue ;
  97. // Set the attribute in the node.
  98. xmlNode.attributes.setNamedItem( oXmlAtt ) ;
  99. }
  100. catch (e)
  101. {}
  102. }
  103. FCKXHtml._AppendChildNodes = function( xmlNode, htmlNode, isBlockElement )
  104. {
  105. var oNode = htmlNode.firstChild ;
  106. while ( oNode )
  107. {
  108. this._AppendNode( xmlNode, oNode ) ;
  109. oNode = oNode.nextSibling ;
  110. }
  111. // Trim block elements. This is also needed to avoid Firefox leaving extra
  112. // BRs at the end of them.
  113. if ( isBlockElement && htmlNode.tagName && htmlNode.tagName.toLowerCase() != 'pre' )
  114. {
  115. FCKDomTools.TrimNode( xmlNode ) ;
  116. if ( FCKConfig.FillEmptyBlocks )
  117. {
  118. var lastChild = xmlNode.lastChild ;
  119. if ( lastChild && lastChild.nodeType == 1 && lastChild.nodeName == 'br' )
  120. this._AppendEntity( xmlNode, this._NbspEntity ) ;
  121. }
  122. }
  123. // If the resulting node is empty.
  124. if ( xmlNode.childNodes.length == 0 )
  125. {
  126. if ( isBlockElement && FCKConfig.FillEmptyBlocks )
  127. {
  128. this._AppendEntity( xmlNode, this._NbspEntity ) ;
  129. return xmlNode ;
  130. }
  131. var sNodeName = xmlNode.nodeName ;
  132. // Some inline elements are required to have something inside (span, strong, etc...).
  133. if ( FCKListsLib.InlineChildReqElements[ sNodeName ] )
  134. return null ;
  135. // We can't use short representation of empty elements that are not marked
  136. // as empty in th XHTML DTD.
  137. if ( !FCKListsLib.EmptyElements[ sNodeName ] )
  138. xmlNode.appendChild( this.XML.createTextNode('') ) ;
  139. }
  140. return xmlNode ;
  141. }
  142. FCKXHtml._AppendNode = function( xmlNode, htmlNode )
  143. {
  144. if ( !htmlNode )
  145. return false ;
  146. switch ( htmlNode.nodeType )
  147. {
  148. // Element Node.
  149. case 1 :
  150. // If we detect a <br> inside a <pre> in Gecko, turn it into a line break instead.
  151. // This is a workaround for the Gecko bug here: https://bugzilla.mozilla.org/show_bug.cgi?id=92921
  152. if ( FCKBrowserInfo.IsGecko
  153. && htmlNode.tagName.toLowerCase() == 'br'
  154. && htmlNode.parentNode.tagName.toLowerCase() == 'pre' )
  155. {
  156. var val = 'r' ;
  157. if ( htmlNode == htmlNode.parentNode.firstChild )
  158. val += 'r' ;
  159. return FCKXHtml._AppendNode( xmlNode, this.XML.createTextNode( val ) ) ;
  160. }
  161. // Here we found an element that is not the real element, but a
  162. // fake one (like the Flash placeholder image), so we must get the real one.
  163. if ( htmlNode.getAttribute('_fckfakelement') )
  164. return FCKXHtml._AppendNode( xmlNode, FCK.GetRealElement( htmlNode ) ) ;
  165. // Ignore bogus BR nodes in the DOM.
  166. if ( FCKBrowserInfo.IsGecko &&
  167. ( htmlNode.hasAttribute('_moz_editor_bogus_node') || htmlNode.getAttribute( 'type' ) == '_moz' ) )
  168. {
  169. if ( htmlNode.nextSibling )
  170. return false ;
  171. else
  172. {
  173. htmlNode.removeAttribute( '_moz_editor_bogus_node' ) ;
  174. htmlNode.removeAttribute( 'type' ) ;
  175. }
  176. }
  177. // This is for elements that are instrumental to FCKeditor and
  178. // must be removed from the final HTML.
  179. if ( htmlNode.getAttribute('_fcktemp') )
  180. return false ;
  181. // Get the element name.
  182. var sNodeName = htmlNode.tagName.toLowerCase()  ;
  183. if ( FCKBrowserInfo.IsIE )
  184. {
  185. // IE doens't include the scope name in the nodeName. So, add the namespace.
  186. if ( htmlNode.scopeName && htmlNode.scopeName != 'HTML' && htmlNode.scopeName != 'FCK' )
  187. sNodeName = htmlNode.scopeName.toLowerCase() + ':' + sNodeName ;
  188. }
  189. else
  190. {
  191. if ( sNodeName.StartsWith( 'fck:' ) )
  192. sNodeName = sNodeName.Remove( 0,4 ) ;
  193. }
  194. // Check if the node name is valid, otherwise ignore this tag.
  195. // If the nodeName starts with a slash, it is a orphan closing tag.
  196. // On some strange cases, the nodeName is empty, even if the node exists.
  197. if ( !FCKRegexLib.ElementName.test( sNodeName ) )
  198. return false ;
  199. // The already processed nodes must be marked to avoid then to be duplicated (bad formatted HTML).
  200. // So here, the "mark" is checked... if the element is Ok, then mark it.
  201. if ( htmlNode._fckxhtmljob && htmlNode._fckxhtmljob == FCKXHtml.CurrentJobNum )
  202. return false ;
  203. var oNode = this.XML.createElement( sNodeName ) ;
  204. // Add all attributes.
  205. FCKXHtml._AppendAttributes( xmlNode, htmlNode, oNode, sNodeName ) ;
  206. htmlNode._fckxhtmljob = FCKXHtml.CurrentJobNum ;
  207. // Tag specific processing.
  208. var oTagProcessor = FCKXHtml.TagProcessors[ sNodeName ] ;
  209. if ( oTagProcessor )
  210. oNode = oTagProcessor( oNode, htmlNode, xmlNode ) ;
  211. else
  212. oNode = this._AppendChildNodes( oNode, htmlNode, Boolean( FCKListsLib.NonEmptyBlockElements[ sNodeName ] ) ) ;
  213. if ( !oNode )
  214. return false ;
  215. xmlNode.appendChild( oNode ) ;
  216. break ;
  217. // Text Node.
  218. case 3 :
  219. if ( htmlNode.parentNode && htmlNode.parentNode.nodeName.IEquals( 'pre' ) )
  220. return this._AppendTextNode( xmlNode, htmlNode.nodeValue ) ;
  221. return this._AppendTextNode( xmlNode, htmlNode.nodeValue.ReplaceNewLineChars(' ') ) ;
  222. // Comment
  223. case 8 :
  224. // IE catches the <!DOTYPE ... > as a comment, but it has no
  225. // innerHTML, so we can catch it, and ignore it.
  226. if ( FCKBrowserInfo.IsIE && !htmlNode.innerHTML )
  227. break ;
  228. try { xmlNode.appendChild( this.XML.createComment( htmlNode.nodeValue ) ) ; }
  229. catch (e) { /* Do nothing... probably this is a wrong format comment. */ }
  230. break ;
  231. // Unknown Node type.
  232. default :
  233. xmlNode.appendChild( this.XML.createComment( "Element not supported - Type: " + htmlNode.nodeType + " Name: " + htmlNode.nodeName ) ) ;
  234. break ;
  235. }
  236. return true ;
  237. }
  238. // Append an item to the SpecialBlocks array and returns the tag to be used.
  239. FCKXHtml._AppendSpecialItem = function( item )
  240. {
  241. return '___FCKsi___' + ( FCKXHtml.SpecialBlocks.push( item ) - 1 ) ;
  242. }
  243. FCKXHtml._AppendEntity = function( xmlNode, entity )
  244. {
  245. xmlNode.appendChild( this.XML.createTextNode( '#?-:' + entity + ';' ) ) ;
  246. }
  247. FCKXHtml._AppendTextNode = function( targetNode, textValue )
  248. {
  249. var bHadText = textValue.length > 0 ;
  250. if ( bHadText )
  251. targetNode.appendChild( this.XML.createTextNode( textValue.replace( FCKXHtmlEntities.EntitiesRegex, FCKXHtml_GetEntity ) ) ) ;
  252. return bHadText ;
  253. }
  254. // Retrieves a entity (internal format) for a given character.
  255. function FCKXHtml_GetEntity( character )
  256. {
  257. // We cannot simply place the entities in the text, because the XML parser
  258. // will translate & to &amp;. So we use a temporary marker which is replaced
  259. // in the end of the processing.
  260. var sEntity = FCKXHtmlEntities.Entities[ character ] || ( '#' + character.charCodeAt(0) ) ;
  261. return '#?-:' + sEntity + ';' ;
  262. }
  263. // An object that hold tag specific operations.
  264. FCKXHtml.TagProcessors =
  265. {
  266. a : function( node, htmlNode )
  267. {
  268. // Firefox may create empty tags when deleting the selection in some special cases (SF-BUG 1556878).
  269. if ( htmlNode.innerHTML.Trim().length == 0 && !htmlNode.name )
  270. return false ;
  271. var sSavedUrl = htmlNode.getAttribute( '_fcksavedurl' ) ;
  272. if ( sSavedUrl != null )
  273. FCKXHtml._AppendAttribute( node, 'href', sSavedUrl ) ;
  274. // Anchors with content has been marked with an additional class, now we must remove it.
  275. if ( FCKBrowserInfo.IsIE )
  276. {
  277. // Buggy IE, doesn't copy the name of changed anchors.
  278. if ( htmlNode.name )
  279. FCKXHtml._AppendAttribute( node, 'name', htmlNode.name ) ;
  280. }
  281. node = FCKXHtml._AppendChildNodes( node, htmlNode, false ) ;
  282. return node ;
  283. },
  284. area : function( node, htmlNode )
  285. {
  286. var sSavedUrl = htmlNode.getAttribute( '_fcksavedurl' ) ;
  287. if ( sSavedUrl != null )
  288. FCKXHtml._AppendAttribute( node, 'href', sSavedUrl ) ;
  289. // IE ignores the "COORDS" and "SHAPE" attribute so we must add it manually.
  290. if ( FCKBrowserInfo.IsIE )
  291. {
  292. if ( ! node.attributes.getNamedItem( 'coords' ) )
  293. {
  294. var sCoords = htmlNode.getAttribute( 'coords', 2 ) ;
  295. if ( sCoords && sCoords != '0,0,0' )
  296. FCKXHtml._AppendAttribute( node, 'coords', sCoords ) ;
  297. }
  298. if ( ! node.attributes.getNamedItem( 'shape' ) )
  299. {
  300. var sShape = htmlNode.getAttribute( 'shape', 2 ) ;
  301. if ( sShape && sShape.length > 0 )
  302. FCKXHtml._AppendAttribute( node, 'shape', sShape.toLowerCase() ) ;
  303. }
  304. }
  305. return node ;
  306. },
  307. body : function( node, htmlNode )
  308. {
  309. node = FCKXHtml._AppendChildNodes( node, htmlNode, false ) ;
  310. // Remove spellchecker attributes added for Firefox when converting to HTML code (Bug #1351).
  311. node.removeAttribute( 'spellcheck' ) ;
  312. return node ;
  313. },
  314. // IE loses contents of iframes, and Gecko does give it back HtmlEncoded
  315. // Note: Opera does lose the content and doesn't provide it in the innerHTML string
  316. iframe : function( node, htmlNode )
  317. {
  318. var sHtml = htmlNode.innerHTML ;
  319. // Gecko does give back the encoded html
  320. if ( FCKBrowserInfo.IsGecko )
  321. sHtml = FCKTools.HTMLDecode( sHtml );
  322. // Remove the saved urls here as the data won't be processed as nodes
  323. sHtml = sHtml.replace( /s_fcksavedurl="[^"]*"/g, '' ) ;
  324. node.appendChild( FCKXHtml.XML.createTextNode( FCKXHtml._AppendSpecialItem( sHtml ) ) ) ;
  325. return node ;
  326. },
  327. img : function( node, htmlNode )
  328. {
  329. // The "ALT" attribute is required in XHTML.
  330. if ( ! node.attributes.getNamedItem( 'alt' ) )
  331. FCKXHtml._AppendAttribute( node, 'alt', '' ) ;
  332. var sSavedUrl = htmlNode.getAttribute( '_fcksavedurl' ) ;
  333. if ( sSavedUrl != null )
  334. FCKXHtml._AppendAttribute( node, 'src', sSavedUrl ) ;
  335. // Bug #768 : If the width and height are defined inline CSS,
  336. // don't define it again in the HTML attributes.
  337. if ( htmlNode.style.width )
  338. node.removeAttribute( 'width' ) ;
  339. if ( htmlNode.style.height )
  340. node.removeAttribute( 'height' ) ;
  341. return node ;
  342. },
  343. // Fix orphaned <li> nodes (Bug #503).
  344. li : function( node, htmlNode, targetNode )
  345. {
  346. // If the XML parent node is already a <ul> or <ol>, then add the <li> as usual.
  347. if ( targetNode.nodeName.IEquals( ['ul', 'ol'] ) )
  348. return FCKXHtml._AppendChildNodes( node, htmlNode, true ) ;
  349. var newTarget = FCKXHtml.XML.createElement( 'ul' ) ;
  350. // Reset the _fckxhtmljob so the HTML node is processed again.
  351. htmlNode._fckxhtmljob = null ;
  352. // Loop through all sibling LIs, adding them to the <ul>.
  353. do
  354. {
  355. FCKXHtml._AppendNode( newTarget, htmlNode ) ;
  356. // Look for the next element following this <li>.
  357. do
  358. {
  359. htmlNode = FCKDomTools.GetNextSibling( htmlNode ) ;
  360. } while ( htmlNode && htmlNode.nodeType == 3 && htmlNode.nodeValue.Trim().length == 0 )
  361. } while ( htmlNode && htmlNode.nodeName.toLowerCase() == 'li' )
  362. return newTarget ;
  363. },
  364. // Fix nested <ul> and <ol>.
  365. ol : function( node, htmlNode, targetNode )
  366. {
  367. if ( htmlNode.innerHTML.Trim().length == 0 )
  368. return false ;
  369. var ePSibling = targetNode.lastChild ;
  370. if ( ePSibling && ePSibling.nodeType == 3 )
  371. ePSibling = ePSibling.previousSibling ;
  372. if ( ePSibling && ePSibling.nodeName.toUpperCase() == 'LI' )
  373. {
  374. htmlNode._fckxhtmljob = null ;
  375. FCKXHtml._AppendNode( ePSibling, htmlNode ) ;
  376. return false ;
  377. }
  378. node = FCKXHtml._AppendChildNodes( node, htmlNode ) ;
  379. return node ;
  380. },
  381. pre : function ( node, htmlNode )
  382. {
  383. var firstChild = htmlNode.firstChild ;
  384. if ( firstChild && firstChild.nodeType == 3 )
  385. node.appendChild( FCKXHtml.XML.createTextNode( FCKXHtml._AppendSpecialItem( 'rn' ) ) ) ;
  386. FCKXHtml._AppendChildNodes( node, htmlNode, true ) ;
  387. return node ;
  388. },
  389. script : function( node, htmlNode )
  390. {
  391. // The "TYPE" attribute is required in XHTML.
  392. if ( ! node.attributes.getNamedItem( 'type' ) )
  393. FCKXHtml._AppendAttribute( node, 'type', 'text/javascript' ) ;
  394. node.appendChild( FCKXHtml.XML.createTextNode( FCKXHtml._AppendSpecialItem( htmlNode.text ) ) ) ;
  395. return node ;
  396. },
  397. span : function( node, htmlNode )
  398. {
  399. // Firefox may create empty tags when deleting the selection in some special cases (SF-BUG 1084404).
  400. if ( htmlNode.innerHTML.length == 0 )
  401. return false ;
  402. node = FCKXHtml._AppendChildNodes( node, htmlNode, false ) ;
  403. return node ;
  404. },
  405. style : function( node, htmlNode )
  406. {
  407. // The "TYPE" attribute is required in XHTML.
  408. if ( ! node.attributes.getNamedItem( 'type' ) )
  409. FCKXHtml._AppendAttribute( node, 'type', 'text/css' ) ;
  410. var cssText = htmlNode.innerHTML ;
  411. if ( FCKBrowserInfo.IsIE ) // Bug #403 : IE always appends a rn to the beginning of StyleNode.innerHTML
  412. cssText = cssText.replace( /^(rn|n|r)/, '' ) ;
  413. node.appendChild( FCKXHtml.XML.createTextNode( FCKXHtml._AppendSpecialItem( cssText ) ) ) ;
  414. return node ;
  415. },
  416. title : function( node, htmlNode )
  417. {
  418. node.appendChild( FCKXHtml.XML.createTextNode( FCK.EditorDocument.title ) ) ;
  419. return node ;
  420. }
  421. } ;
  422. FCKXHtml.TagProcessors.ul = FCKXHtml.TagProcessors.ol ;