亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > HTML > 正文

FCKeditor 源代碼分析附中文注釋

2024-08-26 00:15:56
字體:
來源:轉載
供稿:網友
FCKeditor 源代碼分析(一 ) fckeditor.js的中文注釋分析
 
 
 
這幾天都在研究FCKeditor的源代碼 (FCKeditor就是網絡中應用比較廣泛的網頁編輯器) 這里需要感謝nileaderblog的辛苦翻譯。

幾乎搜遍了Internet,似乎對于fckconfig.js這個文件講解的很多,但對于fckeditor.js這個FCK的核心類文件的資料幾乎為0. 

所以,花了整整一天的時間,以擠牙膏的方式,對fckeditor.js這個fck核心類文件作了自己力所能及的注釋,供同樣學習fck的網友一個參考。 

鑒于筆者水平有限,在此,請廣大高手指出我的注釋中不妥之處,以免誤導他人 。謝謝。 

建議copy到自己的IDE中查看 或者 
注:本文基于FCKeditor2.6.5 
更多權威資料,請參見 FCK 官方Developers Guide 
復制代碼代碼如下:

/** 

* ***********CopyRight************** 
*-------Annotated by nileader----- 
*-----Version 1.00 2009-10-18----- 
*-----Once copied, marked http://www.nileader.cn 

* FCKeditor 類 annotated by nileader 
* @param {Object} instanceName 編輯器的唯一名稱(相當于ID) 是不可省參數, 
* width,height,toolbarset,value 都是 可選參數 
*/ 
var FCKeditor = function( instanceName, width, height, toolbarSet, value ) 

//編輯器的基本屬性 注意:這些東西優先于FCKConfig.js中的配置 

this.InstanceName = instanceName ; //編輯器的唯一名稱(相當于ID)(必須有?。?nbsp;
this.Width = width || '100%' ; //寬度 默認是100% 
this.Height = height || '200' ; //寬度 默認是200 
this.ToolbarSet = toolbarSet || 'Default' ;//工具集名稱,默認值是Default 
this.Value = value || '' ; //初始化編輯器的HTML代碼,默認值為空 
//編輯器初始化的時候默認的根路徑, 其作用是編寫fck中,凡是用到的路徑,均從FCKeditor.BasePath目錄開始 默認為/Fckeditor/ 
this.BasePath = FCKeditor.BasePath ; 
this.CheckBrowser = true ; //是否在顯示編輯器前檢查瀏覽器兼容性,默認為true 
this.DisplayErrors = true ; //是否顯示提示錯誤,默為true 
this.Config = new Object() ; 
// Events 
this.OnError = null ; // function( source, errorNumber, errorDescription )自定義的錯誤處理函數 

FCKeditor.BasePath = '/fckeditor/' ; // fck默認的根目錄 
FCKeditor.MinHeight = 200 ; //高和寬的限制 
FCKeditor.MinWidth = 750 ; 
FCKeditor.prototype.Version = '2.6.5' ; //版本號 
FCKeditor.prototype.VersionBuild = '23959' ; 
/** 
* 調用CreateHtml()來生成編輯器的html代碼并在頁面上輸出編輯器 
*/ 
FCKeditor.prototype.Create = function() 

//調用createhtml()方法 
document.write( this.CreateHtml() ) ; 

/** 
* @return sHtml 用于生成編輯器的html代碼 
*/ 
FCKeditor.prototype.CreateHtml = function() 

// 檢查有無InstanceName 如果沒有則不生成html代碼 
if ( !this.InstanceName || this.InstanceName.length == 0 ) 

this._ThrowError( 701, 'You must specify an instance name.' ) ; 
return '' ; 

//函數的返回值 
var sHtml = '' ; 
/* 
* 當用戶的瀏覽器符合預設的幾種瀏覽器時, 
* 生成一個id="this.instancename" name="this.instancename"的文本框,事實上的內容儲存器 
*/ 
if ( !this.CheckBrowser || this._IsCompatibleBrowser() ) 

//將此時FCK初始值通過轉義之后放入這個input 
sHtml += '<input type="hidden" id="' + this.InstanceName + '" name="' + this.InstanceName + '" value="' + this._HTMLEncode( this.Value ) + '" style="display:none" style="display:none" />' ; 
//生成一個隱藏的INPUT來放置this.config中的內容 
sHtml += this._GetConfigHtml() ; 
//生成編輯器的iframe的代碼 
sHtml += this._GetIFrameHtml() ; 

/** 
* 如果用戶的瀏覽器不兼容FCK默認的幾種瀏覽器 
* 只能有傳統的textarea了 
*/ 
else 

var sWidth = this.Width.toString().indexOf('%') > 0 ? this.Width : this.Width + 'px' ; 
var sHeight = this.Height.toString().indexOf('%') > 0 ? this.Height : this.Height + 'px' ; 
sHtml += '<textarea name="' + this.InstanceName + 
'" rows="4" cols="40" style="width:' + sWidth + 
';height:' + sHeight ; 
if ( this.TabIndex ) 
sHtml += '" tabindex="' + this.TabIndex ; 
sHtml += '">' + 
this._HTMLEncode( this.Value ) + 
'<//textarea>' ; 

return sHtml ; 

/** 
* 用編輯器來替換對應的文本框 
*/ 
FCKeditor.prototype.ReplaceTextarea = function() 

//如果已經有了 id=THIS.INSTANCENAME___Frame 的標簽時,直接返回 
if ( document.getElementById( this.InstanceName + '___Frame' ) ) 
return ; 
//當用戶的瀏覽器符合預設的幾種瀏覽器時 
if ( !this.CheckBrowser || this._IsCompatibleBrowser() ) 

// We must check the elements firstly using the Id and then the name. 
//獲取id=this.InstanceName的html標簽 
var oTextarea = document.getElementById( this.InstanceName ) ; 
//獲取所有name=THIS.instancename的標簽 
var colElementsByName = document.getElementsByName( this.InstanceName ) ; 
var i = 0; 
/* 
* 考慮到用戶html標簽的命名不規范,所以進行以下編歷判斷 筆者指的是用戶在textarea標簽處用了name=this.instancename 
* 在同個頁面的其它標簽上也用了name=this.instancename 
*/ 
while ( oTextarea || i == 0 ) 

//遍歷,直到找到name=this.instancename的textarea標簽,并賦給oTextarea 
if ( oTextarea && oTextarea.tagName.toLowerCase() == 'textarea' ) 
break ; 
oTextarea = colElementsByName[i++] ; 

//如果不存在id或者name為this.instancename的標簽時,彈出錯誤框 
if ( !oTextarea ) 

alert( 'Error: The TEXTAREA with id or name set to "' + this.InstanceName + '" was not found' ) ;
return ; 

/* 
* 確定存在name=this.instancename的textarea標簽后,將編輯器的代碼賦給它 
*/ 
oTextarea.style.display = 'none' ; 
//如果頁面上對這樣的textarea標簽定義了tab鍵的順序,賦給this.TabIndex待用 
if ( oTextarea.tabIndex ) 
this.TabIndex = oTextarea.tabIndex ; 
this._InsertHtmlBefore( this._GetConfigHtml(), oTextarea ) ; 
this._InsertHtmlBefore( this._GetIFrameHtml(), oTextarea ) ; 





/** 
* 在指定的頁面標簽前面插入html代碼 
* @param {Object} 待插入的html代碼 
* @param {Object} 指定的頁面標簽(對象) 
*/ 
FCKeditor.prototype._InsertHtmlBefore = function( html, element ) 

if ( element.insertAdjacentHTML ) // IE 私有的 insertAdjacentHTML 方法 
element.insertAdjacentHTML( 'beforeBegin', html ) ; 
else // 非ie瀏覽器 


var oRange = document.createRange() ; 
oRange.setStartBefore( element ) ; 
var oFragment = oRange.createContextualFragment( html ); 
element.parentNode.insertBefore( oFragment, element ) ; 




/* 
* 通過編歷this.Config[]來生成一個隱藏域, 
* 例如: 
* this.Config['nileader']="1104",this.Config['leaderni']="nichao"…… 
* 那么,sConfig=…… &nileader=1104&leaderni=nichao …… 
* 當然,最終,sConfig會被encodeURIComponent函數轉換成百分比編碼 放入隱藏的INPUT中去 
*/ 
FCKeditor.prototype._GetConfigHtml = function() 

var sConfig = '' ; 
for ( var o in this.Config ) 

if ( sConfig.length > 0 ) sConfig += '&' ; 
//encodeURIComponent函數轉換成百分比編碼 
sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.Config[o] ) ; 

return '<input type="hidden" id="' + this.InstanceName + '___Config" value="' + sConfig + '" style="display:none" style="display:none" />' ; 


/* 
* 生成iframe的html 這里涉及到src的確定 
*/ 
FCKeditor.prototype._GetIFrameHtml = function() 

var sFile = 'fckeditor.html' ; 
//特殊情況 fckedito所在的窗口沒有嵌入在瀏覽器中 
try 

if ( (/fcksource=true/i).test( window.top.location.search ) ) 
sFile = 'fckeditor.original.html' ; 

catch (e) { /* 忽略這個異常. 很多時候,fckedito所在的窗口嵌入在瀏覽器中. */ } 
/* 
* 這里注意的一點: 
* iframe的工作原理: 當iframe處于可編輯狀態時,其實編輯的是src所在的頁面 
* 這里合成一個sLink以放入iframe標簽中 
*/ 
//sLink就是這個事實上的頁面了,從fck的根目錄開始,例如 sLink=/fckeditor/editor/fckeditor.html?InstanceName=nileader&Toolbar=nileadersbar 
var sLink = this.BasePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.InstanceName ) ; 
if (this.ToolbarSet) 
sLink += '&Toolbar=' + this.ToolbarSet ; 
//生成一個真正的編輯iframer的html代碼 當然,放入了src=slink 
var html = '<iframe id="' + this.InstanceName + 
'___Frame" src="' + sLink + 
'" src="' + sLink + 
'" width="' + this.Width + 
'" height="' + this.Height ; 
//如果設定了使用"Tab"鍵的遍歷順序,則賦給iframe 
if ( this.TabIndex ) 
html += '" tabindex="' + this.TabIndex ; 
html += '" frameborder="0" scrolling="no"></iframe>' ; 
return html ; 


/* 
* 檢測用戶的bowser是否是fck的默認 
* 這個方法只是fck公司追求oo,無意義 
*/ 
FCKeditor.prototype._IsCompatibleBrowser = function() 

return FCKeditor_IsCompatibleBrowser() ; 


/** 
* 拋出錯誤 
* @param {Object} errorNumber 錯誤編號 
* @param {Object} errorDescription 錯誤概述 
*/ 
FCKeditor.prototype._ThrowError = function( errorNumber, errorDescription ) 

this.ErrorNumber = errorNumber ; 
this.ErrorDescription = errorDescription ; 
//是否顯示提示錯誤,默為true 
if ( this.DisplayErrors ) 
{ //將錯誤編號和錯誤概述打印出來 
document.write( '<div style="COLOR: #ff0000" style="COLOR: #ff0000">' ) ; 
document.write( '[ FCKeditor Error ' + this.ErrorNumber + ': ' + this.ErrorDescription + ' ]' ) ; 
document.write( '</div>' ) ; 

//OnError是否自定義了錯誤處理函數,若定義了,由其處理 
if ( typeof( this.OnError ) == 'function' ) 
this.OnError( this, errorNumber, errorDescription ) ; 


/** 
* 轉義文本 
* @param {Object} text 待轉義的文本 
* @return String text 轉義完后的文本 
*/ 
FCKeditor.prototype._HTMLEncode = function( text ) 

if ( typeof( text ) != "string" ) 
text = text.toString() ; 
//將字符串中的所有 & " < > 用對應的轉義字符代換 
text = text.replace( 
/&/g, "&").replace( 
/"/g, """).replace( 
/</g, "<").replace( 
/>/g, ">") ; 
return text ; 


;(function() 

//把頁面上的textarea元素賦給editor變量 
var textareaToEditor = function( textarea ) 

var editor = new FCKeditor( textarea.name ) ; 
editor.Width = Math.max( textarea.offsetWidth, FCKeditor.MinWidth ) ; 
editor.Height = Math.max( textarea.offsetHeight, FCKeditor.MinHeight ) ; 
return editor ; 

/** 
* Replace all <textarea> elements available in the document with FCKeditor 
* instances. 

* // Replace all <textarea> elements in the page. 
* FCKeditor.ReplaceAllTextareas() ; 

* // Replace all <textarea class="myClassName"> elements in the page. 
* FCKeditor.ReplaceAllTextareas( 'myClassName' ) ; 

* // Selectively replace <textarea> elements, based on custom assertions. 
* FCKeditor.ReplaceAllTextareas( function( textarea, editor ) 
* { 
* // Custom code to evaluate the replace, returning false if it 
* // must not be done. 
* // It also passes the "editor" parameter, so the developer can 
* // customize the instance. 
* } ) ; 
*/ 
FCKeditor.ReplaceAllTextareas = function() 

//獲取所有的textarea元素 
var textareas = document.getElementsByTagName( 'textarea' ) ; 

for ( var i = 0 ; i < textareas.length ; i++ ) 

var editor = null ; 
var textarea = textareas[i] ; 
var name = textarea.name ; 
// The "name" attribute must exist. 
if ( !name || name.length == 0 ) 
continue ; 
if ( typeof arguments[0] == 'string' ) 

// The textarea class name could be passed as the function 
// parameter. 
var classRegex = new RegExp( '(?:^| )' + arguments[0] + '(?:$| )' ) ; 
if ( !classRegex.test( textarea.className ) ) 
continue ; 

else if ( typeof arguments[0] == 'function' ) 

// An assertion function could be passed as the function parameter. 
// It must explicitly return "false" to ignore a specific <textarea>. 
editor = textareaToEditor( textarea ) ; 
if ( arguments[0]( textarea, editor ) === false ) 
continue ; 

if ( !editor ) 
editor = textareaToEditor( textarea ) ; 
editor.ReplaceTextarea() ; 


})() ; 

/** 
* 檢測瀏覽器的兼容性 
* 利用了navigator對象返回的一些信息sAgent,判斷瀏覽器 返回包括 瀏覽器的碼名 瀏覽器名 瀏覽器版本 語言 等信息 并小寫 
* 例如: 
* mozilla/4.0 (compatible; msie 6.0; windows nt 5.2; sv1; .net clr 1.1.4322) 

* 判斷IE瀏覽器的時候,運用了IE4.0之后支持的增加了對條件編譯, 
* 由于只是IE支持,在W3C標準瀏覽器中,該屬性是不被支持的。因此,適當的利用該特性,判斷IE 
*/ 
function FCKeditor_IsCompatibleBrowser() 

var sAgent = navigator.userAgent.toLowerCase() ; 
// 當前瀏覽器是Internet Explorer 5.5+ 
//利用條件編譯判斷IE 在IE中,/*@cc_on!@*/false == !false == true, 
//如果是非IE瀏覽器,則忽略,/*@cc_on!@*/false == false 
if ( /*@cc_on!@*/false && sAgent.indexOf("mac") == -1 ) //不是apple mac os 

var sBrowserVersion = navigator.appVersion.match(/MSIE (./..)/)[1] ; 
return ( sBrowserVersion >= 5.5 ) ; 

// Gecko (Opera 9 tries to behave like Gecko at this point). 
//檢測是否是OPERA 9 瀏覽器 
if ( navigator.product == "Gecko" && navigator.productSub >= 20030210 && !( typeof(opera) == 'object' && opera.postError ) ) 
return true ; 
// Opera 9.50+ 
if ( window.opera && window.opera.version && parseFloat( window.opera.version() ) >= 9.5 ) 
return true ; 
// Adobe AIR 
// Checked before Safari because AIR have the WebKit rich text editor 
// features from Safari 3.0.4, but the version reported is 420. 
if ( sAgent.indexOf( ' adobeair/' ) != -1 ) 
return ( sAgent.match( / adobeair//(/d+)/ )[1] >= 1 ) ; // Build must be at least v1 
// Safari 3+ 
if ( sAgent.indexOf( ' applewebkit/' ) != -1 ) 
return ( sAgent.match( / applewebkit//(/d+)/ )[1] >= 522 ) ; // Build must be at least 522 (v3) 
return false ; 
}

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲一区二区三| 久久久女人电视剧免费播放下载| 国产成人在线一区二区| 一区二区三区四区精品| 久久人人爽国产| 亚洲精品国产精品国自产观看浪潮| 97香蕉超级碰碰久久免费软件| 欧美性xxxxx极品| 91免费精品视频| 国产91精品视频在线观看| 欧美理论片在线观看| 久久久在线免费观看| 国产成人精品一区二区在线| 97视频人免费观看| 欧美日韩亚洲系列| 欧美精品videos性欧美| 欧美黄色免费网站| 午夜精品在线观看| 国产精品久在线观看| 懂色aⅴ精品一区二区三区蜜月| 中文字幕免费精品一区| 亚洲自拍偷拍一区| 色多多国产成人永久免费网站| 欧美性感美女h网站在线观看免费| 日韩高清电影免费观看完整| 亚洲精品免费一区二区三区| 波霸ol色综合久久| 亚洲va久久久噜噜噜久久天堂| 中文字幕日韩在线观看| 亚洲欧美一区二区三区情侣bbw| 亚洲视频一区二区三区| 欧美大尺度在线观看| 在线视频欧美性高潮| 一本大道香蕉久在线播放29| 日本乱人伦a精品| 久久成年人视频| 欧美精品第一页在线播放| 中文字幕欧美亚洲| 国产精品jvid在线观看蜜臀| 日韩电影在线观看中文字幕| 久久久亚洲福利精品午夜| 97视频在线看| 国产精品亚洲自拍| 亚洲尤物视频网| 欧美成在线观看| 欧美日韩国产一区二区三区| 国产精品成熟老女人| 91精品久久久久| 狠狠操狠狠色综合网| 欧美在线一级视频| 黑丝美女久久久| 成人免费视频网址| 亚洲激情视频在线| 国产精品视频一区国模私拍| 国产精品免费视频久久久| 日韩欧美在线视频免费观看| 国产精品视频一区国模私拍| 91精品国产高清| 欧美一区二区三区免费观看| 九九热精品视频国产| 国产精品久久久久aaaa九色| 久久最新资源网| 国产91精品高潮白浆喷水| 欧美激情综合亚洲一二区| 亚洲无限av看| 日韩电影免费观看中文字幕| 国产精品青草久久久久福利99| 精品一区二区三区三区| 国产999在线观看| 丝袜一区二区三区| 国产精品私拍pans大尺度在线| 日韩免费中文字幕| 另类美女黄大片| 亚洲免费电影在线观看| 国产精品v日韩精品| 亚洲精品国产综合区久久久久久久| 国产精品视频精品视频| 欧美综合第一页| 精品夜色国产国偷在线| 亚洲有声小说3d| 日韩亚洲精品视频| 国产69精品久久久| 精品高清一区二区三区| 亚洲成人黄色网址| 亚洲精品黄网在线观看| 亚洲国产天堂网精品网站| 久久99久久99精品中文字幕| 国产视频久久久久久久| 欧美性极品xxxx娇小| 国产精品精品视频| 亚洲精品久久久久久久久久久久久| 91精品成人久久| 国产精品久久二区| 亚洲色图激情小说| 欧美性猛交丰臀xxxxx网站| 精品国内产的精品视频在线观看| 中文字幕精品国产| 亚洲福利视频在线| 亚洲欧美日韩中文在线| 国产欧美日韩中文| 日韩在线中文视频| 91精品啪aⅴ在线观看国产| 一区二区国产精品视频| 国产一区二区三区中文| 成人xxxx视频| 国产91精品高潮白浆喷水| 欧美精品亚州精品| 国产精品一二三视频| 亚洲bt天天射| 午夜精品久久久久久99热软件| 亚洲自拍偷拍色图| 影音先锋欧美精品| 8090成年在线看片午夜| 国产精品久久久久久久久久免费| 国产精品69精品一区二区三区| 欧美另类极品videosbestfree| 8x拔播拔播x8国产精品| 日韩视频免费在线| 久久精品国产一区二区三区| 国产精品成熟老女人| 一本色道久久88综合亚洲精品ⅰ| 美日韩精品视频免费看| 国内精品伊人久久| 97久久久久久| 国产婷婷色综合av蜜臀av| 中文字幕久久精品| 国产精品一区二区三区久久| 久久影视电视剧免费网站| 亚洲天堂免费视频| 欧美日韩国产麻豆| 成人a级免费视频| 日韩精品在线视频| 韩国国内大量揄拍精品视频| 国产主播喷水一区二区| 亚洲综合第一页| 琪琪第一精品导航| 亚洲第一区在线| www.日韩视频| 久久久久久97| 国产精品入口福利| 久久久精品影院| 亚洲石原莉奈一区二区在线观看| 91夜夜未满十八勿入爽爽影院| 日本道色综合久久影院| 日本a级片电影一区二区| 亚洲香蕉av在线一区二区三区| 国产日韩亚洲欧美| 欧美大尺度在线观看| 欧美日韩免费网站| 亚洲天堂成人在线| 在线观看精品自拍私拍| 日韩av免费在线看| 日韩免费在线看| 亚洲伊人一本大道中文字幕| 欧美精品久久久久久久免费观看| 国产视频一区在线| 在线观看欧美视频| 中文字幕日韩欧美| 欧美日韩亚洲视频| 91天堂在线观看| 欧美视频中文在线看| 4388成人网| 精品一区二区三区三区| 九九九久久久久久|