You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
22478 lines
740 KiB
22478 lines
740 KiB
/*
|
|
Copyright (C) NAVER corp.
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
if(typeof window.nhn=='undefined'){window.nhn = {};}
|
|
if (!nhn.husky){nhn.husky = {};}
|
|
/**
|
|
* @fileOverview This file contains Husky framework core
|
|
* @name HuskyCore.js
|
|
*/
|
|
(function(){
|
|
var _rxMsgHandler = /^\$(LOCAL|BEFORE|ON|AFTER)_/,
|
|
_rxMsgAppReady = /^\$(BEFORE|ON|AFTER)_MSG_APP_READY$/,
|
|
_aHuskyCores = [], // HuskyCore instance list
|
|
_htLoadedFile = {}; // lazy-loaded file list
|
|
|
|
nhn.husky.HuskyCore = jindo.$Class({
|
|
name : "HuskyCore",
|
|
aCallerStack : null,
|
|
bMobile : jindo.$Agent().navigator().mobile || jindo.$Agent().navigator().msafari,
|
|
|
|
$init : function(htOptions){
|
|
this.htOptions = htOptions||{};
|
|
|
|
_aHuskyCores.push(this);
|
|
if( this.htOptions.oDebugger ){
|
|
nhn.husky.HuskyCore.getCore = function() {
|
|
return _aHuskyCores;
|
|
};
|
|
this.htOptions.oDebugger.setApp(this);
|
|
}
|
|
|
|
// To prevent processing a Husky message before all the plugins are registered and ready,
|
|
// Queue up all the messages here until the application's status is changed to READY
|
|
this.messageQueue = [];
|
|
|
|
this.oMessageMap = {};
|
|
this.oDisabledMessage = {};
|
|
this.oLazyMessage = {};
|
|
this.aPlugins = [];
|
|
|
|
this.appStatus = nhn.husky.APP_STATUS.NOT_READY;
|
|
|
|
this.aCallerStack = [];
|
|
|
|
this._fnWaitForPluginReady = jindo.$Fn(this._waitForPluginReady, this).bind();
|
|
|
|
// Register the core as a plugin so it can receive messages
|
|
this.registerPlugin(this);
|
|
},
|
|
|
|
setDebugger: function(oDebugger) {
|
|
this.htOptions.oDebugger = oDebugger;
|
|
oDebugger.setApp(this);
|
|
},
|
|
|
|
exec : function(msg, args, oEvent){
|
|
// If the application is not yet ready just queue the message
|
|
if(this.appStatus == nhn.husky.APP_STATUS.NOT_READY){
|
|
this.messageQueue[this.messageQueue.length] = {'msg':msg, 'args':args, 'event':oEvent};
|
|
return true;
|
|
}
|
|
|
|
this.exec = this._exec;
|
|
this.exec(msg, args, oEvent);
|
|
},
|
|
|
|
delayedExec : function(msg, args, nDelay, oEvent){
|
|
var fExec = jindo.$Fn(this.exec, this).bind(msg, args, oEvent);
|
|
setTimeout(fExec, nDelay);
|
|
},
|
|
|
|
_exec : function(msg, args, oEvent){
|
|
return (this._exec = this.htOptions.oDebugger?this._execWithDebugger:this._execWithoutDebugger).call(this, msg, args, oEvent);
|
|
},
|
|
_execWithDebugger : function(msg, args, oEvent){
|
|
this.htOptions.oDebugger.log_MessageStart(msg, args);
|
|
var bResult = this._doExec(msg, args, oEvent);
|
|
this.htOptions.oDebugger.log_MessageEnd(msg, args);
|
|
return bResult;
|
|
},
|
|
_execWithoutDebugger : function(msg, args, oEvent){
|
|
return this._doExec(msg, args, oEvent);
|
|
},
|
|
_doExec : function(msg, args, oEvent){
|
|
var bContinue = false;
|
|
|
|
// Lazy메시지가 있으면 파일을 로딩한다.
|
|
if(this.oLazyMessage[msg]){
|
|
var htLazyInfo = this.oLazyMessage[msg];
|
|
this._loadLazyFiles(msg, args, oEvent, htLazyInfo.aFilenames, 0);
|
|
return false;
|
|
}
|
|
|
|
if(!this.oDisabledMessage[msg]){
|
|
var allArgs = [];
|
|
if(args && args.length){
|
|
var iLen = args.length;
|
|
for(var i=0; i<iLen; i++){allArgs[i] = args[i];}
|
|
}
|
|
if(oEvent){allArgs[allArgs.length] = oEvent;}
|
|
|
|
bContinue = this._execMsgStep("BEFORE", msg, allArgs);
|
|
if(bContinue){bContinue = this._execMsgStep("ON", msg, allArgs);}
|
|
if(bContinue){bContinue = this._execMsgStep("AFTER", msg, allArgs);}
|
|
}
|
|
|
|
return bContinue;
|
|
},
|
|
|
|
|
|
registerPlugin : function(oPlugin){
|
|
if(!oPlugin){throw("An error occured in registerPlugin(): invalid plug-in");}
|
|
|
|
oPlugin.nIdx = this.aPlugins.length;
|
|
oPlugin.oApp = this;
|
|
this.aPlugins[oPlugin.nIdx] = oPlugin;
|
|
|
|
// If the plugin does not specify that it takes time to be ready, change the stauts to READY right away
|
|
if(oPlugin.status != nhn.husky.PLUGIN_STATUS.NOT_READY){oPlugin.status = nhn.husky.PLUGIN_STATUS.READY;}
|
|
|
|
// If run() function had been called already, need to recreate the message map
|
|
if(this.appStatus != nhn.husky.APP_STATUS.NOT_READY){
|
|
for(var funcName in oPlugin){
|
|
if(_rxMsgHandler.test(funcName)){
|
|
this.addToMessageMap(funcName, oPlugin);
|
|
}
|
|
}
|
|
}
|
|
|
|
this.exec("MSG_PLUGIN_REGISTERED", [oPlugin]);
|
|
|
|
return oPlugin.nIdx;
|
|
},
|
|
|
|
disableMessage : function(sMessage, bDisable){this.oDisabledMessage[sMessage] = bDisable;},
|
|
|
|
registerBrowserEvent : function(obj, sEvent, sMessage, aParams, nDelay){
|
|
aParams = aParams || [];
|
|
var func = (nDelay)?jindo.$Fn(this.delayedExec, this).bind(sMessage, aParams, nDelay):jindo.$Fn(this.exec, this).bind(sMessage, aParams);
|
|
return jindo.$Fn(func, this).attach(obj, sEvent);
|
|
},
|
|
|
|
run : function(htOptions){
|
|
this.htRunOptions = htOptions || {};
|
|
|
|
// Change the status from NOT_READY to let exec to process all the way
|
|
this._changeAppStatus(nhn.husky.APP_STATUS.WAITING_FOR_PLUGINS_READY);
|
|
|
|
// Process all the messages in the queue
|
|
var iQueueLength = this.messageQueue.length;
|
|
for(var i=0; i<iQueueLength; i++){
|
|
var curMsgAndArgs = this.messageQueue[i];
|
|
this.exec(curMsgAndArgs.msg, curMsgAndArgs.args, curMsgAndArgs.event);
|
|
}
|
|
|
|
this._fnWaitForPluginReady();
|
|
},
|
|
|
|
acceptLocalBeforeFirstAgain : function(oPlugin, bAccept){
|
|
// LOCAL_BEFORE_FIRST will be fired again if oPlugin._husky_bRun == false
|
|
oPlugin._husky_bRun = !bAccept;
|
|
},
|
|
|
|
// Use this also to update the mapping
|
|
createMessageMap : function(sMsgHandler){
|
|
this.oMessageMap[sMsgHandler] = [];
|
|
|
|
var nLen = this.aPlugins.length;
|
|
for(var i=0; i<nLen; i++){this._doAddToMessageMap(sMsgHandler, this.aPlugins[i]);}
|
|
},
|
|
|
|
addToMessageMap : function(sMsgHandler, oPlugin){
|
|
// cannot "ADD" unless the map is already created.
|
|
// the message will be added automatically to the mapping when it is first passed anyways, so do not add now
|
|
if(!this.oMessageMap[sMsgHandler]){return;}
|
|
|
|
this._doAddToMessageMap(sMsgHandler, oPlugin);
|
|
},
|
|
|
|
_changeAppStatus : function(appStatus){
|
|
this.appStatus = appStatus;
|
|
|
|
// Initiate MSG_APP_READY if the application's status is being switched to READY
|
|
if(this.appStatus == nhn.husky.APP_STATUS.READY){this.exec("MSG_APP_READY");}
|
|
},
|
|
|
|
|
|
_execMsgStep : function(sMsgStep, sMsg, args){
|
|
return (this._execMsgStep = this.htOptions.oDebugger?this._execMsgStepWithDebugger:this._execMsgStepWithoutDebugger).call(this, sMsgStep, sMsg, args);
|
|
},
|
|
_execMsgStepWithDebugger : function(sMsgStep, sMsg, args){
|
|
this.htOptions.oDebugger.log_MessageStepStart(sMsgStep, sMsg, args);
|
|
var bStatus = this._execMsgHandler("$"+sMsgStep+"_"+sMsg, args);
|
|
this.htOptions.oDebugger.log_MessageStepEnd(sMsgStep, sMsg, args);
|
|
return bStatus;
|
|
},
|
|
_execMsgStepWithoutDebugger : function(sMsgStep, sMsg, args){
|
|
return this._execMsgHandler ("$"+sMsgStep+"_"+sMsg, args);
|
|
},
|
|
_execMsgHandler : function(sMsgHandler, args){
|
|
var i;
|
|
if(!this.oMessageMap[sMsgHandler]){
|
|
this.createMessageMap(sMsgHandler);
|
|
}
|
|
|
|
var aPlugins = this.oMessageMap[sMsgHandler];
|
|
var iNumOfPlugins = aPlugins.length;
|
|
|
|
if(iNumOfPlugins === 0){return true;}
|
|
|
|
var bResult = true;
|
|
|
|
// two similar codes were written twice due to the performace.
|
|
if(_rxMsgAppReady.test(sMsgHandler)){
|
|
for(i=0; i<iNumOfPlugins; i++){
|
|
if(this._execHandler(aPlugins[i], sMsgHandler, args) === false){
|
|
bResult = false;
|
|
break;
|
|
}
|
|
}
|
|
}else{
|
|
for(i=0; i<iNumOfPlugins; i++){
|
|
if(!aPlugins[i]._husky_bRun){
|
|
aPlugins[i]._husky_bRun = true;
|
|
if(typeof aPlugins[i].$LOCAL_BEFORE_FIRST == "function" && this._execHandler(aPlugins[i], "$LOCAL_BEFORE_FIRST", [sMsgHandler, args]) === false){continue;}
|
|
}
|
|
|
|
if(typeof aPlugins[i].$LOCAL_BEFORE_ALL == "function"){
|
|
if(this._execHandler(aPlugins[i], "$LOCAL_BEFORE_ALL", [sMsgHandler, args]) === false){continue;}
|
|
}
|
|
|
|
if(this._execHandler(aPlugins[i], sMsgHandler, args) === false){
|
|
bResult = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return bResult;
|
|
},
|
|
|
|
|
|
_execHandler : function(oPlugin, sHandler, args){
|
|
return (this._execHandler = this.htOptions.oDebugger?this._execHandlerWithDebugger:this._execHandlerWithoutDebugger).call(this, oPlugin, sHandler, args);
|
|
},
|
|
_execHandlerWithDebugger : function(oPlugin, sHandler, args){
|
|
this.htOptions.oDebugger.log_CallHandlerStart(oPlugin, sHandler, args);
|
|
var bResult;
|
|
try{
|
|
this.aCallerStack.push(oPlugin);
|
|
bResult = oPlugin[sHandler].apply(oPlugin, args);
|
|
this.aCallerStack.pop();
|
|
}catch(e){
|
|
this.htOptions.oDebugger.handleException(e);
|
|
bResult = false;
|
|
}
|
|
this.htOptions.oDebugger.log_CallHandlerEnd(oPlugin, sHandler, args);
|
|
return bResult;
|
|
},
|
|
_execHandlerWithoutDebugger : function(oPlugin, sHandler, args){
|
|
this.aCallerStack.push(oPlugin);
|
|
var bResult = oPlugin[sHandler].apply(oPlugin, args);
|
|
this.aCallerStack.pop();
|
|
|
|
return bResult;
|
|
},
|
|
|
|
|
|
_doAddToMessageMap : function(sMsgHandler, oPlugin){
|
|
if(typeof oPlugin[sMsgHandler] != "function"){return;}
|
|
|
|
var aMap = this.oMessageMap[sMsgHandler];
|
|
// do not add if the plugin is already in the mapping
|
|
for(var i=0, iLen=aMap.length; i<iLen; i++){
|
|
if(this.oMessageMap[sMsgHandler][i] == oPlugin){return;}
|
|
}
|
|
this.oMessageMap[sMsgHandler][i] = oPlugin;
|
|
},
|
|
|
|
_waitForPluginReady : function(){
|
|
var bAllReady = true;
|
|
for(var i=0; i<this.aPlugins.length; i++){
|
|
if(this.aPlugins[i].status == nhn.husky.PLUGIN_STATUS.NOT_READY){
|
|
bAllReady = false;
|
|
break;
|
|
}
|
|
}
|
|
if(bAllReady){
|
|
this._changeAppStatus(nhn.husky.APP_STATUS.READY);
|
|
}else{
|
|
setTimeout(this._fnWaitForPluginReady, 100);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Lazy로딩을 실행한다.
|
|
* @param {Object} oPlugin 플러그인 인스턴스
|
|
* @param {String} sMsg 메시지명
|
|
* @param {Array} aArgs 메시지에 전달되는 매개변수
|
|
* @param {Event} oEvent 메시지에 전달되는 이벤트
|
|
* @param {Array} aFilenames Lazy로딩할 파일명
|
|
* @param {Integer} nIdx 로딩할 파일인덱스
|
|
*/
|
|
_loadLazyFiles : function(sMsg, aArgs, oEvent, aFilenames, nIdx){
|
|
var nLen = aFilenames.length;
|
|
if(nLen <= nIdx){
|
|
// 파일이 모두 로딩된 상태라면 oLazyMessage 에서 정보를 제거하고 메시지를 실행한다.
|
|
this.oLazyMessage[sMsg] = null;
|
|
this.oApp.exec(sMsg, aArgs, oEvent);
|
|
return;
|
|
}
|
|
|
|
var sFilename = aFilenames[nIdx];
|
|
|
|
if(_htLoadedFile[sFilename]){
|
|
// 파일이 이미 로딩된 경우 다음 파일을 로딩한다.
|
|
this._loadLazyFiles(sMsg, aArgs, oEvent, aFilenames, nIdx+1);
|
|
}else{
|
|
// 파일을 Lazy로딩한다.
|
|
// TODO: 진도컴포넌트 디펜던시 제거?
|
|
// TODO: 응답결과가 정상적이지 않을 경우에 대한 처리?
|
|
jindo.LazyLoading.load(nhn.husky.SE2M_Configuration.LazyLoad.sJsBaseURI+"/"+sFilename,
|
|
jindo.$Fn(function(sMsg, aArgs, oEvent, aFilenames, nIdx){
|
|
// 로딩완료된 파일은 상태를 변경하고
|
|
var sFilename = aFilenames[nIdx];
|
|
_htLoadedFile[sFilename] = 1;
|
|
// 다음 파일을 로딩한다.
|
|
this._loadLazyFiles(sMsg, aArgs, oEvent, aFilenames, nIdx+1);
|
|
}, this).bind(sMsg, aArgs, oEvent, aFilenames, nIdx),
|
|
"utf-8"
|
|
);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Lazy로딩으로 처리할 메시지를 등록한다.
|
|
* @param {Array} aMsgs 메시지명
|
|
* @param {Array} aFilenames Lazy로딩할 파일명
|
|
*/
|
|
registerLazyMessage : function(aMsgs, aFilenames){
|
|
aMsgs = aMsgs || [];
|
|
aFilenames = aFilenames || [];
|
|
|
|
for(var i = 0, sMsg, htLazyInfo; (sMsg = aMsgs[i]); i++){
|
|
htLazyInfo = this.oLazyMessage[sMsg];
|
|
if(htLazyInfo){
|
|
htLazyInfo.aFilenames = htLazyInfo.aFilenames.concat(aFilenames);
|
|
}else{
|
|
this.oLazyMessage[sMsg] = {
|
|
sMsg : sMsg,
|
|
aFilenames : aFilenames
|
|
};
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Lazy로딩완료된 파일목록
|
|
*/
|
|
nhn.husky.HuskyCore._htLoadedFile = {};
|
|
/**
|
|
* Lazy로딩완료된 파일목록에 파일명을 추가한다.
|
|
* @param {String} sFilename Lazy로딩완료될 경우 마킹할 파일명
|
|
*/
|
|
nhn.husky.HuskyCore.addLoadedFile = function(sFilename){
|
|
_htLoadedFile[sFilename] = 1;
|
|
};
|
|
/**
|
|
* 플러그인 일부분을 Lazy로딩하여 쉽게 확장할 수 있도록 믹스인 기능을 제공한다.
|
|
* @param {Class} oClass 믹스인을 적용할 클래스
|
|
* @param {Object} htMixin 덧붙일 프로토타입 데이터
|
|
* @param {Boolean} bOverride 원본 클래스에 프로토타입을 덮어씌울지 여부
|
|
*/
|
|
nhn.husky.HuskyCore.mixin = function(oClass, htMixin, bOverride, sFilename){
|
|
//TODO: error handling?
|
|
// if(typeof oClass != "function"){
|
|
// throw new Error("SmartEditor: can't mixin (oClass is invalid)");
|
|
// }
|
|
var aPlugins = [];
|
|
// 믹스인을 적용할 클래스가 이미 플러그인으로 등록된 상태라면
|
|
for(var i = 0, oHuskyCore; (oHuskyCore = _aHuskyCores[i]); i++){
|
|
for(var j = 0, oPlugin; (oPlugin = oHuskyCore.aPlugins[j]); j++){
|
|
if(oPlugin instanceof oClass){
|
|
// 1. 메시지 추가등록을 위해 해당 플러그인 인스턴스를 담아두고
|
|
aPlugins.push(oPlugin);
|
|
// 2. 해당 플러그인 인스턴스에 $LOCAL_BEFORE_FIRST 핸들러가 없으면 핸들러처리를 위한 플래그를 리셋한다.
|
|
// if there were no $LOCAL_BEFORE_FIRST in already-loaded script, set to accept $LOCAL_BEFORE_FIRST next time as the function could be included in the lazy-loaded script.
|
|
if(typeof oPlugin["$LOCAL_BEFORE_FIRST"] !== "function"){
|
|
oPlugin.oApp.acceptLocalBeforeFirstAgain(oPlugin, true);
|
|
}
|
|
}else if(oPlugin._$superClass === oClass){
|
|
// [SMARTEDITORSUS-1697]
|
|
// jindo 클래스를 상속받아 확장된 클래스의 경우,
|
|
// 1. instanceof 로 확인이 안되며
|
|
// 2. super 클래스에 mixin 처리한 것이 반영이 안된다.
|
|
// 따라서 상속된 jindo 클래스의 인스턴스는 인스턴스에 직접 mixin 처리한다.
|
|
if(typeof oPlugin["$LOCAL_BEFORE_FIRST"] !== "function"){
|
|
oPlugin.oApp.acceptLocalBeforeFirstAgain(oPlugin, true);
|
|
}
|
|
for(var k in htMixin){
|
|
if(bOverride || !oPlugin.hasOwnProperty(k)){
|
|
oPlugin[k] = htMixin[k];
|
|
if(_rxMsgHandler.test(k)){
|
|
oPlugin.oApp.addToMessageMap(k, oPlugin);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// mixin 처리
|
|
for(var k in htMixin){
|
|
if(bOverride || !oClass.prototype.hasOwnProperty(k)){
|
|
oClass.prototype[k] = htMixin[k];
|
|
// 새로 추가되는 함수가 메시지 핸들러라면 메시지 매핑에 추가 해준다.
|
|
if(_rxMsgHandler.test(k)){
|
|
for(var j = 0, oPlugin; (oPlugin = aPlugins[j]); j++){
|
|
oPlugin.oApp.addToMessageMap(k, oPlugin);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
nhn.husky.APP_STATUS = {
|
|
'NOT_READY' : 0,
|
|
'WAITING_FOR_PLUGINS_READY' : 1,
|
|
'READY' : 2
|
|
};
|
|
|
|
nhn.husky.PLUGIN_STATUS = {
|
|
'NOT_READY' : 0,
|
|
'READY' : 1
|
|
};
|
|
})();
|
|
if(typeof window.nhn=='undefined'){window.nhn = {};}
|
|
|
|
nhn.CurrentSelection_IE = function(){
|
|
this.getCommonAncestorContainer = function(){
|
|
try{
|
|
this._oSelection = this._document.selection;
|
|
if(this._oSelection.type == "Control"){
|
|
return this._oSelection.createRange().item(0);
|
|
}else{
|
|
return this._oSelection.createRangeCollection().item(0).parentElement();
|
|
}
|
|
}catch(e){
|
|
return this._document.body;
|
|
}
|
|
};
|
|
|
|
this.isCollapsed = function(){
|
|
this._oSelection = this._document.selection;
|
|
|
|
return this._oSelection.type == "None";
|
|
};
|
|
};
|
|
|
|
nhn.CurrentSelection_FF = function(){
|
|
this.getCommonAncestorContainer = function(){
|
|
return this._getSelection().commonAncestorContainer;
|
|
};
|
|
|
|
this.isCollapsed = function(){
|
|
var oSelection = this._window.getSelection();
|
|
|
|
if(oSelection.rangeCount<1){ return true; }
|
|
return oSelection.getRangeAt(0).collapsed;
|
|
};
|
|
|
|
this._getSelection = function(){
|
|
try{
|
|
return this._window.getSelection().getRangeAt(0);
|
|
}catch(e){
|
|
return this._document.createRange();
|
|
}
|
|
};
|
|
};
|
|
|
|
nhn.CurrentSelection = new (jindo.$Class({
|
|
$init : function(){
|
|
var oAgentInfo = jindo.$Agent().navigator();
|
|
if(oAgentInfo.ie && document.selection){
|
|
nhn.CurrentSelection_IE.apply(this);
|
|
}else{
|
|
nhn.CurrentSelection_FF.apply(this);
|
|
}
|
|
},
|
|
|
|
setWindow : function(oWin){
|
|
this._window = oWin;
|
|
this._document = oWin.document;
|
|
}
|
|
}))();
|
|
|
|
/**
|
|
* @fileOverview This file contains a cross-browser implementation of W3C's DOM Range
|
|
* @name W3CDOMRange.js
|
|
*/
|
|
nhn.W3CDOMRange = jindo.$Class({
|
|
$init : function(win){
|
|
this.reset(win);
|
|
},
|
|
|
|
reset : function(win){
|
|
this._window = win;
|
|
this._document = this._window.document;
|
|
|
|
this.collapsed = true;
|
|
this.commonAncestorContainer = this._document.body;
|
|
this.endContainer = this._document.body;
|
|
this.endOffset = 0;
|
|
this.startContainer = this._document.body;
|
|
this.startOffset = 0;
|
|
|
|
this.oBrowserSelection = new nhn.BrowserSelection(this._window);
|
|
this.selectionLoaded = this.oBrowserSelection.selectionLoaded;
|
|
},
|
|
|
|
cloneContents : function(){
|
|
var oClonedContents = this._document.createDocumentFragment();
|
|
var oTmpContainer = this._document.createDocumentFragment();
|
|
|
|
var aNodes = this._getNodesInRange();
|
|
|
|
if(aNodes.length < 1){return oClonedContents;}
|
|
|
|
var oClonedContainers = this._constructClonedTree(aNodes, oTmpContainer);
|
|
|
|
// oTopContainer = aNodes[aNodes.length-1].parentNode and this is not part of the initial array and only those child nodes should be cloned
|
|
var oTopContainer = oTmpContainer.firstChild;
|
|
|
|
if(oTopContainer){
|
|
var elCurNode = oTopContainer.firstChild;
|
|
var elNextNode;
|
|
|
|
while(elCurNode){
|
|
elNextNode = elCurNode.nextSibling;
|
|
oClonedContents.appendChild(elCurNode);
|
|
elCurNode = elNextNode;
|
|
}
|
|
}
|
|
|
|
oClonedContainers = this._splitTextEndNodes({oStartContainer: oClonedContainers.oStartContainer, iStartOffset: this.startOffset,
|
|
oEndContainer: oClonedContainers.oEndContainer, iEndOffset: this.endOffset});
|
|
|
|
if(oClonedContainers.oStartContainer && oClonedContainers.oStartContainer.previousSibling){
|
|
nhn.DOMFix.parentNode(oClonedContainers.oStartContainer).removeChild(oClonedContainers.oStartContainer.previousSibling);
|
|
}
|
|
|
|
if(oClonedContainers.oEndContainer && oClonedContainers.oEndContainer.nextSibling){
|
|
nhn.DOMFix.parentNode(oClonedContainers.oEndContainer).removeChild(oClonedContainers.oEndContainer.nextSibling);
|
|
}
|
|
|
|
return oClonedContents;
|
|
},
|
|
|
|
_constructClonedTree : function(aNodes, oClonedParentNode){
|
|
var oClonedStartContainer = null;
|
|
var oClonedEndContainer = null;
|
|
|
|
var oStartContainer = this.startContainer;
|
|
var oEndContainer = this.endContainer;
|
|
|
|
var _recurConstructClonedTree = function(aAllNodes, iCurIdx, oClonedParentNode){
|
|
|
|
if(iCurIdx < 0){return iCurIdx;}
|
|
|
|
var iChildIdx = iCurIdx-1;
|
|
|
|
var oCurNodeCloneWithChildren = aAllNodes[iCurIdx].cloneNode(false);
|
|
|
|
if(aAllNodes[iCurIdx] == oStartContainer){oClonedStartContainer = oCurNodeCloneWithChildren;}
|
|
if(aAllNodes[iCurIdx] == oEndContainer){oClonedEndContainer = oCurNodeCloneWithChildren;}
|
|
|
|
while(iChildIdx >= 0 && nhn.DOMFix.parentNode(aAllNodes[iChildIdx]) == aAllNodes[iCurIdx]){
|
|
iChildIdx = this._recurConstructClonedTree(aAllNodes, iChildIdx, oCurNodeCloneWithChildren);
|
|
}
|
|
|
|
// this may trigger an error message in IE when an erroneous script is inserted
|
|
oClonedParentNode.insertBefore(oCurNodeCloneWithChildren, oClonedParentNode.firstChild);
|
|
|
|
return iChildIdx;
|
|
};
|
|
this._recurConstructClonedTree = _recurConstructClonedTree;
|
|
aNodes[aNodes.length] = nhn.DOMFix.parentNode(aNodes[aNodes.length-1]);
|
|
this._recurConstructClonedTree(aNodes, aNodes.length-1, oClonedParentNode);
|
|
|
|
return {oStartContainer: oClonedStartContainer, oEndContainer: oClonedEndContainer};
|
|
},
|
|
|
|
cloneRange : function(){
|
|
return this._copyRange(new nhn.W3CDOMRange(this._window));
|
|
},
|
|
|
|
_copyRange : function(oClonedRange){
|
|
oClonedRange.collapsed = this.collapsed;
|
|
oClonedRange.commonAncestorContainer = this.commonAncestorContainer;
|
|
oClonedRange.endContainer = this.endContainer;
|
|
oClonedRange.endOffset = this.endOffset;
|
|
oClonedRange.startContainer = this.startContainer;
|
|
oClonedRange.startOffset = this.startOffset;
|
|
oClonedRange._document = this._document;
|
|
|
|
return oClonedRange;
|
|
},
|
|
|
|
collapse : function(toStart){
|
|
if(toStart){
|
|
this.endContainer = this.startContainer;
|
|
this.endOffset = this.startOffset;
|
|
}else{
|
|
this.startContainer = this.endContainer;
|
|
this.startOffset = this.endOffset;
|
|
}
|
|
|
|
this._updateRangeInfo();
|
|
},
|
|
|
|
compareBoundaryPoints : function(how, sourceRange){
|
|
switch(how){
|
|
case nhn.W3CDOMRange.START_TO_START:
|
|
return this._compareEndPoint(this.startContainer, this.startOffset, sourceRange.startContainer, sourceRange.startOffset);
|
|
case nhn.W3CDOMRange.START_TO_END:
|
|
return this._compareEndPoint(this.endContainer, this.endOffset, sourceRange.startContainer, sourceRange.startOffset);
|
|
case nhn.W3CDOMRange.END_TO_END:
|
|
return this._compareEndPoint(this.endContainer, this.endOffset, sourceRange.endContainer, sourceRange.endOffset);
|
|
case nhn.W3CDOMRange.END_TO_START:
|
|
return this._compareEndPoint(this.startContainer, this.startOffset, sourceRange.endContainer, sourceRange.endOffset);
|
|
}
|
|
},
|
|
|
|
_findBody : function(oNode){
|
|
if(!oNode){return null;}
|
|
while(oNode){
|
|
if(oNode.tagName == "BODY"){return oNode;}
|
|
oNode = nhn.DOMFix.parentNode(oNode);
|
|
}
|
|
return null;
|
|
},
|
|
|
|
_compareEndPoint : function(oContainerA, iOffsetA, oContainerB, iOffsetB){
|
|
return this.oBrowserSelection.compareEndPoints(oContainerA, iOffsetA, oContainerB, iOffsetB);
|
|
|
|
var iIdxA, iIdxB;
|
|
|
|
if(!oContainerA || this._findBody(oContainerA) != this._document.body){
|
|
oContainerA = this._document.body;
|
|
iOffsetA = 0;
|
|
}
|
|
|
|
if(!oContainerB || this._findBody(oContainerB) != this._document.body){
|
|
oContainerB = this._document.body;
|
|
iOffsetB = 0;
|
|
}
|
|
|
|
var compareIdx = function(iIdxA, iIdxB){
|
|
// iIdxX == -1 when the node is the commonAncestorNode
|
|
// if iIdxA == -1
|
|
// -> [[<nodeA>...<nodeB></nodeB>]]...</nodeA>
|
|
// if iIdxB == -1
|
|
// -> <nodeB>...[[<nodeA></nodeA>...</nodeB>]]
|
|
if(iIdxB == -1){iIdxB = iIdxA+1;}
|
|
if(iIdxA < iIdxB){return -1;}
|
|
if(iIdxA == iIdxB){return 0;}
|
|
return 1;
|
|
};
|
|
|
|
var oCommonAncestor = this._getCommonAncestorContainer(oContainerA, oContainerB);
|
|
|
|
// ================================================================================================================================================
|
|
// Move up both containers so that both containers are direct child nodes of the common ancestor node. From there, just compare the offset
|
|
// Add 0.5 for each contaienrs that has "moved up" since the actual node is wrapped by 1 or more parent nodes and therefore its position is somewhere between idx & idx+1
|
|
// <COMMON_ANCESTOR>NODE1<P>NODE2</P>NODE3</COMMON_ANCESTOR>
|
|
// The position of NODE2 in COMMON_ANCESTOR is somewhere between after NODE1(idx1) and before NODE3(idx2), so we let that be 1.5
|
|
|
|
// container node A in common ancestor container
|
|
var oNodeA = oContainerA;
|
|
var oTmpNode = null;
|
|
if(oNodeA != oCommonAncestor){
|
|
while((oTmpNode = nhn.DOMFix.parentNode(oNodeA)) != oCommonAncestor){oNodeA = oTmpNode;}
|
|
|
|
iIdxA = this._getPosIdx(oNodeA)+0.5;
|
|
}else{
|
|
iIdxA = iOffsetA;
|
|
}
|
|
|
|
// container node B in common ancestor container
|
|
var oNodeB = oContainerB;
|
|
if(oNodeB != oCommonAncestor){
|
|
while((oTmpNode = nhn.DOMFix.parentNode(oNodeB)) != oCommonAncestor){oNodeB = oTmpNode;}
|
|
|
|
iIdxB = this._getPosIdx(oNodeB)+0.5;
|
|
}else{
|
|
iIdxB = iOffsetB;
|
|
}
|
|
|
|
return compareIdx(iIdxA, iIdxB);
|
|
},
|
|
|
|
_getCommonAncestorContainer : function(oNode1, oNode2){
|
|
oNode1 = oNode1 || this.startContainer;
|
|
oNode2 = oNode2 || this.endContainer;
|
|
|
|
var oComparingNode = oNode2;
|
|
|
|
while(oNode1){
|
|
while(oComparingNode){
|
|
if(oNode1 == oComparingNode){return oNode1;}
|
|
oComparingNode = nhn.DOMFix.parentNode(oComparingNode);
|
|
}
|
|
oComparingNode = oNode2;
|
|
oNode1 = nhn.DOMFix.parentNode(oNode1);
|
|
}
|
|
|
|
return this._document.body;
|
|
},
|
|
|
|
deleteContents : function(){
|
|
if(this.collapsed){return;}
|
|
|
|
this._splitTextEndNodesOfTheRange();
|
|
|
|
var aNodes = this._getNodesInRange();
|
|
|
|
if(aNodes.length < 1){return;}
|
|
var oPrevNode = aNodes[0].previousSibling;
|
|
|
|
while(oPrevNode && this._isBlankTextNode(oPrevNode)){oPrevNode = oPrevNode.previousSibling;}
|
|
|
|
var oNewStartContainer, iNewOffset = -1;
|
|
if(!oPrevNode){
|
|
oNewStartContainer = nhn.DOMFix.parentNode(aNodes[0]);
|
|
iNewOffset = 0;
|
|
}
|
|
|
|
for(var i=0; i<aNodes.length; i++){
|
|
var oNode = aNodes[i];
|
|
|
|
if(!oNode.firstChild || this._isAllChildBlankText(oNode)){
|
|
if(oNewStartContainer == oNode){
|
|
iNewOffset = this._getPosIdx(oNewStartContainer);
|
|
oNewStartContainer = nhn.DOMFix.parentNode(oNode);
|
|
}
|
|
oNode.parentNode.removeChild(oNode);
|
|
}else{
|
|
// move the starting point to out of the parent container if the starting point of parent container is meant to be removed
|
|
// [<span>A]B</span>
|
|
// -> []<span>B</span>
|
|
// without these lines, the result would yeild to
|
|
// -> <span>[]B</span>
|
|
if(oNewStartContainer == oNode && iNewOffset === 0){
|
|
iNewOffset = this._getPosIdx(oNewStartContainer);
|
|
oNewStartContainer = nhn.DOMFix.parentNode(oNode);
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!oPrevNode){
|
|
this.setStart(oNewStartContainer, iNewOffset, true, true);
|
|
}else{
|
|
if(oPrevNode.tagName == "BODY"){
|
|
this.setStartBefore(oPrevNode, true);
|
|
}else{
|
|
this.setStartAfter(oPrevNode, true);
|
|
}
|
|
}
|
|
|
|
this.collapse(true);
|
|
},
|
|
|
|
extractContents : function(){
|
|
var oClonedContents = this.cloneContents();
|
|
this.deleteContents();
|
|
return oClonedContents;
|
|
},
|
|
|
|
getInsertBeforeNodes : function(){
|
|
var oFirstNode = null;
|
|
|
|
var oParentContainer;
|
|
|
|
if(this.startContainer.nodeType == "3"){
|
|
oParentContainer = nhn.DOMFix.parentNode(this.startContainer);
|
|
if(this.startContainer.nodeValue.length <= this.startOffset){
|
|
oFirstNode = this.startContainer.nextSibling;
|
|
}else{
|
|
oFirstNode = this.startContainer.splitText(this.startOffset);
|
|
}
|
|
}else{
|
|
oParentContainer = this.startContainer;
|
|
oFirstNode = nhn.DOMFix.childNodes(this.startContainer)[this.startOffset];
|
|
}
|
|
|
|
if(!oFirstNode || !nhn.DOMFix.parentNode(oFirstNode)){oFirstNode = null;}
|
|
|
|
return {elParent: oParentContainer, elBefore: oFirstNode};
|
|
},
|
|
|
|
insertNode : function(newNode){
|
|
var oInsertBefore = this.getInsertBeforeNodes();
|
|
|
|
oInsertBefore.elParent.insertBefore(newNode, oInsertBefore.elBefore);
|
|
|
|
this.setStartBefore(newNode);
|
|
},
|
|
|
|
selectNode : function(refNode){
|
|
this.reset(this._window);
|
|
|
|
this.setStartBefore(refNode);
|
|
this.setEndAfter(refNode);
|
|
},
|
|
|
|
selectNodeContents : function(refNode){
|
|
this.reset(this._window);
|
|
|
|
this.setStart(refNode, 0, true);
|
|
this.setEnd(refNode, nhn.DOMFix.childNodes(refNode).length);
|
|
},
|
|
|
|
_endsNodeValidation : function(oNode, iOffset){
|
|
if(!oNode || this._findBody(oNode) != this._document.body){throw new Error("INVALID_NODE_TYPE_ERR oNode is not part of current document");}
|
|
|
|
if(oNode.nodeType == 3){
|
|
if(iOffset > oNode.nodeValue.length){iOffset = oNode.nodeValue.length;}
|
|
}else{
|
|
if(iOffset > nhn.DOMFix.childNodes(oNode).length){iOffset = nhn.DOMFix.childNodes(oNode).length;}
|
|
}
|
|
|
|
return iOffset;
|
|
},
|
|
|
|
|
|
setEnd : function(refNode, offset, bSafe, bNoUpdate){
|
|
if(!bSafe){offset = this._endsNodeValidation(refNode, offset);}
|
|
|
|
this.endContainer = refNode;
|
|
this.endOffset = offset;
|
|
|
|
if(!bNoUpdate){
|
|
if(!this.startContainer || this._compareEndPoint(this.startContainer, this.startOffset, this.endContainer, this.endOffset) != -1){
|
|
this.collapse(false);
|
|
}else{
|
|
this._updateRangeInfo();
|
|
}
|
|
}
|
|
},
|
|
|
|
setEndAfter : function(refNode, bNoUpdate){
|
|
if(!refNode){throw new Error("INVALID_NODE_TYPE_ERR in setEndAfter");}
|
|
|
|
if(refNode.tagName == "BODY"){
|
|
this.setEnd(refNode, nhn.DOMFix.childNodes(refNode).length, true, bNoUpdate);
|
|
return;
|
|
}
|
|
this.setEnd(nhn.DOMFix.parentNode(refNode), this._getPosIdx(refNode)+1, true, bNoUpdate);
|
|
},
|
|
|
|
setEndBefore : function(refNode, bNoUpdate){
|
|
if(!refNode){throw new Error("INVALID_NODE_TYPE_ERR in setEndBefore");}
|
|
|
|
if(refNode.tagName == "BODY"){
|
|
this.setEnd(refNode, 0, true, bNoUpdate);
|
|
return;
|
|
}
|
|
|
|
this.setEnd(nhn.DOMFix.parentNode(refNode), this._getPosIdx(refNode), true, bNoUpdate);
|
|
},
|
|
|
|
setStart : function(refNode, offset, bSafe, bNoUpdate){
|
|
if(!bSafe){offset = this._endsNodeValidation(refNode, offset);}
|
|
|
|
this.startContainer = refNode;
|
|
this.startOffset = offset;
|
|
|
|
if(!bNoUpdate){
|
|
if(!this.endContainer || this._compareEndPoint(this.startContainer, this.startOffset, this.endContainer, this.endOffset) != -1){
|
|
this.collapse(true);
|
|
}else{
|
|
this._updateRangeInfo();
|
|
}
|
|
}
|
|
},
|
|
|
|
setStartAfter : function(refNode, bNoUpdate){
|
|
if(!refNode){throw new Error("INVALID_NODE_TYPE_ERR in setStartAfter");}
|
|
|
|
if(refNode.tagName == "BODY"){
|
|
this.setStart(refNode, nhn.DOMFix.childNodes(refNode).length, true, bNoUpdate);
|
|
return;
|
|
}
|
|
|
|
this.setStart(nhn.DOMFix.parentNode(refNode), this._getPosIdx(refNode)+1, true, bNoUpdate);
|
|
},
|
|
|
|
setStartBefore : function(refNode, bNoUpdate){
|
|
if(!refNode){throw new Error("INVALID_NODE_TYPE_ERR in setStartBefore");}
|
|
|
|
if(refNode.tagName == "BODY"){
|
|
this.setStart(refNode, 0, true, bNoUpdate);
|
|
return;
|
|
}
|
|
this.setStart(nhn.DOMFix.parentNode(refNode), this._getPosIdx(refNode), true, bNoUpdate);
|
|
},
|
|
|
|
surroundContents : function(newParent){
|
|
newParent.appendChild(this.extractContents());
|
|
this.insertNode(newParent);
|
|
this.selectNode(newParent);
|
|
},
|
|
|
|
toString : function(){
|
|
var oTmpContainer = this._document.createElement("DIV");
|
|
oTmpContainer.appendChild(this.cloneContents());
|
|
|
|
return oTmpContainer.textContent || oTmpContainer.innerText || "";
|
|
},
|
|
|
|
// this.oBrowserSelection.getCommonAncestorContainer which uses browser's built-in API runs faster but may return an incorrect value.
|
|
// Call this function to fix the problem.
|
|
//
|
|
// In IE, the built-in API would return an incorrect value when,
|
|
// 1. commonAncestorContainer is not selectable
|
|
// AND
|
|
// 2. The selected area will look the same when its child node is selected
|
|
// eg)
|
|
// when <P><SPAN>TEST</SPAN></p> is selected, <SPAN>TEST</SPAN> will be returned as commonAncestorContainer
|
|
fixCommonAncestorContainer : function(){
|
|
if(!jindo.$Agent().navigator().ie){
|
|
return;
|
|
}
|
|
|
|
this.commonAncestorContainer = this._getCommonAncestorContainer();
|
|
},
|
|
|
|
_isBlankTextNode : function(oNode){
|
|
if(oNode.nodeType == 3 && oNode.nodeValue == ""){return true;}
|
|
return false;
|
|
},
|
|
|
|
_isAllChildBlankText : function(elNode){
|
|
for(var i=0, nLen=elNode.childNodes.length; i<nLen; i++){
|
|
if(!this._isBlankTextNode(elNode.childNodes[i])){return false;}
|
|
}
|
|
return true;
|
|
},
|
|
|
|
_getPosIdx : function(refNode){
|
|
var idx = 0;
|
|
for(var node = refNode.previousSibling; node; node = node.previousSibling){idx++;}
|
|
|
|
return idx;
|
|
},
|
|
|
|
_updateRangeInfo : function(){
|
|
if(!this.startContainer){
|
|
this.reset(this._window);
|
|
return;
|
|
}
|
|
|
|
// isCollapsed may not function correctly when the cursor is located,
|
|
// (below a table) AND (at the end of the document where there's no P tag or anything else to actually hold the cursor)
|
|
this.collapsed = this.oBrowserSelection.isCollapsed(this) || (this.startContainer === this.endContainer && this.startOffset === this.endOffset);
|
|
// this.collapsed = this._isCollapsed(this.startContainer, this.startOffset, this.endContainer, this.endOffset);
|
|
this.commonAncestorContainer = this.oBrowserSelection.getCommonAncestorContainer(this);
|
|
// this.commonAncestorContainer = this._getCommonAncestorContainer(this.startContainer, this.endContainer);
|
|
},
|
|
|
|
_isCollapsed : function(oStartContainer, iStartOffset, oEndContainer, iEndOffset){
|
|
var bCollapsed = false;
|
|
|
|
if(oStartContainer == oEndContainer && iStartOffset == iEndOffset){
|
|
bCollapsed = true;
|
|
}else{
|
|
var oActualStartNode = this._getActualStartNode(oStartContainer, iStartOffset);
|
|
var oActualEndNode = this._getActualEndNode(oEndContainer, iEndOffset);
|
|
|
|
// Take the parent nodes on the same level for easier comparison when they're next to each other
|
|
// eg) From
|
|
// <A>
|
|
// <B>
|
|
// <C>
|
|
// </C>
|
|
// </B>
|
|
// <D>
|
|
// <E>
|
|
// <F>
|
|
// </F>
|
|
// </E>
|
|
// </D>
|
|
// </A>
|
|
// , it's easier to compare the position of B and D rather than C and F because they are siblings
|
|
//
|
|
// If the range were collapsed, oActualEndNode will precede oActualStartNode by doing this
|
|
oActualStartNode = this._getNextNode(this._getPrevNode(oActualStartNode));
|
|
oActualEndNode = this._getPrevNode(this._getNextNode(oActualEndNode));
|
|
|
|
if(oActualStartNode && oActualEndNode && oActualEndNode.tagName != "BODY" &&
|
|
(this._getNextNode(oActualEndNode) == oActualStartNode || (oActualEndNode == oActualStartNode && this._isBlankTextNode(oActualEndNode)))
|
|
){
|
|
bCollapsed = true;
|
|
}
|
|
}
|
|
|
|
return bCollapsed;
|
|
},
|
|
|
|
_splitTextEndNodesOfTheRange : function(){
|
|
var oEndPoints = this._splitTextEndNodes({oStartContainer: this.startContainer, iStartOffset: this.startOffset,
|
|
oEndContainer: this.endContainer, iEndOffset: this.endOffset});
|
|
|
|
this.startContainer = oEndPoints.oStartContainer;
|
|
this.startOffset = oEndPoints.iStartOffset;
|
|
|
|
this.endContainer = oEndPoints.oEndContainer;
|
|
this.endOffset = oEndPoints.iEndOffset;
|
|
},
|
|
|
|
_splitTextEndNodes : function(oEndPoints){
|
|
oEndPoints = this._splitStartTextNode(oEndPoints);
|
|
oEndPoints = this._splitEndTextNode(oEndPoints);
|
|
|
|
return oEndPoints;
|
|
},
|
|
|
|
_splitStartTextNode : function(oEndPoints){
|
|
var oStartContainer = oEndPoints.oStartContainer;
|
|
var iStartOffset = oEndPoints.iStartOffset;
|
|
|
|
var oEndContainer = oEndPoints.oEndContainer;
|
|
var iEndOffset = oEndPoints.iEndOffset;
|
|
|
|
if(!oStartContainer){return oEndPoints;}
|
|
if(oStartContainer.nodeType != 3){return oEndPoints;}
|
|
if(iStartOffset === 0){return oEndPoints;}
|
|
|
|
if(oStartContainer.nodeValue.length <= iStartOffset){return oEndPoints;}
|
|
|
|
var oLastPart = oStartContainer.splitText(iStartOffset);
|
|
|
|
if(oStartContainer == oEndContainer){
|
|
iEndOffset -= iStartOffset;
|
|
oEndContainer = oLastPart;
|
|
}
|
|
oStartContainer = oLastPart;
|
|
iStartOffset = 0;
|
|
|
|
return {oStartContainer: oStartContainer, iStartOffset: iStartOffset, oEndContainer: oEndContainer, iEndOffset: iEndOffset};
|
|
},
|
|
|
|
_splitEndTextNode : function(oEndPoints){
|
|
var oStartContainer = oEndPoints.oStartContainer;
|
|
var iStartOffset = oEndPoints.iStartOffset;
|
|
|
|
var oEndContainer = oEndPoints.oEndContainer;
|
|
var iEndOffset = oEndPoints.iEndOffset;
|
|
|
|
if(!oEndContainer){return oEndPoints;}
|
|
if(oEndContainer.nodeType != 3){return oEndPoints;}
|
|
|
|
if(iEndOffset >= oEndContainer.nodeValue.length){return oEndPoints;}
|
|
if(iEndOffset === 0){return oEndPoints;}
|
|
|
|
oEndContainer.splitText(iEndOffset);
|
|
|
|
return {oStartContainer: oStartContainer, iStartOffset: iStartOffset, oEndContainer: oEndContainer, iEndOffset: iEndOffset};
|
|
},
|
|
|
|
_getNodesInRange : function(){
|
|
if(this.collapsed){return [];}
|
|
|
|
var oStartNode = this._getActualStartNode(this.startContainer, this.startOffset);
|
|
var oEndNode = this._getActualEndNode(this.endContainer, this.endOffset);
|
|
|
|
return this._getNodesBetween(oStartNode, oEndNode);
|
|
},
|
|
|
|
_getActualStartNode : function(oStartContainer, iStartOffset){
|
|
var oStartNode = oStartContainer;
|
|
|
|
if(oStartContainer.nodeType == 3){
|
|
if(iStartOffset >= oStartContainer.nodeValue.length){
|
|
oStartNode = this._getNextNode(oStartContainer);
|
|
if(oStartNode.tagName == "BODY"){oStartNode = null;}
|
|
}else{
|
|
oStartNode = oStartContainer;
|
|
}
|
|
}else{
|
|
if(iStartOffset < nhn.DOMFix.childNodes(oStartContainer).length){
|
|
oStartNode = nhn.DOMFix.childNodes(oStartContainer)[iStartOffset];
|
|
}else{
|
|
oStartNode = this._getNextNode(oStartContainer);
|
|
if(oStartNode.tagName == "BODY"){oStartNode = null;}
|
|
}
|
|
}
|
|
|
|
return oStartNode;
|
|
},
|
|
|
|
_getActualEndNode : function(oEndContainer, iEndOffset){
|
|
var oEndNode = oEndContainer;
|
|
|
|
if(iEndOffset === 0){
|
|
oEndNode = this._getPrevNode(oEndContainer);
|
|
if(oEndNode.tagName == "BODY"){oEndNode = null;}
|
|
}else if(oEndContainer.nodeType == 3){
|
|
oEndNode = oEndContainer;
|
|
}else{
|
|
oEndNode = nhn.DOMFix.childNodes(oEndContainer)[iEndOffset-1];
|
|
}
|
|
|
|
return oEndNode;
|
|
},
|
|
|
|
_getNextNode : function(oNode){
|
|
if(!oNode || oNode.tagName == "BODY"){return this._document.body;}
|
|
|
|
if(oNode.nextSibling){return oNode.nextSibling;}
|
|
|
|
return this._getNextNode(nhn.DOMFix.parentNode(oNode));
|
|
},
|
|
|
|
_getPrevNode : function(oNode){
|
|
if(!oNode || oNode.tagName == "BODY"){return this._document.body;}
|
|
|
|
if(oNode.previousSibling){return oNode.previousSibling;}
|
|
|
|
return this._getPrevNode(nhn.DOMFix.parentNode(oNode));
|
|
},
|
|
|
|
// includes partially selected
|
|
// for <div id="a"><div id="b"></div></div><div id="c"></div>, _getNodesBetween(b, c) will yield to b, "a" and c
|
|
_getNodesBetween : function(oStartNode, oEndNode){
|
|
var aNodesBetween = [];
|
|
this._nNodesBetweenLen = 0;
|
|
|
|
if(!oStartNode || !oEndNode){return aNodesBetween;}
|
|
|
|
// IE may throw an exception on "oCurNode = oCurNode.nextSibling;" when oCurNode is 'invalid', not null or undefined but somehow 'invalid'.
|
|
// It happened during browser's build-in UNDO with control range selected(table).
|
|
try{
|
|
this._recurGetNextNodesUntil(oStartNode, oEndNode, aNodesBetween);
|
|
}catch(e){
|
|
return [];
|
|
}
|
|
|
|
return aNodesBetween;
|
|
},
|
|
|
|
_recurGetNextNodesUntil : function(oNode, oEndNode, aNodesBetween){
|
|
if(!oNode){return false;}
|
|
|
|
if(!this._recurGetChildNodesUntil(oNode, oEndNode, aNodesBetween)){return false;}
|
|
|
|
var oNextToChk = oNode.nextSibling;
|
|
|
|
while(!oNextToChk){
|
|
if(!(oNode = nhn.DOMFix.parentNode(oNode))){return false;}
|
|
|
|
aNodesBetween[this._nNodesBetweenLen++] = oNode;
|
|
|
|
if(oNode == oEndNode){return false;}
|
|
|
|
oNextToChk = oNode.nextSibling;
|
|
}
|
|
|
|
return this._recurGetNextNodesUntil(oNextToChk, oEndNode, aNodesBetween);
|
|
},
|
|
|
|
_recurGetChildNodesUntil : function(oNode, oEndNode, aNodesBetween){
|
|
if(!oNode){return false;}
|
|
|
|
var bEndFound = false;
|
|
var oCurNode = oNode;
|
|
if(oCurNode.firstChild){
|
|
oCurNode = oCurNode.firstChild;
|
|
while(oCurNode){
|
|
if(!this._recurGetChildNodesUntil(oCurNode, oEndNode, aNodesBetween)){
|
|
bEndFound = true;
|
|
break;
|
|
}
|
|
oCurNode = oCurNode.nextSibling;
|
|
}
|
|
}
|
|
aNodesBetween[this._nNodesBetweenLen++] = oNode;
|
|
|
|
if(bEndFound){return false;}
|
|
if(oNode == oEndNode){return false;}
|
|
|
|
return true;
|
|
}
|
|
});
|
|
|
|
nhn.W3CDOMRange.START_TO_START = 0;
|
|
nhn.W3CDOMRange.START_TO_END = 1;
|
|
nhn.W3CDOMRange.END_TO_END = 2;
|
|
nhn.W3CDOMRange.END_TO_START = 3;
|
|
|
|
|
|
/**
|
|
* @fileOverview This file contains a cross-browser function that implements all of the W3C's DOM Range specification and some more
|
|
* @name HuskyRange.js
|
|
*/
|
|
nhn.HuskyRange = jindo.$Class({
|
|
_rxCursorHolder : /^(?:\uFEFF|\u00A0|\u200B|<br>)$/i,
|
|
_rxTextAlign : /text-align:[^"';]*;?/i,
|
|
|
|
setWindow : function(win){
|
|
this.reset(win || window);
|
|
},
|
|
|
|
$init : function(win){
|
|
this.HUSKY_BOOMARK_START_ID_PREFIX = "husky_bookmark_start_";
|
|
this.HUSKY_BOOMARK_END_ID_PREFIX = "husky_bookmark_end_";
|
|
|
|
this.sBlockElement = "P|DIV|LI|H[1-6]|PRE";
|
|
this.sBlockContainer = "BODY|TABLE|TH|TR|TD|UL|OL|BLOCKQUOTE|FORM";
|
|
|
|
this.rxBlockElement = new RegExp("^("+this.sBlockElement+")$");
|
|
this.rxBlockContainer = new RegExp("^("+this.sBlockContainer+")$");
|
|
this.rxLineBreaker = new RegExp("^("+this.sBlockElement+"|"+this.sBlockContainer+")$");
|
|
this.rxHasBlock = new RegExp("(?:<(?:"+this.sBlockElement+"|"+this.sBlockContainer+").*?>|style=[\"']?[^>]*?(?:display\\s?:\\s?block)[^>]*?[\"']?)", "i");
|
|
|
|
this.setWindow(win);
|
|
},
|
|
|
|
select : function(){
|
|
try{
|
|
this.oBrowserSelection.selectRange(this);
|
|
}catch(e){}
|
|
},
|
|
|
|
setFromSelection : function(iNum){
|
|
this.setRange(this.oBrowserSelection.getRangeAt(iNum), true);
|
|
},
|
|
|
|
setRange : function(oW3CRange, bSafe){
|
|
this.reset(this._window);
|
|
|
|
this.setStart(oW3CRange.startContainer, oW3CRange.startOffset, bSafe, true);
|
|
this.setEnd(oW3CRange.endContainer, oW3CRange.endOffset, bSafe);
|
|
},
|
|
|
|
setEndNodes : function(oSNode, oENode){
|
|
this.reset(this._window);
|
|
|
|
this.setEndAfter(oENode, true);
|
|
this.setStartBefore(oSNode);
|
|
},
|
|
|
|
splitTextAtBothEnds : function(){
|
|
this._splitTextEndNodesOfTheRange();
|
|
},
|
|
|
|
getStartNode : function(){
|
|
if(this.collapsed){
|
|
if(this.startContainer.nodeType == 3){
|
|
if(this.startOffset === 0){return null;}
|
|
if(this.startContainer.nodeValue.length <= this.startOffset){return null;}
|
|
return this.startContainer;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
if(this.startContainer.nodeType == 3){
|
|
if(this.startOffset >= this.startContainer.nodeValue.length){return this._getNextNode(this.startContainer);}
|
|
return this.startContainer;
|
|
}else{
|
|
if(this.startOffset >= nhn.DOMFix.childNodes(this.startContainer).length){return this._getNextNode(this.startContainer);}
|
|
return nhn.DOMFix.childNodes(this.startContainer)[this.startOffset];
|
|
}
|
|
},
|
|
|
|
getEndNode : function(){
|
|
if(this.collapsed){return this.getStartNode();}
|
|
|
|
if(this.endContainer.nodeType == 3){
|
|
if(this.endOffset === 0){return this._getPrevNode(this.endContainer);}
|
|
return this.endContainer;
|
|
}else{
|
|
if(this.endOffset === 0){return this._getPrevNode(this.endContainer);}
|
|
return nhn.DOMFix.childNodes(this.endContainer)[this.endOffset-1];
|
|
}
|
|
},
|
|
|
|
getNodeAroundRange : function(bBefore, bStrict){
|
|
if(!this.collapsed){return this.getStartNode();}
|
|
|
|
if(this.startContainer && this.startContainer.nodeType == 3){return this.startContainer;}
|
|
//if(this.collapsed && this.startContainer && this.startContainer.nodeType == 3) return this.startContainer;
|
|
//if(!this.collapsed || (this.startContainer && this.startContainer.nodeType == 3)) return this.getStartNode();
|
|
|
|
var oBeforeRange, oAfterRange, oResult;
|
|
|
|
if(this.startOffset >= nhn.DOMFix.childNodes(this.startContainer).length){
|
|
oAfterRange = this._getNextNode(this.startContainer);
|
|
}else{
|
|
oAfterRange = nhn.DOMFix.childNodes(this.startContainer)[this.startOffset];
|
|
}
|
|
|
|
if(this.endOffset === 0){
|
|
oBeforeRange = this._getPrevNode(this.endContainer);
|
|
}else{
|
|
oBeforeRange = nhn.DOMFix.childNodes(this.endContainer)[this.endOffset-1];
|
|
}
|
|
|
|
if(bBefore){
|
|
oResult = oBeforeRange;
|
|
if(!oResult && !bStrict){oResult = oAfterRange;}
|
|
}else{
|
|
oResult = oAfterRange;
|
|
if(!oResult && !bStrict){oResult = oBeforeRange;}
|
|
}
|
|
|
|
return oResult;
|
|
},
|
|
|
|
_getXPath : function(elNode){
|
|
var sXPath = "";
|
|
|
|
while(elNode && elNode.nodeType == 1){
|
|
sXPath = "/" + elNode.tagName+"["+this._getPosIdx4XPath(elNode)+"]" + sXPath;
|
|
elNode = nhn.DOMFix.parentNode(elNode);
|
|
}
|
|
|
|
return sXPath;
|
|
},
|
|
|
|
_getPosIdx4XPath : function(refNode){
|
|
var idx = 0;
|
|
for(var node = refNode.previousSibling; node; node = node.previousSibling){
|
|
if(node.tagName == refNode.tagName){idx++;}
|
|
}
|
|
|
|
return idx;
|
|
},
|
|
|
|
// this was written specifically for XPath Bookmark and it may not perform correctly for general purposes
|
|
_evaluateXPath : function(sXPath, oDoc){
|
|
sXPath = sXPath.substring(1, sXPath.length-1);
|
|
var aXPath = sXPath.split(/\//);
|
|
var elNode = oDoc.body;
|
|
|
|
for(var i=2; i<aXPath.length && elNode; i++){
|
|
aXPath[i].match(/([^\[]+)\[(\d+)/i);
|
|
var sTagName = RegExp.$1;
|
|
var nIdx = RegExp.$2;
|
|
|
|
var aAllNodes = nhn.DOMFix.childNodes(elNode);
|
|
var aNodes = [];
|
|
var nLength = aAllNodes.length;
|
|
var nCount = 0;
|
|
for(var ii=0; ii<nLength; ii++){
|
|
if(aAllNodes[ii].tagName == sTagName){aNodes[nCount++] = aAllNodes[ii];}
|
|
}
|
|
|
|
if(aNodes.length < nIdx){
|
|
elNode = null;
|
|
}else{
|
|
elNode = aNodes[nIdx];
|
|
}
|
|
}
|
|
|
|
return elNode;
|
|
},
|
|
|
|
_evaluateXPathBookmark : function(oBookmark){
|
|
var sXPath = oBookmark["sXPath"];
|
|
var nTextNodeIdx = oBookmark["nTextNodeIdx"];
|
|
var nOffset = oBookmark["nOffset"];
|
|
|
|
var elContainer = this._evaluateXPath(sXPath, this._document);
|
|
|
|
if(nTextNodeIdx > -1 && elContainer){
|
|
var aChildNodes = nhn.DOMFix.childNodes(elContainer);
|
|
var elNode = null;
|
|
|
|
var nIdx = nTextNodeIdx;
|
|
var nOffsetLeft = nOffset;
|
|
|
|
while((elNode = aChildNodes[nIdx]) && elNode.nodeType == 3 && elNode.nodeValue.length < nOffsetLeft){
|
|
nOffsetLeft -= elNode.nodeValue.length;
|
|
nIdx++;
|
|
}
|
|
|
|
elContainer = nhn.DOMFix.childNodes(elContainer)[nIdx];
|
|
nOffset = nOffsetLeft;
|
|
}
|
|
|
|
if(!elContainer){
|
|
elContainer = this._document.body;
|
|
nOffset = 0;
|
|
}
|
|
return {elContainer: elContainer, nOffset: nOffset};
|
|
},
|
|
|
|
// this was written specifically for XPath Bookmark and it may not perform correctly for general purposes
|
|
getXPathBookmark : function(){
|
|
var nTextNodeIdx1 = -1;
|
|
var htEndPt1 = {elContainer: this.startContainer, nOffset: this.startOffset};
|
|
var elNode1 = this.startContainer;
|
|
if(elNode1.nodeType == 3){
|
|
htEndPt1 = this._getFixedStartTextNode();
|
|
nTextNodeIdx1 = this._getPosIdx(htEndPt1.elContainer);
|
|
elNode1 = nhn.DOMFix.parentNode(elNode1);
|
|
}
|
|
var sXPathNode1 = this._getXPath(elNode1);
|
|
var oBookmark1 = {sXPath:sXPathNode1, nTextNodeIdx:nTextNodeIdx1, nOffset: htEndPt1.nOffset};
|
|
|
|
if(this.collapsed){
|
|
var oBookmark2 = {sXPath:sXPathNode1, nTextNodeIdx:nTextNodeIdx1, nOffset: htEndPt1.nOffset};
|
|
}else{
|
|
var nTextNodeIdx2 = -1;
|
|
var htEndPt2 = {elContainer: this.endContainer, nOffset: this.endOffset};
|
|
var elNode2 = this.endContainer;
|
|
if(elNode2.nodeType == 3){
|
|
htEndPt2 = this._getFixedEndTextNode();
|
|
nTextNodeIdx2 = this._getPosIdx(htEndPt2.elContainer);
|
|
elNode2 = nhn.DOMFix.parentNode(elNode2);
|
|
}
|
|
var sXPathNode2 = this._getXPath(elNode2);
|
|
var oBookmark2 = {sXPath:sXPathNode2, nTextNodeIdx:nTextNodeIdx2, nOffset: htEndPt2.nOffset};
|
|
}
|
|
return [oBookmark1, oBookmark2];
|
|
},
|
|
|
|
moveToXPathBookmark : function(aBookmark){
|
|
if(!aBookmark){return false;}
|
|
|
|
var oBookmarkInfo1 = this._evaluateXPathBookmark(aBookmark[0]);
|
|
var oBookmarkInfo2 = this._evaluateXPathBookmark(aBookmark[1]);
|
|
|
|
if(!oBookmarkInfo1["elContainer"] || !oBookmarkInfo2["elContainer"]){return;}
|
|
|
|
this.startContainer = oBookmarkInfo1["elContainer"];
|
|
this.startOffset = oBookmarkInfo1["nOffset"];
|
|
|
|
this.endContainer = oBookmarkInfo2["elContainer"];
|
|
this.endOffset = oBookmarkInfo2["nOffset"];
|
|
|
|
return true;
|
|
},
|
|
|
|
_getFixedTextContainer : function(elNode, nOffset){
|
|
while(elNode && elNode.nodeType == 3 && elNode.previousSibling && elNode.previousSibling.nodeType == 3){
|
|
nOffset += elNode.previousSibling.nodeValue.length;
|
|
elNode = elNode.previousSibling;
|
|
}
|
|
|
|
return {elContainer:elNode, nOffset:nOffset};
|
|
},
|
|
|
|
_getFixedStartTextNode : function(){
|
|
return this._getFixedTextContainer(this.startContainer, this.startOffset);
|
|
},
|
|
|
|
_getFixedEndTextNode : function(){
|
|
return this._getFixedTextContainer(this.endContainer, this.endOffset);
|
|
},
|
|
|
|
placeStringBookmark : function(){
|
|
if(this.collapsed || jindo.$Agent().navigator().ie || jindo.$Agent().navigator().firefox){
|
|
return this.placeStringBookmark_NonWebkit();
|
|
}else{
|
|
return this.placeStringBookmark_Webkit();
|
|
}
|
|
},
|
|
|
|
placeStringBookmark_NonWebkit : function(){
|
|
var sTmpId = (new Date()).getTime();
|
|
|
|
var oInsertionPoint = this.cloneRange();
|
|
oInsertionPoint.collapseToEnd();
|
|
var oEndMarker = this._document.createElement("SPAN");
|
|
oEndMarker.id = this.HUSKY_BOOMARK_END_ID_PREFIX+sTmpId;
|
|
oInsertionPoint.insertNode(oEndMarker);
|
|
|
|
var oInsertionPoint = this.cloneRange();
|
|
oInsertionPoint.collapseToStart();
|
|
var oStartMarker = this._document.createElement("SPAN");
|
|
oStartMarker.id = this.HUSKY_BOOMARK_START_ID_PREFIX+sTmpId;
|
|
oInsertionPoint.insertNode(oStartMarker);
|
|
|
|
// IE에서 빈 SPAN의 앞뒤로 커서가 이동하지 않아 문제가 발생 할 수 있어, 보이지 않는 특수 문자를 임시로 넣어 줌.
|
|
if(jindo.$Agent().navigator().ie){
|
|
// SPAN의 위치가 TD와 TD 사이에 있을 경우, 텍스트 삽입 시 알수 없는 오류가 발생한다.
|
|
// TD와 TD사이에서는 텍스트 삽입이 필요 없음으로 그냥 try/catch로 처리
|
|
try{
|
|
oStartMarker.innerHTML = unescape("%uFEFF");
|
|
}catch(e){}
|
|
|
|
try{
|
|
oEndMarker.innerHTML = unescape("%uFEFF");
|
|
}catch(e){}
|
|
}
|
|
this.moveToBookmark(sTmpId);
|
|
|
|
return sTmpId;
|
|
},
|
|
|
|
placeStringBookmark_Webkit : function(){
|
|
var sTmpId = (new Date()).getTime();
|
|
|
|
var elInsertBefore, elInsertParent;
|
|
|
|
// Do not insert the bookmarks between TDs as it will break the rendering in Chrome/Safari
|
|
// -> modify the insertion position from [<td>abc</td>]<td>abc</td> to <td>[abc]</td><td>abc</td>
|
|
var oInsertionPoint = this.cloneRange();
|
|
oInsertionPoint.collapseToEnd();
|
|
elInsertBefore = this._document.createTextNode("");
|
|
oInsertionPoint.insertNode(elInsertBefore);
|
|
elInsertParent = elInsertBefore.parentNode;
|
|
if(elInsertBefore.previousSibling && elInsertBefore.previousSibling.tagName == "TD"){
|
|
elInsertParent = elInsertBefore.previousSibling;
|
|
elInsertBefore = null;
|
|
}
|
|
var oEndMarker = this._document.createElement("SPAN");
|
|
oEndMarker.id = this.HUSKY_BOOMARK_END_ID_PREFIX+sTmpId;
|
|
elInsertParent.insertBefore(oEndMarker, elInsertBefore);
|
|
|
|
var oInsertionPoint = this.cloneRange();
|
|
oInsertionPoint.collapseToStart();
|
|
elInsertBefore = this._document.createTextNode("");
|
|
oInsertionPoint.insertNode(elInsertBefore);
|
|
elInsertParent = elInsertBefore.parentNode;
|
|
if(elInsertBefore.nextSibling && elInsertBefore.nextSibling.tagName == "TD"){
|
|
elInsertParent = elInsertBefore.nextSibling;
|
|
elInsertBefore = elInsertParent.firstChild;
|
|
}
|
|
var oStartMarker = this._document.createElement("SPAN");
|
|
oStartMarker.id = this.HUSKY_BOOMARK_START_ID_PREFIX+sTmpId;
|
|
elInsertParent.insertBefore(oStartMarker, elInsertBefore);
|
|
|
|
//elInsertBefore.parentNode.removeChild(elInsertBefore);
|
|
|
|
this.moveToBookmark(sTmpId);
|
|
|
|
return sTmpId;
|
|
},
|
|
|
|
cloneRange : function(){
|
|
return this._copyRange(new nhn.HuskyRange(this._window));
|
|
},
|
|
|
|
moveToBookmark : function(vBookmark){
|
|
if(typeof(vBookmark) != "object"){
|
|
return this.moveToStringBookmark(vBookmark);
|
|
}else{
|
|
return this.moveToXPathBookmark(vBookmark);
|
|
}
|
|
},
|
|
|
|
getStringBookmark : function(sBookmarkID, bEndBookmark){
|
|
if(bEndBookmark){
|
|
return this._document.getElementById(this.HUSKY_BOOMARK_END_ID_PREFIX+sBookmarkID);
|
|
}else{
|
|
return this._document.getElementById(this.HUSKY_BOOMARK_START_ID_PREFIX+sBookmarkID);
|
|
}
|
|
},
|
|
|
|
moveToStringBookmark : function(sBookmarkID, bIncludeBookmark){
|
|
var oStartMarker = this.getStringBookmark(sBookmarkID);
|
|
var oEndMarker = this.getStringBookmark(sBookmarkID, true);
|
|
|
|
if(!oStartMarker || !oEndMarker){return false;}
|
|
|
|
this.reset(this._window);
|
|
|
|
if(bIncludeBookmark){
|
|
this.setEndAfter(oEndMarker);
|
|
this.setStartBefore(oStartMarker);
|
|
}else{
|
|
this.setEndBefore(oEndMarker);
|
|
this.setStartAfter(oStartMarker);
|
|
}
|
|
return true;
|
|
},
|
|
|
|
removeStringBookmark : function(sBookmarkID){
|
|
/*
|
|
var oStartMarker = this._document.getElementById(this.HUSKY_BOOMARK_START_ID_PREFIX+sBookmarkID);
|
|
var oEndMarker = this._document.getElementById(this.HUSKY_BOOMARK_END_ID_PREFIX+sBookmarkID);
|
|
|
|
if(oStartMarker) nhn.DOMFix.parentNode(oStartMarker).removeChild(oStartMarker);
|
|
if(oEndMarker) nhn.DOMFix.parentNode(oEndMarker).removeChild(oEndMarker);
|
|
*/
|
|
this._removeAll(this.HUSKY_BOOMARK_START_ID_PREFIX+sBookmarkID);
|
|
this._removeAll(this.HUSKY_BOOMARK_END_ID_PREFIX+sBookmarkID);
|
|
},
|
|
|
|
_removeAll : function(sID){
|
|
var elNode;
|
|
while((elNode = this._document.getElementById(sID))){
|
|
elNode.parentNode.removeChild(elNode);
|
|
}
|
|
},
|
|
|
|
collapseToStart : function(){
|
|
this.collapse(true);
|
|
},
|
|
|
|
collapseToEnd : function(){
|
|
this.collapse(false);
|
|
},
|
|
|
|
createAndInsertNode : function(sTagName){
|
|
var tmpNode = this._document.createElement(sTagName);
|
|
this.insertNode(tmpNode);
|
|
return tmpNode;
|
|
},
|
|
|
|
getNodes : function(bSplitTextEndNodes, fnFilter){
|
|
if(bSplitTextEndNodes){this._splitTextEndNodesOfTheRange();}
|
|
|
|
var aAllNodes = this._getNodesInRange();
|
|
var aFilteredNodes = [];
|
|
|
|
if(!fnFilter){return aAllNodes;}
|
|
|
|
for(var i=0; i<aAllNodes.length; i++){
|
|
if(fnFilter(aAllNodes[i])){aFilteredNodes[aFilteredNodes.length] = aAllNodes[i];}
|
|
}
|
|
|
|
return aFilteredNodes;
|
|
},
|
|
|
|
getTextNodes : function(bSplitTextEndNodes){
|
|
var txtFilter = function(oNode){
|
|
if (oNode.nodeType == 3 && oNode.nodeValue != "\n" && oNode.nodeValue != ""){
|
|
return true;
|
|
}else{
|
|
return false;
|
|
}
|
|
};
|
|
|
|
return this.getNodes(bSplitTextEndNodes, txtFilter);
|
|
},
|
|
|
|
surroundContentsWithNewNode : function(sTagName){
|
|
var oNewParent = this._document.createElement(sTagName);
|
|
this.surroundContents(oNewParent);
|
|
return oNewParent;
|
|
},
|
|
|
|
isRangeinRange : function(oAnoterRange, bIncludePartlySelected){
|
|
var startToStart = this.compareBoundaryPoints(this.W3CDOMRange.START_TO_START, oAnoterRange);
|
|
var startToEnd = this.compareBoundaryPoints(this.W3CDOMRange.START_TO_END, oAnoterRange);
|
|
var endToStart = this.compareBoundaryPoints(this.W3CDOMRange.ND_TO_START, oAnoterRange);
|
|
var endToEnd = this.compareBoundaryPoints(this.W3CDOMRange.END_TO_END, oAnoterRange);
|
|
|
|
if(startToStart <= 0 && endToEnd >= 0){return true;}
|
|
|
|
if(bIncludePartlySelected){
|
|
if(startToEnd == 1){return false;}
|
|
if(endToStart == -1){return false;}
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
isNodeInRange : function(oNode, bIncludePartlySelected, bContentOnly){
|
|
var oTmpRange = new nhn.HuskyRange(this._window);
|
|
|
|
if(bContentOnly && oNode.firstChild){
|
|
oTmpRange.setStartBefore(oNode.firstChild);
|
|
oTmpRange.setEndAfter(oNode.lastChild);
|
|
}else{
|
|
oTmpRange.selectNode(oNode);
|
|
}
|
|
|
|
return this.isRangeInRange(oTmpRange, bIncludePartlySelected);
|
|
},
|
|
|
|
pasteText : function(sText){
|
|
this.pasteHTML(sText.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">").replace(/ /g, " ").replace(/"/g, """));
|
|
},
|
|
|
|
/**
|
|
* TODO: 왜 clone 으로 조작할까?
|
|
* @param {String} sHTML 삽입할 HTML
|
|
* @param {Boolean} bBlock HTML 삽입시 강제로 block 요소 처리할지 여부(true 이면 P태그 안에 삽입될 경우, P태그를 무조건 쪼개고 그 사이에 DIV태그로 감싸서 삽입한다.)
|
|
*/
|
|
pasteHTML : function(sHTML, bBlock){
|
|
var oTmpDiv = this._document.createElement("DIV");
|
|
oTmpDiv.innerHTML = sHTML;
|
|
|
|
if(!oTmpDiv.firstChild){
|
|
this.deleteContents();
|
|
return;
|
|
}
|
|
|
|
// getLineInfo 전에 북마크를 삽입하지 않으면 IE에서 oLineBreaker가 P태그 바깥으로 잡히는 경우가 있음(P태그에 아무것도 없을때)
|
|
var clone = this.cloneRange();
|
|
var sBM = clone.placeStringBookmark();
|
|
|
|
// [SMARTEDITORSUS-1960] PrivateTag, 템플릿삽입등 P태그안에 block 요소 삽입과 관련된 처리
|
|
// P태그인 경우, block요소가 들어오면 안된다.
|
|
// 때문에 현재 위치의 컨테이너가 P태그이고 컨텐츠 내용이 block 요소인 경우 P태그를 쪼개고 그 사이에 컨텐츠를 DIV로 감싸서 넣도록 처리한다.
|
|
// [SMARTEDITORSUS-2026][SMARTEDITORSUS-2061] bBlock = true 이면 삽입되는 HTML 이 block 요소가 아니더라도 무조건 P태그를 쪼개서 DIV로 감싸도록 한다.
|
|
var oLineInfo = clone.getLineInfo(),
|
|
oStart = oLineInfo.oStart,
|
|
oEnd = oLineInfo.oEnd;
|
|
if(oStart.oLineBreaker && oStart.oLineBreaker.nodeName === "P" && (bBlock || clone.rxHasBlock.test(sHTML))){
|
|
// [SMARTEDITORSUS-2169] 선택영역 삭제시 oStart.oLineBreaker도 DOM 트리에서 제거될 수 있기 때문에 필요한 노드를 미리 참조해둔다.
|
|
var oParentNode = oStart.oLineBreaker.parentNode,
|
|
oNextSibling = oStart.oLineBreaker.nextSibling;
|
|
|
|
// 선택영역을 조작해야 하므로 현재 선택된 요소들을 제거한다.
|
|
clone.deleteContents();
|
|
|
|
// 동일한 라인에 있으면 뒷부분은 쪼개서 다음 라인으로 삽입한다.
|
|
if(oStart.oLineBreaker === oEnd.oLineBreaker){
|
|
var elBM = clone.getStringBookmark(sBM);
|
|
clone.setEndNodes(elBM, oEnd.oLineBreaker);
|
|
var oNextContents = clone.extractContents(),
|
|
oNextFirst = oNextContents.firstChild; // oNextSibling 을 교체하기 위해 쪼개진 요소 첫번째 노드를 미리 참조해둔다.
|
|
|
|
// 쪼갠 부분을 삽입하고
|
|
if(oNextSibling){
|
|
oParentNode.insertBefore(oNextContents, oNextSibling);
|
|
}else{
|
|
oParentNode.appendChild(oNextContents);
|
|
}
|
|
// [SMARTEDITORSUS-2145] oNextSibling 을 쪼갠 부분의 첫번째 노드로 교체한다.
|
|
oNextSibling = oNextFirst;
|
|
}
|
|
|
|
// 선택영역 앞쪽이 속한 P태그에서 style과 align 정보를 복사한다.
|
|
// 크롬의 경우 div의 style 에 text-align 이 있으면 align 속성은 무시되는데
|
|
// div 안의 block 요소는 text-align 의 대상이 아니라 정렬되지 않는 문제가 있기 때문에
|
|
// style 복사할 때 text-align 속성은 제외한다.
|
|
oTmpDiv.style.cssText = oStart.oLineBreaker.style.cssText.replace(this._rxTextAlign, ''); // text-align 제외
|
|
oTmpDiv.align = oStart.oLineBreaker.align; // align 복사
|
|
|
|
// 컨텐츠 삽입
|
|
if(oNextSibling){
|
|
oParentNode.insertBefore(oTmpDiv, oNextSibling);
|
|
}else{
|
|
oParentNode.appendChild(oTmpDiv);
|
|
}
|
|
|
|
// 컨텐츠 삽입 후에 북마크를 지운다.
|
|
// 컨텐츠 삽입 전에 지우면 컨텐츠 삽입시 oNextSibling 가 북마크로 잡히는 경우 에러가 발생할 수 있음
|
|
clone.removeStringBookmark(sBM);
|
|
|
|
// 컨텐츠 삽입 후 윗라인 P태그에 아무런 내용이 없으면 제거한다.
|
|
this._removeEmptyP(this._getPrevElement(oTmpDiv));
|
|
// 아래 라인 P태그에 아무런 내용이 없는 경우는 그 다음 아래 라인이 있을때만 제거한다.
|
|
// 아래 라인이 아예없으면 IE에서 커서가 들어가지 않기 때문에 라인을 추가해준다.
|
|
var elNextLine = this._getNextElement(oTmpDiv);
|
|
if(elNextLine){
|
|
var elAfterNext = this._getNextElement(elNextLine);
|
|
if(elAfterNext && this._removeEmptyP(elNextLine)){
|
|
elNextLine = elAfterNext; // 제거되었을 경우만 elNextLine 재할당
|
|
}
|
|
}else{
|
|
// 아래 라인이 없으면 윗 라인 스타일을 복사하여 추가해준다.
|
|
elNextLine = this._document.createElement("P");
|
|
elNextLine.style.cssText = oStart.oLineBreaker.style.cssText;
|
|
elNextLine.align = oStart.oLineBreaker.align;
|
|
oParentNode.appendChild(elNextLine);
|
|
}
|
|
|
|
// 커서를 다음라인으로 앞쪽으로 위치시킨다.
|
|
if(elNextLine.innerHTML === ""){
|
|
// 크롬에서 빈 <p></p> 를 선택해서 collapseToStart 하면 윗라인으로 이동하기 때문에 비어있으면 \uFEFF 를 넣어준다.
|
|
elNextLine.innerHTML = (jindo.$Agent().navigator().ie && jindo.$Agent().navigator().version > 8) ? "\u200B" : "\uFEFF";
|
|
}
|
|
this.selectNodeContents(elNextLine);
|
|
this.collapseToStart();
|
|
|
|
// IE7에서 커서가 다음라인 p태그 앞쪽이 아닌 div태그 끝쪽으로 자동으로 옮겨가는 경우가 있어서
|
|
// 커서가 멋대로 이동하지 않도록 임시북마크를 넣었다가 바로 빼준다.
|
|
// (주의) 북마크를 넣었다 빼면 IE10은 다음라인 p태그 끝쪽으로 이동되기 때문에 IE7인 경우만 넣어줌
|
|
// [SMARTEDITORSUS-2043] SE_EditingArea_WYSIWYG.$ON_PASTE_HTML 에서 IE8의 경우만 삽입시 뒤에 \uFEFF가 추가로 붙어서 들어오는데
|
|
// 이로 인해 템플릿과 커서사이가 한줄 벌어지는 문제가 있어서 \uFEFF 추가하는 부분을 삭제하니 커서가 IE7과 동일하게 동작하여 IE8도 임시북마크처리함
|
|
if(jindo.$Agent().navigator().ie && jindo.$Agent().navigator().version < 9){
|
|
sBM = this.placeStringBookmark();
|
|
this.removeStringBookmark(sBM);
|
|
}
|
|
}else{
|
|
var oFirstNode = oTmpDiv.firstChild;
|
|
var oLastNode = oTmpDiv.lastChild;
|
|
|
|
this.collapseToStart();
|
|
|
|
while(oTmpDiv.lastChild){this.insertNode(oTmpDiv.lastChild);}
|
|
|
|
this.setEndNodes(oFirstNode, oLastNode);
|
|
|
|
// delete the content later as deleting it first may mass up the insertion point
|
|
// eg) <p>[A]BCD</p> ---paste O---> O<p>BCD</p>
|
|
clone.moveToBookmark(sBM);
|
|
clone.deleteContents();
|
|
clone.removeStringBookmark(sBM);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 비어있는 P태그이면 제거한다.
|
|
* @param {Element} el 검사할 Element
|
|
* @returns {Boolean} 제거되었다면 true 를 반환한다.
|
|
*/
|
|
_removeEmptyP : function(el){
|
|
if(el && el.nodeName === "P"){
|
|
var sInner = el.innerHTML;
|
|
if(sInner === "" || this._rxCursorHolder.test(sInner)){
|
|
el.parentNode.removeChild(el);
|
|
return true;
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 인접한 Element 노드를 찾는다.
|
|
* @param {Node} oNode 기준 노드
|
|
* @param {Boolean} bPrev 앞뒤여부(true면 앞, false면 뒤)
|
|
* @return {Element} 인접한 Element, 없으면 null 반환
|
|
*/
|
|
_getSiblingElement : function(oNode, bPrev){
|
|
if(!oNode){
|
|
return null;
|
|
}
|
|
|
|
var oSibling = oNode[bPrev ? "previousSibling" : "nextSibling"];
|
|
if(oSibling && oSibling.nodeType === 1){
|
|
return oSibling;
|
|
}else{
|
|
return arguments.callee(oSibling, bPrev);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 앞쪽 인접한 Element 노드를 찾는다.
|
|
* @param {Node} oNode 기준 노드
|
|
* @return {Element} 인접한 Element, 없으면 null 반환
|
|
*/
|
|
_getPrevElement : function(oNode){
|
|
return this._getSiblingElement(oNode, true);
|
|
},
|
|
|
|
/**
|
|
* 뒤쪽 인접한 Element 노드를 찾는다.
|
|
* @param {Node} oNode 기준 노드
|
|
* @return {Element} 인접한 Element, 없으면 null 반환
|
|
*/
|
|
_getNextElement : function(oNode){
|
|
return this._getSiblingElement(oNode, false);
|
|
},
|
|
|
|
toString : function(){
|
|
this.toString = nhn.W3CDOMRange.prototype.toString;
|
|
return this.toString();
|
|
},
|
|
|
|
toHTMLString : function(){
|
|
var oTmpContainer = this._document.createElement("DIV");
|
|
oTmpContainer.appendChild(this.cloneContents());
|
|
|
|
return oTmpContainer.innerHTML;
|
|
},
|
|
|
|
findAncestorByTagName : function(sTagName){
|
|
var oNode = this.commonAncestorContainer;
|
|
while(oNode && oNode.tagName != sTagName){oNode = nhn.DOMFix.parentNode(oNode);}
|
|
|
|
return oNode;
|
|
},
|
|
|
|
selectNodeContents : function(oNode){
|
|
if(!oNode){return;}
|
|
|
|
var oFirstNode = oNode.firstChild?oNode.firstChild:oNode;
|
|
var oLastNode = oNode.lastChild?oNode.lastChild:oNode;
|
|
|
|
this.reset(this._window);
|
|
if(oFirstNode.nodeType == 3){
|
|
this.setStart(oFirstNode, 0, true);
|
|
}else{
|
|
this.setStartBefore(oFirstNode);
|
|
}
|
|
|
|
if(oLastNode.nodeType == 3){
|
|
this.setEnd(oLastNode, oLastNode.nodeValue.length, true);
|
|
}else{
|
|
this.setEndAfter(oLastNode);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 노드의 취소선/밑줄 정보를 확인한다
|
|
* 관련 BTS [SMARTEDITORSUS-26]
|
|
* @param {Node} oNode 취소선/밑줄을 확인할 노드
|
|
* @param {String} sValue textDecoration 정보
|
|
* @see nhn.HuskyRange#_checkTextDecoration
|
|
*/
|
|
_hasTextDecoration : function(oNode, sValue){
|
|
if(!oNode || !oNode.style){
|
|
return false;
|
|
}
|
|
|
|
if(oNode.style.textDecoration.indexOf(sValue) > -1){
|
|
return true;
|
|
}
|
|
|
|
if(sValue === "underline" && oNode.tagName === "U"){
|
|
return true;
|
|
}
|
|
|
|
if(sValue === "line-through" && (oNode.tagName === "S" || oNode.tagName === "STRIKE")){
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* 노드에 취소선/밑줄을 적용한다
|
|
* 관련 BTS [SMARTEDITORSUS-26]
|
|
* [FF] 노드의 Style 에 textDecoration 을 추가한다
|
|
* [FF 외] U/STRIKE 태그를 추가한다
|
|
* @param {Node} oNode 취소선/밑줄을 적용할 노드
|
|
* @param {String} sValue textDecoration 정보
|
|
* @see nhn.HuskyRange#_checkTextDecoration
|
|
*/
|
|
_setTextDecoration : function(oNode, sValue){
|
|
if (jindo.$Agent().navigator().firefox) { // FF
|
|
oNode.style.textDecoration = (oNode.style.textDecoration) ? oNode.style.textDecoration + " " + sValue : sValue;
|
|
}
|
|
else{
|
|
if(sValue === "underline"){
|
|
oNode.innerHTML = "<U>" + oNode.innerHTML + "</U>"
|
|
}else if(sValue === "line-through"){
|
|
oNode.innerHTML = "<STRIKE>" + oNode.innerHTML + "</STRIKE>"
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 인자로 전달받은 노드 상위의 취소선/밑줄 정보를 확인하여 노드에 적용한다
|
|
* 관련 BTS [SMARTEDITORSUS-26]
|
|
* @param {Node} oNode 취소선/밑줄을 적용할 노드
|
|
*/
|
|
_checkTextDecoration : function(oNode){
|
|
if(oNode.tagName !== "SPAN"){
|
|
return;
|
|
}
|
|
|
|
var bUnderline = false,
|
|
bLineThrough = false,
|
|
sTextDecoration = "",
|
|
oParentNode = null;
|
|
oChildNode = oNode.firstChild;
|
|
|
|
/* check child */
|
|
while(oChildNode){
|
|
if(oChildNode.nodeType === 1){
|
|
bUnderline = (bUnderline || oChildNode.tagName === "U");
|
|
bLineThrough = (bLineThrough || oChildNode.tagName === "S" || oChildNode.tagName === "STRIKE");
|
|
}
|
|
|
|
if(bUnderline && bLineThrough){
|
|
return;
|
|
}
|
|
|
|
oChildNode = oChildNode.nextSibling;
|
|
}
|
|
|
|
oParentNode = nhn.DOMFix.parentNode(oNode);
|
|
|
|
/* check parent */
|
|
while(oParentNode && oParentNode.tagName !== "BODY"){
|
|
if(oParentNode.nodeType !== 1){
|
|
oParentNode = nhn.DOMFix.parentNode(oParentNode);
|
|
continue;
|
|
}
|
|
|
|
if(!bUnderline && this._hasTextDecoration(oParentNode, "underline")){
|
|
bUnderline = true;
|
|
this._setTextDecoration(oNode, "underline"); // set underline
|
|
}
|
|
|
|
if(!bLineThrough && this._hasTextDecoration(oParentNode, "line-through")){
|
|
bLineThrough = true;
|
|
this._setTextDecoration(oNode, "line-through"); // set line-through
|
|
}
|
|
|
|
if(bUnderline && bLineThrough){
|
|
return;
|
|
}
|
|
|
|
oParentNode = nhn.DOMFix.parentNode(oParentNode);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Range에 속한 노드들에 스타일을 적용한다
|
|
* @param {Object} oStyle 적용할 스타일을 가지는 Object (예) 글꼴 색 적용의 경우 { color : "#0075c8" }
|
|
* @param {Object} [oAttribute] 적용할 속성을 가지는 Object (예) 맞춤범 검사의 경우 { _sm2_spchk: "강남콩", class: "se2_check_spell" }
|
|
* @param {String} [sNewSpanMarker] 새로 추가된 SPAN 노드를 나중에 따로 처리해야하는 경우 마킹을 위해 사용하는 문자열
|
|
* @param {Boolean} [bIncludeLI] LI 도 스타일 적용에 포함할 것인지의 여부 [COM-1051] _getStyleParentNodes 메서드 참고하기
|
|
* @param {Boolean} [bCheckTextDecoration] 취소선/밑줄 처리를 적용할 것인지 여부 [SMARTEDITORSUS-26] _setTextDecoration 메서드 참고하기
|
|
*/
|
|
styleRange : function(oStyle, oAttribute, sNewSpanMarker, bIncludeLI, bCheckTextDecoration){
|
|
var aStyleParents = this.aStyleParents = this._getStyleParentNodes(sNewSpanMarker, bIncludeLI);
|
|
if(aStyleParents.length < 1){return;}
|
|
|
|
var sName, sValue;
|
|
|
|
for(var i=0; i<aStyleParents.length; i++){
|
|
for(var x in oStyle){
|
|
sName = x;
|
|
sValue = oStyle[sName];
|
|
|
|
if(typeof sValue != "string"){continue;}
|
|
|
|
// [SMARTEDITORSUS-26] 글꼴 색을 적용할 때 취소선/밑줄의 색상도 처리되도록 추가
|
|
if(bCheckTextDecoration && oStyle.color){
|
|
this._checkTextDecoration(aStyleParents[i]);
|
|
}
|
|
|
|
aStyleParents[i].style[sName] = sValue;
|
|
}
|
|
|
|
if(!oAttribute){continue;}
|
|
|
|
for(var x in oAttribute){
|
|
sName = x;
|
|
sValue = oAttribute[sName];
|
|
|
|
if(typeof sValue != "string"){continue;}
|
|
|
|
if(sName == "class"){
|
|
jindo.$Element(aStyleParents[i]).addClass(sValue);
|
|
}else{
|
|
aStyleParents[i].setAttribute(sName, sValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
this.reset(this._window);
|
|
this.setStartBefore(aStyleParents[0]);
|
|
this.setEndAfter(aStyleParents[aStyleParents.length-1]);
|
|
},
|
|
|
|
expandBothEnds : function(){
|
|
this.expandStart();
|
|
this.expandEnd();
|
|
},
|
|
|
|
expandStart : function(){
|
|
if(this.startContainer.nodeType == 3 && this.startOffset !== 0){return;}
|
|
|
|
var elActualStartNode = this._getActualStartNode(this.startContainer, this.startOffset);
|
|
elActualStartNode = this._getPrevNode(elActualStartNode);
|
|
|
|
if(elActualStartNode.tagName == "BODY"){
|
|
this.setStartBefore(elActualStartNode);
|
|
}else{
|
|
this.setStartAfter(elActualStartNode);
|
|
}
|
|
},
|
|
|
|
expandEnd : function(){
|
|
if(this.endContainer.nodeType == 3 && this.endOffset < this.endContainer.nodeValue.length){return;}
|
|
|
|
var elActualEndNode = this._getActualEndNode(this.endContainer, this.endOffset);
|
|
elActualEndNode = this._getNextNode(elActualEndNode);
|
|
|
|
if(elActualEndNode.tagName == "BODY"){
|
|
this.setEndAfter(elActualEndNode);
|
|
}else{
|
|
this.setEndBefore(elActualEndNode);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Style 을 적용할 노드를 가져온다
|
|
* @param {String} [sNewSpanMarker] 새로 추가하는 SPAN 노드를 마킹을 위해 사용하는 문자열
|
|
* @param {Boolean} [bIncludeLI] LI 도 스타일 적용에 포함할 것인지의 여부
|
|
* @return {Array} Style 을 적용할 노드 배열
|
|
*/
|
|
_getStyleParentNodes : function(sNewSpanMarker, bIncludeLI){
|
|
this._splitTextEndNodesOfTheRange();
|
|
|
|
var oSNode = this.getStartNode();
|
|
var oENode = this.getEndNode();
|
|
|
|
var aAllNodes = this._getNodesInRange();
|
|
var aResult = [];
|
|
var nResult = 0;
|
|
|
|
var oNode, oTmpNode, iStartRelPos, iEndRelPos, oSpan;
|
|
var nInitialLength = aAllNodes.length;
|
|
var arAllBottomNodes = jindo.$A(aAllNodes).filter(function(v){return (!v.firstChild || (bIncludeLI && v.tagName=="LI"));});
|
|
|
|
// [COM-1051] 본문내용을 한 줄만 입력하고 번호 매긴 상태에서 글자크기를 변경하면 번호크기는 변하지 않는 문제
|
|
// 부모 노드 중 LI 가 있고, 해당 LI 의 모든 자식 노드가 선택된 상태라면 LI에도 스타일을 적용하도록 처리함
|
|
// --- Range 에 LI 가 포함되지 않은 경우, LI 를 포함하도록 처리
|
|
var elTmpNode = this.commonAncestorContainer;
|
|
if(bIncludeLI){
|
|
while(elTmpNode){
|
|
if(elTmpNode.tagName == "LI"){
|
|
if(this._isFullyContained(elTmpNode, arAllBottomNodes)){
|
|
aResult[nResult++] = elTmpNode;
|
|
}
|
|
break;
|
|
}
|
|
|
|
elTmpNode = elTmpNode.parentNode;
|
|
}
|
|
}
|
|
|
|
for(var i=0; i<nInitialLength; i++){
|
|
oNode = aAllNodes[i];
|
|
|
|
if(!oNode){continue;}
|
|
|
|
// --- Range 에 LI 가 포함된 경우에 대한 LI 확인
|
|
if(bIncludeLI && oNode.tagName == "LI" && this._isFullyContained(oNode, arAllBottomNodes)){
|
|
aResult[nResult++] = oNode;
|
|
continue;
|
|
}
|
|
|
|
if(oNode.nodeType != 3){continue;}
|
|
if(oNode.nodeValue == "" || oNode.nodeValue.match(/^(\r|\n)+$/)){continue;}
|
|
|
|
var oParentNode = nhn.DOMFix.parentNode(oNode);
|
|
|
|
// 부모 노드가 SPAN 인 경우에는 새로운 SPAN 을 생성하지 않고 SPAN 을 리턴 배열에 추가함
|
|
if(oParentNode.tagName == "SPAN"){
|
|
if(this._isFullyContained(oParentNode, arAllBottomNodes, oNode)){
|
|
aResult[nResult++] = oParentNode;
|
|
continue;
|
|
}
|
|
}else{
|
|
// [SMARTEDITORSUS-1513] 선택된 영역을 single node로 감싸는 상위 span 노드가 있으면 리턴 배열에 추가
|
|
var oParentSingleSpan = this._findParentSingleSpan(oParentNode);
|
|
if(oParentSingleSpan){
|
|
aResult[nResult++] = oParentSingleSpan;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
oSpan = this._document.createElement("SPAN");
|
|
oParentNode.insertBefore(oSpan, oNode);
|
|
oSpan.appendChild(oNode);
|
|
aResult[nResult++] = oSpan;
|
|
|
|
if(sNewSpanMarker){oSpan.setAttribute(sNewSpanMarker, "true");}
|
|
}
|
|
|
|
this.setStartBefore(oSNode);
|
|
this.setEndAfter(oENode);
|
|
|
|
return aResult;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1513][SMARTEDITORSUS-1648] 해당노드가 single child로 묶이는 상위 span 노드가 있는지 찾는다.
|
|
* @param {Node} oNode 검사할 노드
|
|
* @return {Element} 상위 span 노드, 없으면 null
|
|
*/
|
|
_findParentSingleSpan : function(oNode){
|
|
if(!oNode){
|
|
return null;
|
|
}
|
|
// ZWNBSP 문자가 같이 있는 경우도 있기 때문에 실제 노드를 카운팅해야 함
|
|
for(var i = 0, nCnt = 0, sValue, oChild, aChildNodes = oNode.childNodes; (oChild = aChildNodes[i]); i++){
|
|
sValue = oChild.nodeValue;
|
|
if(this._rxCursorHolder.test(sValue)){
|
|
continue;
|
|
}else{
|
|
nCnt++;
|
|
}
|
|
if(nCnt > 1){ // 싱글노드가 아니면 더이상 찾지 않고 null 반환
|
|
return null;
|
|
}
|
|
}
|
|
if(oNode.nodeName === "SPAN"){
|
|
return oNode;
|
|
}else{
|
|
return this._findParentSingleSpan(oNode.parentNode);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 컨테이너 엘리먼트(elContainer)의 모든 자식노드가 노드 배열(waAllNodes)에 속하는지 확인한다
|
|
* 첫 번째 자식 노드와 마지막 자식 노드가 노드 배열에 속하는지를 확인한다
|
|
* @param {Element} elContainer 컨테이너 엘리먼트
|
|
* @param {jindo.$A} waAllNodes Node 의 $A 배열
|
|
* @param {Node} [oNode] 성능을 위한 옵션 노드로 컨테이너의 첫 번째 혹은 마지막 자식 노드와 같으면 indexOf 함수 사용을 줄일 수 있음
|
|
* @return {Array} Style 을 적용할 노드 배열
|
|
*/
|
|
// check if all the child nodes of elContainer are in waAllNodes
|
|
_isFullyContained : function(elContainer, waAllNodes, oNode){
|
|
var nSIdx, nEIdx;
|
|
var oTmpNode = this._getVeryFirstRealChild(elContainer);
|
|
// do quick checks before trying indexOf() because indexOf() function is very slow
|
|
// oNode is optional
|
|
if(oNode && oTmpNode == oNode){
|
|
nSIdx = 1;
|
|
}else{
|
|
nSIdx = waAllNodes.indexOf(oTmpNode);
|
|
}
|
|
|
|
if(nSIdx != -1){
|
|
oTmpNode = this._getVeryLastRealChild(elContainer);
|
|
if(oNode && oTmpNode == oNode){
|
|
nEIdx = 1;
|
|
}else{
|
|
nEIdx = waAllNodes.indexOf(oTmpNode);
|
|
}
|
|
}
|
|
|
|
return (nSIdx != -1 && nEIdx != -1);
|
|
},
|
|
|
|
_getVeryFirstChild : function(oNode){
|
|
if(oNode.firstChild){return this._getVeryFirstChild(oNode.firstChild);}
|
|
return oNode;
|
|
},
|
|
|
|
_getVeryLastChild : function(oNode){
|
|
if(oNode.lastChild){return this._getVeryLastChild(oNode.lastChild);}
|
|
return oNode;
|
|
},
|
|
|
|
_getFirstRealChild : function(oNode){
|
|
var oFirstNode = oNode.firstChild;
|
|
while(oFirstNode && oFirstNode.nodeType == 3 && oFirstNode.nodeValue == ""){oFirstNode = oFirstNode.nextSibling;}
|
|
|
|
return oFirstNode;
|
|
},
|
|
|
|
_getLastRealChild : function(oNode){
|
|
var oLastNode = oNode.lastChild;
|
|
while(oLastNode && oLastNode.nodeType == 3 && oLastNode.nodeValue == ""){oLastNode = oLastNode.previousSibling;}
|
|
|
|
return oLastNode;
|
|
},
|
|
|
|
_getVeryFirstRealChild : function(oNode){
|
|
var oFirstNode = this._getFirstRealChild(oNode);
|
|
if(oFirstNode){return this._getVeryFirstRealChild(oFirstNode);}
|
|
return oNode;
|
|
},
|
|
_getVeryLastRealChild : function(oNode){
|
|
var oLastNode = this._getLastRealChild(oNode);
|
|
if(oLastNode){return this._getVeryLastChild(oLastNode);}
|
|
return oNode;
|
|
},
|
|
|
|
_getLineStartInfo : function(node){
|
|
var frontEndFinal = null;
|
|
var frontEnd = node;
|
|
var lineBreaker = node;
|
|
var bParentBreak = false;
|
|
|
|
var rxLineBreaker = this.rxLineBreaker;
|
|
|
|
// vertical(parent) search
|
|
function getLineStart(node){
|
|
if(!node){return;}
|
|
if(frontEndFinal){return;}
|
|
|
|
if(rxLineBreaker.test(node.tagName)){
|
|
lineBreaker = node;
|
|
frontEndFinal = frontEnd;
|
|
|
|
bParentBreak = true;
|
|
|
|
return;
|
|
}else{
|
|
frontEnd = node;
|
|
}
|
|
|
|
getFrontEnd(node.previousSibling);
|
|
|
|
if(frontEndFinal){return;}
|
|
getLineStart(nhn.DOMFix.parentNode(node));
|
|
}
|
|
|
|
// horizontal(sibling) search
|
|
function getFrontEnd(node){
|
|
if(!node){return;}
|
|
if(frontEndFinal){return;}
|
|
|
|
if(rxLineBreaker.test(node.tagName)){
|
|
lineBreaker = node;
|
|
frontEndFinal = frontEnd;
|
|
|
|
bParentBreak = false;
|
|
return;
|
|
}
|
|
|
|
if(node.firstChild && node.tagName != "TABLE"){
|
|
var curNode = node.lastChild;
|
|
while(curNode && !frontEndFinal){
|
|
getFrontEnd(curNode);
|
|
|
|
curNode = curNode.previousSibling;
|
|
}
|
|
}else{
|
|
frontEnd = node;
|
|
}
|
|
|
|
if(!frontEndFinal){
|
|
getFrontEnd(node.previousSibling);
|
|
}
|
|
}
|
|
|
|
if(rxLineBreaker.test(node.tagName)){
|
|
frontEndFinal = node;
|
|
}else{
|
|
getLineStart(node);
|
|
}
|
|
|
|
return {oNode: frontEndFinal, oLineBreaker: lineBreaker, bParentBreak: bParentBreak};
|
|
},
|
|
|
|
_getLineEndInfo : function(node){
|
|
var backEndFinal = null;
|
|
var backEnd = node;
|
|
var lineBreaker = node;
|
|
var bParentBreak = false;
|
|
|
|
var rxLineBreaker = this.rxLineBreaker;
|
|
|
|
// vertical(parent) search
|
|
function getLineEnd(node){
|
|
if(!node){return;}
|
|
if(backEndFinal){return;}
|
|
|
|
if(rxLineBreaker.test(node.tagName)){
|
|
lineBreaker = node;
|
|
backEndFinal = backEnd;
|
|
|
|
bParentBreak = true;
|
|
|
|
return;
|
|
}else{
|
|
backEnd = node;
|
|
}
|
|
|
|
getBackEnd(node.nextSibling);
|
|
if(backEndFinal){return;}
|
|
|
|
getLineEnd(nhn.DOMFix.parentNode(node));
|
|
}
|
|
|
|
// horizontal(sibling) search
|
|
function getBackEnd(node){
|
|
if(!node){return;}
|
|
if(backEndFinal){return;}
|
|
|
|
if(rxLineBreaker.test(node.tagName)){
|
|
lineBreaker = node;
|
|
backEndFinal = backEnd;
|
|
|
|
bParentBreak = false;
|
|
|
|
return;
|
|
}
|
|
|
|
if(node.firstChild && node.tagName != "TABLE"){
|
|
var curNode = node.firstChild;
|
|
while(curNode && !backEndFinal){
|
|
getBackEnd(curNode);
|
|
|
|
curNode = curNode.nextSibling;
|
|
}
|
|
}else{
|
|
backEnd = node;
|
|
}
|
|
|
|
if(!backEndFinal){
|
|
getBackEnd(node.nextSibling);
|
|
}
|
|
}
|
|
|
|
if(rxLineBreaker.test(node.tagName)){
|
|
backEndFinal = node;
|
|
}else{
|
|
getLineEnd(node);
|
|
}
|
|
|
|
return {oNode: backEndFinal, oLineBreaker: lineBreaker, bParentBreak: bParentBreak};
|
|
},
|
|
|
|
getLineInfo : function(bAfter){
|
|
var bAfter = bAfter || false;
|
|
|
|
var oSNode = this.getStartNode();
|
|
var oENode = this.getEndNode();
|
|
|
|
// oSNode && oENode will be null if the range is currently collapsed and the cursor is not located in the middle of a text node.
|
|
if(!oSNode){oSNode = this.getNodeAroundRange(!bAfter, true);}
|
|
if(!oENode){oENode = this.getNodeAroundRange(!bAfter, true);}
|
|
|
|
var oStart = this._getLineStartInfo(oSNode);
|
|
var oStartNode = oStart.oNode;
|
|
var oEnd = this._getLineEndInfo(oENode);
|
|
var oEndNode = oEnd.oNode;
|
|
|
|
if(oSNode != oStartNode || oENode != oEndNode){
|
|
// check if the start node is positioned after the range's ending point
|
|
// or
|
|
// if the end node is positioned before the range's starting point
|
|
var iRelativeStartPos = this._compareEndPoint(nhn.DOMFix.parentNode(oStartNode), this._getPosIdx(oStartNode), this.endContainer, this.endOffset);
|
|
var iRelativeEndPos = this._compareEndPoint(nhn.DOMFix.parentNode(oEndNode), this._getPosIdx(oEndNode)+1, this.startContainer, this.startOffset);
|
|
|
|
if(!(iRelativeStartPos <= 0 && iRelativeEndPos >= 0)){
|
|
oSNode = this.getNodeAroundRange(false, true);
|
|
oENode = this.getNodeAroundRange(false, true);
|
|
oStart = this._getLineStartInfo(oSNode);
|
|
oEnd = this._getLineEndInfo(oENode);
|
|
}
|
|
}
|
|
|
|
return {oStart: oStart, oEnd: oEnd};
|
|
},
|
|
|
|
/**
|
|
* 커서홀더나 공백을 제외한 child 노드가 하나만 있는 경우만 node 를 반환한다.
|
|
* @param {Node} oNode 확인할 노드
|
|
* @return {Node} single child node를 반환한다. 없거나 두개 이상이면 null 을 반환
|
|
*/
|
|
_findSingleChild : function(oNode){
|
|
if(!oNode){
|
|
return null;
|
|
}
|
|
var oSingleChild = null;
|
|
// ZWNBSP 문자가 같이 있는 경우도 있기 때문에 실제 노드를 카운팅해야 함
|
|
for(var i = 0, nCnt = 0, sValue, oChild, aChildNodes = oNode.childNodes; (oChild = aChildNodes[i]); i++){
|
|
sValue = oChild.nodeValue;
|
|
if(this._rxCursorHolder.test(sValue)){
|
|
continue;
|
|
}else{
|
|
oSingleChild = oChild;
|
|
nCnt++;
|
|
}
|
|
if(nCnt > 1){ // 싱글노드가 아니면 더이상 찾지 않고 null 반환
|
|
return null;
|
|
}
|
|
}
|
|
return oSingleChild;
|
|
},
|
|
|
|
/**
|
|
* 해당요소의 최하위까지 검색해 커서홀더만 감싸고 있는지 여부를 반환
|
|
* @param {Node} oNode 확인할 노드
|
|
* @return {Boolean} 커서홀더만 있는 경우 true 반환
|
|
*/
|
|
_hasCursorHolderOnly : function(oNode){
|
|
if(!oNode || oNode.nodeType !== 1){
|
|
return false;
|
|
}
|
|
if(this._rxCursorHolder.test(oNode.innerHTML)){
|
|
return true;
|
|
}else{
|
|
return this._hasCursorHolderOnly(this._findSingleChild(oNode));
|
|
}
|
|
}
|
|
}).extend(nhn.W3CDOMRange);
|
|
|
|
/**
|
|
* @fileOverview This file contains cross-browser selection function
|
|
* @name BrowserSelection.js
|
|
*/
|
|
nhn.BrowserSelection = function(win){
|
|
this.init = function(win){
|
|
this._window = win || window;
|
|
this._document = this._window.document;
|
|
};
|
|
|
|
this.init(win);
|
|
|
|
// [SMARTEDITORSUS-888] IE9 이후로 document.createRange 를 지원
|
|
/* var oAgentInfo = jindo.$Agent().navigator();
|
|
if(oAgentInfo.ie){
|
|
nhn.BrowserSelectionImpl_IE.apply(this);
|
|
}else{
|
|
nhn.BrowserSelectionImpl_FF.apply(this);
|
|
}*/
|
|
|
|
if(!!this._document.createRange){
|
|
nhn.BrowserSelectionImpl_FF.apply(this);
|
|
}else{
|
|
nhn.BrowserSelectionImpl_IE.apply(this);
|
|
}
|
|
|
|
this.selectRange = function(oRng){
|
|
this.selectNone();
|
|
this.addRange(oRng);
|
|
};
|
|
|
|
this.selectionLoaded = true;
|
|
if(!this._oSelection){this.selectionLoaded = false;}
|
|
};
|
|
|
|
nhn.BrowserSelectionImpl_FF = function(){
|
|
this._oSelection = this._window.getSelection();
|
|
|
|
this.getRangeAt = function(iNum){
|
|
iNum = iNum || 0;
|
|
|
|
try{
|
|
var oFFRange = this._oSelection.getRangeAt(iNum);
|
|
}catch(e){return new nhn.W3CDOMRange(this._window);}
|
|
|
|
return this._FFRange2W3CRange(oFFRange);
|
|
};
|
|
|
|
this.addRange = function(oW3CRange){
|
|
var oFFRange = this._W3CRange2FFRange(oW3CRange);
|
|
this._oSelection.addRange(oFFRange);
|
|
};
|
|
|
|
this.selectNone = function(){
|
|
this._oSelection.removeAllRanges();
|
|
};
|
|
|
|
this.getCommonAncestorContainer = function(oW3CRange){
|
|
var oFFRange = this._W3CRange2FFRange(oW3CRange);
|
|
return oFFRange.commonAncestorContainer;
|
|
};
|
|
|
|
this.isCollapsed = function(oW3CRange){
|
|
var oFFRange = this._W3CRange2FFRange(oW3CRange);
|
|
return oFFRange.collapsed;
|
|
};
|
|
|
|
this.compareEndPoints = function(elContainerA, nOffsetA, elContainerB, nOffsetB){
|
|
var oFFRangeA = this._document.createRange();
|
|
var oFFRangeB = this._document.createRange();
|
|
oFFRangeA.setStart(elContainerA, nOffsetA);
|
|
oFFRangeB.setStart(elContainerB, nOffsetB);
|
|
oFFRangeA.collapse(true);
|
|
oFFRangeB.collapse(true);
|
|
|
|
try{
|
|
return oFFRangeA.compareBoundaryPoints(1, oFFRangeB);
|
|
}catch(e){
|
|
return 1;
|
|
}
|
|
};
|
|
|
|
this._FFRange2W3CRange = function(oFFRange){
|
|
var oW3CRange = new nhn.W3CDOMRange(this._window);
|
|
|
|
oW3CRange.setStart(oFFRange.startContainer, oFFRange.startOffset, true);
|
|
oW3CRange.setEnd(oFFRange.endContainer, oFFRange.endOffset, true);
|
|
|
|
return oW3CRange;
|
|
};
|
|
|
|
this._W3CRange2FFRange = function(oW3CRange){
|
|
var oFFRange = this._document.createRange();
|
|
oFFRange.setStart(oW3CRange.startContainer, oW3CRange.startOffset);
|
|
oFFRange.setEnd(oW3CRange.endContainer, oW3CRange.endOffset);
|
|
|
|
return oFFRange;
|
|
};
|
|
};
|
|
|
|
nhn.BrowserSelectionImpl_IE = function(){
|
|
this._oSelection = this._document.selection;
|
|
this.oLastRange = {
|
|
oBrowserRange : null,
|
|
elStartContainer : null,
|
|
nStartOffset : -1,
|
|
elEndContainer : null,
|
|
nEndOffset : -1
|
|
};
|
|
|
|
this._updateLastRange = function(oBrowserRange, oW3CRange){
|
|
this.oLastRange.oBrowserRange = oBrowserRange;
|
|
this.oLastRange.elStartContainer = oW3CRange.startContainer;
|
|
this.oLastRange.nStartOffset = oW3CRange.startOffset;
|
|
this.oLastRange.elEndContainer = oW3CRange.endContainer;
|
|
this.oLastRange.nEndOffset = oW3CRange.endOffset;
|
|
};
|
|
|
|
this.getRangeAt = function(iNum){
|
|
iNum = iNum || 0;
|
|
|
|
var oW3CRange, oBrowserRange;
|
|
if(this._oSelection.type == "Control"){
|
|
oW3CRange = new nhn.W3CDOMRange(this._window);
|
|
|
|
var oSelectedNode = this._oSelection.createRange().item(iNum);
|
|
|
|
// if the selction occurs in a different document, ignore
|
|
if(!oSelectedNode || oSelectedNode.ownerDocument != this._document){return oW3CRange;}
|
|
|
|
oW3CRange.selectNode(oSelectedNode);
|
|
|
|
return oW3CRange;
|
|
}else{
|
|
//oBrowserRange = this._oSelection.createRangeCollection().item(iNum);
|
|
oBrowserRange = this._oSelection.createRange();
|
|
|
|
var oSelectedNode = oBrowserRange.parentElement();
|
|
|
|
// if the selction occurs in a different document, ignore
|
|
if(!oSelectedNode || oSelectedNode.ownerDocument != this._document){
|
|
oW3CRange = new nhn.W3CDOMRange(this._window);
|
|
return oW3CRange;
|
|
}
|
|
oW3CRange = this._IERange2W3CRange(oBrowserRange);
|
|
|
|
return oW3CRange;
|
|
}
|
|
};
|
|
|
|
this.addRange = function(oW3CRange){
|
|
var oIERange = this._W3CRange2IERange(oW3CRange);
|
|
oIERange.select();
|
|
};
|
|
|
|
this.selectNone = function(){
|
|
this._oSelection.empty();
|
|
};
|
|
|
|
this.getCommonAncestorContainer = function(oW3CRange){
|
|
return this._W3CRange2IERange(oW3CRange).parentElement();
|
|
};
|
|
|
|
this.isCollapsed = function(oW3CRange){
|
|
var oRange = this._W3CRange2IERange(oW3CRange);
|
|
var oRange2 = oRange.duplicate();
|
|
|
|
oRange2.collapse();
|
|
|
|
return oRange.isEqual(oRange2);
|
|
};
|
|
|
|
this.compareEndPoints = function(elContainerA, nOffsetA, elContainerB, nOffsetB){
|
|
var oIERangeA, oIERangeB;
|
|
|
|
if(elContainerA === this.oLastRange.elStartContainer && nOffsetA === this.oLastRange.nStartOffset){
|
|
oIERangeA = this.oLastRange.oBrowserRange.duplicate();
|
|
oIERangeA.collapse(true);
|
|
}else{
|
|
if(elContainerA === this.oLastRange.elEndContainer && nOffsetA === this.oLastRange.nEndOffset){
|
|
oIERangeA = this.oLastRange.oBrowserRange.duplicate();
|
|
oIERangeA.collapse(false);
|
|
}else{
|
|
oIERangeA = this._getIERangeAt(elContainerA, nOffsetA);
|
|
}
|
|
}
|
|
|
|
if(elContainerB === this.oLastRange.elStartContainer && nOffsetB === this.oLastRange.nStartOffset){
|
|
oIERangeB = this.oLastRange.oBrowserRange.duplicate();
|
|
oIERangeB.collapse(true);
|
|
}else{
|
|
if(elContainerB === this.oLastRange.elEndContainer && nOffsetB === this.oLastRange.nEndOffset){
|
|
oIERangeB = this.oLastRange.oBrowserRange.duplicate();
|
|
oIERangeB.collapse(false);
|
|
}else{
|
|
oIERangeB = this._getIERangeAt(elContainerB, nOffsetB);
|
|
}
|
|
}
|
|
|
|
return oIERangeA.compareEndPoints("StartToStart", oIERangeB);
|
|
};
|
|
|
|
this._W3CRange2IERange = function(oW3CRange){
|
|
if(this.oLastRange.elStartContainer === oW3CRange.startContainer &&
|
|
this.oLastRange.nStartOffset === oW3CRange.startOffset &&
|
|
this.oLastRange.elEndContainer === oW3CRange.endContainer &&
|
|
this.oLastRange.nEndOffset === oW3CRange.endOffset){
|
|
return this.oLastRange.oBrowserRange;
|
|
}
|
|
|
|
var oStartIERange = this._getIERangeAt(oW3CRange.startContainer, oW3CRange.startOffset);
|
|
var oEndIERange = this._getIERangeAt(oW3CRange.endContainer, oW3CRange.endOffset);
|
|
oStartIERange.setEndPoint("EndToEnd", oEndIERange);
|
|
|
|
this._updateLastRange(oStartIERange, oW3CRange);
|
|
|
|
return oStartIERange;
|
|
};
|
|
|
|
this._getIERangeAt = function(oW3CContainer, iW3COffset){
|
|
var oIERange = this._document.body.createTextRange();
|
|
|
|
var oEndPointInfoForIERange = this._getSelectableNodeAndOffsetForIE(oW3CContainer, iW3COffset);
|
|
|
|
var oSelectableNode = oEndPointInfoForIERange.oSelectableNodeForIE;
|
|
var iIEOffset = oEndPointInfoForIERange.iOffsetForIE;
|
|
|
|
oIERange.moveToElementText(oSelectableNode);
|
|
|
|
oIERange.collapse(oEndPointInfoForIERange.bCollapseToStart);
|
|
oIERange.moveStart("character", iIEOffset);
|
|
|
|
return oIERange;
|
|
};
|
|
|
|
this._getSelectableNodeAndOffsetForIE = function(oW3CContainer, iW3COffset){
|
|
// var oIERange = this._document.body.createTextRange();
|
|
|
|
var oNonTextNode = null;
|
|
var aChildNodes = null;
|
|
var iNumOfLeftNodesToCount = 0;
|
|
|
|
if(oW3CContainer.nodeType == 3){
|
|
oNonTextNode = nhn.DOMFix.parentNode(oW3CContainer);
|
|
aChildNodes = nhn.DOMFix.childNodes(oNonTextNode);
|
|
iNumOfLeftNodesToCount = aChildNodes.length;
|
|
}else{
|
|
oNonTextNode = oW3CContainer;
|
|
aChildNodes = nhn.DOMFix.childNodes(oNonTextNode);
|
|
//iNumOfLeftNodesToCount = iW3COffset;
|
|
iNumOfLeftNodesToCount = (iW3COffset<aChildNodes.length)?iW3COffset:aChildNodes.length;
|
|
}
|
|
//@ room 4 improvement
|
|
var oNodeTester = null;
|
|
var iResultOffset = 0;
|
|
var bCollapseToStart = true;
|
|
|
|
for(var i=0; i<iNumOfLeftNodesToCount; i++){
|
|
oNodeTester = aChildNodes[i];
|
|
|
|
if(oNodeTester.nodeType == 3){
|
|
if(oNodeTester == oW3CContainer){break;}
|
|
|
|
iResultOffset += oNodeTester.nodeValue.length;
|
|
}else{
|
|
// oIERange.moveToElementText(oNodeTester);
|
|
oNonTextNode = oNodeTester;
|
|
iResultOffset = 0;
|
|
|
|
bCollapseToStart = false;
|
|
}
|
|
}
|
|
|
|
if(oW3CContainer.nodeType == 3){iResultOffset += iW3COffset;}
|
|
|
|
return {oSelectableNodeForIE:oNonTextNode, iOffsetForIE: iResultOffset, bCollapseToStart: bCollapseToStart};
|
|
};
|
|
|
|
this._IERange2W3CRange = function(oIERange){
|
|
var oW3CRange = new nhn.W3CDOMRange(this._window);
|
|
|
|
var oIEPointRange = null;
|
|
var oPosition = null;
|
|
|
|
oIEPointRange = oIERange.duplicate();
|
|
oIEPointRange.collapse(true);
|
|
|
|
oPosition = this._getW3CContainerAndOffset(oIEPointRange, true);
|
|
|
|
oW3CRange.setStart(oPosition.oContainer, oPosition.iOffset, true, true);
|
|
|
|
var oCollapsedChecker = oIERange.duplicate();
|
|
oCollapsedChecker.collapse(true);
|
|
if(oCollapsedChecker.isEqual(oIERange)){
|
|
oW3CRange.collapse(true);
|
|
}else{
|
|
oIEPointRange = oIERange.duplicate();
|
|
oIEPointRange.collapse(false);
|
|
oPosition = this._getW3CContainerAndOffset(oIEPointRange);
|
|
oW3CRange.setEnd(oPosition.oContainer, oPosition.iOffset, true);
|
|
}
|
|
|
|
this._updateLastRange(oIERange, oW3CRange);
|
|
|
|
return oW3CRange;
|
|
};
|
|
|
|
this._getW3CContainerAndOffset = function(oIEPointRange, bStartPt){
|
|
var oRgOrigPoint = oIEPointRange;
|
|
|
|
var oContainer = oRgOrigPoint.parentElement();
|
|
var offset = -1;
|
|
|
|
var oRgTester = this._document.body.createTextRange();
|
|
var aChildNodes = nhn.DOMFix.childNodes(oContainer);
|
|
var oPrevNonTextNode = null;
|
|
var pointRangeIdx = 0;
|
|
|
|
for(var i=0;i<aChildNodes.length;i++){
|
|
if(aChildNodes[i].nodeType == 3){continue;}
|
|
|
|
oRgTester.moveToElementText(aChildNodes[i]);
|
|
|
|
if(oRgTester.compareEndPoints("StartToStart", oIEPointRange)>=0){break;}
|
|
|
|
oPrevNonTextNode = aChildNodes[i];
|
|
}
|
|
|
|
var pointRangeIdx = i;
|
|
|
|
if(pointRangeIdx !== 0 && aChildNodes[pointRangeIdx-1].nodeType == 3){
|
|
var oRgTextStart = this._document.body.createTextRange();
|
|
var oCurTextNode = null;
|
|
if(oPrevNonTextNode){
|
|
oRgTextStart.moveToElementText(oPrevNonTextNode);
|
|
oRgTextStart.collapse(false);
|
|
oCurTextNode = oPrevNonTextNode.nextSibling;
|
|
}else{
|
|
oRgTextStart.moveToElementText(oContainer);
|
|
oRgTextStart.collapse(true);
|
|
oCurTextNode = oContainer.firstChild;
|
|
}
|
|
|
|
var oRgTextsUpToThePoint = oRgOrigPoint.duplicate();
|
|
oRgTextsUpToThePoint.setEndPoint("StartToStart", oRgTextStart);
|
|
|
|
var textCount = oRgTextsUpToThePoint.text.replace(/[\r\n]/g,"").length;
|
|
|
|
while(textCount > oCurTextNode.nodeValue.length && oCurTextNode.nextSibling){
|
|
textCount -= oCurTextNode.nodeValue.length;
|
|
oCurTextNode = oCurTextNode.nextSibling;
|
|
}
|
|
|
|
// this will enforce IE to re-reference oCurTextNode
|
|
var oTmp = oCurTextNode.nodeValue;
|
|
|
|
if(bStartPt && oCurTextNode.nextSibling && oCurTextNode.nextSibling.nodeType == 3 && textCount == oCurTextNode.nodeValue.length){
|
|
textCount -= oCurTextNode.nodeValue.length;
|
|
oCurTextNode = oCurTextNode.nextSibling;
|
|
}
|
|
|
|
oContainer = oCurTextNode;
|
|
offset = textCount;
|
|
}else{
|
|
oContainer = oRgOrigPoint.parentElement();
|
|
offset = pointRangeIdx;
|
|
}
|
|
return {"oContainer" : oContainer, "iOffset" : offset};
|
|
};
|
|
};
|
|
|
|
nhn.DOMFix = new (jindo.$Class({
|
|
$init : function(){
|
|
if(jindo.$Agent().navigator().ie || jindo.$Agent().navigator().opera){
|
|
this.childNodes = this._childNodes_Fix;
|
|
this.parentNode = this._parentNode_Fix;
|
|
}else{
|
|
this.childNodes = this._childNodes_Native;
|
|
this.parentNode = this._parentNode_Native;
|
|
}
|
|
},
|
|
|
|
_parentNode_Native : function(elNode){
|
|
return elNode.parentNode;
|
|
},
|
|
|
|
_parentNode_Fix : function(elNode){
|
|
if(!elNode){return elNode;}
|
|
|
|
while(elNode.previousSibling){elNode = elNode.previousSibling;}
|
|
|
|
return elNode.parentNode;
|
|
},
|
|
|
|
_childNodes_Native : function(elNode){
|
|
return elNode.childNodes;
|
|
},
|
|
|
|
_childNodes_Fix : function(elNode){
|
|
var aResult = null;
|
|
var nCount = 0;
|
|
|
|
if(elNode){
|
|
var aResult = [];
|
|
elNode = elNode.firstChild;
|
|
while(elNode){
|
|
aResult[nCount++] = elNode;
|
|
elNode=elNode.nextSibling;
|
|
}
|
|
}
|
|
|
|
return aResult;
|
|
}
|
|
}))();
|
|
/*[
|
|
* ADD_APP_PROPERTY
|
|
*
|
|
* 주요 오브젝트를 모든 플러그인에서 this.oApp를 통해서 직접 접근 가능 하도록 등록한다.
|
|
*
|
|
* sPropertyName string 등록명
|
|
* oProperty object 등록시킬 오브젝트
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* REGISTER_BROWSER_EVENT
|
|
*
|
|
* 특정 브라우저 이벤트가 발생 했을때 Husky 메시지를 발생 시킨다.
|
|
*
|
|
* obj HTMLElement 브라우저 이벤트를 발생 시킬 HTML 엘리먼트
|
|
* sEvent string 발생 대기 할 브라우저 이벤트
|
|
* sMsg string 발생 할 Husky 메시지
|
|
* aParams array 메시지에 넘길 파라미터
|
|
* nDelay number 브라우저 이벤트 발생 후 Husky 메시지 발생 사이에 딜레이를 주고 싶을 경우 설정. (1/1000초 단위)
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* DISABLE_MESSAGE
|
|
*
|
|
* 특정 메시지를 코어에서 무시하고 라우팅 하지 않도록 비활성화 한다.
|
|
*
|
|
* sMsg string 비활성화 시킬 메시지
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* ENABLE_MESSAGE
|
|
*
|
|
* 무시하도록 설정된 메시지를 무시하지 않도록 활성화 한다.
|
|
*
|
|
* sMsg string 활성화 시킬 메시지
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* EXEC_ON_READY_FUNCTION
|
|
*
|
|
* oApp.run({fnOnAppReady:fnOnAppReady})와 같이 run 호출 시점에 지정된 함수가 있을 경우 이를 MSG_APP_READY 시점에 실행 시킨다.
|
|
* 코어에서 자동으로 발생시키는 메시지로 직접 발생시키지는 않도록 한다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc Husky Framework에서 자주 사용되는 메시지를 처리하는 플러그인
|
|
*/
|
|
nhn.husky.CorePlugin = jindo.$Class({
|
|
name : "CorePlugin",
|
|
|
|
// nStatus = 0(request not sent), 1(request sent), 2(response received)
|
|
// sContents = response
|
|
htLazyLoadRequest_plugins : {},
|
|
htLazyLoadRequest_allFiles : {},
|
|
|
|
htHTMLLoaded : {},
|
|
|
|
$AFTER_MSG_APP_READY : function(){
|
|
this.oApp.exec("EXEC_ON_READY_FUNCTION", []);
|
|
},
|
|
|
|
$ON_ADD_APP_PROPERTY : function(sPropertyName, oProperty){
|
|
this.oApp[sPropertyName] = oProperty;
|
|
},
|
|
|
|
$ON_REGISTER_BROWSER_EVENT : function(obj, sEvent, sMsg, aParams, nDelay){
|
|
this.oApp.registerBrowserEvent(obj, sEvent, sMsg, aParams, nDelay);
|
|
},
|
|
|
|
$ON_DISABLE_MESSAGE : function(sMsg){
|
|
this.oApp.disableMessage(sMsg, true);
|
|
},
|
|
|
|
$ON_ENABLE_MESSAGE : function(sMsg){
|
|
this.oApp.disableMessage(sMsg, false);
|
|
},
|
|
|
|
$ON_LOAD_FULL_PLUGIN : function(aFilenames, sClassName, sMsgName, oThisRef, oArguments){
|
|
var oPluginRef = oThisRef.$this || oThisRef;
|
|
// var nIdx = _nIdx||0;
|
|
|
|
var sFilename = aFilenames[0];
|
|
|
|
if(!this.htLazyLoadRequest_plugins[sFilename]){
|
|
this.htLazyLoadRequest_plugins[sFilename] = {nStatus:1, sContents:""};
|
|
}
|
|
|
|
if(this.htLazyLoadRequest_plugins[sFilename].nStatus === 2){
|
|
//this.oApp.delayedExec("MSG_FULL_PLUGIN_LOADED", [sFilename, sClassName, sMsgName, oThisRef, oArguments, false], 0);
|
|
this.oApp.exec("MSG_FULL_PLUGIN_LOADED", [sFilename, sClassName, sMsgName, oThisRef, oArguments, false]);
|
|
}else{
|
|
this._loadFullPlugin(aFilenames, sClassName, sMsgName, oThisRef, oArguments, 0);
|
|
}
|
|
},
|
|
|
|
_loadFullPlugin : function(aFilenames, sClassName, sMsgName, oThisRef, oArguments, nIdx){
|
|
jindo.LazyLoading.load(nhn.husky.SE2M_Configuration.LazyLoad.sJsBaseURI+"/"+aFilenames[nIdx],
|
|
jindo.$Fn(function(aFilenames, sClassName, sMsgName, oThisRef, oArguments, nIdx){
|
|
var sCurFilename = aFilenames[nIdx];
|
|
|
|
// plugin filename
|
|
var sFilename = aFilenames[0];
|
|
if(nIdx == aFilenames.length-1){
|
|
this.htLazyLoadRequest_plugins[sFilename].nStatus=2;
|
|
this.oApp.exec("MSG_FULL_PLUGIN_LOADED", [aFilenames, sClassName, sMsgName, oThisRef, oArguments]);
|
|
return;
|
|
}
|
|
//this.oApp.exec("LOAD_FULL_PLUGIN", [aFilenames, sClassName, sMsgName, oThisRef, oArguments, nIdx+1]);
|
|
this._loadFullPlugin(aFilenames, sClassName, sMsgName, oThisRef, oArguments, nIdx+1);
|
|
}, this).bind(aFilenames, sClassName, sMsgName, oThisRef, oArguments, nIdx),
|
|
|
|
"utf-8"
|
|
);
|
|
},
|
|
|
|
$ON_MSG_FULL_PLUGIN_LOADED : function(aFilenames, sClassName, sMsgName, oThisRef, oArguments, oRes){
|
|
// oThisRef.$this는 현재 로드되는 플러그인이 parent 인스턴스일 경우 존재 함. oThisRef.$this는 현재 플러그인(oThisRef)를 parent로 삼고 있는 인스턴스
|
|
// oThisRef에 $this 속성이 없다면 parent가 아닌 일반 인스턴스
|
|
// oPluginRef는 결과적으로 상속 관계가 있다면 자식 인스턴스를 아니라면 일반적인 인스턴스를 가짐
|
|
var oPluginRef = oThisRef.$this || oThisRef;
|
|
|
|
var sFilename = aFilenames;
|
|
|
|
// now the source code is loaded, remove the loader handlers
|
|
for(var i=0, nLen=oThisRef._huskyFLT.length; i<nLen; i++){
|
|
var sLoaderHandlerName = "$BEFORE_"+oThisRef._huskyFLT[i];
|
|
|
|
// if child class has its own loader function, remove the loader from current instance(parent) only
|
|
var oRemoveFrom = (oThisRef.$this && oThisRef[sLoaderHandlerName])?oThisRef:oPluginRef;
|
|
oRemoveFrom[sLoaderHandlerName] = null;
|
|
this.oApp.createMessageMap(sLoaderHandlerName);
|
|
}
|
|
|
|
var oPlugin = eval(sClassName+".prototype");
|
|
//var oPlugin = eval("new "+sClassName+"()");
|
|
|
|
var bAcceptLocalBeforeFirstAgain = false;
|
|
// if there were no $LOCAL_BEFORE_FIRST in already-loaded script, set to accept $LOCAL_BEFORE_FIRST next time as the function could be included in the lazy-loaded script.
|
|
if(typeof oPluginRef["$LOCAL_BEFORE_FIRST"] !== "function"){
|
|
this.oApp.acceptLocalBeforeFirstAgain(oPluginRef, true);
|
|
}
|
|
|
|
for(var x in oPlugin){
|
|
// 자식 인스턴스에 parent를 override하는 함수가 없다면 parent 인스턴스에 함수 복사 해 줌. 이때 함수만 복사하고, 나머지 속성들은 현재 인스턴스에 존재 하지 않을 경우에만 복사.
|
|
if(oThisRef.$this && (!oThisRef[x] || (typeof oPlugin[x] === "function" && x != "constructor"))){
|
|
oThisRef[x] = jindo.$Fn(oPlugin[x], oPluginRef).bind();
|
|
}
|
|
|
|
// 현재 인스턴스에 함수 복사 해 줌. 이때 함수만 복사하고, 나머지 속성들은 현재 인스턴스에 존재 하지 않을 경우에만 복사
|
|
if(oPlugin[x] && (!oPluginRef[x] || (typeof oPlugin[x] === "function" && x != "constructor"))){
|
|
oPluginRef[x] = oPlugin[x];
|
|
|
|
// 새로 추가되는 함수가 메시지 핸들러라면 메시지 매핑에 추가 해 줌
|
|
if(x.match(/^\$(LOCAL|BEFORE|ON|AFTER)_/)){
|
|
this.oApp.addToMessageMap(x, oPluginRef);
|
|
}
|
|
}
|
|
}
|
|
|
|
if(bAcceptLocalBeforeFirstAgain){
|
|
this.oApp.acceptLocalBeforeFirstAgain(oPluginRef, true);
|
|
}
|
|
|
|
// re-send the message after all the jindo.$super handlers are executed
|
|
if(!oThisRef.$this){
|
|
this.oApp.exec(sMsgName, oArguments);
|
|
}
|
|
},
|
|
|
|
$ON_LOAD_HTML : function(sId){
|
|
if(this.htHTMLLoaded[sId]) return;
|
|
|
|
var elTextarea = jindo.$("_llh_"+sId);
|
|
if(!elTextarea) return;
|
|
|
|
this.htHTMLLoaded[sId] = true;
|
|
|
|
var elTmp = document.createElement("DIV");
|
|
elTmp.innerHTML = elTextarea.value;
|
|
|
|
while(elTmp.firstChild){
|
|
elTextarea.parentNode.insertBefore(elTmp.firstChild, elTextarea);
|
|
}
|
|
},
|
|
|
|
$ON_EXEC_ON_READY_FUNCTION : function(){
|
|
if(typeof this.oApp.htRunOptions.fnOnAppReady == "function"){this.oApp.htRunOptions.fnOnAppReady();}
|
|
}
|
|
});
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that bridges the HuskyRange function
|
|
* @name hp_HuskyRangeManager.js
|
|
*/
|
|
nhn.husky.HuskyRangeManager = jindo.$Class({
|
|
name : "HuskyRangeManager",
|
|
|
|
oWindow : null,
|
|
|
|
$init : function(win){
|
|
this.oWindow = win || window;
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
if(this.oWindow && this.oWindow.tagName == "IFRAME"){
|
|
this.oWindow = this.oWindow.contentWindow;
|
|
nhn.CurrentSelection.setWindow(this.oWindow);
|
|
}
|
|
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getSelection", jindo.$Fn(this.getSelection, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getEmptySelection", jindo.$Fn(this.getEmptySelection, this).bind()]);
|
|
},
|
|
|
|
$ON_SET_EDITING_WINDOW : function(oWindow){
|
|
this.oWindow = oWindow;
|
|
},
|
|
|
|
getEmptySelection : function(oWindow){
|
|
var oHuskyRange = new nhn.HuskyRange(oWindow || this.oWindow);
|
|
return oHuskyRange;
|
|
},
|
|
|
|
getSelection : function(oWindow){
|
|
this.oApp.exec("RESTORE_IE_SELECTION", []);
|
|
|
|
var oHuskyRange = this.getEmptySelection(oWindow);
|
|
|
|
// this may throw an exception if the selected is area is not yet shown
|
|
try{
|
|
oHuskyRange.setFromSelection();
|
|
}catch(e){}
|
|
|
|
return oHuskyRange;
|
|
}
|
|
});
|
|
//}
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to the tool bar UI
|
|
* @name hp_SE2M_Toolbar.js
|
|
*/
|
|
nhn.husky.SE2M_Toolbar = jindo.$Class({
|
|
name : "SE2M_Toolbar",
|
|
|
|
toolbarArea : null,
|
|
toolbarButton : null,
|
|
uiNameTag : "uiName",
|
|
|
|
// 0: unknown
|
|
// 1: all enabled
|
|
// 2: all disabled
|
|
nUIStatus : 1,
|
|
|
|
sUIClassPrefix : "husky_seditor_ui_",
|
|
|
|
aUICmdMap : null,
|
|
elFirstToolbarItem : null,
|
|
|
|
/**
|
|
* 얼럿 레이어를 닫을 때 사용할 이벤트 핸들러
|
|
* @param {Function} fCallback 레이어를 닫을때 실행할 콜백함수
|
|
* @returns {Boolean} 무조건 false 를 리턴한다. onclick 이벤트에 바로 할당하기 때문에 이벤트 전파를 막기 위해 무조건 false를 리턴
|
|
*/
|
|
_hideAlert : function(fCallback){
|
|
if(typeof fCallback == "function"){
|
|
fCallback();
|
|
}
|
|
this._elAlertLayer.style.display = "none";
|
|
this.oApp.exec("HIDE_EDITING_AREA_COVER");
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* 시스템 alert/confirm 대신 얼럿 레이어를 띄우기 위한 허스키 메시지
|
|
* 얼럿 레이어는 UI 기능버튼이 동작하지 않도록 에디터 전체를 투명 커버로 씌우고
|
|
* 에디터 전체를 기준으로 중앙에 위치한다. (편집영역은 딤드처리)
|
|
* 취소콜백함수(fCancelCallback)를 등록여부에 따라 "취소"버튼을 보여줄지 여부를 결정하기 때문에 alert/confirm 두가지 형태의 레이어를 모두 표현해줄 수 있다.
|
|
* (주의)
|
|
* 특정 UI가 동작한 상태에서 띄워질 수 있기 때문에 DISABLE_ALL_UI / ENABLE_ALL_UI 메시지를 수행하지 않는다.
|
|
* 또한 같은 이유로 포커스 역시 편집영역으로 자동 포커싱하지 않기 때문에 얼럿 레이어가 닫힌 후 편집영역으로 포커스가 필요한 경우 callback 함수에서 실행되도록 해야한다.
|
|
*
|
|
* @param {String} sMsgHTML 얼럿 레이어에 보여줄 문구 (HTML형태로 표현가능하다)
|
|
* @param {HashTable} htOption 얼럿 레이어 옵션 객체
|
|
* @param {Function} htOption.fOkCallback "확인" 버튼 클릭시 콜백함수
|
|
* @param {Function} htOption.fCloseCallback "X" 버튼 클릭시 콜백함수
|
|
* @param {Function} htOption.fCancelCallback "취소" 버튼 클릭시 콜백함수, 등록되지 않으면 "취소"버튼이 나오지 않는다.
|
|
*/
|
|
$ON_ALERT : function(sMsgHTML, htOption){
|
|
if(this._elAlertLayer){
|
|
htOption = htOption || {};
|
|
this._elAlertTxts.innerHTML = sMsgHTML || "";
|
|
|
|
// 각 버튼에 클릭이벤트 핸들러를 달아준다.
|
|
this._elAlertOk.onclick = jindo.$Fn(this._hideAlert, this).bind(htOption.fOkCallback);
|
|
this._elAlertClose.onclick = jindo.$Fn(this._hideAlert, this).bind(htOption.fCloseCallback);
|
|
// 취소 콜백함수가 없는 경우는 버튼을 보여주지 않는다.
|
|
if(htOption.fCancelCallback){
|
|
this._elAlertCancel.onclick = jindo.$Fn(this._hideAlert, this).bind(htOption.fCancelCallback);
|
|
this._elAlertCancel.style.display = "";
|
|
}else{
|
|
this._elAlertCancel.style.display = "none";
|
|
}
|
|
|
|
this.oApp.exec("SHOW_EDITING_AREA_COVER", [true]);
|
|
this._elAlertLayer.style.zIndex = 100;
|
|
this._elAlertLayer.style.display = "block";
|
|
this._elAlertOk.focus();
|
|
}
|
|
},
|
|
|
|
_assignHTMLElements : function(oAppContainer){
|
|
oAppContainer = jindo.$(oAppContainer) || document;
|
|
this.rxUI = new RegExp(this.sUIClassPrefix+"([^ ]+)");
|
|
|
|
this.toolbarArea = jindo.$$.getSingle(".se2_tool", oAppContainer);
|
|
this.aAllUI = jindo.$$("[class*=" + this.sUIClassPrefix + "]", this.toolbarArea);
|
|
this.elTextTool = jindo.$$.getSingle("div.husky_seditor_text_tool", this.toolbarArea); // [SMARTEDITORSUS-1124] 텍스트 툴바 버튼의 라운드 처리
|
|
|
|
// alert 레이어 할당
|
|
this._elAlertLayer = jindo.$$.getSingle(".se2_alert_wrap", oAppContainer);
|
|
if(this._elAlertLayer){
|
|
this._elAlertTxts = jindo.$$.getSingle(".se2_alert_txts", this._elAlertLayer);
|
|
this._elAlertOk = jindo.$$.getSingle(".se2_confirm", this._elAlertLayer);
|
|
this._elAlertCancel = jindo.$$.getSingle(".se2_cancel", this._elAlertLayer);
|
|
this._elAlertClose = jindo.$$.getSingle(".btn_close", this._elAlertLayer);
|
|
}
|
|
|
|
this.welToolbarArea = jindo.$Element(this.toolbarArea);
|
|
for (var i = 0, nCount = this.aAllUI.length; i < nCount; i++) {
|
|
if (this.rxUI.test(this.aAllUI[i].className)) {
|
|
var sUIName = RegExp.$1;
|
|
if(this.htUIList[sUIName] !== undefined){
|
|
continue;
|
|
}
|
|
|
|
this.htUIList[sUIName] = this.aAllUI[i];
|
|
this.htWrappedUIList[sUIName] = jindo.$Element(this.htUIList[sUIName]);
|
|
}
|
|
}
|
|
|
|
if (jindo.$$.getSingle("DIV.se2_icon_tool") != null) {
|
|
this.elFirstToolbarItem = jindo.$$.getSingle("DIV.se2_icon_tool UL.se2_itool1>li>button");
|
|
}
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(sMsg) {
|
|
var aToolItems = jindo.$$(">ul>li[class*=" + this.sUIClassPrefix + "]>button", this.elTextTool);
|
|
var nItemLength = aToolItems.length;
|
|
|
|
this.elFirstToolbarItem = this.elFirstToolbarItem || aToolItems[0];
|
|
this.elLastToolbarItem = aToolItems[nItemLength-1];
|
|
|
|
this.oApp.registerBrowserEvent(this.toolbarArea, "keydown", "NAVIGATE_TOOLBAR", []);
|
|
},
|
|
|
|
/**
|
|
* @param {Element} oAppContainer
|
|
* @param {Object} htOptions
|
|
* @param {Array} htOptions.aDisabled 비활성화할 버튼명 배열
|
|
*/
|
|
$init : function(oAppContainer, htOptions){
|
|
this._htOptions = htOptions || {};
|
|
this.htUIList = {};
|
|
this.htWrappedUIList = {};
|
|
|
|
this.aUICmdMap = {};
|
|
this._assignHTMLElements(oAppContainer);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
if(this.oApp.bMobile){
|
|
this.oApp.registerBrowserEvent(this.toolbarArea, "touchstart", "EVENT_TOOLBAR_TOUCHSTART");
|
|
}else{
|
|
this.oApp.registerBrowserEvent(this.toolbarArea, "mouseover", "EVENT_TOOLBAR_MOUSEOVER");
|
|
this.oApp.registerBrowserEvent(this.toolbarArea, "mouseout", "EVENT_TOOLBAR_MOUSEOUT");
|
|
}
|
|
this.oApp.registerBrowserEvent(this.toolbarArea, "mousedown", "EVENT_TOOLBAR_MOUSEDOWN");
|
|
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getToolbarButtonByUIName", jindo.$Fn(this.getToolbarButtonByUIName, this).bind()]);
|
|
|
|
// [SMARTEDITORSUS-1679] 초기 disabled 처리가 필요한 버튼은 비활성화
|
|
if(this._htOptions.aDisabled){
|
|
this._htOptions._sDisabled = "," + this._htOptions.aDisabled.toString() + ","; // 버튼을 활성화할때 비교하기 위한 문자열구성
|
|
this.oApp.exec("DISABLE_UI", [this._htOptions.aDisabled]);
|
|
}
|
|
},
|
|
|
|
|
|
$ON_NAVIGATE_TOOLBAR : function(weEvent) {
|
|
|
|
var TAB_KEY_CODE = 9;
|
|
//이벤트가 발생한 엘리먼트가 마지막 아이템이고 TAB 키가 눌려졌다면
|
|
if ((weEvent.element == this.elLastToolbarItem) && (weEvent.key().keyCode == TAB_KEY_CODE) ) {
|
|
|
|
|
|
if (weEvent.key().shift) {
|
|
//do nothing
|
|
} else {
|
|
this.elFirstToolbarItem.focus();
|
|
weEvent.stopDefault();
|
|
}
|
|
}
|
|
|
|
|
|
//이벤트가 발생한 엘리먼트가 첫번째 아이템이고 TAB 키가 눌려졌다면
|
|
if (weEvent.element == this.elFirstToolbarItem && (weEvent.key().keyCode == TAB_KEY_CODE)) {
|
|
if (weEvent.key().shift) {
|
|
weEvent.stopDefault();
|
|
this.elLastToolbarItem.focus();
|
|
}
|
|
}
|
|
},
|
|
|
|
$ON_TOGGLE_TOOLBAR_ACTIVE_LAYER : function(elLayer, elBtn, sOpenCmd, aOpenArgs, sCloseCmd, aCloseArgs){
|
|
this.oApp.exec("TOGGLE_ACTIVE_LAYER", [elLayer, "MSG_TOOLBAR_LAYER_SHOWN", [elLayer, elBtn, sOpenCmd, aOpenArgs], sCloseCmd, aCloseArgs]);
|
|
},
|
|
|
|
$ON_MSG_TOOLBAR_LAYER_SHOWN : function(elLayer, elBtn, aOpenCmd, aOpenArgs){
|
|
this.oApp.exec("POSITION_TOOLBAR_LAYER", [elLayer, elBtn]);
|
|
if(aOpenCmd){
|
|
this.oApp.exec(aOpenCmd, aOpenArgs);
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_TOOLBAR_ACTIVE_LAYER : function(elLayer, sCmd, aArgs, elBtn){
|
|
this.oApp.exec("SHOW_ACTIVE_LAYER", [elLayer, sCmd, aArgs]);
|
|
this.oApp.exec("POSITION_TOOLBAR_LAYER", [elLayer, elBtn]);
|
|
},
|
|
|
|
$ON_ENABLE_UI : function(sUIName){
|
|
this._enableUI(sUIName);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1679] 여러개의 버튼을 동시에 비활성화 할 수 있도록 수정
|
|
* @param {String|Array} vUIName 비활성화할 버튼명, 배열일 경우 여러개 동시 적용
|
|
*/
|
|
$ON_DISABLE_UI : function(sUIName){
|
|
if(sUIName instanceof Array){
|
|
for(var i = 0, sName; (sName = sUIName[i]); i++){
|
|
this._disableUI(sName);
|
|
}
|
|
}else{
|
|
this._disableUI(sUIName);
|
|
}
|
|
},
|
|
|
|
$ON_SELECT_UI : function(sUIName){
|
|
var welUI = this.htWrappedUIList[sUIName];
|
|
if(!welUI){
|
|
return;
|
|
}
|
|
welUI.removeClass("hover");
|
|
welUI.addClass("active");
|
|
},
|
|
|
|
$ON_DESELECT_UI : function(sUIName){
|
|
var welUI = this.htWrappedUIList[sUIName];
|
|
if(!welUI){
|
|
return;
|
|
}
|
|
welUI.removeClass("active");
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1646] 툴바버튼 선택상태를 토글링한다.
|
|
* @param {String} sUIName 토글링할 툴바버튼 이름
|
|
*/
|
|
$ON_TOGGLE_UI_SELECTED : function(sUIName){
|
|
var welUI = this.htWrappedUIList[sUIName];
|
|
if(!welUI){
|
|
return;
|
|
}
|
|
if(welUI.hasClass("active")){
|
|
welUI.removeClass("active");
|
|
}else{
|
|
welUI.removeClass("hover");
|
|
welUI.addClass("active");
|
|
}
|
|
},
|
|
|
|
$ON_ENABLE_ALL_UI : function(htOptions){
|
|
if(this.nUIStatus === 1){
|
|
return;
|
|
}
|
|
|
|
var sUIName, className;
|
|
htOptions = htOptions || {};
|
|
var waExceptions = jindo.$A(htOptions.aExceptions || []);
|
|
|
|
for(sUIName in this.htUIList){
|
|
if(sUIName && !waExceptions.has(sUIName)){
|
|
this._enableUI(sUIName);
|
|
}
|
|
// if(sUIName) this.oApp.exec("ENABLE_UI", [sUIName]);
|
|
}
|
|
// jindo.$Element(this.toolbarArea).removeClass("off");
|
|
|
|
this.nUIStatus = 1;
|
|
},
|
|
|
|
$ON_DISABLE_ALL_UI : function(htOptions){
|
|
if(this.nUIStatus === 2){
|
|
return;
|
|
}
|
|
|
|
var sUIName;
|
|
htOptions = htOptions || {};
|
|
var waExceptions = jindo.$A(htOptions.aExceptions || []);
|
|
var bLeavlActiveLayer = htOptions.bLeaveActiveLayer || false;
|
|
|
|
if(!bLeavlActiveLayer){
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
}
|
|
|
|
for(sUIName in this.htUIList){
|
|
if(sUIName && !waExceptions.has(sUIName)){
|
|
this._disableUI(sUIName);
|
|
}
|
|
// if(sUIName) this.oApp.exec("DISABLE_UI", [sUIName]);
|
|
}
|
|
// jindo.$Element(this.toolbarArea).addClass("off");
|
|
|
|
this.nUIStatus = 2;
|
|
},
|
|
|
|
$ON_MSG_STYLE_CHANGED : function(sAttributeName, attributeValue){
|
|
if(attributeValue === "@^"){
|
|
this.oApp.exec("SELECT_UI", [sAttributeName]);
|
|
}else{
|
|
this.oApp.exec("DESELECT_UI", [sAttributeName]);
|
|
}
|
|
},
|
|
|
|
$ON_POSITION_TOOLBAR_LAYER : function(elLayer, htOption){
|
|
var nLayerLeft, nLayerRight, nToolbarLeft, nToolbarRight;
|
|
|
|
elLayer = jindo.$(elLayer);
|
|
htOption = htOption || {};
|
|
var elBtn = jindo.$(htOption.elBtn);
|
|
var sAlign = htOption.sAlign;
|
|
|
|
var nMargin = -1;
|
|
if(!elLayer){
|
|
return;
|
|
}
|
|
if(elBtn && elBtn.tagName && elBtn.tagName == "BUTTON"){
|
|
elBtn.parentNode.appendChild(elLayer);
|
|
}
|
|
|
|
var welLayer = jindo.$Element(elLayer);
|
|
|
|
if(sAlign != "right"){
|
|
elLayer.style.left = "0";
|
|
|
|
nLayerLeft = welLayer.offset().left;
|
|
nLayerRight = nLayerLeft + elLayer.offsetWidth;
|
|
|
|
nToolbarLeft = this.welToolbarArea.offset().left;
|
|
nToolbarRight = nToolbarLeft + this.toolbarArea.offsetWidth;
|
|
|
|
if(nLayerRight > nToolbarRight){
|
|
welLayer.css("left", (nToolbarRight-nLayerRight-nMargin)+"px");
|
|
}
|
|
|
|
if(nLayerLeft < nToolbarLeft){
|
|
welLayer.css("left", (nToolbarLeft-nLayerLeft+nMargin)+"px");
|
|
}
|
|
}else{
|
|
elLayer.style.right = "0";
|
|
|
|
nLayerLeft = welLayer.offset().left;
|
|
nLayerRight = nLayerLeft + elLayer.offsetWidth;
|
|
|
|
nToolbarLeft = this.welToolbarArea.offset().left;
|
|
nToolbarRight = nToolbarLeft + this.toolbarArea.offsetWidth;
|
|
|
|
if(nLayerRight > nToolbarRight){
|
|
welLayer.css("right", -1*(nToolbarRight-nLayerRight-nMargin)+"px");
|
|
}
|
|
|
|
if(nLayerLeft < nToolbarLeft){
|
|
welLayer.css("right", -1*(nToolbarLeft-nLayerLeft+nMargin)+"px");
|
|
}
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_TOOLBAR_MOUSEOVER : function(weEvent){
|
|
if(this.nUIStatus === 2){
|
|
return;
|
|
}
|
|
|
|
var aAffectedElements = this._getAffectedElements(weEvent.element);
|
|
for(var i=0; i<aAffectedElements.length; i++){
|
|
if(!aAffectedElements[i].hasClass("active")){
|
|
aAffectedElements[i].addClass("hover");
|
|
}
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_TOOLBAR_MOUSEOUT : function(weEvent){
|
|
if(this.nUIStatus === 2){
|
|
return;
|
|
}
|
|
var aAffectedElements = this._getAffectedElements(weEvent.element);
|
|
for(var i=0; i<aAffectedElements.length; i++){
|
|
aAffectedElements[i].removeClass("hover");
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_TOOLBAR_MOUSEDOWN : function(weEvent){
|
|
var elTmp = weEvent.element;
|
|
// Check if the button pressed is in active status and has a visible layer i.e. the button had been clicked and its layer is open already. (buttons like font styles-bold, underline-got no sub layer -> childNodes.length<=2)
|
|
// -> In this case, do not close here(mousedown). The layer will be closed on "click". If we close the layer here, the click event will open it again because it toggles the visibility.
|
|
while(elTmp){
|
|
if(elTmp.className && elTmp.className.match(/active/) && (elTmp.childNodes.length>2 || elTmp.parentNode.className.match(/se2_pair/))){
|
|
return;
|
|
}
|
|
elTmp = elTmp.parentNode;
|
|
}
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER_IF_NOT_CHILD", [weEvent.element]);
|
|
},
|
|
|
|
_enableUI : function(sUIName){
|
|
// [SMARTEDITORSUS-1679] 초기 disabled 설정된 버튼은 skip
|
|
if(this._htOptions._sDisabled && this._htOptions._sDisabled.indexOf(","+sUIName+",") > -1){
|
|
return;
|
|
}
|
|
var i, nLen;
|
|
|
|
this.nUIStatus = 0;
|
|
|
|
var welUI = this.htWrappedUIList[sUIName];
|
|
var elUI = this.htUIList[sUIName];
|
|
if(!welUI){
|
|
return;
|
|
}
|
|
welUI.removeClass("off");
|
|
|
|
var aAllBtns = elUI.getElementsByTagName("BUTTON");
|
|
for(i=0, nLen=aAllBtns.length; i<nLen; i++){
|
|
aAllBtns[i].disabled = false;
|
|
}
|
|
|
|
// enable related commands
|
|
var sCmd = "";
|
|
if(this.aUICmdMap[sUIName]){
|
|
for(i=0; i<this.aUICmdMap[sUIName].length;i++){
|
|
sCmd = this.aUICmdMap[sUIName][i];
|
|
this.oApp.exec("ENABLE_MESSAGE", [sCmd]);
|
|
}
|
|
}
|
|
},
|
|
|
|
_disableUI : function(sUIName){
|
|
var i, nLen;
|
|
|
|
this.nUIStatus = 0;
|
|
|
|
var welUI = this.htWrappedUIList[sUIName];
|
|
var elUI = this.htUIList[sUIName];
|
|
if(!welUI){
|
|
return;
|
|
}
|
|
welUI.addClass("off");
|
|
welUI.removeClass("hover");
|
|
|
|
var aAllBtns = elUI.getElementsByTagName("BUTTON");
|
|
for(i=0, nLen=aAllBtns.length; i<nLen; i++){
|
|
aAllBtns[i].disabled = true;
|
|
}
|
|
|
|
// disable related commands
|
|
var sCmd = "";
|
|
if(this.aUICmdMap[sUIName]){
|
|
for(i=0; i<this.aUICmdMap[sUIName].length;i++){
|
|
sCmd = this.aUICmdMap[sUIName][i];
|
|
this.oApp.exec("DISABLE_MESSAGE", [sCmd]);
|
|
}
|
|
}
|
|
},
|
|
|
|
_getAffectedElements : function(el){
|
|
var elLi, welLi;
|
|
|
|
// 버튼 클릭시에 return false를 해 주지 않으면 chrome에서 버튼이 포커스 가져가 버림.
|
|
// 에디터 로딩 시에 일괄처리 할 경우 로딩 속도가 느려짐으로 hover시에 하나씩 처리
|
|
if(!el.bSE2_MDCancelled){
|
|
el.bSE2_MDCancelled = true;
|
|
var aBtns = el.getElementsByTagName("BUTTON");
|
|
|
|
for(var i=0, nLen=aBtns.length; i<nLen; i++){
|
|
aBtns[i].onmousedown = function(){return false;};
|
|
}
|
|
}
|
|
|
|
if(!el || !el.tagName){ return []; }
|
|
|
|
if((elLi = el).tagName == "BUTTON"){
|
|
// typical button
|
|
// <LI>
|
|
// <BUTTON>
|
|
if((elLi = elLi.parentNode) && elLi.tagName == "LI" && this.rxUI.test(elLi.className)){
|
|
return [jindo.$Element(elLi)];
|
|
}
|
|
|
|
// button pair
|
|
// <LI>
|
|
// <SPAN>
|
|
// <BUTTON>
|
|
// <SPAN>
|
|
// <BUTTON>
|
|
elLi = el;
|
|
if((elLi = elLi.parentNode.parentNode) && elLi.tagName == "LI" && (welLi = jindo.$Element(elLi)).hasClass("se2_pair")){
|
|
return [welLi, jindo.$Element(el.parentNode)];
|
|
}
|
|
|
|
return [];
|
|
}
|
|
|
|
// span in a button
|
|
if((elLi = el).tagName == "SPAN"){
|
|
// <LI>
|
|
// <BUTTON>
|
|
// <SPAN>
|
|
if((elLi = elLi.parentNode.parentNode) && elLi.tagName == "LI" && this.rxUI.test(elLi.className)){
|
|
return [jindo.$Element(elLi)];
|
|
}
|
|
|
|
// <LI>
|
|
// <SPAN>
|
|
//글감과 글양식
|
|
if((elLi = elLi.parentNode) && elLi.tagName == "LI" && this.rxUI.test(elLi.className)){
|
|
return [jindo.$Element(elLi)];
|
|
}
|
|
}
|
|
|
|
return [];
|
|
},
|
|
|
|
$ON_REGISTER_UI_EVENT : function(sUIName, sEvent, sCmd, aParams){
|
|
//[SMARTEDITORSUS-966][IE8 표준/IE 10] 호환 모드를 제거하고 사진 첨부 시 에디팅 영역의
|
|
// 커서 주위에 <sub><sup> 태그가 붙어서 글자가 매우 작게 되는 현상
|
|
//원인 : 아래의 [SMARTEDITORSUS-901] 수정 내용에서 윗첨자 아랫첨자 이벤트 등록 시
|
|
//해당 플러그인이 마크업에 없으면 this.htUIList에 존재하지 않아 getsingle 사용시 사진첨부에 이벤트가 걸렸음
|
|
//해결 : this.htUIList에 존재하지 않으면 이벤트를 등록하지 않음
|
|
if(!this.htUIList[sUIName]){
|
|
return;
|
|
}
|
|
// map cmd & ui
|
|
var elButton;
|
|
if(!this.aUICmdMap[sUIName]){this.aUICmdMap[sUIName] = [];}
|
|
this.aUICmdMap[sUIName][this.aUICmdMap[sUIName].length] = sCmd;
|
|
//[SMARTEDITORSUS-901]플러그인 태그 코드 추가 시 <li>태그와<button>태그 사이에 개행이 있으면 이벤트가 등록되지 않는 현상
|
|
//원인 : IE9, Chrome, FF, Safari 에서는 태그를 개행 시 그 개행을 text node로 인식하여 firstchild가 text 노드가 되어 버튼 이벤트가 할당되지 않음
|
|
//해결 : firstchild에 이벤트를 거는 것이 아니라, child 중 button 인 것에 이벤트를 걸도록 변경
|
|
elButton = jindo.$$.getSingle('button', this.htUIList[sUIName]);
|
|
|
|
if(!elButton){return;}
|
|
this.oApp.registerBrowserEvent(elButton, sEvent, sCmd, aParams);
|
|
},
|
|
|
|
getToolbarButtonByUIName : function(sUIName){
|
|
return jindo.$$.getSingle("BUTTON", this.htUIList[sUIName]);
|
|
}
|
|
});
|
|
//}
|
|
/*[
|
|
* LOAD_CONTENTS_FIELD
|
|
*
|
|
* 에디터 초기화 시에 넘어온 Contents(DB 저장 값)필드를 읽어 에디터에 설정한다.
|
|
*
|
|
* bDontAddUndo boolean Contents를 설정하면서 UNDO 히스토리는 추가 하지않는다.
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* UPDATE_IR_FIELD
|
|
*
|
|
* 에디터의 IR값을 IR필드에 설정 한다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* CHANGE_EDITING_MODE
|
|
*
|
|
* 에디터의 편집 모드를 변경한다.
|
|
*
|
|
* sMode string 전환 할 모드명
|
|
* bNoFocus boolean 모드 전환 후에 에디터에 포커스를 강제로 할당하지 않는다.
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* FOCUS
|
|
*
|
|
* 에디터 편집 영역에 포커스를 준다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* SET_IR
|
|
*
|
|
* IR값을 에디터에 설정 한다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* REGISTER_EDITING_AREA
|
|
*
|
|
* 편집 영역을 플러그인을 등록 시킨다. 원활한 모드 전환과 IR값 공유등를 위해서 초기화 시에 등록이 필요하다.
|
|
*
|
|
* oEditingAreaPlugin object 편집 영역 플러그인 인스턴스
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* MSG_EDITING_AREA_RESIZE_STARTED
|
|
*
|
|
* 편집 영역 사이즈 조절이 시작 되었음을 알리는 메시지.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* RESIZE_EDITING_AREA
|
|
*
|
|
* 편집 영역 사이즈를 설정 한다. 변경 전후에 MSG_EDITIING_AREA_RESIZE_STARTED/MSG_EDITING_AREA_RESIZE_ENED를 발생 시켜 줘야 된다.
|
|
*
|
|
* ipNewWidth number 새 폭
|
|
* ipNewHeight number 새 높이
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* RESIZE_EDITING_AREA_BY
|
|
*
|
|
* 편집 영역 사이즈를 늘리거나 줄인다. 변경 전후에 MSG_EDITIING_AREA_RESIZE_STARTED/MSG_EDITING_AREA_RESIZE_ENED를 발생 시켜 줘야 된다.
|
|
* 변경치를 입력하면 원래 사이즈에서 변경하여 px로 적용하며, width가 %로 설정된 경우에는 폭 변경치가 입력되어도 적용되지 않는다.
|
|
*
|
|
* ipWidthChange number 폭 변경치
|
|
* ipHeightChange number 높이 변경치
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* MSG_EDITING_AREA_RESIZE_ENDED
|
|
*
|
|
* 편집 영역 사이즈 조절이 끝났음을 알리는 메시지.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc IR 값과 복수개의 편집 영역을 관리하는 플러그인
|
|
*/
|
|
nhn.husky.SE_EditingAreaManager = jindo.$Class({
|
|
name : "SE_EditingAreaManager",
|
|
|
|
// Currently active plugin instance(SE_EditingArea_???)
|
|
oActivePlugin : null,
|
|
|
|
// Intermediate Representation of the content being edited.
|
|
// This should be a textarea element.
|
|
elContentsField : null,
|
|
|
|
bIsDirty : false,
|
|
bAutoResize : false, // [SMARTEDITORSUS-677] 에디터의 자동확장 기능 On/Off 여부
|
|
|
|
$init : function(sDefaultEditingMode, elContentsField, oDimension, fOnBeforeUnload, elAppContainer){
|
|
this.sDefaultEditingMode = sDefaultEditingMode;
|
|
this.elContentsField = jindo.$(elContentsField);
|
|
this._assignHTMLElements(elAppContainer);
|
|
this.fOnBeforeUnload = fOnBeforeUnload;
|
|
|
|
this.oEditingMode = {};
|
|
|
|
this.elContentsField.style.display = "none";
|
|
|
|
this.nMinWidth = parseInt((oDimension.nMinWidth || 60), 10);
|
|
this.nMinHeight = parseInt((oDimension.nMinHeight || 60), 10);
|
|
|
|
var oWidth = this._getSize([oDimension.nWidth, oDimension.width, this.elEditingAreaContainer.offsetWidth], this.nMinWidth);
|
|
var oHeight = this._getSize([oDimension.nHeight, oDimension.height, this.elEditingAreaContainer.offsetHeight], this.nMinHeight);
|
|
|
|
this.elEditingAreaContainer.style.width = oWidth.nSize + oWidth.sUnit;
|
|
this.elEditingAreaContainer.style.height = oHeight.nSize + oHeight.sUnit;
|
|
|
|
if(oWidth.sUnit === "px"){
|
|
elAppContainer.style.width = (oWidth.nSize + 2) + "px";
|
|
}else if(oWidth.sUnit === "%"){
|
|
elAppContainer.style.minWidth = this.nMinWidth + "px";
|
|
}
|
|
},
|
|
|
|
_getSize : function(aSize, nMin){
|
|
var i, nLen, aRxResult, nSize, sUnit, sDefaultUnit = "px";
|
|
|
|
nMin = parseInt(nMin, 10);
|
|
|
|
for(i=0, nLen=aSize.length; i<nLen; i++){
|
|
if(!aSize[i]){
|
|
continue;
|
|
}
|
|
|
|
if(!isNaN(aSize[i])){
|
|
nSize = parseInt(aSize[i], 10);
|
|
sUnit = sDefaultUnit;
|
|
break;
|
|
}
|
|
|
|
aRxResult = /([0-9]+)(.*)/i.exec(aSize[i]);
|
|
|
|
if(!aRxResult || aRxResult.length < 2 || aRxResult[1] <= 0){
|
|
continue;
|
|
}
|
|
|
|
nSize = parseInt(aRxResult[1], 10);
|
|
sUnit = aRxResult[2];
|
|
|
|
if(!sUnit){
|
|
sUnit = sDefaultUnit;
|
|
}
|
|
|
|
if(nSize < nMin && sUnit === sDefaultUnit){
|
|
nSize = nMin;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
if(!sUnit){
|
|
sUnit = sDefaultUnit;
|
|
}
|
|
|
|
if(isNaN(nSize) || (nSize < nMin && sUnit === sDefaultUnit)){
|
|
nSize = nMin;
|
|
}
|
|
|
|
return {nSize : nSize, sUnit : sUnit};
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
this.elEditingAreaContainer = jindo.$$.getSingle("DIV.husky_seditor_editing_area_container", elAppContainer);
|
|
|
|
// [SMARTEDITORSUS-2036] 로딩 레이어는 하위 호환을 위해 마크업이 존재하는 경우만 동작하도록 한다.
|
|
var f = function(){};
|
|
this.elLoadingLayer = jindo.$$.getSingle(".se2_content_loading", elAppContainer);
|
|
if(!this.elLoadingLayer){
|
|
this.$ON_SHOW_LOADING_LAYER = f;
|
|
this.$ON_HIDE_LOADING_LAYER = f;
|
|
}
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(msg){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["version", nhn.husky.SE_EditingAreaManager.version]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["elEditingAreaContainer", this.elEditingAreaContainer]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["welEditingAreaContainer", jindo.$Element(this.elEditingAreaContainer)]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getEditingAreaHeight", jindo.$Fn(this.getEditingAreaHeight, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getEditingAreaWidth", jindo.$Fn(this.getEditingAreaWidth, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getRawContents", jindo.$Fn(this.getRawContents, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getContents", jindo.$Fn(this.getContents, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getIR", jindo.$Fn(this.getIR, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["setContents", this.setContents]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["setIR", this.setIR]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getEditingMode", jindo.$Fn(this.getEditingMode, this).bind()]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.htOptions = this.oApp.htOptions[this.name] || {};
|
|
this.sDefaultEditingMode = this.htOptions["sDefaultEditingMode"] || this.sDefaultEditingMode;
|
|
this.iframeWindow = this.oApp.getWYSIWYGWindow();
|
|
this.oApp.exec("REGISTER_CONVERTERS", []);
|
|
this.oApp.exec("CHANGE_EDITING_MODE", [this.sDefaultEditingMode, true]);
|
|
this.oApp.exec("LOAD_CONTENTS_FIELD", [false]);
|
|
|
|
// [SMARTEDITORSUS-2089] fOnBeforeUnload === false 인 경우, 아예 window.onbeforeunload 를 등록하지 않도록 수정
|
|
if(this.fOnBeforeUnload !== false){
|
|
if(!!this.fOnBeforeUnload){
|
|
window.onbeforeunload = this.fOnBeforeUnload;
|
|
}else{
|
|
window.onbeforeunload = jindo.$Fn(function(){
|
|
// [SMARTEDITORSUS-1028][SMARTEDITORSUS-1517] QuickEditor 설정 API 개선으로, submit 이후 발생하게 되는 beforeunload 이벤트 핸들링 제거
|
|
//this.oApp.exec("MSG_BEFOREUNLOAD_FIRED");
|
|
// --// [SMARTEDITORSUS-1028][SMARTEDITORSUS-1517]
|
|
//if(this.getContents() != this.elContentsField.value || this.bIsDirty){
|
|
if(this.getRawContents() != this.sCurrentRawContents || this.bIsDirty){
|
|
return this.oApp.$MSG("SE_EditingAreaManager.onExit");
|
|
}
|
|
}, this).bind();
|
|
}
|
|
}
|
|
},
|
|
|
|
$AFTER_MSG_APP_READY : function(){
|
|
this.oApp.exec("UPDATE_RAW_CONTENTS");
|
|
|
|
if(!!this.oApp.htOptions[this.name] && this.oApp.htOptions[this.name].bAutoResize){
|
|
this.bAutoResize = this.oApp.htOptions[this.name].bAutoResize;
|
|
}
|
|
// [SMARTEDITORSUS-941] 아이패드에서는 자동확장기능이 항상 켜져있도록 한다.
|
|
if(this.oApp.oNavigator.msafari){
|
|
this.bAutoResize = true;
|
|
}
|
|
|
|
this.startAutoResize(); // [SMARTEDITORSUS-677] 편집영역 자동 확장 옵션이 TRUE이면 자동확장 시작
|
|
},
|
|
|
|
$ON_LOAD_CONTENTS_FIELD : function(bDontAddUndo){
|
|
var sContentsFieldValue = this.elContentsField.value;
|
|
|
|
// [SMARTEDITORSUS-177] [IE9] 글 쓰기, 수정 시에 elContentsField 에 들어간 공백을 제거
|
|
// [SMARTEDITORSUS-312] [FF4] 인용구 첫번째,두번째 디자인 1회 선택 시 에디터에 적용되지 않음
|
|
sContentsFieldValue = sContentsFieldValue.replace(/^\s+/, "");
|
|
|
|
this.oApp.exec("SET_CONTENTS", [sContentsFieldValue, bDontAddUndo]);
|
|
},
|
|
|
|
// 현재 contents를 form의 textarea에 세팅 해 줌.
|
|
// form submit 전에 이 부분을 실행시켜야 됨.
|
|
$ON_UPDATE_CONTENTS_FIELD : function(){
|
|
//this.oIRField.value = this.oApp.getIR();
|
|
this.elContentsField.value = this.oApp.getContents();
|
|
this.oApp.exec("UPDATE_RAW_CONTENTS");
|
|
//this.sCurrentRawContents = this.elContentsField.value;
|
|
},
|
|
|
|
// 에디터의 현재 상태를 기억해 둠. 페이지를 떠날 때 이 값이 변경 됐는지 확인 해서 내용이 변경 됐다는 경고창을 띄움
|
|
// RawContents 대신 contents를 이용해도 되지만, contents 획득을 위해서는 변환기를 실행해야 되기 때문에 RawContents 이용
|
|
$ON_UPDATE_RAW_CONTENTS : function(){
|
|
this.sCurrentRawContents = this.oApp.getRawContents();
|
|
},
|
|
|
|
$BEFORE_CHANGE_EDITING_MODE : function(sMode){
|
|
if(!this.oEditingMode[sMode]){
|
|
return false;
|
|
}
|
|
|
|
this.stopAutoResize(); // [SMARTEDITORSUS-677] 해당 편집 모드에서의 자동확장을 중지함
|
|
|
|
this._oPrevActivePlugin = this.oActivePlugin;
|
|
this.oActivePlugin = this.oEditingMode[sMode];
|
|
},
|
|
|
|
$AFTER_CHANGE_EDITING_MODE : function(sMode, bNoFocus){
|
|
if(this._oPrevActivePlugin){
|
|
var sIR = this._oPrevActivePlugin.getIR();
|
|
this.oApp.exec("SET_IR", [sIR]);
|
|
|
|
//this.oApp.exec("ENABLE_UI", [this._oPrevActivePlugin.sMode]);
|
|
|
|
this._setEditingAreaDimension();
|
|
}
|
|
//this.oApp.exec("DISABLE_UI", [this.oActivePlugin.sMode]);
|
|
|
|
this.startAutoResize(); // [SMARTEDITORSUS-677] 변경된 편집 모드에서의 자동확장을 시작
|
|
|
|
if(!bNoFocus){
|
|
this.oApp.delayedExec("FOCUS", [], 0);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 페이지를 떠날 때 alert을 표시할지 여부를 셋팅하는 함수.
|
|
*/
|
|
$ON_SET_IS_DIRTY : function(bIsDirty){
|
|
this.bIsDirty = bIsDirty;
|
|
},
|
|
|
|
// [SMARTEDITORSUS-1698] 모바일에서 팝업 형태의 첨부가 사용될 때 포커스 이슈가 있음
|
|
$ON_FOCUS : function(isPopupOpening){
|
|
if(!this.oActivePlugin || typeof this.oActivePlugin.setIR != "function"){
|
|
return;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-599] ipad 대응 이슈.
|
|
// ios5에서는 this.iframe.contentWindow focus가 없어서 생긴 이슈.
|
|
// document가 아닌 window에 focus() 주어야만 본문에 focus가 가고 입력이됨.
|
|
|
|
//[SMARTEDITORSUS-1017] [iOS5대응] 모드 전환 시 textarea에 포커스가 있어도 글자가 입력이 안되는 현상
|
|
//원인 : WYSIWYG모드가 아닐 때에도 iframe의 contentWindow에 focus가 가면서 focus기능이 작동하지 않음
|
|
//해결 : WYSIWYG모드 일때만 실행 되도록 조건식 추가 및 기존에 blur처리 코드 삭제
|
|
//[SMARTEDITORSUS-1594] 크롬에서 웹접근성용 키로 빠져나간 후 다시 진입시 간혹 포커싱이 안되는 문제가 있어 iframe에 포커싱을 먼저 주도록 수정
|
|
if(!!this.iframeWindow && this.iframeWindow.document.hasFocus && !this.iframeWindow.document.hasFocus() && this.oActivePlugin.sMode == "WYSIWYG"){
|
|
this.iframeWindow.focus();
|
|
}else{ // 누락된 [SMARTEDITORSUS-1018] 작업분 반영
|
|
this.oActivePlugin.focus();
|
|
}
|
|
|
|
if(isPopupOpening && this.oApp.bMobile){
|
|
return;
|
|
}
|
|
|
|
this.oActivePlugin.focus();
|
|
},
|
|
// --[SMARTEDITORSUS-1698]
|
|
|
|
$ON_IE_FOCUS : function(){
|
|
var oAgent = this.oApp.oNavigator;
|
|
if(!oAgent.ie && !oAgent.edge){ // [SMARTEDITORSUS-2257] edge도 ie와 동일하게 포커스처리해준다.
|
|
return;
|
|
}
|
|
this.oApp.exec("FOCUS");
|
|
},
|
|
|
|
$ON_SET_CONTENTS : function(sContents, bDontAddUndoHistory){
|
|
this.setContents(sContents, bDontAddUndoHistory);
|
|
},
|
|
|
|
$BEFORE_SET_IR : function(sIR, bDontAddUndoHistory){
|
|
bDontAddUndoHistory = bDontAddUndoHistory || false;
|
|
if(!bDontAddUndoHistory){
|
|
this.oApp.exec("RECORD_UNDO_ACTION", ["BEFORE SET CONTENTS", {sSaveTarget:"BODY"}]);
|
|
}
|
|
},
|
|
|
|
$ON_SET_IR : function(sIR){
|
|
if(!this.oActivePlugin || typeof this.oActivePlugin.setIR != "function"){
|
|
return;
|
|
}
|
|
|
|
this.oActivePlugin.setIR(sIR);
|
|
},
|
|
|
|
$AFTER_SET_IR : function(sIR, bDontAddUndoHistory){
|
|
bDontAddUndoHistory = bDontAddUndoHistory || false;
|
|
if(!bDontAddUndoHistory){
|
|
this.oApp.exec("RECORD_UNDO_ACTION", ["AFTER SET CONTENTS", {sSaveTarget:"BODY"}]);
|
|
}
|
|
},
|
|
|
|
$ON_REGISTER_EDITING_AREA : function(oEditingAreaPlugin){
|
|
this.oEditingMode[oEditingAreaPlugin.sMode] = oEditingAreaPlugin;
|
|
if(oEditingAreaPlugin.sMode == 'WYSIWYG'){
|
|
this.attachDocumentEvents(oEditingAreaPlugin.oEditingArea);
|
|
}
|
|
this._setEditingAreaDimension(oEditingAreaPlugin);
|
|
},
|
|
|
|
$ON_MSG_EDITING_AREA_RESIZE_STARTED : function(){
|
|
this._fitElementInEditingArea(this.elEditingAreaContainer);
|
|
this.oApp.exec("STOP_AUTORESIZE_EDITING_AREA"); // [SMARTEDITORSUS-677] 사용자가 편집영역 사이즈를 변경하면 자동확장 기능 중지
|
|
this.oApp.exec("SHOW_EDITING_AREA_COVER");
|
|
this.elEditingAreaContainer.style.overflow = "hidden";
|
|
|
|
this.iStartingHeight = parseInt(this.elEditingAreaContainer.style.height, 10);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] 편집영역 자동확장 기능을 중지함
|
|
*/
|
|
$ON_STOP_AUTORESIZE_EDITING_AREA : function(){
|
|
if(!this.bAutoResize){
|
|
return;
|
|
}
|
|
|
|
this.stopAutoResize();
|
|
this.bAutoResize = false;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] 해당 편집 모드에서의 자동확장을 시작함
|
|
*/
|
|
startAutoResize : function(){
|
|
if(!this.bAutoResize || !this.oActivePlugin || typeof this.oActivePlugin.startAutoResize != "function"){
|
|
return;
|
|
}
|
|
|
|
this.oActivePlugin.startAutoResize();
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] 해당 편집 모드에서의 자동확장을 중지함
|
|
*/
|
|
stopAutoResize : function(){
|
|
if(!this.bAutoResize || !this.oActivePlugin || typeof this.oActivePlugin.stopAutoResize != "function"){
|
|
return;
|
|
}
|
|
|
|
this.oActivePlugin.stopAutoResize();
|
|
},
|
|
|
|
$ON_RESIZE_EDITING_AREA: function(ipNewWidth, ipNewHeight){
|
|
if(ipNewWidth !== null && typeof ipNewWidth !== "undefined"){
|
|
this._resizeWidth(ipNewWidth, "px");
|
|
}
|
|
if(ipNewHeight !== null && typeof ipNewHeight !== "undefined"){
|
|
this._resizeHeight(ipNewHeight, "px");
|
|
}
|
|
|
|
this._setEditingAreaDimension();
|
|
},
|
|
|
|
_resizeWidth : function(ipNewWidth, sUnit){
|
|
var iNewWidth = parseInt(ipNewWidth, 10);
|
|
|
|
if(iNewWidth < this.nMinWidth){
|
|
iNewWidth = this.nMinWidth;
|
|
}
|
|
|
|
if(ipNewWidth){
|
|
this.elEditingAreaContainer.style.width = iNewWidth + sUnit;
|
|
}
|
|
},
|
|
|
|
_resizeHeight : function(ipNewHeight, sUnit){
|
|
var iNewHeight = parseInt(ipNewHeight, 10);
|
|
|
|
if(iNewHeight < this.nMinHeight){
|
|
iNewHeight = this.nMinHeight;
|
|
}
|
|
|
|
if(ipNewHeight){
|
|
this.elEditingAreaContainer.style.height = iNewHeight + sUnit;
|
|
}
|
|
},
|
|
|
|
$ON_RESIZE_EDITING_AREA_BY : function(ipWidthChange, ipHeightChange){
|
|
var iWidthChange = parseInt(ipWidthChange, 10);
|
|
var iHeightChange = parseInt(ipHeightChange, 10);
|
|
var iWidth;
|
|
var iHeight;
|
|
|
|
if(ipWidthChange !== 0 && this.elEditingAreaContainer.style.width.indexOf("%") === -1){
|
|
iWidth = this.elEditingAreaContainer.style.width?parseInt(this.elEditingAreaContainer.style.width, 10)+iWidthChange:null;
|
|
}
|
|
|
|
if(iHeightChange !== 0){
|
|
iHeight = this.elEditingAreaContainer.style.height?this.iStartingHeight+iHeightChange:null;
|
|
}
|
|
|
|
if(!ipWidthChange && !iHeightChange){
|
|
return;
|
|
}
|
|
|
|
this.oApp.exec("RESIZE_EDITING_AREA", [iWidth, iHeight]);
|
|
},
|
|
|
|
$ON_MSG_EDITING_AREA_RESIZE_ENDED : function(FnMouseDown, FnMouseMove, FnMouseUp){
|
|
this.oApp.exec("HIDE_EDITING_AREA_COVER");
|
|
this.elEditingAreaContainer.style.overflow = "";
|
|
this._setEditingAreaDimension();
|
|
},
|
|
|
|
/**
|
|
* 편집영역에 커서가 들어가지 않도록 투명 커버를 씌운다.
|
|
* @param {Boolean} bDimmed 반투명 처리할지 여부
|
|
*/
|
|
$ON_SHOW_EDITING_AREA_COVER : function(bDimmed){
|
|
if(!this.elEditingAreaCover){
|
|
this.elEditingAreaCover = document.createElement("DIV");
|
|
this.elEditingAreaCover.style.cssText = 'position:absolute;top:0;left:0;z-index:100;background:#000000;filter:alpha(opacity=0);opacity:0.0;-moz-opacity:0.0;-khtml-opacity:0.0;height:100%;width:100%';
|
|
this.elEditingAreaContainer.appendChild(this.elEditingAreaCover);
|
|
}
|
|
if(bDimmed){
|
|
jindo.$Element(this.elEditingAreaCover).opacity(0.4);
|
|
}
|
|
this.elEditingAreaCover.style.display = "block";
|
|
},
|
|
|
|
$ON_HIDE_EDITING_AREA_COVER : function(){
|
|
if(!this.elEditingAreaCover){
|
|
return;
|
|
}
|
|
this.elEditingAreaCover.style.display = "none";
|
|
jindo.$Element(this.elEditingAreaCover).opacity(0);
|
|
},
|
|
|
|
$ON_KEEP_WITHIN_EDITINGAREA : function(elLayer, nHeight){
|
|
var nTop = parseInt(elLayer.style.top, 10);
|
|
if(nTop + elLayer.offsetHeight > this.oApp.elEditingAreaContainer.offsetHeight){
|
|
if(typeof nHeight == "number"){
|
|
elLayer.style.top = nTop - elLayer.offsetHeight - nHeight + "px";
|
|
}else{
|
|
elLayer.style.top = this.oApp.elEditingAreaContainer.offsetHeight - elLayer.offsetHeight + "px";
|
|
}
|
|
}
|
|
|
|
var nLeft = parseInt(elLayer.style.left, 10);
|
|
if(nLeft + elLayer.offsetWidth > this.oApp.elEditingAreaContainer.offsetWidth){
|
|
elLayer.style.left = this.oApp.elEditingAreaContainer.offsetWidth - elLayer.offsetWidth + "px";
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_KEYDOWN : function(){
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_MOUSEDOWN : function(){
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_SCROLL : function(){
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
},
|
|
|
|
_setEditingAreaDimension : function(oEditingAreaPlugin){
|
|
oEditingAreaPlugin = oEditingAreaPlugin || this.oActivePlugin;
|
|
this._fitElementInEditingArea(oEditingAreaPlugin.elEditingArea);
|
|
},
|
|
|
|
_fitElementInEditingArea : function(el){
|
|
el.style.height = this.elEditingAreaContainer.offsetHeight+"px";
|
|
// el.style.width = this.elEditingAreaContainer.offsetWidth+"px";
|
|
// el.style.width = this.elEditingAreaContainer.style.width || (this.elEditingAreaContainer.offsetWidth+"px");
|
|
},
|
|
|
|
attachDocumentEvents : function(doc){
|
|
this.oApp.registerBrowserEvent(doc, "click", "EVENT_EDITING_AREA_CLICK");
|
|
this.oApp.registerBrowserEvent(doc, "dblclick", "EVENT_EDITING_AREA_DBLCLICK");
|
|
this.oApp.registerBrowserEvent(doc, "mousedown", "EVENT_EDITING_AREA_MOUSEDOWN");
|
|
this.oApp.registerBrowserEvent(doc, "mousemove", "EVENT_EDITING_AREA_MOUSEMOVE");
|
|
this.oApp.registerBrowserEvent(doc, "mouseup", "EVENT_EDITING_AREA_MOUSEUP");
|
|
this.oApp.registerBrowserEvent(doc, "mouseout", "EVENT_EDITING_AREA_MOUSEOUT");
|
|
this.oApp.registerBrowserEvent(doc, "mousewheel", "EVENT_EDITING_AREA_MOUSEWHEEL");
|
|
this.oApp.registerBrowserEvent(doc, "keydown", "EVENT_EDITING_AREA_KEYDOWN");
|
|
this.oApp.registerBrowserEvent(doc, "keypress", "EVENT_EDITING_AREA_KEYPRESS");
|
|
this.oApp.registerBrowserEvent(doc, "keyup", "EVENT_EDITING_AREA_KEYUP");
|
|
this.oApp.registerBrowserEvent(doc, "scroll", "EVENT_EDITING_AREA_SCROLL");
|
|
},
|
|
|
|
$ON_GET_COVER_DIV : function(sAttr,oReturn){
|
|
if(!!this.elEditingAreaCover) {
|
|
oReturn[sAttr] = this.elEditingAreaCover;
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_LOADING_LAYER : function(){
|
|
this.elLoadingLayer.style.display = 'block';
|
|
},
|
|
|
|
$ON_HIDE_LOADING_LAYER : function(){
|
|
this.elLoadingLayer.style.display = 'none';
|
|
},
|
|
|
|
getIR : function(){
|
|
if(!this.oActivePlugin){
|
|
return "";
|
|
}
|
|
return this.oActivePlugin.getIR();
|
|
},
|
|
|
|
setIR : function(sIR, bDontAddUndo){
|
|
this.oApp.exec("SET_IR", [sIR, bDontAddUndo]);
|
|
},
|
|
|
|
getRawContents : function(){
|
|
if(!this.oActivePlugin){
|
|
return "";
|
|
}
|
|
return this.oActivePlugin.getRawContents();
|
|
},
|
|
|
|
getContents : function(){
|
|
// [SMARTEDITORSUS-2077]
|
|
this._convertLastBrToNbsp();
|
|
|
|
var sIR = this.oApp.getIR();
|
|
var sContents;
|
|
|
|
if(this.oApp.applyConverter){
|
|
sContents = this.oApp.applyConverter("IR_TO_DB", sIR, this.oApp.getWYSIWYGDocument());
|
|
}else{
|
|
sContents = sIR;
|
|
}
|
|
|
|
sContents = this._cleanContents(sContents);
|
|
|
|
return sContents;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2077]
|
|
* 문단 내 마지막 <br>을 찾아서,
|
|
* 특수한 경우를 제외하고 모두 로 변환한다.
|
|
*
|
|
* -NBSP로 변환해야 할 필요성
|
|
* --[IE 문서모드 8~10] 레이아웃 상 문단 내부 맨 마지막 요소가 <br>인 경우,
|
|
* <br>이 두 줄의 높이를 가진 것처럼 표현되는 현상이 있음
|
|
*
|
|
* -NBSP로 변환하지 않고 <br>을 제거하는 경우
|
|
* --레이아웃 상 <br> 바로 앞에 <img>가 위치하고,
|
|
* 이 <img>가
|
|
* parentNode가 허용하는 최대 가로폭 이상의 가로 크기를 가지고 있어
|
|
* parentNode가 허용하는 최대 가로폭으로 리사이징된 경우
|
|
* ---<br>이 NBSP로 변환되면, <img> 아래줄로 밀려나 한 줄을 더 차지하게 됨
|
|
*
|
|
*/
|
|
_convertLastBrToNbsp : function(){
|
|
var elBody = this.oApp.getWYSIWYGDocument().body,
|
|
aBr, elBr,
|
|
elBrContainer, elImgContainer, elNextToImgContainer, elUndesiredContainer,
|
|
aImg, elImg, nImgWidth,
|
|
elImgParent, nParentWidth,
|
|
bConvertBrToNbsp,
|
|
oNbsp = document.createTextNode('\u00A0'), oNbspClone,
|
|
elBrParent;
|
|
|
|
// br:last-child 탐색
|
|
aBr = jindo.$$('br:last-child', elBody, {oneTimeOffCache : true});
|
|
for(var i = 0, len = aBr.length; i < len; i++){
|
|
elImg = null;
|
|
elBrParent = null;
|
|
elBr = aBr[i];
|
|
|
|
// <br>부터 부모로 거슬러 올라가면서, previousSibling이 있는지 찾는다.
|
|
elImgContainer = this._findNextSiblingRecursive(elBr, {isReverse : true});
|
|
if(!(elImgContainer && (elImgContainer.nodeType === 1))){
|
|
continue;
|
|
}
|
|
|
|
elBrContainer = elImgContainer.nextSibling;
|
|
|
|
/*
|
|
* <br> container에서 부모로 거슬러 올라가면서, nextSibling이 있는지 찾는다.
|
|
* nextSibling이 없다는 것이 확인되어야만,
|
|
* <br> container가 문단 맨 마지막에 있다는 것을
|
|
* 보장할 수 있기 때문이다.
|
|
*/
|
|
elUndesiredContainer = this._findNextSiblingRecursive(elBrContainer);
|
|
// <br> container 뒤의 container는 존재해서는 안 된다.
|
|
if(elUndesiredContainer){
|
|
continue;
|
|
}
|
|
|
|
if(elImgContainer.tagName.toUpperCase() === 'IMG'){
|
|
// previousSibling이 <img>이면 바로 할당
|
|
elImg = elImgContainer;
|
|
}else{
|
|
// previousSibling의 img:last-child 탐색
|
|
aImg = jindo.$$('img:last-child', elImgContainer, {oneTimeOffCache : true});
|
|
|
|
// 가장 마지막 <img>가 고려 대상이다. 즉, <br> 바로 앞에 위치하고 있을 것으로 추정되는 img
|
|
if(aImg.length > 0){
|
|
elImg = aImg[aImg.length - 1];
|
|
}
|
|
}
|
|
|
|
if(elImg){
|
|
/*
|
|
* <img> container 바로 뒤가 실제로 <br> container인지 확인.
|
|
* 레이아웃 상에서 실제로 <img> 뒤에 <br>이 붙어 있는지 확인하는 과정이다.
|
|
*/
|
|
elNextToImgContainer = this._findNextSiblingRecursive(elImg);
|
|
if(elNextToImgContainer == elBrContainer){
|
|
elImgParent = elImg.parentNode;
|
|
if(!elImgParent){
|
|
continue;
|
|
}
|
|
|
|
// <img>의 width 확인
|
|
nImgWidth = jindo.$Element(elImg).width(),
|
|
|
|
// <img> parentNode의 width 확인
|
|
elImgParent = elImg.parentNode,
|
|
nParentWidth = jindo.$Element(elImgParent).width();
|
|
|
|
/*
|
|
* <img>와 parentNode의 width가 같은지 확인하여
|
|
* <img> 뒤의 <br>을 로 변환할지, 아니면 제거할지 판단한다.
|
|
*/
|
|
bConvertBrToNbsp = !(nImgWidth === nParentWidth);
|
|
}else{
|
|
// <img> container와 <br> container 사이에 다른 container가 존재한다면, 로 변환
|
|
bConvertBrToNbsp = true;
|
|
}
|
|
}else{
|
|
// img:last-child 가 존재하지 않으면, <br>을 로 변환
|
|
bConvertBrToNbsp = true;
|
|
}
|
|
|
|
elBrParent = elBr.parentNode;
|
|
if(bConvertBrToNbsp){
|
|
// <br>을 로 변환
|
|
oNbspClone = oNbsp.cloneNode(false);
|
|
|
|
elBrParent.replaceChild(oNbspClone, elBr);
|
|
}else{
|
|
// <br>을 로 변환하지 않고 제거해 버림
|
|
this._recursiveRemoveChild(elBr);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 대상 element의 nextSibling이 있는지
|
|
* tree를 거슬러 올라가며 recursive 탐색
|
|
*
|
|
* @param {HTMLElement} el 탐색 시작점 element
|
|
* @param {Object} htOption
|
|
* @param {Boolean} htOption.isReverse nextSibling 이 아닌, previousSibling을 탐색
|
|
*/
|
|
_findNextSiblingRecursive : function(el, htOption){
|
|
var elTarget = el,
|
|
elSibling,
|
|
elParent,
|
|
isReverse = (htOption && htOption.isReverse) ? true : false,
|
|
rxParagraph = new RegExp('^(P|DIV)$', 'i'),
|
|
rxContainer = new RegExp('^(TD|BODY)$', 'i');
|
|
|
|
/*
|
|
* 종료 조건
|
|
* -container 역할을 하는 태그로 지정한 TD, BODY가 parentNode가 될 때까지 거슬러 올라감
|
|
* -sibling 존재
|
|
*/
|
|
while(!elSibling){
|
|
if(isReverse){
|
|
elSibling = elTarget.previousSibling;
|
|
}else{
|
|
elSibling = elTarget.nextSibling;
|
|
}
|
|
|
|
elTarget = elTarget.parentNode;
|
|
if(rxContainer.test(elTarget.tagName)){
|
|
elSibling = null;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return elSibling;
|
|
},
|
|
|
|
/**
|
|
* 대상 element를 parentNode에서 제거한다.
|
|
*
|
|
* 이후,
|
|
* parentNode 입장에서
|
|
* childNode가 아무것도 없으면
|
|
* 그 parentNode에서 이 parentNode를 제거하는 작업을
|
|
* recursive로 수행한다.
|
|
*/
|
|
_recursiveRemoveChild : function(el){
|
|
var elParent = el.parentNode,
|
|
elChild = el,
|
|
aChild;
|
|
|
|
do{
|
|
elParent.removeChild(elChild);
|
|
}while((aChild = elParent.childNodes) && (aChild.length == 0) && (elChild = elParent) && (elParent = elParent.parentNode))
|
|
},
|
|
|
|
_cleanContents : function(sContents){
|
|
return sContents.replace(new RegExp("(<img [^>]*>)"+unescape("%uFEFF")+"", "ig"), "$1");
|
|
},
|
|
|
|
setContents : function(sContents, bDontAddUndo){
|
|
var sIR;
|
|
|
|
if(this.oApp.applyConverter){
|
|
sIR = this.oApp.applyConverter("DB_TO_IR", sContents, this.oApp.getWYSIWYGDocument());
|
|
}else{
|
|
sIR = sContents;
|
|
}
|
|
|
|
this.oApp.exec("SET_IR", [sIR, bDontAddUndo]);
|
|
},
|
|
|
|
getEditingMode : function(){
|
|
return this.oActivePlugin.sMode;
|
|
},
|
|
|
|
getEditingAreaWidth : function(){
|
|
return this.elEditingAreaContainer.offsetWidth;
|
|
},
|
|
|
|
getEditingAreaHeight : function(){
|
|
return this.elEditingAreaContainer.offsetHeight;
|
|
}
|
|
});
|
|
var nSE2Version = "4a256db";
|
|
nhn.husky.SE_EditingAreaManager.version = {
|
|
revision : "4a256db",
|
|
type : "open",
|
|
number : "2.9.0"
|
|
};
|
|
/*[
|
|
* ENABLE_WYSIWYG
|
|
*
|
|
* 비활성화된 WYSIWYG 편집 영역을 활성화 시킨다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* DISABLE_WYSIWYG
|
|
*
|
|
* WYSIWYG 편집 영역을 비활성화 시킨다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* PASTE_HTML
|
|
*
|
|
* HTML을 편집 영역에 삽입한다.
|
|
*
|
|
* sHTML string 삽입할 HTML
|
|
* oPSelection object 붙여 넣기 할 영역, 생략시 현재 커서 위치
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* RESTORE_IE_SELECTION
|
|
*
|
|
* (IE전용) 에디터에서 포커스가 나가는 시점에 기억해둔 포커스를 복구한다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc WYSIWYG 모드를 제공하는 플러그인
|
|
*/
|
|
nhn.husky.SE_EditingArea_WYSIWYG = jindo.$Class({
|
|
name : "SE_EditingArea_WYSIWYG",
|
|
status : nhn.husky.PLUGIN_STATUS.NOT_READY,
|
|
|
|
sMode : "WYSIWYG",
|
|
iframe : null,
|
|
doc : null,
|
|
|
|
bStopCheckingBodyHeight : false,
|
|
bAutoResize : false, // [SMARTEDITORSUS-677] 해당 편집모드의 자동확장 기능 On/Off 여부
|
|
|
|
nBodyMinHeight : 0,
|
|
nScrollbarWidth : 0,
|
|
|
|
iLastUndoRecorded : 0,
|
|
// iMinUndoInterval : 50,
|
|
|
|
_nIFrameReadyCount : 50,
|
|
|
|
bWYSIWYGEnabled : false,
|
|
|
|
$init : function(iframe){
|
|
this.iframe = jindo.$(iframe);
|
|
var oAgent = jindo.$Agent().navigator();
|
|
// IE에서 에디터 초기화 시에 임의적으로 iframe에 포커스를 반쯤(IME 입력 안되고 커서만 깜박이는 상태) 주는 현상을 막기 위해서 일단 iframe을 숨겨 뒀다가 CHANGE_EDITING_MODE에서 위지윅 전환 시 보여준다.
|
|
// 이런 현상이 다양한 요소에 의해서 발생하며 발견된 몇가지 경우는,
|
|
// - frameset으로 페이지를 구성한 후에 한개의 frame안에 버튼을 두어 에디터로 링크 할 경우
|
|
// - iframe과 동일 페이지에 존재하는 text field에 값을 할당 할 경우
|
|
if(oAgent.ie){
|
|
this.iframe.style.display = "none";
|
|
}
|
|
|
|
// IE8 : 찾기/바꾸기에서 글자 일부에 스타일이 적용된 경우 찾기가 안되는 브라우저 버그로 인해 EmulateIE7 파일을 사용
|
|
// <meta http-equiv="X-UA-Compatible" content="IE=EmulateIE7">
|
|
this.sBlankPageURL = "smart_editor2_inputarea.html";
|
|
this.sBlankPageURL_EmulateIE7 = "smart_editor2_inputarea_ie8.html";
|
|
this.aAddtionalEmulateIE7 = [];
|
|
|
|
this.htOptions = nhn.husky.SE2M_Configuration.SE_EditingAreaManager;
|
|
if (this.htOptions) {
|
|
this.sBlankPageURL = this.htOptions.sBlankPageURL || this.sBlankPageURL;
|
|
this.sBlankPageURL_EmulateIE7 = this.htOptions.sBlankPageURL_EmulateIE7 || this.sBlankPageURL_EmulateIE7;
|
|
this.aAddtionalEmulateIE7 = this.htOptions.aAddtionalEmulateIE7 || this.aAddtionalEmulateIE7;
|
|
}
|
|
|
|
this.aAddtionalEmulateIE7.push(8); // IE8은 Default 사용
|
|
|
|
this.sIFrameSrc = this.sBlankPageURL;
|
|
if(oAgent.ie && jindo.$A(this.aAddtionalEmulateIE7).has(oAgent.nativeVersion)) {
|
|
this.sIFrameSrc = this.sBlankPageURL_EmulateIE7;
|
|
}
|
|
|
|
var sIFrameSrc = this.sIFrameSrc,
|
|
iframe = this.iframe,
|
|
fHandlerSuccess = jindo.$Fn(this.initIframe, this).bind(),
|
|
fHandlerFail =jindo.$Fn(function(){this.iframe.src = sIFrameSrc;}, this).bind();
|
|
|
|
if(!oAgent.ie || (oAgent.version >=9 && !!document.addEventListener)){
|
|
iframe.addEventListener("load", fHandlerSuccess, false);
|
|
iframe.addEventListener("error", fHandlerFail, false);
|
|
}else{
|
|
iframe.attachEvent("onload", fHandlerSuccess);
|
|
iframe.attachEvent("onerror", fHandlerFail);
|
|
}
|
|
iframe.src = sIFrameSrc;
|
|
this.elEditingArea = iframe;
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this.oEditingArea = this.iframe.contentWindow.document;
|
|
this.oApp.exec("REGISTER_EDITING_AREA", [this]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getWYSIWYGWindow", jindo.$Fn(this.getWindow, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getWYSIWYGDocument", jindo.$Fn(this.getDocument, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["isWYSIWYGEnabled", jindo.$Fn(this.isWYSIWYGEnabled, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getRawHTMLContents", jindo.$Fn(this.getRawHTMLContents, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["setRawHTMLContents", jindo.$Fn(this.setRawHTMLContents, this).bind()]);
|
|
|
|
if (!!this.isWYSIWYGEnabled()) {
|
|
this.oApp.exec('ENABLE_WYSIWYG_RULER');
|
|
}
|
|
|
|
this.oApp.registerBrowserEvent(this.getDocument().body, 'paste', 'EVENT_EDITING_AREA_PASTE');
|
|
this.oApp.registerBrowserEvent(this.getDocument().body, 'drop', 'EVENT_EDITING_AREA_DROP');
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
if(!this.oApp.hasOwnProperty("saveSnapShot")){
|
|
this.$ON_EVENT_EDITING_AREA_MOUSEUP = function(){};
|
|
this._recordUndo = function(){};
|
|
}
|
|
|
|
// uncomment this line if you wish to use the IE-style cursor in FF
|
|
// this.getDocument().body.style.cursor = "text";
|
|
|
|
// Do not update this._oIERange until the document is actually clicked (focus was given by mousedown->mouseup)
|
|
// Without this, iframe cannot be re-selected(by RESTORE_IE_SELECTION) if the document hasn't been clicked
|
|
// mousedown on iframe -> focus goes into the iframe doc -> beforedeactivate is fired -> empty selection is saved by the plugin -> empty selection is recovered in RESTORE_IE_SELECTION
|
|
this._bIERangeReset = true;
|
|
|
|
// [SMARTEDITORSUS-2149] win10_edge 추가 (TODO: 추후 win10 정식릴리즈시 ua 재확인필요)
|
|
if(this.oApp.oNavigator.ie || navigator.userAgent.indexOf("Edge") > -1){
|
|
this._bIECursorHide = true;
|
|
jindo.$Fn(
|
|
function(weEvent){
|
|
var oSelection = this.iframe.contentWindow.document.selection;
|
|
if(oSelection && oSelection.type.toLowerCase() === 'control' && weEvent.key().keyCode === 8){
|
|
this.oApp.exec("EXECCOMMAND", ['delete', false, false]);
|
|
weEvent.stop();
|
|
}
|
|
|
|
this._bIERangeReset = false;
|
|
}, this
|
|
).attach(this.iframe.contentWindow.document, "keydown");
|
|
jindo.$Fn(
|
|
function(weEvent){
|
|
this._oIERange = null;
|
|
this._bIERangeReset = true;
|
|
}, this
|
|
).attach(this.iframe.contentWindow.document.body, "mousedown");
|
|
|
|
// [SMARTEDITORSUS-1810] document.createRange 가 없는 경우만(IE8이하) beforedeactivate 이벤트 등록
|
|
if(!this.getDocument().createRange){
|
|
jindo.$Fn(this._onIEBeforeDeactivate, this).attach(this.iframe.contentWindow.document.body, "beforedeactivate");
|
|
}
|
|
|
|
jindo.$Fn(
|
|
function(weEvent){
|
|
this._bIERangeReset = false;
|
|
}, this
|
|
).attach(this.iframe.contentWindow.document.body, "mouseup");
|
|
}else if(this.oApp.oNavigator.bGPadBrowser){
|
|
// [SMARTEDITORSUS-1802] GPad 에서만 툴바 터치시 셀렉션을 저장해둔다.
|
|
this.$ON_EVENT_TOOLBAR_TOUCHSTART = function(){
|
|
this._oIERange = this.oApp.getSelection().cloneRange();
|
|
}
|
|
}
|
|
|
|
// DTD가 quirks가 아닐 경우 body 높이 100%가 제대로 동작하지 않아서 타임아웃을 돌며 높이를 수동으로 계속 할당 해 줌
|
|
// body 높이가 제대로 설정 되지 않을 경우, 보기에는 이상없어 보이나 마우스로 텍스트 선택이 잘 안된다든지 하는 이슈가 있음
|
|
this.fnSetBodyHeight = jindo.$Fn(this._setBodyHeight, this).bind();
|
|
this.fnCheckBodyChange = jindo.$Fn(this._checkBodyChange, this).bind();
|
|
|
|
this.fnSetBodyHeight();
|
|
this._nContainerHeight = this.oApp.getEditingAreaHeight(); // 편집영역이 리사이즈되었는지 체크하기 위해 초기값 할당
|
|
|
|
this._setScrollbarWidth();
|
|
},
|
|
|
|
$ON_REGISTER_CONVERTERS : function(){
|
|
this.oApp.exec("ADD_CONVERTER_DOM", ["DB_TO_IR", jindo.$Fn(this._dbToIrDOM, this).bind()]);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2315] setContents가 될때 폰트태그를 정제해준다.
|
|
*/
|
|
_dbToIrDOM : function(oTmpNode){
|
|
nhn.husky.SE2M_Utils.removeInvalidFont(oTmpNode);
|
|
nhn.husky.SE2M_Utils.convertFontToSpan(oTmpNode);
|
|
},
|
|
|
|
/**
|
|
* 스크롤바의 사이즈 측정하여 설정
|
|
*/
|
|
_setScrollbarWidth : function(){
|
|
var oDocument = this.getDocument(),
|
|
elScrollDiv = oDocument.createElement("div");
|
|
|
|
elScrollDiv.style.width = "100px";
|
|
elScrollDiv.style.height = "100px";
|
|
elScrollDiv.style.overflow = "scroll";
|
|
elScrollDiv.style.position = "absolute";
|
|
elScrollDiv.style.top = "-9999px";
|
|
|
|
oDocument.body.appendChild(elScrollDiv);
|
|
|
|
this.nScrollbarWidth = elScrollDiv.offsetWidth - elScrollDiv.clientWidth;
|
|
|
|
oDocument.body.removeChild(elScrollDiv);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] 붙여넣기나 내용 입력에 대한 편집영역 자동 확장 처리
|
|
*/
|
|
$AFTER_EVENT_EDITING_AREA_KEYUP : function(oEvent){
|
|
if(!this.bAutoResize){
|
|
return;
|
|
}
|
|
|
|
var oKeyInfo = oEvent.key();
|
|
|
|
if((oKeyInfo.keyCode >= 33 && oKeyInfo.keyCode <= 40) || oKeyInfo.alt || oKeyInfo.ctrl || oKeyInfo.keyCode === 16){
|
|
return;
|
|
}
|
|
|
|
this._setAutoResize();
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] 붙여넣기나 내용 입력에 대한 편집영역 자동 확장 처리
|
|
*/
|
|
$AFTER_PASTE_HTML : function(){
|
|
if(!this.bAutoResize){
|
|
return;
|
|
}
|
|
|
|
this._setAutoResize();
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] WYSIWYG 편집 영역 자동 확장 처리 시작
|
|
*/
|
|
startAutoResize : function(){
|
|
this.oApp.exec("STOP_CHECKING_BODY_HEIGHT");
|
|
this.bAutoResize = true;
|
|
|
|
var oBrowser = this.oApp.oNavigator;
|
|
|
|
// [SMARTEDITORSUS-887] [블로그 1단] 자동확장 모드에서 에디터 가로사이즈보다 큰 사진을 추가했을 때 가로스크롤이 안생기는 문제
|
|
if(oBrowser.ie && oBrowser.version < 9){
|
|
jindo.$Element(this.getDocument().body).css({ "overflow" : "visible" });
|
|
|
|
// { "overflowX" : "visible", "overflowY" : "hidden" } 으로 설정하면 세로 스크롤 뿐 아니라 가로 스크롤도 보이지 않는 문제가 있어
|
|
// { "overflow" : "visible" } 로 처리하고 에디터의 container 사이즈를 늘려 세로 스크롤이 보이지 않도록 처리해야 함
|
|
// [한계] 자동 확장 모드에서 내용이 늘어날 때 세로 스크롤이 보였다가 없어지는 문제
|
|
}else{
|
|
jindo.$Element(this.getDocument().body).css({ "overflowX" : "visible", "overflowY" : "hidden" });
|
|
}
|
|
|
|
this._setAutoResize();
|
|
this.nCheckBodyInterval = setInterval(this.fnCheckBodyChange, 500);
|
|
|
|
this.oApp.exec("START_FLOAT_TOOLBAR"); // set scroll event
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] WYSIWYG 편집 영역 자동 확장 처리 종료
|
|
*/
|
|
stopAutoResize : function(){
|
|
this.bAutoResize = false;
|
|
clearInterval(this.nCheckBodyInterval);
|
|
|
|
this.oApp.exec("STOP_FLOAT_TOOLBAR"); // remove scroll event
|
|
|
|
jindo.$Element(this.getDocument().body).css({ "overflow" : "visible", "overflowY" : "visible" });
|
|
|
|
this.oApp.exec("START_CHECKING_BODY_HEIGHT");
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] 편집 영역 Body가 변경되었는지 주기적으로 확인
|
|
*/
|
|
_checkBodyChange : function(){
|
|
if(!this.bAutoResize){
|
|
return;
|
|
}
|
|
|
|
var nBodyLength = this.getDocument().body.innerHTML.length;
|
|
|
|
if(nBodyLength !== this.nBodyLength){
|
|
this.nBodyLength = nBodyLength;
|
|
|
|
this._setAutoResize();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] WYSIWYG 자동 확장 처리
|
|
*/
|
|
_setAutoResize : function(){
|
|
var elBody = this.getDocument().body,
|
|
welBody = jindo.$Element(elBody),
|
|
nBodyHeight,
|
|
nContainerHeight,
|
|
oCurrentStyle,
|
|
nStyleSize,
|
|
bExpand = false,
|
|
oBrowser = this.oApp.oNavigator;
|
|
|
|
this.nTopBottomMargin = this.nTopBottomMargin || (parseInt(welBody.css("marginTop"), 10) + parseInt(welBody.css("marginBottom"), 10));
|
|
this.nBodyMinHeight = this.nBodyMinHeight || (this.oApp.getEditingAreaHeight() - this.nTopBottomMargin);
|
|
|
|
// 내용이 줄었을 경우, height를 줄여주기 위해 height를 0으로 조정하고
|
|
// scrollHeight 를 이용해 내용의 실제 높이값을 구한다.
|
|
welBody.css("height", "0px");
|
|
this.iframe.style.height = "0px";
|
|
nBodyHeight = parseInt(elBody.scrollHeight, 10);
|
|
|
|
if(nBodyHeight < this.nBodyMinHeight){ // 최소높이값 지정
|
|
nBodyHeight = this.nBodyMinHeight;
|
|
}
|
|
|
|
if(oBrowser.ie){
|
|
// 내용 뒤로 공간이 남아 보일 수 있으나 추가로 Container높이를 더하지 않으면
|
|
// 내용 가장 뒤에서 Enter를 하는 경우 아래위로 흔들려 보이는 문제가 발생
|
|
if(nBodyHeight > this.nBodyMinHeight){
|
|
oCurrentStyle = this.oApp.getCurrentStyle();
|
|
// [SMARTEDITORSUS-1756]
|
|
//nStyleSize = parseInt(oCurrentStyle.fontSize, 10) * oCurrentStyle.lineHeight;
|
|
nStyleSize = this._getStyleSize(oCurrentStyle);
|
|
// --[SMARTEDITORSUS-1756]
|
|
|
|
if(nStyleSize < this.nTopBottomMargin){
|
|
nStyleSize = this.nTopBottomMargin;
|
|
}
|
|
|
|
nContainerHeight = nBodyHeight + nStyleSize;
|
|
nContainerHeight += 18;
|
|
|
|
bExpand = true;
|
|
}else{
|
|
nBodyHeight = this.nBodyMinHeight;
|
|
nContainerHeight = this.nBodyMinHeight + this.nTopBottomMargin;
|
|
}
|
|
// }else if(oBrowser.safari){ // -- 사파리에서 내용이 줄어들지 않는 문제가 있어 Firefox 방식으로 변경함
|
|
// // [Chrome/Safari] 크롬이나 사파리에서는 Body와 iframe높이서 서로 연관되어 늘어나므로,
|
|
// // nContainerHeight를 추가로 더하는 경우 setTimeout 시 무한 증식되는 문제가 발생할 수 있음
|
|
// nBodyHeight = nBodyHeight > this.nBodyMinHeight ? nBodyHeight - this.nTopBottomMargin : this.nBodyMinHeight;
|
|
// nContainerHeight = nBodyHeight + this.nTopBottomMargin;
|
|
}else{
|
|
// [FF] nContainerHeight를 추가로 더하였음. setTimeout 시 무한 증식되는 문제가 발생할 수 있음
|
|
if(nBodyHeight > this.nBodyMinHeight){
|
|
oCurrentStyle = this.oApp.getCurrentStyle();
|
|
// [SMARTEDITORSUS-1756]
|
|
//nStyleSize = parseInt(oCurrentStyle.fontSize, 10) * oCurrentStyle.lineHeight;
|
|
nStyleSize = this._getStyleSize(oCurrentStyle);
|
|
// --[SMARTEDITORSUS-1756]
|
|
|
|
if(nStyleSize < this.nTopBottomMargin){
|
|
nStyleSize = this.nTopBottomMargin;
|
|
}
|
|
|
|
nContainerHeight = nBodyHeight + nStyleSize;
|
|
|
|
bExpand = true;
|
|
}else{
|
|
nBodyHeight = this.nBodyMinHeight;
|
|
nContainerHeight = this.nBodyMinHeight + this.nTopBottomMargin;
|
|
}
|
|
}
|
|
|
|
welBody.css("height", nBodyHeight + "px");
|
|
this.iframe.style.height = nContainerHeight + "px"; // 편집영역 IFRAME의 높이 변경
|
|
this.oApp.welEditingAreaContainer.height(nContainerHeight); // 편집영역 IFRAME을 감싸는 DIV 높이 변경
|
|
|
|
// [SMARTEDITORSUS-2036] 자동리사이즈기능으로 편집영역 크기가 변경되면 메시지를 발생시키도록 함
|
|
if(this._nContainerHeight !== nContainerHeight){
|
|
this._nContainerHeight = nContainerHeight;
|
|
this.oApp.exec('MSG_EDITING_AREA_SIZE_CHANGED');
|
|
}
|
|
|
|
//[SMARTEDITORSUS-941][iOS5대응]아이패드의 자동 확장 기능이 동작하지 않을 때 에디터 창보다 긴 내용을 작성하면 에디터를 뚫고 나오는 현상
|
|
//원인 : 자동확장 기능이 정지 될 경우 iframe에 스크롤이 생기지 않고, 창을 뚫고 나옴
|
|
//해결 : 항상 자동확장 기능이 켜져있도록 변경. 자동 확장 기능 관련한 이벤트 코드도 모바일 사파리에서 예외 처리
|
|
if(!this.oApp.oNavigator.msafari){
|
|
this.oApp.checkResizeGripPosition(bExpand);
|
|
}
|
|
},
|
|
|
|
// [SMARTEDITORSUS-1756]
|
|
_getStyleSize : function(oCurrentStyle){
|
|
/**
|
|
* this.iframe의 height style에 반영되는 높이값인
|
|
* nContainerHeight를 결정짓는 nStyleSize의 경우,
|
|
* 기존 로직에서는
|
|
* nStyleSize = parseInt(oCurrentStyle.fontSize, 10) * oCurrentStyle.lineHeight;
|
|
* 와 같이 값을 산정한다.
|
|
*
|
|
* SmartEditor에서만 생산한 컨텐츠의 경우,
|
|
* font-size 값은 px 단위형 숫자이고,
|
|
* line-height 값은 배수형 숫자이다.
|
|
*
|
|
* 따라서 nStyleSize는 이 산정으로 px 단위의 숫자값을 가지게 된다.
|
|
*
|
|
* 하지만 외부에서 붙여넣은 컨텐츠는 다양한 형태의 font-size값과 line-height 값을 가질 수 있다.
|
|
* 그 중 일부 값은 nStyleSize를 NaN으로 만들기 때문에,
|
|
* 컨텐츠가 화면에서 사라진 것처럼 보이는 현상을 일으킨다.
|
|
*
|
|
* 또한 "px 단위형 - 배수형" 이라는 틀에 맞지 않으면
|
|
* 부적절한 결과를 야기할 수 있다.
|
|
*
|
|
* 따라서 font-size 값을 px 단위형 숫자로,
|
|
* line-height 값을 배수형 숫자로 보정해 줘서,
|
|
* nStyleSize가 숫자형이 될 수 있도록 만들어 준다.
|
|
*
|
|
* line-height의 보정은 아래를 참조한다. (http://www.w3schools.com/cssref/pr_dim_line-height.asp)
|
|
* -"normal" : 통상 120%에 대응하며, 정확한 값은 font-family에 좌우 (https://developer.mozilla.org/en-US/docs/Web/CSS/line-height)
|
|
* --ex) verdana 폰트
|
|
* ---12px~15일 때 120% 에 대응
|
|
* ---16일 때 115%
|
|
* ---17일 때 120%
|
|
* ---18~20일 때 125%
|
|
* -배수형 숫자
|
|
* -단위형 숫자 (pt, px, em, cm 등)
|
|
* --pt : 12pt = 16px = 100%
|
|
* --em : 1em = 12pt = 16px = 100%
|
|
* --cm : 1inch = 2.54cm = 96px 이므로 1cm = (1/2.54*96) = 약 37.795px
|
|
* -%형
|
|
* -"initial"
|
|
* -"inherit" : 부모 엘리먼트의 값에 의해 좌우됨
|
|
*
|
|
* font-size의 보정은 아래를 참조한다. (http://www.w3schools.com/cssref/pr_font_font-size.asp)
|
|
* -"medium" : 16px = 100%
|
|
* -단위형은 line-height와 같이 처리
|
|
* */
|
|
var nResult;
|
|
if(oCurrentStyle){
|
|
// line-height 값을 배수형으로 보정
|
|
var nLineHeight = oCurrentStyle.lineHeight;
|
|
if(nLineHeight && /[^\d\.]/.test(nLineHeight)){ // 배수형이 아닌 경우
|
|
if(/\d/.test(nLineHeight) && /[A-Za-z]/.test(nLineHeight)){ // 단위형 : 실제 원하는 최종 결과값인 만큼, px 단위형으로 변환만 거친 뒤 return
|
|
if(/px$/.test(nLineHeight)){ // px 단위형 : 최종 결과값
|
|
return parseFloat(nLineHeight, 10);
|
|
}else if(/pt$/.test(nLineHeight)){ // pt 단위형
|
|
return parseFloat(nLineHeight, 10) * 4 / 3;
|
|
}else if(/em$/.test(nLineHeight)){ // em 단위형
|
|
return parseFloat(nLineHeight, 10) * 16;
|
|
}else if(/cm$/.test(nLineHeight)){ // cm 단위형
|
|
return parseFloat(nLineHeight, 10) * 96 / 2.54;
|
|
}
|
|
}else if(/\d/.test(nLineHeight) && /%/.test(nLineHeight)){ // %형
|
|
nLineHeight = parseFloat(nLineHeight, 10) * 100;
|
|
}else if(!/[^A-Za-z]/.test(nLineHeight)){ // TODO : "normal", "inherit", "initial" 세분화
|
|
nLineHeight = 1.2;
|
|
}
|
|
}
|
|
|
|
// font-size 값을 px 단위형으로 보정
|
|
var sFontSize = oCurrentStyle.fontSize;
|
|
if(sFontSize && !/px$/.test(sFontSize)){ // px 단위형이 아닌 경우
|
|
if(/pt$/.test(sFontSize)){ // pt 단위형
|
|
sFontSize = parseFloat(sFontSize, 10) * 4 / 3 + "px";
|
|
}else if(/em$/.test(sFontSize)){ // em 단위형
|
|
sFontSize = parseFloat(sFontSize, 10) * 16 + "px";
|
|
}else if(/cm$/.test(sFontSize)){ // cm 단위형
|
|
sFontSize = parseFloat(sFontSize, 10) * 96 / 2.54 + "px";
|
|
}else if(sFontSize == "medium"){ // "medium"
|
|
sFontSize = "16px";
|
|
}else{ // TODO : 다양한 small, large 종류가 존재
|
|
sFontSize = "16px";
|
|
}
|
|
}
|
|
|
|
nResult = parseFloat(sFontSize, 10) * nLineHeight;
|
|
}else{
|
|
nResult = 12 * 1.5;
|
|
}
|
|
|
|
return nResult;
|
|
},
|
|
// --[SMARTEDITORSUS-1756]
|
|
|
|
/**
|
|
* 스크롤 처리를 위해 편집영역 Body의 사이즈를 확인하고 설정함
|
|
* 편집영역 자동확장 기능이 Off인 경우에 주기적으로 실행됨
|
|
*/
|
|
_setBodyHeight : function(){
|
|
if( this.bStopCheckingBodyHeight ){ // 멈춰야 하는 경우 true, 계속 체크해야 하면 false
|
|
// 위지윅 모드에서 다른 모드로 변경할 때 "document는 css를 사용 할수 없습니다." 라는 error 가 발생.
|
|
// 그래서 on_change_mode에서 bStopCheckingBodyHeight 를 true로 변경시켜줘야 함.
|
|
return;
|
|
}
|
|
|
|
var elBody = this.getDocument().body,
|
|
welBody = jindo.$Element(elBody),
|
|
nMarginTopBottom = parseInt(welBody.css("marginTop"), 10) + parseInt(welBody.css("marginBottom"), 10),
|
|
nContainerOffset = this.oApp.getEditingAreaHeight(),
|
|
nMinBodyHeight = nContainerOffset - nMarginTopBottom,
|
|
nBodyHeight = welBody.height(),
|
|
nScrollHeight,
|
|
nNewBodyHeight;
|
|
|
|
this.nTopBottomMargin = nMarginTopBottom;
|
|
|
|
if(nBodyHeight === 0){ // [SMARTEDITORSUS-144] height 가 0 이고 내용이 없으면 크롬10 에서 캐럿이 보이지 않음
|
|
welBody.css("height", nMinBodyHeight + "px");
|
|
|
|
setTimeout(this.fnSetBodyHeight, 500);
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1972] [IE 11] 마지막 변경된 body height에서 변화가 없는 경우 0px로 축소하지 않음
|
|
* */
|
|
var htBrowser = jindo.$Agent().navigator(),
|
|
isIE11 = (htBrowser.ie && htBrowser.nativeVersion === 11),
|
|
isShrinkingUnnecessary = (this.nBodyHeight_last === nBodyHeight);
|
|
|
|
if(!(isIE11 && isShrinkingUnnecessary)){
|
|
welBody.css("height", "0px");
|
|
}
|
|
// Previous below
|
|
/*welBody.css("height", "0px");*/
|
|
// --[SMARTEDITORSUS-1972]
|
|
|
|
// [SMARTEDITORSUS-257] IE9, 크롬에서 내용을 삭제해도 스크롤이 남아있는 문제 처리
|
|
// body 에 내용이 없어져도 scrollHeight 가 줄어들지 않아 height 를 강제로 0 으로 설정
|
|
|
|
nScrollHeight = parseInt(elBody.scrollHeight, 10);
|
|
|
|
nNewBodyHeight = (nScrollHeight > nContainerOffset ? nScrollHeight - nMarginTopBottom : nMinBodyHeight);
|
|
// nMarginTopBottom 을 빼지 않으면 스크롤이 계속 늘어나는 경우가 있음 (참고 [BLOGSUS-17421])
|
|
|
|
if(this._isHorizontalScrollbarVisible()){
|
|
nNewBodyHeight -= this.nScrollbarWidth;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1972]
|
|
if(!(isIE11 && isShrinkingUnnecessary)){
|
|
welBody.css("height", nNewBodyHeight + "px");
|
|
}
|
|
this.nBodyHeight_last = nNewBodyHeight;
|
|
// Previous below
|
|
/*welBody.css("height", nNewBodyHeight + "px");*/
|
|
// --[SMARTEDITORSUS-1972]
|
|
|
|
setTimeout(this.fnSetBodyHeight, 500);
|
|
},
|
|
|
|
/**
|
|
* 가로 스크롤바 생성 확인
|
|
*/
|
|
_isHorizontalScrollbarVisible : function(){
|
|
var oDocument = this.getDocument();
|
|
|
|
if(oDocument.documentElement.clientWidth < oDocument.documentElement.scrollWidth){
|
|
//oDocument.body.clientWidth < oDocument.body.scrollWidth ||
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* body의 offset체크를 멈추게 하는 함수.
|
|
*/
|
|
$ON_STOP_CHECKING_BODY_HEIGHT :function(){
|
|
if(!this.bStopCheckingBodyHeight){
|
|
this.bStopCheckingBodyHeight = true;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* body의 offset체크를 계속 진행.
|
|
*/
|
|
$ON_START_CHECKING_BODY_HEIGHT :function(){
|
|
if(this.bStopCheckingBodyHeight){
|
|
this.bStopCheckingBodyHeight = false;
|
|
this.fnSetBodyHeight();
|
|
}
|
|
},
|
|
|
|
$ON_IE_CHECK_EXCEPTION_FOR_SELECTION_PRESERVATION : function(){
|
|
// 현재 선택된 앨리먼트가 iframe이라면, 셀렉션을 따로 기억 해 두지 않아도 유지 됨으로 RESTORE_IE_SELECTION을 타지 않도록 this._oIERange을 지워준다.
|
|
// (필요 없을 뿐더러 저장 시 문제 발생)
|
|
var oSelection = this.getDocument().selection;
|
|
if(oSelection && oSelection.type === "Control"){
|
|
this._oIERange = null;
|
|
}
|
|
},
|
|
|
|
_onIEBeforeDeactivate : function(wev){
|
|
this.oApp.delayedExec("IE_CHECK_EXCEPTION_FOR_SELECTION_PRESERVATION", null, 0);
|
|
|
|
if(this._oIERange){
|
|
return;
|
|
}
|
|
|
|
// without this, cursor won't make it inside a table.
|
|
// mousedown(_oIERange gets reset) -> beforedeactivate(gets fired for table) -> RESTORE_IE_SELECTION
|
|
if(this._bIERangeReset){
|
|
return;
|
|
}
|
|
|
|
this._oIERange = this.oApp.getSelection().cloneRange();
|
|
},
|
|
|
|
$ON_CHANGE_EDITING_MODE : function(sMode, bNoFocus){
|
|
if(sMode === this.sMode){
|
|
// [SMARTEDITORSUS-1213][IE9, 10] 사진 삭제 후 zindex 1000인 div가 잔존하는데, 그 위로 썸네일 drag를 시도하다 보니 drop이 불가능.
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if(htBrowser.ie && htBrowser.nativeVersion > 8){
|
|
var elFirstChild = jindo.$$.getSingle("DIV.husky_seditor_editing_area_container").childNodes[0];
|
|
if((elFirstChild.tagName == "DIV") && (elFirstChild.style.zIndex == 1000)){
|
|
elFirstChild.parentNode.removeChild(elFirstChild);
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1213]
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1889]
|
|
* visibility 속성을 사용해서 Editor를 표시하고 숨김
|
|
* 단, 에디터 초기화 시 필요한 display:block 설정은 유지
|
|
*
|
|
* */
|
|
this.iframe.style.visibility = "visible";
|
|
if(this.iframe.style.display != "block"){ // 초기화 시 최초 1회
|
|
this.iframe.style.display = "block";
|
|
}
|
|
// Previous below
|
|
//this.iframe.style.display = "block";
|
|
// --[SMARTEDITORSUS-1889]
|
|
|
|
this.oApp.exec("SET_EDITING_WINDOW", [this.getWindow()]);
|
|
this.oApp.exec("START_CHECKING_BODY_HEIGHT");
|
|
}else{
|
|
/**
|
|
* [SMARTEDITORSUS-1889]
|
|
* 모드 전환 시 display:none과 display:block을 사용해서
|
|
* Editor 영역을 표시하고 숨기는 경우,
|
|
* iframe 요소가 그 때마다 다시 로드되는 과정에서
|
|
* 스크립트 오류를 유발시킴 (국내지도)
|
|
*
|
|
* 따라서 visibility 속성을 대신 사용하고,
|
|
* 이 경우 Editor 영역이 공간을 여전히 차지하고 있기 때문에
|
|
* 그 아래 위치하게 될 수밖에 없는
|
|
* HTML 영역이나 Text 영역은
|
|
* position:absolute와 top 속성을 사용하여
|
|
* 위로 끌어올리는 방법을 사용
|
|
* */
|
|
this.iframe.style.visibility = "hidden";
|
|
// previous below
|
|
//this.iframe.style.display = "none";
|
|
// --[SMARTEDITORSUS-1889]
|
|
this.oApp.exec("STOP_CHECKING_BODY_HEIGHT");
|
|
}
|
|
},
|
|
|
|
$AFTER_CHANGE_EDITING_MODE : function(sMode, bNoFocus){
|
|
this._oIERange = null;
|
|
},
|
|
|
|
$ON_ENABLE_WYSIWYG : function(){
|
|
this._enableWYSIWYG();
|
|
},
|
|
|
|
$ON_DISABLE_WYSIWYG : function(){
|
|
this._disableWYSIWYG();
|
|
},
|
|
|
|
$ON_IE_HIDE_CURSOR : function(){
|
|
if(!this._bIECursorHide){
|
|
return;
|
|
}
|
|
|
|
this._onIEBeforeDeactivate();
|
|
|
|
// De-select the default selection.
|
|
// [SMARTEDITORSUS-978] IE9에서 removeAllRanges로 제거되지 않아
|
|
// 이전 IE와 동일하게 empty 방식을 사용하도록 하였으나 doc.selection.type이 None인 경우 에러
|
|
// Range를 재설정 해주어 selectNone 으로 처리되도록 예외처리
|
|
var oSelection = this.oApp.getWYSIWYGDocument().selection;
|
|
if(oSelection && oSelection.createRange){
|
|
try{
|
|
oSelection.empty();
|
|
}catch(e){
|
|
// [SMARTEDITORSUS-1003] IE9 / doc.selection.type === "None"
|
|
oSelection = this.oApp.getSelection();
|
|
oSelection.select();
|
|
oSelection.oBrowserSelection.selectNone();
|
|
}
|
|
}else{
|
|
this.oApp.getEmptySelection().oBrowserSelection.selectNone();
|
|
this.getDocument().body.blur(); // [SMARTEDITORSUS-2149] win10_edge 에서 커서가 보이지 않도록 하려면 blur 해줘야 한다.
|
|
}
|
|
},
|
|
|
|
$AFTER_SHOW_ACTIVE_LAYER : function(){
|
|
this.oApp.exec("IE_HIDE_CURSOR");
|
|
this.bActiveLayerShown = true;
|
|
},
|
|
|
|
$BEFORE_EVENT_EDITING_AREA_KEYDOWN : function(oEvent){
|
|
this._bKeyDown = true;
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_KEYDOWN : function(oEvent){
|
|
if(this.oApp.getEditingMode() !== this.sMode){
|
|
return;
|
|
}
|
|
|
|
var oKeyInfo = oEvent.key();
|
|
if(this.oApp.oNavigator.ie){
|
|
//var oKeyInfo = oEvent.key();
|
|
switch(oKeyInfo.keyCode){
|
|
case 33:
|
|
this._pageUp(oEvent);
|
|
break;
|
|
case 34:
|
|
this._pageDown(oEvent);
|
|
break;
|
|
case 8: // [SMARTEDITORSUS-495][SMARTEDITORSUS-548] IE에서 표가 삭제되지 않는 문제
|
|
this._backspace(oEvent);
|
|
break;
|
|
case 46: // [SMARTEDITORSUS-2064] IE11은 delete 로 테이블 삭제가 안되서 추가함
|
|
this._delete(oEvent);
|
|
break;
|
|
default:
|
|
}
|
|
}else if(this.oApp.oNavigator.firefox){
|
|
// [SMARTEDITORSUS-151] FF 에서 표가 삭제되지 않는 문제
|
|
if(oKeyInfo.keyCode === 8){ // backspace
|
|
this._backspace(oEvent);
|
|
}
|
|
}
|
|
|
|
this._recordUndo(oKeyInfo); // 첫번째 Delete 키 입력 전의 상태가 저장되도록 KEYDOWN 시점에 저장
|
|
},
|
|
|
|
/**
|
|
* IE와 FF에서 백스페이스로 테이블이 삭제되지 않기 때문에 강제삭제 처리
|
|
*/
|
|
_backspace : function(weEvent){
|
|
var oPrevNode = this._prepareBackspaceDelete(true);
|
|
if(!oPrevNode){
|
|
return;
|
|
}
|
|
|
|
if(this._removeUnremovable(oPrevNode, true)){
|
|
// table 처럼 키로 삭제가 안되는 경우 강제 삭제하고 이벤트를 중단한다.
|
|
weEvent.stop();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2064] IE11은 delete 로 테이블 삭제가 안되서 추가함
|
|
* [SMARTEDITORSUS-2184] span > p 역전현상으로 인한 오류 보정 로직 추가
|
|
*/
|
|
_delete : function(weEvent){
|
|
var oNextNode = this._prepareBackspaceDelete(false);
|
|
if(!oNextNode){
|
|
return;
|
|
}
|
|
|
|
if(this._removeUnremovable(oNextNode, false)){
|
|
// table 처럼 키로 삭제가 안되는 경우 강제 삭제하고 이벤트를 중단한다.
|
|
weEvent.stop();
|
|
}else if(oNextNode.nodeType === 3){
|
|
// [SMARTEDITORSUS-2184] 텍스트 노드이면 다음 라인을 확인하여 span > p 역전된 경우 span 만 제거하도록 처리
|
|
var oLineInfo = this.oApp.getSelection().getLineInfo(),
|
|
oEnd = oLineInfo.oEnd.oLineBreaker,
|
|
oNextLine = oEnd && oEnd.nextSibling;
|
|
this._removeWrongSpan(oNextLine);
|
|
}else{
|
|
// [SMARTEDITORSUS-2184] span > p 역전된 경우 span 만 제거하도록 처리
|
|
this._removeWrongSpan(oNextNode);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* backspace/delete 키에 대한 공통 전처리 과정으로
|
|
* 셀렉션레인지가 collapsed 상태인 경우 주변의 노드를 반환한다.
|
|
* @param {Boolean} bBackspace 백스페이스키 여부 (true 면 앞쪽노드를 찾고 false 면 뒤쪽노드를 찾는다.)
|
|
* @returns {Node} 찾은 주변 노드
|
|
*/
|
|
_prepareBackspaceDelete : function(bBackspace){
|
|
var oSelection = this.oApp.getSelection();
|
|
if(!oSelection.collapsed){
|
|
return;
|
|
}
|
|
|
|
var oNode = oSelection.getNodeAroundRange(bBackspace, false);
|
|
// LineFeed 텍스트노드라면 다음 노드를 할당
|
|
if(this._isLineFeed(oNode)){
|
|
oNode = bBackspace ? oSelection._getPrevNode(oNode) : oSelection._getNextNode(oNode);
|
|
}
|
|
/*
|
|
* [SMARTEDITORSUS-1575] 빈라인에 커서홀더가 삽입된 상태에서는
|
|
* 키를 두번 쳐야 빈줄이 삭제되기 때문에 미리 커서홀더문자는 제거한다.
|
|
*/
|
|
this._clearCursorHolderValue(oNode);
|
|
|
|
return oNode;
|
|
},
|
|
|
|
/**
|
|
* 해당 텍스트 노드가 LineFeed(\n) 로만 이루어졌는지 여부
|
|
* @param {Node} oNode 확인할 노드
|
|
* @returns {Boolean} LineFeed(\n) 로만 이루어진 텍스트노드인지 여부
|
|
*/
|
|
_isLineFeed : function(oNode){
|
|
return (oNode && oNode.nodeType === 3 && /^[\n]*$/.test(oNode.nodeValue));
|
|
},
|
|
|
|
/**
|
|
* 해당 텍스트 노드의 값이 커서홀더 문자이면 값을 비운다. (노드자체를 제거하지 않고 문자값만 비운다.)
|
|
* @param {Node} oNode 확인할 노드
|
|
*/
|
|
_clearCursorHolderValue : function(oNode){
|
|
if(oNode && oNode.nodeType === 3 &&
|
|
(oNode.nodeValue === "\u200B" || oNode.nodeValue === "\uFEFF")){
|
|
oNode.nodeValue = "";
|
|
}
|
|
},
|
|
|
|
/**
|
|
* backspace나 delete 키로 삭제가 안되는 요소를 강제 삭제한다.
|
|
* @param {Node} oNode 확인할 노드
|
|
* @param {Boolean} bBackspace 백스페이스키 여부
|
|
* @returns {Boolean} 삭제되었으면 true 반환
|
|
*/
|
|
_removeUnremovable : function(oNode, bBackspace){
|
|
var bRemoved = false;
|
|
if(!oNode){
|
|
return false;
|
|
}
|
|
|
|
if(oNode.nodeName === "TABLE"){
|
|
oNode.parentNode.removeChild(oNode);
|
|
bRemoved = true;
|
|
}else if(oNode.nodeName === "DIV"){
|
|
/*
|
|
* IE의 경우 텍스트가 없는 블럭요소가 삭제되지 않기 때문에 별도 처리함
|
|
* TODO: div 뿐만 아니라 다른 블럭요소도 마찬가지일 것으로 추정되나 일단 div에 대해서만 한정 처리함
|
|
*/
|
|
var oChild = bBackspace ? oNode.lastChild : oNode.firstChild;
|
|
if(!oChild){
|
|
oNode.parentNode.removeChild(oNode);
|
|
bRemoved = true;
|
|
}else if(oChild.nodeName === "TABLE"){
|
|
oNode.removeChild(oChild);
|
|
bRemoved = true;
|
|
}else if(oChild.nodeType === 1 && jindo.$S(oChild.innerHTML).trim() == ""){
|
|
oNode.removeChild(oChild);
|
|
bRemoved = true;
|
|
}
|
|
}
|
|
|
|
return bRemoved;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2184] span 안쪽에 p태그가 있는 경우 span의 모든 child 노드를 밖으로 빼내고 제거한다.
|
|
* known-issue: span에 스타일이 적용되어 있을 경우, 적용된 스타일이 풀려버린다.
|
|
* 잘못된 span에 적용된 스타일을 무조건 안쪽에 넣어주기에는 위험도가 있어서 별도 처리하지 않음
|
|
* @param {Node} oNode 확인할 노드
|
|
*/
|
|
_removeWrongSpan : function(oNode){
|
|
if(oNode && oNode.nodeName === "SPAN" && oNode.firstChild && oNode.firstChild.nodeName === "P"){
|
|
var oParentNode = oNode.parentNode;
|
|
while(oNode.firstChild){
|
|
oParentNode.insertBefore(oNode.firstChild, oNode);
|
|
}
|
|
oParentNode.removeChild(oNode);
|
|
}
|
|
},
|
|
|
|
$BEFORE_EVENT_EDITING_AREA_KEYUP : function(oEvent){
|
|
// IE(6) sometimes fires keyup events when it should not and when it happens the keyup event gets fired without a keydown event
|
|
if(!this._bKeyDown){
|
|
return false;
|
|
}
|
|
this._bKeyDown = false;
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_MOUSEUP : function(oEvent){
|
|
this.oApp.saveSnapShot();
|
|
},
|
|
|
|
$BEFORE_PASTE_HTML : function(){
|
|
if(this.oApp.getEditingMode() !== this.sMode){
|
|
this.oApp.exec("CHANGE_EDITING_MODE", [this.sMode]);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* @param {String} sHTML 삽입할 HTML
|
|
* @param {HuskyRange} oPSelection 재사용할 Selection 객체
|
|
* @param {HashTable} htOption 추가옵션
|
|
* @param {Boolean} htOption.bNoUndo UNDO 히스토리를 저장하지 않을지 여부
|
|
* @param {Boolean} htOption.bBlock HTML 삽입시 강제로 block 요소 처리할지 여부(true 이면 P태그 안에 삽입될 경우, P태그를 무조건 쪼개고 그 사이에 DIV태그로 감싸서 삽입한다.)
|
|
*/
|
|
$ON_PASTE_HTML : function(sHTML, oPSelection, htOption){
|
|
htOption = htOption || {};
|
|
var oSelection, oNavigator, sTmpBookmark,
|
|
oStartContainer, aImgChild, elLastImg, elChild, elNextChild;
|
|
|
|
if(this.oApp.getEditingMode() !== this.sMode){
|
|
return;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-2023] 편집영역에 포커스가 없는 상태에서 PASTE_HTML 을 수행하면
|
|
// <p>태그 바깥쪽으로 삽입되기 때문에 pasteHTML 전에 포커스를 준다.
|
|
this.focus();
|
|
|
|
if(!htOption.bNoUndo){
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["PASTE HTML"]);
|
|
}
|
|
|
|
oNavigator = jindo.$Agent().navigator();
|
|
oSelection = oPSelection || this.oApp.getSelection();
|
|
|
|
//[SMARTEDITORSUS-888] 브라우저 별 테스트 후 아래 부분이 불필요하여 제거함
|
|
// - [SMARTEDITORSUS-387] IE9 표준모드에서 엘리먼트 뒤에 어떠한 엘리먼트도 없는 상태에서 커서가 안들어가는 현상.
|
|
// if(oNavigator.ie && oNavigator.nativeVersion >= 9 && document.documentMode >= 9){
|
|
// sHTML = sHTML + unescape("%uFEFF");
|
|
// }
|
|
//[SMARTEDITORSUS-2043] IE8의 경우 FEFF 로 인해 한줄 내려가는 현상이 생겨서 아래 부분 제거함
|
|
//if(oNavigator.ie && oNavigator.nativeVersion == 8 && document.documentMode == 8){
|
|
// sHTML = sHTML + unescape("%uFEFF");
|
|
//}
|
|
|
|
oSelection.pasteHTML(sHTML, htOption.bBlock);
|
|
|
|
// every browser except for IE may modify the innerHTML when it is inserted
|
|
if(!oNavigator.ie){
|
|
sTmpBookmark = oSelection.placeStringBookmark();
|
|
this.oApp.getWYSIWYGDocument().body.innerHTML = this.oApp.getWYSIWYGDocument().body.innerHTML;
|
|
oSelection.moveToBookmark(sTmpBookmark);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(sTmpBookmark);
|
|
// [SMARTEDITORSUS-56] 사진을 연속으로 첨부할 경우 연이어 삽입되지 않는 현상으로 이슈를 발견하게 되었습니다.
|
|
// 그러나 이는 비단 '다수의 사진을 첨부할 경우'에만 발생하는 문제는 아니었고,
|
|
// 원인 확인 결과 컨텐츠 삽입 후 기존 Bookmark 삭제 시 갱신된 Selection 이 제대로 반영되지 않는 점이 있었습니다.
|
|
// 이에, Selection 을 갱신하는 코드를 추가하였습니다.
|
|
oSelection = this.oApp.getSelection();
|
|
|
|
//[SMARTEDITORSUS-831] 비IE 계열 브라우저에서 스크롤바가 생기게 문자입력 후 엔터 클릭하지 않은 상태에서
|
|
//이미지 하나 삽입 시 이미지에 포커싱이 놓이지 않습니다.
|
|
//원인 : parameter로 넘겨 받은 oPSelecion에 변경된 값을 복사해 주지 않아서 발생
|
|
//해결 : parameter로 넘겨 받은 oPSelecion에 변경된 값을 복사해준다
|
|
// call by reference로 넘겨 받았으므로 직접 객체 안의 인자 값을 바꿔주는 setRange 함수 사용
|
|
if(!!oPSelection){
|
|
oPSelection.setRange(oSelection);
|
|
}
|
|
}else{
|
|
// [SMARTEDITORSUS-428] [IE9.0] IE9에서 포스트 쓰기에 접근하여 맨위에 임의의 글감 첨부 후 엔터를 클릭 시 글감이 사라짐
|
|
// PASTE_HTML 후에 IFRAME 부분이 선택된 상태여서 Enter 시 내용이 제거되어 발생한 문제
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
|
|
this._oIERange = null;
|
|
this._bIERangeReset = false;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-639] 사진 첨부 후 이미지 뒤의 공백으로 인해 스크롤이 생기는 문제
|
|
if(sHTML.indexOf("<img") > -1){
|
|
oStartContainer = oSelection.startContainer;
|
|
|
|
if(oStartContainer.nodeType === 1 && oStartContainer.tagName === "P"){
|
|
aImgChild = jindo.$Element(oStartContainer).child(function(v){
|
|
return (v.$value().nodeType === 1 && v.$value().tagName === "IMG");
|
|
}, 1);
|
|
|
|
if(aImgChild.length > 0){
|
|
elLastImg = aImgChild[aImgChild.length - 1].$value();
|
|
elChild = elLastImg.nextSibling;
|
|
|
|
while(elChild){
|
|
elNextChild = elChild.nextSibling;
|
|
|
|
if (elChild.nodeType === 3 && (elChild.nodeValue === " " || elChild.nodeValue === unescape("%u00A0"))) {
|
|
oStartContainer.removeChild(elChild);
|
|
}
|
|
|
|
elChild = elNextChild;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!htOption.bNoUndo){
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["PASTE HTML"]);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-344]사진/동영상/지도 연속첨부시 포커싱 개선이슈로 추가되 함수.
|
|
*/
|
|
$ON_FOCUS_N_CURSOR : function (bEndCursor, sId){
|
|
var el, oSelection;
|
|
if(sId && ( el = jindo.$(sId, this.getDocument()) )){
|
|
// ID가 지정된 경우, 무조건 해당 부분으로 커서 이동
|
|
clearTimeout(this._nTimerFocus); // 연속 삽입될 경우, 미완료 타이머는 취소한다.
|
|
this._nTimerFocus = setTimeout(jindo.$Fn(function(el){
|
|
this._scrollIntoView(el);
|
|
this.oApp.exec("FOCUS");
|
|
}, this).bind(el), 300);
|
|
return;
|
|
}
|
|
|
|
oSelection = this.oApp.getSelection();
|
|
if(!oSelection.collapsed){ // select 영역이 있는 경우
|
|
if(bEndCursor){
|
|
oSelection.collapseToEnd();
|
|
} else {
|
|
oSelection.collapseToStart();
|
|
}
|
|
oSelection.select();
|
|
}else if(bEndCursor){ // select 영역이 없는 상태에서 bEndCursor 이면 body 맨 뒤로 이동시킨다.
|
|
this.oApp.exec("FOCUS");
|
|
el = this.getDocument().body;
|
|
oSelection.selectNode(el);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
this._scrollIntoView(el);
|
|
}else{ // select 영역이 없는 상태라면 focus만 준다.
|
|
this.oApp.exec("FOCUS");
|
|
}
|
|
},
|
|
|
|
/*
|
|
* 엘리먼트의 top, bottom 값을 반환
|
|
*/
|
|
_getElementVerticalPosition : function(el){
|
|
var nTop = 0,
|
|
elParent = el,
|
|
htPos = {nTop : 0, nBottom : 0};
|
|
|
|
if(!el){
|
|
return htPos;
|
|
}
|
|
|
|
// 테스트코드를 실행하면 IE8 이하에서 offsetParent 접근시 다음과 같이 알 수 없는 exception 이 발생함
|
|
// "SCRIPT16389: 지정되지 않은 오류입니다."
|
|
// TODO: 해결방법이 없어서 일단 try/catch 처리했지만 추후 정확한 이유를 파악할 필요가 있음
|
|
try{
|
|
while(elParent) {
|
|
nTop += elParent.offsetTop;
|
|
elParent = elParent.offsetParent;
|
|
}
|
|
}catch(e){}
|
|
|
|
htPos.nTop = nTop;
|
|
htPos.nBottom = nTop + jindo.$Element(el).height();
|
|
|
|
return htPos;
|
|
},
|
|
|
|
/*
|
|
* Window에서 현재 보여지는 영역의 top, bottom 값을 반환
|
|
*/
|
|
_getVisibleVerticalPosition : function(){
|
|
var oWindow, oDocument, nVisibleHeight,
|
|
htPos = {nTop : 0, nBottom : 0};
|
|
|
|
oWindow = this.getWindow();
|
|
oDocument = this.getDocument();
|
|
nVisibleHeight = oWindow.innerHeight ? oWindow.innerHeight : oDocument.documentElement.clientHeight || oDocument.body.clientHeight;
|
|
|
|
htPos.nTop = oWindow.pageYOffset || oDocument.documentElement.scrollTop;
|
|
htPos.nBottom = htPos.nTop + nVisibleHeight;
|
|
|
|
return htPos;
|
|
},
|
|
|
|
/*
|
|
* 엘리먼트가 WYSIWYG Window의 Visible 부분에서 완전히 보이는 상태인지 확인 (일부만 보이면 false)
|
|
*/
|
|
_isElementVisible : function(htElementPos, htVisiblePos){
|
|
return (htElementPos.nTop >= htVisiblePos.nTop && htElementPos.nBottom <= htVisiblePos.nBottom);
|
|
},
|
|
|
|
/*
|
|
* [SMARTEDITORSUS-824] [SMARTEDITORSUS-828] 자동 스크롤 처리
|
|
*/
|
|
_scrollIntoView : function(el){
|
|
var htElementPos = this._getElementVerticalPosition(el),
|
|
htVisiblePos = this._getVisibleVerticalPosition(),
|
|
nScroll = 0;
|
|
|
|
if(this._isElementVisible(htElementPos, htVisiblePos)){
|
|
return;
|
|
}
|
|
|
|
if((nScroll = htElementPos.nBottom - htVisiblePos.nBottom) > 0){
|
|
this.getWindow().scrollTo(0, htVisiblePos.nTop + nScroll); // Scroll Down
|
|
return;
|
|
}
|
|
|
|
this.getWindow().scrollTo(0, htElementPos.nTop); // Scroll Up
|
|
},
|
|
|
|
$BEFORE_MSG_EDITING_AREA_RESIZE_STARTED : function(){
|
|
// FF에서 Height조정 시에 본문의 _fitElementInEditingArea()함수 부분에서 selection이 깨지는 현상을 잡기 위해서
|
|
// StringBookmark를 사용해서 위치를 저장해둠. (step1)
|
|
if(!jindo.$Agent().navigator().ie){
|
|
var oSelection = null;
|
|
oSelection = this.oApp.getSelection();
|
|
this.sBM = oSelection.placeStringBookmark();
|
|
}
|
|
},
|
|
|
|
$AFTER_MSG_EDITING_AREA_RESIZE_ENDED : function(FnMouseDown, FnMouseMove, FnMouseUp){
|
|
if(this.oApp.getEditingMode() !== this.sMode){
|
|
return;
|
|
}
|
|
|
|
// bts.nhncorp.com/nhnbts/browse/COM-1042
|
|
// $BEFORE_MSG_EDITING_AREA_RESIZE_STARTED에서 저장한 StringBookmark를 셋팅해주고 삭제함.(step2)
|
|
if(!jindo.$Agent().navigator().ie){
|
|
var oSelection = this.oApp.getEmptySelection();
|
|
oSelection.moveToBookmark(this.sBM);
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(this.sBM);
|
|
}
|
|
},
|
|
|
|
$ON_CLEAR_IE_BACKUP_SELECTION : function(){
|
|
this._oIERange = null;
|
|
},
|
|
|
|
$ON_RESTORE_IE_SELECTION : function(){
|
|
if(this._oIERange){
|
|
// changing the visibility of the iframe can cause an exception
|
|
try{
|
|
this._oIERange.select();
|
|
|
|
this._oPrevIERange = this._oIERange;
|
|
this._oIERange = null;
|
|
}catch(e){}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* EVENT_EDITING_AREA_PASTE 의 ON 메시지 핸들러
|
|
* 위지윅 모드에서 에디터 본문의 paste 이벤트에 대한 메시지를 처리한다.
|
|
* paste 시에 내용이 붙여진 본문의 내용을 바로 가져올 수 없어 delay 를 준다.
|
|
*/
|
|
$ON_EVENT_EDITING_AREA_PASTE : function(oEvent){
|
|
this.oApp.delayedExec('EVENT_EDITING_AREA_PASTE_DELAY', [oEvent], 0);
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_PASTE_DELAY : function(weEvent) {
|
|
this._replaceBlankToNbsp(weEvent.element);
|
|
},
|
|
|
|
// [SMARTEDITORSUS-855] IE에서 특정 블로그 글을 복사하여 붙여넣기 했을 때 개행이 제거되는 문제
|
|
_replaceBlankToNbsp : function(el){
|
|
var oNavigator = this.oApp.oNavigator;
|
|
|
|
if(!oNavigator.ie){
|
|
return;
|
|
}
|
|
|
|
if(oNavigator.nativeVersion !== 9 || document.documentMode !== 7) { // IE9 호환모드에서만 발생
|
|
return;
|
|
}
|
|
|
|
if(el.nodeType !== 1){
|
|
return;
|
|
}
|
|
|
|
if(el.tagName === "BR"){
|
|
return;
|
|
}
|
|
|
|
var aEl = jindo.$$("p:empty()", this.oApp.getWYSIWYGDocument().body, { oneTimeOffCache:true });
|
|
|
|
jindo.$A(aEl).forEach(function(value, index, array) {
|
|
value.innerHTML = " ";
|
|
});
|
|
},
|
|
|
|
_pageUp : function(we){
|
|
var nEditorHeight = this._getEditorHeight(),
|
|
htPos = jindo.$Document(this.oApp.getWYSIWYGDocument()).scrollPosition(),
|
|
nNewTop;
|
|
|
|
if(htPos.top <= nEditorHeight){
|
|
nNewTop = 0;
|
|
}else{
|
|
nNewTop = htPos.top - nEditorHeight;
|
|
}
|
|
this.oApp.getWYSIWYGWindow().scrollTo(0, nNewTop);
|
|
we.stop();
|
|
},
|
|
|
|
_pageDown : function(we){
|
|
var nEditorHeight = this._getEditorHeight(),
|
|
htPos = jindo.$Document(this.oApp.getWYSIWYGDocument()).scrollPosition(),
|
|
nBodyHeight = this._getBodyHeight(),
|
|
nNewTop;
|
|
|
|
if(htPos.top+nEditorHeight >= nBodyHeight){
|
|
nNewTop = nBodyHeight - nEditorHeight;
|
|
}else{
|
|
nNewTop = htPos.top + nEditorHeight;
|
|
}
|
|
this.oApp.getWYSIWYGWindow().scrollTo(0, nNewTop);
|
|
we.stop();
|
|
},
|
|
|
|
_getEditorHeight : function(){
|
|
return this.oApp.elEditingAreaContainer.offsetHeight - this.nTopBottomMargin;
|
|
},
|
|
|
|
_getBodyHeight : function(){
|
|
return parseInt(this.getDocument().body.scrollHeight, 10);
|
|
},
|
|
|
|
initIframe : function(){
|
|
try {
|
|
if (!this.iframe.contentWindow.document || !this.iframe.contentWindow.document.body || this.iframe.contentWindow.document.location.href === 'about:blank'){
|
|
throw new Error('Access denied');
|
|
}
|
|
|
|
var sCSSBaseURI = (!!nhn.husky.SE2M_Configuration.SE2M_CSSLoader && nhn.husky.SE2M_Configuration.SE2M_CSSLoader.sCSSBaseURI) ?
|
|
nhn.husky.SE2M_Configuration.SE2M_CSSLoader.sCSSBaseURI : "";
|
|
|
|
if(!!nhn.husky.SE2M_Configuration.SE_EditingAreaManager.sCSSBaseURI){
|
|
sCSSBaseURI = nhn.husky.SE2M_Configuration.SE_EditingAreaManager.sCSSBaseURI;
|
|
}
|
|
|
|
// add link tag
|
|
if (sCSSBaseURI){
|
|
var sCssUrl = sCSSBaseURI;
|
|
var sLocale = this.oApp && this.oApp.htOptions.I18N_LOCALE;
|
|
if(sLocale){
|
|
sCssUrl += "/" + sLocale;
|
|
}
|
|
sCssUrl += "/smart_editor2_in.css";
|
|
|
|
var doc = this.getDocument();
|
|
var headNode = doc.getElementsByTagName("head")[0];
|
|
var linkNode = doc.createElement('link');
|
|
linkNode.type = 'text/css';
|
|
linkNode.rel = 'stylesheet';
|
|
linkNode.href = sCssUrl;
|
|
linkNode.onload = jindo.$Fn(function(){
|
|
// [SMARTEDITORSUS-1853] IE의 경우 css가 로드되어 반영되는데 시간이 걸려서 브라우저 기본폰트가 세팅되는 경우가 있음
|
|
// 때문에 css가 로드되면 SE_WYSIWYGStylerGetter 플러그인의 스타일정보를 RESET 해준다.
|
|
// 주의: 크롬의 경우, css 로딩이 더 먼저 발생해서 SE_WYSIWYGStylerGetter 플러그인에서 오류가 발생할 수 있기 때문에 RESET_STYLE_STATUS 메시지 호출이 가능한 상태인지 체크함
|
|
if(this.oApp && this.oApp.getEditingMode && this.oApp.getEditingMode() === this.sMode){
|
|
this.oApp.exec("RESET_STYLE_STATUS");
|
|
}
|
|
/*
|
|
* [SMARTEDITORSUS-2298]
|
|
* IE에서 웹폰트용 css가 import될때 이벤트핸들러가 실행되어 툴바에 선택된 폰트가 리셋되는 문제가 있음
|
|
* 때문에 한번 실행되고 난 후에는 연결된 이벤트핸들러를 클리어처리함
|
|
*/
|
|
linkNode.onload = null;
|
|
}, this).bind();
|
|
headNode.appendChild(linkNode);
|
|
}
|
|
|
|
this._enableWYSIWYG();
|
|
|
|
this.status = nhn.husky.PLUGIN_STATUS.READY;
|
|
} catch(e) {
|
|
if(this._nIFrameReadyCount-- > 0){
|
|
setTimeout(jindo.$Fn(this.initIframe, this).bind(), 100);
|
|
}else{
|
|
throw("iframe for WYSIWYG editing mode can't be initialized. Please check if the iframe document exists and is also accessable(cross-domain issues). ");
|
|
}
|
|
}
|
|
},
|
|
|
|
getIR : function(){
|
|
var sContent = this.iframe.contentWindow.document.body.innerHTML,
|
|
sIR;
|
|
|
|
if(this.oApp.applyConverter){
|
|
sIR = this.oApp.applyConverter(this.sMode+"_TO_IR", sContent, this.oApp.getWYSIWYGDocument());
|
|
}else{
|
|
sIR = sContent;
|
|
}
|
|
|
|
return sIR;
|
|
},
|
|
|
|
setIR : function(sIR){
|
|
// [SMARTEDITORSUS-875] HTML 모드의 beautify에서 추가된 공백을 다시 제거
|
|
//sIR = sIR.replace(/(>)([\n\r\t\s]*)([^<]?)/g, "$1$3").replace(/([\n\r\t\s]*)(<)/g, "$2")
|
|
// --[SMARTEDITORSUS-875]
|
|
|
|
var sContent,
|
|
oNavigator = this.oApp.oNavigator,
|
|
bUnderIE11 = oNavigator.ie && document.documentMode < 11, // IE11미만
|
|
sCursorHolder = bUnderIE11 ? "" : "<br>";
|
|
|
|
if(this.oApp.applyConverter){
|
|
sContent = this.oApp.applyConverter("IR_TO_"+this.sMode, sIR, this.oApp.getWYSIWYGDocument());
|
|
}else{
|
|
sContent = sIR;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1279] [IE9/10] pre 태그 아래에 \n이 포함되면 개행이 되지 않는 이슈
|
|
/*if(oNavigator.ie && oNavigator.nativeVersion >= 9 && document.documentMode >= 9){
|
|
// [SMARTEDITORSUS-704] \r\n이 있는 경우 IE9 표준모드에서 정렬 시 브라우저가 <p>를 추가하는 문제
|
|
sContent = sContent.replace(/[\r\n]/g,"");
|
|
}*/
|
|
|
|
// 편집내용이 없는 경우 커서홀더로 대체
|
|
if(sContent.replace(/[\r\n\t\s]*/,"") === ""){
|
|
if(this.oApp.sLineBreaker !== "BR"){
|
|
sCursorHolder = "<p>" + sCursorHolder + "</p>";
|
|
}
|
|
sContent = sCursorHolder;
|
|
}
|
|
this.iframe.contentWindow.document.body.innerHTML = sContent;
|
|
|
|
// [COM-1142] IE의 경우 <p> </p> 를 <p></p> 로 변환
|
|
// [SMARTEDITORSUS-1623] IE11은 <p></p>로 변환하면 라인이 붙어버리기 때문에 IE10만 적용하도록 수정
|
|
if(bUnderIE11 && this.oApp.getEditingMode() === this.sMode){
|
|
var pNodes = this.oApp.getWYSIWYGDocument().body.getElementsByTagName("P");
|
|
|
|
for(var i=0, nMax = pNodes.length; i < nMax; i++){
|
|
if(pNodes[i].childNodes.length === 1 && pNodes[i].innerHTML === " "){
|
|
pNodes[i].innerHTML = '';
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
getRawContents : function(){
|
|
return this.iframe.contentWindow.document.body.innerHTML;
|
|
},
|
|
|
|
getRawHTMLContents : function(){
|
|
return this.getRawContents();
|
|
},
|
|
|
|
setRawHTMLContents : function(sContents){
|
|
this.iframe.contentWindow.document.body.innerHTML = sContents;
|
|
},
|
|
|
|
getWindow : function(){
|
|
return this.iframe.contentWindow;
|
|
},
|
|
|
|
getDocument : function(){
|
|
return this.iframe.contentWindow.document;
|
|
},
|
|
|
|
focus : function(){
|
|
//this.getWindow().focus();
|
|
this.getDocument().body.focus();
|
|
this.oApp.exec("RESTORE_IE_SELECTION");
|
|
},
|
|
|
|
_recordUndo : function(oKeyInfo){
|
|
/**
|
|
* 229: Korean/Eng
|
|
* 16: shift
|
|
* 33,34: page up/down
|
|
* 35,36: end/home
|
|
* 37,38,39,40: left, up, right, down
|
|
* 32: space
|
|
* 46: delete
|
|
* 8: bksp
|
|
*/
|
|
if(oKeyInfo.keyCode >= 33 && oKeyInfo.keyCode <= 40){ // record snapshot
|
|
this.oApp.saveSnapShot();
|
|
return;
|
|
}
|
|
|
|
if(oKeyInfo.alt || oKeyInfo.ctrl || oKeyInfo.keyCode === 16){
|
|
return;
|
|
}
|
|
|
|
if(this.oApp.getLastKey() === oKeyInfo.keyCode){
|
|
return;
|
|
}
|
|
|
|
this.oApp.setLastKey(oKeyInfo.keyCode);
|
|
|
|
// && oKeyInfo.keyCode != 32 // 속도 문제로 인하여 Space 는 제외함
|
|
if(!oKeyInfo.enter && oKeyInfo.keyCode !== 46 && oKeyInfo.keyCode !== 8){
|
|
return;
|
|
}
|
|
|
|
this.oApp.exec("RECORD_UNDO_ACTION", ["KEYPRESS(" + oKeyInfo.keyCode + ")", {bMustBlockContainer:true}]);
|
|
},
|
|
|
|
_enableWYSIWYG : function(){
|
|
//if (this.iframe.contentWindow.document.body.hasOwnProperty("contentEditable")){
|
|
if (this.iframe.contentWindow.document.body.contentEditable !== null) {
|
|
this.iframe.contentWindow.document.body.contentEditable = true;
|
|
} else {
|
|
this.iframe.contentWindow.document.designMode = "on";
|
|
}
|
|
|
|
this.bWYSIWYGEnabled = true;
|
|
if(jindo.$Agent().navigator().firefox){
|
|
setTimeout(jindo.$Fn(function(){
|
|
//enableInlineTableEditing : Enables or disables the table row and column insertion and deletion controls.
|
|
this.iframe.contentWindow.document.execCommand('enableInlineTableEditing', false, false);
|
|
}, this).bind(), 0);
|
|
}
|
|
},
|
|
|
|
_disableWYSIWYG : function(){
|
|
//if (this.iframe.contentWindow.document.body.hasOwnProperty("contentEditable")){
|
|
if (this.iframe.contentWindow.document.body.contentEditable !== null){
|
|
this.iframe.contentWindow.document.body.contentEditable = false;
|
|
} else {
|
|
this.iframe.contentWindow.document.designMode = "off";
|
|
}
|
|
this.bWYSIWYGEnabled = false;
|
|
},
|
|
|
|
isWYSIWYGEnabled : function(){
|
|
return this.bWYSIWYGEnabled;
|
|
}
|
|
});
|
|
//}
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations directly related to editing the HTML source code using Textarea element
|
|
* @name hp_SE_EditingArea_HTMLSrc.js
|
|
* @required SE_EditingAreaManager
|
|
*/
|
|
nhn.husky.SE_EditingArea_HTMLSrc = jindo.$Class({
|
|
name : "SE_EditingArea_HTMLSrc",
|
|
sMode : "HTMLSrc",
|
|
bAutoResize : false, // [SMARTEDITORSUS-677] 해당 편집모드의 자동확장 기능 On/Off 여부
|
|
nMinHeight : null, // [SMARTEDITORSUS-677] 편집 영역의 최소 높이
|
|
|
|
$init : function(sTextArea) {
|
|
this.elEditingArea = jindo.$(sTextArea);
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function() {
|
|
this.oNavigator = jindo.$Agent().navigator();
|
|
this.oApp.exec("REGISTER_EDITING_AREA", [this]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function() {
|
|
if(!!this.oApp.getEditingAreaHeight){
|
|
this.nMinHeight = this.oApp.getEditingAreaHeight(); // [SMARTEDITORSUS-677] 편집 영역의 최소 높이를 가져와 자동 확장 처리를 할 때 사용
|
|
}
|
|
},
|
|
|
|
$ON_CHANGE_EDITING_MODE : function(sMode) {
|
|
if (sMode == this.sMode) {
|
|
this.elEditingArea.style.display = "block";
|
|
/**
|
|
* [SMARTEDITORSUS-1889] Editor 영역을 표시하고 숨기는 데 있어서
|
|
* display 속성 대신 visibility 속성을 사용하게 되면서,
|
|
* Editor 영역이 화면에서 사라지지만
|
|
* 공간을 차지하게 되므로
|
|
* 그 아래로 위치하는 HTML 영역을 끌어올려 준다.
|
|
*
|
|
* @see hp_SE_EditingArea_WYSIWYG.js
|
|
* */
|
|
this.elEditingArea.style.position = "absolute";
|
|
this.elEditingArea.style.top = "0px";
|
|
// --[SMARTEDITORSUS-1889]
|
|
} else {
|
|
this.elEditingArea.style.display = "none";
|
|
// [SMARTEDITORSUS-1889]
|
|
this.elEditingArea.style.position = "";
|
|
this.elEditingArea.style.top = "";
|
|
// --[SMARTEDITORSUS-1889]
|
|
}
|
|
},
|
|
|
|
$AFTER_CHANGE_EDITING_MODE : function(sMode, bNoFocus) {
|
|
if (sMode == this.sMode && !bNoFocus) {
|
|
var o = new TextRange(this.elEditingArea);
|
|
o.setSelection(0, 0);
|
|
|
|
//[SMARTEDITORSUS-1017] [iOS5대응] 모드 전환 시 textarea에 포커스가 있어도 글자가 입력이 안되는 현상
|
|
//원인 : WYSIWYG모드가 아닐 때에도 iframe의 contentWindow에 focus가 가면서 focus기능이 작동하지 않음
|
|
//해결 : WYSIWYG모드 일때만 실행 되도록 조건식 추가 및 기존에 blur처리 코드 삭제
|
|
//모바일 textarea에서는 직접 클릭을해야만 키보드가 먹히기 때문에 우선은 커서가 안보이게 해서 사용자가 직접 클릭을 유도.
|
|
// if(!!this.oNavigator.msafari){
|
|
// this.elEditingArea.blur();
|
|
// }
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] HTML 편집 영역 자동 확장 처리 시작
|
|
*/
|
|
startAutoResize : function(){
|
|
var htOption = {
|
|
nMinHeight : this.nMinHeight,
|
|
wfnCallback : jindo.$Fn(this.oApp.checkResizeGripPosition, this).bind()
|
|
};
|
|
//[SMARTEDITORSUS-941][iOS5대응]아이패드의 자동 확장 기능이 동작하지 않을 때 에디터 창보다 긴 내용을 작성하면 에디터를 뚫고 나오는 현상
|
|
//원인 : 자동확장 기능이 정지 될 경우 iframe에 스크롤이 생기지 않고, 창을 뚫고 나옴
|
|
//해결 : 항상 자동확장 기능이 켜져있도록 변경. 자동 확장 기능 관련한 이벤트 코드도 모바일 사파리에서 예외 처리
|
|
if(this.oNavigator.msafari){
|
|
htOption.wfnCallback = function(){};
|
|
}
|
|
|
|
this.bAutoResize = true;
|
|
this.AutoResizer = new nhn.husky.AutoResizer(this.elEditingArea, htOption);
|
|
this.AutoResizer.bind();
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] HTML 편집 영역 자동 확장 처리 종료
|
|
*/
|
|
stopAutoResize : function(){
|
|
this.AutoResizer.unbind();
|
|
},
|
|
|
|
getIR : function() {
|
|
var sIR = this.getRawContents();
|
|
if (this.oApp.applyConverter) {
|
|
sIR = this.oApp.applyConverter(this.sMode + "_TO_IR", sIR, this.oApp.getWYSIWYGDocument());
|
|
}
|
|
|
|
return sIR;
|
|
},
|
|
|
|
setIR : function(sIR) {
|
|
if(sIR.toLowerCase() === "<br>" || sIR.toLowerCase() === "<p> </p>" || sIR.toLowerCase() === "<p><br></p>" || sIR.toLowerCase() === "<p></p>"){
|
|
sIR="";
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1589] 문서 모드가 Edge인 IE11에서 WYSIWYG 모드와 HTML 모드 전환 시, 문말에 무의미한 <br> 두 개가 첨가되는 현상으로 필터링 추가
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if(htBrowser.ie && htBrowser.nativeVersion == 11 && document.documentMode == 11){ // Edge 모드의 documentMode 값은 11
|
|
sIR = sIR.replace(/(<br><br>$)/, "");
|
|
}
|
|
// --[SMARTEDITORSUS-1589]
|
|
|
|
var sContent = sIR;
|
|
if (this.oApp.applyConverter) {
|
|
sContent = this.oApp.applyConverter("IR_TO_" + this.sMode, sContent, this.oApp.getWYSIWYGDocument());
|
|
}
|
|
|
|
this.setRawContents(sContent);
|
|
},
|
|
|
|
setRawContents : function(sContent) {
|
|
if (typeof sContent !== 'undefined') {
|
|
this.elEditingArea.value = sContent;
|
|
}
|
|
},
|
|
|
|
getRawContents : function() {
|
|
return this.elEditingArea.value;
|
|
},
|
|
|
|
focus : function() {
|
|
this.elEditingArea.focus();
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Selection for textfield
|
|
* @author hooriza
|
|
*/
|
|
if (typeof window.TextRange == 'undefined') { window.TextRange = {}; }
|
|
TextRange = function(oEl, oDoc) {
|
|
this._o = oEl;
|
|
this._oDoc = (oDoc || document);
|
|
};
|
|
|
|
TextRange.prototype.getSelection = function() {
|
|
var obj = this._o;
|
|
var ret = [-1, -1];
|
|
|
|
if(isNaN(this._o.selectionStart)) {
|
|
obj.focus();
|
|
|
|
// textarea support added by nagoon97
|
|
var range = this._oDoc.body.createTextRange();
|
|
var rangeField = null;
|
|
|
|
rangeField = this._oDoc.selection.createRange().duplicate();
|
|
range.moveToElementText(obj);
|
|
rangeField.collapse(true);
|
|
range.setEndPoint("EndToEnd", rangeField);
|
|
ret[0] = range.text.length;
|
|
|
|
rangeField = this._oDoc.selection.createRange().duplicate();
|
|
range.moveToElementText(obj);
|
|
rangeField.collapse(false);
|
|
range.setEndPoint("EndToEnd", rangeField);
|
|
ret[1] = range.text.length;
|
|
|
|
obj.blur();
|
|
} else {
|
|
ret[0] = obj.selectionStart;
|
|
ret[1] = obj.selectionEnd;
|
|
}
|
|
|
|
return ret;
|
|
};
|
|
|
|
TextRange.prototype.setSelection = function(start, end) {
|
|
var obj = this._o;
|
|
if (typeof end == 'undefined') {
|
|
end = start;
|
|
}
|
|
|
|
if (obj.setSelectionRange) {
|
|
obj.setSelectionRange(start, end);
|
|
} else if (obj.createTextRange) {
|
|
var range = obj.createTextRange();
|
|
range.collapse(true);
|
|
range.moveStart("character", start);
|
|
range.moveEnd("character", end - start);
|
|
range.select();
|
|
obj.blur();
|
|
}
|
|
};
|
|
|
|
TextRange.prototype.copy = function() {
|
|
var r = this.getSelection();
|
|
return this._o.value.substring(r[0], r[1]);
|
|
};
|
|
|
|
TextRange.prototype.paste = function(sStr) {
|
|
var obj = this._o;
|
|
var sel = this.getSelection();
|
|
var value = obj.value;
|
|
var pre = value.substr(0, sel[0]);
|
|
var post = value.substr(sel[1]);
|
|
|
|
value = pre + sStr + post;
|
|
obj.value = value;
|
|
|
|
var n = 0;
|
|
if (typeof this._oDoc.body.style.maxHeight == "undefined") {
|
|
var a = pre.match(/\n/gi);
|
|
n = ( a !== null ? a.length : 0 );
|
|
}
|
|
|
|
this.setSelection(sel[0] + sStr.length - n);
|
|
};
|
|
|
|
TextRange.prototype.cut = function() {
|
|
var r = this.copy();
|
|
this.paste('');
|
|
return r;
|
|
};
|
|
//}
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations directly related to editing the HTML source code using Textarea element
|
|
* @name hp_SE_EditingArea_TEXT.js
|
|
* @required SE_EditingAreaManager
|
|
*/
|
|
nhn.husky.SE_EditingArea_TEXT = jindo.$Class({
|
|
name : "SE_EditingArea_TEXT",
|
|
sMode : "TEXT",
|
|
sRxConverter : '@[0-9]+@',
|
|
bAutoResize : false, // [SMARTEDITORSUS-677] 해당 편집모드의 자동확장 기능 On/Off 여부
|
|
nMinHeight : null, // [SMARTEDITORSUS-677] 편집 영역의 최소 높이
|
|
|
|
$init : function(sTextArea) {
|
|
this.elEditingArea = jindo.$(sTextArea);
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function() {
|
|
this.oNavigator = jindo.$Agent().navigator();
|
|
this.oApp.exec("REGISTER_EDITING_AREA", [this]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getTextAreaContents", jindo.$Fn(this.getRawContents, this).bind()]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function() {
|
|
if(!!this.oApp.getEditingAreaHeight){
|
|
this.nMinHeight = this.oApp.getEditingAreaHeight(); // [SMARTEDITORSUS-677] 편집 영역의 최소 높이를 가져와 자동 확장 처리를 할 때 사용
|
|
}
|
|
},
|
|
|
|
$ON_REGISTER_CONVERTERS : function() {
|
|
this.oApp.exec("ADD_CONVERTER", ["IR_TO_TEXT", jindo.$Fn(this.irToText, this).bind()]);
|
|
this.oApp.exec("ADD_CONVERTER", ["TEXT_TO_IR", jindo.$Fn(this.textToIr, this).bind()]);
|
|
},
|
|
|
|
$ON_CHANGE_EDITING_MODE : function(sMode) {
|
|
if (sMode == this.sMode) {
|
|
this.elEditingArea.style.display = "block";
|
|
/**
|
|
* [SMARTEDITORSUS-1889] Editor 영역을 표시하고 숨기는 데 있어서
|
|
* display 속성 대신 visibility 속성을 사용하게 되면서,
|
|
* Editor 영역이 화면에서 사라지지만
|
|
* 공간을 차지하게 되므로
|
|
* 그 아래로 위치하는 Text 영역을 끌어올려 준다.
|
|
*
|
|
* @see hp_SE_EditingArea_WYSIWYG.js
|
|
* */
|
|
this.elEditingArea.style.position = "absolute";
|
|
this.elEditingArea.style.top = "0px";
|
|
// --[SMARTEDITORSUS-1889]
|
|
} else {
|
|
this.elEditingArea.style.display = "none";
|
|
// [SMARTEDITORSUS-1889]
|
|
this.elEditingArea.style.position = "";
|
|
this.elEditingArea.style.top = "";
|
|
// --[SMARTEDITORSUS-1889]
|
|
}
|
|
},
|
|
|
|
$AFTER_CHANGE_EDITING_MODE : function(sMode, bNoFocus) {
|
|
if (sMode == this.sMode && !bNoFocus) {
|
|
var o = new TextRange(this.elEditingArea);
|
|
o.setSelection(0, 0);
|
|
}
|
|
|
|
//[SMARTEDITORSUS-1017] [iOS5대응] 모드 전환 시 textarea에 포커스가 있어도 글자가 입력이 안되는 현상
|
|
//원인 : WYSIWYG모드가 아닐 때에도 iframe의 contentWindow에 focus가 가면서 focus기능이 작동하지 않음
|
|
//해결 : WYSIWYG모드 일때만 실행 되도록 조건식 추가 및 기존에 blur처리 코드 삭제
|
|
//모바일 textarea에서는 직접 클릭을해야만 키보드가 먹히기 때문에 우선은 커서가 안보이게 해서 사용자가 직접 클릭을 유도.
|
|
// if(!!this.oNavigator.msafari){
|
|
// this.elEditingArea.blur();
|
|
// }
|
|
},
|
|
|
|
irToText : function(sHtml) {
|
|
var sContent = sHtml, nIdx = 0;
|
|
var aTemp = sContent.match(new RegExp(this.sRxConverter)); // applyConverter에서 추가한 sTmpStr를 잠시 제거해준다.
|
|
if (aTemp !== null) {
|
|
sContent = sContent.replace(new RegExp(this.sRxConverter), "");
|
|
}
|
|
|
|
//0.안보이는 값들에 대한 정리. (에디터 모드에 view와 text모드의 view를 동일하게 해주기 위해서)
|
|
sContent = sContent.replace(/\r/g, '');// MS엑셀 테이블에서 tr별로 분리해주는 역할이\r이기 때문에 text모드로 변경시에 가독성을 위해 \r 제거하는 것은 임시 보류. - 11.01.28 by cielo
|
|
sContent = sContent.replace(/[\n|\t]/g, ''); // 개행문자, 안보이는 공백 제거
|
|
sContent = sContent.replace(/[\v|\f]/g, ''); // 개행문자, 안보이는 공백 제거
|
|
//1. 먼저, 빈 라인 처리 .
|
|
sContent = sContent.replace(/<p><br><\/p>/gi, '\n');
|
|
sContent = sContent.replace(/<P> <\/P>/gi, '\n');
|
|
|
|
//2. 빈 라인 이외에 linebreak 처리.
|
|
sContent = sContent.replace(/<br(\s)*\/?>/gi, '\n'); // br 태그를 개행문자로
|
|
sContent = sContent.replace(/<br(\s[^\/]*)?>/gi, '\n'); // br 태그를 개행문자로
|
|
sContent = sContent.replace(/<\/p(\s[^\/]*)?>/gi, '\n'); // p 태그를 개행문자로
|
|
|
|
sContent = sContent.replace(/<\/li(\s[^\/]*)?>/gi, '\n'); // li 태그를 개행문자로 [SMARTEDITORSUS-107]개행 추가
|
|
sContent = sContent.replace(/<\/tr(\s[^\/]*)?>/gi, '\n'); // tr 태그를 개행문자로 [SMARTEDITORSUS-107]개행 추가
|
|
|
|
// 마지막 \n은 로직상 불필요한 linebreak를 제공하므로 제거해준다.
|
|
nIdx = sContent.lastIndexOf('\n');
|
|
if (nIdx > -1 && sContent.substring(nIdx) == '\n') {
|
|
sContent = sContent.substring(0, nIdx);
|
|
}
|
|
|
|
sContent = jindo.$S(sContent).stripTags().toString();
|
|
sContent = this.unhtmlSpecialChars(sContent);
|
|
if (aTemp !== null) { // 제거했던sTmpStr를 추가해준다.
|
|
sContent = aTemp[0] + sContent;
|
|
}
|
|
|
|
return sContent;
|
|
},
|
|
|
|
textToIr : function(sHtml) {
|
|
if (!sHtml) {
|
|
return;
|
|
}
|
|
|
|
var sContent = sHtml, aTemp = null;
|
|
|
|
// applyConverter에서 추가한 sTmpStr를 잠시 제거해준다. sTmpStr도 하나의 string으로 인식하는 경우가 있기 때문.
|
|
aTemp = sContent.match(new RegExp(this.sRxConverter));
|
|
if (aTemp !== null) {
|
|
sContent = sContent.replace(aTemp[0], "");
|
|
}
|
|
|
|
sContent = this.htmlSpecialChars(sContent);
|
|
sContent = this._addLineBreaker(sContent);
|
|
|
|
if (aTemp !== null) {
|
|
sContent = aTemp[0] + sContent;
|
|
}
|
|
|
|
return sContent;
|
|
},
|
|
|
|
_addLineBreaker : function(sContent){
|
|
if(this.oApp.sLineBreaker === "BR"){
|
|
return sContent.replace(/\r?\n/g, "<BR>");
|
|
}
|
|
|
|
var oContent = new StringBuffer(),
|
|
aContent = sContent.split('\n'), // \n을 기준으로 블럭을 나눈다.
|
|
aContentLng = aContent.length,
|
|
sTemp = "";
|
|
|
|
for (var i = 0; i < aContentLng; i++) {
|
|
sTemp = jindo.$S(aContent[i]).trim().$value();
|
|
if (i === aContentLng -1 && sTemp === "") {
|
|
break;
|
|
}
|
|
|
|
if (sTemp !== null && sTemp !== "") {
|
|
oContent.append('<P>');
|
|
oContent.append(aContent[i]);
|
|
oContent.append('</P>');
|
|
} else {
|
|
if (!jindo.$Agent().navigator().ie) {
|
|
oContent.append('<P><BR></P>');
|
|
} else {
|
|
oContent.append('<P> <\/P>');
|
|
}
|
|
}
|
|
}
|
|
|
|
return oContent.toString();
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] HTML 편집 영역 자동 확장 처리 시작
|
|
*/
|
|
startAutoResize : function(){
|
|
var htOption = {
|
|
nMinHeight : this.nMinHeight,
|
|
wfnCallback : jindo.$Fn(this.oApp.checkResizeGripPosition, this).bind()
|
|
};
|
|
|
|
//[SMARTEDITORSUS-941][iOS5대응]아이패드의 자동 확장 기능이 동작하지 않을 때 에디터 창보다 긴 내용을 작성하면 에디터를 뚫고 나오는 현상
|
|
//원인 : 자동확장 기능이 정지 될 경우 iframe에 스크롤이 생기지 않고, 창을 뚫고 나옴
|
|
//해결 : 항상 자동확장 기능이 켜져있도록 변경. 자동 확장 기능 관련한 이벤트 코드도 모바일 사파리에서 예외 처리
|
|
if(this.oNavigator.msafari){
|
|
htOption.wfnCallback = function(){};
|
|
}
|
|
|
|
this.bAutoResize = true;
|
|
this.AutoResizer = new nhn.husky.AutoResizer(this.elEditingArea, htOption);
|
|
this.AutoResizer.bind();
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] HTML 편집 영역 자동 확장 처리 종료
|
|
*/
|
|
stopAutoResize : function(){
|
|
this.AutoResizer.unbind();
|
|
},
|
|
|
|
getIR : function() {
|
|
var sIR = this.getRawContents();
|
|
if (this.oApp.applyConverter) {
|
|
sIR = this.oApp.applyConverter(this.sMode + "_TO_IR", sIR, this.oApp.getWYSIWYGDocument());
|
|
}
|
|
return sIR;
|
|
},
|
|
|
|
setIR : function(sIR) {
|
|
var sContent = sIR;
|
|
if (this.oApp.applyConverter) {
|
|
sContent = this.oApp.applyConverter("IR_TO_" + this.sMode, sContent, this.oApp.getWYSIWYGDocument());
|
|
}
|
|
|
|
this.setRawContents(sContent);
|
|
},
|
|
|
|
setRawContents : function(sContent) {
|
|
if (typeof sContent !== 'undefined') {
|
|
this.elEditingArea.value = sContent;
|
|
}
|
|
},
|
|
|
|
getRawContents : function() {
|
|
return this.elEditingArea.value;
|
|
},
|
|
|
|
focus : function() {
|
|
this.elEditingArea.focus();
|
|
},
|
|
|
|
/**
|
|
* HTML 태그에 해당하는 글자가 먹히지 않도록 바꿔주기
|
|
*
|
|
* 동작) & 를 & 로, < 를 < 로, > 를 > 로 바꿔준다
|
|
*
|
|
* @param {String} sText
|
|
* @return {String}
|
|
*/
|
|
htmlSpecialChars : function(sText) {
|
|
return sText.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>').replace(/ /g, ' ');
|
|
},
|
|
|
|
/**
|
|
* htmlSpecialChars 의 반대 기능의 함수
|
|
*
|
|
* 동작) &, <, >,   를 각각 &, <, >, 빈칸으로 바꿔준다
|
|
*
|
|
* @param {String} sText
|
|
* @return {String}
|
|
*/
|
|
unhtmlSpecialChars : function(sText) {
|
|
return sText.replace(/</g, '<').replace(/>/g, '>').replace(/ /g, ' ').replace(/&/g, '&');
|
|
}
|
|
});
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to resizing the editing area vertically
|
|
* @name hp_SE_EditingAreaVerticalResizer.js
|
|
*/
|
|
nhn.husky.SE_EditingAreaVerticalResizer = jindo.$Class({
|
|
name : "SE_EditingAreaVerticalResizer",
|
|
|
|
oResizeGrip : null,
|
|
sCookieNotice : "bHideResizeNotice",
|
|
|
|
nEditingAreaMinHeight : null, // [SMARTEDITORSUS-677] 편집 영역의 최소 높이
|
|
htConversionMode : null,
|
|
|
|
$init : function(elAppContainer, htConversionMode){
|
|
this.htConversionMode = htConversionMode;
|
|
this._assignHTMLElements(elAppContainer);
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["isUseVerticalResizer", jindo.$Fn(this.isUseVerticalResizer, this).bind()]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
if(this.oApp.bMobile){
|
|
// [SMARTEDITORSUS-941] 모바일에서는 자동확장기능이 항상 켜져있도록 한다.
|
|
// [SMARTEDITORSUS-1679] 하지만 사용자가 조절하지는 못하도록 버튼은 비활성화 한다.
|
|
this.oResizeGrip.disabled = true;
|
|
this.oResizeGrip.style.height = '0'; // 버튼의 문구를 가림. display:none을 하면 안드로이드에서 높이 계산오류 발생
|
|
}else{
|
|
this.oApp.exec("REGISTER_HOTKEY", ["shift+esc", "FOCUS_RESIZER"]);
|
|
|
|
// [SMARTEDITORSUS-906][SMARTEDITORSUS-1433] Resizbar 사용 여부 처리 (true:사용함/ false:사용하지 않음)
|
|
if(this.isUseVerticalResizer()){
|
|
this.oResizeGrip.style.display = 'block';
|
|
if(!!this.welNoticeLayer && !Number(jindo.$Cookie().get(this.sCookieNotice))){
|
|
this.welNoticeLayer.delegate("click", "BUTTON.bt_clse", jindo.$Fn(this._closeNotice, this).bind());
|
|
this.welNoticeLayer.show();
|
|
}
|
|
this.$FnMouseDown = jindo.$Fn(this._mousedown, this);
|
|
this.$FnMouseMove = jindo.$Fn(this._mousemove, this);
|
|
this.$FnMouseUp = jindo.$Fn(this._mouseup, this);
|
|
this.$FnMouseOver = jindo.$Fn(this._mouseover, this);
|
|
this.$FnMouseOut = jindo.$Fn(this._mouseout, this);
|
|
|
|
this.$FnMouseDown.attach(this.oResizeGrip, "mousedown");
|
|
this.$FnMouseOver.attach(this.oResizeGrip, "mouseover");
|
|
this.$FnMouseOut.attach(this.oResizeGrip, "mouseout");
|
|
|
|
}else{
|
|
this.oResizeGrip.style.display = 'none';
|
|
if(!this.oApp.isUseModeChanger()){
|
|
this.elModeToolbar.style.display = "none";
|
|
}
|
|
}
|
|
}
|
|
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["checkResizeGripPosition", jindo.$Fn(this.checkResizeGripPosition, this).bind()]); // [SMARTEDITORSUS-677]
|
|
|
|
if(!!this.oApp.getEditingAreaHeight){
|
|
this.nEditingAreaMinHeight = this.oApp.getEditingAreaHeight(); // [SMARTEDITORSUS-677] 편집 영역의 최소 높이를 가져와 Gap 처리 시 사용
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2036][SMARTEDITORSUS-1585] DISABLE_ALL_UI 메시지가 발생하면 기능을 비활성화한다.
|
|
*/
|
|
$ON_DISABLE_ALL_UI : function(){
|
|
this.oResizeGrip.style.cursor = "default";
|
|
this.welConversionMode.addClass("off");
|
|
this.oResizeGrip.disabled = true;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2036][SMARTEDITORSUS-1585] ENABLE_ALL_UI 메시지가 발생하면 기능을 활성화한다.
|
|
*/
|
|
$ON_ENABLE_ALL_UI : function(){
|
|
this.oResizeGrip.style.cursor = "n-resize";
|
|
this.welConversionMode.removeClass("off");
|
|
this.oResizeGrip.disabled = false;
|
|
},
|
|
|
|
isUseVerticalResizer : function(){
|
|
return (typeof(this.htConversionMode) === 'undefined' || typeof(this.htConversionMode.bUseVerticalResizer) === 'undefined' || this.htConversionMode.bUseVerticalResizer === true) ? true : false;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-677] [에디터 자동확장 ON인 경우]
|
|
* 입력창 크기 조절 바의 위치를 확인하여 브라우저 하단에 위치한 경우 자동확장을 멈춤
|
|
*/
|
|
checkResizeGripPosition : function(bExpand){
|
|
var oDocument = jindo.$Document();
|
|
var nGap = (jindo.$Element(this.oResizeGrip).offset().top - oDocument.scrollPosition().top + 25) - oDocument.clientSize().height;
|
|
|
|
if(nGap <= 0){
|
|
return;
|
|
}
|
|
|
|
if(bExpand){
|
|
if(this.nEditingAreaMinHeight > this.oApp.getEditingAreaHeight() - nGap){ // [SMARTEDITORSUS-822] 수정 모드인 경우에 대비
|
|
nGap = (-1) * (this.nEditingAreaMinHeight - this.oApp.getEditingAreaHeight());
|
|
}
|
|
|
|
// Gap 만큼 편집영역 사이즈를 조절하여
|
|
// 사진 첨부나 붙여넣기 등의 사이즈가 큰 내용 추가가 있었을 때 입력창 크기 조절 바가 숨겨지지 않도록 함
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_STARTED");
|
|
this.oApp.exec("RESIZE_EDITING_AREA_BY", [0, (-1) * nGap]);
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_ENDED");
|
|
}
|
|
|
|
this.oApp.exec("STOP_AUTORESIZE_EDITING_AREA");
|
|
},
|
|
|
|
$ON_FOCUS_RESIZER : function(){
|
|
this.oApp.exec("IE_HIDE_CURSOR");
|
|
this.oResizeGrip.focus();
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer, htConversionMode){
|
|
//@ec[
|
|
this.oResizeGrip = jindo.$$.getSingle("BUTTON.husky_seditor_editingArea_verticalResizer", elAppContainer);
|
|
this.elModeToolbar = jindo.$$.getSingle("DIV.se2_conversion_mode", elAppContainer);
|
|
//@ec]
|
|
|
|
this.welNoticeLayer = jindo.$Element(jindo.$$.getSingle("DIV.husky_seditor_resize_notice", elAppContainer));
|
|
this.welConversionMode = jindo.$Element(this.oResizeGrip.parentNode);
|
|
},
|
|
|
|
_mouseover : function(oEvent){
|
|
oEvent.stopBubble();
|
|
this.welConversionMode.addClass("controller_on");
|
|
},
|
|
|
|
_mouseout : function(oEvent){
|
|
oEvent.stopBubble();
|
|
this.welConversionMode.removeClass("controller_on");
|
|
},
|
|
|
|
_mousedown : function(oEvent){
|
|
this.iStartHeight = oEvent.pos().clientY;
|
|
this.iStartHeightOffset = oEvent.pos().layerY;
|
|
|
|
this.$FnMouseMove.attach(document, "mousemove");
|
|
this.$FnMouseUp.attach(document, "mouseup");
|
|
|
|
this.iStartHeight = oEvent.pos().clientY;
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
this.oApp.exec("HIDE_ALL_DIALOG_LAYER");
|
|
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_STARTED", [this.$FnMouseDown, this.$FnMouseMove, this.$FnMouseUp]);
|
|
},
|
|
|
|
_mousemove : function(oEvent){
|
|
var iHeightChange = oEvent.pos().clientY - this.iStartHeight;
|
|
|
|
this.oApp.exec("RESIZE_EDITING_AREA_BY", [0, iHeightChange]);
|
|
},
|
|
|
|
_mouseup : function(oEvent){
|
|
this.$FnMouseMove.detach(document, "mousemove");
|
|
this.$FnMouseUp.detach(document, "mouseup");
|
|
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_ENDED", [this.$FnMouseDown, this.$FnMouseMove, this.$FnMouseUp]);
|
|
},
|
|
|
|
_closeNotice : function(){
|
|
this.welNoticeLayer.hide();
|
|
jindo.$Cookie().set(this.sCookieNotice, 1, 365*10);
|
|
}
|
|
});
|
|
//}
|
|
/**
|
|
* @pluginDesc Enter키 입력시에 현재 줄을 P 태그로 감거나 <br> 태그를 삽입한다.
|
|
*/
|
|
nhn.husky.SE_WYSIWYGEnterKey = jindo.$Class({
|
|
name : "SE_WYSIWYGEnterKey",
|
|
|
|
$init : function(sLineBreaker){
|
|
if(sLineBreaker == "BR"){
|
|
this.sLineBreaker = "BR";
|
|
}else{
|
|
this.sLineBreaker = "P";
|
|
}
|
|
|
|
this.htBrowser = jindo.$Agent().navigator();
|
|
|
|
// [SMARTEDITORSUS-227] IE 인 경우에도 에디터 Enter 처리 로직을 사용하도록 수정
|
|
if(this.htBrowser.opera && this.sLineBreaker == "P"){
|
|
this.$ON_MSG_APP_READY = function(){};
|
|
}
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-230] 밑줄+색상변경 후, 엔터치면 스크립트 오류
|
|
* [SMARTEDITORSUS-180] [IE9] 배경색 적용 후, 엔터키 2회이상 입력시 커서위치가 다음 라인으로 이동하지 않음
|
|
* 오류 현상 : IE9 에서 엔터 후 생성된 P 태그가 "빈 SPAN 태그만 가지는 경우" P 태그 영역이 보이지 않거나 포커스가 위로 올라가 보임
|
|
* 해결 방법 : 커서 홀더로 IE 이외에서는 <br> 을 사용
|
|
* - IE 에서는 렌더링 시 <br> 부분에서 비정상적인 P 태그가 생성되어 [SMARTEDITORSUS-230] 오류 발생
|
|
* unescape("%uFEFF") (BOM) 을 추가
|
|
* - IE9 표준모드에서 [SMARTEDITORSUS-180] 의 문제가 발생함
|
|
* (unescape("%u2028") (Line separator) 를 사용하면 P 가 보여지나 사이드이펙트가 우려되어 사용하지 않음)
|
|
* IE 브라우저에서 Enter 처리 시, 를 넣어주므로 해당 방식을 그대로 사용하도록 수정함
|
|
*/
|
|
if(this.htBrowser.ie){
|
|
this._addCursorHolder = this._addCursorHolderSpace;
|
|
|
|
//[SMARTEDITORSUS-1652] 글자크기 지정후 엔터를 치면 빈SPAN으로 감싸지는데 IE에서 빈SPAN은 높이값을 갖지 않아 커서가 올라가 보이게 됨
|
|
// 따라서, IE의 경우 브라우저모드와 상관없이 다음라인의 SPAN에 무조건 ExtraCursorHolder 를 넣어주도록 코멘트처리함
|
|
// if(this.htBrowser.nativeVersion < 9 || document.documentMode < 9){
|
|
// this._addExtraCursorHolder = function(){};
|
|
// }
|
|
}else{
|
|
this._addExtraCursorHolder = function(){};
|
|
this._addBlankText = function(){};
|
|
}
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["sLineBreaker", this.sLineBreaker]);
|
|
|
|
this.oSelection = this.oApp.getEmptySelection();
|
|
this.tmpTextNode = this.oSelection._document.createTextNode(unescape("%u00A0")); // 공백( ) 추가 시 사용할 노드
|
|
jindo.$Fn(this._onKeyDown, this).attach(this.oApp.getWYSIWYGDocument(), "keydown");
|
|
},
|
|
|
|
_onKeyDown : function(oEvent){
|
|
var oKeyInfo = oEvent.key();
|
|
|
|
if(oKeyInfo.shift){
|
|
return;
|
|
}
|
|
|
|
if(oKeyInfo.enter){
|
|
if(this.sLineBreaker == "BR"){
|
|
this._insertBR(oEvent);
|
|
}else{
|
|
this._wrapBlock(oEvent);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-950] 에디터 적용 페이지의 Compatible meta IE=edge 설정 시 줄간격 벌어짐 이슈 (<BR>)
|
|
*/
|
|
$ON_REGISTER_CONVERTERS : function(){
|
|
this.oApp.exec("ADD_CONVERTER", ["IR_TO_DB", jindo.$Fn(this.onIrToDB, this).bind()]);
|
|
},
|
|
|
|
/**
|
|
* IR_TO_DB 변환기 처리
|
|
*/
|
|
onIrToDB : function(sHTML){
|
|
var sContents = sHTML,
|
|
rxEmptyP = /(<p[^>]*>)(?:\s*)(<\/p>)/gi;
|
|
|
|
// [SMARTEDITORSUS-2258] 글작성시 보여지는 그대로 저장이 될 수 있도록 브라우저에 따라 빈 P 태그에 대해 구분처리
|
|
if(this.htBrowser.ie && this.htBrowser.version < 11){
|
|
// IE10이하 인 경우 빈 P 태그는 공백을 넣어준다.
|
|
sContents = sContents.replace(rxEmptyP, "$1 $2");
|
|
}else{
|
|
// 모던브라우저는 빈 P 태그가 높이값을 갖지 않기 때문에 제거해준다.
|
|
sContents = sContents.replace(rxEmptyP, "");
|
|
}
|
|
|
|
return sContents;
|
|
},
|
|
|
|
// [IE] Selection 내의 노드를 가져와 빈 노드에 unescape("%uFEFF") (BOM) 을 추가
|
|
_addBlankText : function(oSelection){
|
|
var oNodes = oSelection.getNodes(),
|
|
i, nLen, oNode, oNodeChild, tmpTextNode;
|
|
|
|
for(i=0, nLen=oNodes.length; i<nLen; i++){
|
|
oNode = oNodes[i];
|
|
|
|
if(oNode.nodeType !== 1 || oNode.tagName !== "SPAN"){
|
|
continue;
|
|
}
|
|
|
|
if(oNode.id.indexOf(oSelection.HUSKY_BOOMARK_START_ID_PREFIX) > -1 ||
|
|
oNode.id.indexOf(oSelection.HUSKY_BOOMARK_END_ID_PREFIX) > -1){
|
|
continue;
|
|
}
|
|
|
|
oNodeChild = oNode.firstChild;
|
|
|
|
if(!oNodeChild ||
|
|
(oNodeChild.nodeType == 3 && nhn.husky.SE2M_Utils.isBlankTextNode(oNodeChild)) ||
|
|
(oNodeChild.nodeType == 1 && oNode.childNodes.length == 1 &&
|
|
(oNodeChild.id.indexOf(oSelection.HUSKY_BOOMARK_START_ID_PREFIX) > -1 || oNodeChild.id.indexOf(oSelection.HUSKY_BOOMARK_END_ID_PREFIX) > -1))){
|
|
tmpTextNode = oSelection._document.createTextNode(unescape("%uFEFF"));
|
|
oNode.appendChild(tmpTextNode);
|
|
}
|
|
}
|
|
},
|
|
|
|
// [IE 이외] 빈 노드 내에 커서를 표시하기 위한 처리
|
|
_addCursorHolder : function(elWrapper){
|
|
var elStyleOnlyNode = elWrapper;
|
|
|
|
if(elWrapper.innerHTML == "" || (elStyleOnlyNode = this._getStyleOnlyNode(elWrapper))){
|
|
elStyleOnlyNode.innerHTML = "<br>";
|
|
}
|
|
if(!elStyleOnlyNode){
|
|
elStyleOnlyNode = this._getStyleNode(elWrapper);
|
|
}
|
|
|
|
return elStyleOnlyNode;
|
|
},
|
|
|
|
// [IE] 빈 노드 내에 커서를 표시하기 위한 처리 (_addSpace 사용)
|
|
_addCursorHolderSpace : function(elWrapper){
|
|
var elNode;
|
|
|
|
this._addSpace(elWrapper);
|
|
|
|
elNode = this._getStyleNode(elWrapper);
|
|
|
|
if(elNode.innerHTML == "" && elNode.nodeName.toLowerCase() != "param"){
|
|
try{
|
|
elNode.innerHTML = unescape("%uFEFF");
|
|
}catch(e) {
|
|
}
|
|
}
|
|
|
|
return elNode;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1513] 시작노드와 끝노드 사이에 첫번째 BR을 찾는다. BR이 없는 경우 끝노드를 반환한다.
|
|
* @param {Node} oStart 검사할 시작노드
|
|
* @param {Node} oEnd 검사할 끝노드
|
|
* @return {Node} 첫번째 BR 혹은 끝노드를 반환한다.
|
|
*/
|
|
_getBlockEndNode : function(oStart, oEnd){
|
|
if(!oStart){
|
|
return oEnd;
|
|
}else if(oStart.nodeName === "BR"){
|
|
return oStart;
|
|
}else if(oStart === oEnd){
|
|
return oEnd;
|
|
}else{
|
|
return this._getBlockEndNode(oStart.nextSibling, oEnd);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1797] 북마크 다음노드가가 텍스트노드인 경우, 문자열 앞쪽의 공백문자(\u0020)를 (\u00A0) 문자로 변환한다.
|
|
* @param {Node} oNode 변환할 텍스트노드
|
|
*/
|
|
_convertHeadSpace : function(oNode){
|
|
if(oNode && oNode.nodeType === 3){
|
|
var sText = oNode.nodeValue, sSpaces = "";
|
|
for(var i = 0, ch;(ch = sText[i]); i++){
|
|
if(ch !== "\u0020"){
|
|
break;
|
|
}
|
|
sSpaces += "\u00A0";
|
|
}
|
|
if(i > 0){
|
|
oNode.nodeValue = sSpaces + sText.substring(i);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 기준요소의 nextSibling을 찾는데 빈텍스트요소이면 그 다음 nextSibling을 찾는다.
|
|
* @param {Node} oNode nextSibling을 찾을 기준요소
|
|
* @returns {Node} 빈텍스트가 아닌 nextSibling 을 반환한다.
|
|
*/
|
|
_getValidNextSibling : function(oNode){
|
|
var oNext = oNode.nextSibling;
|
|
if(!oNext){
|
|
return null;
|
|
}else if(oNext.nodeType == 3 && oNext.nodeValue == ""){
|
|
return arguments.callee(oNext);
|
|
}
|
|
return oNext;
|
|
},
|
|
|
|
_wrapBlock : function(oEvent){
|
|
var oSelection = this.oApp.getSelection(),
|
|
sBM = oSelection.placeStringBookmark(),
|
|
oLineInfo = oSelection.getLineInfo(),
|
|
oStart = oLineInfo.oStart,
|
|
oEnd = oLineInfo.oEnd,
|
|
oSWrapper,
|
|
oEWrapper,
|
|
elStyleOnlyNode;
|
|
|
|
// line broke by sibling
|
|
// or
|
|
// the parent line breaker is just a block container
|
|
if(!oStart.bParentBreak || oSelection.rxBlockContainer.test(oStart.oLineBreaker.tagName)){
|
|
oEvent.stop();
|
|
|
|
// 선택된 내용은 삭제
|
|
oSelection.deleteContents();
|
|
if(!!oStart.oNode.parentNode && oStart.oNode.parentNode.nodeType !== 11){
|
|
// LineBreaker 로 감싸서 분리
|
|
oSWrapper = this.oApp.getWYSIWYGDocument().createElement(this.sLineBreaker);
|
|
oSelection.moveToBookmark(sBM); //oSelection.moveToStringBookmark(sBM, true);
|
|
oSelection.setStartBefore(oStart.oNode);
|
|
oSelection.surroundContents(oSWrapper);
|
|
oSelection.collapseToEnd();
|
|
|
|
oEWrapper = this.oApp.getWYSIWYGDocument().createElement(this.sLineBreaker);
|
|
// [SMARTEDITORSUS-1513] oStart.oNode와 oEnd.oNode 사이에 BR이 있는 경우, 다음 엔터시 스타일이 비정상으로 복사되기 때문에 중간에 BR이 있으면 BR까지만 잘라서 세팅한다.
|
|
var oEndNode = this._getBlockEndNode(oStart.oNode, oEnd.oNode);
|
|
// [SMARTEDITORSUS-1743] oStart.oNode가 BR인 경우, setStartBefore와 setEndAfter에 모두 oStart.oNode로 세팅을 시도하기 때문에 스크립트 오류가 발생한다.
|
|
// 따라서, _getBlockEndNode 메서드를 통해 찾은 BR이 oStart.oNode인 경우, oEnd.oNode 를 세팅한다.
|
|
if(oEndNode === oStart.oNode){
|
|
oEndNode = oEnd.oNode;
|
|
}
|
|
oSelection.setEndAfter(oEndNode);
|
|
this._addBlankText(oSelection);
|
|
oSelection.surroundContents(oEWrapper);
|
|
oSelection.moveToStringBookmark(sBM, true); // [SMARTEDITORSUS-180] 포커스 리셋
|
|
oSelection.collapseToEnd(); // [SMARTEDITORSUS-180] 포커스 리셋
|
|
oSelection.removeStringBookmark(sBM);
|
|
oSelection.select();
|
|
|
|
// [SMARTEDITORSUS-2312] 위쪽 P에 커서홀더가 없으면 추가 (북마크가 지워진 후에 해야함)
|
|
this._addCursorHolder(oSWrapper);
|
|
|
|
// P로 분리했기 때문에 BR이 들어있으면 제거한다.
|
|
if(oEWrapper.lastChild !== null && oEWrapper.lastChild.tagName == "BR"){
|
|
oEWrapper.removeChild(oEWrapper.lastChild);
|
|
}
|
|
|
|
// Cursor Holder 추가
|
|
// insert a cursor holder(br) if there's an empty-styling-only-tag surrounding current cursor
|
|
elStyleOnlyNode = this._addCursorHolder(oEWrapper);
|
|
|
|
if(oEWrapper.nextSibling && oEWrapper.nextSibling.tagName == "BR"){
|
|
oEWrapper.parentNode.removeChild(oEWrapper.nextSibling);
|
|
}
|
|
|
|
oSelection.selectNodeContents(elStyleOnlyNode);
|
|
oSelection.collapseToStart();
|
|
oSelection.select();
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE");
|
|
|
|
sBM = oSelection.placeStringBookmark();
|
|
setTimeout(jindo.$Fn(function(sBM){
|
|
var elBookmark = oSelection.getStringBookmark(sBM);
|
|
if(!elBookmark){return;}
|
|
|
|
oSelection.moveToStringBookmark(sBM);
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(sBM);
|
|
}, this).bind(sBM), 0);
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
var elBookmark = oSelection.getStringBookmark(sBM, true);
|
|
|
|
// 아래는 기본적으로 브라우저 기본 기능에 맡겨서 처리함
|
|
if(this.htBrowser.firefox){
|
|
if(elBookmark && elBookmark.nextSibling && elBookmark.nextSibling.tagName == "IFRAME"){
|
|
// [WOEDITOR-1603] FF에서 본문에 글감 삽입 후 엔터키 입력하면 글감이 복사되는 문제
|
|
setTimeout(jindo.$Fn(function(sBM){
|
|
var elBookmark = oSelection.getStringBookmark(sBM);
|
|
if(!elBookmark){return;}
|
|
|
|
oSelection.moveToStringBookmark(sBM);
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(sBM);
|
|
}, this).bind(sBM), 0);
|
|
}else{
|
|
// [SMARTEDITORSUS-1797] 엔터시 공백문자를 로 변환
|
|
// FF의 경우 2번이상 엔터치면 앞쪽공백이 사라져서 setTimeout으로 처리
|
|
setTimeout(jindo.$Fn(function(elNext){
|
|
this._convertHeadSpace(elNext);
|
|
}, this).bind(elBookmark.nextSibling), 0);
|
|
// [SMARTEDITORSUS-2070] 북마크를 setTimeout 으로 지우면 연속 엔터시 글꼴이 풀리기 때문에 SMARTEDITORSUS-1797 이슈 처리 로직과 분리
|
|
oSelection.removeStringBookmark(sBM);
|
|
}
|
|
}else if(this.htBrowser.ie){
|
|
var elParentNode = elBookmark.parentNode,
|
|
bAddUnderline = false,
|
|
bAddLineThrough = false;
|
|
|
|
// [SMARTEDITORSUS-2190] 첫번째 자식노드로 BR 노드가 있으면 블럭라인으로 변환처리
|
|
this._firstBR2Line(elParentNode, oSelection, oEnd.oLineBreaker);
|
|
|
|
if(!elBookmark || !elParentNode){// || elBookmark.nextSibling){
|
|
oSelection.removeStringBookmark(sBM);
|
|
return;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1973] 북마크를 바로 제거하면 커서 위치가 잘못되어 정렬이 풀리기 때문에 setTimeout 으로 제거
|
|
setTimeout(jindo.$Fn(function(){
|
|
this.oApp.getSelection().removeStringBookmark(sBM);
|
|
},this).bind(sBM),0);
|
|
|
|
bAddUnderline = (elParentNode.tagName === "U" || nhn.husky.SE2M_Utils.findAncestorByTagName("U", elParentNode) !== null);
|
|
bAddLineThrough = (elParentNode.tagName === "S" || elParentNode.tagName === "STRIKE" ||
|
|
(nhn.husky.SE2M_Utils.findAncestorByTagName("S", elParentNode) !== null && nhn.husky.SE2M_Utils.findAncestorByTagName("STRIKE", elParentNode) !== null));
|
|
|
|
// [SMARTEDITORSUS-26] Enter 후에 밑줄/취소선이 복사되지 않는 문제를 처리 (브라우저 Enter 처리 후 실행되도록 setTimeout 사용)
|
|
if(bAddUnderline || bAddLineThrough){
|
|
setTimeout(jindo.$Fn(this._addTextDecorationTag, this).bind(bAddUnderline, bAddLineThrough), 0);
|
|
|
|
return;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-180] 빈 SPAN 태그에 의해 엔터 후 엔터가 되지 않은 것으로 보이는 문제 (브라우저 Enter 처리 후 실행되도록 setTimeout 사용)
|
|
setTimeout(jindo.$Fn(this._addExtraCursorHolder, this).bind(elParentNode), 0);
|
|
}else{
|
|
var elParentNode = elBookmark.parentNode,
|
|
oNextSibling = this._getValidNextSibling(elBookmark);
|
|
|
|
/*
|
|
* [SMARTEDITORSUS-2046] 크롬에서 span으로 감싸있고 다음요소가 br 인 경우
|
|
* 커서 앞에 내용이 없으면 br로 line-break가 발생하고
|
|
* 커서 앞에 내용이 있으면 상위 p태그로 line-break가 발생하는데 p태그가 쪼개지지 않고 그냥 다음 라인에 p태그가 생긴다.
|
|
* 때문에 span으로 쌓여있고 br태그가 근접해있는 경우 브라우저자체 엔터처리를 막고 강제로 p태그를 쪼개주도록 처리함
|
|
* [SMARTEDITORSUS-2070] span 에 감싸진 br 뒤에 다른요소가 있는 경우만 해당 로직을 타도록 조건 추가
|
|
*/
|
|
if(elParentNode.tagName == "SPAN" && oNextSibling && oNextSibling.nodeName == "BR" && oNextSibling.nextSibling){
|
|
// 현재 선택된 요소들을 먼저 제거한다.
|
|
oSelection.deleteContents();
|
|
// 셀렉션을 커서부터 라인 끝까지 확장한다.
|
|
oSelection.setEndNodes(elBookmark, oEnd.oLineBreaker);
|
|
// 확장된 셀렉션(즉, 커서 뒷부분)을 잘라낸다.
|
|
var oNextContents = oSelection.extractContents(),
|
|
elNextP = oNextContents.firstChild; // oNextContents 는 fragment 이므로 나중에 커서를 위치시키기 위해 firstChild를 미리 할당해 둔다.
|
|
|
|
// 잘라낸 부분을 다음 라인으로 삽입한다.
|
|
elParentNode = oStart.oLineBreaker.parentNode; // 커서가 속한 p의 parent
|
|
oNextSibling = oStart.oLineBreaker.nextSibling; // 커서가 속한 p의 다음요소
|
|
if(oNextSibling){
|
|
elParentNode.insertBefore(oNextContents, oNextSibling);
|
|
}else{
|
|
elParentNode.appendChild(oNextContents);
|
|
}
|
|
|
|
// 삽입된 라인의 앞쪽에 커서를 위치시킨다.
|
|
oSelection.selectNodeContents(elNextP);
|
|
oSelection.collapseToStart();
|
|
oSelection.select();
|
|
|
|
oSelection.removeStringBookmark(sBM);
|
|
oEvent.stop(); // 브라우저의 엔터처리를 막는다.
|
|
}else{
|
|
// [SMARTEDITORSUS-1797] 엔터시 공백문자를 로 변환
|
|
this._convertHeadSpace(elBookmark.nextSibling);
|
|
oSelection.removeStringBookmark(sBM);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2190] IE8이하에서 P > SPAN > BR 이 첫번째에 있으면 엔터시 SPAN 역전현상(IE버그)이 발생한다.
|
|
* 이 SPAN 역전현상은 잘못된 태그 구조이기 때문에 다음 엔터시 HuskyRange 에서 오류를 유발한다.
|
|
* 따라서 P태그 안쪽에 BR 이 첫번째노드로 존재하면 엔터가 동작하기 전에 미리 BR 를 빈 P태그로 변환해두어야 한다.
|
|
*
|
|
* @param {Element} elParentNode 검사할 노드
|
|
* @param {HuskyRange} oSelection 허스키레인지
|
|
* @param {Element} oLineBreaker 검사할 노드가 포함된 블럭노드
|
|
*/
|
|
_firstBR2Line : function(elParentNode, oSelection, oLineBreaker){
|
|
while(elParentNode.firstChild && elParentNode.firstChild.nodeName === "BR"){
|
|
// 커서홀더노드를 만들어 미리 BR과 교체해둔다.
|
|
var oCursorHolder = oSelection._document.createTextNode('\u200B');
|
|
elParentNode.replaceChild(oCursorHolder, elParentNode.firstChild);
|
|
// P에서 커서홀더노드 까지를 추출하여 앞 라인으로 붙여넣는다. (P이하에 감싸진 SPAN이나 다른 태그들을 온전히 복사하기 위해 추출하여 삽입하는 방식을 사용함)
|
|
oSelection.setStartBefore(oLineBreaker);
|
|
oSelection.setEndAfter(oCursorHolder);
|
|
oLineBreaker.parentNode.insertBefore(oSelection.extractContents(), oLineBreaker);
|
|
}
|
|
},
|
|
|
|
// [IE9 standard mode] 엔터 후의 상/하단 P 태그를 확인하여 BOM, 공백( ) 추가
|
|
_addExtraCursorHolder : function(elUpperNode){
|
|
var oNodeChild,
|
|
oPrevChild,
|
|
elHtml;
|
|
|
|
elUpperNode = this._getStyleOnlyNode(elUpperNode);
|
|
|
|
// 엔터 후의 상단 SPAN 노드에 BOM 추가
|
|
//if(!!elUpperNode && /^(B|EM|I|LABEL|SPAN|STRONG|SUB|SUP|U|STRIKE)$/.test(elUpperNode.tagName) === false){
|
|
if(!!elUpperNode && elUpperNode.tagName === "SPAN"){ // SPAN 인 경우에만 발생함
|
|
oNodeChild = elUpperNode.lastChild;
|
|
|
|
while(!!oNodeChild){ // 빈 Text 제거
|
|
oPrevChild = oNodeChild.previousSibling;
|
|
|
|
if(oNodeChild.nodeType !== 3){
|
|
oNodeChild = oPrevChild;
|
|
continue;
|
|
}
|
|
|
|
if(nhn.husky.SE2M_Utils.isBlankTextNode(oNodeChild)){
|
|
oNodeChild.parentNode.removeChild(oNodeChild);
|
|
}
|
|
|
|
oNodeChild = oPrevChild;
|
|
}
|
|
|
|
elHtml = elUpperNode.innerHTML;
|
|
|
|
if(elHtml.replace("\u200B","").replace("\uFEFF","") === ""){
|
|
elUpperNode.innerHTML = "\u200B";
|
|
}
|
|
}
|
|
|
|
// 엔터 후에 비어있는 하단 SPAN 노드에 BOM 추가
|
|
var oSelection = this.oApp.getSelection(),
|
|
sBM,
|
|
elLowerNode,
|
|
elParent;
|
|
|
|
if(!oSelection.collapsed){
|
|
return;
|
|
}
|
|
|
|
oSelection.fixCommonAncestorContainer();
|
|
elLowerNode = oSelection.commonAncestorContainer;
|
|
|
|
if(!elLowerNode){
|
|
return;
|
|
}
|
|
|
|
elLowerNode = oSelection._getVeryFirstRealChild(elLowerNode);
|
|
|
|
if(elLowerNode.nodeType === 3){
|
|
elLowerNode = elLowerNode.parentNode;
|
|
}
|
|
|
|
if(!elLowerNode || elLowerNode.tagName !== "SPAN"){
|
|
return;
|
|
}
|
|
|
|
elHtml = elLowerNode.innerHTML;
|
|
|
|
if(elHtml.replace("\u200B","").replace("\uFEFF","") === ""){
|
|
elLowerNode.innerHTML = "\u200B";
|
|
}
|
|
|
|
// 백스페이스시 커서가 움직이지 않도록 커서를 커서홀더 앞쪽으로 옮긴다.
|
|
oSelection.selectNodeContents(elLowerNode);
|
|
oSelection.collapseToStart();
|
|
oSelection.select();
|
|
},
|
|
|
|
// [IE] P 태그 가장 뒤 자식노드로 공백( )을 값으로 하는 텍스트 노드를 추가
|
|
_addSpace : function(elNode){
|
|
var tmpTextNode, elChild, elNextChild, bHasNBSP, aImgChild, elLastImg;
|
|
|
|
if(!elNode){
|
|
return;
|
|
}
|
|
|
|
if(elNode.nodeType === 3){
|
|
return elNode.parentNode;
|
|
}
|
|
|
|
if(elNode.tagName !== "P"){
|
|
return elNode;
|
|
}
|
|
|
|
aImgChild = jindo.$Element(elNode).child(function(v){
|
|
return (v.$value().nodeType === 1 && v.$value().tagName === "IMG");
|
|
}, 1);
|
|
|
|
if(aImgChild.length > 0){
|
|
elLastImg = aImgChild[aImgChild.length - 1].$value();
|
|
elChild = elLastImg.nextSibling;
|
|
|
|
while(elChild){
|
|
elNextChild = elChild.nextSibling;
|
|
|
|
if (elChild.nodeType === 3 && (elChild.nodeValue === " " || elChild.nodeValue === unescape("%u00A0") || elChild.nodeValue === "\u200B")) {
|
|
elNode.removeChild(elChild);
|
|
}
|
|
|
|
elChild = elNextChild;
|
|
}
|
|
return elNode;
|
|
}
|
|
|
|
elChild = elNode.firstChild;
|
|
elNextChild = elChild;
|
|
bHasNBSP = false;
|
|
|
|
while(elChild){ // 를 붙일꺼니까 P 바로 아래의 "%uFEFF"는 제거함
|
|
elNextChild = elChild.nextSibling;
|
|
|
|
if(elChild.nodeType === 3){
|
|
if(elChild.nodeValue === unescape("%uFEFF")){
|
|
elNode.removeChild(elChild);
|
|
}
|
|
|
|
if(!bHasNBSP && (elChild.nodeValue === " " || elChild.nodeValue === unescape("%u00A0") || elChild.nodeValue === "\u200B")){
|
|
bHasNBSP = true;
|
|
}
|
|
}
|
|
|
|
elChild = elNextChild;
|
|
}
|
|
|
|
if(!bHasNBSP){
|
|
tmpTextNode = this.tmpTextNode.cloneNode();
|
|
elNode.appendChild(tmpTextNode);
|
|
}
|
|
|
|
return elNode; // [SMARTEDITORSUS-418] return 엘리먼트 추가
|
|
},
|
|
|
|
// [IE] 엔터 후에 취소선/밑줄 태그를 임의로 추가 (취소선/밑줄에 색상을 표시하기 위함)
|
|
_addTextDecorationTag : function(bAddUnderline, bAddLineThrough){
|
|
var oTargetNode, oNewNode,
|
|
oSelection = this.oApp.getSelection();
|
|
|
|
if(!oSelection.collapsed){
|
|
return;
|
|
}
|
|
|
|
oTargetNode = oSelection.startContainer;
|
|
|
|
while(oTargetNode){
|
|
if(oTargetNode.nodeType === 3){
|
|
oTargetNode = nhn.DOMFix.parentNode(oTargetNode);
|
|
break;
|
|
}
|
|
|
|
if(!oTargetNode.childNodes || oTargetNode.childNodes.length === 0){
|
|
// oTargetNode.innerHTML = "\u200B";
|
|
break;
|
|
}
|
|
|
|
oTargetNode = oTargetNode.firstChild;
|
|
}
|
|
|
|
if(!oTargetNode){
|
|
return;
|
|
}
|
|
|
|
if(oTargetNode.tagName === "U" || oTargetNode.tagName === "S" || oTargetNode.tagName === "STRIKE"){
|
|
return;
|
|
}
|
|
|
|
if(bAddUnderline){
|
|
oNewNode = oSelection._document.createElement("U");
|
|
oTargetNode.appendChild(oNewNode);
|
|
oTargetNode = oNewNode;
|
|
}
|
|
|
|
if(bAddLineThrough){
|
|
oNewNode = oSelection._document.createElement("STRIKE");
|
|
oTargetNode.appendChild(oNewNode);
|
|
}
|
|
|
|
oNewNode.innerHTML = "\u200B";
|
|
oSelection.selectNodeContents(oNewNode);
|
|
oSelection.collapseToEnd(); // End 로 해야 새로 생성된 노드 안으로 Selection 이 들어감
|
|
oSelection.select();
|
|
},
|
|
|
|
// returns inner-most styling node
|
|
// -> returns span3 from <span1><span2><span3>aaa</span></span></span>
|
|
_getStyleNode : function(elNode){
|
|
while(elNode.firstChild && this.oSelection._isBlankTextNode(elNode.firstChild)){
|
|
elNode.removeChild(elNode.firstChild);
|
|
}
|
|
|
|
var elFirstChild = elNode.firstChild;
|
|
|
|
if(!elFirstChild){
|
|
return elNode;
|
|
}
|
|
|
|
if(elFirstChild.nodeType === 3 ||
|
|
(elFirstChild.nodeType === 1 &&
|
|
(elFirstChild.tagName == "IMG" || elFirstChild.tagName == "BR" || elFirstChild.tagName == "HR" || elFirstChild.tagName == "IFRAME"))){
|
|
return elNode;
|
|
}
|
|
|
|
return this._getStyleNode(elNode.firstChild);
|
|
},
|
|
|
|
// returns inner-most styling only node if there's any.
|
|
// -> returns span3 from <span1><span2><span3></span></span></span>
|
|
_getStyleOnlyNode : function(elNode){
|
|
if(!elNode){
|
|
return null;
|
|
}
|
|
|
|
// the final styling node must allow appending children
|
|
// -> this doesn't seem to work for FF
|
|
if(!elNode.insertBefore){
|
|
return null;
|
|
}
|
|
|
|
if(elNode.tagName == "IMG" || elNode.tagName == "BR" || elNode.tagName == "HR" || elNode.tagName == "IFRAME"){
|
|
return null;
|
|
}
|
|
|
|
while(elNode.firstChild && this.oSelection._isBlankTextNode(elNode.firstChild)){
|
|
elNode.removeChild(elNode.firstChild);
|
|
}
|
|
|
|
if(elNode.childNodes.length>1){
|
|
return null;
|
|
}
|
|
|
|
if(!elNode.firstChild){
|
|
return elNode;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-227] TEXT_NODE 가 return 되는 문제를 수정함. IE 에서 TEXT_NODE 의 innrHTML 에 접근하면 오류 발생
|
|
if(elNode.firstChild.nodeType === 3){
|
|
return nhn.husky.SE2M_Utils.isBlankTextNode(elNode.firstChild) ? elNode : null;
|
|
//return (elNode.firstChild.textContents === null || elNode.firstChild.textContents === "") ? elNode : null;
|
|
}
|
|
|
|
return this._getStyleOnlyNode(elNode.firstChild);
|
|
},
|
|
|
|
_insertBR : function(oEvent){
|
|
oEvent.stop();
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
|
|
var elBR = this.oApp.getWYSIWYGDocument().createElement("BR");
|
|
oSelection.insertNode(elBR);
|
|
oSelection.selectNode(elBR);
|
|
oSelection.collapseToEnd();
|
|
|
|
if(!this.htBrowser.ie){
|
|
var oLineInfo = oSelection.getLineInfo();
|
|
var oEnd = oLineInfo.oEnd;
|
|
|
|
// line break by Parent
|
|
// <div> 1234<br></div>인경우, FF에서는 다음 라인으로 커서 이동이 안 일어남.
|
|
// 그래서 <div> 1234<br><br type='_moz'/></div> 이와 같이 생성해주어야 에디터 상에 2줄로 되어 보임.
|
|
if(oEnd.bParentBreak){
|
|
while(oEnd.oNode && oEnd.oNode.nodeType == 3 && oEnd.oNode.nodeValue == ""){
|
|
oEnd.oNode = oEnd.oNode.previousSibling;
|
|
}
|
|
|
|
var nTmp = 1;
|
|
if(oEnd.oNode == elBR || oEnd.oNode.nextSibling == elBR){
|
|
nTmp = 0;
|
|
}
|
|
|
|
if(nTmp === 0){
|
|
oSelection.pasteHTML("<br type='_moz'/>");
|
|
oSelection.collapseToEnd();
|
|
}
|
|
}
|
|
}
|
|
|
|
// the text cursor won't move to the next line without this
|
|
oSelection.insertNode(this.oApp.getWYSIWYGDocument().createTextNode(""));
|
|
oSelection.select();
|
|
}
|
|
});
|
|
//}
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to changing the editing mode using a Button element
|
|
* @name hp_SE2M_EditingModeChanger.js
|
|
*/
|
|
nhn.husky.SE2M_EditingModeChanger = jindo.$Class({
|
|
name : "SE2M_EditingModeChanger",
|
|
htConversionMode : null,
|
|
|
|
$init : function(elAppContainer, htConversionMode){
|
|
this.htConversionMode = htConversionMode;
|
|
this._assignHTMLElements(elAppContainer);
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
elAppContainer = jindo.$(elAppContainer) || document;
|
|
|
|
//@ec[
|
|
this.elWYSIWYGButton = jindo.$$.getSingle("BUTTON.se2_to_editor", elAppContainer);
|
|
this.elHTMLSrcButton = jindo.$$.getSingle("BUTTON.se2_to_html", elAppContainer);
|
|
this.elTEXTButton = jindo.$$.getSingle("BUTTON.se2_to_text", elAppContainer);
|
|
this.elModeToolbar = jindo.$$.getSingle("DIV.se2_conversion_mode", elAppContainer);
|
|
//@ec]
|
|
|
|
this.welWYSIWYGButtonLi = jindo.$Element(this.elWYSIWYGButton.parentNode);
|
|
this.welHTMLSrcButtonLi = jindo.$Element(this.elHTMLSrcButton.parentNode);
|
|
this.welTEXTButtonLi = jindo.$Element(this.elTEXTButton.parentNode);
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["isUseModeChanger", jindo.$Fn(this.isUseModeChanger, this).bind()]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
if(this.oApp.htOptions.bOnlyTextMode){
|
|
this.elWYSIWYGButton.style.display = 'none';
|
|
this.elHTMLSrcButton.style.display = 'none';
|
|
this.elTEXTButton.style.display = 'block';
|
|
this.oApp.exec("CHANGE_EDITING_MODE", ["TEXT"]);
|
|
}else{
|
|
this.oApp.registerBrowserEvent(this.elWYSIWYGButton, "click", "EVENT_CHANGE_EDITING_MODE_CLICKED", ["WYSIWYG"]);
|
|
this.oApp.registerBrowserEvent(this.elHTMLSrcButton, "click", "EVENT_CHANGE_EDITING_MODE_CLICKED", ["HTMLSrc"]);
|
|
this.oApp.registerBrowserEvent(this.elTEXTButton, "click", "EVENT_CHANGE_EDITING_MODE_CLICKED", ["TEXT", false]);
|
|
|
|
this.showModeChanger();
|
|
|
|
if(this.isUseModeChanger() === false && this.oApp.isUseVerticalResizer() === false){
|
|
this.elModeToolbar.style.display = "none";
|
|
}
|
|
}
|
|
},
|
|
|
|
// [SMARTEDITORSUS-906][SMARTEDITORSUS-1433] Editing Mode 사용 여부 처리 (true:사용함/ false:사용하지 않음)
|
|
showModeChanger : function(){
|
|
if(this.isUseModeChanger()){
|
|
this.elWYSIWYGButton.style.display = 'block';
|
|
this.elHTMLSrcButton.style.display = 'block';
|
|
this.elTEXTButton.style.display = 'block';
|
|
}else{
|
|
this.elWYSIWYGButton.style.display = 'none';
|
|
this.elHTMLSrcButton.style.display = 'none';
|
|
this.elTEXTButton.style.display = 'none';
|
|
}
|
|
},
|
|
|
|
isUseModeChanger : function(){
|
|
return (typeof(this.htConversionMode) === 'undefined' || typeof(this.htConversionMode.bUseModeChanger) === 'undefined' || this.htConversionMode.bUseModeChanger === true) ? true : false;
|
|
},
|
|
|
|
$ON_EVENT_CHANGE_EDITING_MODE_CLICKED : function(sMode, bNoAlertMsg){
|
|
if (sMode == 'TEXT') {
|
|
//에디터 영역 내에 모든 내용 가져옴.
|
|
var sContent = this.oApp.getIR();
|
|
|
|
// 내용이 있으면 경고창 띄우기
|
|
if (sContent.length > 0 && !bNoAlertMsg) {
|
|
if ( !confirm(this.oApp.$MSG("SE2M_EditingModeChanger.confirmTextMode")) ) {
|
|
return false;
|
|
}
|
|
}
|
|
this.oApp.exec("CHANGE_EDITING_MODE", [sMode]);
|
|
}else{
|
|
this.oApp.exec("CHANGE_EDITING_MODE", [sMode]);
|
|
}
|
|
|
|
if ('HTMLSrc' == sMode) {
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['htmlmode']);
|
|
} else if ('TEXT' == sMode) {
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['textmode']);
|
|
} else {
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['editormode']);
|
|
}
|
|
},
|
|
|
|
$ON_DISABLE_ALL_UI : function(htOptions){
|
|
htOptions = htOptions || {};
|
|
var waExceptions = jindo.$A(htOptions.aExceptions || []);
|
|
|
|
if(waExceptions.has("mode_switcher")){
|
|
return;
|
|
}
|
|
if(this.oApp.getEditingMode() == "WYSIWYG"){
|
|
this.welWYSIWYGButtonLi.removeClass("active");
|
|
this.elHTMLSrcButton.disabled = true;
|
|
this.elTEXTButton.disabled = true;
|
|
} else if (this.oApp.getEditingMode() == 'TEXT') {
|
|
this.welTEXTButtonLi.removeClass("active");
|
|
this.elWYSIWYGButton.disabled = true;
|
|
this.elHTMLSrcButton.disabled = true;
|
|
}else{
|
|
this.welHTMLSrcButtonLi.removeClass("active");
|
|
this.elWYSIWYGButton.disabled = true;
|
|
this.elTEXTButton.disabled = true;
|
|
}
|
|
},
|
|
|
|
$ON_ENABLE_ALL_UI : function(){
|
|
if(this.oApp.getEditingMode() == "WYSIWYG"){
|
|
this.welWYSIWYGButtonLi.addClass("active");
|
|
this.elHTMLSrcButton.disabled = false;
|
|
this.elTEXTButton.disabled = false;
|
|
} else if (this.oApp.getEditingMode() == 'TEXT') {
|
|
this.welTEXTButtonLi.addClass("active");
|
|
this.elWYSIWYGButton.disabled = false;
|
|
this.elHTMLSrcButton.disabled = false;
|
|
}else{
|
|
this.welHTMLSrcButtonLi.addClass("active");
|
|
this.elWYSIWYGButton.disabled = false;
|
|
this.elTEXTButton.disabled = false;
|
|
}
|
|
},
|
|
|
|
$ON_CHANGE_EDITING_MODE : function(sMode){
|
|
if(sMode == "HTMLSrc"){
|
|
this.welWYSIWYGButtonLi.removeClass("active");
|
|
this.welHTMLSrcButtonLi.addClass("active");
|
|
this.welTEXTButtonLi.removeClass("active");
|
|
|
|
this.elWYSIWYGButton.disabled = false;
|
|
this.elHTMLSrcButton.disabled = true;
|
|
this.elTEXTButton.disabled = false;
|
|
this.oApp.exec("HIDE_ALL_DIALOG_LAYER");
|
|
|
|
this.oApp.exec("DISABLE_ALL_UI", [{aExceptions:["mode_switcher"]}]);
|
|
} else if (sMode == 'TEXT') {
|
|
this.welWYSIWYGButtonLi.removeClass("active");
|
|
this.welHTMLSrcButtonLi.removeClass("active");
|
|
this.welTEXTButtonLi.addClass("active");
|
|
|
|
this.elWYSIWYGButton.disabled = false;
|
|
this.elHTMLSrcButton.disabled = false;
|
|
this.elTEXTButton.disabled = true;
|
|
this.oApp.exec("HIDE_ALL_DIALOG_LAYER");
|
|
this.oApp.exec("DISABLE_ALL_UI", [{aExceptions:["mode_switcher"]}]);
|
|
}else{
|
|
this.welWYSIWYGButtonLi.addClass("active");
|
|
this.welHTMLSrcButtonLi.removeClass("active");
|
|
this.welTEXTButtonLi.removeClass("active");
|
|
|
|
this.elWYSIWYGButton.disabled = true;
|
|
this.elHTMLSrcButton.disabled = false;
|
|
this.elTEXTButton.disabled = false;
|
|
|
|
this.oApp.exec("RESET_STYLE_STATUS");
|
|
this.oApp.exec("ENABLE_ALL_UI", []);
|
|
}
|
|
}
|
|
});
|
|
//}
|
|
/**
|
|
* @pluginDesc WYSIWYG 영역에 붙여넣어지는 외부 컨텐츠를 정제하는 플러그인
|
|
*/
|
|
nhn.husky.SE_PasteHandler = jindo.$Class({
|
|
name : "SE_PasteHandler",
|
|
_rxStyleTag : /<style(?:\s+[^>]*)?>(?:.|\r|\n)*?<\/style>/gi,
|
|
_rxStyleTagStrip : /<\/?style(?:\s+[^>]*)?>/gi,
|
|
_rxClassSelector : /\w*\.\w+/g,
|
|
_rxClassSelectorStart : /\w*\./g,
|
|
|
|
$init : function(){
|
|
},
|
|
|
|
/**
|
|
* 크롬에서 복/붙시 style분석을 위한 임시 document
|
|
* @returns {Document} 임시 document 반환
|
|
*/
|
|
_getTmpDocument : function(){
|
|
if(!this._oTmpDoc){
|
|
var elIframe = document.createElement("IFRAME");
|
|
elIframe.style.display = "none";
|
|
document.body.appendChild(elIframe);
|
|
var oDoc = elIframe.contentWindow.document;
|
|
oDoc.open();
|
|
oDoc.write('<html><head></head><body></body></html>');
|
|
oDoc.close();
|
|
|
|
this._oTmpDoc = oDoc;
|
|
}
|
|
|
|
return this._oTmpDoc;
|
|
},
|
|
|
|
/**
|
|
* CSSRuleList에서 CSSRule.STYLE_RULE만 가져와 selector키 맵핑형태로 반환한다.
|
|
* @param {CSSRuleList} cssRules 확인할 CSSRuleList
|
|
* @returns {HashTable} key(셀렉터명)/value(CSSRule) 맵핑객체
|
|
*/
|
|
_getCSSStyleRule : function(cssRules) {
|
|
var htStyleRule = {};
|
|
for(var i = 0, cssRule; (cssRule = cssRules[i]); i++){
|
|
// @see https://developer.mozilla.org/en-US/docs/Web/API/CSSRule#Type_constants
|
|
if(cssRule.type === CSSRule.STYLE_RULE){
|
|
htStyleRule[cssRule.selectorText] = cssRule.style;
|
|
}
|
|
}
|
|
return htStyleRule;
|
|
},
|
|
|
|
/**
|
|
* html문자열에 style태그가 있으면 CSSStyleRule을 추출하여 저장해둔다.
|
|
* @param {String} html 확인할 HTML문자열
|
|
*/
|
|
_extractStyle : function(html) {
|
|
if(!html) {
|
|
return;
|
|
}
|
|
var aMatch = html.match(this._rxStyleTag) || [];
|
|
var sStyles = aMatch.join("\n");
|
|
sStyles = sStyles.replace(this._rxStyleTagStrip, "");
|
|
|
|
var oTmpDoc = this._getTmpDocument();
|
|
var elHead = oTmpDoc.getElementsByTagName("HEAD")[0];
|
|
var elStyle = oTmpDoc.createElement("STYLE");
|
|
elStyle.innerHTML = sStyles;
|
|
elHead.appendChild(elStyle);
|
|
|
|
this._htStyleRule = this._getCSSStyleRule(elStyle.sheet.cssRules);
|
|
elStyle.parentNode.removeChild(elStyle);
|
|
},
|
|
|
|
/**
|
|
* 해당 요소에 inline style을 적용한다.
|
|
* @param {Element} el 스타일을 적용할 대상 요소
|
|
* @param {CSSStyleDeclaration} oStyle 반영할 스타일 정의 객체
|
|
* @param {Booleam} bOverride 스타일을 덮어씌울 것인지 여부
|
|
*/
|
|
_applyInlineStyle : function(el, oStyle, bOverride) {
|
|
for(var i = 0, len = oStyle.length, sStyleName; i < len; i++){
|
|
sStyleName = oStyle[i];
|
|
if(bOverride || !el.style[sStyleName]){
|
|
el.style[sStyleName] = oStyle[sStyleName];
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* CSSStyleRule을 inline style로 반영해준다.
|
|
* @param {String} sSelector CSS셀렉터명
|
|
* @param {CSSStyleDeclaration} oStyle 셀렉터명에 해당하는 스타일 정의 객체
|
|
* @param {Element} el 스타일반영을 제한할 상위 요소
|
|
*/
|
|
_applyStyleRuleToInline : function(sSelector, oStyle, el) {
|
|
var aClassSelector = sSelector.match(this._rxClassSelector) || [];
|
|
// 태그 셀렉터는 잘못된 영향을 줄 수 있기 때문에 클래스 셀렉터만 적용한다.
|
|
if(aClassSelector.length < 1) {
|
|
return;
|
|
}
|
|
|
|
var sRemoveClass = aClassSelector.join(" ").replace(this._rxClassSelectorStart, "");
|
|
var aelTarget = jindo.$$(sSelector, el);
|
|
for(var i = 0, elTarget; (elTarget = aelTarget[i]); i++){
|
|
this._applyInlineStyle(elTarget, oStyle);
|
|
if(sRemoveClass){
|
|
jindo.$Element(elTarget).removeClass(sRemoveClass);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 저장해둔 CSSStyleRule를 적용한다.
|
|
* @param {Element} el 적용범위를 제한할 상위 요소
|
|
*/
|
|
_applyStyle : function(el) {
|
|
var htStyleRule = this._htStyleRule || {};
|
|
for (var sSelector in htStyleRule) {
|
|
this._applyStyleRuleToInline(sSelector, htStyleRule[sSelector], el);
|
|
}
|
|
this._htStyleRule = null;
|
|
},
|
|
|
|
/**
|
|
* 붙여넣기시 생성된 폰트태그를 보정한다.
|
|
* @param {Element} el 태그정제를 제한할 상위 요소
|
|
*/
|
|
_revertFontAfterPaste : function(el){
|
|
nhn.husky.SE2M_Utils.removeInvalidFont(el);
|
|
|
|
if(document.documentMode < 11){
|
|
// [MUG-7757] IE11미만은 span 이 남아있기 때문에 그냥 제거
|
|
nhn.husky.SE2M_Utils.stripTags(el, "FONT");
|
|
}else{
|
|
// IE11미만이외는 font를 span으로 재변환
|
|
nhn.husky.SE2M_Utils.convertFontToSpan(el);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* SPAN태그에 line-height 스타일속성이 있으면 제거한다.
|
|
* @param {Element} el 태그정제를 제한할 상위 요소
|
|
*/
|
|
_removeLineHeightInSpan : function(el){
|
|
var aelSpans = jindo.$$('span[style*="line-height:"]', el);
|
|
for(var i = 0, elSpan; (elSpan = aelSpans[i]); i++){
|
|
elSpan.style.lineHeight = null;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* paste이벤트에서 클립보드의 style태그를 추출할 수 있으면 CSSStyleRule을 저장해둔다.
|
|
*/
|
|
$ON_EVENT_EDITING_AREA_PASTE : function(we) {
|
|
var oClipboard = we.$value().clipboardData;
|
|
if (!oClipboard) {
|
|
return;
|
|
}
|
|
var sClipboardHTML = oClipboard.getData("text/html");
|
|
if (!sClipboardHTML) {
|
|
return;
|
|
}
|
|
|
|
this._extractStyle(sClipboardHTML);
|
|
},
|
|
|
|
/**
|
|
* paste된 후 정제처리한다.
|
|
* 1. font태그가 삽입되었다면 보정한다.
|
|
* 2. 저장해둔 CSSStyleRule이 있다면 inline style로 반영해준다.
|
|
* 3. line-height스타일이 지정된 span이 있으면 해당 속성 제거
|
|
*/
|
|
$ON_EVENT_EDITING_AREA_PASTE_DELAY : function() {
|
|
var el = this.oApp.getWYSIWYGDocument().body;
|
|
|
|
this._revertFontAfterPaste(el);
|
|
this._applyStyle(el);
|
|
this._removeLineHeightInSpan(el);
|
|
}
|
|
});
|
|
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the basic editor commands
|
|
* @name hp_SE_ExecCommand.js
|
|
*/
|
|
nhn.husky.SE2M_ExecCommand = jindo.$Class({
|
|
name : "SE2M_ExecCommand",
|
|
oEditingArea : null,
|
|
oUndoOption : null,
|
|
_rxCmdInline : /^(?:bold|underline|italic|strikethrough|superscript|subscript)$/i, // inline element 가 생성되는 command
|
|
|
|
$init : function(oEditingArea){
|
|
this.oEditingArea = oEditingArea;
|
|
this.nIndentSpacing = 40;
|
|
|
|
this.rxClickCr = new RegExp('^bold|underline|italic|strikethrough|justifyleft|justifycenter|justifyright|justifyfull|insertorderedlist|insertunorderedlist|outdent|indent$', 'i');
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
// the right document will be available only when the src is completely loaded
|
|
if(this.oEditingArea && this.oEditingArea.tagName == "IFRAME"){
|
|
this.oEditingArea = this.oEditingArea.contentWindow.document;
|
|
}
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
// [SMARTEDITORSUS-2260] 메일 > Mac에서 ctrl 조합 단축키 모두 meta 조합으로 변경
|
|
if (jindo.$Agent().os().mac) {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+b", "EXECCOMMAND", ["bold", false, false]]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+u", "EXECCOMMAND", ["underline", false, false]]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+i", "EXECCOMMAND", ["italic", false, false]]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+d", "EXECCOMMAND", ["strikethrough", false, false]]);
|
|
} else {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+b", "EXECCOMMAND", ["bold", false, false]]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+u", "EXECCOMMAND", ["underline", false, false]]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+i", "EXECCOMMAND", ["italic", false, false]]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+d", "EXECCOMMAND", ["strikethrough", false, false]]);
|
|
}
|
|
|
|
this.oApp.exec("REGISTER_HOTKEY", ["tab", "INDENT"]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["shift+tab", "OUTDENT"]);
|
|
//this.oApp.exec("REGISTER_HOTKEY", ["tab", "EXECCOMMAND", ["indent", false, false]]);
|
|
//this.oApp.exec("REGISTER_HOTKEY", ["shift+tab", "EXECCOMMAND", ["outdent", false, false]]);
|
|
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["bold", "click", "EXECCOMMAND", ["bold", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["underline", "click", "EXECCOMMAND", ["underline", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["italic", "click", "EXECCOMMAND", ["italic", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["lineThrough", "click", "EXECCOMMAND", ["strikethrough", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["superscript", "click", "EXECCOMMAND", ["superscript", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["subscript", "click", "EXECCOMMAND", ["subscript", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["justifyleft", "click", "EXECCOMMAND", ["justifyleft", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["justifycenter", "click", "EXECCOMMAND", ["justifycenter", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["justifyright", "click", "EXECCOMMAND", ["justifyright", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["justifyfull", "click", "EXECCOMMAND", ["justifyfull", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["orderedlist", "click", "EXECCOMMAND", ["insertorderedlist", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["unorderedlist", "click", "EXECCOMMAND", ["insertunorderedlist", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["outdent", "click", "EXECCOMMAND", ["outdent", false, false]]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["indent", "click", "EXECCOMMAND", ["indent", false, false]]);
|
|
|
|
// this.oApp.exec("REGISTER_UI_EVENT", ["styleRemover", "click", "EXECCOMMAND", ["RemoveFormat", false, false]]);
|
|
|
|
this.oNavigator = jindo.$Agent().navigator();
|
|
|
|
if(!this.oNavigator.safari && !this.oNavigator.chrome){
|
|
this._getDocumentBR = function(){};
|
|
this._fixDocumentBR = function(){};
|
|
}
|
|
|
|
if(!this.oNavigator.ie){
|
|
this._fixCorruptedBlockQuote = function(){};
|
|
|
|
if(!this.oNavigator.safari && !this.oNavigator.chrome){
|
|
this._insertBlankLine = function(){};
|
|
}
|
|
}
|
|
|
|
if(!this.oNavigator.firefox){
|
|
this._extendBlock = function(){};
|
|
}
|
|
},
|
|
|
|
$ON_INDENT : function(){
|
|
this.oApp.delayedExec("EXECCOMMAND", ["indent", false, false], 0);
|
|
},
|
|
|
|
$ON_OUTDENT : function(){
|
|
this.oApp.delayedExec("EXECCOMMAND", ["outdent", false, false], 0);
|
|
},
|
|
|
|
$BEFORE_EXECCOMMAND : function(sCommand, bUserInterface, vValue, htOptions){
|
|
var elTmp, oSelection;
|
|
|
|
//본문에 전혀 클릭이 한번도 안 일어난 상태에서 크롬과 IE에서 EXECCOMMAND가 정상적으로 안 먹히는 현상.
|
|
this.oApp.exec("FOCUS");
|
|
this._bOnlyCursorChanged = false;
|
|
oSelection = this.oApp.getSelection();
|
|
// [SMARTEDITORSUS-1584] IE에서 테이블관련 태그 사이의 텍스트노드가 포함된 채로 execCommand 가 실행되면
|
|
// 테이블 태그들 사이에 더미 P 태그가 추가된다.
|
|
// 테이블관련 태그 사이에 태그가 있으면 문법에 어긋나기 때문에 getContents 시 이 더미 P 태그들이 밖으로 빠져나가게 된다.
|
|
// 때문에 execCommand 실행되기 전에 셀렉션에 테이블관련 태그 사이의 텍스트노드를 찾아내 지워준다.
|
|
for(var i = 0, aNodes = oSelection.getNodes(), oNode;(oNode = aNodes[i]); i++){
|
|
nhn.husky.SE2M_Utils.removeInvalidNodeInTable(oNode);
|
|
}
|
|
|
|
if(/^insertorderedlist|insertunorderedlist$/i.test(sCommand)){
|
|
this._getDocumentBR();
|
|
|
|
// [SMARTEDITORSUS-985][SMARTEDITORSUS-1740]
|
|
this._checkBlockQuoteCondition_IE();
|
|
// --[SMARTEDITORSUS-985][SMARTEDITORSUS-1740]
|
|
}
|
|
|
|
if(/^justify*/i.test(sCommand)){
|
|
// [SMARTEDITORSUS-704][SMARTEDITORSUS-2050] 메서드 통합
|
|
this._removeElementAlign('span');
|
|
}
|
|
|
|
if(this._rxCmdInline.test(sCommand)){
|
|
this.oUndoOption = {bMustBlockElement:true};
|
|
|
|
if(nhn.CurrentSelection.isCollapsed()){
|
|
this._bOnlyCursorChanged = true;
|
|
}
|
|
}
|
|
|
|
if(sCommand == "indent" || sCommand == "outdent"){
|
|
if(!htOptions){htOptions = {};}
|
|
htOptions["bDontAddUndoHistory"] = true;
|
|
}
|
|
if((!htOptions || !htOptions["bDontAddUndoHistory"]) && !this._bOnlyCursorChanged){
|
|
if(/^justify*/i.test(sCommand)){
|
|
this.oUndoOption = {sSaveTarget:"BODY"};
|
|
}else if(sCommand === "insertorderedlist" || sCommand === "insertunorderedlist"){
|
|
this.oUndoOption = {bMustBlockContainer:true};
|
|
}
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", [sCommand, this.oUndoOption]);
|
|
}
|
|
if(this.oNavigator.ie && this.oApp.getWYSIWYGDocument().selection){
|
|
if(this.oApp.getWYSIWYGDocument().selection.type === "Control"){
|
|
oSelection = this.oApp.getSelection();
|
|
oSelection.select();
|
|
}
|
|
}
|
|
|
|
if(sCommand == "insertorderedlist" || sCommand == "insertunorderedlist"){
|
|
this._insertBlankLine();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-985][SMARTEDITORSUS-1740][SMARTEDITORSUS-1798][SMARTEDITORSUS-2157]
|
|
* [Win XP - IE 8][IE 9~11][IE edge] 인용구 안에서 번호매기기, 글머리기호를 적용할 때 필요한 조치이다.
|
|
*
|
|
* 인용구 안의 선택한 영역을 기준으로,
|
|
*
|
|
* 선택한 영역이 없는 경우에는 해당 줄을 제외했을 때,
|
|
* 선택한 영역이 있는 경우에는 선택한 줄을 제외했을 때
|
|
*
|
|
* 더 이상의 <P>가 없는 경우
|
|
* execCommand("insertorderedlist"), execCommand("insertunorderedlist")가 오동작한다.
|
|
*
|
|
* 이러한 오동작을 방지하기 위해
|
|
* 인용구 안에서 번호매기기, 글머리기호를 삽입할 때는
|
|
* execCommand() 실행 전에 빈 <P>를 삽입해 주고,
|
|
* execCommand() 실행 후 빈 <P>를 제거해 준다.
|
|
* */
|
|
_checkBlockQuoteCondition_IE : function(){
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
var bProcess = false;
|
|
var elBlockquote;
|
|
|
|
if((htBrowser.ie && (htBrowser.nativeVersion >= 9 && htBrowser.nativeVersion <= 11) && (htBrowser.version >= 9 && htBrowser.version <= 11))
|
|
|| (this.oApp.oAgent.os().winxp && htBrowser.ie && htBrowser.nativeVersion <= 8) || htBrowser.edge){
|
|
var oSelection = this.oApp.getSelection();
|
|
var elCommonAncestorContainer = oSelection.commonAncestorContainer;
|
|
var htAncestor_blockquote = nhn.husky.SE2M_Utils.findAncestorByTagNameWithCount("BLOCKQUOTE", elCommonAncestorContainer);
|
|
elBlockquote = htAncestor_blockquote.elNode;
|
|
|
|
if(elBlockquote){
|
|
var htAncestor_cell = nhn.husky.SE2M_Utils.findClosestAncestorAmongTagNamesWithCount(["td", "th"], elCommonAncestorContainer);
|
|
if(htAncestor_cell.elNode){
|
|
if(htAncestor_cell.nRecursiveCount > htAncestor_blockquote.nRecursiveCount){
|
|
// blockquote가 cell 안에서 생성된 경우
|
|
bProcess = true;
|
|
}
|
|
}else{
|
|
// blockquote가 cell 안에서 생성되지 않은 경우
|
|
bProcess = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(bProcess){
|
|
this._insertDummyParagraph_IE(elBlockquote);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-985][SMARTEDITORSUS-1740]
|
|
* [IE 9~10] 대상 엘리먼트에 빈 <P>를 삽입
|
|
* */
|
|
_insertDummyParagraph_IE : function(el){
|
|
this._elDummyParagraph = document.createElement("P");
|
|
el.appendChild(this._elDummyParagraph);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-985][SMARTEDITORSUS-1740]
|
|
* [IE 9~10] 빈 <P>를 제거
|
|
* */
|
|
_removeDummyParagraph_IE : function(){
|
|
if(this._elDummyParagraph && this._elDummyParagraph.parentNode){
|
|
this._elDummyParagraph.parentNode.removeChild(this._elDummyParagraph);
|
|
}
|
|
},
|
|
|
|
$ON_EXECCOMMAND : function(sCommand, bUserInterface, vValue){
|
|
var bSelectedBlock = false;
|
|
var htSelectedTDs = {};
|
|
var oSelection = this.oApp.getSelection();
|
|
|
|
bUserInterface = (bUserInterface == "" || bUserInterface)?bUserInterface:false;
|
|
vValue = (vValue == "" || vValue)?vValue:false;
|
|
|
|
this.oApp.exec("IS_SELECTED_TD_BLOCK",['bIsSelectedTd',htSelectedTDs]);
|
|
bSelectedBlock = htSelectedTDs.bIsSelectedTd;
|
|
|
|
if( bSelectedBlock){
|
|
if(sCommand == "indent"){
|
|
this.oApp.exec("SET_LINE_BLOCK_STYLE", [null, jindo.$Fn(this._indentMargin, this).bind()]);
|
|
}else if(sCommand == "outdent"){
|
|
this.oApp.exec("SET_LINE_BLOCK_STYLE", [null, jindo.$Fn(this._outdentMargin, this).bind()]);
|
|
}else{
|
|
this._setBlockExecCommand(sCommand, bUserInterface, vValue);
|
|
}
|
|
} else {
|
|
switch(sCommand){
|
|
case "indent":
|
|
case "outdent":
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", [sCommand]);
|
|
|
|
// bookmark 설정
|
|
var sBookmark = oSelection.placeStringBookmark();
|
|
|
|
if(sCommand === "indent"){
|
|
this.oApp.exec("SET_LINE_STYLE", [null, jindo.$Fn(this._indentMargin, this).bind(), {bDoNotSelect : true, bDontAddUndoHistory : true}]);
|
|
}else{
|
|
this.oApp.exec("SET_LINE_STYLE", [null, jindo.$Fn(this._outdentMargin, this).bind(), {bDoNotSelect : true, bDontAddUndoHistory : true}]);
|
|
}
|
|
|
|
oSelection.moveToStringBookmark(sBookmark);
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(sBookmark); //bookmark 삭제
|
|
|
|
setTimeout(jindo.$Fn(function(sCommand){
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", [sCommand]);
|
|
}, this).bind(sCommand), 25);
|
|
|
|
break;
|
|
|
|
case "justifyleft":
|
|
case "justifycenter":
|
|
case "justifyright":
|
|
case "justifyfull":
|
|
var oSelectionClone = this._extendBlock(); // FF
|
|
|
|
this.oEditingArea.execCommand(sCommand, bUserInterface, vValue);
|
|
|
|
if(!!oSelectionClone){
|
|
oSelectionClone.select();
|
|
}
|
|
|
|
// [SMARTEDITORSUS-2050] 선택된 범위 내에 table이 있다면 align 제거
|
|
// [SMARTEDITORSUS-2114] 크롬에서 표 가운데 정렬이 되지 않는 문제가 있어서 해당 처리를 IE에서만 하도록 수정
|
|
if(this.oNavigator.ie) {
|
|
this._removeElementAlign('table');
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
//if(this.oNavigator.firefox){
|
|
//this.oEditingArea.execCommand("styleWithCSS", bUserInterface, false);
|
|
//}
|
|
// [SMARTEDITORSUS-1646] [SMARTEDITORSUS-1653] collapsed 상태이면 execCommand 가 실행되기 전에 ZWSP를 넣어준다.
|
|
// [SMARTEDITORSUS-1702] ul, ol 처럼 block element 가 바로 생성되는 경우는 ZWSP 삽입 제외
|
|
if(oSelection.collapsed && this._rxCmdInline.test(sCommand)){
|
|
// collapsed 인 경우
|
|
var sBM = oSelection.placeStringBookmark(),
|
|
oBM = oSelection.getStringBookmark(sBM),
|
|
oHolderNode = oBM.previousSibling;
|
|
|
|
// execCommand를 실행할때마다 ZWSP가 포함된 더미 태그가 자꾸 생길 수 있기 때문에 이미 있으면 있는 걸로 사용한다.
|
|
if(!oHolderNode || oHolderNode.nodeValue !== "\u200B"){
|
|
oHolderNode = this.oApp.getWYSIWYGDocument().createTextNode("\u200B");
|
|
oSelection.insertNode(oHolderNode);
|
|
}
|
|
oSelection.removeStringBookmark(sBM); // 미리 지워주지 않으면 더미 태그가 생길 수 있다.
|
|
oSelection.selectNodeContents(oHolderNode);
|
|
oSelection.select();
|
|
this.oEditingArea.execCommand(sCommand, bUserInterface, vValue);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
|
|
// [SMARTEDITORSUS-1658] 뒤쪽에 더미태그가 있으면 제거해준다.
|
|
var oSingleNode = this._findSingleNode(oHolderNode);
|
|
if(oSingleNode && oSelection._hasCursorHolderOnly(oSingleNode.nextSibling)){
|
|
oSingleNode.parentNode.removeChild(oSingleNode.nextSibling);
|
|
}
|
|
}else{
|
|
this.oEditingArea.execCommand(sCommand, bUserInterface, vValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
this._countClickCr(sCommand);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1658] 해당노드의 상위로 검색해 single child 만 갖는 최상위 노드를 찾는다.
|
|
* @param {Node} oNode 확인할 노드
|
|
* @return {Node} single child 만 감싸고 있는 최상위 노드를 반환한다. 없으면 입력한 노드 반환
|
|
*/
|
|
_findSingleNode : function(oNode){
|
|
if(!oNode){
|
|
return null;
|
|
}
|
|
if(oNode.parentNode.childNodes.length === 1){
|
|
return this._findSingleNode(oNode.parentNode);
|
|
}else{
|
|
return oNode;
|
|
}
|
|
},
|
|
|
|
$AFTER_EXECCOMMAND : function(sCommand, bUserInterface, vValue, htOptions){
|
|
if(this.elP1 && this.elP1.parentNode){
|
|
this.elP1.parentNode.removeChild(this.elP1);
|
|
}
|
|
|
|
if(this.elP2 && this.elP2.parentNode){
|
|
this.elP2.parentNode.removeChild(this.elP2);
|
|
}
|
|
|
|
if(/^insertorderedlist|insertunorderedlist$/i.test(sCommand)){
|
|
// this._fixDocumentBR(); // Chrome/Safari
|
|
// [SMARTEDITORSUS-985][SMARTEDITORSUS-1740]
|
|
this._removeDummyParagraph_IE();
|
|
// --[SMARTEDITORSUS-985][SMARTEDITORSUS-1740]
|
|
this._fixCorruptedBlockQuote(sCommand === "insertorderedlist" ? "OL" : "UL"); // IE
|
|
// [SMARTEDITORSUS-1795] 갤럭시노트_Android4.1.2 기본브라우저일 경우 내부에 생성된 BLOCKQUOTE 제거
|
|
if(this.oNavigator.bGalaxyBrowser){
|
|
this._removeBlockQuote();
|
|
}
|
|
}
|
|
|
|
if((/^justify*/i.test(sCommand))){
|
|
this._fixAlign(sCommand === "justifyfull" ? "justify" : sCommand.substring(7));
|
|
}
|
|
|
|
if(sCommand == "indent" || sCommand == "outdent"){
|
|
if(!htOptions){htOptions = {};}
|
|
htOptions["bDontAddUndoHistory"] = true;
|
|
}
|
|
|
|
if((!htOptions || !htOptions["bDontAddUndoHistory"]) && !this._bOnlyCursorChanged){
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", [sCommand, this.oUndoOption]);
|
|
}
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-704][SMARTEDITORSUS-2050] Element에 적용된 align 제거
|
|
* @param String sTagName 대상 element의 tagName
|
|
*/
|
|
_removeElementAlign : function(sTagName){
|
|
var oSelection = this.oApp.getSelection(),
|
|
aNodes = oSelection.getNodes(),
|
|
elNode = null,
|
|
rxTagName = new RegExp('^' + sTagName + '$', 'i');
|
|
|
|
for(var i=0, nLen=aNodes.length; i<nLen; i++){
|
|
elNode = aNodes[i];
|
|
|
|
if(elNode.tagName && rxTagName.test(elNode.tagName)){
|
|
elNode.style.textAlign = "";
|
|
elNode.removeAttribute("align");
|
|
}
|
|
}
|
|
},
|
|
|
|
// [SMARTEDITORSUS-851] align, text-align을 fix해야 할 대상 노드를 찾음
|
|
_getAlignNode : function(elNode){
|
|
if(elNode.tagName && (elNode.tagName === "P" || elNode.tagName === "DIV")){
|
|
return elNode;
|
|
}
|
|
|
|
elNode = elNode.parentNode;
|
|
|
|
while(elNode && elNode.tagName){
|
|
if(elNode.tagName === "P" || elNode.tagName === "DIV"){
|
|
return elNode;
|
|
}
|
|
|
|
elNode = elNode.parentNode;
|
|
}
|
|
},
|
|
|
|
_fixAlign : function(sAlign){
|
|
var oSelection = this.oApp.getSelection(),
|
|
aNodes = [],
|
|
elNode = null,
|
|
elParentNode = null;
|
|
|
|
var removeTableAlign = !this.oNavigator.ie ? function(){} : function(elNode){
|
|
if(elNode.tagName && elNode.tagName === "TABLE"){
|
|
elNode.removeAttribute("align");
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
if(oSelection.collapsed){
|
|
aNodes[0] = oSelection.startContainer; // collapsed인 경우에는 getNodes의 결과는 []
|
|
}else{
|
|
aNodes = oSelection.getNodes();
|
|
}
|
|
|
|
for(var i=0, nLen=aNodes.length; i<nLen; i++){
|
|
elNode = aNodes[i];
|
|
|
|
if(elNode.nodeType === 3){
|
|
elNode = elNode.parentNode;
|
|
}
|
|
|
|
if(elParentNode && (elNode === elParentNode || jindo.$Element(elNode).isChildOf(elParentNode))){
|
|
continue;
|
|
}
|
|
|
|
elParentNode = this._getAlignNode(elNode);
|
|
|
|
if(elParentNode && elParentNode.align !== elParentNode.style.textAlign){ // [SMARTEDITORSUS-704] align 속성과 text-align 속성의 값을 맞춰줌
|
|
elParentNode.style.textAlign = sAlign;
|
|
elParentNode.setAttribute("align", sAlign);
|
|
}
|
|
}
|
|
},
|
|
|
|
_getDocumentBR : function(){
|
|
var i, nLen;
|
|
|
|
// [COM-715] <Chrome/Safari> 요약글 삽입 > 더보기 영역에서 기호매기기, 번호매기기 설정할때마다 요약글 박스가 아래로 이동됨
|
|
// ExecCommand를 처리하기 전에 현재의 BR을 저장
|
|
|
|
this.aBRs = this.oApp.getWYSIWYGDocument().getElementsByTagName("BR");
|
|
this.aBeforeBRs = [];
|
|
|
|
for(i=0, nLen=this.aBRs.length; i<nLen; i++){
|
|
this.aBeforeBRs[i] = this.aBRs[i];
|
|
}
|
|
},
|
|
|
|
_fixDocumentBR : function(){
|
|
// [COM-715] ExecCommand가 처리된 후에 업데이트된 BR을 처리 전에 저장한 BR과 비교하여 생성된 BR을 제거
|
|
|
|
if(this.aBeforeBRs.length === this.aBRs.length){ // this.aBRs gets updated automatically when the document is updated
|
|
return;
|
|
}
|
|
|
|
var waBeforeBRs = jindo.$A(this.aBeforeBRs),
|
|
i, iLen = this.aBRs.length;
|
|
|
|
for(i=iLen-1; i>=0; i--){
|
|
if(waBeforeBRs.indexOf(this.aBRs[i])<0){
|
|
this.aBRs[i].parentNode.removeChild(this.aBRs[i]);
|
|
}
|
|
}
|
|
},
|
|
|
|
_setBlockExecCommand : function(sCommand, bUserInterface, vValue){
|
|
var aNodes, aChildrenNode, htSelectedTDs = {};
|
|
this.oSelection = this.oApp.getSelection();
|
|
this.oApp.exec("GET_SELECTED_TD_BLOCK",['aTdCells',htSelectedTDs]);
|
|
aNodes = htSelectedTDs.aTdCells;
|
|
|
|
for( var j = 0; j < aNodes.length ; j++){
|
|
|
|
this.oSelection.selectNodeContents(aNodes[j]);
|
|
this.oSelection.select();
|
|
|
|
if(this.oNavigator.firefox){
|
|
this.oEditingArea.execCommand("styleWithCSS", bUserInterface, false); //styleWithCSS는 ff전용임.
|
|
}
|
|
|
|
aChildrenNode = this.oSelection.getNodes();
|
|
for( var k = 0; k < aChildrenNode.length ; k++ ) {
|
|
if(aChildrenNode[k].tagName == "UL" || aChildrenNode[k].tagName == "OL" ){
|
|
jindo.$Element(aChildrenNode[k]).css("color",vValue);
|
|
}
|
|
}
|
|
this.oEditingArea.execCommand(sCommand, bUserInterface, vValue);
|
|
}
|
|
},
|
|
|
|
_indentMargin : function(elDiv){
|
|
var elTmp = elDiv,
|
|
aAppend, i, nLen, elInsertTarget, elDeleteTarget, nCurMarginLeft;
|
|
|
|
while(elTmp){
|
|
if(elTmp.tagName && elTmp.tagName === "LI"){
|
|
elDiv = elTmp;
|
|
break;
|
|
}
|
|
elTmp = elTmp.parentNode;
|
|
}
|
|
|
|
if(elDiv.tagName === "LI"){
|
|
//<OL>
|
|
// <OL>
|
|
// <LI>22</LI>
|
|
// </OL>
|
|
// <LI>33</LI>
|
|
//</OL>
|
|
//와 같은 형태라면 33을 들여쓰기 했을 때, 상단의 silbling OL과 합쳐서 아래와 같이 만들어 줌.
|
|
//<OL>
|
|
// <OL>
|
|
// <LI>22</LI>
|
|
// <LI>33</LI>
|
|
// </OL>
|
|
//</OL>
|
|
if(elDiv.previousSibling && elDiv.previousSibling.tagName && elDiv.previousSibling.tagName === elDiv.parentNode.tagName){
|
|
// 하단에 또다른 OL이 있어 아래와 같은 형태라면,
|
|
//<OL>
|
|
// <OL>
|
|
// <LI>22</LI>
|
|
// </OL>
|
|
// <LI>33</LI>
|
|
// <OL>
|
|
// <LI>44</LI>
|
|
// </OL>
|
|
//</OL>
|
|
//22,33,44를 합쳐서 아래와 같이 만들어 줌.
|
|
//<OL>
|
|
// <OL>
|
|
// <LI>22</LI>
|
|
// <LI>33</LI>
|
|
// <LI>44</LI>
|
|
// </OL>
|
|
//</OL>
|
|
if(elDiv.nextSibling && elDiv.nextSibling.tagName && elDiv.nextSibling.tagName === elDiv.parentNode.tagName){
|
|
aAppend = [elDiv];
|
|
|
|
for(i=0, nLen=elDiv.nextSibling.childNodes.length; i<nLen; i++){
|
|
aAppend.push(elDiv.nextSibling.childNodes[i]);
|
|
}
|
|
|
|
elInsertTarget = elDiv.previousSibling;
|
|
elDeleteTarget = elDiv.nextSibling;
|
|
|
|
for(i=0, nLen=aAppend.length; i<nLen; i++){
|
|
elInsertTarget.insertBefore(aAppend[i], null);
|
|
}
|
|
|
|
elDeleteTarget.parentNode.removeChild(elDeleteTarget);
|
|
}else{
|
|
elDiv.previousSibling.insertBefore(elDiv, null);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
//<OL>
|
|
// <LI>22</LI>
|
|
// <OL>
|
|
// <LI>33</LI>
|
|
// </OL>
|
|
//</OL>
|
|
//와 같은 형태라면 22을 들여쓰기 했을 때, 하단의 silbling OL과 합친다.
|
|
if(elDiv.nextSibling && elDiv.nextSibling.tagName && elDiv.nextSibling.tagName === elDiv.parentNode.tagName){
|
|
elDiv.nextSibling.insertBefore(elDiv, elDiv.nextSibling.firstChild);
|
|
return;
|
|
}
|
|
|
|
elTmp = elDiv.parentNode.cloneNode(false);
|
|
elDiv.parentNode.insertBefore(elTmp, elDiv);
|
|
elTmp.appendChild(elDiv);
|
|
return;
|
|
}
|
|
|
|
nCurMarginLeft = parseInt(elDiv.style.marginLeft, 10);
|
|
|
|
if(!nCurMarginLeft){
|
|
nCurMarginLeft = 0;
|
|
}
|
|
|
|
nCurMarginLeft += this.nIndentSpacing;
|
|
elDiv.style.marginLeft = nCurMarginLeft+"px";
|
|
},
|
|
|
|
_outdentMargin : function(elDiv){
|
|
var elTmp = elDiv,
|
|
elParentNode, elInsertBefore, elNewParent, elInsertParent, oDoc, nCurMarginLeft;
|
|
|
|
while(elTmp){
|
|
if(elTmp.tagName && elTmp.tagName === "LI"){
|
|
elDiv = elTmp;
|
|
break;
|
|
}
|
|
elTmp = elTmp.parentNode;
|
|
}
|
|
|
|
if(elDiv.tagName === "LI"){
|
|
elParentNode = elDiv.parentNode;
|
|
elInsertBefore = elDiv.parentNode;
|
|
|
|
// LI를 적절 위치로 이동.
|
|
// 위에 다른 li/ol/ul가 있는가?
|
|
if(elDiv.previousSibling && elDiv.previousSibling.tagName && elDiv.previousSibling.tagName.match(/LI|UL|OL/)){
|
|
// 위아래로 sibling li/ol/ul가 있다면 ol/ul를 2개로 나누어야됨
|
|
if(elDiv.nextSibling && elDiv.nextSibling.tagName && elDiv.nextSibling.tagName.match(/LI|UL|OL/)){
|
|
elNewParent = elParentNode.cloneNode(false);
|
|
|
|
while(elDiv.nextSibling){
|
|
elNewParent.insertBefore(elDiv.nextSibling, null);
|
|
}
|
|
|
|
elParentNode.parentNode.insertBefore(elNewParent, elParentNode.nextSibling);
|
|
elInsertBefore = elNewParent;
|
|
// 현재 LI가 마지막 LI라면 부모 OL/UL 하단에 삽입
|
|
}else{
|
|
elInsertBefore = elParentNode.nextSibling;
|
|
}
|
|
}
|
|
elParentNode.parentNode.insertBefore(elDiv, elInsertBefore);
|
|
|
|
// 내어쓰기 한 LI 외에 다른 LI가 존재 하지 않을 경우 부모 노드 지워줌
|
|
if(!elParentNode.innerHTML.match(/LI/i)){
|
|
elParentNode.parentNode.removeChild(elParentNode);
|
|
}
|
|
|
|
// OL이나 UL 위로까지 내어쓰기가 된 상태라면 LI를 벗겨냄
|
|
if(!elDiv.parentNode.tagName.match(/OL|UL/)){
|
|
elInsertParent = elDiv.parentNode;
|
|
elInsertBefore = elDiv;
|
|
|
|
// 내용물을 P로 감싸기
|
|
oDoc = this.oApp.getWYSIWYGDocument();
|
|
elInsertParent = oDoc.createElement("P");
|
|
elInsertBefore = null;
|
|
|
|
elDiv.parentNode.insertBefore(elInsertParent, elDiv);
|
|
|
|
while(elDiv.firstChild){
|
|
elInsertParent.insertBefore(elDiv.firstChild, elInsertBefore);
|
|
}
|
|
elDiv.parentNode.removeChild(elDiv);
|
|
}
|
|
return;
|
|
}
|
|
nCurMarginLeft = parseInt(elDiv.style.marginLeft, 10);
|
|
|
|
if(!nCurMarginLeft){
|
|
nCurMarginLeft = 0;
|
|
}
|
|
|
|
nCurMarginLeft -= this.nIndentSpacing;
|
|
|
|
if(nCurMarginLeft < 0){
|
|
nCurMarginLeft = 0;
|
|
}
|
|
|
|
elDiv.style.marginLeft = nCurMarginLeft+"px";
|
|
},
|
|
|
|
// Fix IE's execcommand bug
|
|
// When insertorderedlist/insertunorderedlist is executed on a blockquote, the blockquote will "suck in" directly neighboring OL, UL's if there's any.
|
|
// To prevent this, insert empty P tags right before and after the blockquote and remove them after the execution.
|
|
// [SMARTEDITORSUS-793] Chrome 에서 동일한 이슈 발생, Chrome 은 빈 P 태그로는 처리되지 않으 추가
|
|
// [SMARTEDITORSUS-1857] 인용구내에 UL/OL이 있고 바깥에서 UL/OL을 실행하는 경우도 동일한 문제가 발생하여 동일한 방식으로 해결하도록 해당 케이스 추가
|
|
_insertBlankLine : function(){
|
|
var oSelection = this.oApp.getSelection();
|
|
var elNode = oSelection.commonAncestorContainer;
|
|
this.elP1 = null;
|
|
this.elP2 = null;
|
|
|
|
// [SMARTEDITORSUS-793] 인용구 안에서 글머리기호/번호매기기하는 경우에 대한 처리
|
|
while(elNode){
|
|
if(elNode.tagName == "BLOCKQUOTE"){
|
|
this.elP1 = jindo.$("<p> </p>", this.oApp.getWYSIWYGDocument());
|
|
elNode.parentNode.insertBefore(this.elP1, elNode);
|
|
|
|
this.elP2 = jindo.$("<p> </p>", this.oApp.getWYSIWYGDocument());
|
|
elNode.parentNode.insertBefore(this.elP2, elNode.nextSibling);
|
|
|
|
break;
|
|
}
|
|
elNode = elNode.parentNode;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1857] 인용구 바깥에서 글머리기호/번호매기기하는 경우에 대한 처리
|
|
if(!this.elP1 && !this.elP2){
|
|
elNode = oSelection.commonAncestorContainer;
|
|
elNode = (elNode.nodeType !== 1) ? elNode.parentNode : elNode;
|
|
var elPrev = elNode.previousSibling,
|
|
elNext = elNode.nextSibling;
|
|
|
|
if(elPrev && elPrev.tagName === "BLOCKQUOTE"){
|
|
this.elP1 = jindo.$("<p> </p>", this.oApp.getWYSIWYGDocument());
|
|
elPrev.parentNode.insertBefore(this.elP1, elPrev.nextSibling);
|
|
}
|
|
if(elNext && elNext.tagName === "BLOCKQUOTE"){
|
|
this.elP1 = jindo.$("<p> </p>", this.oApp.getWYSIWYGDocument());
|
|
elNext.parentNode.insertBefore(this.elP1, elNext);
|
|
}
|
|
}
|
|
},
|
|
|
|
// Fix IE's execcommand bug
|
|
// When insertorderedlist/insertunorderedlist is executed on a blockquote with all the child nodes selected,
|
|
// eg:<blockquote>[selection starts here]blah...[selection ends here]</blockquote>
|
|
// , IE will change the blockquote with the list tag and create <OL><OL><LI>blah...</LI></OL></OL>.
|
|
// (two OL's or two UL's depending on which command was executed)
|
|
//
|
|
// It can also happen when the cursor is located at bogus positions like
|
|
// * below blockquote when the blockquote is the last element in the document
|
|
//
|
|
// [IE] 인용구 안에서 글머리 기호를 적용했을 때, 인용구 밖에 적용된 번호매기기/글머리 기호가 인용구 안으로 빨려 들어가는 문제 처리
|
|
_fixCorruptedBlockQuote : function(sTagName){
|
|
var aNodes = this.oApp.getWYSIWYGDocument().getElementsByTagName(sTagName),
|
|
elCorruptedBlockQuote, elTmpParent, elNewNode, aLists,
|
|
i, nLen, nPos, el, oSelection;
|
|
|
|
for(i=0, nLen=aNodes.length; i<nLen; i++){
|
|
if(aNodes[i].firstChild && aNodes[i].firstChild.tagName == sTagName){
|
|
elCorruptedBlockQuote = aNodes[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(!elCorruptedBlockQuote){return;}
|
|
|
|
elTmpParent = elCorruptedBlockQuote.parentNode;
|
|
|
|
// (1) changing outerHTML will cause loss of the reference to the node, so remember the idx position here
|
|
nPos = this._getPosIdx(elCorruptedBlockQuote);
|
|
el = this.oApp.getWYSIWYGDocument().createElement("DIV");
|
|
el.innerHTML = elCorruptedBlockQuote.outerHTML.replace("<"+sTagName, "<BLOCKQUOTE");
|
|
elCorruptedBlockQuote.parentNode.insertBefore(el.firstChild, elCorruptedBlockQuote);
|
|
elCorruptedBlockQuote.parentNode.removeChild(elCorruptedBlockQuote);
|
|
|
|
// (2) and retrieve the new node here
|
|
elNewNode = elTmpParent.childNodes[nPos];
|
|
|
|
// garbage <OL></OL> or <UL></UL> will be left over after setting the outerHTML, so remove it here.
|
|
aLists = elNewNode.getElementsByTagName(sTagName);
|
|
|
|
for(i=0, nLen=aLists.length; i<nLen; i++){
|
|
if(aLists[i].childNodes.length<1){
|
|
aLists[i].parentNode.removeChild(aLists[i]);
|
|
}
|
|
}
|
|
|
|
oSelection = this.oApp.getEmptySelection();
|
|
oSelection.selectNodeContents(elNewNode);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1795] UL/OL 삽입시 LI 하위에 BLOCKQUOTE 가 있으면 제거한다.
|
|
* <blockquote><p><ul><li><span class="Apple-style-span"><blockquote><p style="display: inline !important;">선택영역</p></blockquote></span></li></ul></p><blockquote>
|
|
* 삭제될때도 복사됨
|
|
* <blockquote><p><span class="Apple-style-span"><blockquote><p style="display: inline !important;">선택영역</p></blockquote></span></p><blockquote>
|
|
*/
|
|
_removeBlockQuote : function(){
|
|
var sVendorSpanClass = "Apple-style-span",
|
|
elVendorSpan,
|
|
aelVendorSpanDummy,
|
|
oSelection = this.oApp.getSelection(),
|
|
elNode = oSelection.commonAncestorContainer,
|
|
elChild = elNode,
|
|
elLi;
|
|
|
|
// LI 와 SPAN.Apple-style-span 를 찾는다.
|
|
while(elNode){
|
|
if(elNode.tagName === "LI"){
|
|
elLi = elNode;
|
|
elNode = (elNode.style.cssText === "display: inline !important; ") ? elNode.parentNode : null;
|
|
}else if(elNode.tagName === "SPAN" && elNode.className === sVendorSpanClass){
|
|
elVendorSpan = elNode;
|
|
elNode = (!elLi) ? elNode.parentNode : null;
|
|
}else{
|
|
elNode = elNode.parentNode;
|
|
}
|
|
}
|
|
// SPAN.Apple-style-span 을 selection 된 텍스트로 교체한 후 다시 selection을 준다.
|
|
if(elLi && elVendorSpan){
|
|
elNode = elVendorSpan.parentNode;
|
|
elNode.replaceChild(elChild, elVendorSpan);
|
|
oSelection.selectNodeContents(elNode);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
}
|
|
// BLOCKQUOTE 내에 남아있는 SPAN.Apple-style-span 을 제거한다.(UL과 OL 교체시 남게되는 더미 SPAN 제거용)
|
|
while(elNode){
|
|
if(elNode.tagName === "BLOCKQUOTE"){
|
|
aelVendorSpanDummy = elNode.getElementsByClassName(sVendorSpanClass);
|
|
for(var i = 0;(elVendorSpan = aelVendorSpanDummy[i]); i++){
|
|
elVendorSpan.parentNode.removeChild(elVendorSpan);
|
|
}
|
|
elNode = null;
|
|
}else{
|
|
elNode = elNode.parentNode;
|
|
}
|
|
}
|
|
},
|
|
|
|
_getPosIdx : function(refNode){
|
|
var idx = 0;
|
|
for(var node = refNode.previousSibling; node; node = node.previousSibling){idx++;}
|
|
|
|
return idx;
|
|
},
|
|
|
|
_countClickCr : function(sCommand) {
|
|
if (!sCommand.match(this.rxClickCr)) {
|
|
return;
|
|
}
|
|
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', [sCommand.replace(/^insert/i, '')]);
|
|
},
|
|
|
|
_extendBlock : function(){
|
|
// [SMARTEDITORSUS-663] [FF] block단위로 확장하여 Range를 새로 지정해주는것이 원래 스펙이므로
|
|
// 해결을 위해서는 현재 선택된 부분을 Block으로 extend하여 execCommand API가 처리될 수 있도록 함
|
|
|
|
var oSelection = this.oApp.getSelection(),
|
|
oStartContainer = oSelection.startContainer,
|
|
oEndContainer = oSelection.endContainer,
|
|
aChildImg = [],
|
|
aSelectedImg = [],
|
|
oSelectionClone = oSelection.cloneRange();
|
|
|
|
// <p><img><br/><img><br/><img></p> 일 때 이미지가 일부만 선택되면 발생
|
|
// - container 노드는 P 이고 container 노드의 자식노드 중 이미지가 여러개인데 선택된 이미지가 그 중 일부인 경우
|
|
|
|
if(!(oStartContainer === oEndContainer && oStartContainer.nodeType === 1 && oStartContainer.tagName === "P")){
|
|
return;
|
|
}
|
|
|
|
aChildImg = jindo.$A(oStartContainer.childNodes).filter(function(value, index, array){
|
|
return (value.nodeType === 1 && value.tagName === "IMG");
|
|
}).$value();
|
|
|
|
aSelectedImg = jindo.$A(oSelection.getNodes()).filter(function(value, index, array){
|
|
return (value.nodeType === 1 && value.tagName === "IMG");
|
|
}).$value();
|
|
|
|
if(aChildImg.length <= aSelectedImg.length){
|
|
return;
|
|
}
|
|
|
|
oSelection.selectNode(oStartContainer);
|
|
oSelection.select();
|
|
|
|
return oSelectionClone;
|
|
}
|
|
});
|
|
//}
|
|
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to styling the font
|
|
* @name hp_SE_WYSIWYGStyler.js
|
|
* @required SE_EditingArea_WYSIWYG, HuskyRangeManager
|
|
*/
|
|
nhn.husky.SE_WYSIWYGStyler = jindo.$Class({
|
|
name : "SE_WYSIWYGStyler",
|
|
_sCursorHolder : "\uFEFF",
|
|
|
|
$init : function(){
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
|
|
if(htBrowser.ie && htBrowser.version > 8){
|
|
// [SMARTEDITORSUS-178] ZWNBSP(\uFEFF) 를 사용하면 IE9 이상의 경우 높이값을 갖지 못해 커서위치가 이상함
|
|
// [SMARTEDITORSUS-1704] ZWSP(\u200B) 를 사용할 경우 줄바꿈이 됨
|
|
// 기본적으로 \uFEFF 를 사용하고 IE9 이상만 \u2060 사용 (\u2060 은 \uFEFF 와 동일한 역할을 하지만 크롬에서는 깨짐)
|
|
// *주의* 작성자가 IE9이상에서 작성하고 독자가 크롬에서 볼 경우 \u2060 가 깨진 문자로 보여질 수 있기 때문에 컨버터를 통해 \u2060 를 \uFEFF 로 변환한다.
|
|
// FIXME: 단, \u2060 를 \uFEFF 변환으로 인해 SPAN태그만 들어있는 상태에서 모드를 변환하면 커서 위치가 다시 이상해질 수 있음
|
|
// 참고:
|
|
// http://en.wikipedia.org/wiki/Universal_Character_Set_characters#Word_joiners_and_separators
|
|
// http://en.wikipedia.org/wiki/Zero-width_no-break_space
|
|
// https://www.cs.tut.fi/~jkorpela/chars/spaces.html
|
|
this._sCursorHolder = "\u2060";
|
|
this.$ON_REGISTER_CONVERTERS = function(){
|
|
var rx2060 = /\u2060/g;
|
|
this.oApp.exec("ADD_CONVERTER", ["WYSIWYG_TO_IR", jindo.$Fn(function(sContents){
|
|
return sContents.replace(rx2060, "\uFEFF");
|
|
}, this).bind()]);
|
|
};
|
|
}
|
|
},
|
|
|
|
$PRECONDITION : function(sFullCommand, aArgs){
|
|
return (this.oApp.getEditingMode() == "WYSIWYG");
|
|
},
|
|
|
|
$ON_SET_WYSIWYG_STYLE : function(oStyles){
|
|
var oSelection = this.oApp.getSelection();
|
|
var htSelectedTDs = {};
|
|
this.oApp.exec("IS_SELECTED_TD_BLOCK",['bIsSelectedTd',htSelectedTDs]);
|
|
var bSelectedBlock = htSelectedTDs.bIsSelectedTd;
|
|
|
|
// style cursor or !(selected block)
|
|
if(oSelection.collapsed && !bSelectedBlock){
|
|
this.oApp.exec("RECORD_UNDO_ACTION", ["FONT STYLE", {bMustBlockElement : true}]);
|
|
|
|
var oSpan, bNewSpan = false;
|
|
var elCAC = oSelection.commonAncestorContainer;
|
|
//var elCAC = nhn.CurrentSelection.getCommonAncestorContainer();
|
|
if(elCAC.nodeType == 3){
|
|
elCAC = elCAC.parentNode;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1648] SPAN > 굵게/밑줄/기울림/취소선이 있는 경우, 상위 SPAN을 찾는다.
|
|
if(elCAC && oSelection._rxCursorHolder.test(elCAC.innerHTML)){
|
|
oSpan = oSelection._findParentSingleSpan(elCAC);
|
|
}
|
|
// 스타일을 적용할 SPAN이 없으면 새로 생성
|
|
if(!oSpan){
|
|
oSpan = this.oApp.getWYSIWYGDocument().createElement("SPAN");
|
|
oSpan.innerHTML = this._sCursorHolder;
|
|
bNewSpan = true;
|
|
}else if(oSpan.innerHTML == ""){ // 내용이 아예 없으면 크롬에서 커서가 위치하지 못함
|
|
oSpan.innerHTML = this._sCursorHolder;
|
|
}
|
|
|
|
var sValue;
|
|
for(var sName in oStyles){
|
|
sValue = oStyles[sName];
|
|
|
|
if(typeof sValue != "string"){
|
|
continue;
|
|
}
|
|
|
|
oSpan.style[sName] = sValue;
|
|
}
|
|
|
|
if(bNewSpan){
|
|
if(oSelection.startContainer.tagName == "BODY" && oSelection.startOffset === 0){
|
|
var oVeryFirstNode = oSelection._getVeryFirstRealChild(this.oApp.getWYSIWYGDocument().body);
|
|
|
|
var bAppendable = true;
|
|
var elTmp = oVeryFirstNode.cloneNode(false);
|
|
// some browsers may throw an exception for trying to set the innerHTML of BR/IMG tags
|
|
try{
|
|
elTmp.innerHTML = "test";
|
|
|
|
if(elTmp.innerHTML != "test"){
|
|
bAppendable = false;
|
|
}
|
|
}catch(e){
|
|
bAppendable = false;
|
|
}
|
|
|
|
if(bAppendable && elTmp.nodeType == 1 && elTmp.tagName == "BR"){// [SMARTEDITORSUS-311] [FF4] Cursor Holder 인 BR 의 하위노드로 SPAN 을 추가하여 발생하는 문제
|
|
oSelection.selectNode(oVeryFirstNode);
|
|
oSelection.collapseToStart();
|
|
oSelection.insertNode(oSpan);
|
|
}else if(bAppendable && oVeryFirstNode.tagName != "IFRAME" && oVeryFirstNode.appendChild && typeof oVeryFirstNode.innerHTML == "string"){
|
|
oVeryFirstNode.appendChild(oSpan);
|
|
}else{
|
|
oSelection.selectNode(oVeryFirstNode);
|
|
oSelection.collapseToStart();
|
|
oSelection.insertNode(oSpan);
|
|
}
|
|
}else{
|
|
oSelection.collapseToStart();
|
|
oSelection.insertNode(oSpan);
|
|
}
|
|
}else{
|
|
oSelection = this.oApp.getEmptySelection();
|
|
}
|
|
|
|
// [SMARTEDITORSUS-229] 새로 생성되는 SPAN 에도 취소선/밑줄 처리 추가
|
|
if(!!oStyles.color){
|
|
oSelection._checkTextDecoration(oSpan);
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1648] oSpan이 굵게//밑줄/기울임/취소선태그보다 상위인 경우, IE에서 굵게//밑줄/기울임/취소선태그 밖으로 나가게 된다. 때문에 SPAN을 새로 만든 경우 oSpan을, 그렇지 않은 경우 elCAC를 잡는다.
|
|
oSelection.selectNodeContents(bNewSpan?oSpan:elCAC);
|
|
oSelection.collapseToEnd();
|
|
// TODO: focus 는 왜 있는 것일까? => IE에서 style 적용후 포커스가 날아가서 글작성이 안됨???
|
|
oSelection._window.focus();
|
|
oSelection._window.document.body.focus();
|
|
oSelection.select();
|
|
|
|
// 영역으로 스타일이 잡혀 있는 경우(예:현재 커서가 B블럭 안에 존재) 해당 영역이 사라져 버리는 오류 발생해서 제거
|
|
// http://bts.nhncorp.com/nhnbts/browse/COM-912
|
|
/*
|
|
var oCursorStyle = this.oApp.getCurrentStyle();
|
|
if(oCursorStyle.bold == "@^"){
|
|
this.oApp.delayedExec("EXECCOMMAND", ["bold"], 0);
|
|
}
|
|
if(oCursorStyle.underline == "@^"){
|
|
this.oApp.delayedExec("EXECCOMMAND", ["underline"], 0);
|
|
}
|
|
if(oCursorStyle.italic == "@^"){
|
|
this.oApp.delayedExec("EXECCOMMAND", ["italic"], 0);
|
|
}
|
|
if(oCursorStyle.lineThrough == "@^"){
|
|
this.oApp.delayedExec("EXECCOMMAND", ["strikethrough"], 0);
|
|
}
|
|
*/
|
|
// FF3 will actually display %uFEFF when it is followed by a number AND certain font-family is used(like Gulim), so remove the character for FF3
|
|
//if(jindo.$Agent().navigator().firefox && jindo.$Agent().navigator().version == 3){
|
|
// FF4+ may have similar problems, so ignore the version number
|
|
// [SMARTEDITORSUS-416] 커서가 올라가지 않도록 BR 을 살려둠
|
|
// if(jindo.$Agent().navigator().firefox){
|
|
// oSpan.innerHTML = "";
|
|
// }
|
|
return;
|
|
}
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["FONT STYLE", {bMustBlockElement:true}]);
|
|
|
|
if(bSelectedBlock){
|
|
var aNodes;
|
|
|
|
this.oApp.exec("GET_SELECTED_TD_BLOCK",['aTdCells',htSelectedTDs]);
|
|
aNodes = htSelectedTDs.aTdCells;
|
|
|
|
for( var j = 0; j < aNodes.length ; j++){
|
|
oSelection.selectNodeContents(aNodes[j]);
|
|
oSelection.styleRange(oStyles);
|
|
oSelection.select();
|
|
}
|
|
} else {
|
|
var bCheckTextDecoration = !!oStyles.color; // [SMARTEDITORSUS-26] 취소선/밑줄 색상 적용 처리
|
|
var bIncludeLI = oStyles.fontSize || oStyles.fontFamily;
|
|
// [SMARTEDITORSUS-2237] IE11 에서 엑셀을 복사붙여넣기하면 유효하지 않은 위치(tbody, tr, td 사이)에 font 태그가 삽입된다.
|
|
// oSelection.styleRange 을 실행하면 유효하지 않은 위치의 태그들에 스타일지정용 span 이 삽입되어 테이블이 깨지게 된다.
|
|
// 때문에 styleRange 실행전에 잘못된 노드들을 제거해준다.
|
|
for(var i = 0, aNodes = oSelection.getNodes(), oNode;(oNode = aNodes[i]); i++){
|
|
nhn.husky.SE2M_Utils.removeInvalidNodeInTable(oNode);
|
|
}
|
|
|
|
oSelection.styleRange(oStyles, null, null, bIncludeLI, bCheckTextDecoration);
|
|
|
|
// http://bts.nhncorp.com/nhnbts/browse/COM-964
|
|
//
|
|
// In FF when,
|
|
// 1) Some text was wrapped with a styling SPAN and a bogus BR is followed
|
|
// eg: <span style="XXX">TEST</span><br>
|
|
// 2) And some place outside the span is clicked.
|
|
//
|
|
// The text cursor will be located outside the SPAN like the following,
|
|
// <span style="XXX">TEST</span>[CURSOR]<br>
|
|
//
|
|
// which is not what the user would expect
|
|
// Desired result: <span style="XXX">TEST[CURSOR]</span><br>
|
|
//
|
|
// To make the cursor go inside the styling SPAN, remove the bogus BR when the styling SPAN is created.
|
|
// -> Style TEST<br> as <span style="XXX">TEST</span> (remove unnecessary BR)
|
|
// -> Cannot monitor clicks/cursor position real-time so make the contents error-proof instead.
|
|
if(jindo.$Agent().navigator().firefox){
|
|
var aStyleParents = oSelection.aStyleParents;
|
|
for(var i=0, nLen=aStyleParents.length; i<nLen; i++){
|
|
var elNode = aStyleParents[i];
|
|
if(elNode.nextSibling && elNode.nextSibling.tagName == "BR" && !elNode.nextSibling.nextSibling){
|
|
elNode.parentNode.removeChild(elNode.nextSibling);
|
|
}
|
|
}
|
|
}
|
|
|
|
oSelection._window.focus();
|
|
oSelection.select();
|
|
}
|
|
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["FONT STYLE", {bMustBlockElement:true}]);
|
|
}
|
|
});
|
|
//}
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to detecting the style change
|
|
* @name hp_SE_WYSIWYGStyleGetter.js
|
|
*/
|
|
nhn.husky.SE_WYSIWYGStyleGetter = jindo.$Class({
|
|
name : "SE_WYSIWYGStyleGetter",
|
|
|
|
hKeyUp : null,
|
|
|
|
getStyleInterval : 200,
|
|
|
|
oStyleMap : {
|
|
fontFamily : {
|
|
type : "Value",
|
|
css : "fontFamily"
|
|
},
|
|
fontSize : {
|
|
type : "Value",
|
|
css : "fontSize"
|
|
},
|
|
lineHeight : {
|
|
type : "Value",
|
|
css : "lineHeight",
|
|
converter : function(sValue, oStyle){
|
|
if(!sValue.match(/px$/)){
|
|
return sValue;
|
|
}
|
|
|
|
return Math.ceil((parseInt(sValue, 10)/parseInt(oStyle.fontSize, 10))*10)/10;
|
|
}
|
|
},
|
|
bold : {
|
|
command : "bold"
|
|
},
|
|
underline : {
|
|
command : "underline"
|
|
},
|
|
italic : {
|
|
command : "italic"
|
|
},
|
|
lineThrough : {
|
|
command : "strikethrough"
|
|
},
|
|
superscript : {
|
|
command : "superscript"
|
|
},
|
|
subscript : {
|
|
command : "subscript"
|
|
},
|
|
justifyleft : {
|
|
command : "justifyleft"
|
|
},
|
|
justifycenter : {
|
|
command : "justifycenter"
|
|
},
|
|
justifyright : {
|
|
command : "justifyright"
|
|
},
|
|
justifyfull : {
|
|
command : "justifyfull"
|
|
},
|
|
orderedlist : {
|
|
command : "insertorderedlist"
|
|
},
|
|
unorderedlist : {
|
|
command : "insertunorderedlist"
|
|
}
|
|
},
|
|
|
|
$init : function(){
|
|
this.oStyle = this._getBlankStyle();
|
|
},
|
|
|
|
$LOCAL_BEFORE_ALL : function(){
|
|
return (this.oApp.getEditingMode() == "WYSIWYG");
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oDocument = this.oApp.getWYSIWYGDocument();
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getCurrentStyle", jindo.$Fn(this.getCurrentStyle, this).bind()]);
|
|
|
|
if(jindo.$Agent().navigator().safari || jindo.$Agent().navigator().chrome || jindo.$Agent().navigator().ie){
|
|
this.oStyleMap.textAlign = {
|
|
type : "Value",
|
|
css : "textAlign"
|
|
};
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_MOUSEUP : function(oEvnet){
|
|
/*
|
|
if(this.hKeyUp){
|
|
clearTimeout(this.hKeyUp);
|
|
}
|
|
this.oApp.delayedExec("CHECK_STYLE_CHANGE", [], 100);
|
|
*/
|
|
this.oApp.exec("CHECK_STYLE_CHANGE");
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_KEYPRESS : function(oEvent){
|
|
// ctrl+a in FF triggers keypress event with keyCode 97, other browsers don't throw keypress event for ctrl+a
|
|
var oKeyInfo;
|
|
if(this.oApp.oNavigator.firefox){
|
|
oKeyInfo = oEvent.key();
|
|
if(oKeyInfo.ctrl && oKeyInfo.keyCode == 97){
|
|
return;
|
|
}
|
|
}
|
|
|
|
if(this.bAllSelected){
|
|
this.bAllSelected = false;
|
|
return;
|
|
}
|
|
|
|
/*
|
|
// queryCommandState often fails to return correct result for Korean/Enter. So just ignore them.
|
|
if(this.oApp.oNavigator.firefox && (oKeyInfo.keyCode == 229 || oKeyInfo.keyCode == 13)){
|
|
return;
|
|
}
|
|
*/
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE");
|
|
//this.oApp.delayedExec("CHECK_STYLE_CHANGE", [], 0);
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_KEYDOWN : function(oEvent){
|
|
var oKeyInfo = oEvent.key();
|
|
|
|
// ctrl+a
|
|
if((this.oApp.oNavigator.ie || this.oApp.oNavigator.firefox) && oKeyInfo.ctrl && oKeyInfo.keyCode == 65){
|
|
this.oApp.exec("RESET_STYLE_STATUS");
|
|
this.bAllSelected = true;
|
|
return;
|
|
}
|
|
|
|
/*
|
|
backspace 8
|
|
enter 13
|
|
page up 33
|
|
page down 34
|
|
end 35
|
|
home 36
|
|
left arrow 37
|
|
up arrow 38
|
|
right arrow 39
|
|
down arrow 40
|
|
insert 45
|
|
delete 46
|
|
*/
|
|
// other key strokes are taken care by keypress event
|
|
if(!(oKeyInfo.keyCode == 8 || (oKeyInfo.keyCode >= 33 && oKeyInfo.keyCode <= 40) || oKeyInfo.keyCode == 45 || oKeyInfo.keyCode == 46)) return;
|
|
|
|
// [SMARTEDITORSUS-1841] IE11에서 테이블 첫번째 셀에서 shift+end 를 두번 실행하면 오류 발생
|
|
// ctrl+a 를 다루는 방식대로 RESET_STYLE_STATUS 를 수행하고 CHECK_STYLE_CHANGE 는 수행하지 않도록 처리
|
|
if(oKeyInfo.shift && oKeyInfo.keyCode === 35){
|
|
this.oApp.exec("RESET_STYLE_STATUS");
|
|
this.bAllSelected = true;
|
|
return;
|
|
}
|
|
|
|
// take care of ctrl+a -> delete/bksp sequence
|
|
if(this.bAllSelected){
|
|
// firefox will throw both keydown and keypress events for those input(keydown first), so let keypress take care of them
|
|
if(this.oApp.oNavigator.firefox){
|
|
return;
|
|
}
|
|
|
|
this.bAllSelected = false;
|
|
return;
|
|
}
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE");
|
|
},
|
|
|
|
$ON_CHECK_STYLE_CHANGE : function(){
|
|
this._getStyle();
|
|
},
|
|
|
|
$ON_RESET_STYLE_STATUS : function(){
|
|
this.oStyle = this._getBlankStyle();
|
|
var oBodyStyle = this._getStyleOf(this.oApp.getWYSIWYGDocument().body);
|
|
this.oStyle.fontFamily = oBodyStyle.fontFamily;
|
|
this.oStyle.fontSize = oBodyStyle.fontSize;
|
|
this.oStyle["justifyleft"]="@^";
|
|
for(var sAttributeName in this.oStyle){
|
|
//this.oApp.exec("SET_STYLE_STATUS", [sAttributeName, this.oStyle[sAttributeName]]);
|
|
this.oApp.exec("MSG_STYLE_CHANGED", [sAttributeName, this.oStyle[sAttributeName]]);
|
|
}
|
|
},
|
|
|
|
getCurrentStyle : function(){
|
|
return this.oStyle;
|
|
},
|
|
|
|
_check_style_change : function(){
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
},
|
|
|
|
_getBlankStyle : function(){
|
|
var oBlankStyle = {};
|
|
for(var attributeName in this.oStyleMap){
|
|
if(this.oStyleMap[attributeName].type == "Value"){
|
|
oBlankStyle[attributeName] = "";
|
|
}else{
|
|
oBlankStyle[attributeName] = 0;
|
|
}
|
|
}
|
|
|
|
return oBlankStyle;
|
|
},
|
|
|
|
_getStyle : function(){
|
|
var oStyle;
|
|
if(nhn.CurrentSelection.isCollapsed()){
|
|
oStyle = this._getStyleOf(nhn.CurrentSelection.getCommonAncestorContainer());
|
|
}else{
|
|
var oSelection = this.oApp.getSelection();
|
|
|
|
var funcFilter = function(oNode){
|
|
if (!oNode.childNodes || oNode.childNodes.length == 0)
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
var aBottomNodes = oSelection.getNodes(false, funcFilter);
|
|
|
|
if(aBottomNodes.length == 0){
|
|
oStyle = this._getStyleOf(oSelection.commonAncestorContainer);
|
|
}else{
|
|
oStyle = this._getStyleOf(aBottomNodes[0]);
|
|
}
|
|
}
|
|
|
|
for(attributeName in oStyle){
|
|
if(this.oStyleMap[attributeName].converter){
|
|
oStyle[attributeName] = this.oStyleMap[attributeName].converter(oStyle[attributeName], oStyle);
|
|
}
|
|
|
|
if(this.oStyle[attributeName] != oStyle[attributeName]){
|
|
/**
|
|
* [SMARTEDITORSUS-1803] 글꼴을 변경할 때는 글자크기 변경사항은 반영되지 않도록 함 - getComputedStyle() 버그
|
|
*
|
|
* 글꼴이나 글자 크기를 변경할 때마다,
|
|
* this.oApp.exec("CHECK_STYLE_CHANGE")가 호출되는데,
|
|
* 이 때 대상 스타일 뿐 아니라 모든 요소의 변화를 확인하게 된다.
|
|
*
|
|
* 글꼴만 변경하는 경우에도
|
|
* getComputedStyle() 반올림 방식으로 인한 오차로 인해
|
|
* pt 단위의 글자크기가 px로 바뀌게 되는데,
|
|
*
|
|
* 스타일 변화 확인에는 jindo.$Element().css()를 사용하는데,
|
|
* el.currentStyle - getComputedStyle(el)의 순위로 존재여부를 확인하여 사용한다.
|
|
*
|
|
* getComputedStyle(el)을 사용하는 경우,
|
|
* 대상 엘리먼트에 pt 단위의 값이 지정되어 있었다면
|
|
* 다음의 순서를 거친다.
|
|
* - pt 단위를 px 단위로 변환
|
|
* - 소수점 이하 값을 반올림
|
|
*
|
|
* 글자 크기의 경우 이 영향으로
|
|
* 산술적인 pt-px 변환이 아닌 값으로 변경되어
|
|
* 툴바에 노출되는 값 계산에 사용될 수 있다.
|
|
* */
|
|
if((typeof(document.body.currentStyle) != "object") && (typeof(getComputedStyle) == "function")){
|
|
if((attributeName == "fontSize") && (this.oStyle["fontFamily"] != oStyle["fontFamily"])){
|
|
continue;
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1803]
|
|
this.oApp.exec("MSG_STYLE_CHANGED", [attributeName, oStyle[attributeName]]);
|
|
}
|
|
}
|
|
|
|
this.oStyle = oStyle;
|
|
},
|
|
|
|
_getStyleOf : function(oNode){
|
|
var oStyle = this._getBlankStyle();
|
|
|
|
// this must not happen
|
|
if(!oNode){
|
|
return oStyle;
|
|
}
|
|
|
|
if( oNode.nodeType == 3 ){
|
|
oNode = oNode.parentNode;
|
|
}else if( oNode.nodeType == 9 ){
|
|
//document에는 css를 적용할 수 없음.
|
|
oNode = oNode.body;
|
|
}
|
|
|
|
var welNode = jindo.$Element(oNode);
|
|
var attribute, cssName;
|
|
|
|
for(var styleName in this.oStyle){
|
|
attribute = this.oStyleMap[styleName];
|
|
if(attribute.type && attribute.type == "Value"){
|
|
try{
|
|
if(attribute.css){
|
|
var sValue = welNode.css(attribute.css);
|
|
if(styleName == "fontFamily"){
|
|
sValue = sValue.split(/,/)[0];
|
|
}
|
|
|
|
oStyle[styleName] = sValue;
|
|
} else if(attribute.command){
|
|
oStyle[styleName] = this.oDocument.queryCommandState(attribute.command);
|
|
} else {
|
|
// todo
|
|
}
|
|
}catch(e){}
|
|
}else{
|
|
if(attribute.command){
|
|
try{
|
|
if(this.oDocument.queryCommandState(attribute.command)){
|
|
oStyle[styleName] = "@^";
|
|
}else{
|
|
oStyle[styleName] = "@-";
|
|
}
|
|
}catch(e){}
|
|
}else{
|
|
// todo
|
|
}
|
|
}
|
|
}
|
|
|
|
switch(oStyle["textAlign"]){
|
|
case "left":
|
|
oStyle["justifyleft"]="@^";
|
|
break;
|
|
case "center":
|
|
oStyle["justifycenter"]="@^";
|
|
break;
|
|
case "right":
|
|
oStyle["justifyright"]="@^";
|
|
break;
|
|
case "justify":
|
|
oStyle["justifyfull"]="@^";
|
|
break;
|
|
}
|
|
|
|
// IE에서는 기본 정렬이 queryCommandState로 넘어오지 않아서 정렬이 없다면, 왼쪽 정렬로 가정함
|
|
if(oStyle["justifyleft"]=="@-" && oStyle["justifycenter"]=="@-" && oStyle["justifyright"]=="@-" && oStyle["justifyfull"]=="@-"){oStyle["justifyleft"]="@^";}
|
|
|
|
return oStyle;
|
|
}
|
|
});
|
|
//}
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to changing the font size using Select element
|
|
* @name SE2M_FontSizeWithLayerUI.js
|
|
*/
|
|
nhn.husky.SE2M_FontSizeWithLayerUI = jindo.$Class({
|
|
name : "SE2M_FontSizeWithLayerUI",
|
|
|
|
$init : function(elAppContainer){
|
|
this._assignHTMLElements(elAppContainer);
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
//@ec
|
|
this.oDropdownLayer = jindo.$$.getSingle("DIV.husky_se_fontSize_layer", elAppContainer);
|
|
|
|
//@ec[
|
|
this.elFontSizeLabel = jindo.$$.getSingle("SPAN.husky_se2m_current_fontSize", elAppContainer);
|
|
this.aLIFontSizes = jindo.$A(jindo.$$("LI", this.oDropdownLayer)).filter(function(v,i,a){return (v.firstChild != null);})._array;
|
|
//@ec]
|
|
|
|
this.sDefaultText = this.elFontSizeLabel.innerHTML;
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["fontSize", "click", "SE2M_TOGGLE_FONTSIZE_LAYER"]);
|
|
this.oApp.exec("SE2_ATTACH_HOVER_EVENTS", [this.aLIFontSizes]);
|
|
|
|
for(var i=0; i<this.aLIFontSizes.length; i++){
|
|
this.oApp.registerBrowserEvent(this.aLIFontSizes[i], "click", "SET_FONTSIZE", [this._getFontSizeFromLI(this.aLIFontSizes[i])]);
|
|
}
|
|
},
|
|
|
|
$ON_SE2M_TOGGLE_FONTSIZE_LAYER : function(){
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.oDropdownLayer, null, "SELECT_UI", ["fontSize"], "DESELECT_UI", ["fontSize"]]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['size']);
|
|
},
|
|
|
|
_rxPX : /px$/i,
|
|
_rxPT : /pt$/i,
|
|
|
|
$ON_MSG_STYLE_CHANGED : function(sAttributeName, sAttributeValue){
|
|
if(sAttributeName == "fontSize"){
|
|
// [SMARTEDITORSUS-1600]
|
|
if(this._rxPX.test(sAttributeValue)){
|
|
// as-is
|
|
/*
|
|
if(sAttributeValue.match(/px$/)){
|
|
var num = parseFloat(sAttributeValue.replace("px", "")).toFixed(0);
|
|
if(this.mapPX2PT[num]){
|
|
sAttributeValue = this.mapPX2PT[num] + "pt";
|
|
}else{
|
|
if(sAttributeValue > 0){
|
|
sAttributeValue = num + "px";
|
|
}else{
|
|
sAttributeValue = this.sDefaultText;
|
|
}
|
|
}*/
|
|
|
|
/**
|
|
* Chrome의 경우,
|
|
* jindo.$Element().css()에서 대상 Element에 구하고자 하는 style 값이 명시되어 있지 않다면,
|
|
* 실제 수행되는 메서드는 window.getComputedStyle()이다.
|
|
*
|
|
* 이 메서드를 거치면 px 단위로 값을 가져오게 되는데,
|
|
* WYSIWYGDocument.body에 pt 단위로 값이 설정되어 있었다면
|
|
* px : pt = 72 : 96 의 비례에 의해
|
|
* 변환된 px 값을 획득하게 되며,
|
|
*
|
|
* 아래 parseFloat()의 특성 상
|
|
* 소수점 16자리부터는 버려질 수 있으며,
|
|
*
|
|
* 이 경우 발생할 수 있는 오차는
|
|
* pt 기준으로 3.75E-16 pt이다.
|
|
*
|
|
* 0.5pt 크기로 구간을 설정하였기 때문에
|
|
* 이 오차는 설정에 지장을 주지 않는다.
|
|
*
|
|
* 위의 기존 방식은 계산을 거치지 않을 뿐 아니라,
|
|
* 소수점 첫째 자리부터 무조건 반올림하기 때문에
|
|
* 결과에 따라 0.375 pt의 오차가 발생할 수 있었다.
|
|
* */
|
|
var num = parseFloat(sAttributeValue.replace(this._rxPX, ""));
|
|
if(num > 0){
|
|
// px : pt = 72 : 96
|
|
sAttributeValue = num * 72 / 96 + "pt";
|
|
}else{
|
|
sAttributeValue = this.sDefaultText;
|
|
}
|
|
// --[SMARTEDITORSUS-1600]
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1600]
|
|
// 산술 계산을 통해 일차적으로 pt로 변환된 값을 0.5pt 구간을 적용하여 보정하되, 보다 가까운 쪽으로 설정한다.
|
|
if(this._rxPT.test(sAttributeValue)){
|
|
var num = parseFloat(sAttributeValue.replace(this._rxPT, ""));
|
|
var integerPart = Math.floor(num); // 정수 부분
|
|
var decimalPart = num - integerPart; // 소수 부분
|
|
|
|
// 보정 기준은 소수 부분이며, 반올림 단위는 0.25pt
|
|
if(decimalPart >= 0 && decimalPart < 0.25){
|
|
num = integerPart + 0;
|
|
}else if(decimalPart >= 0.25 && decimalPart < 0.75){
|
|
num = integerPart + 0.5;
|
|
}else{
|
|
num = integerPart + 1;
|
|
}
|
|
|
|
// 보정된 pt
|
|
sAttributeValue = num + "pt";
|
|
}
|
|
// --[SMARTEDITORSUS-1600]
|
|
|
|
if(!sAttributeValue){
|
|
sAttributeValue = this.sDefaultText;
|
|
}
|
|
var elLi = this._getMatchingLI(sAttributeValue);
|
|
this._clearFontSizeSelection();
|
|
if(elLi){
|
|
this.elFontSizeLabel.innerHTML = sAttributeValue;
|
|
jindo.$Element(elLi).addClass("active");
|
|
}else{
|
|
this.elFontSizeLabel.innerHTML = sAttributeValue;
|
|
}
|
|
}
|
|
},
|
|
|
|
$ON_SET_FONTSIZE : function(sFontSize){
|
|
if(!sFontSize){return;}
|
|
|
|
this.oApp.exec("SET_WYSIWYG_STYLE", [{"fontSize":sFontSize}]);
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
},
|
|
|
|
_getMatchingLI : function(sFontSize){
|
|
var elLi;
|
|
|
|
sFontSize = sFontSize.toLowerCase();
|
|
for(var i=0; i<this.aLIFontSizes.length; i++){
|
|
elLi = this.aLIFontSizes[i];
|
|
if(this._getFontSizeFromLI(elLi).toLowerCase() == sFontSize){return elLi;}
|
|
}
|
|
|
|
return null;
|
|
},
|
|
|
|
_getFontSizeFromLI : function(elLi){
|
|
return elLi.firstChild.firstChild.style.fontSize;
|
|
},
|
|
|
|
_clearFontSizeSelection : function(elLi){
|
|
for(var i=0; i<this.aLIFontSizes.length; i++){
|
|
jindo.$Element(this.aLIFontSizes[i]).removeClass("active");
|
|
}
|
|
}
|
|
});
|
|
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to setting/changing the line style
|
|
* @name hp_SE_LineStyler.js
|
|
*/
|
|
nhn.husky.SE2M_LineStyler = jindo.$Class({
|
|
name : "SE2M_LineStyler",
|
|
|
|
$BEFORE_MSG_APP_READY : function() {
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getLineStyle", jindo.$Fn(this.getLineStyle, this).bind()]);
|
|
},
|
|
|
|
$ON_SET_LINE_STYLE : function(sStyleName, styleValue, htOptions){
|
|
this.oSelection = this.oApp.getSelection();
|
|
var nodes = this._getSelectedNodes(false);
|
|
this.setLineStyle(sStyleName, styleValue, htOptions, nodes);
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
},
|
|
|
|
$ON_SET_LINE_BLOCK_STYLE : function(sStyleName, styleValue, htOptions){
|
|
this.oSelection = this.oApp.getSelection();
|
|
this.setLineBlockStyle(sStyleName, styleValue, htOptions);
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
},
|
|
|
|
/**
|
|
* SE2M_TableEditor 플러그인에 의해 선택된 TD를 SE2M_TableBlockStyler 플러그인을 통해 가져온다.
|
|
* 선택된 TD가 없으면 Empty Array 를 반환한다.
|
|
* @returns {Array} SE2M_TableEditor 플러그인에 의해 선택된 TD 요소 배열
|
|
*/
|
|
_getSelectedTDs : function(){
|
|
var htSelectedTDs = {};
|
|
this.oApp.exec("GET_SELECTED_TD_BLOCK",['aTdCells',htSelectedTDs]);
|
|
return htSelectedTDs.aTdCells || [];
|
|
},
|
|
|
|
getLineStyle : function(sStyle){
|
|
var nodes = this._getSelectedNodes(false);
|
|
|
|
var curWrapper, prevWrapper;
|
|
var sCurStyle, sStyleValue;
|
|
|
|
if(nodes.length === 0){return null;}
|
|
|
|
var iLength = nodes.length;
|
|
|
|
if(iLength === 0){
|
|
sStyleValue = null;
|
|
}else{
|
|
prevWrapper = this._getLineWrapper(nodes[0]);
|
|
sStyleValue = this._getWrapperLineStyle(sStyle, prevWrapper);
|
|
}
|
|
|
|
var firstNode = this.oSelection.getStartNode();
|
|
|
|
if(sStyleValue != null){
|
|
for(var i=1; i<iLength; i++){
|
|
if(this._isChildOf(nodes[i], curWrapper)){continue;}
|
|
if(!nodes[i]){continue;}
|
|
|
|
curWrapper = this._getLineWrapper(nodes[i]);
|
|
if(curWrapper == prevWrapper){continue;}
|
|
|
|
sCurStyle = this._getWrapperLineStyle(sStyle, curWrapper);
|
|
|
|
if(sCurStyle != sStyleValue){
|
|
sStyleValue = null;
|
|
break;
|
|
}
|
|
|
|
prevWrapper = curWrapper;
|
|
}
|
|
}
|
|
|
|
curWrapper = this._getLineWrapper(nodes[iLength-1]);
|
|
|
|
var lastNode = this.oSelection.getEndNode();
|
|
|
|
setTimeout(jindo.$Fn(function(firstNode, lastNode){
|
|
// [SMARTEDITORSUS-1606] 테이블 셀 일부가 선택되었는지 확인
|
|
var aNodes = this._getSelectedTDs();
|
|
if(aNodes.length > 0){
|
|
// [SMARTEDITORSUS-1822] 테이블 셀이 일부가 선택되었다면
|
|
// 현재 Selection의 fisrtNode 와 lastNode 가 셀 내부에 있는지 확인하고
|
|
// 셀 내부에 있으면 노드를 선택된 테이블 셀 노드로 교체한다.
|
|
var elFirstTD = nhn.husky.SE2M_Utils.findAncestorByTagName("TD", firstNode);
|
|
var elLastTD = nhn.husky.SE2M_Utils.findAncestorByTagName("TD", lastNode);
|
|
firstNode = (elFirstTD || !firstNode) ? aNodes[0].firstChild : firstNode;
|
|
lastNode = (elLastTD || !lastNode) ? aNodes[aNodes.length - 1].lastChild : lastNode;
|
|
}
|
|
|
|
this.oSelection.setEndNodes(firstNode, lastNode);
|
|
this.oSelection.select();
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
}, this).bind(firstNode, lastNode), 0);
|
|
|
|
return sStyleValue;
|
|
},
|
|
|
|
// height in percentage. For example pass 1 to set the line height to 100% and 1.5 to set it to 150%
|
|
setLineStyle : function(sStyleName, styleValue, htOptions, nodes){
|
|
thisRef = this;
|
|
|
|
var bWrapperCreated = false;
|
|
|
|
function _setLineStyle(div, sStyleName, styleValue){
|
|
if(!div){
|
|
bWrapperCreated = true;
|
|
|
|
// try to wrap with P first
|
|
try{
|
|
div = thisRef.oSelection.surroundContentsWithNewNode("P");
|
|
// if the range contains a block-level tag, wrap it with a DIV
|
|
}catch(e){
|
|
div = thisRef.oSelection.surroundContentsWithNewNode("DIV");
|
|
}
|
|
}
|
|
|
|
if(typeof styleValue == "function"){
|
|
styleValue(div);
|
|
}else{
|
|
div.style[sStyleName] = styleValue;
|
|
}
|
|
|
|
if(div.childNodes.length === 0){
|
|
div.innerHTML = " ";
|
|
}
|
|
|
|
return div;
|
|
}
|
|
|
|
function isInBody(node){
|
|
while(node && node.tagName != "BODY"){
|
|
node = nhn.DOMFix.parentNode(node);
|
|
}
|
|
if(!node){return false;}
|
|
|
|
return true;
|
|
}
|
|
|
|
if(nodes.length === 0){
|
|
return;
|
|
}
|
|
|
|
var curWrapper, prevWrapper;
|
|
var iLength = nodes.length;
|
|
|
|
if((!htOptions || !htOptions["bDontAddUndoHistory"])){
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["LINE STYLE"]);
|
|
}
|
|
|
|
prevWrapper = this._getLineWrapper(nodes[0]);
|
|
prevWrapper = _setLineStyle(prevWrapper, sStyleName, styleValue);
|
|
|
|
var startNode = prevWrapper;
|
|
var endNode = prevWrapper;
|
|
|
|
for(var i=1; i<iLength; i++){
|
|
// Skip the node if a copy of the node were wrapped and the actual node no longer exists within the document.
|
|
try{
|
|
if(!isInBody(nhn.DOMFix.parentNode(nodes[i]))){continue;}
|
|
}catch(e){continue;}
|
|
|
|
if(this._isChildOf(nodes[i], curWrapper)){continue;}
|
|
|
|
curWrapper = this._getLineWrapper(nodes[i]);
|
|
|
|
if(curWrapper == prevWrapper){continue;}
|
|
|
|
curWrapper = _setLineStyle(curWrapper, sStyleName, styleValue);
|
|
|
|
prevWrapper = curWrapper;
|
|
}
|
|
|
|
endNode = curWrapper || startNode;
|
|
|
|
if(bWrapperCreated && (!htOptions || !htOptions.bDoNotSelect)) {
|
|
setTimeout(jindo.$Fn(function(startNode, endNode, htOptions){
|
|
if(startNode == endNode){
|
|
this.oSelection.selectNodeContents(startNode);
|
|
|
|
if(startNode.childNodes.length==1 && startNode.firstChild.tagName == "BR"){
|
|
this.oSelection.collapseToStart();
|
|
}
|
|
}else{
|
|
this.oSelection.setEndNodes(startNode, endNode);
|
|
}
|
|
|
|
this.oSelection.select();
|
|
|
|
if((!htOptions || !htOptions["bDontAddUndoHistory"])){
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["LINE STYLE"]);
|
|
}
|
|
}, this).bind(startNode, endNode, htOptions), 0);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Block Style 적용
|
|
*/
|
|
setLineBlockStyle : function(sStyleName, styleValue, htOptions) {
|
|
//var aTempNodes = aTextnodes = [];
|
|
var aTempNodes = [];
|
|
var aTextnodes = [];
|
|
var aNodes = this._getSelectedTDs();
|
|
|
|
for( var j = 0; j < aNodes.length ; j++){
|
|
this.oSelection.selectNode(aNodes[j]);
|
|
aTempNodes = this.oSelection.getNodes();
|
|
|
|
for(var k = 0, m = 0; k < aTempNodes.length ; k++){
|
|
if(aTempNodes[k].nodeType == 3 || (aTempNodes[k].tagName == "BR" && k == 0)) {
|
|
aTextnodes[m] = aTempNodes[k];
|
|
m ++;
|
|
}
|
|
}
|
|
this.setLineStyle(sStyleName, styleValue, htOptions, aTextnodes);
|
|
aTempNodes = aTextnodes = [];
|
|
}
|
|
},
|
|
|
|
getTextNodes : function(bSplitTextEndNodes, oSelection){
|
|
var txtFilter = function(oNode){
|
|
// 편집 중에 생겨난 빈 LI/P에도 스타일 먹이도록 포함함
|
|
// [SMARTEDITORSUS-1861] 커서홀더용 BOM문자 제외하도록 함
|
|
if((oNode.nodeType == 3 && oNode.nodeValue != "\n" && oNode.nodeValue != "" && oNode.nodeValue != "\uFEFF") || (oNode.tagName == "LI" && oNode.innerHTML == "") || (oNode.tagName == "P" && oNode.innerHTML == "")){
|
|
return true;
|
|
}else{
|
|
return false;
|
|
}
|
|
};
|
|
|
|
return oSelection.getNodes(bSplitTextEndNodes, txtFilter);
|
|
},
|
|
|
|
_getSelectedNodes : function(bDontUpdate){
|
|
if(!bDontUpdate){
|
|
this.oSelection = this.oApp.getSelection();
|
|
}
|
|
|
|
// 페이지 최하단에 빈 LI 있을 경우 해당 LI 포함하도록 expand
|
|
if(this.oSelection.endContainer.tagName == "LI" && this.oSelection.endOffset == 0 && this.oSelection.endContainer.innerHTML == ""){
|
|
this.oSelection.setEndAfter(this.oSelection.endContainer);
|
|
}
|
|
|
|
if(this.oSelection.collapsed){
|
|
// [SMARTEDITORSUS-1822] SE2M_TableEditor 플러그인에 의해 선택된 TD가 없는지 확인
|
|
// IE의 경우 SE2M_TableEditor 플러그인에 의해 TD가 선택되면 기존 selection 영역을 리셋해버리기 때문에 TD 내의 노드를 반환한다.
|
|
var aNodes = this._getSelectedTDs();
|
|
if(aNodes.length > 0){
|
|
return [aNodes[0].firstChild, aNodes[aNodes.length - 1].lastChild];
|
|
}
|
|
this.oSelection.selectNode(this.oSelection.commonAncestorContainer);
|
|
}
|
|
|
|
//var nodes = this.oSelection.getTextNodes();
|
|
var nodes = this.getTextNodes(false, this.oSelection);
|
|
|
|
if(nodes.length === 0){
|
|
var tmp = this.oSelection.getStartNode();
|
|
if(tmp){
|
|
nodes[0] = tmp;
|
|
}else{
|
|
var elTmp = this.oSelection._document.createTextNode("\u00A0");
|
|
this.oSelection.insertNode(elTmp);
|
|
nodes = [elTmp];
|
|
}
|
|
}
|
|
return nodes;
|
|
},
|
|
|
|
_getWrapperLineStyle : function(sStyle, div){
|
|
var sStyleValue = null;
|
|
if(div && div.style[sStyle]){
|
|
sStyleValue = div.style[sStyle];
|
|
}else{
|
|
div = this.oSelection.commonAncesterContainer;
|
|
while(div && !this.oSelection.rxLineBreaker.test(div.tagName)){
|
|
if(div && div.style[sStyle]){
|
|
sStyleValue = div.style[sStyle];
|
|
break;
|
|
}
|
|
div = nhn.DOMFix.parentNode(div);
|
|
}
|
|
}
|
|
|
|
return sStyleValue;
|
|
},
|
|
|
|
_isChildOf : function(node, container){
|
|
while(node && node.tagName != "BODY"){
|
|
if(node == container){return true;}
|
|
node = nhn.DOMFix.parentNode(node);
|
|
}
|
|
|
|
return false;
|
|
},
|
|
_getLineWrapper : function(node){
|
|
var oTmpSelection = this.oApp.getEmptySelection();
|
|
oTmpSelection.selectNode(node);
|
|
var oLineInfo = oTmpSelection.getLineInfo();
|
|
var oStart = oLineInfo.oStart;
|
|
var oEnd = oLineInfo.oEnd;
|
|
|
|
var a, b;
|
|
var breakerA, breakerB;
|
|
var div = null;
|
|
|
|
a = oStart.oNode;
|
|
breakerA = oStart.oLineBreaker;
|
|
b = oEnd.oNode;
|
|
breakerB = oEnd.oLineBreaker;
|
|
|
|
this.oSelection.setEndNodes(a, b);
|
|
|
|
if(breakerA == breakerB){
|
|
if(breakerA.tagName == "P" || breakerA.tagName == "DIV" || breakerA.tagName == "LI"){
|
|
// if(breakerA.tagName == "P" || breakerA.tagName == "DIV"){
|
|
div = breakerA;
|
|
}else{
|
|
this.oSelection.setEndNodes(breakerA.firstChild, breakerA.lastChild);
|
|
}
|
|
}
|
|
|
|
return div;
|
|
}
|
|
});
|
|
//}
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to changing the lineheight using layer
|
|
* @name hp_SE2M_LineHeightWithLayerUI.js
|
|
*/
|
|
nhn.husky.SE2M_LineHeightWithLayerUI = jindo.$Class({
|
|
name : "SE2M_LineHeightWithLayerUI",
|
|
MIN_LINE_HEIGHT : 50,
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["lineHeight", "click", "SE2M_TOGGLE_LINEHEIGHT_LAYER"]);
|
|
this.oApp.registerLazyMessage(["SE2M_TOGGLE_LINEHEIGHT_LAYER"], ["hp_SE2M_LineHeightWithLayerUI$Lazy.js"]);
|
|
}
|
|
});
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations directly related to the color palette
|
|
* @name hp_SE2M_ColorPalette.js
|
|
*/
|
|
nhn.husky.SE2M_ColorPalette = jindo.$Class({
|
|
name : "SE2M_ColorPalette",
|
|
elAppContainer : null,
|
|
bUseRecentColor : false,
|
|
nLimitRecentColor : 17,
|
|
rxRGBColorPattern : /rgb\((\d+), ?(\d+), ?(\d+)\)/i,
|
|
rxColorPattern : /^#?[0-9a-fA-F]{6}$|^rgb\(\d+, ?\d+, ?\d+\)$/i,
|
|
aRecentColor : [], // 최근 사용한 색 목록, 가장 최근에 등록한 색의 index가 가장 작음
|
|
URL_COLOR_LIST : "",
|
|
URL_COLOR_ADD : "",
|
|
URL_COLOR_UPDATE : "",
|
|
sRecentColorTemp : "<li><button type=\"button\" title=\"{RGB_CODE}\" style=\"background:{RGB_CODE}\"><span><span>{RGB_CODE}</span></span></button></li>",
|
|
|
|
$init : function(elAppContainer){
|
|
this.elAppContainer = elAppContainer;
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){},
|
|
|
|
_assignHTMLElements : function(oAppContainer){
|
|
var htConfiguration = nhn.husky.SE2M_Configuration.SE2M_ColorPalette;
|
|
if(htConfiguration){
|
|
this.bUseRecentColor = htConfiguration.bUseRecentColor || false;
|
|
this.URL_COLOR_ADD = htConfiguration.addColorURL || "http://api.se2.naver.com/1/colortable/TextAdd.nhn";
|
|
this.URL_COLOR_UPDATE = htConfiguration.updateColorURL || "http://api.se2.naver.com/1/colortable/TextUpdate.nhn";
|
|
this.URL_COLOR_LIST = htConfiguration.colorListURL || "http://api.se2.naver.com/1/colortable/TextList.nhn";
|
|
}
|
|
|
|
this.elColorPaletteLayer = jindo.$$.getSingle("DIV.husky_se2m_color_palette", oAppContainer);
|
|
|
|
this.elColorPaletteLayerColorPicker = jindo.$$.getSingle("DIV.husky_se2m_color_palette_colorpicker", this.elColorPaletteLayer);
|
|
this.elRecentColorForm = jindo.$$.getSingle("form", this.elColorPaletteLayerColorPicker);
|
|
|
|
this.elBackgroundColor = jindo.$$.getSingle("ul.husky_se2m_bgcolor_list", oAppContainer);
|
|
this.elInputColorCode = jindo.$$.getSingle("INPUT.husky_se2m_cp_colorcode", this.elColorPaletteLayerColorPicker);
|
|
|
|
this.elPreview = jindo.$$.getSingle("SPAN.husky_se2m_cp_preview", this.elColorPaletteLayerColorPicker);
|
|
this.elCP_ColPanel = jindo.$$.getSingle("DIV.husky_se2m_cp_colpanel", this.elColorPaletteLayerColorPicker);
|
|
this.elCP_HuePanel = jindo.$$.getSingle("DIV.husky_se2m_cp_huepanel", this.elColorPaletteLayerColorPicker);
|
|
|
|
this.elCP_ColPanel.style.position = "relative";
|
|
this.elCP_HuePanel.style.position = "relative";
|
|
|
|
this.elColorPaletteLayerColorPicker.style.display = "none";
|
|
|
|
this.elMoreBtn = jindo.$$.getSingle("BUTTON.husky_se2m_color_palette_more_btn", this.elColorPaletteLayer);
|
|
this.welMoreBtn = jindo.$Element(this.elMoreBtn);
|
|
|
|
this.elOkBtn = jindo.$$.getSingle("BUTTON.husky_se2m_color_palette_ok_btn", this.elColorPaletteLayer);
|
|
|
|
if(this.bUseRecentColor){
|
|
this.elColorPaletteLayerRecent = jindo.$$.getSingle("DIV.husky_se2m_color_palette_recent", this.elColorPaletteLayer);
|
|
this.elRecentColor = jindo.$$.getSingle("ul.se2_pick_color", this.elColorPaletteLayerRecent);
|
|
this.elDummyNode = jindo.$$.getSingle("ul.se2_pick_color > li", this.elColorPaletteLayerRecent) || null;
|
|
|
|
this.elColorPaletteLayerRecent.style.display = "none";
|
|
}
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(){
|
|
this._assignHTMLElements(this.elAppContainer);
|
|
|
|
if(this.elDummyNode){
|
|
jindo.$Element(jindo.$$.getSingle("ul.se2_pick_color > li", this.elColorPaletteLayerRecent)).leave();
|
|
}
|
|
|
|
if( this.bUseRecentColor ){
|
|
this._ajaxRecentColor(this._ajaxRecentColorCallback);
|
|
}
|
|
|
|
this.oApp.registerBrowserEvent(this.elColorPaletteLayer, "click", "EVENT_CLICK_COLOR_PALETTE");
|
|
// [SMARTEDITORSUS-1833] 아이패드에서 mouseover 이벤트리스너를 등록하면 후속 click 이벤트가 바로 동작하지 않음
|
|
// 모바일환경에서 hover 처리는 의미가 없으므로 PC 환경에서만 hover 처리하도록 함
|
|
if(!this.oApp.bMobile){
|
|
this.oApp.registerBrowserEvent(this.elBackgroundColor, "mouseover", "EVENT_MOUSEOVER_COLOR_PALETTE");
|
|
this.oApp.registerBrowserEvent(this.elColorPaletteLayer, "mouseover", "EVENT_MOUSEOVER_COLOR_PALETTE");
|
|
this.oApp.registerBrowserEvent(this.elBackgroundColor, "mouseout", "EVENT_MOUSEOUT_COLOR_PALETTE");
|
|
this.oApp.registerBrowserEvent(this.elColorPaletteLayer, "mouseout", "EVENT_MOUSEOUT_COLOR_PALETTE");
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_MOUSEOVER_COLOR_PALETTE : function(oEvent){
|
|
var elHovered = oEvent.element;
|
|
while(elHovered && elHovered.tagName && elHovered.tagName.toLowerCase() != "li"){
|
|
elHovered = elHovered.parentNode;
|
|
}
|
|
//조건 추가-by cielo 2010.04.20
|
|
if(!elHovered || !elHovered.nodeType || elHovered.nodeType == 9){return;}
|
|
if(elHovered.className == "" || (!elHovered.className) || typeof(elHovered.className) == 'undefined'){jindo.$Element(elHovered).addClass("hover");}
|
|
},
|
|
|
|
$ON_EVENT_MOUSEOUT_COLOR_PALETTE : function(oEvent){
|
|
var elHovered = oEvent.element;
|
|
|
|
while(elHovered && elHovered.tagName && elHovered.tagName.toLowerCase() != "li"){
|
|
elHovered = elHovered.parentNode;
|
|
}
|
|
if(!elHovered){return;}
|
|
if(elHovered.className == "hover"){jindo.$Element(elHovered).removeClass("hover");}
|
|
},
|
|
|
|
$ON_EVENT_CLICK_COLOR_PALETTE : function(oEvent){
|
|
var elClicked = oEvent.element;
|
|
|
|
while(elClicked.tagName == "SPAN"){elClicked = elClicked.parentNode;}
|
|
|
|
if(elClicked.tagName && elClicked.tagName == "BUTTON"){
|
|
if(elClicked == this.elMoreBtn){
|
|
this.oApp.exec("TOGGLE_COLOR_PICKER");
|
|
return;
|
|
}
|
|
|
|
this.oApp.exec("APPLY_COLOR", [elClicked]);
|
|
}
|
|
},
|
|
|
|
$ON_APPLY_COLOR : function(elButton){
|
|
var sColorCode = this.elInputColorCode.value,
|
|
welColorParent = null;
|
|
|
|
if(sColorCode.indexOf("#") == -1){
|
|
sColorCode = "#" + sColorCode;
|
|
this.elInputColorCode.value = sColorCode;
|
|
}
|
|
|
|
// 입력 버튼인 경우
|
|
if(elButton == this.elOkBtn){
|
|
if(!this._verifyColorCode(sColorCode)){
|
|
this.elInputColorCode.value = "";
|
|
alert(this.oApp.$MSG("SE_Color.invalidColorCode"));
|
|
this.elInputColorCode.focus();
|
|
|
|
return;
|
|
}
|
|
|
|
this.oApp.exec("COLOR_PALETTE_APPLY_COLOR", [sColorCode,true]);
|
|
|
|
return;
|
|
}
|
|
|
|
// 색상 버튼인 경우
|
|
welColorParent = jindo.$Element(elButton.parentNode.parentNode.parentNode);
|
|
sColorCode = elButton.title;
|
|
|
|
if(welColorParent.hasClass("husky_se2m_color_palette")){ // 템플릿 색상 적용
|
|
/*
|
|
* [SMARTEDITORSUS-1884][SMARTEDITORSUS-2117]
|
|
* 설정값 제거(r12236) 전에도
|
|
* 모든 타입에서
|
|
* nhn.husky.SE2M_Configuration.SE2M_ColorPalette.bAddRecentColorFromDefault 값이
|
|
* undefined인 상태로 동작하고 있었기 때문에
|
|
* false로 처리
|
|
*/
|
|
this.oApp.exec("COLOR_PALETTE_APPLY_COLOR", [sColorCode, false]);
|
|
}else if(welColorParent.hasClass("husky_se2m_color_palette_recent")){ // 최근 색상 적용
|
|
this.oApp.exec("COLOR_PALETTE_APPLY_COLOR", [sColorCode,true]);
|
|
}
|
|
},
|
|
|
|
$ON_RESET_COLOR_PALETTE : function(){
|
|
this._initColor();
|
|
},
|
|
|
|
$ON_TOGGLE_COLOR_PICKER : function(){
|
|
if(this.elColorPaletteLayerColorPicker.style.display == "none"){
|
|
this.oApp.exec("SHOW_COLOR_PICKER");
|
|
}else{
|
|
this.oApp.exec("HIDE_COLOR_PICKER");
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_COLOR_PICKER : function(){
|
|
this.elColorPaletteLayerColorPicker.style.display = "";
|
|
|
|
this.cpp = new nhn.ColorPicker(this.elCP_ColPanel, {huePanel:this.elCP_HuePanel});
|
|
var fn = jindo.$Fn(function(oEvent) {
|
|
this.elPreview.style.backgroundColor = oEvent.hexColor;
|
|
this.elInputColorCode.value = oEvent.hexColor;
|
|
}, this).bind();
|
|
this.cpp.attach("colorchange", fn);
|
|
|
|
this.$ON_SHOW_COLOR_PICKER = this._showColorPickerMain;
|
|
this.$ON_SHOW_COLOR_PICKER();
|
|
},
|
|
|
|
$ON_HIDE_COLOR_PICKER : function(){
|
|
this.elColorPaletteLayerColorPicker.style.display = "none";
|
|
this.welMoreBtn.addClass("se2_view_more");
|
|
this.welMoreBtn.removeClass("se2_view_more2");
|
|
},
|
|
|
|
$ON_SHOW_COLOR_PALETTE : function(sCallbackCmd, oLayerContainer){
|
|
this.sCallbackCmd = sCallbackCmd;
|
|
this.oLayerContainer = oLayerContainer;
|
|
|
|
this.oLayerContainer.insertBefore(this.elColorPaletteLayer, null);
|
|
|
|
this.elColorPaletteLayer.style.display = "block";
|
|
|
|
this.oApp.delayedExec("POSITION_TOOLBAR_LAYER", [this.elColorPaletteLayer.parentNode.parentNode], 0);
|
|
},
|
|
|
|
$ON_HIDE_COLOR_PALETTE : function(){
|
|
this.elColorPaletteLayer.style.display = "none";
|
|
},
|
|
|
|
$ON_COLOR_PALETTE_APPLY_COLOR : function(sColorCode , bAddRecentColor){
|
|
bAddRecentColor = (!bAddRecentColor)? false : bAddRecentColor;
|
|
sColorCode = this._getHexColorCode(sColorCode);
|
|
|
|
//더보기 레이어에서 적용한 색상만 최근 사용한 색에 추가한다.
|
|
if( this.bUseRecentColor && !!bAddRecentColor ){
|
|
this.oApp.exec("ADD_RECENT_COLOR", [sColorCode]);
|
|
}
|
|
this.oApp.exec(this.sCallbackCmd, [sColorCode]);
|
|
},
|
|
|
|
$ON_EVENT_MOUSEUP_COLOR_PALETTE : function(oEvent){
|
|
var elButton = oEvent.element;
|
|
if(! elButton.style.backgroundColor){return;}
|
|
|
|
this.oApp.exec("COLOR_PALETTE_APPLY_COLOR", [elButton.style.backgroundColor,false]);
|
|
},
|
|
|
|
$ON_ADD_RECENT_COLOR : function(sRGBCode){
|
|
var bAdd = (this.aRecentColor.length === 0);
|
|
|
|
this._addRecentColor(sRGBCode);
|
|
|
|
if(bAdd){
|
|
this._ajaxAddColor();
|
|
}else{
|
|
this._ajaxUpdateColor();
|
|
}
|
|
|
|
this._redrawRecentColorElement();
|
|
},
|
|
|
|
_verifyColorCode : function(sColorCode){
|
|
return this.rxColorPattern.test(sColorCode);
|
|
},
|
|
|
|
_getHexColorCode : function(sColorCode){
|
|
if(this.rxRGBColorPattern.test(sColorCode)){
|
|
var dec2Hex = function(sDec){
|
|
var sTmp = parseInt(sDec, 10).toString(16);
|
|
if(sTmp.length<2){sTmp = "0"+sTmp;}
|
|
return sTmp.toUpperCase();
|
|
};
|
|
|
|
var sR = dec2Hex(RegExp.$1);
|
|
var sG = dec2Hex(RegExp.$2);
|
|
var sB = dec2Hex(RegExp.$3);
|
|
sColorCode = "#"+sR+sG+sB;
|
|
}
|
|
|
|
return sColorCode;
|
|
},
|
|
|
|
_addRecentColor : function(sRGBCode){
|
|
var waRecentColor = jindo.$A(this.aRecentColor);
|
|
|
|
waRecentColor = waRecentColor.refuse(sRGBCode);
|
|
waRecentColor.unshift(sRGBCode);
|
|
|
|
if(waRecentColor.length() > this.nLimitRecentColor){
|
|
waRecentColor.length(this.nLimitRecentColor);
|
|
}
|
|
|
|
this.aRecentColor = waRecentColor.$value();
|
|
},
|
|
|
|
_redrawRecentColorElement : function(){
|
|
var aRecentColorHtml = [],
|
|
nRecentColor = this.aRecentColor.length,
|
|
i;
|
|
|
|
if(nRecentColor === 0){
|
|
return;
|
|
}
|
|
|
|
for(i=0; i<nRecentColor; i++){
|
|
aRecentColorHtml.push(this.sRecentColorTemp.replace(/\{RGB_CODE\}/gi, this.aRecentColor[i]));
|
|
}
|
|
|
|
this.elRecentColor.innerHTML = aRecentColorHtml.join("");
|
|
|
|
this.elColorPaletteLayerRecent.style.display = "block";
|
|
},
|
|
|
|
_ajaxAddColor : function(){
|
|
jindo.$Ajax(this.URL_COLOR_ADD, {
|
|
type : "jsonp",
|
|
onload: function(){}
|
|
}).request({
|
|
text_key : "colortable",
|
|
text_data : this.aRecentColor.join(",")
|
|
});
|
|
},
|
|
|
|
_ajaxUpdateColor : function(){
|
|
jindo.$Ajax(this.URL_COLOR_UPDATE, {
|
|
type : "jsonp",
|
|
onload: function(){}
|
|
}).request({
|
|
text_key : "colortable",
|
|
text_data : this.aRecentColor.join(",")
|
|
});
|
|
},
|
|
|
|
_showColorPickerMain : function(){
|
|
this._initColor();
|
|
this.elColorPaletteLayerColorPicker.style.display = "";
|
|
this.welMoreBtn.removeClass("se2_view_more");
|
|
this.welMoreBtn.addClass("se2_view_more2");
|
|
},
|
|
|
|
_initColor : function(){
|
|
if(this.cpp){this.cpp.rgb({r:0,g:0,b:0});}
|
|
this.elPreview.style.backgroundColor = '#'+'000000';
|
|
this.elInputColorCode.value = '#'+'000000';
|
|
this.oApp.exec("HIDE_COLOR_PICKER");
|
|
},
|
|
|
|
_ajaxRecentColor : function(fCallback){
|
|
jindo.$Ajax(this.URL_COLOR_LIST, {
|
|
type : "jsonp",
|
|
onload : jindo.$Fn(fCallback, this).bind()
|
|
}).request();
|
|
},
|
|
|
|
_ajaxRecentColorCallback : function(htResponse){
|
|
var aColorList = htResponse.json()["result"],
|
|
waColorList,
|
|
i, nLen;
|
|
|
|
if(!aColorList || !!aColorList.error){
|
|
return;
|
|
}
|
|
|
|
waColorList = jindo.$A(aColorList).filter(this._verifyColorCode, this);
|
|
|
|
if(waColorList.length() > this.nLimitRecentColor){
|
|
waColorList.length(this.nLimitRecentColor);
|
|
}
|
|
|
|
aColorList = waColorList.reverse().$value();
|
|
|
|
for(i = 0, nLen = aColorList.length; i < nLen; i++){
|
|
this._addRecentColor(this._getHexColorCode(aColorList[i]));
|
|
}
|
|
|
|
this._redrawRecentColorElement();
|
|
}
|
|
}).extend(jindo.Component);
|
|
//}
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to changing the font color
|
|
* @name hp_SE_FontColor.js
|
|
*/
|
|
nhn.husky.SE2M_FontColor = jindo.$Class({
|
|
name : "SE2M_FontColor",
|
|
rxColorPattern : /^#?[0-9a-fA-F]{6}$|^rgb\(\d+, ?\d+, ?\d+\)$/i,
|
|
|
|
$init : function(elAppContainer){
|
|
this._assignHTMLElements(elAppContainer);
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
//@ec[
|
|
this.elLastUsed = jindo.$$.getSingle("SPAN.husky_se2m_fontColor_lastUsed", elAppContainer);
|
|
|
|
this.elDropdownLayer = jindo.$$.getSingle("DIV.husky_se2m_fontcolor_layer", elAppContainer);
|
|
this.elPaletteHolder = jindo.$$.getSingle("DIV.husky_se2m_fontcolor_paletteHolder", this.elDropdownLayer);
|
|
//@ec]
|
|
|
|
this._setLastUsedFontColor("#000000");
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function() {
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getLastUsedFontColor", jindo.$Fn(this.getLastUsedFontColor, this).bind()]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["fontColorA", "click", "APPLY_LAST_USED_FONTCOLOR"]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["fontColorB", "click", "TOGGLE_FONTCOLOR_LAYER"]);
|
|
this.oApp.registerLazyMessage(["APPLY_LAST_USED_FONTCOLOR", "TOGGLE_FONTCOLOR_LAYER"], ["hp_SE2M_FontColor$Lazy.js"]);
|
|
},
|
|
|
|
_setLastUsedFontColor : function(sFontColor){
|
|
this.sLastUsedColor = sFontColor;
|
|
this.elLastUsed.style.backgroundColor = this.sLastUsedColor;
|
|
},
|
|
|
|
getLastUsedFontColor : function(){
|
|
return (!!this.sLastUsedColor) ? this.sLastUsedColor : '#000000';
|
|
}
|
|
});
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of changing the background color
|
|
* @name hp_SE2M_BGColor.js
|
|
*/
|
|
nhn.husky.SE2M_BGColor = jindo.$Class({
|
|
name : "SE2M_BGColor",
|
|
rxColorPattern : /^#?[0-9a-fA-F]{6}$|^rgb\(\d+, ?\d+, ?\d+\)$/i,
|
|
|
|
$init : function(elAppContainer){
|
|
this._assignHTMLElements(elAppContainer);
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
//@ec[
|
|
this.elLastUsed = jindo.$$.getSingle("SPAN.husky_se2m_BGColor_lastUsed", elAppContainer);
|
|
|
|
this.elDropdownLayer = jindo.$$.getSingle("DIV.husky_se2m_BGColor_layer", elAppContainer);
|
|
this.elBGColorList = jindo.$$.getSingle("UL.husky_se2m_bgcolor_list", elAppContainer);
|
|
this.elPaletteHolder = jindo.$$.getSingle("DIV.husky_se2m_BGColor_paletteHolder", this.elDropdownLayer);
|
|
//@ec]
|
|
|
|
this._setLastUsedBGColor("#777777");
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function() {
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getLastUsedBackgroundColor", jindo.$Fn(this.getLastUsedBGColor, this).bind()]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["BGColorA", "click", "APPLY_LAST_USED_BGCOLOR"]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["BGColorB", "click", "TOGGLE_BGCOLOR_LAYER"]);
|
|
|
|
this.oApp.registerBrowserEvent(this.elBGColorList, "click", "EVENT_APPLY_BGCOLOR", []);
|
|
this.oApp.registerLazyMessage(["APPLY_LAST_USED_BGCOLOR", "TOGGLE_BGCOLOR_LAYER"], ["hp_SE2M_BGColor$Lazy.js"]);
|
|
},
|
|
|
|
_setLastUsedBGColor : function(sBGColor){
|
|
this.sLastUsedColor = sBGColor;
|
|
this.elLastUsed.style.backgroundColor = this.sLastUsedColor;
|
|
},
|
|
|
|
getLastUsedBGColor : function(){
|
|
return (!!this.sLastUsedColor) ? this.sLastUsedColor : '#777777';
|
|
}
|
|
});
|
|
//}
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to hyperlink
|
|
* @name hp_SE_Hyperlink.js
|
|
*/
|
|
nhn.husky.SE2M_Hyperlink = jindo.$Class({
|
|
name : "SE2M_Hyperlink",
|
|
sATagMarker : "HTTP://HUSKY_TMP.MARKER/",
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
this.oHyperlinkButton = jindo.$$.getSingle("li.husky_seditor_ui_hyperlink", elAppContainer);
|
|
this.oHyperlinkLayer = jindo.$$.getSingle("div.se2_layer", this.oHyperlinkButton);
|
|
this.oLinkInput = jindo.$$.getSingle("INPUT[type=text]", this.oHyperlinkLayer);
|
|
|
|
this.oBtnConfirm = jindo.$$.getSingle("button.se2_apply", this.oHyperlinkLayer);
|
|
this.oBtnCancel = jindo.$$.getSingle("button.se2_cancel", this.oHyperlinkLayer);
|
|
|
|
//this.oCbNewWin = jindo.$$.getSingle("INPUT[type=checkbox]", this.oHyperlinkLayer) || null;
|
|
},
|
|
|
|
_generateAutoLink : function(sAll, sBreaker, sURL, sWWWURL, sHTTPURL) {
|
|
sBreaker = sBreaker || "";
|
|
|
|
var sResult;
|
|
if (sWWWURL){
|
|
sResult = '<a href="http://'+sWWWURL+'">'+sURL+'</a>';
|
|
} else {
|
|
sResult = '<a href="'+sHTTPURL+'">'+sURL+'</a>';
|
|
}
|
|
|
|
return sBreaker+sResult;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1405] 자동링크 비활성화 옵션을 체크해서 처리한다.
|
|
* $ON_REGISTER_CONVERTERS 메시지가 SE_EditingAreaManager.$ON_MSG_APP_READY 에서 수행되므로 먼저 처리한다.
|
|
*/
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
var htOptions = nhn.husky.SE2M_Configuration.SE2M_Hyperlink;
|
|
if(htOptions && htOptions.bAutolink === false){
|
|
// 자동링크 컨버터 비활성화
|
|
this.$ON_REGISTER_CONVERTERS = null;
|
|
// UI enable/disable 처리 제외
|
|
this.$ON_DISABLE_MESSAGE = null;
|
|
this.$ON_ENABLE_MESSAGE = null;
|
|
// 브라우저의 자동링크기능 비활성화
|
|
try{ this.oApp.getWYSIWYGDocument().execCommand("AutoUrlDetect", false, false); } catch(e){}
|
|
}
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.bLayerShown = false;
|
|
|
|
// [SMARTEDITORSUS-2260] 메일 > Mac에서 ctrl 조합 단축키 모두 meta 조합으로 변경
|
|
if (jindo.$Agent().os().mac) {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+k", "TOGGLE_HYPERLINK_LAYER", []]);
|
|
} else {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+k", "TOGGLE_HYPERLINK_LAYER", []]);
|
|
}
|
|
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["hyperlink", "click", "TOGGLE_HYPERLINK_LAYER"]);
|
|
this.oApp.registerLazyMessage(["TOGGLE_HYPERLINK_LAYER", "APPLY_HYPERLINK"], ["hp_SE2M_Hyperlink$Lazy.js"]);
|
|
},
|
|
|
|
$ON_REGISTER_CONVERTERS : function(){
|
|
this.oApp.exec("ADD_CONVERTER_DOM", ["IR_TO_DB", jindo.$Fn(this.irToDb, this).bind()]);
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(sMsg){
|
|
if(!!sMsg.match(/(REGISTER_CONVERTERS)/)){
|
|
this.oApp.acceptLocalBeforeFirstAgain(this, true);
|
|
return true;
|
|
}
|
|
|
|
this._assignHTMLElements(this.oApp.htOptions.elAppContainer);
|
|
this.sRXATagMarker = this.sATagMarker.replace(/\//g, "\\/").replace(/\./g, "\\.");
|
|
this.oApp.registerBrowserEvent(this.oBtnConfirm, "click", "APPLY_HYPERLINK");
|
|
this.oApp.registerBrowserEvent(this.oBtnCancel, "click", "HIDE_ACTIVE_LAYER");
|
|
this.oApp.registerBrowserEvent(this.oLinkInput, "keydown", "EVENT_HYPERLINK_KEYDOWN");
|
|
},
|
|
|
|
$ON_EVENT_HYPERLINK_KEYDOWN : function(oEvent){
|
|
if (oEvent.key().enter){
|
|
this.oApp.exec("APPLY_HYPERLINK");
|
|
oEvent.stop();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [MUG-1265] 버튼이 사용불가 상태이면 자동변환기능을 막는다.
|
|
* @see http://stackoverflow.com/questions/7556007/avoid-transformation-text-to-link-ie-contenteditable-mode
|
|
* IE9 이전 버전은 AutoURlDetect을 사용할 수 없어 오류 발생되기 때문에, try catch로 블럭 처리(http://msdn.microsoft.com/en-us/library/aa769893%28VS.85%29.aspx)
|
|
*/
|
|
$ON_DISABLE_MESSAGE : function(sCmd) {
|
|
if(sCmd !== "TOGGLE_HYPERLINK_LAYER"){
|
|
return;
|
|
}
|
|
try{ this.oApp.getWYSIWYGDocument().execCommand("AutoUrlDetect", false, false); } catch(e){}
|
|
this._bDisabled = true;
|
|
},
|
|
|
|
/**
|
|
* [MUG-1265] 버튼이 사용가능 상태이면 자동변환기능을 복원해준다.
|
|
*/
|
|
$ON_ENABLE_MESSAGE : function(sCmd) {
|
|
if(sCmd !== "TOGGLE_HYPERLINK_LAYER"){
|
|
return;
|
|
}
|
|
try{ this.oApp.getWYSIWYGDocument().execCommand("AutoUrlDetect", false, true); } catch(e){}
|
|
this._bDisabled = false;
|
|
},
|
|
|
|
irToDb : function(oTmpNode){
|
|
if(this._bDisabled){ // [MUG-1265] 버튼이 사용불가 상태이면 자동변환하지 않는다.
|
|
return;
|
|
}
|
|
//저장 시점에 자동 링크를 위한 함수.
|
|
//[SMARTEDITORSUS-1207][IE][메일] object 삽입 후 글을 저장하면 IE 브라우저가 죽어버리는 현상
|
|
//원인 : 확인 불가. IE 저작권 관련 이슈로 추정
|
|
//해결 : contents를 가지고 있는 div 태그를 이 함수 내부에서 복사하여 수정 후 call by reference로 넘어온 변수의 innerHTML을 변경
|
|
var oCopyNode = oTmpNode.cloneNode(true);
|
|
try{
|
|
oCopyNode.innerHTML;
|
|
}catch(e) {
|
|
oCopyNode = jindo.$(oTmpNode.outerHTML);
|
|
}
|
|
|
|
var oTmpRange = this.oApp.getEmptySelection();
|
|
var elFirstNode = oTmpRange._getFirstRealChild(oCopyNode);
|
|
var elLastNode = oTmpRange._getLastRealChild(oCopyNode);
|
|
var waAllNodes = jindo.$A(oTmpRange._getNodesBetween(elFirstNode, elLastNode));
|
|
var aAllTextNodes = waAllNodes.filter(function(elNode){return (elNode && elNode.nodeType === 3);}).$value();
|
|
var a = aAllTextNodes;
|
|
|
|
/*
|
|
// 텍스트 검색이 용이 하도록 끊어진 텍스트 노드가 있으면 합쳐줌. (화면상으로 ABC라고 보이나 상황에 따라 실제 2개의 텍스트 A, BC로 이루어져 있을 수 있음. 이를 ABC 하나의 노드로 만들어 줌.)
|
|
// 문제 발생 가능성에 비해서 퍼포먼스나 사이드 이펙트 가능성 높아 일단 주석
|
|
var aCleanTextNodes = [];
|
|
for(var i=0, nLen=aAllTextNodes.length; i<nLen; i++){
|
|
if(a[i].nextSibling && a[i].nextSibling.nodeType === 3){
|
|
a[i].nextSibling.nodeValue += a[i].nodeValue;
|
|
a[i].parentNode.removeChild(a[i]);
|
|
}else{
|
|
aCleanTextNodes[aCleanTextNodes.length] = a[i];
|
|
}
|
|
}
|
|
*/
|
|
var aCleanTextNodes = aAllTextNodes;
|
|
|
|
// IE에서 PRE를 제외한 다른 태그 하위에 있는 텍스트 노드는 줄바꿈 등의 값을 변질시킴
|
|
var elTmpDiv = this.oApp.getWYSIWYGDocument().createElement("DIV");
|
|
var elParent, bAnchorFound;
|
|
var sTmpStr = "@"+(new Date()).getTime()+"@";
|
|
var rxTmpStr = new RegExp(sTmpStr, "g");
|
|
for(var i=0, nLen=aAllTextNodes.length; i<nLen; i++){
|
|
// Anchor가 이미 걸려 있는 텍스트이면 링크를 다시 걸지 않음.
|
|
elParent = a[i].parentNode;
|
|
bAnchorFound = false;
|
|
while(elParent){
|
|
if(elParent.tagName === "A" || elParent.tagName === "PRE"){
|
|
bAnchorFound = true;
|
|
break;
|
|
}
|
|
elParent = elParent.parentNode;
|
|
}
|
|
if(bAnchorFound){
|
|
continue;
|
|
}
|
|
// www.또는 http://으로 시작하는 텍스트에 링크 걸어 줌
|
|
// IE에서 텍스트 노드 앞쪽의 스페이스나 주석등이 사라지는 현상이 있어 sTmpStr을 앞에 붙여줌.
|
|
elTmpDiv.innerHTML = "";
|
|
|
|
try {
|
|
elTmpDiv.appendChild(a[i].cloneNode(true));
|
|
|
|
// IE에서 innerHTML를 이용 해 직접 텍스트 노드 값을 할당 할 경우 줄바꿈등이 깨질 수 있어, 텍스트 노드로 만들어서 이를 바로 append 시켜줌
|
|
// [SMARTEDITORSUS-1649] https:// URL을 입력한 경우에도 자동링크 지원
|
|
//elTmpDiv.innerHTML = (sTmpStr+elTmpDiv.innerHTML).replace(/( |\s)?(((?!http:\/\/)www\.(?:(?!\ |\s|"|').)+)|(http:\/\/(?:(?! |\s|"|').)+))/ig, this._generateAutoLink);
|
|
elTmpDiv.innerHTML = (sTmpStr+elTmpDiv.innerHTML).replace(/( |\s)?(((?!http[s]?:\/\/)www\.(?:(?!\ |\s|"|').)+)|(http[s]?:\/\/(?:(?! |\s|"|').)+))/ig, this._generateAutoLink);
|
|
// --[SMARTEDITORSUS-1649]
|
|
|
|
// innerHTML 내에 텍스트가 있을 경우 insert 시에 주변 텍스트 노드와 합쳐지는 현상이 있어 div로 위치를 먼저 잡고 하나씩 삽입
|
|
a[i].parentNode.insertBefore(elTmpDiv, a[i]);
|
|
a[i].parentNode.removeChild(a[i]);
|
|
} catch(e1) {
|
|
|
|
}
|
|
|
|
while(elTmpDiv.firstChild){
|
|
elTmpDiv.parentNode.insertBefore(elTmpDiv.firstChild, elTmpDiv);
|
|
}
|
|
elTmpDiv.parentNode.removeChild(elTmpDiv);
|
|
// alert(a[i].nodeValue);
|
|
}
|
|
elTmpDiv = oTmpRange = elFirstNode = elLastNode = waAllNodes = aAllTextNodes = a = aCleanTextNodes = elParent = null;
|
|
oCopyNode.innerHTML = oCopyNode.innerHTML.replace(rxTmpStr, "");
|
|
oTmpNode.innerHTML = oCopyNode.innerHTML;
|
|
oCopyNode = null;
|
|
//alert(oTmpNode.innerHTML);
|
|
}
|
|
});
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to changing the font name using Select element
|
|
* @name SE2M_FontNameWithLayerUI.js
|
|
* @trigger MSG_STYLE_CHANGED,SE2M_TOGGLE_FONTNAME_LAYER
|
|
*/
|
|
nhn.husky.SE2M_FontNameWithLayerUI = jindo.$Class({
|
|
name : "SE2M_FontNameWithLayerUI",
|
|
FONT_SEPARATOR : "husky_seditor_font_separator",
|
|
_rxQuote : /['"]/g,
|
|
_rxComma : /\s*,\s*/g,
|
|
|
|
$init : function(elAppContainer, aAdditionalFontList){
|
|
this.elLastHover = null;
|
|
this._assignHTMLElements(elAppContainer);
|
|
|
|
this.htBrowser = jindo.$Agent().navigator();
|
|
this.aAdditionalFontList = aAdditionalFontList || [];
|
|
},
|
|
|
|
addAllFonts : function(){
|
|
var aDefaultFontList, aFontList, htMainFont, aFontInUse, i;
|
|
|
|
// family name -> display name 매핑 (웹폰트는 두개가 다름)
|
|
this.htFamilyName2DisplayName = {};
|
|
this.htAllFonts = {};
|
|
|
|
this.aBaseFontList = [];
|
|
this.aDefaultFontList = [];
|
|
this.aTempSavedFontList = [];
|
|
|
|
this.htOptions = this.oApp.htOptions.SE2M_FontName;
|
|
|
|
if(this.htOptions){
|
|
aDefaultFontList = this.htOptions.aDefaultFontList || [];
|
|
aFontList = this.htOptions.aFontList;
|
|
htMainFont = this.htOptions.htMainFont;
|
|
aFontInUse = this.htOptions.aFontInUse;
|
|
|
|
//add Font
|
|
if(this.htBrowser.ie && aFontList){
|
|
for(i=0; i<aFontList.length; i++){
|
|
this.addFont(aFontList[i].id, aFontList[i].name, aFontList[i].size, aFontList[i].url, aFontList[i].cssUrl);
|
|
}
|
|
}
|
|
|
|
for(i=0; i<aDefaultFontList.length; i++){
|
|
this.addFont(aDefaultFontList[i][0], aDefaultFontList[i][1], 0, "", "", 1);
|
|
}
|
|
|
|
//set Main Font
|
|
//if(mainFontSelected=='true') {
|
|
if(htMainFont && htMainFont.id) {
|
|
//this.setMainFont(mainFontId, mainFontName, mainFontSize, mainFontUrl, mainFontCssUrl);
|
|
this.setMainFont(htMainFont.id, htMainFont.name, htMainFont.size, htMainFont.url, htMainFont.cssUrl);
|
|
}
|
|
// add font in use
|
|
if(this.htBrowser.ie && aFontInUse){
|
|
for(i=0; i<aFontInUse.length; i++){
|
|
this.addFontInUse(aFontInUse[i].id, aFontInUse[i].name, aFontInUse[i].size, aFontInUse[i].url, aFontInUse[i].cssUrl);
|
|
}
|
|
}
|
|
}
|
|
|
|
// [SMARTEDITORSUS-245] 서비스 적용 시 글꼴정보를 넘기지 않으면 기본 글꼴 목록이 보이지 않는 오류
|
|
if(!this.htOptions || !this.htOptions.aDefaultFontList || this.htOptions.aDefaultFontList.length === 0){
|
|
this.addFont("돋움,Dotum", "돋움", 0, "", "", 1, null, true);
|
|
this.addFont("돋움체,DotumChe,AppleGothic", "돋움체", 0, "", "", 1, null, true);
|
|
this.addFont("굴림,Gulim", "굴림", 0, "", "", 1, null, true);
|
|
this.addFont("굴림체,GulimChe", "굴림체", 0, "", "", 1, null, true);
|
|
this.addFont("바탕,Batang,AppleMyungjo", "바탕", 0, "", "", 1, null, true);
|
|
this.addFont("바탕체,BatangChe", "바탕체", 0, "", "", 1, null, true);
|
|
this.addFont("궁서,Gungsuh,GungSeo", "궁서", 0, "", "", 1, null, true);
|
|
this.addFont('Arial', 'Arial', 0, "", "", 1, "abcd", true);
|
|
this.addFont('Tahoma', 'Tahoma', 0, "", "", 1, "abcd", true);
|
|
this.addFont('Times New Roman', 'Times New Roman', 0, "", "", 1, "abcd", true);
|
|
this.addFont('Verdana', 'Verdana', 0, "", "", 1, "abcd", true);
|
|
this.addFont('Courier New', 'Courier New', 0, "", "", 1, "abcd", true);
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1436] 글꼴 리스트에 글꼴 종류 추가하기 기능
|
|
if(!!this.aAdditionalFontList && this.aAdditionalFontList.length > 0){
|
|
for(i = 0, nLen = this.aAdditionalFontList.length; i < nLen; i++){
|
|
this.addFont(this.aAdditionalFontList[i][0], this.aAdditionalFontList[i][1], 0, "", "", 1);
|
|
}
|
|
}
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.bDoNotRecordUndo = false;
|
|
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["addFont", jindo.$Fn(this.addFont, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["addFontInUse", jindo.$Fn(this.addFontInUse, this).bind()]);
|
|
// 블로그등 팩토리 폰트 포함 용
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["setMainFont", jindo.$Fn(this.setMainFont, this).bind()]);
|
|
// 메일등 단순 폰트 지정 용
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["setDefaultFont", jindo.$Fn(this.setDefaultFont, this).bind()]);
|
|
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["fontName", "click", "SE2M_TOGGLE_FONTNAME_LAYER"]);
|
|
|
|
this._initFontName(); // [SMARTEDITORSUS-2111] 메일쪽 요청으로 글꼴목록 초기화시점 변경
|
|
},
|
|
|
|
$AFTER_MSG_APP_READY : function(){
|
|
this._attachIEEvent();
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
//@ec[
|
|
this.oDropdownLayer = jindo.$$.getSingle("DIV.husky_se_fontName_layer", elAppContainer);
|
|
|
|
this.elFontNameLabel = jindo.$$.getSingle("SPAN.husky_se2m_current_fontName", elAppContainer);
|
|
|
|
this.elFontNameList = jindo.$$.getSingle("UL", this.oDropdownLayer);
|
|
this.elInnerLayer = this.elFontNameList.parentNode;
|
|
this.aelFontInMarkup = jindo.$$("LI", this.oDropdownLayer); // 마크업에 있는 LI
|
|
this.elFontItemTemplate = this.aelFontInMarkup.shift(); // 맨앞에 있는 LI 는 템플릿
|
|
this.aLIFontNames = jindo.$A(jindo.$$("LI", this.oDropdownLayer)).filter(function(v,i,a){return (v.firstChild !== null);})._array;
|
|
//@ec]
|
|
|
|
this.sDefaultText = this.elFontNameLabel.innerHTML;
|
|
},
|
|
|
|
//$LOCAL_BEFORE_FIRST : function(){
|
|
_initFontName : function(){
|
|
this._addFontInMarkup();
|
|
|
|
this.addAllFonts();
|
|
|
|
// [SMARTEDITORSUS-1853] 폰트가 초기화되면 현재 스타일정보를 가져와서 툴바에 반영해준다.
|
|
var oStyle;
|
|
if(this.oApp.getCurrentStyle && (oStyle = this.oApp.getCurrentStyle())){
|
|
this.$ON_MSG_STYLE_CHANGED("fontFamily", oStyle.fontFamily);
|
|
}
|
|
|
|
this.oApp.registerBrowserEvent(this.oDropdownLayer, "mouseover", "EVENT_FONTNAME_LAYER_MOUSEOVER", []);
|
|
this.oApp.registerBrowserEvent(this.oDropdownLayer, "click", "EVENT_FONTNAME_LAYER_CLICKED", []);
|
|
},
|
|
|
|
/**
|
|
* 해당 글꼴이 존재하면 LI 요소를 보여주고 true 를 반환한다.
|
|
* @param {Element} el 글꼴리스트의 LI 요소
|
|
* @param {String} sFontName 확인할 글꼴이름
|
|
* @return {Boolean} LI 요소가 있고 글꼴이 OS에 존재하면 true 반환
|
|
*/
|
|
_checkFontLI : function(el, sFontName){
|
|
if(!el){
|
|
return false;
|
|
}
|
|
|
|
var bInstalled = IsInstalledFont(sFontName);
|
|
el.style.display = bInstalled ? "block" : "none";
|
|
return bInstalled;
|
|
},
|
|
|
|
/**
|
|
* 마크업에 있는 글꼴 목록을 추가해준다.
|
|
*/
|
|
_addFontInMarkup : function(){
|
|
for(var i = 0, elLi, sFontFamily, elSeparator, bUseSeparator; (elLi = this.aelFontInMarkup[i]); i++){
|
|
if(elLi.firstChild){
|
|
sFontFamily = this._getFontFamilyFromLI(elLi).replace(this._rxQuote, "").replace(this._rxComma, ",");
|
|
// 폰트패밀리값으로 OS에 폰트가 설치되어있는지 확인하여 노출하고 노출되면 구분선노출플래그를 true 로 세팅한다.
|
|
bUseSeparator |= this._checkFontLI(elLi, sFontFamily);
|
|
}else if(elLi.className.indexOf(this.FONT_SEPARATOR) > -1){
|
|
if(elSeparator){ // 이전에 구분선이 있었으면 구분선 노출여부 판단
|
|
elSeparator.style.display = bUseSeparator ? "block" : "none";
|
|
}
|
|
|
|
elSeparator = elLi; // 새로운 구분선 저장
|
|
bUseSeparator = false; // 구분선노출플래그 리셋
|
|
}else{
|
|
elLi.style.display = "none";
|
|
}
|
|
}
|
|
// 마지막 구분선 노출여부를 확인한다.
|
|
if(elSeparator){
|
|
elSeparator.style.display = bUseSeparator ? "block" : "none";
|
|
}
|
|
},
|
|
|
|
_attachIEEvent : function(){
|
|
if(!this.htBrowser.ie){
|
|
return;
|
|
}
|
|
|
|
if(this.htBrowser.nativeVersion < 9){ // [SMARTEDITORSUS-187] [< IE9] 최초 paste 시점에 웹폰트 파일을 로드
|
|
this._wfOnPasteWYSIWYGBody = jindo.$Fn(this._onPasteWYSIWYGBody, this);
|
|
this._wfOnPasteWYSIWYGBody.attach(this.oApp.getWYSIWYGDocument().body, "paste");
|
|
|
|
return;
|
|
}
|
|
|
|
if(document.documentMode < 9){ // [SMARTEDITORSUS-169] [>= IE9] 최초 포커스 시점에 웹폰트 로드
|
|
this._wfOnFocusWYSIWYGBody = jindo.$Fn(this._onFocusWYSIWYGBody, this);
|
|
this._wfOnFocusWYSIWYGBody.attach(this.oApp.getWYSIWYGDocument().body, "focus");
|
|
|
|
return;
|
|
}
|
|
|
|
// documentMode === 9
|
|
// http://blogs.msdn.com/b/ie/archive/2010/08/17/ie9-opacity-and-alpha.aspx // opacity:0.0;
|
|
this.welEditingAreaCover = jindo.$Element('<DIV style="width:100%; height:100%; position:absolute; top:0px; left:0px; z-index:1000;"></DIV>');
|
|
|
|
this.oApp.welEditingAreaContainer.prepend(this.welEditingAreaCover);
|
|
jindo.$Fn(this._onMouseupCover, this).attach(this.welEditingAreaCover.$value(), "mouseup");
|
|
},
|
|
|
|
_onFocusWYSIWYGBody : function(e){
|
|
this._wfOnFocusWYSIWYGBody.detach(this.oApp.getWYSIWYGDocument().body, "focus");
|
|
this._loadAllBaseFont();
|
|
},
|
|
|
|
_onPasteWYSIWYGBody : function(e){
|
|
this._wfOnPasteWYSIWYGBody.detach(this.oApp.getWYSIWYGDocument().body, "paste");
|
|
this._loadAllBaseFont();
|
|
},
|
|
|
|
_onMouseupCover : function(e){
|
|
e.stop();
|
|
|
|
// [SMARTEDITORSUS-1632] 문서 모드가 9 이상일 때, 경우에 따라 this.welEditingAreaContainer가 없을 때 스크립트 오류 발생
|
|
if(this.welEditingAreaCover){
|
|
this.welEditingAreaCover.leave();
|
|
}
|
|
//this.welEditingAreaCover.leave();
|
|
// --[SMARTEDITORSUS-1632]
|
|
|
|
var oMouse = e.mouse(),
|
|
elBody = this.oApp.getWYSIWYGDocument().body,
|
|
welBody = jindo.$Element(elBody),
|
|
oSelection = this.oApp.getEmptySelection();
|
|
|
|
// [SMARTEDITORSUS-363] 강제로 Selection 을 주도록 처리함
|
|
oSelection.selectNode(elBody);
|
|
oSelection.collapseToStart();
|
|
oSelection.select();
|
|
|
|
welBody.fireEvent("mousedown", {left : oMouse.left, middle : oMouse.middle, right : oMouse.right});
|
|
welBody.fireEvent("mouseup", {left : oMouse.left, middle : oMouse.middle, right : oMouse.right});
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1691]
|
|
* [IE 10-] 에디터가 초기화되고 나서 <p></p>로만 innerHTML을 설정하는데,
|
|
* 이 경우 실제 커서는 <p></p> 내부에 있는 것이 아니라 그 앞에 위치한다.
|
|
* 따라서 임시 북마크를 사용해서 <p></p> 내부로 커서를 이동시켜 준다.
|
|
*
|
|
* [SMARTEDITORSUS-1781]
|
|
* [IE 11] 문서 모드가 Edge인 경우에 한하여
|
|
* <p><br></p>로 innerHTML을 설정하는데,
|
|
* 실제 커서는 <p><br></p> 앞에 위치한다.
|
|
* 이 경우에는 임시 북마크를 삽입할 필요 없이 <br> 앞에 커서를 위치시켜 준다.
|
|
* */
|
|
if(this.oApp.oNavigator.ie && document.documentMode < 11 && this.oApp.getEditingMode() === "WYSIWYG"){
|
|
if(this.oApp.getWYSIWYGDocument().body.innerHTML == "<p></p>"){
|
|
this.oApp.getWYSIWYGDocument().body.innerHTML = '<p><span id="husky_bookmark_start_INIT"></span><span id="husky_bookmark_end_INIT"></span></p>';
|
|
var oSelection = this.oApp.getSelection();
|
|
oSelection.moveToStringBookmark("INIT");
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark("INIT");
|
|
}
|
|
}else if(this.oApp.oNavigator.ie && this.oApp.oNavigator.nativeVersion == 11 && document.documentMode == 11 && this.oApp.getEditingMode() === "WYSIWYG"){
|
|
if(this.oApp.getWYSIWYGDocument().body.innerHTML == "<p><br></p>"){
|
|
var elCursorHolder_br = jindo.$$.getSingle("br", elBody);
|
|
oSelection.setStartBefore(elCursorHolder_br);
|
|
oSelection.setEndBefore(elCursorHolder_br);
|
|
oSelection.select();
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1781][SMARTEDITORSUS-1691]
|
|
},
|
|
|
|
$ON_EVENT_TOOLBAR_MOUSEDOWN : function(){
|
|
if(this.htBrowser.nativeVersion < 9 || document.documentMode < 9){
|
|
return;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1632] 문서 모드가 9 이상일 때, 경우에 따라 this.welEditingAreaContainer가 없을 때 스크립트 오류 발생
|
|
if(this.welEditingAreaCover){
|
|
this.welEditingAreaCover.leave();
|
|
}
|
|
//this.welEditingAreaCover.leave();
|
|
// --[SMARTEDITORSUS-1632]
|
|
},
|
|
|
|
_loadAllBaseFont : function(){
|
|
var i, nFontLen;
|
|
|
|
if(!this.htBrowser.ie){
|
|
return;
|
|
}
|
|
|
|
if(this.htBrowser.nativeVersion < 9){
|
|
for(i=0, nFontLen=this.aBaseFontList.length; i<nFontLen; i++){
|
|
this.aBaseFontList[i].loadCSS(this.oApp.getWYSIWYGDocument());
|
|
}
|
|
}else if(document.documentMode < 9){
|
|
for(i=0, nFontLen=this.aBaseFontList.length; i<nFontLen; i++){
|
|
this.aBaseFontList[i].loadCSSToMenu();
|
|
}
|
|
}
|
|
|
|
this._loadAllBaseFont = function(){};
|
|
},
|
|
|
|
_addFontToMenu: function(sDisplayName, sFontFamily, sSampleText){
|
|
var elItem = document.createElement("LI");
|
|
elItem.innerHTML = this.elFontItemTemplate.innerHTML.replace("@DisplayName@", sDisplayName).replace("FontFamily", sFontFamily).replace("@SampleText@", sSampleText);
|
|
this.elFontNameList.insertBefore(elItem, this.elFontItemTemplate);
|
|
|
|
this.aLIFontNames[this.aLIFontNames.length] = elItem;
|
|
|
|
if(this.aLIFontNames.length > 20){
|
|
this.oDropdownLayer.style.overflowX = 'hidden';
|
|
this.oDropdownLayer.style.overflowY = 'auto';
|
|
this.oDropdownLayer.style.height = '400px';
|
|
this.oDropdownLayer.style.width = '204px'; // [SMARTEDITORSUS-155] 스크롤을 포함하여 206px 이 되도록 처리
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_FONTNAME_LAYER_MOUSEOVER : function(wev){
|
|
var elTmp = this._findLI(wev.element);
|
|
if(!elTmp){
|
|
return;
|
|
}
|
|
|
|
this._clearLastHover();
|
|
|
|
elTmp.className = "hover";
|
|
this.elLastHover = elTmp;
|
|
},
|
|
|
|
$ON_EVENT_FONTNAME_LAYER_CLICKED : function(wev){
|
|
var elTmp = this._findLI(wev.element);
|
|
if(!elTmp){
|
|
return;
|
|
}
|
|
|
|
var sFontFamily = this._getFontFamilyFromLI(elTmp);
|
|
// [SMARTEDITORSUS-169] 웹폰트의 경우 fontFamily 에 ' 을 붙여주는 처리를 함
|
|
var htFontInfo = this.htAllFonts[sFontFamily.replace(/\"/g, nhn.husky.SE2M_FontNameWithLayerUI.CUSTOM_FONT_MARKS)];
|
|
var nDefaultFontSize;
|
|
if(htFontInfo){
|
|
nDefaultFontSize = htFontInfo.defaultSize+"pt";
|
|
}else{
|
|
nDefaultFontSize = 0;
|
|
}
|
|
this.oApp.exec("SET_FONTFAMILY", [sFontFamily, nDefaultFontSize]);
|
|
},
|
|
|
|
_findLI : function(elTmp){
|
|
while(elTmp.tagName != "LI"){
|
|
if(!elTmp || elTmp === this.oDropdownLayer){
|
|
return null;
|
|
}
|
|
elTmp = elTmp.parentNode;
|
|
}
|
|
|
|
if(elTmp.className.indexOf(this.FONT_SEPARATOR) > -1){
|
|
return null;
|
|
}
|
|
return elTmp;
|
|
},
|
|
|
|
_clearLastHover : function(){
|
|
if(this.elLastHover){
|
|
this.elLastHover.className = "";
|
|
}
|
|
},
|
|
|
|
$ON_SE2M_TOGGLE_FONTNAME_LAYER : function(){
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.oDropdownLayer, null, "MSG_FONTNAME_LAYER_OPENED", [], "MSG_FONTNAME_LAYER_CLOSED", []]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['font']);
|
|
},
|
|
|
|
$ON_MSG_FONTNAME_LAYER_OPENED : function(){
|
|
this.oApp.exec("SELECT_UI", ["fontName"]);
|
|
},
|
|
|
|
$ON_MSG_FONTNAME_LAYER_CLOSED : function(){
|
|
this._clearLastHover();
|
|
this.oApp.exec("DESELECT_UI", ["fontName"]);
|
|
},
|
|
|
|
$ON_MSG_STYLE_CHANGED : function(sAttributeName, sAttributeValue){
|
|
if(sAttributeName == "fontFamily"){
|
|
sAttributeValue = sAttributeValue.replace(/["']/g, "");
|
|
var elLi = this._getMatchingLI(sAttributeValue);
|
|
this._clearFontNameSelection();
|
|
if(elLi){
|
|
this.elFontNameLabel.innerHTML = this._getFontNameLabelFromLI(elLi);
|
|
jindo.$Element(elLi).addClass("active");
|
|
}else{
|
|
//var sDisplayName = this.htFamilyName2DisplayName[sAttributeValue] || sAttributeValue;
|
|
var sDisplayName = this.sDefaultText;
|
|
this.elFontNameLabel.innerHTML = sDisplayName;
|
|
}
|
|
}
|
|
},
|
|
|
|
$BEFORE_RECORD_UNDO_BEFORE_ACTION : function(){
|
|
return !this.bDoNotRecordUndo;
|
|
},
|
|
$BEFORE_RECORD_UNDO_AFTER_ACTION : function(){
|
|
return !this.bDoNotRecordUndo;
|
|
},
|
|
$BEFORE_RECORD_UNDO_ACTION : function(){
|
|
return !this.bDoNotRecordUndo;
|
|
},
|
|
|
|
$ON_SET_FONTFAMILY : function(sFontFamily, sDefaultSize){
|
|
if(!sFontFamily){return;}
|
|
|
|
// [SMARTEDITORSUS-169] 웹폰트의 경우 fontFamily 에 ' 을 붙여주는 처리를 함
|
|
var oFontInfo = this.htAllFonts[sFontFamily.replace(/\"/g, nhn.husky.SE2M_FontNameWithLayerUI.CUSTOM_FONT_MARKS)];
|
|
if(!!oFontInfo){
|
|
oFontInfo.loadCSS(this.oApp.getWYSIWYGDocument());
|
|
}
|
|
|
|
// fontFamily와 fontSize 두개의 액션을 하나로 묶어서 undo history 저장
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["SET FONTFAMILY", {bMustBlockElement:true}]);
|
|
this.bDoNotRecordUndo = true;
|
|
|
|
if(parseInt(sDefaultSize, 10) > 0){
|
|
this.oApp.exec("SET_WYSIWYG_STYLE", [{"fontSize":sDefaultSize}]);
|
|
}
|
|
this.oApp.exec("SET_WYSIWYG_STYLE", [{"fontFamily":sFontFamily}]);
|
|
|
|
this.bDoNotRecordUndo = false;
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["SET FONTFAMILY", {bMustBlockElement:true}]);
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
},
|
|
|
|
_getMatchingLI : function(sFontName){
|
|
sFontName = sFontName.toLowerCase();
|
|
var elLi, aFontFamily;
|
|
for(var i=0; i<this.aLIFontNames.length; i++){
|
|
elLi = this.aLIFontNames[i];
|
|
aFontFamily = this._getFontFamilyFromLI(elLi).toLowerCase().split(",");
|
|
for(var h=0; h < aFontFamily.length;h++){
|
|
if( !!aFontFamily[h] && jindo.$S(aFontFamily[h].replace(/['"]/ig, "")).trim().$value() == sFontName){
|
|
return elLi;
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
},
|
|
|
|
_getFontFamilyFromLI : function(elLi){
|
|
//return elLi.childNodes[1].innerHTML.toLowerCase();
|
|
// <li><button type="button"><span>돋음</span>(</span><em style="font-family:'돋음',Dotum,'굴림',Gulim,Helvetica,Sans-serif;">돋음</em><span>)</span></span></button></li>
|
|
return (elLi.getElementsByTagName("EM")[0]).style.fontFamily;
|
|
},
|
|
|
|
_getFontNameLabelFromLI : function(elLi){
|
|
return elLi.firstChild.firstChild.firstChild.nodeValue;
|
|
},
|
|
|
|
_clearFontNameSelection : function(elLi){
|
|
for(var i=0; i<this.aLIFontNames.length; i++){
|
|
jindo.$Element(this.aLIFontNames[i]).removeClass("active");
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Add the font to the list
|
|
* @param fontId {String} value of font-family in style
|
|
* @param fontName {String} name of font list in editor
|
|
* @param defaultSize
|
|
* @param fontURL
|
|
* @param fontCSSURL
|
|
* @param fontType fontType == null, custom font (sent from the server)
|
|
* fontType == 1, default font
|
|
* fontType == 2, tempSavedFont
|
|
* @param sSampleText {String} sample text of font list in editor
|
|
* @param bCheck {Boolean}
|
|
*/
|
|
addFont : function (fontId, fontName, defaultSize, fontURL, fontCSSURL, fontType, sSampleText, bCheck) {
|
|
// custom font feature only available in IE
|
|
if(!this.htBrowser.ie && fontCSSURL){
|
|
return null;
|
|
}
|
|
|
|
// OS에 해당 폰트가 존재하는지 여부를 확인한다.
|
|
if(bCheck && !IsInstalledFont(fontId)){
|
|
return null;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-2235] 폰트패밀리가 대소문자 구분되어 삽입될 수 있도록 주석처리
|
|
//fontId = fontId.toLowerCase();
|
|
|
|
var newFont = new fontProperty(fontId, fontName, defaultSize, fontURL, fontCSSURL);
|
|
|
|
var sFontFamily;
|
|
var sDisplayName;
|
|
if(defaultSize>0){
|
|
sFontFamily = fontId+"_"+defaultSize;
|
|
sDisplayName = fontName+"_"+defaultSize;
|
|
}else{
|
|
sFontFamily = fontId;
|
|
sDisplayName = fontName;
|
|
}
|
|
|
|
if(!fontType){
|
|
sFontFamily = nhn.husky.SE2M_FontNameWithLayerUI.CUSTOM_FONT_MARKS + sFontFamily + nhn.husky.SE2M_FontNameWithLayerUI.CUSTOM_FONT_MARKS;
|
|
}
|
|
|
|
if(this.htAllFonts[sFontFamily]){
|
|
return this.htAllFonts[sFontFamily];
|
|
}
|
|
this.htAllFonts[sFontFamily] = newFont;
|
|
/*
|
|
// do not add again, if the font is already in the list
|
|
for(var i=0; i<this._allFontList.length; i++){
|
|
if(newFont.fontFamily == this._allFontList[i].fontFamily){
|
|
return this._allFontList[i];
|
|
}
|
|
}
|
|
|
|
this._allFontList[this._allFontList.length] = newFont;
|
|
*/
|
|
// [SMARTEDITORSUS-169] [IE9] 웹폰트A 선택>웹폰트B 선택>웹폰트 A를 다시 선택하면 웹폰트 A가 적용되지 않는 문제가 발생
|
|
//
|
|
// [원인]
|
|
// - IE9의 웹폰트 로드/언로드 시점
|
|
// 웹폰트 로드 시점: StyleSheet 의 @font-face 구문이 해석된 이후, DOM Tree 상에서 해당 웹폰트가 최초로 사용된 시점
|
|
// 웹폰트 언로드 시점: StyleSheet 의 @font-face 구문이 해석된 이후, DOM Tree 상에서 해당 웬폰트가 더이상 사용되지 않는 시점
|
|
// - 메뉴 리스트에 적용되는 스타일은 @font-face 이전에 처리되는 것이어서 언로드에 영향을 미치지 않음
|
|
//
|
|
// 스마트에디터의 경우, 웹폰트를 선택할 때마다 SPAN 이 새로 추가되는 것이 아닌 선택된 SPAN 의 fontFamily 를 변경하여 처리하므로
|
|
// fontFamily 변경 후 DOM Tree 상에서 더이상 사용되지 않는 것으로 브라우저 판단하여 언로드 해버림.
|
|
// [해결]
|
|
// 언로드가 발생하지 않도록 메뉴 리스트에 스타일을 적용하는 것을 @font-face 이후로 하도록 처리하여 DOM Tree 상에 항상 적용될 수 있도록 함
|
|
//
|
|
// [SMARTEDITORSUS-969] [IE10] 웹폰트를 사용하여 글을 등록하고, 수정모드로 들어갔을 때 웹폰트가 적용되지 않는 문제
|
|
// - IE10에서도 웹폰트 언로드가 발생하지 않도록 조건을 수정함
|
|
// -> 기존 : nativeVersion === 9 && documentMode === 9
|
|
// -> 수정 : nativeVersion >= 9 && documentMode >= 9
|
|
if(this.htBrowser.ie && this.htBrowser.nativeVersion >= 9 && document.documentMode >= 9) {
|
|
newFont.loadCSSToMenu();
|
|
}
|
|
|
|
this.htFamilyName2DisplayName[sFontFamily] = fontName;
|
|
|
|
sSampleText = sSampleText || this.oApp.$MSG('SE2M_FontNameWithLayerUI.sSampleText');
|
|
this._addFontToMenu(sDisplayName, sFontFamily, sSampleText);
|
|
|
|
if(!fontType){
|
|
this.aBaseFontList[this.aBaseFontList.length] = newFont;
|
|
}else{
|
|
if(fontType == 1){
|
|
this.aDefaultFontList[this.aDefaultFontList.length] = newFont;
|
|
}else{
|
|
this.aTempSavedFontList[this.aTempSavedFontList.length] = newFont;
|
|
}
|
|
}
|
|
|
|
return newFont;
|
|
},
|
|
// Add the font AND load it right away
|
|
addFontInUse : function (fontId, fontName, defaultSize, fontURL, fontCSSURL, fontType) {
|
|
var newFont = this.addFont(fontId, fontName, defaultSize, fontURL, fontCSSURL, fontType);
|
|
if(!newFont){
|
|
return null;
|
|
}
|
|
|
|
newFont.loadCSS(this.oApp.getWYSIWYGDocument());
|
|
|
|
return newFont;
|
|
},
|
|
// Add the font AND load it right away AND THEN set it as the default font
|
|
setMainFont : function (fontId, fontName, defaultSize, fontURL, fontCSSURL, fontType) {
|
|
var newFont = this.addFontInUse(fontId, fontName, defaultSize, fontURL, fontCSSURL, fontType);
|
|
if(!newFont){
|
|
return null;
|
|
}
|
|
|
|
this.setDefaultFont(newFont.fontFamily, defaultSize);
|
|
|
|
return newFont;
|
|
},
|
|
|
|
setDefaultFont : function(sFontFamily, nFontSize){
|
|
var elBody = this.oApp.getWYSIWYGDocument().body;
|
|
elBody.style.fontFamily = sFontFamily;
|
|
if(nFontSize>0){elBody.style.fontSize = nFontSize + 'pt';}
|
|
}
|
|
});
|
|
|
|
nhn.husky.SE2M_FontNameWithLayerUI.CUSTOM_FONT_MARKS = "'"; // [SMARTEDITORSUS-169] 웹폰트의 경우 fontFamily 에 ' 을 붙여주는 처리를 함
|
|
|
|
// property function for all fonts - including the default fonts and the custom fonts
|
|
// non-custom fonts will have the defaultSize of 0 and empty string for fontURL/fontCSSURL
|
|
function fontProperty(fontId, fontName, defaultSize, fontURL, fontCSSURL){
|
|
this.fontId = fontId;
|
|
this.fontName = fontName;
|
|
this.defaultSize = defaultSize;
|
|
this.fontURL = fontURL;
|
|
this.fontCSSURL = fontCSSURL;
|
|
|
|
this.displayName = fontName;
|
|
this.isLoaded = true;
|
|
this.fontFamily = this.fontId;
|
|
|
|
// it is custom font
|
|
if(this.fontCSSURL != ""){
|
|
this.displayName += '' + defaultSize;
|
|
this.fontFamily += '_' + defaultSize;
|
|
// custom fonts requires css loading
|
|
this.isLoaded = false;
|
|
|
|
// load the css that loads the custom font
|
|
this.loadCSS = function(doc){
|
|
// if the font is loaded already, return
|
|
if(this.isLoaded){
|
|
return;
|
|
}
|
|
|
|
this._importCSS(doc);
|
|
this.isLoaded = true;
|
|
};
|
|
|
|
// [SMARTEDITORSUS-169] [IE9]
|
|
// addImport 후에 처음 적용된 DOM-Tree 가 iframe 내부인 경우 (setMainFont || addFontInUse 에서 호출된 경우)
|
|
// 해당 폰트에 대한 언로드 문제가 계속 발생하여 IE9에서 addFont 에서 호출하는 loadCSS 의 경우에는 isLoaded를 true 로 변경하지 않음.
|
|
this.loadCSSToMenu = function(){
|
|
this._importCSS(document);
|
|
};
|
|
|
|
this._importCSS = function(doc){
|
|
var nStyleSheet = doc.styleSheets.length;
|
|
var oStyleSheet = doc.styleSheets[nStyleSheet - 1];
|
|
|
|
if(nStyleSheet === 0 || oStyleSheet.imports.length > 30){ // imports limit
|
|
// [SMARTEDITORSUS-1828] IE11에서 document.createStyleSheet API가 제거되었음
|
|
// [SMARTEDITORSUS-2181] createStyleSheet 를 사용하면 순서보장이 안되어 createElement("style") 로 삽입하는 방식만 사용
|
|
// 참고1 : http://msdn.microsoft.com/en-us/library/ie/bg182625(v=vs.85).aspx#legacyapis
|
|
// 참고2 : http://msdn.microsoft.com/en-us/library/ie/ms531194(v=vs.85).aspx
|
|
oStyleSheet = doc.createElement("style");
|
|
doc.documentElement.firstChild.appendChild(oStyleSheet);
|
|
// [SMARTEDITORSUS-2189] sheet은 모던브라우저, IE9이상부터 지원 (IE8이하는 styleSheet 으로 가져와야함)
|
|
// 참고 : http://help.dottoro.com/ljpatulu.php
|
|
oStyleSheet = oStyleSheet.sheet || oStyleSheet.styleSheet;
|
|
}
|
|
|
|
oStyleSheet.addImport(this.fontCSSURL);
|
|
};
|
|
}else{
|
|
this.loadCSS = function(){};
|
|
this.loadCSSToMenu = function(){};
|
|
}
|
|
|
|
this.toStruct = function(){
|
|
return {fontId:this.fontId, fontName:this.fontName, defaultSize:this.defaultSize, fontURL:this.fontURL, fontCSSURL:this.fontCSSURL};
|
|
};
|
|
}
|
|
/**
|
|
* ColorPicker Component
|
|
* @author gony
|
|
*/
|
|
nhn.ColorPicker = jindo.$Class({
|
|
elem : null,
|
|
huePanel : null,
|
|
canvasType : "Canvas",
|
|
_hsvColor : null,
|
|
$init : function(oElement, oOptions) {
|
|
this.elem = jindo.$Element(oElement).empty();
|
|
this.huePanel = null;
|
|
this.cursor = jindo.$Element("<div>").css("overflow", "hidden");
|
|
this.canvasType = jindo.$(oElement).filters?"Filter":jindo.$("<canvas>").getContext?"Canvas":null;
|
|
|
|
if(!this.canvasType) {
|
|
return false;
|
|
}
|
|
|
|
this.option({
|
|
huePanel : null,
|
|
huePanelType : "horizontal"
|
|
});
|
|
|
|
this.option(oOptions);
|
|
if (this.option("huePanel")) {
|
|
this.huePanel = jindo.$Element(this.option("huePanel")).empty();
|
|
}
|
|
|
|
// rgb
|
|
this._hsvColor = this._hsv(0,100,100); // #FF0000
|
|
|
|
// event binding
|
|
for(var name in this) {
|
|
if (/^_on[A-Z][a-z]+[A-Z][a-z]+$/.test(name)) {
|
|
this[name+"Fn"] = jindo.$Fn(this[name], this);
|
|
}
|
|
}
|
|
|
|
this._onDownColorFn.attach(this.elem, "mousedown");
|
|
if (this.huePanel) {
|
|
this._onDownHueFn.attach(this.huePanel, "mousedown");
|
|
}
|
|
|
|
// paint
|
|
this.paint();
|
|
},
|
|
rgb : function(rgb) {
|
|
this.hsv(this._rgb2hsv(rgb.r, rgb.g, rgb.b));
|
|
},
|
|
hsv : function(hsv) {
|
|
if (typeof hsv == "undefined") {
|
|
return this._hsvColor;
|
|
}
|
|
|
|
var rgb = null;
|
|
var w = this.elem.width();
|
|
var h = this.elem.height();
|
|
var cw = this.cursor.width();
|
|
var ch = this.cursor.height();
|
|
var x = 0, y = 0;
|
|
|
|
if (this.huePanel) {
|
|
rgb = this._hsv2rgb(hsv.h, 100, 100);
|
|
this.elem.css("background", "#"+this._rgb2hex(rgb.r, rgb.g, rgb.b));
|
|
|
|
x = hsv.s/100 * w;
|
|
y = (100-hsv.v)/100 * h;
|
|
} else {
|
|
var hw = w / 2;
|
|
if (hsv.v > hsv.s) {
|
|
hsv.v = 100;
|
|
x = hsv.s/100 * hw;
|
|
} else {
|
|
hsv.s = 100;
|
|
x = (100-hsv.v)/100 * hw + hw;
|
|
}
|
|
y = hsv.h/360 * h;
|
|
}
|
|
|
|
x = Math.max(Math.min(x-1,w-cw), 1);
|
|
y = Math.max(Math.min(y-1,h-ch), 1);
|
|
|
|
this.cursor.css({left:x+"px",top:y+"px"});
|
|
|
|
this._hsvColor = hsv;
|
|
rgb = this._hsv2rgb(hsv.h, hsv.s, hsv.v);
|
|
|
|
this.fireEvent("colorchange", {type:"colorchange", element:this, currentElement:this, rgbColor:rgb, hexColor:"#"+this._rgb2hex(rgb.r, rgb.g, rgb.b), hsvColor:hsv} );
|
|
},
|
|
paint : function() {
|
|
if (this.huePanel) {
|
|
// paint color panel
|
|
this["_paintColWith"+this.canvasType]();
|
|
|
|
// paint hue panel
|
|
this["_paintHueWith"+this.canvasType]();
|
|
} else {
|
|
// paint color panel
|
|
this["_paintOneWith"+this.canvasType]();
|
|
}
|
|
|
|
// draw cursor
|
|
this.cursor.appendTo(this.elem);
|
|
this.cursor.css({position:"absolute",top:"1px",left:"1px",background:"white",border:"1px solid black"}).width(3).height(3);
|
|
|
|
this.hsv(this._hsvColor);
|
|
},
|
|
_paintColWithFilter : function() {
|
|
// white : left to right
|
|
jindo.$Element("<div>").css({
|
|
position : "absolute",
|
|
top : 0,
|
|
left : 0,
|
|
width : "100%",
|
|
height : "100%",
|
|
filter : "progid:DXImageTransform.Microsoft.Gradient(GradientType=1,StartColorStr='#FFFFFFFF',EndColorStr='#00FFFFFF')"
|
|
}).appendTo(this.elem);
|
|
|
|
// black : down to up
|
|
jindo.$Element("<div>").css({
|
|
position : "absolute",
|
|
top : 0,
|
|
left : 0,
|
|
width : "100%",
|
|
height : "100%",
|
|
filter : "progid:DXImageTransform.Microsoft.Gradient(GradientType=0,StartColorStr='#00000000',EndColorStr='#FF000000')"
|
|
}).appendTo(this.elem);
|
|
},
|
|
_paintColWithCanvas : function() {
|
|
var cvs = jindo.$Element("<canvas>").css({width:"100%",height:"100%"});
|
|
cvs.appendTo(this.elem.empty());
|
|
|
|
var ctx = cvs.attr("width", cvs.width()).attr("height", cvs.height()).$value().getContext("2d");
|
|
var lin = null;
|
|
var w = cvs.width();
|
|
var h = cvs.height();
|
|
|
|
// white : left to right
|
|
lin = ctx.createLinearGradient(0,0,w,0);
|
|
lin.addColorStop(0, "rgba(255,255,255,1)");
|
|
lin.addColorStop(1, "rgba(255,255,255,0)");
|
|
ctx.fillStyle = lin;
|
|
ctx.fillRect(0,0,w,h);
|
|
|
|
// black : down to top
|
|
lin = ctx.createLinearGradient(0,0,0,h);
|
|
lin.addColorStop(0, "rgba(0,0,0,0)");
|
|
lin.addColorStop(1, "rgba(0,0,0,1)");
|
|
ctx.fillStyle = lin;
|
|
ctx.fillRect(0,0,w,h);
|
|
},
|
|
_paintOneWithFilter : function() {
|
|
var sp, ep, s_rgb, e_rgb, s_hex, e_hex;
|
|
var h = this.elem.height();
|
|
|
|
for(var i=1; i < 7; i++) {
|
|
sp = Math.floor((i-1)/6 * h);
|
|
ep = Math.floor(i/6 * h);
|
|
|
|
s_rgb = this._hsv2rgb((i-1)/6*360, 100, 100);
|
|
e_rgb = this._hsv2rgb(i/6*360, 100, 100);
|
|
s_hex = "#FF"+this._rgb2hex(s_rgb.r, s_rgb.g, s_rgb.b);
|
|
e_hex = "#FF"+this._rgb2hex(e_rgb.r, e_rgb.g, e_rgb.b);
|
|
|
|
jindo.$Element("<div>").css({
|
|
position : "absolute",
|
|
left : 0,
|
|
width : "100%",
|
|
top : sp + "px",
|
|
height : (ep-sp) + "px",
|
|
filter : "progid:DXImageTransform.Microsoft.Gradient(GradientType=0,StartColorStr='"+s_hex+"',EndColorStr='"+e_hex+"')"
|
|
}).appendTo(this.elem);
|
|
}
|
|
|
|
// white : left to right
|
|
jindo.$Element("<div>").css({
|
|
position : "absolute",
|
|
top : 0,
|
|
left : 0,
|
|
width : "50%",
|
|
height : "100%",
|
|
filter : "progid:DXImageTransform.Microsoft.Gradient(GradientType=1,StartColorStr='#FFFFFFFF',EndColorStr='#00FFFFFF')"
|
|
}).appendTo(this.elem);
|
|
|
|
// black : down to up
|
|
jindo.$Element("<div>").css({
|
|
position : "absolute",
|
|
top : 0,
|
|
right : 0,
|
|
width : "50%",
|
|
height : "100%",
|
|
filter : "progid:DXImageTransform.Microsoft.Gradient(GradientType=1,StartColorStr='#00000000',EndColorStr='#FF000000')"
|
|
}).appendTo(this.elem);
|
|
},
|
|
_paintOneWithCanvas : function() {
|
|
var rgb = {r:0, g:0, b:0};
|
|
var cvs = jindo.$Element("<canvas>").css({width:"100%",height:"100%"});
|
|
cvs.appendTo(this.elem.empty());
|
|
|
|
var ctx = cvs.attr("width", cvs.width()).attr("height", cvs.height()).$value().getContext("2d");
|
|
|
|
var w = cvs.width();
|
|
var h = cvs.height();
|
|
var lin = ctx.createLinearGradient(0,0,0,h);
|
|
|
|
for(var i=0; i < 7; i++) {
|
|
rgb = this._hsv2rgb(i/6*360, 100, 100);
|
|
lin.addColorStop(i/6, "rgb("+rgb.join(",")+")");
|
|
}
|
|
ctx.fillStyle = lin;
|
|
ctx.fillRect(0,0,w,h);
|
|
|
|
lin = ctx.createLinearGradient(0,0,w,0);
|
|
lin.addColorStop(0, "rgba(255,255,255,1)");
|
|
lin.addColorStop(0.5, "rgba(255,255,255,0)");
|
|
lin.addColorStop(0.5, "rgba(0,0,0,0)");
|
|
lin.addColorStop(1, "rgba(0,0,0,1)");
|
|
ctx.fillStyle = lin;
|
|
ctx.fillRect(0,0,w,h);
|
|
},
|
|
_paintHueWithFilter : function() {
|
|
var sp, ep, s_rgb, e_rgb, s_hex, e_hex;
|
|
var vert = (this.option().huePanelType == "vertical");
|
|
var w = this.huePanel.width();
|
|
var h = this.huePanel.height();
|
|
var elDiv = null;
|
|
|
|
var nPanelBorderWidth = parseInt(this.huePanel.css('borderWidth'), 10);
|
|
if (!!isNaN(nPanelBorderWidth)) { nPanelBorderWidth = 0; }
|
|
w -= nPanelBorderWidth * 2; // borderWidth를 제외한 내측 폭을 구함
|
|
|
|
for(var i=1; i < 7; i++) {
|
|
sp = Math.floor((i-1)/6 * (vert?h:w));
|
|
ep = Math.floor(i/6 * (vert?h:w));
|
|
|
|
s_rgb = this._hsv2rgb((i-1)/6*360, 100, 100);
|
|
e_rgb = this._hsv2rgb(i/6*360, 100, 100);
|
|
s_hex = "#FF"+this._rgb2hex(s_rgb.r, s_rgb.g, s_rgb.b);
|
|
e_hex = "#FF"+this._rgb2hex(e_rgb.r, e_rgb.g, e_rgb.b);
|
|
|
|
elDiv = jindo.$Element("<div>").css({
|
|
position : "absolute",
|
|
filter : "progid:DXImageTransform.Microsoft.Gradient(GradientType="+(vert?0:1)+",StartColorStr='"+s_hex+"',EndColorStr='"+e_hex+"')"
|
|
});
|
|
|
|
var width = (ep - sp) + 1; // IE에서 폭을 넓혀주지 않으면 확대 시 벌어짐, 그래서 1px 보정
|
|
elDiv.appendTo(this.huePanel);
|
|
elDiv.css(vert?"left":"top", 0).css(vert?"width":"height", '100%');
|
|
elDiv.css(vert?"top":"left", sp + "px").css(vert?"height":"width", width + "px");
|
|
}
|
|
},
|
|
_paintHueWithCanvas : function() {
|
|
var opt = this.option(), rgb;
|
|
var vtc = (opt.huePanelType == "vertical");
|
|
|
|
var cvs = jindo.$Element("<canvas>").css({width:"100%",height:"100%"});
|
|
cvs.appendTo(this.huePanel.empty());
|
|
|
|
var ctx = cvs.attr("width", cvs.width()).attr("height", cvs.height()).$value().getContext("2d");
|
|
var lin = ctx.createLinearGradient(0,0,vtc?0:cvs.width(),vtc?cvs.height():0);
|
|
|
|
for(var i=0; i < 7; i++) {
|
|
rgb = this._hsv2rgb(i/6*360, 100, 100);
|
|
lin.addColorStop(i/6, "rgb("+rgb.join(",")+")");
|
|
}
|
|
ctx.fillStyle = lin;
|
|
ctx.fillRect(0,0,cvs.width(),cvs.height());
|
|
},
|
|
_rgb2hsv : function(r,g,b) {
|
|
var h = 0, s = 0, v = Math.max(r,g,b), min = Math.min(r,g,b), delta = v - min;
|
|
s = (v ? delta/v : 0);
|
|
|
|
if (s) {
|
|
if (r == v) {
|
|
h = 60 * (g - b) / delta;
|
|
} else if (g == v) {
|
|
h = 120 + 60 * (b - r) / delta;
|
|
} else if (b == v) {
|
|
h = 240 + 60 * (r - g) / delta;
|
|
}
|
|
|
|
if (h < 0) {
|
|
h += 360;
|
|
}
|
|
}
|
|
|
|
h = Math.floor(h);
|
|
s = Math.floor(s * 100);
|
|
v = Math.floor(v / 255 * 100);
|
|
|
|
return this._hsv(h,s,v);
|
|
},
|
|
_hsv2rgb : function(h,s,v) {
|
|
h = (h % 360) / 60; s /= 100; v /= 100;
|
|
|
|
var r=0, g=0, b=0;
|
|
var i = Math.floor(h);
|
|
var f = h-i;
|
|
var p = v*(1-s);
|
|
var q = v*(1-s*f);
|
|
var t = v*(1-s*(1-f));
|
|
|
|
switch (i) {
|
|
case 0: r=v; g=t; b=p; break;
|
|
case 1: r=q; g=v; b=p; break;
|
|
case 2: r=p; g=v; b=t; break;
|
|
case 3: r=p; g=q; b=v; break;
|
|
case 4: r=t; g=p; b=v; break;
|
|
case 5: r=v; g=p; b=q;break;
|
|
case 6: break;
|
|
}
|
|
|
|
r = Math.floor(r*255);
|
|
g = Math.floor(g*255);
|
|
b = Math.floor(b*255);
|
|
|
|
return this._rgb(r,g,b);
|
|
},
|
|
_rgb2hex : function(r,g,b) {
|
|
r = r.toString(16);
|
|
if (r.length == 1) {
|
|
r = '0'+r;
|
|
}
|
|
|
|
g = g.toString(16);
|
|
if (g.length==1) {
|
|
g = '0'+g;
|
|
}
|
|
|
|
b = b.toString(16);
|
|
if (b.length==1) {
|
|
b = '0'+b;
|
|
}
|
|
|
|
return r+g+b;
|
|
},
|
|
_hex2rgb : function(hex) {
|
|
var m = hex.match(/#?([0-9a-f]{6}|[0-9a-f]{3})/i);
|
|
if (m[1].length == 3) {
|
|
m = m[1].match(/./g).filter(function(c) {
|
|
return c+c;
|
|
});
|
|
} else {
|
|
m = m[1].match(/../g);
|
|
}
|
|
return {
|
|
r : Number("0x" + m[0]),
|
|
g : Number("0x" + m[1]),
|
|
b : Number("0x" + m[2])
|
|
};
|
|
},
|
|
_rgb : function(r,g,b) {
|
|
var ret = [r,g,b];
|
|
|
|
ret.r = r;
|
|
ret.g = g;
|
|
ret.b = b;
|
|
|
|
return ret;
|
|
},
|
|
_hsv : function(h,s,v) {
|
|
var ret = [h,s,v];
|
|
|
|
ret.h = h;
|
|
ret.s = s;
|
|
ret.v = v;
|
|
|
|
return ret;
|
|
},
|
|
_onDownColor : function(e) {
|
|
if (!e.mouse().left) {
|
|
return false;
|
|
}
|
|
|
|
var pos = e.pos();
|
|
|
|
this._colPagePos = [pos.pageX, pos.pageY];
|
|
this._colLayerPos = [pos.layerX, pos.layerY];
|
|
|
|
this._onUpColorFn.attach(document, "mouseup");
|
|
this._onMoveColorFn.attach(document, "mousemove");
|
|
|
|
this._onMoveColor(e);
|
|
},
|
|
_onUpColor : function(e) {
|
|
this._onUpColorFn.detach(document, "mouseup");
|
|
this._onMoveColorFn.detach(document, "mousemove");
|
|
},
|
|
_onMoveColor : function(e) {
|
|
var hsv = this._hsvColor;
|
|
var pos = e.pos();
|
|
var x = this._colLayerPos[0] + (pos.pageX - this._colPagePos[0]);
|
|
var y = this._colLayerPos[1] + (pos.pageY - this._colPagePos[1]);
|
|
var w = this.elem.width();
|
|
var h = this.elem.height();
|
|
|
|
x = Math.max(Math.min(x, w), 0);
|
|
y = Math.max(Math.min(y, h), 0);
|
|
|
|
if (this.huePanel) {
|
|
hsv.s = hsv[1] = x / w * 100;
|
|
hsv.v = hsv[2] = (h - y) / h * 100;
|
|
} else {
|
|
hsv.h = y/h*360;
|
|
|
|
var hw = w/2;
|
|
|
|
if (x < hw) {
|
|
hsv.s = x/hw * 100;
|
|
hsv.v = 100;
|
|
} else {
|
|
hsv.s = 100;
|
|
hsv.v = (w-x)/hw * 100;
|
|
}
|
|
}
|
|
|
|
this.hsv(hsv);
|
|
|
|
e.stop();
|
|
},
|
|
_onDownHue : function(e) {
|
|
if (!e.mouse().left) {
|
|
return false;
|
|
}
|
|
|
|
var pos = e.pos();
|
|
|
|
this._huePagePos = [pos.pageX, pos.pageY];
|
|
this._hueLayerPos = [pos.layerX, pos.layerY];
|
|
|
|
this._onUpHueFn.attach(document, "mouseup");
|
|
this._onMoveHueFn.attach(document, "mousemove");
|
|
|
|
this._onMoveHue(e);
|
|
},
|
|
_onUpHue : function(e) {
|
|
this._onUpHueFn.detach(document, "mouseup");
|
|
this._onMoveHueFn.detach(document, "mousemove");
|
|
},
|
|
_onMoveHue : function(e) {
|
|
var hsv = this._hsvColor;
|
|
var pos = e.pos();
|
|
var cur = 0, len = 0;
|
|
var x = this._hueLayerPos[0] + (pos.pageX - this._huePagePos[0]);
|
|
var y = this._hueLayerPos[1] + (pos.pageY - this._huePagePos[1]);
|
|
|
|
if (this.option().huePanelType == "vertical") {
|
|
cur = y;
|
|
len = this.huePanel.height();
|
|
} else {
|
|
cur = x;
|
|
len = this.huePanel.width();
|
|
}
|
|
|
|
hsv.h = hsv[0] = (Math.min(Math.max(cur, 0), len)/len * 360)%360;
|
|
|
|
this.hsv(hsv);
|
|
|
|
e.stop();
|
|
}
|
|
}).extend(jindo.Component);
|
|
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of Accessibility about SmartEditor2.
|
|
* @name hp_SE2M_Accessibility.js
|
|
*/
|
|
nhn.husky.SE2M_Accessibility = jindo.$Class({
|
|
name : "SE2M_Accessibility",
|
|
|
|
/*
|
|
* elAppContainer : mandatory
|
|
* sLocale, sEditorType : optional
|
|
*/
|
|
$init: function(elAppContainer, sLocale, sEditorType) {
|
|
this._assignHTMLElements(elAppContainer);
|
|
|
|
if(!!sLocale){
|
|
this.sLang = sLocale;
|
|
}
|
|
|
|
if(!!sEditorType){
|
|
this.sType = sEditorType;
|
|
}
|
|
},
|
|
|
|
_assignHTMLElements : function(elAppContainer){
|
|
this.elHelpPopupLayer = jindo.$$.getSingle("DIV.se2_accessibility", elAppContainer);
|
|
this.welHelpPopupLayer = jindo.$Element(this.elHelpPopupLayer);
|
|
|
|
//close buttons
|
|
this.oCloseButton = jindo.$$.getSingle("BUTTON.se2_close", this.elHelpPopupLayer);
|
|
this.oCloseButton2 = jindo.$$.getSingle("BUTTON.se2_close2", this.elHelpPopupLayer);
|
|
|
|
this.nDefaultTop = 150;
|
|
|
|
// [SMARTEDITORSUS-1594] 포커스 탐색에 사용하기 위해 할당
|
|
this.elAppContainer = elAppContainer;
|
|
// --[SMARTEDITORSUS-1594]
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.htAccessOption = nhn.husky.SE2M_Configuration.SE2M_Accessibility || {};
|
|
this.oApp.exec("REGISTER_HOTKEY", ["alt+F10", "FOCUS_TOOLBAR_AREA", []]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["alt+COMMA", "FOCUS_BEFORE_ELEMENT", []]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["alt+PERIOD", "FOCUS_NEXT_ELEMENT", []]);
|
|
|
|
if (this.sLang && this.sLang !== 'ko_KR') {
|
|
//do nothing
|
|
return;
|
|
} else {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["alt+0", "OPEN_HELP_POPUP", []]);
|
|
|
|
//[SMARTEDITORSUS-1327] IE 7/8에서 ALT+0으로 팝업 띄우고 esc클릭시 팝업창 닫히게 하려면 아래 부분 꼭 필요함. (target은 document가 되어야 함!)
|
|
this.oApp.exec("REGISTER_HOTKEY", ["esc", "CLOSE_HELP_POPUP", [], document]);
|
|
}
|
|
|
|
//[SMARTEDITORSUS-1353]
|
|
if (this.htAccessOption.sTitleElementId) {
|
|
this.oApp.registerBrowserEvent(document.getElementById(this.htAccessOption.sTitleElementId), "keydown", "MOVE_TO_EDITAREA", []);
|
|
}
|
|
},
|
|
|
|
$ON_MOVE_TO_EDITAREA : function(weEvent) {
|
|
var TAB_KEY_CODE = 9;
|
|
if (weEvent.key().keyCode == TAB_KEY_CODE) {
|
|
if(weEvent.key().shift) {return;}
|
|
this.oApp.delayedExec("FOCUS", [], 0);
|
|
}
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(sMsg){
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("CLOSE_HELP_POPUP", [this.oCloseButton]), this).attach(this.oCloseButton, "click");
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("CLOSE_HELP_POPUP", [this.oCloseButton2]), this).attach(this.oCloseButton2, "click");
|
|
|
|
//레이어의 이동 범위 설정.
|
|
var elIframe = this.oApp.getWYSIWYGWindow().frameElement;
|
|
this.htOffsetPos = jindo.$Element(elIframe).offset();
|
|
this.nEditorWidth = elIframe.offsetWidth;
|
|
|
|
this.htInitialPos = this.welHelpPopupLayer.offset();
|
|
var htScrollXY = this.oApp.oUtils.getScrollXY();
|
|
|
|
this.nLayerWidth = 590;
|
|
this.nLayerHeight = 480;
|
|
|
|
this.htTopLeftCorner = {x:parseInt(this.htOffsetPos.left, 10), y:parseInt(this.htOffsetPos.top, 10)};
|
|
//[css markup] left:11 top:74로 되어 있음
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1594]
|
|
* SE2M_Configuration_General에서 포커스를 이동할 에디터 영역 이후의 엘레먼트를 지정해 두었다면, 설정값을 따른다.
|
|
* 지정하지 않았거나 빈 String이라면, elAppContainer를 기준으로 자동 탐색한다.
|
|
* */
|
|
$ON_FOCUS_NEXT_ELEMENT : function() {
|
|
// 포커스 캐싱
|
|
this._currentNextFocusElement = null; // 새로운 포커스 이동이 발생할 때마다 캐싱 초기화
|
|
|
|
if(this.htAccessOption.sNextElementId){
|
|
this._currentNextFocusElement = document.getElementById(this.htAccessOption.sNextElementId);
|
|
}else{
|
|
this._currentNextFocusElement = this._findNextFocusElement(this.elAppContainer);
|
|
}
|
|
|
|
if(this._currentNextFocusElement){
|
|
window.focus(); // [SMARTEDITORSUS-1360] IE7에서는 element에 대한 focus를 주기 위해 선행되어야 한다.
|
|
this._currentNextFocusElement.focus();
|
|
}else if(parent && parent.nhn && parent.nhn.husky && parent.nhn.husky.EZCreator && parent.nhn.husky.EZCreator.elIFrame){
|
|
parent.focus();
|
|
if(this._currentNextFocusElement = this._findNextFocusElement(parent.nhn.husky.EZCreator.elIFrame)){
|
|
this._currentNextFocusElement.focus();
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1594] DIV#smart_editor2 다음 요소에서 가장 가까운 포커스용 태그를 탐색
|
|
* */
|
|
_findNextFocusElement : function(targetElement){
|
|
var target = null;
|
|
|
|
var el = targetElement.nextSibling;
|
|
|
|
while(el){
|
|
if(el.nodeType !== 1){ // Element Node만을 대상으로 한다.
|
|
// 대상 노드 대신 nextSibling을 찾되, 부모를 거슬러 올라갈 수도 있다.
|
|
// document.body까지 거슬러 올라가게 되면 탐색 종료
|
|
el = this._switchToSiblingOrNothing(el);
|
|
if(!el){
|
|
break;
|
|
}else{
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// 대상 노드를 기준으로, 전위순회로 조건에 부합하는 노드 탐색
|
|
this._recursivePreorderTraversalFilter(el, this._isFocusTag);
|
|
|
|
if(this._nextFocusElement){
|
|
target = this._nextFocusElement;
|
|
|
|
// 탐색에 사용했던 변수 초기화
|
|
this._bStopFindingNextElement = false;
|
|
this._nextFocusElement = null;
|
|
|
|
break;
|
|
}else{
|
|
// 대상 노드 대신 nextSibling을 찾되, 부모를 거슬러 올라갈 수도 있다.
|
|
// document.body까지 거슬러 올라가게 되면 탐색 종료
|
|
el = this._switchToSiblingOrNothing(el);
|
|
if(!el){
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// target이 존재하지 않으면 null 반환
|
|
return target;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1594] 대상 노드를 기준으로 하여, nextSibling 또는 previousSibling을 찾는다.
|
|
* nextSibling 또는 previousSibling이 없다면,
|
|
* 부모를 거슬러 올라가면서 첫 nextSibling 또는 previousSibling을 찾는다.
|
|
* document의 body까지 올라가도 nextSibling 또는 previousSibling이 나타나지 않는다면
|
|
* 탐색 대상으로 null을 반환한다.
|
|
* @param {NodeElement} 대상 노드 (주의:NodeElement에 대한 null 체크 안함)
|
|
* @param {Boolean} 생략하거나 false이면 nextSibling을 찾고, true이면 previousSibling을 찾는다.
|
|
* */
|
|
_switchToSiblingOrNothing : function(targetElement, isPreviousOrdered){
|
|
var el = targetElement;
|
|
|
|
if(isPreviousOrdered){
|
|
if(el.previousSibling){
|
|
el = el.previousSibling;
|
|
}else{
|
|
// 형제가 없다면 부모를 거슬러 올라가면서 탐색
|
|
|
|
// 이 루프의 종료 조건
|
|
// 1. 부모를 거슬러 올라가다가 el이 document.body가 되는 시점
|
|
// - 더 이상 previousSibling을 탐색할 수 없음
|
|
// 2. el이 부모로 대체된 뒤 previousSibling이 존재하는 경우
|
|
while(el.nodeName.toUpperCase() != "BODY" && !el.previousSibling){
|
|
el = el.parentNode;
|
|
}
|
|
|
|
if(el.nodeName.toUpperCase() == "BODY"){
|
|
el = null;
|
|
}else{
|
|
el = el.previousSibling;
|
|
}
|
|
}
|
|
}else{
|
|
if(el.nextSibling){
|
|
el = el.nextSibling;
|
|
}else{
|
|
// 형제가 없다면 부모를 거슬러 올라가면서 탐색
|
|
|
|
// 이 루프의 종료 조건
|
|
// 1. 부모를 거슬러 올라가다가 el이 document.body가 되는 시점
|
|
// - 더 이상 nextSibling을 탐색할 수 없음
|
|
// 2. el이 부모로 대체된 뒤 nextSibling이 존재하는 경우
|
|
while(el.nodeName.toUpperCase() != "BODY" && !el.nextSibling){
|
|
el = el.parentNode;
|
|
}
|
|
|
|
if(el.nodeName.toUpperCase() == "BODY"){
|
|
el = null;
|
|
}else{
|
|
el = el.nextSibling;
|
|
}
|
|
}
|
|
}
|
|
|
|
return el;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1594] 대상 노드를 기준으로 하는 트리를 전위순회를 거쳐, 필터 조건에 부합하는 첫 노드를 찾는다.
|
|
* @param {NodeElement} 탐색하려는 트리의 루트 노드
|
|
* @param {Function} 필터 조건으로 사용할 함수
|
|
* @param {Boolean} 생략하거나 false이면 순수 전위순회(루트 - 좌측 - 우측 순)로 탐색하고, true이면 반대 방향의 전위순회(루트 - 우측 - 좌측)로 탐색한다.
|
|
* */
|
|
_recursivePreorderTraversalFilter : function(node, filterFunction, isReversed){
|
|
var self = this;
|
|
|
|
// 현재 노드를 기준으로 필터링
|
|
var _bStopFindingNextElement = filterFunction.apply(node);
|
|
|
|
if(_bStopFindingNextElement){
|
|
// 최초로 포커스 태그를 찾는다면 탐색 중단용 flag 변경
|
|
self._bStopFindingNextElement = true;
|
|
|
|
if(isReversed){
|
|
self._previousFocusElement = node;
|
|
}else{
|
|
self._nextFocusElement = node;
|
|
}
|
|
|
|
return;
|
|
}else{
|
|
// 필터링 조건에 부합하지 않는다면, 자식들을 기준으로 반복하게 된다.
|
|
if(isReversed){
|
|
for(var len = node.childNodes.length, i = len - 1; i >= 0; i--){
|
|
self._recursivePreorderTraversalFilter(node.childNodes[i], filterFunction, true);
|
|
if(!!this._bStopFindingNextElement){
|
|
break;
|
|
}
|
|
}
|
|
}else{
|
|
for(var i=0, len = node.childNodes.length; i < len; i++){
|
|
self._recursivePreorderTraversalFilter(node.childNodes[i], filterFunction);
|
|
if(!!this._bStopFindingNextElement){
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1594] 필터 함수로, 이 노드가 tab 키로 포커스를 이동하는 태그에 해당하는지 확인한다.
|
|
* */
|
|
_isFocusTag : function(){
|
|
var self = this;
|
|
|
|
// tab 키로 포커스를 잡아주는 태그 목록
|
|
var aFocusTagViaTabKey = ["A", "BUTTON", "INPUT", "TEXTAREA"];
|
|
|
|
// 포커스 태그가 현재 노드에 존재하는지 확인하기 위한 flag
|
|
var bFocusTagExists = false;
|
|
|
|
for(var i = 0, len = aFocusTagViaTabKey.length; i < len; i++){
|
|
if(self.nodeType === 1 && self.nodeName && self.nodeName.toUpperCase() == aFocusTagViaTabKey[i] && !self.disabled && jindo.$Element(self).visible()){
|
|
bFocusTagExists = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return bFocusTagExists;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1594]
|
|
* SE2M_Configuration_General에서 포커스를 이동할 에디터 영역 이전의 엘레먼트를 지정해 두었다면, 설정값을 따른다.
|
|
* 지정하지 않았거나 빈 String이라면, elAppContainer를 기준으로 자동 탐색한다.
|
|
* */
|
|
$ON_FOCUS_BEFORE_ELEMENT : function() {
|
|
// 포커스 캐싱
|
|
this._currentPreviousFocusElement = null; // 새로운 포커스 이동이 발생할 때마다 캐싱 초기화
|
|
|
|
if(this.htAccessOption.sBeforeElementId){
|
|
this._currentPreviousFocusElement = document.getElementById(this.htAccessOption.sBeforeElementId);
|
|
}else{
|
|
this._currentPreviousFocusElement = this._findPreviousFocusElement(this.elAppContainer); // 삽입될 대상
|
|
}
|
|
|
|
if(this._currentPreviousFocusElement){
|
|
window.focus(); // [SMARTEDITORSUS-1360] IE7에서는 element에 대한 focus를 주기 위해 선행되어야 한다.
|
|
this._currentPreviousFocusElement.focus();
|
|
}else if(parent && parent.nhn && parent.nhn.husky && parent.nhn.husky.EZCreator && parent.nhn.husky.EZCreator.elIFrame){
|
|
parent.focus();
|
|
if(this._currentPreviousFocusElement = this._findPreviousFocusElement(parent.nhn.husky.EZCreator.elIFrame)){
|
|
this._currentPreviousFocusElement.focus();
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1594] DIV#smart_editor2 이전 요소에서 가장 가까운 포커스용 태그를 탐색
|
|
* */
|
|
_findPreviousFocusElement : function(targetElement){
|
|
var target = null;
|
|
|
|
var el = targetElement.previousSibling;
|
|
|
|
while(el){
|
|
if(el.nodeType !== 1){ // Element Node만을 대상으로 한다.
|
|
// 대상 노드 대신 previousSibling을 찾되, 부모를 거슬러 올라갈 수도 있다.
|
|
// document.body까지 거슬러 올라가게 되면 탐색 종료
|
|
el = this._switchToSiblingOrNothing(el, /*isReversed*/true);
|
|
if(!el){
|
|
break;
|
|
}else{
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// 대상 노드를 기준으로, 역 전위순회로 조건에 부합하는 노드 탐색
|
|
this._recursivePreorderTraversalFilter(el, this._isFocusTag, true);
|
|
|
|
if(this._previousFocusElement){
|
|
target = this._previousFocusElement;
|
|
|
|
// 탐색에 사용했던 변수 초기화
|
|
this._bStopFindingNextElement = false;
|
|
this._previousFocusElement = null;
|
|
|
|
break;
|
|
}else{
|
|
// 대상 노드 대신 previousSibling을 찾되, 부모를 거슬러 올라갈 수도 있다.
|
|
// document.body까지 거슬러 올라가게 되면 탐색 종료
|
|
el = this._switchToSiblingOrNothing(el, /*isReversed*/true);
|
|
if(!el){
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// target이 존재하지 않으면 null 반환
|
|
return target;
|
|
},
|
|
|
|
$ON_FOCUS_TOOLBAR_AREA : function(){
|
|
this.oButton = jindo.$$.getSingle("BUTTON.se2_font_family", this.elAppContainer);
|
|
if(this.oButton && !this.oButton.disabled){ // [SMARTEDITORSUS-1369] IE9이하에서 disabled 요소에 포커스를 주면 오류 발생
|
|
window.focus();
|
|
this.oButton.focus();
|
|
}
|
|
},
|
|
|
|
$ON_OPEN_HELP_POPUP : function() {
|
|
this.oApp.exec("DISABLE_ALL_UI", [{aExceptions: ["se2_accessibility"]}]);
|
|
this.oApp.exec("SHOW_EDITING_AREA_COVER");
|
|
this.oApp.exec("SELECT_UI", ["se2_accessibility"]);
|
|
|
|
//아래 코드 없어야 블로그에서도 동일한 위치에 팝업 뜸..
|
|
//this.elHelpPopupLayer.style.top = this.nDefaultTop+"px";
|
|
|
|
this.nCalcX = this.htTopLeftCorner.x + this.oApp.getEditingAreaWidth() - this.nLayerWidth;
|
|
this.nCalcY = this.htTopLeftCorner.y - 30; // 블로그버전이 아닌 경우 에디터영역을 벗어나는 문제가 있기 때문에 기본툴바(30px) 크기만큼 올려줌
|
|
|
|
this.oApp.exec("SHOW_DIALOG_LAYER", [this.elHelpPopupLayer, {
|
|
elHandle: this.elTitle,
|
|
nMinX : this.htTopLeftCorner.x + 0,
|
|
nMinY : this.nDefaultTop + 77,
|
|
nMaxX : this.nCalcX,
|
|
nMaxY : this.nCalcY
|
|
}]);
|
|
|
|
// offset (nTop:Numeric, nLeft:Numeric)
|
|
this.welHelpPopupLayer.offset(this.nCalcY, (this.nCalcX)/2);
|
|
|
|
//[SMARTEDITORSUS-1327] IE에서 포커스 이슈로 IE에 대해서만 window.focus실행함.
|
|
if(jindo.$Agent().navigator().ie) {
|
|
window.focus();
|
|
}
|
|
|
|
var self = this;
|
|
setTimeout(function(){
|
|
try{
|
|
self.oCloseButton2.focus();
|
|
}catch(e){
|
|
}
|
|
},200);
|
|
},
|
|
|
|
$ON_CLOSE_HELP_POPUP : function() {
|
|
this.oApp.exec("ENABLE_ALL_UI"); // 모든 UI 활성화.
|
|
this.oApp.exec("DESELECT_UI", ["helpPopup"]);
|
|
this.oApp.exec("HIDE_ALL_DIALOG_LAYER", []);
|
|
this.oApp.exec("HIDE_EDITING_AREA_COVER"); // 편집 영역 활성화.
|
|
|
|
this.oApp.exec("FOCUS");
|
|
}
|
|
});
|
|
//}
|
|
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to inserting special characters
|
|
* @name hp_SE2M_SCharacter.js
|
|
* @required HuskyRangeManager
|
|
*/
|
|
nhn.husky.SE2M_SCharacter = jindo.$Class({
|
|
name : "SE2M_SCharacter",
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["sCharacter", "click", "TOGGLE_SCHARACTER_LAYER"]);
|
|
this.oApp.registerLazyMessage(["TOGGLE_SCHARACTER_LAYER"], ["hp_SE2M_SCharacter$Lazy.js"]);
|
|
}
|
|
});
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to Find/Replace
|
|
* @name hp_SE2M_FindReplacePlugin.js
|
|
*/
|
|
nhn.husky.SE2M_FindReplacePlugin = jindo.$Class({
|
|
name : "SE2M_FindReplacePlugin",
|
|
oEditingWindow : null,
|
|
oFindReplace : null,
|
|
bFindMode : true,
|
|
bLayerShown : false,
|
|
|
|
$init : function(){
|
|
this.nDefaultTop = 20;
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
// the right document will be available only when the src is completely loaded
|
|
this.oEditingWindow = this.oApp.getWYSIWYGWindow();
|
|
|
|
// [SMARTEDITORSUS-2260] 메일 > Mac에서 ctrl 조합 단축키 모두 meta 조합으로 변경
|
|
if (jindo.$Agent().os().mac) {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+f", "SHOW_FIND_LAYER", []]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+h", "SHOW_REPLACE_LAYER", []]);
|
|
} else {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+f", "SHOW_FIND_LAYER", []]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+h", "SHOW_REPLACE_LAYER", []]);
|
|
}
|
|
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["findAndReplace", "click", "TOGGLE_FIND_REPLACE_LAYER"]);
|
|
this.oApp.registerLazyMessage(["TOGGLE_FIND_REPLACE_LAYER","SHOW_FIND_LAYER","SHOW_REPLACE_LAYER","SHOW_FIND_REPLACE_LAYER"], ["hp_SE2M_FindReplacePlugin$Lazy.js","N_FindReplace.js"]);
|
|
},
|
|
|
|
$ON_SHOW_ACTIVE_LAYER : function(){
|
|
this.oApp.exec("HIDE_DIALOG_LAYER", [this.elDropdownLayer]);
|
|
}
|
|
});
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to quote
|
|
* @name hp_SE_Quote.js
|
|
* @required SE_EditingArea_WYSIWYG
|
|
*/
|
|
nhn.husky.SE2M_Quote = jindo.$Class({
|
|
name : "SE2M_Quote",
|
|
|
|
htQuoteStyles_view : null,
|
|
|
|
$init : function(){
|
|
var htConfig = nhn.husky.SE2M_Configuration.Quote || {};
|
|
var sImageBaseURL = htConfig.sImageBaseURL;
|
|
|
|
this.nMaxLevel = htConfig.nMaxLevel || 14;
|
|
|
|
this.htQuoteStyles_view = {};
|
|
this.htQuoteStyles_view["se2_quote1"] = "_zoom:1;padding:0 8px; margin:0 0 30px 20px; margin-right:15px; border-left:2px solid #cccccc;color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote2"] = "_zoom:1;margin:0 0 30px 13px;padding:0 8px 0 16px;background:url("+sImageBaseURL+"/bg_quote2.gif) 0 3px no-repeat;color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote3"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:1px dashed #cccccc;color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote4"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:1px dashed #66b246;color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote5"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:1px dashed #cccccc;background:url("+sImageBaseURL+"/bg_b1.png) repeat;_background:none;_filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='"+sImageBaseURL+"/bg_b1.png',sizingMethod='scale');color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote6"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:1px solid #e5e5e5;color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote7"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:1px solid #66b246;color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote8"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:1px solid #e5e5e5;background:url("+sImageBaseURL+"/bg_b1.png) repeat;_background:none;_filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='"+sImageBaseURL+"/bg_b1.png',sizingMethod='scale');color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote9"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:2px solid #e5e5e5;color:#888888;";
|
|
this.htQuoteStyles_view["se2_quote10"] = "_zoom:1;margin:0 0 30px 0;padding:10px;border:2px solid #e5e5e5;background:url("+sImageBaseURL+"/bg_b1.png) repeat;_background:none;_filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='"+sImageBaseURL+"/bg_b1.png',sizingMethod='scale');color:#888888;";
|
|
},
|
|
|
|
_assignHTMLElements : function(){
|
|
//@ec
|
|
this.elDropdownLayer = jindo.$$.getSingle("DIV.husky_seditor_blockquote_layer", this.oApp.htOptions.elAppContainer);
|
|
this.aLI = jindo.$$("LI", this.elDropdownLayer);
|
|
},
|
|
|
|
$ON_REGISTER_CONVERTERS : function(){
|
|
this.oApp.exec("ADD_CONVERTER", ["DB_TO_IR", jindo.$Fn(function(sContents){
|
|
sContents = sContents.replace(/<(blockquote)[^>]*class=['"]?(se2_quote[0-9]+)['"]?[^>]*>/gi, "<$1 class=$2>");
|
|
return sContents;
|
|
}, this).bind()]);
|
|
|
|
this.oApp.exec("ADD_CONVERTER", ["IR_TO_DB", jindo.$Fn(function(sContents){
|
|
var htQuoteStyles_view = this.htQuoteStyles_view;
|
|
sContents = sContents.replace(/<(blockquote)[^>]*class=['"]?(se2_quote[0-9]+)['"]?[^>]*>/gi, function(sAll, sTag, sClassName){
|
|
return '<'+sTag+' class='+sClassName+' style="'+htQuoteStyles_view[sClassName]+'">';
|
|
});
|
|
return sContents;
|
|
}, this).bind()]);
|
|
|
|
this.htSE1toSE2Map = {
|
|
"01" : "1",
|
|
"02" : "2",
|
|
"03" : "6",
|
|
"04" : "8",
|
|
"05" : "9",
|
|
"07" : "3",
|
|
"08" : "5"
|
|
};
|
|
// convert SE1's quotes to SE2's
|
|
// -> 블로그 개발 쪽에서 처리 하기로 함.
|
|
/*
|
|
this.oApp.exec("ADD_CONVERTER", ["DB_TO_IR", jindo.$Fn(function(sContents){
|
|
return sContents.replace(/<blockquote[^>]* class="?vview_quote([0-9]+)"?[^>]*>((?:\s|.)*?)<\/blockquote>/ig, jindo.$Fn(function(m0,sQuoteType,sQuoteContents){
|
|
if (/<!--quote_txt-->((?:\s|.)*?)<!--\/quote_txt-->/ig.test(sQuoteContents)){
|
|
if(!this.htSE1toSE2Map[sQuoteType]){
|
|
return m0;
|
|
}
|
|
|
|
return '<blockquote class="se2_quote'+this.htSE1toSE2Map[sQuoteType]+'">'+RegExp.$1+'</blockquote>';
|
|
}else{
|
|
return '';
|
|
}
|
|
}, this).bind());
|
|
}, this).bind()]);
|
|
*/
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(){
|
|
this._assignHTMLElements();
|
|
|
|
this.oApp.registerBrowserEvent(this.elDropdownLayer, "click", "EVENT_SE2_BLOCKQUOTE_LAYER_CLICK", []);
|
|
this.oApp.delayedExec("SE2_ATTACH_HOVER_EVENTS", [this.aLI], 0);
|
|
},
|
|
|
|
$ON_MSG_APP_READY: function(){
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["quote", "click", "TOGGLE_BLOCKQUOTE_LAYER"]);
|
|
this.oApp.registerLazyMessage(["TOGGLE_BLOCKQUOTE_LAYER"], ["hp_SE2M_Quote$Lazy.js"]);
|
|
},
|
|
|
|
// [SMARTEDITORSUS-209] 인용구 내에 내용이 없을 때 Backspace 로 인용구가 삭제되도록 처리
|
|
$ON_EVENT_EDITING_AREA_KEYDOWN : function(weEvent) {
|
|
var oSelection,
|
|
elParentQuote;
|
|
|
|
if ('WYSIWYG' !== this.oApp.getEditingMode()){
|
|
return;
|
|
}
|
|
|
|
if(8 !== weEvent.key().keyCode){
|
|
return;
|
|
}
|
|
|
|
oSelection = this.oApp.getSelection();
|
|
oSelection.fixCommonAncestorContainer();
|
|
elParentQuote = this._findParentQuote(oSelection.commonAncestorContainer);
|
|
|
|
if(!elParentQuote){
|
|
return;
|
|
}
|
|
|
|
if(this._isBlankQuote(elParentQuote)){
|
|
weEvent.stop(jindo.$Event.CANCEL_DEFAULT);
|
|
|
|
oSelection.selectNode(elParentQuote);
|
|
oSelection.collapseToStart();
|
|
|
|
jindo.$Element(elParentQuote).leave();
|
|
|
|
oSelection.select();
|
|
}
|
|
},
|
|
|
|
// [SMARTEDITORSUS-215] Delete 로 인용구 뒤의 P 가 제거되지 않도록 처리
|
|
$ON_EVENT_EDITING_AREA_KEYUP : function(weEvent) {
|
|
var oSelection,
|
|
elParentQuote,
|
|
oP;
|
|
|
|
if ('WYSIWYG' !== this.oApp.getEditingMode()){
|
|
return;
|
|
}
|
|
|
|
if(46 !== weEvent.key().keyCode){
|
|
return;
|
|
}
|
|
|
|
oSelection = this.oApp.getSelection();
|
|
oSelection.fixCommonAncestorContainer();
|
|
elParentQuote = this._findParentQuote(oSelection.commonAncestorContainer);
|
|
|
|
if(!elParentQuote){
|
|
return false;
|
|
}
|
|
|
|
if(!elParentQuote.nextSibling){
|
|
weEvent.stop(jindo.$Event.CANCEL_DEFAULT);
|
|
|
|
oP = oSelection._document.createElement("P");
|
|
oP.innerHTML = " ";
|
|
|
|
jindo.$Element(elParentQuote).after(oP);
|
|
|
|
setTimeout(jindo.$Fn(function(oSelection){
|
|
var sBookmarkID = oSelection.placeStringBookmark();
|
|
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(sBookmarkID);
|
|
},this).bind(oSelection), 0);
|
|
}
|
|
},
|
|
|
|
_isBlankQuote : function(elParentQuote){
|
|
var elChild,
|
|
aChildNodes,
|
|
i, nLen,
|
|
bChrome = this.oApp.oNavigator.chrome,
|
|
bSafari = this.oApp.oNavigator.safari,
|
|
isBlankText = function(sText){
|
|
sText = sText.replace(/[\r\n]/ig, '').replace(unescape("%uFEFF"), '');
|
|
|
|
if(sText === ""){
|
|
return true;
|
|
}
|
|
|
|
if(sText === " " || sText === " "){ // [SMARTEDITORSUS-479]
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
isBlank = function(oNode){
|
|
if(oNode.nodeType === 3 && isBlankText(oNode.nodeValue)){
|
|
return true;
|
|
}
|
|
|
|
if((oNode.tagName === "P" || oNode.tagName === "SPAN") &&
|
|
(isBlankText(oNode.innerHTML) || oNode.innerHTML === "<br>")){
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
isBlankTable = function(oNode){
|
|
if((jindo.$$("tr", oNode)).length === 0){
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
if(isBlankText(elParentQuote.innerHTML) || elParentQuote.innerHTML === "<br>"){
|
|
return true;
|
|
}
|
|
|
|
if(bChrome || bSafari){ // [SMARTEDITORSUS-352], [SMARTEDITORSUS-502]
|
|
var aTable = jindo.$$("TABLE", elParentQuote),
|
|
nTable = aTable.length,
|
|
elTable;
|
|
|
|
for(i=0; i<nTable; i++){
|
|
elTable = aTable[i];
|
|
|
|
if(isBlankTable(elTable)){
|
|
jindo.$Element(elTable).leave();
|
|
}
|
|
}
|
|
}
|
|
|
|
aChildNodes = elParentQuote.childNodes;
|
|
|
|
for(i=0, nLen=aChildNodes.length; i<nLen; i++){
|
|
elChild = aChildNodes[i];
|
|
|
|
if(!isBlank(elChild)){
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
},
|
|
|
|
_findParentQuote : function(el){
|
|
return this._findAncestor(jindo.$Fn(function(elNode){
|
|
if(!elNode){return false;}
|
|
if(elNode.tagName !== "BLOCKQUOTE"){return false;}
|
|
if(!elNode.className){return false;}
|
|
|
|
var sClassName = elNode.className;
|
|
if(!this.htQuoteStyles_view[sClassName]){return false;}
|
|
|
|
return true;
|
|
}, this).bind(), el);
|
|
},
|
|
|
|
_findAncestor : function(fnCondition, elNode){
|
|
while(elNode && !fnCondition(elNode)){elNode = elNode.parentNode;}
|
|
|
|
return elNode;
|
|
}
|
|
});
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to table creation
|
|
* @name hp_SE_Table.js
|
|
*/
|
|
nhn.husky.SE2M_TableCreator = jindo.$Class({
|
|
name : "SE2M_TableCreator",
|
|
|
|
_sSETblClass : "__se_tbl",
|
|
|
|
nRows : 3,
|
|
nColumns : 4,
|
|
nBorderSize : 1,
|
|
sBorderColor : "#000000",
|
|
sBGColor: "#000000",
|
|
|
|
nBorderStyleIdx : 3,
|
|
nTableStyleIdx : 1,
|
|
|
|
nMinRows : 1,
|
|
nMaxRows : 20,
|
|
nMinColumns : 1,
|
|
nMaxColumns : 20,
|
|
nMinBorderWidth : 1,
|
|
nMaxBorderWidth : 10,
|
|
|
|
rxLastDigits : null,
|
|
sReEditGuideMsg_table : null,
|
|
|
|
// 테두리 스타일 목록
|
|
// 표 스타일 스타일 목록
|
|
oSelection : null,
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.sReEditGuideMsg_table = this.oApp.$MSG("SE2M_ReEditAction.reEditGuideMsg.table");
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["table", "click", "TOGGLE_TABLE_LAYER"]);
|
|
this.oApp.registerLazyMessage(["TOGGLE_TABLE_LAYER"], ["hp_SE2M_TableCreator$Lazy.js"]);
|
|
},
|
|
|
|
// [SMARTEDITORSUS-365] 테이블퀵에디터 > 속성 직접입력 > 테두리 스타일
|
|
// - 테두리 없음을 선택하는 경우 본문에 삽입하는 표에 가이드 라인을 표시해 줍니다. 보기 시에는 테두리가 보이지 않습니다.
|
|
$ON_REGISTER_CONVERTERS : function(){
|
|
this.oApp.exec("ADD_CONVERTER_DOM", ["IR_TO_DB", jindo.$Fn(this.irToDbDOM, this).bind()]);
|
|
this.oApp.exec("ADD_CONVERTER_DOM", ["DB_TO_IR", jindo.$Fn(this.dbToIrDOM, this).bind()]);
|
|
},
|
|
|
|
irToDbDOM : function(oTmpNode){
|
|
/**
|
|
* 저장을 위한 Table Tag 는 아래와 같이 변경됩니다.
|
|
* (1) <TABLE>
|
|
* <table border="1" cellpadding="0" cellspacing="0" style="border:1px dashed #c7c7c7; border-left:0; border-bottom:0;" attr_no_border_tbl="1" class="__se_tbl">
|
|
* --> <table border="0" cellpadding="1" cellspacing="0" attr_no_border_tbl="1" class="__se_tbl">
|
|
* (2) <TD>
|
|
* <td style="border:1px dashed #c7c7c7; border-top:0; border-right:0; background-color:#ffef00" width="245"><p> </p></td>
|
|
* --> <td style="background-color:#ffef00" width="245"> </td>
|
|
*/
|
|
var aNoBorderTable = [];
|
|
var aTables = jindo.$$('table[class=__se_tbl]', oTmpNode, {oneTimeOffCache:true});
|
|
|
|
// 테두리가 없음 속성의 table (임의로 추가한 attr_no_border_tbl 속성이 있는 table 을 찾음)
|
|
jindo.$A(aTables).forEach(function(oValue, nIdx, oArray) {
|
|
if(jindo.$Element(oValue).attr("attr_no_border_tbl")){
|
|
aNoBorderTable.push(oValue);
|
|
}
|
|
}, this);
|
|
|
|
if(aNoBorderTable.length < 1){
|
|
return;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-410] 글 저장 시, 테두리 없음 속성을 선택할 때 임의로 표시한 가이드 라인 property 만 style 에서 제거해 준다.
|
|
// <TABLE> 과 <TD> 의 속성값을 변경 및 제거
|
|
var aTDs = [], oTable;
|
|
for(var i = 0, nCount = aNoBorderTable.length; i < nCount; i++){
|
|
oTable = aNoBorderTable[i];
|
|
|
|
// <TABLE> 에서 border, cellpadding 속성값 변경, style property 제거
|
|
jindo.$Element(oTable).css({"border": "", "borderLeft": "", "borderBottom": ""});
|
|
jindo.$Element(oTable).attr({"border": 0, "cellpadding": 1});
|
|
|
|
// <TD> 에서는 background-color 를 제외한 style 을 모두 제거
|
|
aTDs = jindo.$$('tbody>tr>td', oTable);
|
|
jindo.$A(aTDs).forEach(function(oTD, nIdx, oTDArray) {
|
|
jindo.$Element(oTD).css({"border": "", "borderTop": "", "borderRight": ""});
|
|
});
|
|
}
|
|
},
|
|
|
|
dbToIrDOM : function(oTmpNode){
|
|
/**
|
|
* 수정을 위한 Table Tag 는 아래와 같이 변경됩니다.
|
|
* (1) <TABLE>
|
|
* <table border="0" cellpadding="1" cellspacing="0" attr_no_border_tbl="1" class="__se_tbl">
|
|
* --> <table border="1" cellpadding="0" cellspacing="0" style="border:1px dashed #c7c7c7; border-left:0; border-bottom:0;" attr_no_border_tbl="1" class="__se_tbl">
|
|
* (2) <TD>
|
|
* <td style="background-color:#ffef00" width="245"> </td>
|
|
* --> <td style="border:1px dashed #c7c7c7; border-top:0; border-right:0; background-color:#ffef00" width="245"><p> </p></td>
|
|
*/
|
|
var aNoBorderTable = [];
|
|
var aTables = jindo.$$('table[class=__se_tbl]', oTmpNode, {oneTimeOffCache:true});
|
|
|
|
// 테두리가 없음 속성의 table (임의로 추가한 attr_no_border_tbl 속성이 있는 table 을 찾음)
|
|
jindo.$A(aTables).forEach(function(oValue, nIdx, oArray) {
|
|
if(jindo.$Element(oValue).attr("attr_no_border_tbl")){
|
|
aNoBorderTable.push(oValue);
|
|
}
|
|
}, this);
|
|
|
|
if(aNoBorderTable.length < 1){
|
|
return;
|
|
}
|
|
|
|
// <TABLE> 과 <TD> 의 속성값을 변경/추가
|
|
var aTDs = [], oTable;
|
|
for(var i = 0, nCount = aNoBorderTable.length; i < nCount; i++){
|
|
oTable = aNoBorderTable[i];
|
|
|
|
// <TABLE> 에서 border, cellpadding 속성값 변경/ style 속성 추가
|
|
jindo.$Element(oTable).css({"border": "1px dashed #c7c7c7", "borderLeft": 0, "borderBottom": 0});
|
|
jindo.$Element(oTable).attr({"border": 1, "cellpadding": 0});
|
|
|
|
// <TD> 에서 style 속성값 추가
|
|
aTDs = jindo.$$('tbody>tr>td', oTable);
|
|
jindo.$A(aTDs).forEach(function(oTD, nIdx, oTDArray) {
|
|
jindo.$Element(oTD).css({"border": "1px dashed #c7c7c7", "borderTop": 0, "borderRight": 0});
|
|
});
|
|
}
|
|
}
|
|
});
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to changing the font style in the table.
|
|
* @requires SE2M_TableEditor.js
|
|
* @name SE2M_TableBlockManager
|
|
*/
|
|
nhn.husky.SE2M_TableBlockStyler = jindo.$Class({
|
|
name : "SE2M_TableBlockStyler",
|
|
nSelectedTD : 0,
|
|
htSelectedTD : {},
|
|
aTdRange : [],
|
|
|
|
$init : function(){ },
|
|
|
|
$LOCAL_BEFORE_ALL : function(){
|
|
return (this.oApp.getEditingMode() == "WYSIWYG");
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oDocument = this.oApp.getWYSIWYGDocument();
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_MOUSEUP : function(wevE){
|
|
if(this.oApp.getEditingMode() != "WYSIWYG") return;
|
|
this.setTdBlock();
|
|
},
|
|
|
|
/**
|
|
* selected Area가 td block인지 체크하는 함수.
|
|
*/
|
|
$ON_IS_SELECTED_TD_BLOCK : function(sAttr,oReturn) {
|
|
if( this.nSelectedTD > 0){
|
|
oReturn[sAttr] = true;
|
|
return oReturn[sAttr];
|
|
}else{
|
|
oReturn[sAttr] = false;
|
|
return oReturn[sAttr];
|
|
}
|
|
},
|
|
|
|
/**
|
|
*
|
|
*/
|
|
$ON_GET_SELECTED_TD_BLOCK : function(sAttr,oReturn){
|
|
//use : this.oApp.exec("GET_SELECTED_TD_BLOCK",['aCells',this.htSelectedTD]);
|
|
oReturn[sAttr] = this.htSelectedTD.aTdCells;
|
|
},
|
|
|
|
setTdBlock : function() {
|
|
this.oApp.exec("GET_SELECTED_CELLS",['aTdCells',this.htSelectedTD]); //tableEditor로 부터 얻어온다.
|
|
var aNodes = this.htSelectedTD.aTdCells;
|
|
if(aNodes){
|
|
this.nSelectedTD = aNodes.length;
|
|
}
|
|
},
|
|
|
|
$ON_DELETE_BLOCK_CONTENTS : function(){
|
|
var self = this, welParent, oBlockNode, oChildNode;
|
|
|
|
this.setTdBlock();
|
|
for (var j = 0; j < this.nSelectedTD ; j++){
|
|
jindo.$Element(this.htSelectedTD.aTdCells[j]).child( function(elChild){
|
|
|
|
welParent = jindo.$Element(elChild._element.parentNode);
|
|
welParent.remove(elChild);
|
|
|
|
oBlockNode = self.oDocument.createElement('P');
|
|
|
|
if (jindo.$Agent().navigator().firefox) {
|
|
oChildNode = self.oDocument.createElement('BR');
|
|
} else {
|
|
oChildNode = self.oDocument.createTextNode('\u00A0');
|
|
}
|
|
|
|
oBlockNode.appendChild(oChildNode);
|
|
welParent.append(oBlockNode);
|
|
}, 1);
|
|
}
|
|
}
|
|
|
|
});
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin with test handlers
|
|
* @name hp_SE2M_StyleRemover.js
|
|
*/
|
|
nhn.husky.SE2M_StyleRemover = jindo.$Class({
|
|
name: "SE2M_StyleRemover",
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["styleRemover", "click", "CHOOSE_REMOVE_STYLE", []]);
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(){
|
|
// The plugin may be used in view and when it is used there, EditingAreaManager plugin is not loaded.
|
|
// So, get the document from the selection instead of EditingAreaManager.
|
|
this.oHuskyRange = this.oApp.getEmptySelection();
|
|
this._document = this.oHuskyRange._document;
|
|
},
|
|
|
|
$ON_CHOOSE_REMOVE_STYLE : function(oSelection){
|
|
var bSelectedBlock = false;
|
|
var htSelectedTDs = {};
|
|
this.oApp.exec("IS_SELECTED_TD_BLOCK",['bIsSelectedTd',htSelectedTDs]);
|
|
bSelectedBlock = htSelectedTDs.bIsSelectedTd;
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["REMOVE STYLE", {bMustBlockElement:true}]);
|
|
|
|
if( bSelectedBlock ){
|
|
this.oApp.exec("REMOVE_STYLE_IN_BLOCK", []);
|
|
}else{
|
|
this.oApp.exec("REMOVE_STYLE", []);
|
|
}
|
|
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["REMOVE STYLE", {bMustBlockElement:true}]);
|
|
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['noeffect']);
|
|
},
|
|
|
|
$ON_REMOVE_STYLE_IN_BLOCK : function(oSelection){
|
|
var htSelectedTDs = {};
|
|
this.oSelection = this.oApp.getSelection();
|
|
this.oApp.exec("GET_SELECTED_TD_BLOCK",['aTdCells',htSelectedTDs]);
|
|
var aNodes = htSelectedTDs.aTdCells;
|
|
|
|
for( var j = 0; j < aNodes.length ; j++){
|
|
this.oSelection.selectNodeContents(aNodes[j]);
|
|
this.oSelection.select();
|
|
this.oApp.exec("REMOVE_STYLE", []);
|
|
}
|
|
},
|
|
|
|
$ON_REMOVE_STYLE : function(oSelection){
|
|
if(!oSelection || !oSelection.commonAncestorContainer){
|
|
oSelection = this.oApp.getSelection();
|
|
}
|
|
|
|
if(oSelection.collapsed){return;}
|
|
|
|
oSelection.expandBothEnds();
|
|
|
|
var sBookmarkID = oSelection.placeStringBookmark();
|
|
var aNodes = oSelection.getNodes(true);
|
|
|
|
this._removeStyle(aNodes);
|
|
oSelection.moveToBookmark(sBookmarkID);
|
|
|
|
var aNodes = oSelection.getNodes(true);
|
|
for(var i=0; i<aNodes.length; i++){
|
|
var oNode = aNodes[i];
|
|
|
|
if(oNode.style && oNode.tagName != "BR" && oNode.tagName != "TD" && oNode.tagName != "TR" && oNode.tagName != "TBODY" && oNode.tagName != "TABLE"){
|
|
oNode.removeAttribute("align");
|
|
oNode.removeAttribute("style");
|
|
if((jindo.$Element(oNode).css("display") == "inline" && oNode.tagName != "IMG" && oNode.tagName != "IFRAME") && (!oNode.firstChild || oSelection._isBlankTextNode(oNode.firstChild))){
|
|
oNode.parentNode.removeChild(oNode);
|
|
}
|
|
}
|
|
}
|
|
|
|
oSelection.moveToBookmark(sBookmarkID);
|
|
|
|
// [SMARTEDITORSUS-1750] 스타일제거를 위해 selection을 확장(oSelection.expandBothEnds)하면 TR까지 확장되는데 IE10에서만 execCommand 가 제대로 동작하지 않는 문제가 발생하기 때문에 확장전 selection으로 복원하도록 수정
|
|
// [SMARTEDITORSUS-1893] 테이블밖에서는 마지막라인이 풀리는 이슈가 발생하여 commonAncestorContainer가 TBODY 인 경우에만 selection을 복원하도록 제한
|
|
if(oSelection.commonAncestorContainer.tagName === "TBODY"){
|
|
oSelection = this.oApp.getSelection();
|
|
}
|
|
oSelection.select();
|
|
|
|
// use a custom removeStringBookmark here as the string bookmark could've been cloned and there are some additional cases that need to be considered
|
|
|
|
// remove start marker
|
|
var oMarker = this._document.getElementById(oSelection.HUSKY_BOOMARK_START_ID_PREFIX+sBookmarkID);
|
|
while(oMarker){
|
|
oParent = nhn.DOMFix.parentNode(oMarker);
|
|
oParent.removeChild(oMarker);
|
|
while(oParent && (!oParent.firstChild || (!oParent.firstChild.nextSibling && oSelection._isBlankTextNode(oParent.firstChild)))){
|
|
var oNextParent = oParent.parentNode;
|
|
oParent.parentNode.removeChild(oParent);
|
|
oParent = oNextParent;
|
|
}
|
|
oMarker = this._document.getElementById(oSelection.HUSKY_BOOMARK_START_ID_PREFIX+sBookmarkID);
|
|
}
|
|
|
|
// remove end marker
|
|
var oMarker = this._document.getElementById(oSelection.HUSKY_BOOMARK_END_ID_PREFIX+sBookmarkID);
|
|
while(oMarker){
|
|
oParent = nhn.DOMFix.parentNode(oMarker);
|
|
oParent.removeChild(oMarker);
|
|
while(oParent && (!oParent.firstChild || (!oParent.firstChild.nextSibling && oSelection._isBlankTextNode(oParent.firstChild)))){
|
|
var oNextParent = oParent.parentNode;
|
|
oParent.parentNode.removeChild(oParent);
|
|
oParent = oNextParent;
|
|
}
|
|
oMarker = this._document.getElementById(oSelection.HUSKY_BOOMARK_END_ID_PREFIX+sBookmarkID);
|
|
}
|
|
|
|
this.oApp.exec("CHECK_STYLE_CHANGE");
|
|
},
|
|
|
|
$ON_REMOVE_STYLE2 : function(aNodes){
|
|
this._removeStyle(aNodes);
|
|
},
|
|
|
|
$ON_REMOVE_STYLE_AND_PASTE_HTML : function(sHtml, bNoUndo){
|
|
var htBrowser,
|
|
elDivHolder,
|
|
elFirstTD,
|
|
aNodesInSelection,
|
|
oSelection;
|
|
|
|
htBrowser = jindo.$Agent().navigator();
|
|
|
|
if(!sHtml) {return false;}
|
|
if(this.oApp.getEditingMode() != "WYSIWYG"){
|
|
this.oApp.exec("CHANGE_EDITING_MODE", ["WYSIWYG"]);
|
|
}
|
|
|
|
if(!bNoUndo){
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["REMOVE STYLE AND PASTE HTML"]);
|
|
}
|
|
|
|
oSelection = this.oApp.getSelection();
|
|
oSelection.deleteContents(); // remove select node - for dummy image, reedit object
|
|
|
|
// If the table were inserted within a styled(strikethough & etc) paragraph, the table may inherit the style in IE.
|
|
elDivHolder = this.oApp.getWYSIWYGDocument().createElement("DIV");
|
|
oSelection.insertNode(elDivHolder);
|
|
|
|
if (!!htBrowser.webkit) {
|
|
elDivHolder.innerHTML = " "; // for browser bug! - summary reiteration
|
|
}
|
|
|
|
oSelection.selectNode(elDivHolder);
|
|
this.oApp.exec("REMOVE_STYLE", [oSelection]);
|
|
|
|
//[SMARTEDITORSUS-181][IE9] 표나 요약글 등의 테이블에서 > 테이블 외부로 커서 이동 불가
|
|
if( htBrowser.ie ){
|
|
sHtml += "<p> </p>";
|
|
}else if(htBrowser.firefox){
|
|
//[SMARTEDITORSUS-477][개별블로그](파폭특정)포스트쓰기>요약글을 삽입 후 요약글 아래 임의의 본문영역에 마우스 클릭 시 커서가 요약안에 노출됩니다.
|
|
// 본문에 table만 있는 경우, 커서가 밖으로 못나오는 현상이 있음.FF버그임.
|
|
sHtml += "<p><br></p>";
|
|
}
|
|
|
|
oSelection.selectNode(elDivHolder);
|
|
oSelection.pasteHTML(sHtml);
|
|
|
|
//Table인경우, 커서를 테이블 첫 TD에 넣기 위한 작업.
|
|
aNodesInSelection = oSelection.getNodes() || [];
|
|
for(var i = 0; i < aNodesInSelection.length ; i++){
|
|
if(!!aNodesInSelection[i].tagName && aNodesInSelection[i].tagName.toLowerCase() == 'td'){
|
|
elFirstTD = aNodesInSelection[i];
|
|
oSelection.selectNodeContents(elFirstTD.firstChild || elFirstTD);
|
|
oSelection.collapseToStart();
|
|
oSelection.select();
|
|
break;
|
|
}
|
|
}
|
|
|
|
oSelection.collapseToEnd(); //파란색 커버 제거.
|
|
oSelection.select();
|
|
this.oApp.exec("FOCUS");
|
|
if (!elDivHolder) {// 임시 div 삭제.
|
|
elDivHolder.parentNode.removeChild(elDivHolder);
|
|
}
|
|
|
|
if(!bNoUndo){
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["REMOVE STYLE AND PASTE HTML"]);
|
|
}
|
|
},
|
|
|
|
_removeStyle : function(aNodes){
|
|
var arNodes = jindo.$A(aNodes);
|
|
for(var i=0; i<aNodes.length; i++){
|
|
var oNode = aNodes[i];
|
|
|
|
// oNode had been removed from the document already
|
|
if(!oNode || !oNode.parentNode || !oNode.parentNode.tagName){continue;}
|
|
|
|
var bDontSplit = false;
|
|
// If oNode is direct child of a block level node, don't do anything. (should not move up the hierarchy anymore)
|
|
if(jindo.$Element(oNode.parentNode).css("display") != "inline"){
|
|
continue;
|
|
}
|
|
|
|
var parent = oNode.parentNode;
|
|
|
|
// do not proceed if oNode is not completely selected
|
|
if(oNode.firstChild){
|
|
if(arNodes.indexOf(this.oHuskyRange._getVeryLastRealChild(oNode)) == -1){continue;}
|
|
if(arNodes.indexOf(this.oHuskyRange._getVeryFirstRealChild(oNode)) == -1){continue;}
|
|
}
|
|
|
|
// Case 1: oNode is the right most node
|
|
//
|
|
// If oNode were C(right most node) from
|
|
// H
|
|
// |
|
|
// P
|
|
// / | \
|
|
// A B C
|
|
//
|
|
// and B and C were selected, bring up all the (selected) left siblings to the right of the parent and and make it
|
|
// H
|
|
// / | \
|
|
// P B C
|
|
// |
|
|
// A
|
|
// ===========================================================
|
|
// If A, B and C were selected from
|
|
// H
|
|
// |
|
|
// P
|
|
// / | \
|
|
// A B C
|
|
//
|
|
// append them to the right of the parent and make it
|
|
// H
|
|
// / | | \
|
|
// P A B C
|
|
//
|
|
// and then remove P as it's got no child and make it
|
|
// H
|
|
// / | \
|
|
// A B C
|
|
if(!oNode.nextSibling){
|
|
i--;
|
|
var tmp = oNode;
|
|
// bring up left siblings
|
|
while(tmp){
|
|
var prevNode = tmp.previousSibling;
|
|
parent.parentNode.insertBefore(tmp, parent.nextSibling);
|
|
if(!prevNode){break;}
|
|
if(arNodes.indexOf(this._getVeryFirst(prevNode)) == -1){break;}
|
|
tmp = prevNode;
|
|
}
|
|
|
|
// remove the parent if it's got no child now
|
|
if(parent.childNodes.length === 0){parent.parentNode.removeChild(parent);}
|
|
|
|
continue;
|
|
}
|
|
|
|
// Case 2: oNode's got a right sibling that is included in the selection
|
|
//
|
|
// if the next sibling is included in the selection, stop current iteration
|
|
// -> current node will be handled in the next iteration
|
|
if(arNodes.indexOf(this._getVeryLast(oNode.nextSibling)) != -1){continue;}
|
|
|
|
// Since the case
|
|
// 1. oNode is the right most node
|
|
// 2. oNode's got a right sibling that is included in the selection
|
|
// were all taken care of above, so from here we just need take care of the case when oNode is NOT the right most node and oNode's right sibling is NOT included in the selection
|
|
|
|
// Case 3: the rest
|
|
// When all of the left siblings were selected, take all the left siblings and current node and append them to the left of the parent node.
|
|
// H
|
|
// |
|
|
// P
|
|
// / | | \
|
|
// A B C D
|
|
// -> if A, B and C were selected, then make it
|
|
// H
|
|
// / | | \
|
|
// A B C P
|
|
// |
|
|
// D
|
|
i--;
|
|
// bring up selected prev siblings
|
|
if(arNodes.indexOf(this._getVeryFirst(oNode.parentNode)) != -1){
|
|
// move
|
|
var tmp = oNode;
|
|
var lastInserted = parent;
|
|
while(tmp){
|
|
var prevNode = tmp.previousSibling;
|
|
parent.parentNode.insertBefore(tmp, lastInserted);
|
|
lastInserted = tmp;
|
|
|
|
if(!prevNode){break;}
|
|
tmp = prevNode;
|
|
}
|
|
if(parent.childNodes.length === 0){parent.parentNode.removeChild(parent);}
|
|
// When NOT all of the left siblings were selected, split the parent node and insert the selected nodes in between.
|
|
// H
|
|
// |
|
|
// P
|
|
// / | | \
|
|
// A B C D
|
|
// -> if B and C were selected, then make it
|
|
// H
|
|
// / | | \
|
|
// P B C P
|
|
// | |
|
|
// A D
|
|
}else{
|
|
//split
|
|
if(bDontSplit){
|
|
i++;
|
|
continue;
|
|
}
|
|
|
|
var oContainer = this._document.createElement("SPAN");
|
|
var tmp = oNode;
|
|
parent.insertBefore(oContainer, tmp.nextSibling);
|
|
while(tmp){
|
|
var prevNode = tmp.previousSibling;
|
|
oContainer.insertBefore(tmp, oContainer.firstChild);
|
|
|
|
if(!prevNode){break;}
|
|
if(arNodes.indexOf(this._getVeryFirst(prevNode)) == -1){break;}
|
|
tmp = prevNode;
|
|
}
|
|
|
|
this._splitAndAppendAtTop(oContainer);
|
|
while(oContainer.firstChild){
|
|
oContainer.parentNode.insertBefore(oContainer.firstChild, oContainer);
|
|
}
|
|
oContainer.parentNode.removeChild(oContainer);
|
|
}
|
|
}
|
|
},
|
|
|
|
_splitAndAppendAtTop : function(oSpliter){
|
|
var targetNode = oSpliter;
|
|
var oTmp = targetNode;
|
|
var oCopy = oTmp;
|
|
|
|
while(jindo.$Element(oTmp.parentNode).css("display") == "inline"){
|
|
var oNode = oTmp.parentNode.cloneNode(false);
|
|
|
|
while(oTmp.nextSibling){
|
|
oNode.appendChild(oTmp.nextSibling);
|
|
}
|
|
|
|
oTmp = oTmp.parentNode;
|
|
|
|
oNode.insertBefore(oCopy, oNode.firstChild);
|
|
oCopy = oNode;
|
|
}
|
|
|
|
oTop = oTmp.parentNode;
|
|
oTop.insertBefore(targetNode, oTmp.nextSibling);
|
|
oTop.insertBefore(oCopy, targetNode.nextSibling);
|
|
},
|
|
|
|
_getVeryFirst : function(oNode){
|
|
if(!oNode){return null;}
|
|
|
|
if(oNode.firstChild){
|
|
return this.oHuskyRange._getVeryFirstRealChild(oNode);
|
|
}else{
|
|
return oNode;
|
|
}
|
|
},
|
|
|
|
_getVeryLast : function(oNode){
|
|
if(!oNode){return null;}
|
|
|
|
if(oNode.lastChild){
|
|
return this.oHuskyRange._getVeryLastRealChild(oNode);
|
|
}else{
|
|
return oNode;
|
|
}
|
|
}
|
|
});
|
|
//}
|
|
|
|
nhn.husky.SE2M_TableEditor = jindo.$Class({
|
|
name : "SE2M_TableEditor",
|
|
|
|
_sSETblClass : "__se_tbl",
|
|
_sSEReviewTblClass : "__se_tbl_review",
|
|
|
|
STATUS : {
|
|
S_0 : 1, // neither cell selection nor cell resizing is active
|
|
MOUSEDOWN_CELL : 2, // mouse down on a table cell
|
|
CELL_SELECTING : 3, // cell selection is in progress
|
|
CELL_SELECTED : 4, // cell selection was (completely) made
|
|
MOUSEOVER_BORDER : 5, // mouse is over a table/cell border and the cell resizing grip is shown
|
|
MOUSEDOWN_BORDER : 6 // mouse down on the cell resizing grip (cell resizing is in progress)
|
|
},
|
|
|
|
CELL_SELECTION_CLASS : "se2_te_selection",
|
|
|
|
MIN_CELL_WIDTH : 5,
|
|
MIN_CELL_HEIGHT : 5,
|
|
|
|
TMP_BGC_ATTR : "_se2_tmp_te_bgc",
|
|
TMP_BGIMG_ATTR : "_se2_tmp_te_bg_img",
|
|
ATTR_TBL_TEMPLATE : "_se2_tbl_template",
|
|
|
|
nStatus : 1,
|
|
nMouseEventsStatus : 0,
|
|
|
|
aSelectedCells : [],
|
|
|
|
$ON_REGISTER_CONVERTERS : function(){
|
|
// remove the cell selection class
|
|
this.oApp.exec("ADD_CONVERTER_DOM", ["WYSIWYG_TO_IR", jindo.$Fn(function(elTmpNode){
|
|
if(this.aSelectedCells.length < 1){
|
|
//return sContents;
|
|
return;
|
|
}
|
|
|
|
var aCells;
|
|
var aCellType = ["TD", "TH"];
|
|
|
|
for(var n = 0; n < aCellType.length; n++){
|
|
aCells = elTmpNode.getElementsByTagName(aCellType[n]);
|
|
for(var i = 0, nLen = aCells.length; i < nLen; i++){
|
|
if(aCells[i].className){
|
|
aCells[i].className = aCells[i].className.replace(this.CELL_SELECTION_CLASS, "");
|
|
if(aCells[i].getAttribute(this.TMP_BGC_ATTR)){
|
|
aCells[i].style.backgroundColor = aCells[i].getAttribute(this.TMP_BGC_ATTR);
|
|
aCells[i].removeAttribute(this.TMP_BGC_ATTR);
|
|
}else if(aCells[i].getAttribute(this.TMP_BGIMG_ATTR)){
|
|
jindo.$Element(this.aCells[i]).css("backgroundImage",aCells[i].getAttribute(this.TMP_BGIMG_ATTR));
|
|
aCells[i].removeAttribute(this.TMP_BGIMG_ATTR);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// this.wfnMouseDown.attach(this.elResizeCover, "mousedown");
|
|
|
|
// return elTmpNode.innerHTML;
|
|
// var rxSelectionColor = new RegExp("<(TH|TD)[^>]*)("+this.TMP_BGC_ATTR+"=[^> ]*)([^>]*>)", "gi");
|
|
}, this).bind()]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.registerLazyMessage(["EVENT_EDITING_AREA_MOUSEMOVE","STYLE_TABLE"], ["hp_SE2M_TableEditor$Lazy.js","SE2M_TableTemplate.js"]);
|
|
}
|
|
});
|
|
/**
|
|
* @name SE2M_QuickEditor_Common
|
|
* @class
|
|
* @description Quick Editor Common function Class
|
|
* @author NHN AjaxUI Lab - mixed
|
|
* @version 1.0
|
|
* @since 2009.09.29
|
|
* */
|
|
nhn.husky.SE2M_QuickEditor_Common = jindo.$Class({
|
|
/**
|
|
* class 이름
|
|
* @type {String}
|
|
*/
|
|
name : "SE2M_QuickEditor_Common",
|
|
/**
|
|
* 환경 정보.
|
|
* @type {Object}
|
|
*/
|
|
_environmentData : "",
|
|
/**
|
|
* 현재 타입 (table|img)
|
|
* @type {String}
|
|
*/
|
|
_currentType :"",
|
|
/**
|
|
* 이벤트가 레이어 안에서 호출되었는지 알기 위한 변수
|
|
* @type {Boolean}
|
|
*/
|
|
_in_event : false,
|
|
/**
|
|
* Ajax처리를 하지 않음
|
|
* @type {Boolean}
|
|
*/
|
|
_bUseConfig : false,
|
|
|
|
/**
|
|
* 공통 서버에서 개인 설정 받아오는 AjaxUrl
|
|
* @See SE2M_Configuration.js
|
|
*/
|
|
_sBaseAjaxUrl : "",
|
|
_sAddTextAjaxUrl : "",
|
|
|
|
/**
|
|
* 초기 인스턴스 생성 실행되는 함수.
|
|
*/
|
|
$init : function() {
|
|
this.waHotkeys = new jindo.$A([]);
|
|
this.waHotkeyLayers = new jindo.$A([]);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function() {
|
|
var htConfiguration = nhn.husky.SE2M_Configuration.QuickEditor;
|
|
|
|
if(htConfiguration){
|
|
this._bUseConfig = (!!htConfiguration.common && typeof htConfiguration.common.bUseConfig !== "undefined") ? htConfiguration.common.bUseConfig : true;
|
|
}
|
|
|
|
if(!this._bUseConfig){
|
|
this.setData("{table:'full',img:'full',review:'full'}");
|
|
} else {
|
|
this._sBaseAjaxUrl = htConfiguration.common.sBaseAjaxUrl;
|
|
this._sAddTextAjaxUrl = htConfiguration.common.sAddTextAjaxUrl;
|
|
|
|
this.getData();
|
|
}
|
|
this.oApp.registerLazyMessage(["OPEN_QE_LAYER"], ["hp_SE2M_QuickEditor_Common$Lazy.js"]);
|
|
},
|
|
|
|
//삭제 시에 qe layer close
|
|
$ON_EVENT_EDITING_AREA_KEYDOWN : function(oEvent){
|
|
var oKeyInfo = oEvent.key();
|
|
//Backspace : 8, Delete :46
|
|
if (oKeyInfo.keyCode == 8 || oKeyInfo.keyCode == 46 ) {
|
|
// [SMARTEDITORSUS-1213][IE9, 10] 사진 삭제 후 zindex 1000인 div가 잔존하는데, 그 위로 썸네일 drag를 시도하다 보니 drop이 불가능.
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if(htBrowser.ie && htBrowser.nativeVersion > 8){
|
|
var elFirstChild = jindo.$$.getSingle("DIV.husky_seditor_editing_area_container").childNodes[0];
|
|
if((elFirstChild.tagName == "DIV") && (elFirstChild.style.zIndex == 1000)){
|
|
elFirstChild.parentNode.removeChild(elFirstChild);
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1213]
|
|
this.oApp.exec("CLOSE_QE_LAYER", [oEvent]);
|
|
}
|
|
},
|
|
|
|
getData : function() {
|
|
var self = this;
|
|
jindo.$Ajax(self._sBaseAjaxUrl, {
|
|
type : "jsonp",
|
|
timeout : 1,
|
|
onload: function(rp) {
|
|
var result = rp.json().result;
|
|
// [SMARTEDITORSUS-1028][SMARTEDITORSUS-1517] QuickEditor 설정 API 개선
|
|
//if (!!result && !!result.length) {
|
|
if (!!result && !!result.text_data) {
|
|
//self.setData(result[result.length - 1]);
|
|
self.setData(result.text_data);
|
|
} else {
|
|
self.setData("{table:'full',img:'full',review:'full'}");
|
|
}
|
|
// --[SMARTEDITORSUS-1028][SMARTEDITORSUS-1517]
|
|
},
|
|
|
|
onerror : function() {
|
|
self.setData("{table:'full',img:'full',review:'full'}");
|
|
},
|
|
|
|
ontimeout : function() {
|
|
self.setData("{table:'full',img:'full',review:'full'}");
|
|
}
|
|
}).request({ text_key : "qeditor_fold" });
|
|
},
|
|
|
|
setData : function(sResult){
|
|
var oResult = {
|
|
table : "full",
|
|
img : "full",
|
|
review : "full"
|
|
};
|
|
|
|
if(sResult){
|
|
oResult = eval("("+sResult+")");
|
|
}
|
|
|
|
this._environmentData = {
|
|
table : {
|
|
isOpen : false,
|
|
type : oResult["table"],//full,fold,
|
|
isFixed : false,
|
|
position : []
|
|
},
|
|
img : {
|
|
isOpen : false,
|
|
type : oResult["img"],//full,fold
|
|
isFixed : false
|
|
},
|
|
review : {
|
|
isOpen : false,
|
|
type : oResult["review"],//full,fold
|
|
isFixed : false,
|
|
position : []
|
|
}
|
|
};
|
|
|
|
|
|
this.waTableTagNames =jindo.$A(["table","tbody","td","tfoot","th","thead","tr"]);
|
|
},
|
|
|
|
/**
|
|
* 위지윅 영역에 단축키가 등록될 때,
|
|
* tab 과 shift+tab (들여쓰기 / 내어쓰기 ) 를 제외한 단축키 리스트를 저장한다.
|
|
*/
|
|
$ON_REGISTER_HOTKEY : function(sHotkey, sCMD, aArgs){
|
|
if(sHotkey != "tab" && sHotkey != "shift+tab"){
|
|
this.waHotkeys.push([sHotkey, sCMD, aArgs]);
|
|
}
|
|
}
|
|
});
|
|
/**
|
|
* @classDescription shortcut
|
|
* @author AjaxUI Lab - mixed
|
|
*/
|
|
|
|
function Shortcut(sKey,sId){
|
|
var sKey = sKey.replace(/\s+/g,"");
|
|
var store = Shortcut.Store;
|
|
var action = Shortcut.Action;
|
|
if(typeof sId === "undefined"&&sKey.constructor == String){
|
|
store.set("document",sKey,document);
|
|
return action.init(store.get("document"),sKey);
|
|
}else if(sId.constructor == String&&sKey.constructor == String){
|
|
store.set(sId,sKey,jindo.$(sId));
|
|
return action.init(store.get(sId),sKey);
|
|
}else if(sId.constructor != String&&sKey.constructor == String){
|
|
var fakeId = "nonID"+new Date().getTime();
|
|
fakeId = Shortcut.Store.searchId(fakeId,sId);
|
|
store.set(fakeId,sKey,sId);
|
|
return action.init(store.get(fakeId),sKey);
|
|
}
|
|
alert(sId+"는 반드시 string이거나 없어야 됩니다.");
|
|
};
|
|
|
|
|
|
Shortcut.Store = {
|
|
anthorKeyHash:{},
|
|
datas:{},
|
|
currentId:"",
|
|
currentKey:"",
|
|
searchId:function(sId,oElement){
|
|
// [SMARTEDITORSUS-2103]
|
|
var isElementInData = false;
|
|
jindo.$H(this.datas).forEach(function(oValue,sKey){
|
|
if(oElement == oValue.element){
|
|
sId = sKey;
|
|
isElementInData = true;
|
|
jindo.$H.Break();
|
|
}
|
|
});
|
|
|
|
/*
|
|
* element context는 서로 다르지만
|
|
* 이미 추가된 Shortcut.Data 객체와 timestamp가 중복되는 경우,
|
|
*
|
|
* 원래 동작해야 할 element context가 아닌,
|
|
* 이미 추가된 element context에서 잘못 동작할 수 있다.
|
|
*
|
|
* 이 경우에는 timestamp 기반 id를 새롭개 생성해서
|
|
* 원래 동작해야 할 element context에서 잘 동작할 수 있도록 한다.
|
|
*/
|
|
if(!isElementInData && (sId in this.datas)){
|
|
var newFakeId = sId;
|
|
while(newFakeId in this.datas){
|
|
newFakeId = "nonID"+new Date().getTime();
|
|
}
|
|
return newFakeId;
|
|
}
|
|
|
|
return sId;
|
|
// --[SMARTEDITORSUS-2103]
|
|
},
|
|
set:function(sId,sKey,oElement){
|
|
this.currentId = sId;
|
|
this.currentKey = sKey;
|
|
var idData = this.get(sId);
|
|
this.datas[sId] = idData?idData.createKey(sKey):new Shortcut.Data(sId,sKey,oElement);
|
|
},
|
|
get:function(sId,sKey){
|
|
if(sKey){
|
|
return this.datas[sId].keys[sKey];
|
|
}else{
|
|
return this.datas[sId];
|
|
}
|
|
},
|
|
reset:function(sId){
|
|
var data = this.datas[sId];
|
|
Shortcut.Helper.bind(data.func,data.element,"detach");
|
|
|
|
delete this.datas[sId];
|
|
},
|
|
allReset: function(){
|
|
jindo.$H(this.datas).forEach(jindo.$Fn(function(value,key) {
|
|
this.reset(key);
|
|
},this).bind());
|
|
}
|
|
};
|
|
|
|
Shortcut.Data = jindo.$Class({
|
|
$init:function(sId,sKey,oElement){
|
|
this.id = sId;
|
|
this.element = oElement;
|
|
this.func = jindo.$Fn(this.fire,this).bind();
|
|
|
|
Shortcut.Helper.bind(this.func,oElement,"attach");
|
|
this.keys = {};
|
|
this.keyStemp = {};
|
|
this.createKey(sKey);
|
|
},
|
|
createKey:function(sKey){
|
|
this.keyStemp[Shortcut.Helper.keyInterpretor(sKey)] = sKey;
|
|
this.keys[sKey] = {};
|
|
var data = this.keys[sKey];
|
|
data.key = sKey;
|
|
data.events = [];
|
|
data.commonExceptions = [];
|
|
// data.keyAnalysis = Shortcut.Helper.keyInterpretor(sKey);
|
|
data.stopDefalutBehavior = true;
|
|
|
|
return this;
|
|
},
|
|
getKeyStamp : function(eEvent){
|
|
|
|
|
|
var sKey = eEvent.keyCode || eEvent.charCode;
|
|
var returnVal = "";
|
|
|
|
returnVal += eEvent.altKey?"1":"0";
|
|
returnVal += eEvent.ctrlKey?"1":"0";
|
|
returnVal += eEvent.metaKey?"1":"0";
|
|
returnVal += eEvent.shiftKey?"1":"0";
|
|
returnVal += sKey;
|
|
return returnVal;
|
|
},
|
|
fire:function(eEvent){
|
|
eEvent = eEvent||window.eEvent;
|
|
|
|
var oMatchKeyData = this.keyStemp[this.getKeyStamp(eEvent)];
|
|
|
|
if(oMatchKeyData){
|
|
this.excute(new jindo.$Event(eEvent),oMatchKeyData);
|
|
}
|
|
|
|
},
|
|
excute:function(weEvent,sRawKey){
|
|
var isExcute = true;
|
|
var staticFun = Shortcut.Helper;
|
|
var data = this.keys[sRawKey];
|
|
|
|
if(staticFun.notCommonException(weEvent,data.commonExceptions)){
|
|
jindo.$A(data.events).forEach(function(v){
|
|
if(data.stopDefalutBehavior){
|
|
var leng = v.exceptions.length;
|
|
if(leng){
|
|
for(var i=0;i<leng;i++){
|
|
if(!v.exception[i](weEvent)){
|
|
isExcute = false;
|
|
break;
|
|
}
|
|
}
|
|
if(isExcute){
|
|
v.event(weEvent);
|
|
if(jindo.$Agent().navigator().ie){
|
|
var e = weEvent._event;
|
|
e.keyCode = "";
|
|
e.charCode = "";
|
|
}
|
|
weEvent.stop();
|
|
}else{
|
|
jindo.$A.Break();
|
|
}
|
|
}else{
|
|
v.event(weEvent);
|
|
if(jindo.$Agent().navigator().ie){
|
|
var e = weEvent._event;
|
|
e.keyCode = "";
|
|
e.charCode = "";
|
|
}
|
|
weEvent.stop();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
},
|
|
addEvent:function(fpEvent,sRawKey){
|
|
var events = this.keys[sRawKey].events;
|
|
if(!Shortcut.Helper.hasEvent(fpEvent,events)){
|
|
events.push({
|
|
event:fpEvent,
|
|
exceptions:[]
|
|
});
|
|
};
|
|
},
|
|
addException:function(fpException,sRawKey){
|
|
var commonExceptions = this.keys[sRawKey].commonExceptions;
|
|
if(!Shortcut.Helper.hasException(fpException,commonExceptions)){
|
|
commonExceptions.push(fpException);
|
|
};
|
|
},
|
|
removeException:function(fpException,sRawKey){
|
|
var commonExceptions = this.keys[sRawKey].commonExceptions;
|
|
commonExceptions = jindo.$A(commonExceptions).filter(function(exception){
|
|
return exception!=fpException;
|
|
}).$value();
|
|
},
|
|
removeEvent:function(fpEvent,sRawKey){
|
|
var events = this.keys[sRawKey].events;
|
|
events = jindo.$A(events).filter(function(event) {
|
|
return event!=fpEvent;
|
|
}).$value();
|
|
this.unRegister(sRawKey);
|
|
},
|
|
unRegister:function(sRawKey){
|
|
var aEvents = this.keys[sRawKey].events;
|
|
|
|
if(aEvents.length)
|
|
delete this.keys[sRawKey];
|
|
|
|
var hasNotKey = true;
|
|
for(var i in this.keys){
|
|
hasNotKey =false;
|
|
break;
|
|
}
|
|
|
|
if(hasNotKey){
|
|
Shortcut.Helper.bind(this.func,this.element,"detach");
|
|
delete Shortcut.Store.datas[this.id];
|
|
}
|
|
|
|
},
|
|
startDefalutBehavior: function(sRawKey){
|
|
this._setDefalutBehavior(sRawKey,false);
|
|
},
|
|
stopDefalutBehavior: function(sRawKey){
|
|
this._setDefalutBehavior(sRawKey,true);
|
|
},
|
|
_setDefalutBehavior: function(sRawKey,bType){
|
|
this.keys[sRawKey].stopDefalutBehavior = bType;
|
|
}
|
|
});
|
|
|
|
Shortcut.Helper = {
|
|
keyInterpretor:function(sKey){
|
|
var keyArray = sKey.split("+");
|
|
var wKeyArray = jindo.$A(keyArray);
|
|
|
|
var returnVal = "";
|
|
|
|
returnVal += wKeyArray.has("alt")?"1":"0";
|
|
returnVal += wKeyArray.has("ctrl")?"1":"0";
|
|
returnVal += wKeyArray.has("meta")?"1":"0";
|
|
returnVal += wKeyArray.has("shift")?"1":"0";
|
|
|
|
var wKeyArray = wKeyArray.filter(function(v){
|
|
return !(v=="alt"||v=="ctrl"||v=="meta"||v=="shift")
|
|
});
|
|
var key = wKeyArray.$value()[0];
|
|
|
|
if(key){
|
|
|
|
var sKey = Shortcut.Store.anthorKeyHash[key.toUpperCase()]||key.toUpperCase().charCodeAt(0);
|
|
returnVal += sKey;
|
|
}
|
|
|
|
return returnVal;
|
|
},
|
|
notCommonException:function(e,exceptions){
|
|
var leng = exceptions.length;
|
|
for(var i=0;i<leng ;i++){
|
|
if(!exceptions[i](e))
|
|
return false;
|
|
}
|
|
return true;
|
|
},
|
|
hasEvent:function(fpEvent,aEvents){
|
|
var nLength = aEvents.length;
|
|
for(var i=0; i<nLength; ++i){
|
|
if(aEvents.event==fpEvent){
|
|
return true;
|
|
}
|
|
};
|
|
return false;
|
|
},
|
|
hasException:function(fpException,commonExceptions){
|
|
var nLength = commonExceptions.length;
|
|
for(var i=0; i<nLength; ++i){
|
|
if(commonExceptions[i]==fpException){
|
|
return true;
|
|
}
|
|
};
|
|
return false;
|
|
},
|
|
bind:function(wfFunc,oElement,sType){
|
|
if(sType=="attach"){
|
|
domAttach(oElement,"keydown",wfFunc);
|
|
}else{
|
|
domDetach(oElement,"keydown",wfFunc);
|
|
}
|
|
}
|
|
|
|
};
|
|
|
|
(function domAttach (){
|
|
if(document.addEventListener){
|
|
window.domAttach = function(dom,ev,fn){
|
|
dom.addEventListener(ev, fn, false);
|
|
}
|
|
}else{
|
|
window.domAttach = function(dom,ev,fn){
|
|
dom.attachEvent("on"+ev, fn);
|
|
}
|
|
}
|
|
})();
|
|
|
|
(function domDetach (){
|
|
if(document.removeEventListener){
|
|
window.domDetach = function(dom,ev,fn){
|
|
dom.removeEventListener(ev, fn, false);
|
|
}
|
|
}else{
|
|
window.domDetach = function(dom,ev,fn){
|
|
dom.detachEvent("on"+ev, fn);
|
|
}
|
|
}
|
|
})();
|
|
|
|
|
|
|
|
Shortcut.Action ={
|
|
init:function(oData,sRawKey){
|
|
this.dataInstance = oData;
|
|
this.rawKey = sRawKey;
|
|
return this;
|
|
},
|
|
addEvent:function(fpEvent){
|
|
this.dataInstance.addEvent(fpEvent,this.rawKey);
|
|
return this;
|
|
},
|
|
removeEvent:function(fpEvent){
|
|
this.dataInstance.removeEvent(fpEvent,this.rawKey);
|
|
return this;
|
|
},
|
|
addException : function(fpException){
|
|
this.dataInstance.addException(fpException,this.rawKey);
|
|
return this;
|
|
},
|
|
removeException : function(fpException){
|
|
this.dataInstance.removeException(fpException,this.rawKey);
|
|
return this;
|
|
},
|
|
// addCommonException : function(fpException){
|
|
// return this;
|
|
// },
|
|
// removeCommonEexception : function(fpException){
|
|
// return this;
|
|
// },
|
|
startDefalutBehavior: function(){
|
|
this.dataInstance.startDefalutBehavior(this.rawKey);
|
|
return this;
|
|
},
|
|
stopDefalutBehavior: function(){
|
|
this.dataInstance.stopDefalutBehavior(this.rawKey);
|
|
return this;
|
|
},
|
|
resetElement: function(){
|
|
Shortcut.Store.reset(this.dataInstance.id);
|
|
return this;
|
|
},
|
|
resetAll: function(){
|
|
Shortcut.Store.allReset();
|
|
return this;
|
|
}
|
|
};
|
|
|
|
(function (){
|
|
Shortcut.Store.anthorKeyHash = {
|
|
BACKSPACE : 8,
|
|
TAB : 9,
|
|
ENTER : 13,
|
|
ESC : 27,
|
|
SPACE : 32,
|
|
PAGEUP : 33,
|
|
PAGEDOWN : 34,
|
|
END : 35,
|
|
HOME : 36,
|
|
LEFT : 37,
|
|
UP : 38,
|
|
RIGHT : 39,
|
|
DOWN : 40,
|
|
DEL : 46,
|
|
COMMA : 188,//(,)
|
|
PERIOD : 190,//(.)
|
|
SLASH : 191//(/),
|
|
};
|
|
var hash = Shortcut.Store.anthorKeyHash;
|
|
for(var i=1 ; i < 13 ; i++){
|
|
Shortcut.Store.anthorKeyHash["F"+i] = i+111;
|
|
}
|
|
var agent = jindo.$Agent().navigator();
|
|
if(agent.ie||agent.safari||agent.chrome){
|
|
hash.HYPHEN = 189;//(-)
|
|
hash.EQUAL = 187;//(=)
|
|
}else{
|
|
hash.HYPHEN = 109;
|
|
hash.EQUAL = 61;
|
|
}
|
|
})();
|
|
var shortcut = Shortcut;
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the hotkey feature
|
|
* @name hp_Hotkey.js
|
|
*/
|
|
nhn.husky.Hotkey = jindo.$Class({
|
|
name : "Hotkey",
|
|
|
|
$init : function(){
|
|
this.oShortcut = shortcut;
|
|
},
|
|
|
|
$ON_ADD_HOTKEY : function(sHotkey, sCMD, aArgs, elTarget){
|
|
if(!aArgs){aArgs = [];}
|
|
|
|
var func = jindo.$Fn(this.oApp.exec, this.oApp).bind(sCMD, aArgs);
|
|
this.oShortcut(sHotkey, elTarget).addEvent(func);
|
|
}
|
|
});
|
|
//}
|
|
/*[
|
|
* UNDO
|
|
*
|
|
* UNDO 히스토리에 저장되어 있는 이전 IR을 복구한다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* REDO
|
|
*
|
|
* UNDO 히스토리에 저장되어 있는 다음 IR을 복구한다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* RECORD_UNDO_ACTION
|
|
*
|
|
* 현재 IR을 UNDO 히스토리에 추가한다.
|
|
*
|
|
* sAction string 실행 할 액션(어떤 이유로 IR에 변경이 있었는지 참고용)
|
|
* oSaveOption object 저장 옵션(htRecordOption 참고)
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* RECORD_UNDO_BEFORE_ACTION
|
|
*
|
|
* 현재 IR을 UNDO 히스토리에 추가한다. 액션 전후 따로 저장 할 경우 전 단계.
|
|
*
|
|
* sAction string 실행 할 액션(어떤 이유로 IR에 변경이 있었는지 참고용)
|
|
* oSaveOption object 저장 옵션(htRecordOption 참고)
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* RECORD_UNDO_AFTER_ACTION
|
|
*
|
|
* 현재 IR을 UNDO 히스토리에 추가한다. 액션 전후 따로 저장 할 경우 후 단계.
|
|
*
|
|
* sAction string 실행 할 액션(어떤 이유로 IR에 변경이 있었는지 참고용)
|
|
* oSaveOption object 저장 옵션(htRecordOption 참고)
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* RESTORE_UNDO_HISTORY
|
|
*
|
|
* UNDO 히스토리에 저장되어 있는 IR을 복구한다.
|
|
*
|
|
* nUndoIdx number 몇번째 히스토리를 복구할지
|
|
* nUndoStateStep number 히스토리 내에 몇번째 스텝을 복구 할지. (before:0, after:1)
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* DO_RECORD_UNDO_HISTORY
|
|
*
|
|
* 현재 IR을 UNDO 히스토리에 추가한다.
|
|
*
|
|
* sAction string 실행 할 액션(어떤 이유로 IR에 변경이 있었는지 참고용)
|
|
* htRecordOption object 저장 옵션
|
|
* nStep (number) 0 | 1 액션의 스텝 인덱스 (보통 1단계이나 Selection 의 저장이 필요한 경우 1, 2단계로 나누어짐)
|
|
* bSkipIfEqual (bool) false | true 변경이 없다면 히스토리에 추가하지 않음 (현재 길이로 판단하여 저장함)
|
|
* bTwoStepAction (bool) false | true 2단계 액션인 경우
|
|
* sSaveTarget (string) [TAG] | null 저장 타겟을 지정하는 경우 사용 (해당 태그를 포함하여 저장)
|
|
* elSaveTarget : [Element] | null 저장 타겟을 지정하는 경우 사용 (해당 엘리먼트의 innerHTML을 저장)
|
|
* bDontSaveSelection : false | true Selection을 추가하지 않는 경우 (예, 표 편집)
|
|
* bMustBlockElement : false | true 반드시 Block 엘리먼트에 대해서만 저장함, 없으면 BODY 영역 (예, 글자 스타일 편집)
|
|
* bMustBlockContainer : false | true 반드시 Block 엘리먼트(그 중 컨테이너로 사용되는)에 대해서만 저장함, 없으면 BODY 영역 (예, 엔터)
|
|
* oUndoCallback : null | [Object] Undo 처리할 때 호출해야할 콜백 메시지 정보
|
|
* oRedoCallback : null | [Object] Redo 처리할 때 호출해야할 콜백 메시지 정보
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* DO_RECORD_UNDO_HISTORY_AT
|
|
*
|
|
* 현재 IR을 UNDO 히스토리의 지정된 위치에 추가한다.
|
|
*
|
|
* oInsertionIdx object 삽입할 위치({nIdx:히스토리 번호, nStep: 히스토리 내에 액션 번호})
|
|
* sAction string 실행 할 액션(어떤 이유로 IR에 변경이 있었는지 참고용)
|
|
* sContent string 저장할 내용
|
|
* oBookmark object oSelection.getXPathBookmark()를 통해 얻어진 북마크
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc Husky Framework에서 자주 사용되는 메시지를 처리하는 플러그인
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to Undo/Redo
|
|
* @name hp_SE_UndoRedo.js
|
|
* @required SE_EditingAreaManager, HuskyRangeManager
|
|
*/
|
|
nhn.husky.SE_UndoRedo = jindo.$Class({
|
|
name : "SE_UndoRedo",
|
|
|
|
oCurStateIdx : null,
|
|
iMinimumSizeChange : 1,
|
|
|
|
// limit = nMaxUndoCount + nAfterMaxDeleteBuffer. When the limit is reached delete [0...nAfterMaxDeleteBuffer] so only nMaxUndoCount histories will be left
|
|
nMaxUndoCount : 20, // 1000
|
|
nAfterMaxDeleteBuffer : 100,
|
|
|
|
sBlankContentsForFF : "<br>",
|
|
sDefaultXPath : "/HTML[0]/BODY[0]",
|
|
|
|
$init : function(){
|
|
this.aUndoHistory = [];
|
|
this.oCurStateIdx = {nIdx: 0, nStep: 0};
|
|
this.nHardLimit = this.nMaxUndoCount + this.nAfterMaxDeleteBuffer;
|
|
},
|
|
|
|
$LOCAL_BEFORE_ALL : function(sCmd){
|
|
if(sCmd.match(/_DO_RECORD_UNDO_HISTORY_AT$/)){
|
|
return true;
|
|
}
|
|
|
|
try{
|
|
if(this.oApp.getEditingMode() != "WYSIWYG"){
|
|
return false;
|
|
}
|
|
}catch(e){
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this._historyLength = 0;
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getUndoHistory", jindo.$Fn(this._getUndoHistory, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getUndoStateIdx", jindo.$Fn(this._getUndoStateIdx, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["saveSnapShot", jindo.$Fn(this._saveSnapShot, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["getLastKey", jindo.$Fn(this._getLastKey, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["setLastKey", jindo.$Fn(this._setLastKey, this).bind()]);
|
|
|
|
this._saveSnapShot();
|
|
|
|
this.oApp.exec("DO_RECORD_UNDO_HISTORY_AT", [this.oCurStateIdx, "", "", "", null, this.sDefaultXPath]);
|
|
},
|
|
|
|
_getLastKey : function(){
|
|
return this.sLastKey;
|
|
},
|
|
|
|
_setLastKey : function(sLastKey){
|
|
this.sLastKey = sLastKey;
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
var oNavigator = jindo.$Agent().navigator();
|
|
this.bIE = oNavigator.ie;
|
|
this.bFF = oNavigator.firefox;
|
|
//this.bChrome = oNavigator.chrome;
|
|
//this.bSafari = oNavigator.safari;
|
|
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["undo", "click", "UNDO"]);
|
|
this.oApp.exec("REGISTER_UI_EVENT", ["redo", "click", "REDO"]);
|
|
|
|
// [SMARTEDITORSUS-2260] 메일 > Mac에서 ctrl 조합 단축키 모두 meta 조합으로 변경
|
|
if (jindo.$Agent().os().mac) {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+z", "UNDO"]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["meta+y", "REDO"]);
|
|
} else {
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+z", "UNDO"]);
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+y", "REDO"]);
|
|
}
|
|
|
|
// this.htOptions = this.oApp.htOptions["SE_UndoRedo"] || {};
|
|
},
|
|
|
|
$ON_UNDO : function(){
|
|
this._doRecordUndoHistory("UNDO", { nStep : 0, bSkipIfEqual : true, bMustBlockContainer : true });
|
|
|
|
if(this.oCurStateIdx.nIdx <= 0){
|
|
return;
|
|
}
|
|
|
|
// 현재의 상태에서 Undo 했을 때 처리해야 할 메시지 호출
|
|
var oUndoCallback = this.aUndoHistory[this.oCurStateIdx.nIdx].oUndoCallback[this.oCurStateIdx.nStep];
|
|
var sCurrentPath = this.aUndoHistory[this.oCurStateIdx.nIdx].sParentXPath[this.oCurStateIdx.nStep];
|
|
|
|
if(oUndoCallback){
|
|
this.oApp.exec(oUndoCallback.sMsg, oUndoCallback.aParams);
|
|
}
|
|
|
|
if(this.oCurStateIdx.nStep > 0){
|
|
this.oCurStateIdx.nStep--;
|
|
}else{
|
|
var oTmpHistory = this.aUndoHistory[this.oCurStateIdx.nIdx];
|
|
|
|
this.oCurStateIdx.nIdx--;
|
|
|
|
if(oTmpHistory.nTotalSteps>1){
|
|
this.oCurStateIdx.nStep = 0;
|
|
}else{
|
|
oTmpHistory = this.aUndoHistory[this.oCurStateIdx.nIdx];
|
|
this.oCurStateIdx.nStep = oTmpHistory.nTotalSteps-1;
|
|
}
|
|
}
|
|
|
|
var sUndoHistoryPath = this.aUndoHistory[this.oCurStateIdx.nIdx].sParentXPath[this.oCurStateIdx.nStep];
|
|
var bUseDefault = false;
|
|
|
|
if(sUndoHistoryPath !== sCurrentPath && sUndoHistoryPath.indexOf(sCurrentPath) === 0){ // 현재의 Path가 Undo의 Path보다 범위가 큰 경우
|
|
bUseDefault = true;
|
|
}
|
|
|
|
this.oApp.exec("RESTORE_UNDO_HISTORY", [this.oCurStateIdx.nIdx, this.oCurStateIdx.nStep, bUseDefault]);
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
|
|
this.sLastKey = null;
|
|
},
|
|
|
|
|
|
$ON_REDO : function(){
|
|
if(this.oCurStateIdx.nIdx >= this.aUndoHistory.length){
|
|
return;
|
|
}
|
|
|
|
var oCurHistory = this.aUndoHistory[this.oCurStateIdx.nIdx];
|
|
|
|
if(this.oCurStateIdx.nIdx == this.aUndoHistory.length-1 && this.oCurStateIdx.nStep >= oCurHistory.nTotalSteps-1){
|
|
return;
|
|
}
|
|
|
|
if(this.oCurStateIdx.nStep < oCurHistory.nTotalSteps-1){
|
|
this.oCurStateIdx.nStep++;
|
|
}else{
|
|
this.oCurStateIdx.nIdx++;
|
|
oCurHistory = this.aUndoHistory[this.oCurStateIdx.nIdx];
|
|
this.oCurStateIdx.nStep = oCurHistory.nTotalSteps-1;
|
|
}
|
|
|
|
// 원복될 상태에서 Redo 했을 때 처리해야 할 메시지 호출
|
|
var oRedoCallback = this.aUndoHistory[this.oCurStateIdx.nIdx].oRedoCallback[this.oCurStateIdx.nStep];
|
|
|
|
if(oRedoCallback){
|
|
this.oApp.exec(oRedoCallback.sMsg, oRedoCallback.aParams);
|
|
}
|
|
|
|
this.oApp.exec("RESTORE_UNDO_HISTORY", [this.oCurStateIdx.nIdx, this.oCurStateIdx.nStep]);
|
|
this.oApp.exec("CHECK_STYLE_CHANGE", []);
|
|
|
|
this.sLastKey = null;
|
|
},
|
|
|
|
$ON_RECORD_UNDO_ACTION : function(sAction, oSaveOption){
|
|
oSaveOption = oSaveOption || { sSaveTarget : null, elSaveTarget : null, bMustBlockElement : false, bMustBlockContainer : false, bDontSaveSelection : false };
|
|
oSaveOption.nStep = 0;
|
|
oSaveOption.bSkipIfEqual = false;
|
|
oSaveOption.bTwoStepAction = false;
|
|
|
|
this._doRecordUndoHistory(sAction, oSaveOption);
|
|
},
|
|
|
|
$ON_RECORD_UNDO_BEFORE_ACTION : function(sAction, oSaveOption){
|
|
oSaveOption = oSaveOption || { sSaveTarget : null, elSaveTarget : null, bMustBlockElement : false, bMustBlockContainer : false, bDontSaveSelection : false };
|
|
oSaveOption.nStep = 0;
|
|
oSaveOption.bSkipIfEqual = false;
|
|
oSaveOption.bTwoStepAction = true;
|
|
|
|
this._doRecordUndoHistory(sAction, oSaveOption);
|
|
},
|
|
|
|
$ON_RECORD_UNDO_AFTER_ACTION : function(sAction, oSaveOption){
|
|
oSaveOption = oSaveOption || { sSaveTarget : null, elSaveTarget : null, bMustBlockElement : false, bMustBlockContainer : false, bDontSaveSelection : false };
|
|
oSaveOption.nStep = 1;
|
|
oSaveOption.bSkipIfEqual = false;
|
|
oSaveOption.bTwoStepAction = true;
|
|
|
|
this._doRecordUndoHistory(sAction, oSaveOption);
|
|
},
|
|
|
|
$ON_RESTORE_UNDO_HISTORY : function(nUndoIdx, nUndoStateStep, bUseDefault){
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
|
|
this.oCurStateIdx.nIdx = nUndoIdx;
|
|
this.oCurStateIdx.nStep = nUndoStateStep;
|
|
|
|
var oCurHistory = this.aUndoHistory[this.oCurStateIdx.nIdx],
|
|
sContent = oCurHistory.sContent[this.oCurStateIdx.nStep],
|
|
sFullContents = oCurHistory.sFullContents[this.oCurStateIdx.nStep],
|
|
oBookmark = oCurHistory.oBookmark[this.oCurStateIdx.nStep],
|
|
sParentXPath = oCurHistory.sParentXPath[this.oCurStateIdx.nStep],
|
|
oParent = null,
|
|
sCurContent = "",
|
|
oSelection = this.oApp.getEmptySelection();
|
|
|
|
this.oApp.exec("RESTORE_IE_SELECTION"); // this is done to null the ie selection
|
|
|
|
if(bUseDefault){
|
|
this.oApp.getWYSIWYGDocument().body.innerHTML = sFullContents;
|
|
sFullContents = this.oApp.getWYSIWYGDocument().body.innerHTML;
|
|
sCurContent = sFullContents;
|
|
sParentXPath = this.sDefaultXPath;
|
|
}else{
|
|
oParent = oSelection._evaluateXPath(sParentXPath, oSelection._document);
|
|
try{
|
|
oParent.innerHTML = sContent;
|
|
sCurContent = oParent.innerHTML;
|
|
}catch(e){ // Path 노드를 찾지 못하는 경우
|
|
this.oApp.getWYSIWYGDocument().body.innerHTML = sFullContents;
|
|
sFullContents = this.oApp.getWYSIWYGDocument().body.innerHTML; // setting the innerHTML may change the internal DOM structure, so save the value again.
|
|
sCurContent = sFullContents;
|
|
sParentXPath = this.sDefaultXPath;
|
|
}
|
|
}
|
|
|
|
if(this.bFF && sCurContent == this.sBlankContentsForFF){
|
|
sCurContent = "";
|
|
}
|
|
|
|
oCurHistory.sFullContents[this.oCurStateIdx.nStep] = sFullContents;
|
|
oCurHistory.sContent[this.oCurStateIdx.nStep] = sCurContent;
|
|
oCurHistory.sParentXPath[this.oCurStateIdx.nStep] = sParentXPath;
|
|
|
|
if(oBookmark && oBookmark.sType == "scroll"){
|
|
setTimeout(jindo.$Fn(function(){this.oApp.getWYSIWYGDocument().documentElement.scrollTop = oBookmark.nScrollTop;}, this).bind(), 0);
|
|
}else{
|
|
oSelection = this.oApp.getEmptySelection();
|
|
if(oSelection.selectionLoaded){
|
|
if(oBookmark){
|
|
oSelection.moveToXPathBookmark(oBookmark);
|
|
}else{
|
|
oSelection = this.oApp.getEmptySelection();
|
|
}
|
|
|
|
oSelection.select();
|
|
}
|
|
}
|
|
},
|
|
|
|
_doRecordUndoHistory : function(sAction, htRecordOption){
|
|
/*
|
|
htRecordOption = {
|
|
nStep : 0 | 1,
|
|
bSkipIfEqual : false | true,
|
|
bTwoStepAction : false | true,
|
|
sSaveTarget : [TAG] | null
|
|
elSaveTarget : [Element] | null
|
|
bDontSaveSelection : false | true
|
|
bMustBlockElement : false | true
|
|
bMustBlockContainer : false | true
|
|
};
|
|
*/
|
|
|
|
htRecordOption = htRecordOption || {};
|
|
|
|
var nStep = htRecordOption.nStep || 0,
|
|
bSkipIfEqual = htRecordOption.bSkipIfEqual || false,
|
|
bTwoStepAction = htRecordOption.bTwoStepAction || false,
|
|
sSaveTarget = htRecordOption.sSaveTarget || null,
|
|
elSaveTarget = htRecordOption.elSaveTarget || null,
|
|
bDontSaveSelection = htRecordOption.bDontSaveSelection || false,
|
|
bMustBlockElement = htRecordOption.bMustBlockElement || false,
|
|
bMustBlockContainer = htRecordOption.bMustBlockContainer || false,
|
|
oUndoCallback = htRecordOption.oUndoCallback,
|
|
oRedoCallback = htRecordOption.oRedoCallback;
|
|
|
|
// if we're in the middle of some action history,
|
|
// remove everything after current idx if any "little" change is made
|
|
this._historyLength = this.aUndoHistory.length;
|
|
|
|
if(this.oCurStateIdx.nIdx !== this._historyLength-1){
|
|
bSkipIfEqual = true;
|
|
}
|
|
|
|
var oCurHistory = this.aUndoHistory[this.oCurStateIdx.nIdx],
|
|
sHistoryFullContents = oCurHistory.sFullContents[this.oCurStateIdx.nStep],
|
|
sCurContent = "",
|
|
sFullContents = "",
|
|
sParentXPath = "",
|
|
oBookmark = null,
|
|
oSelection = null,
|
|
oInsertionIdx = {nIdx:this.oCurStateIdx.nIdx, nStep:this.oCurStateIdx.nStep}; // 히스토리를 저장할 위치
|
|
|
|
oSelection = this.oApp.getSelection();
|
|
|
|
if(oSelection.selectionLoaded){
|
|
oBookmark = oSelection.getXPathBookmark();
|
|
}
|
|
|
|
if(elSaveTarget){
|
|
sParentXPath = oSelection._getXPath(elSaveTarget);
|
|
}else if(sSaveTarget){
|
|
sParentXPath = this._getTargetXPath(oBookmark, sSaveTarget);
|
|
}else{
|
|
sParentXPath = this._getParentXPath(oBookmark, bMustBlockElement, bMustBlockContainer);
|
|
}
|
|
|
|
sFullContents = this.oApp.getWYSIWYGDocument().body.innerHTML;
|
|
// sCurContent = this.oApp.getWYSIWYGDocument().body.innerHTML.replace(/ *_cssquery_UID="[^"]+" */g, "");
|
|
|
|
if(sParentXPath === this.sDefaultXPath){
|
|
sCurContent = sFullContents;
|
|
}else{
|
|
sCurContent = oSelection._evaluateXPath(sParentXPath, oSelection._document).innerHTML;
|
|
}
|
|
|
|
if(this.bFF && sCurContent == this.sBlankContentsForFF){
|
|
sCurContent = "";
|
|
}
|
|
|
|
// every TwoStepAction needs to be recorded
|
|
if(!bTwoStepAction && bSkipIfEqual){
|
|
if(sHistoryFullContents.length === sFullContents.length){
|
|
return;
|
|
}
|
|
|
|
// 저장된 데이터와 같음에도 다르다고 처리되는 경우에 대한 처리
|
|
// (예, P안에 Block엘리먼트가 추가된 경우 P를 분리)
|
|
//if(this.bChrome || this.bSafari){
|
|
var elCurrentDiv = document.createElement("div");
|
|
var elHistoryDiv = document.createElement("div");
|
|
|
|
elCurrentDiv.innerHTML = sFullContents;
|
|
elHistoryDiv.innerHTML = sHistoryFullContents;
|
|
|
|
var elDocFragment = document.createDocumentFragment();
|
|
elDocFragment.appendChild(elCurrentDiv);
|
|
elDocFragment.appendChild(elHistoryDiv);
|
|
|
|
sFullContents = elCurrentDiv.innerHTML;
|
|
sHistoryFullContents = elHistoryDiv.innerHTML;
|
|
|
|
elCurrentDiv = null;
|
|
elHistoryDiv = null;
|
|
elDocFragment = null;
|
|
|
|
if(sHistoryFullContents.length === sFullContents.length){
|
|
return;
|
|
}
|
|
//}
|
|
}
|
|
|
|
if(bDontSaveSelection){
|
|
oBookmark = { sType : "scroll", nScrollTop : this.oApp.getWYSIWYGDocument().documentElement.scrollTop };
|
|
}
|
|
|
|
oInsertionIdx.nStep = nStep;
|
|
|
|
if(oInsertionIdx.nStep === 0 && this.oCurStateIdx.nStep === oCurHistory.nTotalSteps-1){
|
|
oInsertionIdx.nIdx = this.oCurStateIdx.nIdx+1;
|
|
}
|
|
|
|
this._doRecordUndoHistoryAt(oInsertionIdx, sAction, sCurContent, sFullContents, oBookmark, sParentXPath, oUndoCallback, oRedoCallback);
|
|
},
|
|
|
|
$ON_DO_RECORD_UNDO_HISTORY_AT : function(oInsertionIdx, sAction, sContent, sFullContents, oBookmark, sParentXPath){
|
|
this._doRecordUndoHistoryAt(oInsertionIdx, sAction, sContent, sFullContents, oBookmark, sParentXPath);
|
|
},
|
|
|
|
_doRecordUndoHistoryAt : function(oInsertionIdx, sAction, sContent, sFullContents, oBookmark, sParentXPath, oUndoCallback, oRedoCallback){
|
|
if(oInsertionIdx.nStep !== 0){
|
|
this.aUndoHistory[oInsertionIdx.nIdx].nTotalSteps = oInsertionIdx.nStep+1;
|
|
this.aUndoHistory[oInsertionIdx.nIdx].sContent[oInsertionIdx.nStep] = sContent;
|
|
this.aUndoHistory[oInsertionIdx.nIdx].sFullContents[oInsertionIdx.nStep] = sFullContents;
|
|
this.aUndoHistory[oInsertionIdx.nIdx].oBookmark[oInsertionIdx.nStep] = oBookmark;
|
|
this.aUndoHistory[oInsertionIdx.nIdx].sParentXPath[oInsertionIdx.nStep] = sParentXPath;
|
|
this.aUndoHistory[oInsertionIdx.nIdx].oUndoCallback[oInsertionIdx.nStep] = oUndoCallback;
|
|
this.aUndoHistory[oInsertionIdx.nIdx].oRedoCallback[oInsertionIdx.nStep] = oRedoCallback;
|
|
}else{
|
|
var oNewHistory = {sAction:sAction, nTotalSteps: 1};
|
|
oNewHistory.sContent = [];
|
|
oNewHistory.sContent[0] = sContent;
|
|
|
|
oNewHistory.sFullContents = [];
|
|
oNewHistory.sFullContents[0] = sFullContents;
|
|
|
|
oNewHistory.oBookmark = [];
|
|
oNewHistory.oBookmark[0] = oBookmark;
|
|
|
|
oNewHistory.sParentXPath = [];
|
|
oNewHistory.sParentXPath[0] = sParentXPath;
|
|
|
|
oNewHistory.oUndoCallback = [];
|
|
oNewHistory.oUndoCallback[0] = oUndoCallback;
|
|
|
|
oNewHistory.oRedoCallback = [];
|
|
oNewHistory.oRedoCallback[0] = oRedoCallback;
|
|
|
|
this.aUndoHistory.splice(oInsertionIdx.nIdx, this._historyLength - oInsertionIdx.nIdx, oNewHistory);
|
|
this._historyLength = this.aUndoHistory.length;
|
|
}
|
|
|
|
if(this._historyLength > this.nHardLimit){
|
|
this.aUndoHistory.splice(0, this.nAfterMaxDeleteBuffer);
|
|
oInsertionIdx.nIdx -= this.nAfterMaxDeleteBuffer;
|
|
}
|
|
this.oCurStateIdx.nIdx = oInsertionIdx.nIdx;
|
|
this.oCurStateIdx.nStep = oInsertionIdx.nStep;
|
|
},
|
|
|
|
_saveSnapShot : function(){
|
|
this.oSnapShot = {
|
|
oBookmark : this.oApp.getSelection().getXPathBookmark()
|
|
};
|
|
},
|
|
|
|
_getTargetXPath : function(oBookmark, sSaveTarget){ // ex. A, TABLE ...
|
|
var sParentXPath = this.sDefaultXPath,
|
|
aStartXPath = oBookmark[0].sXPath.split("/"),
|
|
aEndXPath = oBookmark[1].sXPath.split("/"),
|
|
aParentPath = [],
|
|
nPathLen = aStartXPath.length < aEndXPath.length ? aStartXPath.length : aEndXPath.length,
|
|
nPathIdx = 0, nTargetIdx = -1;
|
|
|
|
if(sSaveTarget === "BODY"){
|
|
return sParentXPath;
|
|
}
|
|
|
|
for(nPathIdx=0; nPathIdx<nPathLen; nPathIdx++){
|
|
if(aStartXPath[nPathIdx] !== aEndXPath[nPathIdx]){
|
|
break;
|
|
}
|
|
|
|
aParentPath.push(aStartXPath[nPathIdx]);
|
|
|
|
if(aStartXPath[nPathIdx] === "" || aStartXPath[nPathIdx] === "HTML" || aStartXPath[nPathIdx] === "BODY"){
|
|
continue;
|
|
}
|
|
|
|
if(aStartXPath[nPathIdx].indexOf(sSaveTarget) > -1){
|
|
nTargetIdx = nPathIdx;
|
|
}
|
|
}
|
|
|
|
if(nTargetIdx > -1){
|
|
aParentPath.length = nTargetIdx; // Target 의 상위 노드까지 지정
|
|
}
|
|
|
|
sParentXPath = aParentPath.join("/");
|
|
|
|
if(sParentXPath.length < this.sDefaultXPath.length){
|
|
sParentXPath = this.sDefaultXPath;
|
|
}
|
|
|
|
return sParentXPath;
|
|
},
|
|
|
|
_getParentXPath : function(oBookmark, bMustBlockElement, bMustBlockContainer){
|
|
var sParentXPath = this.sDefaultXPath,
|
|
aStartXPath, aEndXPath,
|
|
aSnapShotStart, aSnapShotEnd,
|
|
nSnapShotLen, nPathLen,
|
|
aParentPath = ["", "HTML[0]", "BODY[0]"],
|
|
nPathIdx = 0, nBlockIdx = -1,
|
|
// rxBlockContainer = /\bUL|OL|TD|TR|TABLE|BLOCKQUOTE\[/i, // DL
|
|
// rxBlockElement = /\bP|LI|DIV|UL|OL|TD|TR|TABLE|BLOCKQUOTE\[/i, // H[1-6]|DD|DT|DL|PRE
|
|
// rxBlock,
|
|
sPath, sTag;
|
|
|
|
if(!oBookmark){
|
|
return sParentXPath;
|
|
}
|
|
|
|
// 가능한 중복되는 Parent 를 검색
|
|
if(oBookmark[0].sXPath === sParentXPath || oBookmark[1].sXPath === sParentXPath){
|
|
return sParentXPath;
|
|
}
|
|
|
|
aStartXPath = oBookmark[0].sXPath.split("/");
|
|
aEndXPath = oBookmark[1].sXPath.split("/");
|
|
aSnapShotStart = this.oSnapShot.oBookmark[0].sXPath.split("/");
|
|
aSnapShotEnd = this.oSnapShot.oBookmark[1].sXPath.split("/");
|
|
|
|
nSnapShotLen = aSnapShotStart.length < aSnapShotEnd.length ? aSnapShotStart.length : aSnapShotEnd.length;
|
|
nPathLen = aStartXPath.length < aEndXPath.length ? aStartXPath.length : aEndXPath.length;
|
|
nPathLen = nPathLen < nSnapShotLen ? nPathLen : nSnapShotLen;
|
|
|
|
if(nPathLen < 3){ // BODY
|
|
return sParentXPath;
|
|
}
|
|
|
|
bMustBlockElement = bMustBlockElement || false;
|
|
bMustBlockContainer = bMustBlockContainer || false;
|
|
// rxBlock = bMustBlockElement ? rxBlockElement : rxBlockContainer;
|
|
|
|
for(nPathIdx=3; nPathIdx<nPathLen; nPathIdx++){
|
|
sPath = aStartXPath[nPathIdx];
|
|
|
|
if(sPath !== aEndXPath[nPathIdx] ||
|
|
sPath !== aSnapShotStart[nPathIdx] ||
|
|
sPath !== aSnapShotEnd[nPathIdx] ||
|
|
aEndXPath[nPathIdx] !== aSnapShotStart[nPathIdx] ||
|
|
aEndXPath[nPathIdx] !== aSnapShotEnd[nPathIdx] ||
|
|
aSnapShotStart[nPathIdx] !== aSnapShotEnd[nPathIdx]){
|
|
|
|
break;
|
|
}
|
|
|
|
aParentPath.push(sPath);
|
|
|
|
sTag = sPath.substring(0, sPath.indexOf("["));
|
|
|
|
if(bMustBlockElement && (sTag === "P" || sTag === "LI" || sTag === "DIV")){
|
|
nBlockIdx = nPathIdx;
|
|
}else if(sTag === "UL" || sTag === "OL" || sTag === "TD" || sTag === "TR" || sTag === "TABLE" || sTag === "BLOCKQUOTE"){
|
|
nBlockIdx = nPathIdx;
|
|
}
|
|
|
|
// if(rxBlock.test(sPath)){
|
|
// nBlockIdx = nPathIdx;
|
|
// }
|
|
}
|
|
|
|
if(nBlockIdx > -1){
|
|
aParentPath.length = nBlockIdx + 1;
|
|
}else if(bMustBlockElement || bMustBlockContainer){
|
|
return sParentXPath;
|
|
}
|
|
|
|
return aParentPath.join("/");
|
|
},
|
|
|
|
_getUndoHistory : function(){
|
|
return this.aUndoHistory;
|
|
},
|
|
|
|
_getUndoStateIdx : function(){
|
|
return this.oCurStateIdx;
|
|
}
|
|
});
|
|
/*[
|
|
* ATTACH_HOVER_EVENTS
|
|
*
|
|
* 주어진 HTML엘리먼트에 Hover 이벤트 발생시 특정 클래스가 할당 되도록 설정
|
|
*
|
|
* aElms array Hover 이벤트를 걸 HTML Element 목록
|
|
* sHoverClass string Hover 시에 할당 할 클래스
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc Husky Framework에서 자주 사용되는 유틸성 메시지를 처리하는 플러그인
|
|
*/
|
|
nhn.husky.Utils = jindo.$Class({
|
|
name : "Utils",
|
|
|
|
$init : function(){
|
|
var oAgentInfo = jindo.$Agent();
|
|
var oNavigatorInfo = oAgentInfo.navigator();
|
|
|
|
if(oNavigatorInfo.ie && oNavigatorInfo.version == 6){
|
|
try{
|
|
document.execCommand('BackgroundImageCache', false, true);
|
|
}catch(e){}
|
|
}
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["htBrowser", jindo.$Agent().navigator()]);
|
|
},
|
|
|
|
$ON_ATTACH_HOVER_EVENTS : function(aElms, htOptions){
|
|
htOptions = htOptions || [];
|
|
var sHoverClass = htOptions.sHoverClass || "hover";
|
|
var fnElmToSrc = htOptions.fnElmToSrc || function(el){return el};
|
|
var fnElmToTarget = htOptions.fnElmToTarget || function(el){return el};
|
|
|
|
if(!aElms) return;
|
|
|
|
var wfAddClass = jindo.$Fn(function(wev){
|
|
jindo.$Element(fnElmToTarget(wev.currentElement)).addClass(sHoverClass);
|
|
}, this);
|
|
|
|
var wfRemoveClass = jindo.$Fn(function(wev){
|
|
jindo.$Element(fnElmToTarget(wev.currentElement)).removeClass(sHoverClass);
|
|
}, this);
|
|
|
|
for(var i=0, len = aElms.length; i<len; i++){
|
|
var elSource = fnElmToSrc(aElms[i]);
|
|
|
|
wfAddClass.attach(elSource, "mouseover");
|
|
wfRemoveClass.attach(elSource, "mouseout");
|
|
|
|
wfAddClass.attach(elSource, "focus");
|
|
wfRemoveClass.attach(elSource, "blur");
|
|
}
|
|
}
|
|
});
|
|
|
|
/*[
|
|
* SHOW_DIALOG_LAYER
|
|
*
|
|
* 다이얼로그 레이어를 화면에 보여준다.
|
|
*
|
|
* oLayer HTMLElement 다이얼로그 레이어로 사용 할 HTML 엘리먼트
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* HIDE_DIALOG_LAYER
|
|
*
|
|
* 다이얼로그 레이어를 화면에 숨긴다.
|
|
*
|
|
* oLayer HTMLElement 숨길 다이얼로그 레이어에 해당 하는 HTML 엘리먼트
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* HIDE_LAST_DIALOG_LAYER
|
|
*
|
|
* 마지막으로 화면에 표시한 다이얼로그 레이어를 숨긴다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* HIDE_ALL_DIALOG_LAYER
|
|
*
|
|
* 표시 중인 모든 다이얼로그 레이어를 숨긴다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc 드래그가 가능한 레이어를 컨트롤 하는 플러그인
|
|
*/
|
|
nhn.husky.DialogLayerManager = jindo.$Class({
|
|
name : "DialogLayerManager",
|
|
aMadeDraggable : null,
|
|
aOpenedLayers : null,
|
|
|
|
$init : function(){
|
|
this.aMadeDraggable = [];
|
|
this.aDraggableLayer = [];
|
|
this.aOpenedLayers = [];
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function() {
|
|
this.oApp.registerLazyMessage(["SHOW_DIALOG_LAYER","TOGGLE_DIALOG_LAYER"], ["hp_DialogLayerManager$Lazy.js", "N_DraggableLayer.js"]);
|
|
}
|
|
});
|
|
/*[
|
|
* TOGGLE_ACTIVE_LAYER
|
|
*
|
|
* 액티브 레이어가 화면에 보이는 여부를 토글 한다.
|
|
*
|
|
* oLayer HTMLElement 레이어로 사용할 HTML Element
|
|
* sOnOpenCmd string 화면에 보이는 경우 발생 할 메시지(옵션)
|
|
* aOnOpenParam array sOnOpenCmd와 함께 넘겨줄 파라미터(옵션)
|
|
* sOnCloseCmd string 해당 레이어가 화면에서 숨겨질 때 발생 할 메시지(옵션)
|
|
* aOnCloseParam array sOnCloseCmd와 함께 넘겨줄 파라미터(옵션)
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* SHOW_ACTIVE_LAYER
|
|
*
|
|
* 액티브 레이어가 화면에 보이는 여부를 토글 한다.
|
|
*
|
|
* oLayer HTMLElement 레이어로 사용할 HTML Element
|
|
* sOnCloseCmd string 해당 레이어가 화면에서 숨겨질 때 발생 할 메시지(옵션)
|
|
* aOnCloseParam array sOnCloseCmd와 함께 넘겨줄 파라미터(옵션)
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/*[
|
|
* HIDE_ACTIVE_LAYER
|
|
*
|
|
* 현재 화면에 보이는 액티브 레이어를 화면에서 숨긴다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc 한번에 한개만 화면에 보여야 하는 레이어를 관리하는 플러그인
|
|
*/
|
|
nhn.husky.ActiveLayerManager = jindo.$Class({
|
|
name : "ActiveLayerManager",
|
|
oCurrentLayer : null,
|
|
|
|
$BEFORE_MSG_APP_READY : function() {
|
|
this.oNavigator = jindo.$Agent().navigator();
|
|
},
|
|
|
|
$ON_TOGGLE_ACTIVE_LAYER : function(oLayer, sOnOpenCmd, aOnOpenParam, sOnCloseCmd, aOnCloseParam){
|
|
if(oLayer == this.oCurrentLayer){
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
}else{
|
|
this.oApp.exec("SHOW_ACTIVE_LAYER", [oLayer, sOnCloseCmd, aOnCloseParam]);
|
|
if(sOnOpenCmd){this.oApp.exec(sOnOpenCmd, aOnOpenParam);}
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_ACTIVE_LAYER : function(oLayer, sOnCloseCmd, aOnCloseParam){
|
|
oLayer = jindo.$(oLayer);
|
|
|
|
var oPrevLayer = this.oCurrentLayer;
|
|
if(oLayer == oPrevLayer){return;}
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
|
|
this.sOnCloseCmd = sOnCloseCmd;
|
|
this.aOnCloseParam = aOnCloseParam;
|
|
|
|
oLayer.style.display = "block";
|
|
this.oCurrentLayer = oLayer;
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["oToolBarLayer", this.oCurrentLayer]);
|
|
},
|
|
|
|
$ON_HIDE_ACTIVE_LAYER : function(){
|
|
var oLayer = this.oCurrentLayer;
|
|
if(!oLayer){return;}
|
|
oLayer.style.display = "none";
|
|
this.oCurrentLayer = null;
|
|
if(this.sOnCloseCmd){
|
|
this.oApp.exec(this.sOnCloseCmd, this.aOnCloseParam);
|
|
}
|
|
},
|
|
|
|
$ON_HIDE_ACTIVE_LAYER_IF_NOT_CHILD : function(el){
|
|
var elTmp = el;
|
|
while(elTmp){
|
|
if(elTmp == this.oCurrentLayer){
|
|
return;
|
|
}
|
|
elTmp = elTmp.parentNode;
|
|
}
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
},
|
|
|
|
// for backward compatibility only.
|
|
// use HIDE_ACTIVE_LAYER instead!
|
|
$ON_HIDE_CURRENT_ACTIVE_LAYER : function(){
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
}
|
|
});
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of the operations related to string conversion. Ususally used to convert the IR value.
|
|
* @name hp_StringConverterManager.js
|
|
*/
|
|
nhn.husky.StringConverterManager = jindo.$Class({
|
|
name : "StringConverterManager",
|
|
|
|
oConverters : null,
|
|
|
|
$init : function(){
|
|
this.oConverters = {};
|
|
this.oConverters_DOM = {};
|
|
this.oAgent = jindo.$Agent().navigator();
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["applyConverter", jindo.$Fn(this.applyConverter, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["addConverter", jindo.$Fn(this.addConverter, this).bind()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["addConverter_DOM", jindo.$Fn(this.addConverter_DOM, this).bind()]);
|
|
},
|
|
|
|
applyConverter : function(sRuleName, sContents, oDocument){
|
|
//string을 넣는 이유:IE의 경우,본문 앞에 있는 html 주석이 삭제되는 경우가 있기때문에 임시 string을 추가해준것임.
|
|
var sTmpStr = "@"+(new Date()).getTime()+"@";
|
|
var rxTmpStr = new RegExp(sTmpStr, "g");
|
|
|
|
var oRes = {sContents:sTmpStr+sContents};
|
|
|
|
oDocument = oDocument || document;
|
|
|
|
this.oApp.exec("MSG_STRING_CONVERTER_STARTED", [sRuleName, oRes]);
|
|
// this.oApp.exec("MSG_STRING_CONVERTER_STARTED_"+sRuleName, [oRes]);
|
|
|
|
var aConverters;
|
|
sContents = oRes.sContents;
|
|
aConverters = this.oConverters_DOM[sRuleName];
|
|
if(aConverters){
|
|
var elContentsHolder = oDocument.createElement("DIV");
|
|
elContentsHolder.innerHTML = sContents;
|
|
|
|
for(var i=0; i<aConverters.length; i++){
|
|
aConverters[i](elContentsHolder);
|
|
}
|
|
sContents = elContentsHolder.innerHTML;
|
|
// 내용물에 EMBED등이 있을 경우 IE에서 페이지 나갈 때 권한 오류 발생 할 수 있어 명시적으로 노드 삭제.
|
|
|
|
if(!!elContentsHolder.parentNode){
|
|
elContentsHolder.parentNode.removeChild(elContentsHolder);
|
|
}
|
|
elContentsHolder = null;
|
|
|
|
|
|
//IE의 경우, sContents를 innerHTML로 넣는 경우 string과 <p>tag 사이에 '\n\'개행문자를 넣어준다.
|
|
if( jindo.$Agent().navigator().ie ){
|
|
sTmpStr = sTmpStr +'(\r\n)?'; //ie+win에서는 개행이 \r\n로 들어감.
|
|
rxTmpStr = new RegExp(sTmpStr , "g");
|
|
}
|
|
}
|
|
|
|
aConverters = this.oConverters[sRuleName];
|
|
if(aConverters){
|
|
for(var i=0; i<aConverters.length; i++){
|
|
var sTmpContents = aConverters[i](sContents);
|
|
if(typeof sTmpContents != "undefined"){
|
|
sContents = sTmpContents;
|
|
}
|
|
}
|
|
}
|
|
|
|
oRes = {sContents:sContents};
|
|
this.oApp.exec("MSG_STRING_CONVERTER_ENDED", [sRuleName, oRes]);
|
|
|
|
oRes.sContents = oRes.sContents.replace(rxTmpStr, "");
|
|
return oRes.sContents;
|
|
},
|
|
|
|
$ON_ADD_CONVERTER : function(sRuleName, funcConverter){
|
|
var aCallerStack = this.oApp.aCallerStack;
|
|
funcConverter.sPluginName = aCallerStack[aCallerStack.length-2].name;
|
|
this.addConverter(sRuleName, funcConverter);
|
|
},
|
|
|
|
$ON_ADD_CONVERTER_DOM : function(sRuleName, funcConverter){
|
|
var aCallerStack = this.oApp.aCallerStack;
|
|
funcConverter.sPluginName = aCallerStack[aCallerStack.length-2].name;
|
|
this.addConverter_DOM(sRuleName, funcConverter);
|
|
},
|
|
|
|
addConverter : function(sRuleName, funcConverter){
|
|
var aConverters = this.oConverters[sRuleName];
|
|
if(!aConverters){
|
|
this.oConverters[sRuleName] = [];
|
|
}
|
|
|
|
this.oConverters[sRuleName][this.oConverters[sRuleName].length] = funcConverter;
|
|
},
|
|
|
|
addConverter_DOM : function(sRuleName, funcConverter){
|
|
var aConverters = this.oConverters_DOM[sRuleName];
|
|
if(!aConverters){
|
|
this.oConverters_DOM[sRuleName] = [];
|
|
}
|
|
|
|
this.oConverters_DOM[sRuleName][this.oConverters_DOM[sRuleName].length] = funcConverter;
|
|
}
|
|
});
|
|
//}
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that maps a message code to the actual message
|
|
* @name hp_MessageManager.js
|
|
*/
|
|
nhn.husky.MessageManager = jindo.$Class({
|
|
name : "MessageManager",
|
|
|
|
_oMessageMapSet : {},
|
|
_sDefaultLocale : "ko_KR",
|
|
|
|
$init : function(oMessageMap, sLocale){
|
|
// 하위호환을 위해 기존 코드 유지
|
|
var oTmpMessageMap;
|
|
switch(sLocale) {
|
|
case "ja_JP" :
|
|
oTmpMessageMap = oMessageMap_ja_JP;
|
|
break;
|
|
case "en_US" :
|
|
oTmpMessageMap = oMessageMap_en_US;
|
|
break;
|
|
case "zh_CN" :
|
|
oTmpMessageMap = oMessageMap_zh_CN;
|
|
break;
|
|
default : // Korean
|
|
oTmpMessageMap = oMessageMap;
|
|
break;
|
|
}
|
|
oTmpMessageMap = oTmpMessageMap || oMessageMap;
|
|
|
|
this._sDefaultLocale = sLocale || this._sDefaultLocale;
|
|
this._setMessageMap(oTmpMessageMap, this._sDefaultLocale);
|
|
},
|
|
|
|
/**
|
|
* 로케일에 해당하는 메시지맵을 세팅한다.
|
|
* @param {Object} oMessageMap 세팅할 메시지맵 객체
|
|
* @param {String} [sLocale] 메시지맵 객체를 세팅할 로케일 정보
|
|
*/
|
|
_setMessageMap : function(oMessageMap, sLocale){
|
|
sLocale = sLocale || this._sDefaultLocale;
|
|
if(oMessageMap){
|
|
this._oMessageMapSet[sLocale] = oMessageMap;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 로케일에 해당하는 메시지맵을 가져온다.
|
|
* @param {String} [sLocale] 가져올 메시지맵 객체의 로케일 정보
|
|
* @returns {Object} 메시지맵
|
|
*/
|
|
_getMessageMap : function(sLocale){
|
|
return this._oMessageMapSet[sLocale] || this._oMessageMapSet[this._sDefaultLocale] || {};
|
|
},
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["$MSG", jindo.$Fn(this.getMessage, this).bind()]);
|
|
},
|
|
|
|
/**
|
|
* 메시지문자열을 가져온다.
|
|
* @param {String} sMsg 메시지키
|
|
* @param {String} [sLocale] 가져올 메시지의 로케일 정보
|
|
* @returns {String} 해당로케일의 메시지문자열
|
|
*/
|
|
getMessage : function(sMsg, sLocale){
|
|
var oMessageMap = this._getMessageMap(sLocale);
|
|
|
|
if(oMessageMap[sMsg]){return unescape(oMessageMap[sMsg]);}
|
|
return sMsg;
|
|
}
|
|
});
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains
|
|
* @name hp_LazyLoader.js
|
|
*/
|
|
nhn.husky.LazyLoader = jindo.$Class({
|
|
name : "LazyLoader",
|
|
|
|
// sMsg : KEY
|
|
// contains htLoadingInfo : {}
|
|
htMsgInfo : null,
|
|
|
|
// contains objects
|
|
// sURL : HTML to be loaded
|
|
// elTarget : where to append the HTML
|
|
// sSuccessCallback : message name
|
|
// sFailureCallback : message name
|
|
// nLoadingStatus :
|
|
// 0 : loading not started
|
|
// 1 : loading started
|
|
// 2 : loading ended
|
|
aLoadingInfo : null,
|
|
|
|
// aToDo : [{aMsgs: ["EXECCOMMAND"], sURL: "http://127.0.0.1/html_snippet.txt", elTarget: elPlaceHolder}, ...]
|
|
$init : function(aToDo){
|
|
this.htMsgInfo = {};
|
|
this.aLoadingInfo = [];
|
|
this.aToDo = aToDo;
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
for(var i=0; i<this.aToDo.length; i++){
|
|
var htToDoDetail = this.aToDo[i];
|
|
this._createBeforeHandlersAndSaveURLInfo(htToDoDetail.oMsgs, htToDoDetail.sURL, htToDoDetail.elTarget, htToDoDetail.htOptions);
|
|
}
|
|
},
|
|
|
|
$LOCAL_BEFORE_ALL : function(sMsgHandler, aParams){
|
|
var sMsg = sMsgHandler.replace("$BEFORE_", "");
|
|
|
|
var htCurMsgInfo = this.htMsgInfo[sMsg];
|
|
|
|
// ignore current message
|
|
if(htCurMsgInfo.nLoadingStatus == 1){return true;}
|
|
|
|
// the HTML was loaded before(probably by another message), remove the loading handler and re-send the message
|
|
if(htCurMsgInfo.nLoadingStatus == 2){
|
|
this[sMsgHandler] = function(){
|
|
this._removeHandler(sMsgHandler);
|
|
this.oApp.delayedExec(sMsg, aParams, 0);
|
|
return false;
|
|
};
|
|
return true;
|
|
}
|
|
|
|
htCurMsgInfo.bLoadingStatus = 1;
|
|
(new jindo.$Ajax(htCurMsgInfo.sURL, {
|
|
onload : jindo.$Fn(this._onload, this).bind(sMsg, aParams)
|
|
})).request();
|
|
|
|
return true;
|
|
},
|
|
|
|
_onload : function(sMsg, aParams, oResponse){
|
|
if(oResponse._response.readyState == 4) {
|
|
this.htMsgInfo[sMsg].elTarget.innerHTML = oResponse.text();
|
|
this.htMsgInfo[sMsg].nLoadingStatus = 2;
|
|
this._removeHandler("$BEFORE_"+sMsg);
|
|
this.oApp.exec("sMsg", aParams);
|
|
}else{
|
|
this.oApp.exec(this.htMsgInfo[sMsg].sFailureCallback, []);
|
|
}
|
|
},
|
|
|
|
_removeHandler : function(sMsgHandler){
|
|
delete this[sMsgHandler];
|
|
this.oApp.createMessageMap(sMsgHandler);
|
|
},
|
|
|
|
_createBeforeHandlersAndSaveURLInfo : function(oMsgs, sURL, elTarget, htOptions){
|
|
htOptions = htOptions || {};
|
|
|
|
var htNewInfo = {
|
|
sURL : sURL,
|
|
elTarget : elTarget,
|
|
sSuccessCallback : htOptions.sSuccessCallback,
|
|
sFailureCallback : htOptions.sFailureCallback,
|
|
nLoadingStatus : 0
|
|
};
|
|
this.aLoadingInfo[this.aLoadingInfo.legnth] = htNewInfo;
|
|
|
|
// extract msgs if plugin is given
|
|
if(!(oMsgs instanceof Array)){
|
|
var oPlugin = oMsgs;
|
|
|
|
oMsgs = [];
|
|
var htMsgAdded = {};
|
|
for(var sFunctionName in oPlugin){
|
|
if(sFunctionName.match(/^\$(BEFORE|ON|AFTER)_(.+)$/)){
|
|
var sMsg = RegExp.$2;
|
|
if(sMsg == "MSG_APP_READY"){continue;}
|
|
|
|
if(!htMsgAdded[sMsg]){
|
|
oMsgs[oMsgs.length] = RegExp.$2;
|
|
htMsgAdded[sMsg] = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for(var i=0; i<oMsgs.length; i++){
|
|
// create HTML loading handler
|
|
var sTmpMsg = "$BEFORE_"+oMsgs[i];
|
|
this[sTmpMsg] = function(){return false;};
|
|
this.oApp.createMessageMap(sTmpMsg);
|
|
|
|
// add loading info
|
|
this.htMsgInfo[oMsgs[i]] = htNewInfo;
|
|
}
|
|
}
|
|
});
|
|
//}
|
|
/**
|
|
* @name nhn.husky.PopUpManager
|
|
* @namespace
|
|
* @description 팝업 매니저 클래스.
|
|
* <dt><strong>Spec Code</strong></dt>
|
|
* <dd><a href="http://ajaxui.nhndesign.com/svnview/SmartEditor2_Official/tags/SE2M_popupManager/0.1/test/spec/hp_popupManager_spec.html" target="_new">Spec</a></dd>
|
|
* <dt><strong>wiki</strong></dt>
|
|
* <dd><a href="http://wikin.nhncorp.com/pages/viewpage.action?pageId=63501152" target="_new">wiki</a></dd>
|
|
* @author NHN AjaxUI Lab - sung jong min
|
|
* @version 0.1
|
|
* @since 2009.07.06
|
|
*/
|
|
nhn.husky.PopUpManager = {};
|
|
|
|
/** * @ignore */
|
|
nhn.husky.PopUpManager._instance = null;
|
|
/** * @ignore */
|
|
nhn.husky.PopUpManager._pluginKeyCnt = 0;
|
|
|
|
/**
|
|
* @description 팝업 매니저 인스턴스 호출 메소드, nhn.husky js framework 기반 코드
|
|
* @public
|
|
* @param {Object} oApp 허스키 코어 객체를 넘겨준다.(this.oApp)
|
|
* @return {Object} nhn.husky.PopUpManager Instance
|
|
* @example 팝업관련 플러그인 제작 예제
|
|
* nhn.husky.NewPlugin = function(){
|
|
* this.$ON_APP_READY = function(){
|
|
* // 팝업 매니저 getInstance 메소드를 호출한다.
|
|
* // 허스키 코어의 참조값을 넘겨준다(this.oApp)
|
|
* this.oPopupMgr = nhn.husky.PopUpMaganer.getInstance(this.oApp);
|
|
* };
|
|
*
|
|
* // 팝업을 요청하는 메시지 메소드는 아래와 같음
|
|
* this.$ON_NEWPLUGIN_OPEN_WINDOW = function(){
|
|
* var oWinOp = {
|
|
* oApp : this.oApp, // oApp this.oApp 허스키 참조값
|
|
* sUrl : "", // sUrl : 페이지 URL
|
|
* sName : "", // sName : 페이지 name
|
|
* nWidth : 400,
|
|
* nHeight : 400,
|
|
* bScroll : true
|
|
* }
|
|
* this.oPopUpMgr.openWindow(oWinOp);
|
|
* };
|
|
*
|
|
* // 팝업페이지 응답데이타 반환 메시지 메소드를 정의함.
|
|
* // 각 플러그인 팝업페이지에서 해당 메시지와 데이타를 넘기게 됨.
|
|
* this.@ON_NEWPLUGIN_WINDOW_CALLBACK = function(){
|
|
* // 팝업페이지별로 정의된 형태의 아규먼트 데이타가 넘어오면 처리한다.
|
|
* }
|
|
* }
|
|
* @example 팝업 페이지와 opener 호출 인터페이스 예제
|
|
* onclick시
|
|
* "nhn.husky.PopUpManager.setCallback(window, "NEWPLUGIN_WINDOW_CALLBACK", oData);"
|
|
* 형태로 호출함.
|
|
*
|
|
*
|
|
*/
|
|
nhn.husky.PopUpManager.getInstance = function(oApp) {
|
|
if (this._instance==null) {
|
|
|
|
this._instance = new (function(){
|
|
|
|
this._whtPluginWin = new jindo.$H();
|
|
this._whtPlugin = new jindo.$H();
|
|
this.addPlugin = function(sKey, vValue){
|
|
this._whtPlugin.add(sKey, vValue);
|
|
};
|
|
|
|
this.getPlugin = function() {
|
|
return this._whtPlugin;
|
|
};
|
|
this.getPluginWin = function() {
|
|
return this._whtPluginWin;
|
|
};
|
|
|
|
this.openWindow = function(oWinOpt) {
|
|
var op= {
|
|
oApp : null,
|
|
sUrl : "",
|
|
sName : "popup",
|
|
sLeft : null,
|
|
sTop : null,
|
|
nWidth : 400,
|
|
nHeight : 400,
|
|
sProperties : null,
|
|
bScroll : true
|
|
};
|
|
for(var i in oWinOpt) op[i] = oWinOpt[i];
|
|
|
|
if(op.oApp == null) {
|
|
alert("팝업 요청시 옵션으로 oApp(허스키 reference) 값을 설정하셔야 합니다.");
|
|
}
|
|
|
|
var left = op.sLeft || (screen.availWidth-op.nWidth)/2;
|
|
var top = op.sTop ||(screen.availHeight-op.nHeight)/2;
|
|
|
|
var sProperties = op.sProperties != null ? op.sProperties :
|
|
"top="+ top +",left="+ left +",width="+op.nWidth+",height="+op.nHeight+",scrollbars="+(op.bScroll?"yes":"no")+",status=yes";
|
|
var win = window.open(op.sUrl, op.sName,sProperties);
|
|
if (!!win) {
|
|
setTimeout( function(){
|
|
try{win.focus();}catch(e){}
|
|
}, 100);
|
|
}
|
|
|
|
this.removePluginWin(win);
|
|
this._whtPluginWin.add(this.getCorrectKey(this._whtPlugin, op.oApp), win);
|
|
|
|
return win;
|
|
};
|
|
this.getCorrectKey = function(whtData, oCompare) {
|
|
var key = null;
|
|
whtData.forEach(function(v,k){
|
|
if (v == oCompare) {
|
|
key = k;
|
|
return;
|
|
}
|
|
});
|
|
return key;
|
|
};
|
|
this.removePluginWin = function(vValue) {
|
|
var list = this._whtPluginWin.search(vValue);
|
|
if (list) {
|
|
this._whtPluginWin.remove(list);
|
|
this.removePluginWin(vValue);
|
|
}
|
|
}
|
|
})();
|
|
}
|
|
|
|
this._instance.addPlugin("plugin_" + (this._pluginKeyCnt++), oApp);
|
|
return nhn.husky.PopUpManager._instance;
|
|
};
|
|
|
|
/**
|
|
* @description opener 연동 interface
|
|
* @public
|
|
* @param {Object} oOpenWin 팝업 페이지의 window 객체
|
|
* @param {Object} sMsg 플러그인 메시지명
|
|
* @param {Object} oData 응답 데이타
|
|
*/
|
|
nhn.husky.PopUpManager.setCallback = function(oOpenWin, sMsg, oData) {
|
|
if (this._instance.getPluginWin().hasValue(oOpenWin)) {
|
|
var key = this._instance.getCorrectKey(this._instance.getPluginWin(), oOpenWin);
|
|
if (key) {
|
|
this._instance.getPlugin().$(key).exec(sMsg, oData);
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @description opener에 허스키 함수를 실행시키고 데이터 값을 리턴 받음.
|
|
* @param
|
|
*/
|
|
nhn.husky.PopUpManager.getFunc = function(oOpenWin, sFunc) {
|
|
if (this._instance.getPluginWin().hasValue(oOpenWin)) {
|
|
var key = this._instance.getCorrectKey(this._instance.getPluginWin(), oOpenWin);
|
|
if (key) {
|
|
return this._instance.getPlugin().$(key)[sFunc]();
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
if(typeof window.nhn == 'undefined') { window.nhn = {}; }
|
|
if(!nhn.husky) { nhn.husky = {}; }
|
|
|
|
(function(){
|
|
// 구버전 jindo.$Agent polyfill
|
|
var ua = navigator.userAgent,
|
|
oAgent = jindo.$Agent(),
|
|
browser = oAgent.navigator(),
|
|
os = oAgent.os();
|
|
|
|
// [SMARTEDITORSUS-1795] 갤럭시노트 기본브라우저 구분을 위해 구분필드 추가
|
|
var aMatch = ua.match(/(SHW-|Chrome|Safari)/gi) || "";
|
|
if(aMatch.length === 2 && aMatch[0] === "SHW-" && aMatch[1] === "Safari"){
|
|
// 갤럭시노트 기본브라우저
|
|
browser.bGalaxyBrowser = true;
|
|
}else if(ua.indexOf("LG-V500") > -1 && ua.indexOf("Version/4.0") > -1){
|
|
// [SMARTEDITORSUS-1802] G패드 기본브라우저
|
|
browser.bGPadBrowser = true;
|
|
}
|
|
// [SMARTEDITORSUS-1860] iOS 버전 확인용
|
|
// os 에서 ios 여부 및 version 정보는 jindo2.3.0 부터 추가되었음
|
|
if(typeof os.ios === 'undefined'){
|
|
os.ios = ua.indexOf("iPad") > -1 || ua.indexOf("iPhone") > -1;
|
|
if(os.ios){
|
|
aMatch = ua.match(/(iPhone )?OS ([\d|_]+)/);
|
|
if(aMatch != null && aMatch[2] != undefined){
|
|
os.version = String(aMatch[2]).split("_").join(".");
|
|
}
|
|
}
|
|
}
|
|
})();
|
|
|
|
nhn.husky.SE2M_UtilPlugin = jindo.$Class({
|
|
name : "SE2M_UtilPlugin",
|
|
|
|
$BEFORE_MSG_APP_READY : function(){
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["oAgent", jindo.$Agent()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["oNavigator", jindo.$Agent().navigator()]);
|
|
this.oApp.exec("ADD_APP_PROPERTY", ["oUtils", this]);
|
|
},
|
|
|
|
$ON_REGISTER_HOTKEY : function(sHotkey, sCMD, aArgs, elTarget) {
|
|
this.oApp.exec("ADD_HOTKEY", [sHotkey, sCMD, aArgs, (elTarget || this.oApp.getWYSIWYGDocument())]);
|
|
},
|
|
|
|
$ON_SE2_ATTACH_HOVER_EVENTS : function(aElms){
|
|
this.oApp.exec("ATTACH_HOVER_EVENTS", [aElms, {fnElmToSrc: this._elm2Src, fnElmToTarget: this._elm2Target}]);
|
|
},
|
|
|
|
_elm2Src : function(el){
|
|
if(el.tagName == "LI" && el.firstChild && el.firstChild.tagName == "BUTTON"){
|
|
return el.firstChild;
|
|
}else{
|
|
return el;
|
|
}
|
|
},
|
|
|
|
_elm2Target : function(el){
|
|
if(el.tagName == "BUTTON" && el.parentNode.tagName == "LI"){
|
|
return el.parentNode;
|
|
}else{
|
|
return el;
|
|
}
|
|
},
|
|
|
|
getScrollXY : function(){
|
|
var scrollX,scrollY;
|
|
var oAppWindow = this.oApp.getWYSIWYGWindow();
|
|
if(typeof oAppWindow.scrollX == "undefined"){
|
|
scrollX = oAppWindow.document.documentElement.scrollLeft;
|
|
scrollY = oAppWindow.document.documentElement.scrollTop;
|
|
}else{
|
|
scrollX = oAppWindow.scrollX;
|
|
scrollY = oAppWindow.scrollY;
|
|
}
|
|
|
|
return {x:scrollX, y:scrollY};
|
|
}
|
|
});
|
|
|
|
nhn.husky.SE2M_Utils = {
|
|
sURLPattern : '(http|https|ftp|mailto):(?:\\/\\/)?((:?\\w|-)+(:?\\.(:?\\w|-)+)+)([^ <>]+)?',
|
|
rxDateFormat : /^(?:\d{4}\.)?\d{1,2}\.\d{1,2}$/,
|
|
_rxTable : /^(?:CAPTION|TBODY|THEAD|TFOOT|TR|TD|TH|COLGROUP|COL)$/i,
|
|
_rxSpaceOnly : /^\s+$/,
|
|
_rxFontStart : /<font(?:\s+[^>]*)?>/i,
|
|
_htFontSize : { // @see http://jerekdain.com/fontconversion.html
|
|
"1" : "7pt",
|
|
"2" : "10pt",
|
|
"3" : "12pt",
|
|
"4" : "13.5pt",
|
|
"5" : "18pt",
|
|
"6" : "24pt"
|
|
},
|
|
|
|
/**
|
|
* 유효하지 않은 노드가 TBODY, TR, TD 등등 사이에 있는지 판별한다.
|
|
* @param oNode {Node} 검사할 노드
|
|
* @return {Boolean} TBODY, TR, TD 등등 사이에 있는 유효하지 않은 노드인지 여부
|
|
*/
|
|
isInvalidNodeInTable : function(oNode){
|
|
// 해당노드가 table 관련노드들이 아닌데 table 관련노드들 사이에 있는지 검사한다.
|
|
// TODO: 좀 더 정확하게 검사할 필요가 있을까?
|
|
if(oNode && !this._rxTable.test(oNode.nodeName)){
|
|
var oTmpNode;
|
|
if((oTmpNode = oNode.previousSibling) && this._rxTable.test(oTmpNode.nodeName)){
|
|
return true;
|
|
}
|
|
if((oTmpNode = oNode.nextSibling) && this._rxTable.test(oTmpNode.nodeName)){
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1584][SMARTEDITORSUS-2237]
|
|
* TBODY, TR, TD 사이에 있는 유효하지 않은 노드이면 제거한다.
|
|
* @param oNode {Node} 검사할 노드
|
|
*/
|
|
removeInvalidNodeInTable : function(oNode){
|
|
if(this.isInvalidNodeInTable(oNode) && oNode.parentNode){
|
|
oNode.parentNode.removeChild(oNode);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2315] 유효하지 않은 폰트태그를 제거한다.
|
|
* @param {Element} el 태그정제를 제한할 상위 요소
|
|
*/
|
|
removeInvalidFont : function(el){
|
|
if(!el){
|
|
return;
|
|
}
|
|
|
|
this._removeInvalidFontInTable(el);
|
|
this._removeEmptyFont(el);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2237] IE11 에서 엑셀을 복사붙여넣기하면 유효하지 않은 위치(tbody, tr, td 사이)에 font 태그가 삽입되므로 제거
|
|
* @param {Element} el 태그제거를 제한할 상위 요소
|
|
*/
|
|
_removeInvalidFontInTable : function(el){
|
|
var aelFonts = jindo.$$("table font", el);
|
|
for(var i = 0, elFont; (elFont = aelFonts[i]); i++){
|
|
this.removeInvalidNodeInTable(elFont);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 빈 폰트태그를 제거한다.
|
|
* @param {Element} el 태그제거를 제한할 상위 요소
|
|
*/
|
|
_removeEmptyFont : function(el){
|
|
var aelFonts = jindo.$$("font", el);
|
|
for(var i = 0, elFont, sInner; (elFont = aelFonts[i]); i++){
|
|
sInner = elFont.innerHTML || "";
|
|
sInner = sInner.replace(this._rxSpaceOnly, "");
|
|
if (!sInner) {
|
|
elFont.parentNode.removeChild(elFont);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2315] font 태그를 span 으로 변환
|
|
* 포스트이슈[MUG-7757] 처리 로직 참고
|
|
* @param {Element} el 태그변환을 제한할 상위 요소
|
|
*/
|
|
convertFontToSpan : function(el){
|
|
if(!el){
|
|
return;
|
|
}
|
|
|
|
var oDoc = el.ownerDocument || document;
|
|
var aelTarget = jindo.$$("font", el);
|
|
|
|
for(var i = 0, elTarget, elSpan, sAttrValue; (elTarget = aelTarget[i]); i++){
|
|
elSpan = elTarget.parentNode;
|
|
if(elSpan.tagName !== "SPAN" || elSpan.childNodes.length > 1){
|
|
elSpan = oDoc.createElement("SPAN");
|
|
elTarget.parentNode.insertBefore(elSpan, elTarget);
|
|
}
|
|
|
|
sAttrValue = elTarget.getAttribute("face");
|
|
if(sAttrValue){
|
|
elSpan.style.fontFamily = sAttrValue;
|
|
}
|
|
sAttrValue = this._htFontSize[elTarget.getAttribute("size")];
|
|
if(sAttrValue){
|
|
elSpan.style.fontSize = sAttrValue;
|
|
}
|
|
sAttrValue = elTarget.getAttribute("color");
|
|
if(sAttrValue){
|
|
elSpan.style.color = sAttrValue;
|
|
}
|
|
|
|
// font태그 안쪽 내용을 span에 넣는다.
|
|
this._switchFontInnerToSpan(elTarget, elSpan);
|
|
|
|
if(elTarget.parentNode){
|
|
elTarget.parentNode.removeChild(elTarget);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* font태그 안쪽 내용을 span에 넣는다.
|
|
* @param {Element} elFont 대상FONT태그
|
|
* @param {Element} elSpan 빈SPAN태그
|
|
*/
|
|
_switchFontInnerToSpan : function(elFont, elSpan){
|
|
var sInnerHTML = elFont.innerHTML;
|
|
/**
|
|
* 폰트태그안에 폰트태그가 있을때 innerHTML으로 넣으면 안쪽 폰트태그는 span변환작업에서 누락될 수 있기 때문에
|
|
* 폰트태그가 중첩해서 있으면 appendChild를 이용하고 그렇지 않으면 innerHTML을 이용
|
|
*/
|
|
if(this._rxFontStart.test(sInnerHTML)){
|
|
for(var elChild; (elChild = elFont.firstChild);){
|
|
elSpan.appendChild(elChild);
|
|
}
|
|
}else{
|
|
elSpan.innerHTML = sInnerHTML;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 대상요소의 하위 요소들을 모두 밖으로 빼내고 대상요소를 제거한다.
|
|
* @param {Element} el 제거 대상 요소
|
|
*/
|
|
stripTag : function(el){
|
|
for(var elChild; (elChild = el.firstChild);){
|
|
el.parentNode.insertBefore(elChild, el);
|
|
}
|
|
el.parentNode.removeChild(el);
|
|
},
|
|
|
|
/**
|
|
* 노드 하위의 특정태그를 모두 제거한다.
|
|
* @param {Element} el 확인 대상 요소
|
|
* @param {String} sTagName 제거 대상 태그명 (대문자)
|
|
*/
|
|
stripTags :function(el, sTagName){
|
|
var aelTarget = jindo.$$(sTagName, el);
|
|
for(var i = 0, elTarget; (elTarget = aelTarget[i]); i++){
|
|
this.stripTag(elTarget);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2203] 날짜표기를 바로 잡는다. ex) 2015.10.13 -> 2015.10.13.
|
|
* @param {String} sDate 확인할 문자열
|
|
* @returns {String} 올바른 표기로 변환한 문자열
|
|
*/
|
|
reviseDateFormat : function(sDate){
|
|
if(sDate && sDate.replace){
|
|
sDate = sDate.replace(this.rxDateFormat, "$&.");
|
|
}
|
|
return sDate;
|
|
},
|
|
|
|
/**
|
|
* 사용자 클래스 정보를 추출한다.
|
|
* @param {String} sStr 추출 String
|
|
* @param {rx} rxValue rx type 형식의 값
|
|
* @param {String} sDivision value의 split 형식
|
|
* @return {Array}
|
|
*/
|
|
getCustomCSS : function(sStr, rxValue, sDivision) {
|
|
var ret = [];
|
|
if('undefined' == typeof(sStr) || 'undefined' == typeof(rxValue) || !sStr || !rxValue) {
|
|
return ret;
|
|
}
|
|
|
|
var aMatch = sStr.match(rxValue);
|
|
if(aMatch && aMatch[0]&&aMatch[1]) {
|
|
if(sDivision) {
|
|
ret = aMatch[1].split(sDivision);
|
|
} else {
|
|
ret[0] = aMatch[1];
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
},
|
|
/**
|
|
* HashTable로 구성된 Array의 같은 프로퍼티를 sSeperator 로 구분된 String 값으로 변환
|
|
* @param {Object} v
|
|
* @param {Object} sKey
|
|
* @author senxation
|
|
* @example
|
|
a = [{ b : "1" }, { b : "2" }]
|
|
|
|
toStringSamePropertiesOfArray(a, "b", ", ");
|
|
==> "1, 2"
|
|
*/
|
|
toStringSamePropertiesOfArray : function(v, sKey, sSeperator) {
|
|
if(!v){
|
|
return "";
|
|
}
|
|
if (v instanceof Array) {
|
|
var a = [];
|
|
for (var i = 0; i < v.length; i++) {
|
|
a.push(v[i][sKey]);
|
|
}
|
|
return a.join(",").replace(/,/g, sSeperator);
|
|
}
|
|
else {
|
|
if (typeof v[sKey] == "undefined") {
|
|
return "";
|
|
}
|
|
if (typeof v[sKey] == "string") {
|
|
return v[sKey];
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 단일 객체를 배열로 만들어줌
|
|
* @param {Object} v
|
|
* @return {Array}
|
|
* @author senxation
|
|
* @example
|
|
makeArray("test"); ==> ["test"]
|
|
*/
|
|
makeArray : function(v) {
|
|
if (v === null || typeof v === "undefined"){
|
|
return [];
|
|
}
|
|
if (v instanceof Array) {
|
|
return v;
|
|
}
|
|
var a = [];
|
|
a.push(v);
|
|
return a;
|
|
},
|
|
|
|
/**
|
|
* 말줄임을 할때 줄일 내용과 컨테이너가 다를 경우 처리
|
|
* 컨테이너의 css white-space값이 "normal"이어야한다. (컨테이너보다 텍스트가 길면 여러행으로 표현되는 상태)
|
|
* @param {HTMLElement} elText 말줄임할 엘리먼트
|
|
* @param {HTMLElement} elContainer 말줄임할 엘리먼트를 감싸는 컨테이너
|
|
* @param {String} sStringTail 말줄임을 표현할 문자열 (미지정시 ...)
|
|
* @param {Number} nLine 최대 라인수 (미지정시 1)
|
|
* @author senxation
|
|
* @example
|
|
//div가 2줄 이하가 되도록 strong 내부의 내용을 줄임
|
|
<div>
|
|
<strong id="a">말줄임을적용할내용말줄임을적용할내용말줄임을적용할내용</strong><span>상세보기</span>
|
|
<div>
|
|
ellipsis(jindo.$("a"), jindo.$("a").parentNode, "...", 2);
|
|
*/
|
|
ellipsis : function(elText, elContainer, sStringTail, nLine) {
|
|
sStringTail = sStringTail || "...";
|
|
if (typeof nLine == "undefined") {
|
|
nLine = 1;
|
|
}
|
|
var welText = jindo.$Element(elText);
|
|
var welContainer = jindo.$Element(elContainer);
|
|
|
|
var sText = welText.html();
|
|
var nLength = sText.length;
|
|
var nCurrentHeight = welContainer.height();
|
|
var nIndex = 0;
|
|
welText.html('A');
|
|
var nHeight = welContainer.height();
|
|
|
|
if (nCurrentHeight < nHeight * (nLine + 0.5)) {
|
|
return welText.html(sText);
|
|
}
|
|
|
|
/**
|
|
* 지정된 라인보다 커질때까지 전체 남은 문자열의 절반을 더해나감
|
|
*/
|
|
nCurrentHeight = nHeight;
|
|
while(nCurrentHeight < nHeight * (nLine + 0.5)) {
|
|
nIndex += Math.max(Math.ceil((nLength - nIndex)/2), 1);
|
|
welText.html(sText.substring(0, nIndex) + sStringTail);
|
|
nCurrentHeight = welContainer.height();
|
|
}
|
|
|
|
/**
|
|
* 지정된 라인이 될때까지 한글자씩 잘라냄
|
|
*/
|
|
while(nCurrentHeight > nHeight * (nLine + 0.5)) {
|
|
nIndex--;
|
|
welText.html(sText.substring(0, nIndex) + sStringTail);
|
|
nCurrentHeight = welContainer.height();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 최대 가로사이즈를 지정하여 말줄임한다.
|
|
* elText의 css white-space값이 "nowrap"이어야한다. (컨테이너보다 텍스트가 길면 행변환되지않고 가로로 길게 표현되는 상태)
|
|
* @param {HTMLElement} elText 말줄임할 엘리먼트
|
|
* @param {String} sStringTail 말줄임을 표현할 문자열 (미지정시 ...)
|
|
* @param {Function} fCondition 조건 함수. 내부에서 true를 리턴하는 동안에만 말줄임을 진행한다.
|
|
* @author senxation
|
|
* @example
|
|
//150픽셀 이하가 되도록 strong 내부의 내용을 줄임
|
|
<strong id="a">말줄임을적용할내용말줄임을적용할내용말줄임을적용할내용</strong>>
|
|
ellipsisByPixel(jindo.$("a"), "...", 150);
|
|
*/
|
|
ellipsisByPixel : function(elText, sStringTail, nPixel, fCondition) {
|
|
sStringTail = sStringTail || "...";
|
|
var welText = jindo.$Element(elText);
|
|
var nCurrentWidth = welText.width();
|
|
if (nCurrentWidth < nPixel) {
|
|
return;
|
|
}
|
|
|
|
var sText = welText.html();
|
|
var nLength = sText.length;
|
|
|
|
var nIndex = 0;
|
|
if (typeof fCondition == "undefined") {
|
|
var nWidth = welText.html('A').width();
|
|
nCurrentWidth = nWidth;
|
|
|
|
while(nCurrentWidth < nPixel) {
|
|
nIndex += Math.max(Math.ceil((nLength - nIndex)/2), 1);
|
|
welText.html(sText.substring(0, nIndex) + sStringTail);
|
|
nCurrentWidth = welText.width();
|
|
}
|
|
|
|
fCondition = function() {
|
|
return true;
|
|
};
|
|
}
|
|
|
|
nIndex = welText.html().length - sStringTail.length;
|
|
|
|
while(nCurrentWidth > nPixel) {
|
|
if (!fCondition()) {
|
|
break;
|
|
}
|
|
nIndex--;
|
|
welText.html(sText.substring(0, nIndex) + sStringTail);
|
|
nCurrentWidth = welText.width();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 여러개의 엘리먼트를 각각의 지정된 최대너비로 말줄임한다.
|
|
* 말줄임할 엘리먼트의 css white-space값이 "nowrap"이어야한다. (컨테이너보다 텍스트가 길면 행변환되지않고 가로로 길게 표현되는 상태)
|
|
* @param {Array} aElement 말줄임할 엘리먼트의 배열. 지정된 순서대로 말줄임한다.
|
|
* @param {String} sStringTail 말줄임을 표현할 문자열 (미지정시 ...)
|
|
* @param {Array} aMinWidth 말줄임할 너비의 배열.
|
|
* @param {Function} fCondition 조건 함수. 내부에서 true를 리턴하는 동안에만 말줄임을 진행한다.
|
|
* @example
|
|
//#a #b #c의 너비를 각각 100, 50, 50픽셀로 줄임 (div#parent 가 200픽셀 이하이면 중단)
|
|
//#c의 너비를 줄이는 동안 fCondition에서 false를 리턴하면 b, a는 말줄임 되지 않는다.
|
|
<div id="parent">
|
|
<strong id="a">말줄임을적용할내용</strong>
|
|
<strong id="b">말줄임을적용할내용</strong>
|
|
<strong id="c">말줄임을적용할내용</strong>
|
|
<div>
|
|
ellipsisElementsToDesinatedWidth([jindo.$("c"), jindo.$("b"), jindo.$("a")], "...", [100, 50, 50], function(){
|
|
if (jindo.$Element("parent").width() > 200) {
|
|
return true;
|
|
}
|
|
return false;
|
|
});
|
|
*/
|
|
ellipsisElementsToDesinatedWidth : function(aElement, sStringTail, aMinWidth, fCondition) {
|
|
jindo.$A(aElement).forEach(function(el, i){
|
|
if (!el) {
|
|
jindo.$A.Continue();
|
|
}
|
|
nhn.husky.SE2M_Utils.ellipsisByPixel(el, sStringTail, aMinWidth[i], fCondition);
|
|
});
|
|
},
|
|
|
|
/**
|
|
* 숫자를 입력받아 정해진 길이만큼 앞에 "0"이 추가된 문자열을 구한다.
|
|
* @param {Number} nNumber
|
|
* @param {Number} nLength
|
|
* @return {String}
|
|
* @example
|
|
paddingZero(10, 5); ==> "00010" (String)
|
|
*/
|
|
paddingZero : function(nNumber, nLength) {
|
|
var sResult = nNumber.toString();
|
|
while (sResult.length < nLength) {
|
|
sResult = ("0" + sResult);
|
|
}
|
|
return sResult;
|
|
},
|
|
|
|
/**
|
|
* string을 byte 단위로 짤라서 tail를 붙힌다.
|
|
* @param {String} sString
|
|
* @param {Number} nByte
|
|
* @param {String} sTail
|
|
* @example
|
|
cutStringToByte('일이삼사오육', 6, '...') ==> '일이삼...' (string)
|
|
*/
|
|
cutStringToByte : function(sString, nByte, sTail){
|
|
if(sString === null || sString.length === 0) {
|
|
return sString;
|
|
}
|
|
|
|
sString = sString.replace(/ +/g, " ");
|
|
if (!sTail && sTail != "") {
|
|
sTail = "...";
|
|
}
|
|
|
|
var maxByte = nByte;
|
|
var n=0;
|
|
var nLen = sString.length;
|
|
for(var i=0; i<nLen;i++){
|
|
n += this.getCharByte(sString.charAt(i));
|
|
if(n == maxByte){
|
|
if(i == nLen-1) {
|
|
return sString;
|
|
} else {
|
|
return sString.substring(0,i)+sTail;
|
|
}
|
|
} else if( n > maxByte ) {
|
|
return sString.substring(0, i)+sTail;
|
|
}
|
|
}
|
|
return sString;
|
|
},
|
|
|
|
/**
|
|
* 입력받은 문자의 byte 구한다.
|
|
* @param {String} ch
|
|
*
|
|
*/
|
|
getCharByte : function(ch){
|
|
if (ch === null || ch.length < 1) {
|
|
return 0;
|
|
}
|
|
|
|
var byteSize = 0;
|
|
var str = escape(ch);
|
|
|
|
if ( str.length == 1 ) { // when English then 1byte
|
|
byteSize ++;
|
|
} else if ( str.indexOf("%u") != -1 ) { // when Korean then 2byte
|
|
byteSize += 2;
|
|
} else if ( str.indexOf("%") != -1 ) { // else 3byte
|
|
byteSize += str.length/3;
|
|
}
|
|
return byteSize;
|
|
},
|
|
|
|
/**
|
|
* Hash Table에서 원하는 키값만을 가지는 필터된 새로운 Hash Table을 구한다.
|
|
* @param {HashTable} htUnfiltered
|
|
* @param {Array} aKey
|
|
* @return {HashTable}
|
|
* @author senxation
|
|
* @example
|
|
getFilteredHashTable({
|
|
a : 1,
|
|
b : 2,
|
|
c : 3,
|
|
d : 4
|
|
}, ["a", "c"]);
|
|
==> { a : 1, c : 3 }
|
|
*/
|
|
getFilteredHashTable : function(htUnfiltered, vKey) {
|
|
if (!(vKey instanceof Array)) {
|
|
return arguments.callee.call(this, htUnfiltered, [ vKey ]);
|
|
}
|
|
|
|
var waKey = jindo.$A(vKey);
|
|
return jindo.$H(htUnfiltered).filter(function(vValue, sKey){
|
|
if (waKey.has(sKey) && vValue) {
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}).$value();
|
|
},
|
|
|
|
isBlankNode : function(elNode){
|
|
var isBlankTextNode = this.isBlankTextNode;
|
|
|
|
var bEmptyContent = function(elNode){
|
|
if(!elNode) {
|
|
return true;
|
|
}
|
|
|
|
if(isBlankTextNode(elNode)){
|
|
return true;
|
|
}
|
|
|
|
if(elNode.tagName == "BR") {
|
|
return true;
|
|
}
|
|
|
|
if(elNode.innerHTML == " " || elNode.innerHTML == "") {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
var bEmptyP = function(elNode){
|
|
if(elNode.tagName == "IMG" || elNode.tagName == "IFRAME"){
|
|
return false;
|
|
}
|
|
|
|
if(bEmptyContent(elNode)){
|
|
return true;
|
|
}
|
|
|
|
if(elNode.tagName == "P"){
|
|
for(var i=elNode.childNodes.length-1; i>=0; i--){
|
|
var elTmp = elNode.childNodes[i];
|
|
if(isBlankTextNode(elTmp)){
|
|
elTmp.parentNode.removeChild(elTmp);
|
|
}
|
|
}
|
|
|
|
if(elNode.childNodes.length == 1){
|
|
if(elNode.firstChild.tagName == "IMG" || elNode.firstChild.tagName == "IFRAME"){
|
|
return false;
|
|
}
|
|
if(bEmptyContent(elNode.firstChild)){
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
if(bEmptyP(elNode)){
|
|
return true;
|
|
}
|
|
|
|
for(var i=0, nLen=elNode.childNodes.length; i<nLen; i++){
|
|
var elTmp = elNode.childNodes[i];
|
|
if(!bEmptyP(elTmp)){
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
},
|
|
|
|
isBlankTextNode : function(oNode){
|
|
var sText;
|
|
|
|
if(oNode.nodeType == 3){
|
|
sText = oNode.nodeValue;
|
|
sText = sText.replace(unescape("%uFEFF"), '');
|
|
|
|
if(sText == "") {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
isFirstChildOfNode : function(sTagName, sParentTagName, elNode){
|
|
if(!elNode){
|
|
return false;
|
|
}
|
|
|
|
if(elNode.tagName == sParentTagName && elNode.firstChild.tagName == sTagName){
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* elNode의 상위 노드 중 태그명이 sTagName과 일치하는 것이 있다면 반환.
|
|
* @param {String} sTagName 검색 할 태그명(반드시 대문자를 사용할 것)
|
|
* @param {HTMLElement} elNode 검색 시작점으로 사용 할 노드
|
|
* @return {HTMLElement} 부모 노드 중 태그명이 sTagName과 일치하는 노드. 없을 경우 null 반환
|
|
*/
|
|
findAncestorByTagName : function(sTagName, elNode){
|
|
while(elNode && elNode.tagName != sTagName) {
|
|
elNode = elNode.parentNode;
|
|
}
|
|
|
|
return elNode;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1735]
|
|
* elNode의 상위 노드 중 태그명이 sTagName과 일치하는 것이 있다면
|
|
* 해당 노드와 재귀 탐색 횟수가 담긴 객체를 반환.
|
|
* @param {String} sTagName 검색 할 태그명(반드시 대문자를 사용할 것)
|
|
* @param {HTMLElement} elNode 검색 시작점으로 사용 할 노드
|
|
* @return {Object}
|
|
* {HTMLElement} Object.elNode 부모 노드 중 태그명이 sTagName과 일치하는 노드. 없을 경우 null 반환
|
|
* {Number} Object.nRecursiveCount 재귀 탐색 횟수
|
|
*/
|
|
findAncestorByTagNameWithCount : function(sTagName, elNode){
|
|
var nRecursiveCount = 0;
|
|
var htResult = {};
|
|
|
|
while(elNode && elNode.tagName != sTagName) {
|
|
elNode = elNode.parentNode;
|
|
nRecursiveCount += 1;
|
|
}
|
|
|
|
htResult = {
|
|
elNode : elNode,
|
|
nRecursiveCount : nRecursiveCount
|
|
}
|
|
|
|
return htResult;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1672] elNode의 상위 노드 중 태그명이 aTagName 의 요소 중 하나와 일치하는 것이 있다면 반환.
|
|
* @param {String} aTagName 검색 할 태그명이 담긴 배열
|
|
* @param {HTMLElement} elNode 검색 시작점으로 사용 할 노드
|
|
* @return {HTMLElement} 부모 노드 중 태그명이 aTagName의 요소 중 하나와 일치하는 노드. 없을 경우 null 반환
|
|
*/
|
|
findClosestAncestorAmongTagNames : function(aTagName, elNode){
|
|
var rxTagNames = new RegExp("^(" + aTagName.join("|") + ")$", "i");
|
|
|
|
while(elNode && !rxTagNames.test(elNode.tagName)){
|
|
elNode = elNode.parentNode;
|
|
}
|
|
|
|
return elNode;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1735]
|
|
* elNode의 상위 노드 중 태그명이 aTagName 의 요소 중 하나와 일치하는 것이 있다면
|
|
* 해당 노드와 재귀 탐색 횟수가 담긴 객체를 반환.
|
|
* @param {String} aTagName 검색 할 태그명이 담긴 배열
|
|
* @param {HTMLElement} elNode 검색 시작점으로 사용 할 노드
|
|
* @return {Object}
|
|
* {HTMLElement} Object.elNode 부모 노드 중 태그명이 aTagName의 요소 중 하나와 일치하는 노드. 없을 경우 null 반환
|
|
* {Number} Object.nRecursiveCount 재귀 탐색 횟수
|
|
*/
|
|
findClosestAncestorAmongTagNamesWithCount : function(aTagName, elNode){
|
|
var nRecursiveCount = 0;
|
|
var htResult = {};
|
|
|
|
var rxTagNames = new RegExp("^(" + aTagName.join("|") + ")$", "i");
|
|
|
|
while(elNode && !rxTagNames.test(elNode.tagName)){
|
|
elNode = elNode.parentNode;
|
|
nRecursiveCount += 1;
|
|
}
|
|
|
|
htResult = {
|
|
elNode : elNode,
|
|
nRecursiveCount : nRecursiveCount
|
|
}
|
|
|
|
return htResult;
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2136] 대상이 Number인지 확인한다.
|
|
*
|
|
* @param {Number} n 판별 대상
|
|
* @return {Boolean} 대상이 Number이다.
|
|
* @see http://stackoverflow.com/questions/18082/validate-decimal-numbers-in-javascript-isnumeric/1830844#1830844
|
|
* */
|
|
isNumber : function(n){
|
|
return !isNaN(parseFloat(n)) && isFinite(n);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2136] 객체의 property를 delete 연산자로 제거하는 polyfill
|
|
*
|
|
* @param {Object} oTarget 대상 객체
|
|
* @param {String} sProp property 명
|
|
*
|
|
* @example nhn.husky.SE2M_Utils.deleteProperty(elTable, 'propA'); // elTable.propA를 delete
|
|
*
|
|
* @see http://stackoverflow.com/questions/1073414/deleting-a-window-property-in-ie
|
|
* */
|
|
deleteProperty : function(oTarget, sProp){
|
|
if((typeof(oTarget) !== 'object') || (typeof(sProp) !== 'string') || (typeof(oTarget[sProp]) === 'undefined')){
|
|
return;
|
|
}
|
|
|
|
oTarget[sProp] = undefined;
|
|
try{
|
|
delete oTarget[sProp];
|
|
}catch(e){
|
|
// [IE 8-]
|
|
}
|
|
},
|
|
|
|
loadCSS : function(url, fnCallback){
|
|
var oDoc = document;
|
|
var elHead = oDoc.getElementsByTagName("HEAD")[0];
|
|
var elStyle = oDoc.createElement ("LINK");
|
|
elStyle.setAttribute("type", "text/css");
|
|
elStyle.setAttribute("rel", "stylesheet");
|
|
elStyle.setAttribute("href", url);
|
|
if(fnCallback){
|
|
if ('onload' in elStyle) {
|
|
elStyle.onload = function(){
|
|
fnCallback();
|
|
// [SMARTEDITORSUS-2032] IE10호환성보기(IE7) 에서 onload 이벤트가 2번 발생할 수 있는 가능성이 있기 때문에
|
|
// 콜백실행 후 onload 이벤트 핸들러를 제거
|
|
this.onload = null;
|
|
};
|
|
} else {
|
|
elStyle.onreadystatechange = function(){
|
|
if(elStyle.readyState != "complete"){
|
|
return;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-308] [IE9] 응답이 304인 경우
|
|
// onreadystatechage 핸들러에서 readyState 가 complete 인 경우가 두 번 발생
|
|
// LINK 엘리먼트의 속성으로 콜백 실행 여부를 플래그로 남겨놓아 처리함
|
|
if(elStyle.getAttribute("_complete")){
|
|
return;
|
|
}
|
|
|
|
elStyle.setAttribute("_complete", true);
|
|
|
|
fnCallback();
|
|
};
|
|
}
|
|
}
|
|
elHead.appendChild (elStyle);
|
|
},
|
|
|
|
getUniqueId : function(sPrefix) {
|
|
return (sPrefix || '') + jindo.$Date().time() + (Math.random() * 100000).toFixed();
|
|
},
|
|
|
|
/**
|
|
* @param {Object} oSrc value copy할 object
|
|
* @return {Object}
|
|
* @example
|
|
* var oSource = [1, 3, 4, { a:1, b:2, c: { a:1 }}];
|
|
var oTarget = oSource; // call by reference
|
|
oTarget = nhn.husky.SE2M_Utils.clone(oSource);
|
|
|
|
oTarget[1] = 2;
|
|
oTarget[3].a = 100;
|
|
console.log(oSource); // check for deep copy
|
|
console.log(oTarget, oTarget instanceof Object); // check instance type!
|
|
*/
|
|
clone : function(oSrc, oChange) {
|
|
if ('undefined' != typeof(oSrc) && !!oSrc && (oSrc.constructor == Array || oSrc.constructor == Object)) {
|
|
var oCopy = (oSrc.constructor == Array ? [] : {} );
|
|
for (var property in oSrc) {
|
|
if ('undefined' != typeof(oChange) && !!oChange[property]) {
|
|
oCopy[property] = arguments.callee(oChange[property]);
|
|
} else {
|
|
oCopy[property] = arguments.callee(oSrc[property]);
|
|
}
|
|
}
|
|
|
|
return oCopy;
|
|
}
|
|
|
|
return oSrc;
|
|
},
|
|
|
|
getHtmlTagAttr : function(sHtmlTag, sAttr) {
|
|
var rx = new RegExp('\\s' + sAttr + "=('([^']*)'|\"([^\"]*)\"|([^\"' >]*))", 'i');
|
|
var aResult = rx.exec(sHtmlTag);
|
|
|
|
if (!aResult) {
|
|
return '';
|
|
}
|
|
|
|
var sAttrTmp = (aResult[1] || aResult[2] || aResult[3]); // for chrome 5.x bug!
|
|
if (!!sAttrTmp) {
|
|
sAttrTmp = sAttrTmp.replace(/[\"]/g, '');
|
|
}
|
|
|
|
return sAttrTmp;
|
|
},
|
|
|
|
|
|
/**
|
|
* iframe 영역의 aling 정보를 다시 세팅하는 부분.
|
|
* iframe 형태의 산출물을 에디터에 삽입 이후에 에디터 정렬기능을 추가 하였을때 ir_to_db 이전 시점에서 div태그에 정렬을 넣어주는 로직임.
|
|
* 브라우저 형태에 따라 정렬 태그가 iframe을 감싸는 div 혹은 p 태그에 정렬이 추가된다.
|
|
* @param {HTMLElement} el iframe의 parentNode
|
|
* @param {Document} oDoc document
|
|
*/
|
|
// [COM-1151] SE2M_PreStringConverter 에서 수정하도록 변경
|
|
iframeAlignConverter : function(el, oDoc){
|
|
var sTagName = el.tagName.toUpperCase();
|
|
|
|
if(sTagName == "DIV" || sTagName == 'P'){
|
|
//irToDbDOM 에서 최상위 노드가 div 엘리먼트 이므로 parentNode가 없으면 최상의 div 노드 이므로 리턴한다.
|
|
if(el.parentNode === null ){
|
|
return;
|
|
}
|
|
var elWYSIWYGDoc = oDoc;
|
|
var wel = jindo.$Element(el);
|
|
var sHtml = wel.html();
|
|
//현재 align을 얻어오기.
|
|
var sAlign = jindo.$Element(el).attr('align') || jindo.$Element(el).css('text-align');
|
|
//if(!sAlign){ // P > DIV의 경우 문제 발생, 수정 화면에 들어 왔을 때 태그 깨짐
|
|
// return;
|
|
//}
|
|
//새로운 div 노드 생성한다.
|
|
var welAfter = jindo.$Element(jindo.$('<div></div>', elWYSIWYGDoc));
|
|
welAfter.html(sHtml).attr('align', sAlign);
|
|
wel.replace(welAfter);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* jindo.$JSON.fromXML을 변환한 메서드.
|
|
* 소숫점이 있는 경우의 처리 시에 숫자로 변환하지 않도록 함(parseFloat 사용 안하도록 수정)
|
|
* 관련 BTS : [COM-1093]
|
|
* @param {String} sXML XML 형태의 문자열
|
|
* @return {jindo.$JSON}
|
|
*/
|
|
getJsonDatafromXML : function(sXML) {
|
|
var o = {};
|
|
var re = /\s*<(\/?[\w:\-]+)((?:\s+[\w:\-]+\s*=\s*(?:"(?:\\"|[^"])*"|'(?:\\'|[^'])*'))*)\s*((?:\/>)|(?:><\/\1>|\s*))|\s*<!\[CDATA\[([\w\W]*?)\]\]>\s*|\s*>?([^<]*)/ig;
|
|
var re2= /^[0-9]+(?:\.[0-9]+)?$/;
|
|
var re3= /^\s+$/g;
|
|
var ec = {"&":"&"," ":" ",""":"\"","<":"<",">":">"};
|
|
var fg = {tags:["/"],stack:[o]};
|
|
var es = function(s){
|
|
if (typeof s == "undefined") {
|
|
return "";
|
|
}
|
|
return s.replace(/&[a-z]+;/g, function(m){ return (typeof ec[m] == "string")?ec[m]:m; });
|
|
};
|
|
var at = function(s,c) {
|
|
s.replace(/([\w\:\-]+)\s*=\s*(?:"((?:\\"|[^"])*)"|'((?:\\'|[^'])*)')/g, function($0,$1,$2,$3) {
|
|
c[$1] = es(($2?$2.replace(/\\"/g,'"'):undefined)||($3?$3.replace(/\\'/g,"'"):undefined));
|
|
});
|
|
};
|
|
|
|
var em = function(o) {
|
|
for(var x in o){
|
|
if (o.hasOwnProperty(x)) {
|
|
if(Object.prototype[x]) {
|
|
continue;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
// $0 : 전체
|
|
// $1 : 태그명
|
|
// $2 : 속성문자열
|
|
// $3 : 닫는태그
|
|
// $4 : CDATA바디값
|
|
// $5 : 그냥 바디값
|
|
var cb = function($0,$1,$2,$3,$4,$5) {
|
|
var cur, cdata = "";
|
|
var idx = fg.stack.length - 1;
|
|
|
|
if (typeof $1 == "string" && $1) {
|
|
if ($1.substr(0,1) != "/") {
|
|
var has_attr = (typeof $2 == "string" && $2);
|
|
var closed = (typeof $3 == "string" && $3);
|
|
var newobj = (!has_attr && closed)?"":{};
|
|
|
|
cur = fg.stack[idx];
|
|
|
|
if (typeof cur[$1] == "undefined") {
|
|
cur[$1] = newobj;
|
|
cur = fg.stack[idx+1] = cur[$1];
|
|
} else if (cur[$1] instanceof Array) {
|
|
var len = cur[$1].length;
|
|
cur[$1][len] = newobj;
|
|
cur = fg.stack[idx+1] = cur[$1][len];
|
|
} else {
|
|
cur[$1] = [cur[$1], newobj];
|
|
cur = fg.stack[idx+1] = cur[$1][1];
|
|
}
|
|
|
|
if (has_attr) {
|
|
at($2,cur);
|
|
}
|
|
|
|
fg.tags[idx+1] = $1;
|
|
|
|
if (closed) {
|
|
fg.tags.length--;
|
|
fg.stack.length--;
|
|
}
|
|
} else {
|
|
fg.tags.length--;
|
|
fg.stack.length--;
|
|
}
|
|
} else if (typeof $4 == "string" && $4) {
|
|
cdata = $4;
|
|
} else if (typeof $5 == "string" && $5.replace(re3, "")) { // [SMARTEDITORSUS-1525] 닫는 태그인데 공백문자가 들어있어 cdata 값을 덮어쓰는 경우 방지
|
|
cdata = es($5);
|
|
}
|
|
|
|
if (cdata.length > 0) {
|
|
var par = fg.stack[idx-1];
|
|
var tag = fg.tags[idx];
|
|
|
|
if (re2.test(cdata)) {
|
|
//cdata = parseFloat(cdata);
|
|
}else if (cdata == "true" || cdata == "false"){
|
|
cdata = new Boolean(cdata);
|
|
}
|
|
|
|
if(typeof par =='undefined') {
|
|
return;
|
|
}
|
|
|
|
if (par[tag] instanceof Array) {
|
|
var o = par[tag];
|
|
if (typeof o[o.length-1] == "object" && !em(o[o.length-1])) {
|
|
o[o.length-1].$cdata = cdata;
|
|
o[o.length-1].toString = function(){ return cdata; };
|
|
} else {
|
|
o[o.length-1] = cdata;
|
|
}
|
|
} else {
|
|
if (typeof par[tag] == "object" && !em(par[tag])) {
|
|
par[tag].$cdata = cdata;
|
|
par[tag].toString = function() { return cdata; };
|
|
} else {
|
|
par[tag] = cdata;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
sXML = sXML.replace(/<(\?|\!-)[^>]*>/g, "");
|
|
sXML.replace(re, cb);
|
|
|
|
return jindo.$Json(o);
|
|
},
|
|
/**
|
|
* 문자열내 자주 사용되는 특수문자 5개 (", ', &, <, >)를 HTML Entity Code 로 변경하여 반환
|
|
* @see http://www.w3.org/TR/html4/charset.html#entities
|
|
* @param {String} sString 원본 문자열
|
|
* @returns {String} 변경된 문자열
|
|
* @example
|
|
* replaceSpecialChar() or replaceSpecialChar(123)
|
|
* // 결과: ""
|
|
*
|
|
* replaceSpecialChar("", ', &, <, >")
|
|
* // 결과: &quot;, &#39;, &amp;, &lt;, &gt;
|
|
*/
|
|
replaceSpecialChar : function(sString){
|
|
return (typeof(sString) == "string") ? (sString.replace(/\&/g, "&").replace(/\"/g, """).replace(/\'/g, "'").replace(/</g, "<").replace(/\>/g, ">")) : "";
|
|
},
|
|
/**
|
|
* 문자열내 자주 사용되는 HTML Entity Code 5개를 원래 문자로 (", ', &, <, >)로 변경하여 반환
|
|
* @see http://www.w3.org/TR/html4/charset.html#entities
|
|
* @param {String} sString 원본 문자열
|
|
* @returns {String} 변경된 문자열
|
|
* @example
|
|
* restoreSpecialChar() or restoreSpecialChar(123)
|
|
* // 결과: ""
|
|
*
|
|
* restoreSpecialChar("&quot;, &#39;, &amp;, &lt;, &gt;")
|
|
* // 결과: ", ', &, <, >
|
|
*/
|
|
restoreSpecialChar : function(sString){
|
|
return (typeof(sString) == "string") ? (sString.replace(/"/g, "\"").replace(/'/g, "'").replace(/</g, "<").replace(/>/g, ">").replace(/&/g, "&")) : "";
|
|
}
|
|
};
|
|
|
|
/**
|
|
* nhn.husky.AutoResizer
|
|
* HTML모드와 TEXT 모드의 편집 영역인 TEXTAREA에 대한 자동확장 처리
|
|
*/
|
|
nhn.husky.AutoResizer = jindo.$Class({
|
|
welHiddenDiv : null,
|
|
welCloneDiv : null,
|
|
elContainer : null,
|
|
$init : function(el, htOption){
|
|
var aCopyStyle = [
|
|
'lineHeight', 'textDecoration', 'letterSpacing',
|
|
'fontSize', 'fontFamily', 'fontStyle', 'fontWeight',
|
|
'textTransform', 'textAlign', 'direction', 'wordSpacing', 'fontSizeAdjust',
|
|
'paddingTop', 'paddingLeft', 'paddingBottom', 'paddingRight', 'width'
|
|
],
|
|
i = aCopyStyle.length,
|
|
oCss = {
|
|
"position" : "absolute",
|
|
"top" : -9999,
|
|
"left" : -9999,
|
|
"opacity": 0,
|
|
"overflow": "hidden",
|
|
"wordWrap" : "break-word"
|
|
};
|
|
|
|
this.nMinHeight = htOption.nMinHeight;
|
|
this.wfnCallback = htOption.wfnCallback;
|
|
|
|
this.elContainer = el.parentNode;
|
|
this.welTextArea = jindo.$Element(el); // autoresize를 적용할 TextArea
|
|
this.welHiddenDiv = jindo.$Element('<div>');
|
|
|
|
this.wfnResize = jindo.$Fn(this._resize, this);
|
|
|
|
this.sOverflow = this.welTextArea.css("overflow");
|
|
this.welTextArea.css("overflow", "hidden");
|
|
|
|
while(i--){
|
|
oCss[aCopyStyle[i]] = this.welTextArea.css(aCopyStyle[i]);
|
|
}
|
|
|
|
this.welHiddenDiv.css(oCss);
|
|
|
|
this.nLastHeight = this.welTextArea.height();
|
|
},
|
|
bind : function(){
|
|
this.welCloneDiv = jindo.$Element(this.welHiddenDiv.$value().cloneNode(false));
|
|
|
|
this.wfnResize.attach(this.welTextArea, "keyup");
|
|
this.welCloneDiv.appendTo(this.elContainer);
|
|
|
|
this._resize();
|
|
},
|
|
unbind : function(){
|
|
this.wfnResize.detach(this.welTextArea, "keyup");
|
|
this.welTextArea.css("overflow", this.sOverflow);
|
|
|
|
if(this.welCloneDiv){
|
|
this.welCloneDiv.leave();
|
|
}
|
|
},
|
|
_resize : function(){
|
|
var sContents = this.welTextArea.$value().value,
|
|
bExpand = false,
|
|
nHeight;
|
|
|
|
if(sContents === this.sContents){
|
|
return;
|
|
}
|
|
|
|
this.sContents = sContents.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>').replace(/ /g, ' ').replace(/\n/g, '<br>');
|
|
this.sContents += "<br>"; // 마지막 개행 뒤에 <br>을 더 붙여주어야 늘어나는 높이가 동일함
|
|
|
|
this.welCloneDiv.html(this.sContents);
|
|
nHeight = this.welCloneDiv.height();
|
|
|
|
if(nHeight < this.nMinHeight){
|
|
nHeight = this.nMinHeight;
|
|
}
|
|
|
|
this.welTextArea.css("height", nHeight + "px");
|
|
this.elContainer.style.height = nHeight + "px";
|
|
|
|
if(this.nLastHeight < nHeight){
|
|
bExpand = true;
|
|
}
|
|
|
|
this.wfnCallback(bExpand);
|
|
}
|
|
});
|
|
|
|
/**
|
|
* 문자를 연결하는 '+' 대신에 java와 유사하게 처리하도록 문자열 처리하도록 만드는 object
|
|
* @author nox
|
|
* @example
|
|
var sTmp1 = new StringBuffer();
|
|
sTmp1.append('1').append('2').append('3');
|
|
|
|
var sTmp2 = new StringBuffer('1');
|
|
sTmp2.append('2').append('3');
|
|
|
|
var sTmp3 = new StringBuffer('1').append('2').append('3');
|
|
*/
|
|
if ('undefined' != typeof(StringBuffer)) {
|
|
StringBuffer = {};
|
|
}
|
|
|
|
StringBuffer = function(str) {
|
|
this._aString = [];
|
|
if ('undefined' != typeof(str)) {
|
|
this.append(str);
|
|
}
|
|
};
|
|
|
|
StringBuffer.prototype.append = function(str) {
|
|
this._aString.push(str);
|
|
return this;
|
|
};
|
|
|
|
StringBuffer.prototype.toString = function() {
|
|
return this._aString.join('');
|
|
};
|
|
|
|
StringBuffer.prototype.setLength = function(nLen) {
|
|
if('undefined' == typeof(nLen) || 0 >= nLen) {
|
|
this._aString.length = 0;
|
|
} else {
|
|
this._aString.length = nLen;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Installed Font Detector
|
|
* @author hooriza
|
|
*
|
|
* @see http://remysharp.com/2008/07/08/how-to-detect-if-a-font-is-installed-only-using-javascript/
|
|
*/
|
|
|
|
(function() {
|
|
|
|
var oDummy = null, rx = /,/gi;
|
|
|
|
IsInstalledFont = function(sFont) {
|
|
|
|
var sDefFont = sFont == 'Comic Sans MS' ? 'Courier New' : 'Comic Sans MS';
|
|
if (!oDummy) {
|
|
oDummy = document.createElement('div');
|
|
}
|
|
|
|
var sStyle = 'position:absolute !important; font-size:200px !important; left:-9999px !important; top:-9999px !important;';
|
|
oDummy.innerHTML = 'mmmmiiiii'+unescape('%uD55C%uAE00');
|
|
oDummy.style.cssText = sStyle + 'font-family:"' + sDefFont + '" !important';
|
|
|
|
var elBody = document.body || document.documentElement;
|
|
if(elBody.firstChild){
|
|
elBody.insertBefore(oDummy, elBody.firstChild);
|
|
}else{
|
|
document.body.appendChild(oDummy);
|
|
}
|
|
|
|
var sOrg = oDummy.offsetWidth + '-' + oDummy.offsetHeight;
|
|
|
|
oDummy.style.cssText = sStyle + 'font-family:"' + sFont.replace(rx, '","') + '", "' + sDefFont + '" !important';
|
|
|
|
var bInstalled = sOrg != (oDummy.offsetWidth + '-' + oDummy.offsetHeight);
|
|
|
|
document.body.removeChild(oDummy);
|
|
|
|
return bInstalled;
|
|
|
|
};
|
|
})();
|
|
|
|
//{
|
|
/**
|
|
* @fileOverview This file contains Husky plugin that takes care of loading css files dynamically
|
|
* @name hp_SE2B_CSSLoader.js
|
|
*/
|
|
nhn.husky.SE2B_CSSLoader = jindo.$Class({
|
|
name : "SE2B_CSSLoader",
|
|
bCssLoaded : false,
|
|
|
|
// load & continue with the message right away.
|
|
aInstantLoadTrigger : ["OPEN_QE_LAYER", "SHOW_ACTIVE_LAYER", "SHOW_DIALOG_LAYER", "START_SPELLCHECK"],
|
|
// if a rendering bug occurs in IE, give some delay before continue processing the message.
|
|
aDelayedLoadTrigger : ["MSG_SE_OBJECT_EDIT_REQUESTED", "OBJECT_MODIFY", "MSG_SE_DUMMY_OBJECT_EDIT_REQUESTED", "TOGGLE_TOOLBAR_ACTIVE_LAYER", "SHOW_TOOLBAR_ACTIVE_LAYER"],
|
|
|
|
$init : function(){
|
|
this.htOptions = nhn.husky.SE2M_Configuration.SE2B_CSSLoader;
|
|
|
|
// only IE's slow
|
|
if(!jindo.$Agent().navigator().ie){
|
|
this.$ON_MSG_APP_READY = jindo.$Fn(function(){
|
|
this.loadSE2CSS();
|
|
}, this).bind()
|
|
}else{
|
|
for(var i=0, nLen = this.aInstantLoadTrigger.length; i<nLen; i++){
|
|
this["$BEFORE_"+this.aInstantLoadTrigger[i]] = jindo.$Fn(function(){
|
|
this.loadSE2CSS();
|
|
}, this).bind();
|
|
}
|
|
|
|
for(var i=0, nLen = this.aDelayedLoadTrigger.length; i<nLen; i++){
|
|
var sMsg = this.aDelayedLoadTrigger[i];
|
|
|
|
this["$BEFORE_"+this.aDelayedLoadTrigger[i]] = jindo.$Fn(function(sMsg){
|
|
var aArgs = jindo.$A(arguments).$value();
|
|
aArgs = aArgs.splice(1, aArgs.length-1);
|
|
return this.loadSE2CSS(sMsg, aArgs);
|
|
}, this).bind(sMsg);
|
|
}
|
|
}
|
|
},
|
|
|
|
/*
|
|
$BEFORE_REEDIT_ITEM_ACTION : function(){
|
|
return this.loadSE2CSS("REEDIT_ITEM_ACTION", arguments);
|
|
},
|
|
$BEFORE_OBJECT_MODIFY : function(){
|
|
return this.loadSE2CSS("OBJECT_MODIFY", arguments);
|
|
},
|
|
$BEFORE_MSG_SE_DUMMY_OBJECT_EDIT_REQUESTED : function(){
|
|
return this.loadSE2CSS("MSG_SE_DUMMY_OBJECT_EDIT_REQUESTED", arguments);
|
|
},
|
|
$BEFORE_TOGGLE_DBATTACHMENT_LAYER : function(){
|
|
return this.loadSE2CSS("TOGGLE_DBATTACHMENT_LAYER", arguments);
|
|
},
|
|
$BEFORE_SHOW_WRITE_REVIEW_DESIGN_SELECT_LAYER : function(){
|
|
this.loadSE2CSS();
|
|
},
|
|
$BEFORE_OPEN_QE_LAYER : function(){
|
|
this.loadSE2CSS();
|
|
},
|
|
$BEFORE_TOGGLE_TOOLBAR_ACTIVE_LAYER : function(){
|
|
return this.loadSE2CSS("TOGGLE_TOOLBAR_ACTIVE_LAYER", arguments);
|
|
},
|
|
$BEFORE_SHOW_TOOLBAR_ACTIVE_LAYER : function(){
|
|
return this.loadSE2CSS("SHOW_TOOLBAR_ACTIVE_LAYER", arguments);
|
|
},
|
|
$BEFORE_SHOW_ACTIVE_LAYER : function(){
|
|
this.loadSE2CSS();
|
|
},
|
|
$BEFORE_SHOW_DIALOG_LAYER : function(){
|
|
this.loadSE2CSS();
|
|
},
|
|
$BEFORE_TOGGLE_ITEM_LAYER : function(){
|
|
return this.loadSE2CSS("TOGGLE_ITEM_LAYER", arguments);
|
|
},
|
|
*/
|
|
|
|
// if a rendering bug occurs in IE, pass sMsg and oArgs to give some delay before the message is processed.
|
|
loadSE2CSS : function(sMsg, oArgs){
|
|
if(this.bCssLoaded){return true;}
|
|
this.bCssLoaded = true;
|
|
|
|
var fnCallback = null;
|
|
if(sMsg){
|
|
fnCallback = jindo.$Fn(this.oApp.exec, this.oApp).bind(sMsg, oArgs);
|
|
}
|
|
|
|
//nhn.husky.SE2M_Utils.loadCSS("css/smart_editor2.css");
|
|
var sCssUrl = this.htOptions.sCSSBaseURI;
|
|
var sLocale = this.oApp && this.oApp.htOptions.I18N_LOCALE;
|
|
if(sLocale){
|
|
sCssUrl += "/" + sLocale;
|
|
}
|
|
sCssUrl += "/smart_editor2_items.css";
|
|
nhn.husky.SE2M_Utils.loadCSS(sCssUrl, fnCallback);
|
|
|
|
return false;
|
|
}
|
|
});
|
|
//}
|
|
if(typeof window.nhn=='undefined'){window.nhn = {};}
|
|
/**
|
|
* @fileOverview This file contains a message mapping(Korean), which is used to map the message code to the actual message
|
|
* @name husky_SE2B_Lang_ko_KR.js
|
|
* @ unescape
|
|
*/
|
|
var oMessageMap = {
|
|
'SE_EditingAreaManager.onExit' : '내용이 변경되었습니다.',
|
|
'SE_Color.invalidColorCode' : '색상 코드를 올바르게 입력해 주세요. \n\n 예) #000000, #FF0000, #FFFFFF, #ffffff, ffffff',
|
|
'SE_Hyperlink.invalidURL' : '입력하신 URL이 올바르지 않습니다.',
|
|
'SE_FindReplace.keywordMissing' : '찾으실 단어를 입력해 주세요.',
|
|
'SE_FindReplace.keywordNotFound' : '찾으실 단어가 없습니다.',
|
|
'SE_FindReplace.replaceAllResultP1' : '일치하는 내용이 총 ',
|
|
'SE_FindReplace.replaceAllResultP2' : '건 바뀌었습니다.',
|
|
'SE_FindReplace.notSupportedBrowser' : '현재 사용하고 계신 브라우저에서는 사용하실수 없는 기능입니다.\n\n이용에 불편을 드려 죄송합니다.',
|
|
'SE_FindReplace.replaceKeywordNotFound' : '바뀔 단어가 없습니다',
|
|
'SE_LineHeight.invalidLineHeight' : '잘못된 값입니다.',
|
|
'SE_Footnote.defaultText' : '각주내용을 입력해 주세요',
|
|
'SE.failedToLoadFlash' : '플래시가 차단되어 있어 해당 기능을 사용할 수 없습니다.',
|
|
'SE2M_EditingModeChanger.confirmTextMode' : '텍스트 모드로 전환하면 작성된 내용은 유지되나, \n\n글꼴 등의 편집효과와 이미지 등의 첨부내용이 모두 사라지게 됩니다.\n\n전환하시겠습니까?',
|
|
'SE2M_FontNameWithLayerUI.sSampleText' : '가나다라'
|
|
};
|
|
if(typeof window.nhn=='undefined'){window.nhn = {};}
|
|
/**
|
|
* @fileOverview This file contains a message mapping(Korean), which is used to map the message code to the actual message
|
|
* @name husky_SE2B_Lang_en_US.js
|
|
* @ unescape
|
|
*/
|
|
var oMessageMap_en_US = {
|
|
'SE_EditingAreaManager.onExit' : 'Contents have been changed.',
|
|
'SE_Color.invalidColorCode' : 'Enter the correct color code. \n\n ex) #000000, #FF0000, #FFFFFF, #ffffff, ffffff',
|
|
'SE_Hyperlink.invalidURL' : 'You have entered an incorrect URL.',
|
|
'SE_FindReplace.keywordMissing' : 'Enter the word you wish to find.',
|
|
'SE_FindReplace.keywordNotFound' : 'The word does not exist.',
|
|
'SE_FindReplace.replaceAllResultP1' : 'A total of ',
|
|
'SE_FindReplace.replaceAllResultP2' : ' matching contents have been changed.',
|
|
'SE_FindReplace.notSupportedBrowser' : 'Function cannot be used in the browser you are currently using. \n\nSorry for the inconvenience.',
|
|
'SE_FindReplace.replaceKeywordNotFound' : 'No word to change.',
|
|
'SE_LineHeight.invalidLineHeight' : 'Incorrect value.',
|
|
'SE_Footnote.defaultText' : 'Enter footnote details.',
|
|
'SE.failedToLoadFlash' : 'The function cannot be used because flash has been blocked.',
|
|
'SE2M_EditingModeChanger.confirmTextMode' : 'The contents remain, but editing effects, including fonts, and attachments, \n\nsuch as images, will disappear when changed to text mode. \n\n Make changes?',
|
|
'SE2M_FontNameWithLayerUI.sSampleText' : 'ABCD'
|
|
};
|
|
if(typeof window.nhn=='undefined'){window.nhn = {};}
|
|
/**
|
|
* @fileOverview This file contains a message mapping(Korean), which is used to map the message code to the actual message
|
|
* @name husky_SE2B_Lang_ja_JP.js
|
|
* @ unescape
|
|
*/
|
|
var oMessageMap_ja_JP = {
|
|
'SE_EditingAreaManager.onExit' : '内容が変更されました。',
|
|
'SE_Color.invalidColorCode' : 'カラーコードを正しく入力してください。 \n\n 例) #000000, #FF0000, #FFFFFF, #ffffff, ffffff',
|
|
'SE_Hyperlink.invalidURL' : '入力したURLが正しくありません。',
|
|
'SE_FindReplace.keywordMissing' : 'お探しの単語を入力してください。',
|
|
'SE_FindReplace.keywordNotFound' : 'お探しの単語がありません。',
|
|
'SE_FindReplace.replaceAllResultP1' : '一致する内容が計',
|
|
'SE_FindReplace.replaceAllResultP2' : '件変わりました。',
|
|
'SE_FindReplace.notSupportedBrowser' : '現在ご使用中のブラウザーではご利用いただけない機能です。\n\nご不便をおかけしまして申し訳ございません。',
|
|
'SE_FindReplace.replaceKeywordNotFound' : '変更される単語がありません。',
|
|
'SE_LineHeight.invalidLineHeight' : '誤った値です。',
|
|
'SE_Footnote.defaultText' : '脚注内容を入力してください。',
|
|
'SE.failedToLoadFlash' : 'フラッシュが遮断されているため、この機能は使用できません。',
|
|
'SE2M_EditingModeChanger.confirmTextMode' : 'テキストモードに切り換えると、作成された内容は維持されますが、\n\nフォント等の編集効果と画像等の添付内容が消えることになります。\n\n切り換えますか?',
|
|
'SE2M_FontNameWithLayerUI.sSampleText' : 'あいうえお'
|
|
};
|
|
if(typeof window.nhn=='undefined'){window.nhn = {};}
|
|
/**
|
|
* @fileOverview This file contains a message mapping(Korean), which is used to map the message code to the actual message
|
|
* @name husky_SE2B_Lang_zh_CN.js
|
|
* @ unescape
|
|
*/
|
|
var oMessageMap_zh_CN = {
|
|
'SE_EditingAreaManager.onExit' : '内容有了变化。',
|
|
'SE_Color.invalidColorCode' : '请你输入正确的色相代码。 \n\n 例) #000000, #FF0000, #FFFFFF, #ffffff, ffffff',
|
|
'SE_Hyperlink.invalidURL' : '你输入的URL不符条件。',
|
|
'SE_FindReplace.keywordMissing' : '请你输入要找的词汇。',
|
|
'SE_FindReplace.keywordNotFound' : '没有词汇符合条件。',
|
|
'SE_FindReplace.replaceAllResultP1' : '符合条件的内容改编为',
|
|
'SE_FindReplace.replaceAllResultP2' : '件',
|
|
'SE_FindReplace.notSupportedBrowser' : '这是你现在使用的浏览器不可支持的功能。\n\n麻烦你很道歉。',
|
|
'SE_FindReplace.replaceKeywordNotFound' : '没有词汇要改变。',
|
|
'SE_LineHeight.invalidLineHeight' : '这是有问题的值。',
|
|
'SE_Footnote.defaultText' : '请你输入脚注内容。',
|
|
'SE.failedToLoadFlash' : 'flash被隔绝,不能使用该功能。',
|
|
'SE2M_EditingModeChanger.confirmTextMode' : '转换为text模式就能维持制作内容,\n\n但字体等编辑效果和图像等附件内容都会消失。\n\n你还要继续吗?',
|
|
'SE2M_FontNameWithLayerUI.sSampleText' : 'ABCD'
|
|
};
|
|
if(typeof window.nhn=='undefined'){window.nhn = {};}
|
|
/**
|
|
* @fileOverview This file contains a message mapping(Korean), which is used to map the message code to the actual message
|
|
* @name husky_SE2B_Lang_zh_TW.js
|
|
* @ unescape
|
|
*/
|
|
var oMessageMap_zh_TW = {
|
|
'SE_EditingAreaManager.onExit' : '內容已被更改。',
|
|
'SE_Color.invalidColorCode' : '請輸入正確的顔色代碼。\n\n例子)#000000, #FF0000, #FFFFFF, #ffffff, ffffff',
|
|
'SE_Hyperlink.invalidURL' : '輸錯URL',
|
|
'SE_FindReplace.keywordMissing' : '請輸入要查詢的單詞。',
|
|
'SE_FindReplace.keywordNotFound' : '要查詢的單詞不存在。',
|
|
'SE_FindReplace.replaceAllResultP1' : '一致的內容總有',
|
|
'SE_FindReplace.replaceAllResultP2' : '件已被更改。',
|
|
'SE_FindReplace.notSupportedBrowser' : '當前的浏覽器上無法使用此功能。\n\n很抱歉給你添麻煩了。',
|
|
'SE_FindReplace.replaceKeywordNotFound' : '要更改的詞彙不存在。',
|
|
'SE_LineHeight.invalidLineHeight' : '爲有錯誤的值。',
|
|
'SE_Footnote.defaultText' : '請輸入注腳內容。',
|
|
'SE.failedToLoadFlash' : 'Flash被屏蔽,導致無法使用此功能。',
|
|
'SE2M_EditingModeChanger.confirmTextMode' : '如轉換爲文本模式,雖然維持已寫成的內容,\n\n但字號等編輯效果和圖像等附加內容都會被消失。\n\n是否轉換?',
|
|
'SE2M_FontNameWithLayerUI.sSampleText' : 'ABCD'
|
|
};
|
|
|
|
/*[
|
|
* SE_FIT_IFRAME
|
|
*
|
|
* 스마트에디터 사이즈에 맞게 iframe사이즈를 조절한다.
|
|
*
|
|
* none
|
|
*
|
|
---------------------------------------------------------------------------]*/
|
|
/**
|
|
* @pluginDesc 에디터를 싸고 있는 iframe 사이즈 조절을 담당하는 플러그인
|
|
*/
|
|
nhn.husky.SE_OuterIFrameControl = $Class({
|
|
name : "SE_OuterIFrameControl",
|
|
oResizeGrip : null,
|
|
|
|
$init : function(oAppContainer){
|
|
// page up, page down, home, end, left, up, right, down
|
|
this.aHeightChangeKeyMap = [-100, 100, 500, -500, -1, -10, 1, 10];
|
|
|
|
this._assignHTMLObjects(oAppContainer);
|
|
|
|
//키보드 이벤트
|
|
this.$FnKeyDown = $Fn(this._keydown, this);
|
|
if(this.oResizeGrip){
|
|
this.$FnKeyDown.attach(this.oResizeGrip, "keydown");
|
|
}
|
|
|
|
//마우스 이벤트
|
|
if(!!jindo.$Agent().navigator().ie){
|
|
this.$FnMouseDown = $Fn(this._mousedown, this);
|
|
this.$FnMouseMove = $Fn(this._mousemove, this);
|
|
this.$FnMouseMove_Parent = $Fn(this._mousemove_parent, this);
|
|
this.$FnMouseUp = $Fn(this._mouseup, this);
|
|
|
|
if(this.oResizeGrip){
|
|
this.$FnMouseDown.attach(this.oResizeGrip, "mousedown");
|
|
}
|
|
}
|
|
},
|
|
|
|
_assignHTMLObjects : function(oAppContainer){
|
|
oAppContainer = jindo.$(oAppContainer) || document;
|
|
|
|
this.oResizeGrip = cssquery.getSingle(".husky_seditor_editingArea_verticalResizer", oAppContainer);
|
|
|
|
this.elIFrame = window.frameElement;
|
|
this.welIFrame = $Element(this.elIFrame);
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("SE_FIT_IFRAME", []);
|
|
},
|
|
|
|
$ON_MSG_EDITING_AREA_SIZE_CHANGED : function(){
|
|
this.oApp.exec("SE_FIT_IFRAME", []);
|
|
},
|
|
|
|
$ON_SE_FIT_IFRAME : function(){
|
|
this.elIFrame.style.height = document.body.offsetHeight+"px";
|
|
},
|
|
|
|
$AFTER_RESIZE_EDITING_AREA_BY : function(ipWidthChange, ipHeightChange){
|
|
this.oApp.exec("SE_FIT_IFRAME", []);
|
|
},
|
|
|
|
_keydown : function(oEvent){
|
|
var oKeyInfo = oEvent.key();
|
|
|
|
// 33, 34: page up/down, 35,36: end/home, 37,38,39,40: left, up, right, down
|
|
if(oKeyInfo.keyCode >= 33 && oKeyInfo.keyCode <= 40){
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_STARTED", []);
|
|
this.oApp.exec("RESIZE_EDITING_AREA_BY", [0, this.aHeightChangeKeyMap[oKeyInfo.keyCode-33]]);
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_ENDED", []);
|
|
|
|
oEvent.stop();
|
|
}
|
|
},
|
|
|
|
_mousedown : function(oEvent){
|
|
this.iStartHeight = oEvent.pos().clientY;
|
|
this.iStartHeightOffset = oEvent.pos().layerY;
|
|
|
|
this.$FnMouseMove.attach(document, "mousemove");
|
|
this.$FnMouseMove_Parent.attach(parent.document, "mousemove");
|
|
|
|
this.$FnMouseUp.attach(document, "mouseup");
|
|
this.$FnMouseUp.attach(parent.document, "mouseup");
|
|
|
|
this.iStartHeight = oEvent.pos().clientY;
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_STARTED", [this.$FnMouseDown, this.$FnMouseMove, this.$FnMouseUp]);
|
|
},
|
|
|
|
_mousemove : function(oEvent){
|
|
var iHeightChange = oEvent.pos().clientY - this.iStartHeight;
|
|
this.oApp.exec("RESIZE_EDITING_AREA_BY", [0, iHeightChange]);
|
|
},
|
|
|
|
_mousemove_parent : function(oEvent){
|
|
var iHeightChange = oEvent.pos().pageY - (this.welIFrame.offset().top + this.iStartHeight);
|
|
this.oApp.exec("RESIZE_EDITING_AREA_BY", [0, iHeightChange]);
|
|
},
|
|
|
|
_mouseup : function(oEvent){
|
|
this.$FnMouseMove.detach(document, "mousemove");
|
|
this.$FnMouseMove_Parent.detach(parent.document, "mousemove");
|
|
this.$FnMouseUp.detach(document, "mouseup");
|
|
this.$FnMouseUp.detach(parent.document, "mouseup");
|
|
|
|
this.oApp.exec("MSG_EDITING_AREA_RESIZE_ENDED", [this.$FnMouseDown, this.$FnMouseMove, this.$FnMouseUp]);
|
|
}
|
|
});
|
|
// Sample plugin. Use CTRL+T to toggle the toolbar
|
|
nhn.husky.SE_ToolbarToggler = $Class({
|
|
name : "SE_ToolbarToggler",
|
|
bUseToolbar : true,
|
|
|
|
$init : function(oAppContainer, bUseToolbar){
|
|
this._assignHTMLObjects(oAppContainer, bUseToolbar);
|
|
},
|
|
|
|
_assignHTMLObjects : function(oAppContainer, bUseToolbar){
|
|
oAppContainer = jindo.$(oAppContainer) || document;
|
|
|
|
this.toolbarArea = cssquery.getSingle(".se2_tool", oAppContainer);
|
|
|
|
//설정이 없거나, 사용하겠다고 표시한 경우 block 처리
|
|
if( typeof(bUseToolbar) == 'undefined' || bUseToolbar === true){
|
|
this.toolbarArea.style.display = "block";
|
|
}else{
|
|
this.toolbarArea.style.display = "none";
|
|
}
|
|
},
|
|
|
|
$ON_MSG_APP_READY : function(){
|
|
this.oApp.exec("REGISTER_HOTKEY", ["ctrl+t", "SE_TOGGLE_TOOLBAR", []]);
|
|
},
|
|
|
|
$ON_SE_TOGGLE_TOOLBAR : function(){
|
|
this.toolbarArea.style.display = (this.toolbarArea.style.display == "none")?"block":"none";
|
|
this.oApp.exec("MSG_EDITING_AREA_SIZE_CHANGED", []);
|
|
}
|
|
});
|
|
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_FindReplacePlugin$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_FindReplacePlugin
|
|
* this.oApp.registerLazyMessage(["TOGGLE_FIND_REPLACE_LAYER","SHOW_FIND_LAYER","SHOW_REPLACE_LAYER","SHOW_FIND_REPLACE_LAYER"], ["hp_SE2M_FindReplacePlugin$Lazy.js","N_FindReplace.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_FindReplacePlugin, {
|
|
//@lazyload_js TOGGLE_FIND_REPLACE_LAYER,SHOW_FIND_LAYER,SHOW_REPLACE_LAYER,SHOW_FIND_REPLACE_LAYER:N_FindReplace.js[
|
|
_assignHTMLElements : function(){
|
|
var oAppContainer = this.oApp.htOptions.elAppContainer;
|
|
|
|
this.oApp.exec("LOAD_HTML", ["find_and_replace"]);
|
|
// this.oEditingWindow = jindo.$$.getSingle("IFRAME", oAppContainer);
|
|
this.elDropdownLayer = jindo.$$.getSingle("DIV.husky_se2m_findAndReplace_layer", oAppContainer);
|
|
this.welDropdownLayer = jindo.$Element(this.elDropdownLayer);
|
|
var oTmp = jindo.$$("LI", this.elDropdownLayer);
|
|
|
|
this.oFindTab = oTmp[0];
|
|
this.oReplaceTab = oTmp[1];
|
|
|
|
oTmp = jindo.$$(".container > .bx", this.elDropdownLayer);
|
|
|
|
this.oFindInputSet = jindo.$$.getSingle(".husky_se2m_find_ui", this.elDropdownLayer);
|
|
this.oReplaceInputSet = jindo.$$.getSingle(".husky_se2m_replace_ui", this.elDropdownLayer);
|
|
|
|
this.elTitle = jindo.$$.getSingle("H3", this.elDropdownLayer);
|
|
|
|
this.oFindInput_Keyword = jindo.$$.getSingle("INPUT", this.oFindInputSet);
|
|
|
|
oTmp = jindo.$$("INPUT", this.oReplaceInputSet);
|
|
this.oReplaceInput_Original = oTmp[0];
|
|
this.oReplaceInput_Replacement = oTmp[1];
|
|
|
|
this.oFindNextButton = jindo.$$.getSingle("BUTTON.husky_se2m_find_next", this.elDropdownLayer);
|
|
|
|
this.oReplaceFindNextButton = jindo.$$.getSingle("BUTTON.husky_se2m_replace_find_next", this.elDropdownLayer);
|
|
|
|
this.oReplaceButton = jindo.$$.getSingle("BUTTON.husky_se2m_replace", this.elDropdownLayer);
|
|
this.oReplaceAllButton = jindo.$$.getSingle("BUTTON.husky_se2m_replace_all", this.elDropdownLayer);
|
|
|
|
this.aCloseButtons = jindo.$$("BUTTON.husky_se2m_cancel", this.elDropdownLayer);
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(sMsg){
|
|
this._assignHTMLElements();
|
|
|
|
this.oFindReplace = new nhn.FindReplace(this.oEditingWindow);
|
|
|
|
for(var i=0; i<this.aCloseButtons.length; i++){
|
|
// var func = jindo.$Fn(this.oApp.exec, this.oApp).bind("HIDE_DIALOG_LAYER", [this.elDropdownLayer]);
|
|
var func = jindo.$Fn(this.oApp.exec, this.oApp).bind("HIDE_FIND_REPLACE_LAYER", [this.elDropdownLayer]);
|
|
jindo.$Fn(func, this).attach(this.aCloseButtons[i], "click");
|
|
}
|
|
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("SHOW_FIND", []), this).attach(this.oFindTab, "click");
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("SHOW_REPLACE", []), this).attach(this.oReplaceTab, "click");
|
|
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("FIND", []), this).attach(this.oFindNextButton, "click");
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("FIND", []), this).attach(this.oReplaceFindNextButton, "click");
|
|
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("REPLACE", []), this).attach(this.oReplaceButton, "click");
|
|
jindo.$Fn(jindo.$Fn(this.oApp.exec, this.oApp).bind("REPLACE_ALL", []), this).attach(this.oReplaceAllButton, "click");
|
|
|
|
this.oFindInput_Keyword.value = "";
|
|
this.oReplaceInput_Original.value = "";
|
|
this.oReplaceInput_Replacement.value = "";
|
|
|
|
//레이어의 이동 범위 설정.
|
|
var elIframe = this.oApp.getWYSIWYGWindow().frameElement;
|
|
this.htOffsetPos = jindo.$Element(elIframe).offset();
|
|
this.nEditorWidth = elIframe.offsetWidth;
|
|
|
|
this.elDropdownLayer.style.display = "block";
|
|
this.htInitialPos = this.welDropdownLayer.offset();
|
|
var htScrollXY = this.oApp.oUtils.getScrollXY();
|
|
// this.welDropdownLayer.offset(this.htOffsetPos.top-htScrollXY.y, this.htOffsetPos.left-htScrollXY.x);
|
|
this.welDropdownLayer.offset(this.htOffsetPos.top, this.htOffsetPos.left);
|
|
this.htTopLeftCorner = {x:parseInt(this.elDropdownLayer.style.left, 10), y:parseInt(this.elDropdownLayer.style.top, 10)};
|
|
|
|
// offset width가 IE에서 css lazy loading 때문에 제대로 잡히지 않아 상수로 설정
|
|
//this.nLayerWidth = this.elDropdownLayer.offsetWidth;
|
|
this.nLayerWidth = 258;
|
|
this.nLayerHeight = 160;
|
|
|
|
//this.nLayerWidth = Math.abs(parseInt(this.elDropdownLayer.style.marginLeft))+20;
|
|
this.elDropdownLayer.style.display = "none";
|
|
},
|
|
|
|
// [SMARTEDITORSUS-728] 찾기/바꾸기 레이어 오픈 툴바 버튼 active/inactive 처리 추가
|
|
$ON_TOGGLE_FIND_REPLACE_LAYER : function(){
|
|
if(!this.bLayerShown) {
|
|
this.oApp.exec("SHOW_FIND_REPLACE_LAYER");
|
|
} else {
|
|
this.oApp.exec("HIDE_FIND_REPLACE_LAYER");
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_FIND_REPLACE_LAYER : function(){
|
|
this.bLayerShown = true;
|
|
this.oApp.exec("DISABLE_ALL_UI", [{aExceptions: ["findAndReplace"]}]);
|
|
this.oApp.exec("SELECT_UI", ["findAndReplace"]);
|
|
|
|
this.oApp.exec("HIDE_ALL_DIALOG_LAYER", []);
|
|
this.elDropdownLayer.style.top = this.nDefaultTop+"px";
|
|
|
|
this.oApp.exec("SHOW_DIALOG_LAYER", [this.elDropdownLayer, {
|
|
elHandle: this.elTitle,
|
|
fnOnDragStart : jindo.$Fn(this.oApp.exec, this.oApp).bind("SHOW_EDITING_AREA_COVER"),
|
|
fnOnDragEnd : jindo.$Fn(this.oApp.exec, this.oApp).bind("HIDE_EDITING_AREA_COVER"),
|
|
nMinX : this.htTopLeftCorner.x,
|
|
nMinY : this.nDefaultTop,
|
|
nMaxX : this.htTopLeftCorner.x + this.oApp.getEditingAreaWidth() - this.nLayerWidth,
|
|
nMaxY : this.htTopLeftCorner.y + this.oApp.getEditingAreaHeight() - this.nLayerHeight,
|
|
sOnShowMsg : "FIND_REPLACE_LAYER_SHOWN"
|
|
}]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['findreplace']);
|
|
},
|
|
|
|
$ON_HIDE_FIND_REPLACE_LAYER : function() {
|
|
this.oApp.exec("ENABLE_ALL_UI");
|
|
this.oApp.exec("DESELECT_UI", ["findAndReplace"]);
|
|
this.oApp.exec("HIDE_ALL_DIALOG_LAYER", []);
|
|
this.bLayerShown = false;
|
|
},
|
|
|
|
$ON_FIND_REPLACE_LAYER_SHOWN : function(){
|
|
this.oApp.exec("POSITION_TOOLBAR_LAYER", [this.elDropdownLayer]);
|
|
if(this.bFindMode){
|
|
this.oFindInput_Keyword.value = "_clear_";
|
|
this.oFindInput_Keyword.value = "";
|
|
this.oFindInput_Keyword.focus();
|
|
}else{
|
|
this.oReplaceInput_Original.value = "_clear_";
|
|
this.oReplaceInput_Original.value = "";
|
|
this.oReplaceInput_Replacement.value = "";
|
|
this.oReplaceInput_Original.focus();
|
|
}
|
|
|
|
this.oApp.exec("HIDE_CURRENT_ACTIVE_LAYER", []);
|
|
},
|
|
|
|
$ON_SHOW_FIND_LAYER : function(){
|
|
this.oApp.exec("SHOW_FIND");
|
|
this.oApp.exec("SHOW_FIND_REPLACE_LAYER");
|
|
},
|
|
|
|
$ON_SHOW_REPLACE_LAYER : function(){
|
|
this.oApp.exec("SHOW_REPLACE");
|
|
this.oApp.exec("SHOW_FIND_REPLACE_LAYER");
|
|
},
|
|
|
|
$ON_SHOW_FIND : function(){
|
|
this.bFindMode = true;
|
|
this.oFindInput_Keyword.value = this.oReplaceInput_Original.value;
|
|
|
|
jindo.$Element(this.oFindTab).addClass("active");
|
|
jindo.$Element(this.oReplaceTab).removeClass("active");
|
|
|
|
jindo.$Element(this.oFindNextButton).removeClass("normal");
|
|
jindo.$Element(this.oFindNextButton).addClass("strong");
|
|
|
|
this.oFindInputSet.style.display = "block";
|
|
this.oReplaceInputSet.style.display = "none";
|
|
|
|
this.oReplaceButton.style.display = "none";
|
|
this.oReplaceAllButton.style.display = "none";
|
|
|
|
jindo.$Element(this.elDropdownLayer).removeClass("replace");
|
|
jindo.$Element(this.elDropdownLayer).addClass("find");
|
|
},
|
|
|
|
$ON_SHOW_REPLACE : function(){
|
|
this.bFindMode = false;
|
|
this.oReplaceInput_Original.value = this.oFindInput_Keyword.value;
|
|
|
|
jindo.$Element(this.oFindTab).removeClass("active");
|
|
jindo.$Element(this.oReplaceTab).addClass("active");
|
|
|
|
jindo.$Element(this.oFindNextButton).removeClass("strong");
|
|
jindo.$Element(this.oFindNextButton).addClass("normal");
|
|
|
|
this.oFindInputSet.style.display = "none";
|
|
this.oReplaceInputSet.style.display = "block";
|
|
|
|
this.oReplaceButton.style.display = "inline";
|
|
this.oReplaceAllButton.style.display = "inline";
|
|
|
|
jindo.$Element(this.elDropdownLayer).removeClass("find");
|
|
jindo.$Element(this.elDropdownLayer).addClass("replace");
|
|
},
|
|
|
|
$ON_FIND : function(){
|
|
var sKeyword;
|
|
if(this.bFindMode){
|
|
sKeyword = this.oFindInput_Keyword.value;
|
|
}else{
|
|
sKeyword = this.oReplaceInput_Original.value;
|
|
}
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
oSelection.select();
|
|
|
|
switch(this.oFindReplace.find(sKeyword, false)){
|
|
case 1:
|
|
alert(this.oApp.$MSG("SE_FindReplace.keywordNotFound"));
|
|
oSelection.select();
|
|
break;
|
|
case 2:
|
|
alert(this.oApp.$MSG("SE_FindReplace.keywordMissing"));
|
|
break;
|
|
}
|
|
},
|
|
|
|
$ON_REPLACE : function(){
|
|
var sOriginal = this.oReplaceInput_Original.value;
|
|
var sReplacement = this.oReplaceInput_Replacement.value;
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["REPLACE"]);
|
|
var iReplaceResult = this.oFindReplace.replace(sOriginal, sReplacement, false);
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["REPLACE"]);
|
|
|
|
switch(iReplaceResult){
|
|
case 1:
|
|
case 3:
|
|
alert(this.oApp.$MSG("SE_FindReplace.keywordNotFound"));
|
|
oSelection.select();
|
|
break;
|
|
case 4:
|
|
alert(this.oApp.$MSG("SE_FindReplace.keywordMissing"));
|
|
break;
|
|
}
|
|
},
|
|
|
|
$ON_REPLACE_ALL : function(){
|
|
var sOriginal = this.oReplaceInput_Original.value;
|
|
var sReplacement = this.oReplaceInput_Replacement.value;
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["REPLACE ALL", {sSaveTarget:"BODY"}]);
|
|
var iReplaceAllResult = this.oFindReplace.replaceAll(sOriginal, sReplacement, false);
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["REPLACE ALL", {sSaveTarget:"BODY"}]);
|
|
|
|
if(iReplaceAllResult === 0){
|
|
alert(this.oApp.$MSG("SE_FindReplace.replaceKeywordNotFound"));
|
|
oSelection.select();
|
|
this.oApp.exec("FOCUS");
|
|
}else{
|
|
if(iReplaceAllResult<0){
|
|
alert(this.oApp.$MSG("SE_FindReplace.keywordMissing"));
|
|
oSelection.select();
|
|
}else{
|
|
alert(this.oApp.$MSG("SE_FindReplace.replaceAllResultP1")+iReplaceAllResult+this.oApp.$MSG("SE_FindReplace.replaceAllResultP2"));
|
|
oSelection = this.oApp.getEmptySelection();
|
|
oSelection.select();
|
|
this.oApp.exec("FOCUS");
|
|
}
|
|
}
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_Quote$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_Quote
|
|
* this.oApp.registerLazyMessage(["TOGGLE_BLOCKQUOTE_LAYER"], ["hp_SE2M_Quote$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_Quote, {
|
|
//@lazyload_js TOGGLE_BLOCKQUOTE_LAYER[
|
|
$ON_TOGGLE_BLOCKQUOTE_LAYER : function(){
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.elDropdownLayer, null, "SELECT_UI", ["quote"], "DESELECT_UI", ["quote"]]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['quote']);
|
|
},
|
|
|
|
$ON_EVENT_SE2_BLOCKQUOTE_LAYER_CLICK : function(weEvent){
|
|
var elButton = nhn.husky.SE2M_Utils.findAncestorByTagName("BUTTON", weEvent.element);
|
|
|
|
if(!elButton || elButton.tagName != "BUTTON"){return;}
|
|
|
|
var sClass = elButton.className;
|
|
this.oApp.exec("APPLY_BLOCKQUOTE", [sClass]);
|
|
},
|
|
|
|
$ON_APPLY_BLOCKQUOTE : function(sClass){
|
|
if(sClass.match(/(se2_quote[0-9]+)/)){
|
|
this._wrapBlock("BLOCKQUOTE", RegExp.$1);
|
|
}else{
|
|
this._unwrapBlock("BLOCKQUOTE");
|
|
}
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
},
|
|
|
|
/**
|
|
* 인용구의 중첩 가능한 최대 개수를 넘었는지 확인함
|
|
* 인용구 내부에서 인용구를 적용하면 중첩되지 않으므로 자식노드에 대해서만 확인함
|
|
*/
|
|
_isExceedMaxDepth : function(elNode){
|
|
var countChildQuote = function(elNode){
|
|
var elChild = elNode.firstChild;
|
|
var nCount = 0;
|
|
var nMaxCount = 0;
|
|
|
|
if(!elChild){
|
|
if(elNode.tagName && elNode.tagName === "BLOCKQUOTE"){
|
|
return 1;
|
|
}else{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
while(elChild){
|
|
if(elChild.nodeType === 1){
|
|
nCount = countChildQuote(elChild);
|
|
|
|
if(elChild.tagName === "BLOCKQUOTE"){
|
|
nCount += 1;
|
|
}
|
|
|
|
if(nMaxCount < nCount){
|
|
nMaxCount = nCount;
|
|
}
|
|
|
|
if(nMaxCount >= this.nMaxLevel){
|
|
return nMaxCount;
|
|
}
|
|
}
|
|
|
|
elChild = elChild.nextSibling;
|
|
}
|
|
|
|
return nMaxCount;
|
|
};
|
|
|
|
return (countChildQuote(elNode) >= this.nMaxLevel);
|
|
},
|
|
|
|
_unwrapBlock : function(tag){
|
|
var oSelection = this.oApp.getSelection();
|
|
var elCommonAncestor = oSelection.commonAncestorContainer;
|
|
|
|
while(elCommonAncestor && elCommonAncestor.tagName != tag){elCommonAncestor = elCommonAncestor.parentNode;}
|
|
if(!elCommonAncestor){return;}
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["CANCEL BLOCK QUOTE", {sSaveTarget:"BODY"}]);
|
|
|
|
// [SMARTEDITORSUS-1782] 인용구가 제거되기 전에 선택 영역안에 있는 마지막 텍스트노드를 미리 찾아둔다.
|
|
var oLastTextNode = oSelection.commonAncestorContainer;
|
|
if(oLastTextNode.nodeType !== 3){ // 텍스트노드가 아니면
|
|
var aTextNodesInRange = oSelection.getTextNodes() || "",
|
|
nLastIndex = aTextNodesInRange.length - 1;
|
|
oLastTextNode = (nLastIndex > -1) ? aTextNodesInRange[nLastIndex] : null;
|
|
}
|
|
|
|
// 인용구내의 요소들을 바깥으로 모두 꺼낸 후 인용구요소를 제거
|
|
while(elCommonAncestor.firstChild){elCommonAncestor.parentNode.insertBefore(elCommonAncestor.firstChild, elCommonAncestor);}
|
|
elCommonAncestor.parentNode.removeChild(elCommonAncestor);
|
|
|
|
// [SMARTEDITORSUS-1782] 찾아둔 마지막 텍스트노드 끝으로 커서를 이동시킨다.
|
|
if(oLastTextNode){
|
|
oSelection.selectNodeContents(oLastTextNode);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
}
|
|
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["CANCEL BLOCK QUOTE", {sSaveTarget:"BODY"}]);
|
|
},
|
|
|
|
_wrapBlock : function(tag, className){
|
|
var oSelection,
|
|
oLineInfo,
|
|
oStart, oEnd,
|
|
rxDontUseAsWhole = /BODY|TD|LI/i,
|
|
oStartNode, oEndNode, oNode,
|
|
elCommonAncestor,
|
|
elCommonNode,
|
|
elParentQuote,
|
|
elInsertBefore,
|
|
oFormattingNode,
|
|
elNextNode,
|
|
elParentNode,
|
|
aQuoteChild,
|
|
aQuoteCloneChild,
|
|
i, nLen, oP,
|
|
sBookmarkID;
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["BLOCK QUOTE", {sSaveTarget:"BODY"}]);
|
|
oSelection = this.oApp.getSelection();
|
|
// var sBookmarkID = oSelection.placeStringBookmark();
|
|
|
|
// [SMARTEDITORSUS-430] 문자를 입력하고 Enter 후 인용구를 적용할 때 위의 문자들이 인용구 안에 들어가는 문제
|
|
// [SMARTEDITORSUS-1323] 사진 첨부 후 인용구 적용 시 첨부한 사진이 삭제되는 현상
|
|
if(oSelection.startContainer === oSelection.endContainer &&
|
|
oSelection.startContainer.nodeType === 1 &&
|
|
oSelection.startContainer.tagName === "P"){
|
|
if(nhn.husky.SE2M_Utils.isBlankNode(oSelection.startContainer) ||
|
|
nhn.husky.SE2M_Utils.isFirstChildOfNode("IMG", oSelection.startContainer.tagName, oSelection.startContainer) ||
|
|
nhn.husky.SE2M_Utils.isFirstChildOfNode("IFRAME", oSelection.startContainer.tagName, oSelection.startContainer)){
|
|
oLineInfo = oSelection.getLineInfo(true);
|
|
}else{
|
|
oLineInfo = oSelection.getLineInfo(false);
|
|
}
|
|
}else{
|
|
oLineInfo = oSelection.getLineInfo(false);
|
|
}
|
|
|
|
oStart = oLineInfo.oStart;
|
|
oEnd = oLineInfo.oEnd;
|
|
|
|
if(oStart.bParentBreak && !rxDontUseAsWhole.test(oStart.oLineBreaker.tagName)){
|
|
oStartNode = oStart.oNode.parentNode;
|
|
}else{
|
|
oStartNode = oStart.oNode;
|
|
}
|
|
|
|
if(oEnd.bParentBreak && !rxDontUseAsWhole.test(oEnd.oLineBreaker.tagName)){
|
|
oEndNode = oEnd.oNode.parentNode;
|
|
}else{
|
|
oEndNode = oEnd.oNode;
|
|
}
|
|
|
|
oSelection.setStartBefore(oStartNode);
|
|
oSelection.setEndAfter(oEndNode);
|
|
|
|
oNode = this._expandToTableStart(oSelection, oEndNode);
|
|
if(oNode){
|
|
oEndNode = oNode;
|
|
oSelection.setEndAfter(oNode);
|
|
}
|
|
|
|
oNode = this._expandToTableStart(oSelection, oStartNode);
|
|
if(oNode){
|
|
oStartNode = oNode;
|
|
oSelection.setStartBefore(oNode);
|
|
}
|
|
|
|
oNode = oStartNode;
|
|
// IE에서는 commonAncestorContainer 자체는 select 가능하지 않고, 하위에 commonAncestorContainer를 대체 하더라도 똑같은 영역이 셀렉트 되어 보이는
|
|
// 노드가 있을 경우 하위 노드가 commonAncestorContainer로 반환됨.
|
|
// 그래서, 스크립트로 commonAncestorContainer 계산 하도록 함.
|
|
// 예)
|
|
// <P><SPAN>TEST</SPAN></p>를 선택 할 경우, <SPAN>TEST</SPAN>가 commonAncestorContainer로 잡힘
|
|
oSelection.fixCommonAncestorContainer();
|
|
elCommonAncestor = oSelection.commonAncestorContainer;
|
|
|
|
if(oSelection.startContainer == oSelection.endContainer && oSelection.endOffset-oSelection.startOffset == 1){
|
|
elCommonNode = oSelection.startContainer.childNodes[oSelection.startOffset];
|
|
}else{
|
|
elCommonNode = oSelection.commonAncestorContainer;
|
|
}
|
|
|
|
elParentQuote = this._findParentQuote(elCommonNode);
|
|
|
|
if(elParentQuote){
|
|
elParentQuote.className = className;
|
|
|
|
// [SMARTEDITORSUS-1239] blockquote 태그교체시 style 적용
|
|
this._setStyle(elParentQuote, this.htQuoteStyles_view[className]);
|
|
// --[SMARTEDITORSUS-1239]
|
|
return;
|
|
}
|
|
|
|
while(!elCommonAncestor.tagName || (elCommonAncestor.tagName && elCommonAncestor.tagName.match(/UL|OL|LI|IMG|IFRAME/))){
|
|
elCommonAncestor = elCommonAncestor.parentNode;
|
|
}
|
|
|
|
// find the insertion position for the formatting tag right beneath the common ancestor container
|
|
while(oNode && oNode != elCommonAncestor && oNode.parentNode != elCommonAncestor){oNode = oNode.parentNode;}
|
|
|
|
if(oNode == elCommonAncestor){
|
|
elInsertBefore = elCommonAncestor.firstChild;
|
|
}else{
|
|
elInsertBefore = oNode;
|
|
}
|
|
|
|
oFormattingNode = oSelection._document.createElement(tag);
|
|
if(className){
|
|
oFormattingNode.className = className;
|
|
// [SMARTEDITORSUS-1239] 에디터에서 인용구 5개이상 상입 시 에디터를 뚫고 노출되는 현상
|
|
// [SMARTEDITORSUS-1229] 인용구 여러 개 중첩하면 에디터 본문 영역을 벗어나는 현상
|
|
// blockquate style 적용
|
|
this._setStyle(oFormattingNode, this.htQuoteStyles_view[className]);
|
|
}
|
|
|
|
elCommonAncestor.insertBefore(oFormattingNode, elInsertBefore);
|
|
|
|
oSelection.setStartAfter(oFormattingNode);
|
|
|
|
oSelection.setEndAfter(oEndNode);
|
|
oSelection.surroundContents(oFormattingNode);
|
|
|
|
if(this._isExceedMaxDepth(oFormattingNode)){
|
|
alert(this.oApp.$MSG("SE2M_Quote.exceedMaxCount").replace("#MaxCount#", (this.nMaxLevel + 1)));
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
|
|
elNextNode = oFormattingNode.nextSibling;
|
|
elParentNode = oFormattingNode.parentNode;
|
|
aQuoteChild = oFormattingNode.childNodes;
|
|
aQuoteCloneChild = [];
|
|
|
|
jindo.$Element(oFormattingNode).leave();
|
|
for(i = 0, nLen = aQuoteChild.length; i < nLen; i++){
|
|
aQuoteCloneChild[i] = aQuoteChild[i];
|
|
}
|
|
for(i = 0, nLen = aQuoteCloneChild.length; i < nLen; i++){
|
|
if(!!elNextNode){
|
|
jindo.$Element(elNextNode).before(aQuoteCloneChild[i]);
|
|
}else{
|
|
jindo.$Element(elParentNode).append(aQuoteCloneChild[i]);
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
oSelection.selectNodeContents(oFormattingNode);
|
|
|
|
// insert an empty line below, so the text cursor can move there
|
|
if(oFormattingNode && oFormattingNode.parentNode && oFormattingNode.parentNode.tagName == "BODY" && !oFormattingNode.nextSibling){
|
|
oP = oSelection._document.createElement("P");
|
|
//oP.innerHTML = unescape("<br/>");
|
|
oP.innerHTML = " ";
|
|
oFormattingNode.parentNode.insertBefore(oP, oFormattingNode.nextSibling);
|
|
}
|
|
|
|
// oSelection.removeStringBookmark(sBookmarkID);
|
|
// Insert an empty line inside the blockquote if it's empty.
|
|
// This is done to position the cursor correctly when the contents of the blockquote is empty in Chrome.
|
|
if(nhn.husky.SE2M_Utils.isBlankNode(oFormattingNode)){
|
|
// [SMARTEDITORSUS-1751] 현재 undo/redo 기능을 사용하지 않고 ie7은 주요브라우저에서 제외되었기 때문에 다른 이슈들 처리시 복잡도를 줄이기 위해 코멘트처리함
|
|
// [SMARTEDITORSUS-645] 편집영역 포커스 없이 인용구 추가했을 때 IE7에서 박스가 늘어나는 문제
|
|
//oFormattingNode.innerHTML = " ";
|
|
|
|
// [SMARTEDITORSUS-1567] P 태그로 감싸주지 않으면 크롬에서 blockquote 태그에 정렬이 적용되는데 IR_TO_DB 컨버터에서 style을 리셋하고 있기 때문에 저장되는 시점에 정렬이 제거된다.
|
|
// [SMARTEDITORSUS-1229] 인용구 여러 개 중첩하면 에디터 본문 영역을 벗어나는 현상
|
|
oFormattingNode.innerHTML = " ";
|
|
// [SMARTEDITORSUS-1741] 커서가 p태그 안으로 들어가도록 세팅
|
|
oSelection.selectNodeContents(oFormattingNode.firstChild);
|
|
oSelection.collapseToStart();
|
|
oSelection.select();
|
|
}
|
|
|
|
//oSelection.select();
|
|
setTimeout(jindo.$Fn(function(oSelection){
|
|
sBookmarkID = oSelection.placeStringBookmark();
|
|
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(sBookmarkID);
|
|
|
|
this.oApp.exec("FOCUS"); // [SMARTEDITORSUS-469] [SMARTEDITORSUS-434] 에디터 로드 후 최초 삽입한 인용구 안에 포커스가 가지 않는 문제
|
|
},this).bind(oSelection), 0);
|
|
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["BLOCK QUOTE", {sSaveTarget:"BODY"}]);
|
|
|
|
return oFormattingNode;
|
|
},
|
|
|
|
_expandToTableStart : function(oSelection, oNode){
|
|
var elCommonAncestor = oSelection.commonAncestorContainer;
|
|
var oResultNode = null;
|
|
|
|
var bLastIteration = false;
|
|
while(oNode && !bLastIteration){
|
|
if(oNode == elCommonAncestor){bLastIteration = true;}
|
|
|
|
if(/TBODY|TFOOT|THEAD|TR/i.test(oNode.tagName)){
|
|
oResultNode = this._getTableRoot(oNode);
|
|
break;
|
|
}
|
|
oNode = oNode.parentNode;
|
|
}
|
|
|
|
return oResultNode;
|
|
},
|
|
|
|
_getTableRoot : function(oNode){
|
|
while(oNode && oNode.tagName != "TABLE"){oNode = oNode.parentNode;}
|
|
|
|
return oNode;
|
|
},
|
|
|
|
_setStyle : function(el, sStyle) {
|
|
el.setAttribute("style", sStyle);
|
|
el.style.cssText = sStyle;
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_SCharacter$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_SCharacter
|
|
* this.oApp.registerLazyMessage(["TOGGLE_SCHARACTER_LAYER"], ["hp_SE2M_SCharacter$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_SCharacter, {
|
|
//@lazyload_js TOGGLE_SCHARACTER_LAYER[
|
|
_assignHTMLObjects : function(oAppContainer){
|
|
oAppContainer = jindo.$(oAppContainer) || document;
|
|
|
|
this.elDropdownLayer = jindo.$$.getSingle("DIV.husky_seditor_sCharacter_layer", oAppContainer);
|
|
|
|
this.oTextField = jindo.$$.getSingle("INPUT", this.elDropdownLayer);
|
|
this.oInsertButton = jindo.$$.getSingle("BUTTON.se2_confirm", this.elDropdownLayer);
|
|
this.aCloseButton = jindo.$$("BUTTON.husky_se2m_sCharacter_close", this.elDropdownLayer);
|
|
this.aSCharList = jindo.$$("UL.husky_se2m_sCharacter_list", this.elDropdownLayer);
|
|
var oLabelUL = jindo.$$.getSingle("UL.se2_char_tab", this.elDropdownLayer);
|
|
this.aLabel = jindo.$$(">LI", oLabelUL);
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(sFullMsg){
|
|
this.bIE = jindo.$Agent().navigator().ie;
|
|
|
|
this._assignHTMLObjects(this.oApp.htOptions.elAppContainer);
|
|
|
|
this.charSet = [];
|
|
this.charSet[0] = unescape('FF5B FF5D 3014 3015 3008 3009 300A 300B 300C 300D 300E 300F 3010 3011 2018 2019 201C 201D 3001 3002 %B7 2025 2026 %A7 203B 2606 2605 25CB 25CF 25CE 25C7 25C6 25A1 25A0 25B3 25B2 25BD 25BC 25C1 25C0 25B7 25B6 2664 2660 2661 2665 2667 2663 2299 25C8 25A3 25D0 25D1 2592 25A4 25A5 25A8 25A7 25A6 25A9 %B1 %D7 %F7 2260 2264 2265 221E 2234 %B0 2032 2033 2220 22A5 2312 2202 2261 2252 226A 226B 221A 223D 221D 2235 222B 222C 2208 220B 2286 2287 2282 2283 222A 2229 2227 2228 FFE2 21D2 21D4 2200 2203 %B4 FF5E 02C7 02D8 02DD 02DA 02D9 %B8 02DB %A1 %BF 02D0 222E 2211 220F 266D 2669 266A 266C 327F 2192 2190 2191 2193 2194 2195 2197 2199 2196 2198 321C 2116 33C7 2122 33C2 33D8 2121 2668 260F 260E 261C 261E %B6 2020 2021 %AE %AA %BA 2642 2640').replace(/(\S{4})/g, function(a){return "%u"+a;}).split(' ');
|
|
this.charSet[1] = unescape('%BD 2153 2154 %BC %BE 215B 215C 215D 215E %B9 %B2 %B3 2074 207F 2081 2082 2083 2084 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172 2173 2174 2175 2176 2177 2178 2179 FFE6 %24 FFE5 FFE1 20AC 2103 212B 2109 FFE0 %A4 2030 3395 3396 3397 2113 3398 33C4 33A3 33A4 33A5 33A6 3399 339A 339B 339C 339D 339E 339F 33A0 33A1 33A2 33CA 338D 338E 338F 33CF 3388 3389 33C8 33A7 33A8 33B0 33B1 33B2 33B3 33B4 33B5 33B6 33B7 33B8 33B9 3380 3381 3382 3383 3384 33BA 33BB 33BC 33BD 33BE 33BF 3390 3391 3392 3393 3394 2126 33C0 33C1 338A 338B 338C 33D6 33C5 33AD 33AE 33AF 33DB 33A9 33AA 33AB 33AC 33DD 33D0 33D3 33C3 33C9 33DC 33C6').replace(/(\S{4})/g, function(a){return "%u"+a;}).split(' ');
|
|
this.charSet[2] = unescape('3260 3261 3262 3263 3264 3265 3266 3267 3268 3269 326A 326B 326C 326D 326E 326F 3270 3271 3272 3273 3274 3275 3276 3277 3278 3279 327A 327B 24D0 24D1 24D2 24D3 24D4 24D5 24D6 24D7 24D8 24D9 24DA 24DB 24DC 24DD 24DE 24DF 24E0 24E1 24E2 24E3 24E4 24E5 24E6 24E7 24E8 24E9 2460 2461 2462 2463 2464 2465 2466 2467 2468 2469 246A 246B 246C 246D 246E 3200 3201 3202 3203 3204 3205 3206 3207 3208 3209 320A 320B 320C 320D 320E 320F 3210 3211 3212 3213 3214 3215 3216 3217 3218 3219 321A 321B 249C 249D 249E 249F 24A0 24A1 24A2 24A3 24A4 24A5 24A6 24A7 24A8 24A9 24AA 24AB 24AC 24AD 24AE 24AF 24B0 24B1 24B2 24B3 24B4 24B5 2474 2475 2476 2477 2478 2479 247A 247B 247C 247D 247E 247F 2480 2481 2482').replace(/(\S{4})/g, function(a){return "%u"+a;}).split(' ');
|
|
this.charSet[3] = unescape('3131 3132 3133 3134 3135 3136 3137 3138 3139 313A 313B 313C 313D 313E 313F 3140 3141 3142 3143 3144 3145 3146 3147 3148 3149 314A 314B 314C 314D 314E 314F 3150 3151 3152 3153 3154 3155 3156 3157 3158 3159 315A 315B 315C 315D 315E 315F 3160 3161 3162 3163 3165 3166 3167 3168 3169 316A 316B 316C 316D 316E 316F 3170 3171 3172 3173 3174 3175 3176 3177 3178 3179 317A 317B 317C 317D 317E 317F 3180 3181 3182 3183 3184 3185 3186 3187 3188 3189 318A 318B 318C 318D 318E').replace(/(\S{4})/g, function(a){return "%u"+a;}).split(' ');
|
|
this.charSet[4] = unescape('0391 0392 0393 0394 0395 0396 0397 0398 0399 039A 039B 039C 039D 039E 039F 03A0 03A1 03A3 03A4 03A5 03A6 03A7 03A8 03A9 03B1 03B2 03B3 03B4 03B5 03B6 03B7 03B8 03B9 03BA 03BB 03BC 03BD 03BE 03BF 03C0 03C1 03C3 03C4 03C5 03C6 03C7 03C8 03C9 %C6 %D0 0126 0132 013F 0141 %D8 0152 %DE 0166 014A %E6 0111 %F0 0127 I 0133 0138 0140 0142 0142 0153 %DF %FE 0167 014B 0149 0411 0413 0414 0401 0416 0417 0418 0419 041B 041F 0426 0427 0428 0429 042A 042B 042C 042D 042E 042F 0431 0432 0433 0434 0451 0436 0437 0438 0439 043B 043F 0444 0446 0447 0448 0449 044A 044B 044C 044D 044E 044F').replace(/(\S{4})/g, function(a){return "%u"+a;}).split(' ');
|
|
this.charSet[5] = unescape('3041 3042 3043 3044 3045 3046 3047 3048 3049 304A 304B 304C 304D 304E 304F 3050 3051 3052 3053 3054 3055 3056 3057 3058 3059 305A 305B 305C 305D 305E 305F 3060 3061 3062 3063 3064 3065 3066 3067 3068 3069 306A 306B 306C 306D 306E 306F 3070 3071 3072 3073 3074 3075 3076 3077 3078 3079 307A 307B 307C 307D 307E 307F 3080 3081 3082 3083 3084 3085 3086 3087 3088 3089 308A 308B 308C 308D 308E 308F 3090 3091 3092 3093 30A1 30A2 30A3 30A4 30A5 30A6 30A7 30A8 30A9 30AA 30AB 30AC 30AD 30AE 30AF 30B0 30B1 30B2 30B3 30B4 30B5 30B6 30B7 30B8 30B9 30BA 30BB 30BC 30BD 30BE 30BF 30C0 30C1 30C2 30C3 30C4 30C5 30C6 30C7 30C8 30C9 30CA 30CB 30CC 30CD 30CE 30CF 30D0 30D1 30D2 30D3 30D4 30D5 30D6 30D7 30D8 30D9 30DA 30DB 30DC 30DD 30DE 30DF 30E0 30E1 30E2 30E3 30E4 30E5 30E6 30E7 30E8 30E9 30EA 30EB 30EC 30ED 30EE 30EF 30F0 30F1 30F2 30F3 30F4 30F5 30F6').replace(/(\S{4})/g, function(a){return "%u"+a;}).split(' ');
|
|
|
|
var funcInsert = jindo.$Fn(this.oApp.exec, this.oApp).bind("INSERT_SCHARACTERS", [this.oTextField.value]);
|
|
jindo.$Fn(funcInsert, this).attach(this.oInsertButton, "click");
|
|
|
|
this.oApp.exec("SET_SCHARACTER_LIST", [this.charSet]);
|
|
|
|
for(var i=0; i<this.aLabel.length; i++){
|
|
var func = jindo.$Fn(this.oApp.exec, this.oApp).bind("CHANGE_SCHARACTER_SET", [i]);
|
|
jindo.$Fn(func, this).attach(this.aLabel[i].firstChild, "mousedown");
|
|
}
|
|
|
|
for(var i=0; i<this.aCloseButton.length; i++){
|
|
this.oApp.registerBrowserEvent(this.aCloseButton[i], "click", "HIDE_ACTIVE_LAYER", []);
|
|
}
|
|
|
|
this.oApp.registerBrowserEvent(this.elDropdownLayer, "click", "EVENT_SCHARACTER_CLICKED", []);
|
|
|
|
// [SMARTEDITORSUS-1767]
|
|
this.oApp.registerBrowserEvent(this.oTextField, "keydown", "EVENT_SCHARACTER_KEYDOWN");
|
|
// --[SMARTEDITORSUS-1767]
|
|
},
|
|
|
|
// [SMARTEDITORSUS-1767]
|
|
$ON_EVENT_SCHARACTER_KEYDOWN : function(oEvent){
|
|
if (oEvent.key().enter){
|
|
this.oApp.exec("INSERT_SCHARACTERS");
|
|
oEvent.stop();
|
|
}
|
|
},
|
|
// --[SMARTEDITORSUS-1767]
|
|
|
|
$ON_TOGGLE_SCHARACTER_LAYER : function(){
|
|
this.oTextField.value = "";
|
|
this.oSelection = this.oApp.getSelection();
|
|
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.elDropdownLayer, null, "MSG_SCHARACTER_LAYER_SHOWN", [], "MSG_SCHARACTER_LAYER_HIDDEN", [""]]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['symbol']);
|
|
},
|
|
|
|
$ON_MSG_SCHARACTER_LAYER_SHOWN : function(){
|
|
this.oTextField.focus();
|
|
this.oApp.exec("SELECT_UI", ["sCharacter"]);
|
|
},
|
|
|
|
$ON_MSG_SCHARACTER_LAYER_HIDDEN : function(){
|
|
this.oApp.exec("DESELECT_UI", ["sCharacter"]);
|
|
},
|
|
|
|
$ON_EVENT_SCHARACTER_CLICKED : function(weEvent){
|
|
var elButton = nhn.husky.SE2M_Utils.findAncestorByTagName("BUTTON", weEvent.element);
|
|
if(!elButton || elButton.tagName != "BUTTON"){return;}
|
|
|
|
if(elButton.parentNode.tagName != "LI"){return;}
|
|
|
|
var sChar = elButton.firstChild.innerHTML;
|
|
if(sChar.length > 1){return;}
|
|
|
|
this.oApp.exec("SELECT_SCHARACTER", [sChar]);
|
|
weEvent.stop();
|
|
},
|
|
|
|
$ON_SELECT_SCHARACTER : function(schar){
|
|
this.oTextField.value += schar;
|
|
|
|
if(this.oTextField.createTextRange){
|
|
var oTextRange = this.oTextField.createTextRange();
|
|
oTextRange.collapse(false);
|
|
oTextRange.select();
|
|
}else{
|
|
if(this.oTextField.selectionEnd){
|
|
this.oTextField.selectionEnd = this.oTextField.value.length;
|
|
this.oTextField.focus();
|
|
}
|
|
}
|
|
},
|
|
|
|
$ON_INSERT_SCHARACTERS : function(){
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["INSERT SCHARACTER"]);
|
|
this.oApp.exec("PASTE_HTML", [this.oTextField.value]);
|
|
this.oApp.exec("FOCUS");
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["INSERT SCHARACTER"]);
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
},
|
|
|
|
$ON_CHANGE_SCHARACTER_SET : function(nSCharSet){
|
|
for(var i=0; i<this.aSCharList.length; i++){
|
|
if(jindo.$Element(this.aLabel[i]).hasClass("active")){
|
|
if(i == nSCharSet){return;}
|
|
|
|
jindo.$Element(this.aLabel[i]).removeClass("active");
|
|
}
|
|
}
|
|
|
|
this._drawSCharList(nSCharSet);
|
|
jindo.$Element(this.aLabel[nSCharSet]).addClass("active");
|
|
},
|
|
|
|
$ON_SET_SCHARACTER_LIST : function(charSet){
|
|
this.charSet = charSet;
|
|
this.bSCharSetDrawn = new Array(this.charSet.length);
|
|
this._drawSCharList(0);
|
|
},
|
|
|
|
_drawSCharList : function(i){
|
|
if(this.bSCharSetDrawn[i]){return;}
|
|
this.bSCharSetDrawn[i] = true;
|
|
|
|
var len = this.charSet[i].length;
|
|
var aLI = new Array(len);
|
|
|
|
this.aSCharList[i].innerHTML = '';
|
|
|
|
var button, span;
|
|
for(var ii=0; ii<len; ii++){
|
|
aLI[ii] = jindo.$("<LI>");
|
|
|
|
if(this.bIE){
|
|
button = jindo.$("<BUTTON>");
|
|
button.setAttribute('type', 'button');
|
|
}else{
|
|
button = jindo.$("<BUTTON>");
|
|
button.type = "button";
|
|
}
|
|
span = jindo.$("<SPAN>");
|
|
span.innerHTML = unescape(this.charSet[i][ii]);
|
|
button.appendChild(span);
|
|
|
|
aLI[ii].appendChild(button);
|
|
aLI[ii].onmouseover = function(){this.className='hover'};
|
|
aLI[ii].onmouseout = function(){this.className=''};
|
|
|
|
this.aSCharList[i].appendChild(aLI[ii]);
|
|
}
|
|
|
|
//this.oApp.delayedExec("SE2_ATTACH_HOVER_EVENTS", [jindo.$$(">LI", this.aSCharList[i]), 0]);
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_TableCreator$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_TableCreator
|
|
* this.oApp.registerLazyMessage(["TOGGLE_TABLE_LAYER"], ["hp_SE2M_TableCreator$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_TableCreator, {
|
|
//@lazyload_js TOGGLE_TABLE_LAYER[
|
|
_assignHTMLObjects : function(oAppContainer){
|
|
this.oApp.exec("LOAD_HTML", ["create_table"]);
|
|
var tmp = null;
|
|
|
|
this.elDropdownLayer = jindo.$$.getSingle("DIV.husky_se2m_table_layer", oAppContainer);
|
|
this.welDropdownLayer = jindo.$Element(this.elDropdownLayer);
|
|
|
|
tmp = jindo.$$("INPUT", this.elDropdownLayer);
|
|
this.elText_row = tmp[0];
|
|
this.elText_col = tmp[1];
|
|
this.elRadio_manualStyle = tmp[2];
|
|
this.elText_borderSize = tmp[3];
|
|
this.elText_borderColor = tmp[4];
|
|
this.elText_BGColor = tmp[5];
|
|
this.elRadio_templateStyle = tmp[6];
|
|
|
|
tmp = jindo.$$("BUTTON", this.elDropdownLayer);
|
|
this.elBtn_rowInc = tmp[0];
|
|
this.elBtn_rowDec = tmp[1];
|
|
this.elBtn_colInc = tmp[2];
|
|
this.elBtn_colDec = tmp[3];
|
|
this.elBtn_borderStyle = tmp[4];
|
|
this.elBtn_incBorderSize = jindo.$$.getSingle("BUTTON.se2m_incBorder", this.elDropdownLayer);
|
|
this.elBtn_decBorderSize = jindo.$$.getSingle("BUTTON.se2m_decBorder", this.elDropdownLayer);
|
|
|
|
this.elLayer_Dim1 = jindo.$$.getSingle("DIV.se2_t_dim0", this.elDropdownLayer);
|
|
this.elLayer_Dim2 = jindo.$$.getSingle("DIV.se2_t_dim3", this.elDropdownLayer);
|
|
|
|
// border style layer contains btn elm's
|
|
|
|
tmp = jindo.$$("SPAN.se2_pre_color>BUTTON", this.elDropdownLayer);
|
|
|
|
this.elBtn_borderColor = tmp[0];
|
|
this.elBtn_BGColor = tmp[1];
|
|
|
|
this.elBtn_tableStyle = jindo.$$.getSingle("DIV.se2_select_ty2>BUTTON", this.elDropdownLayer);
|
|
|
|
tmp = jindo.$$("P.se2_btn_area>BUTTON", this.elDropdownLayer);
|
|
this.elBtn_apply = tmp[0];
|
|
this.elBtn_cancel = tmp[1];
|
|
|
|
this.elTable_preview = jindo.$$.getSingle("TABLE.husky_se2m_table_preview", this.elDropdownLayer);
|
|
this.elLayer_borderStyle = jindo.$$.getSingle("DIV.husky_se2m_table_border_style_layer", this.elDropdownLayer);
|
|
this.elPanel_borderStylePreview = jindo.$$.getSingle("SPAN.husky_se2m_table_border_style_preview", this.elDropdownLayer);
|
|
this.elPanel_borderColorPallet = jindo.$$.getSingle("DIV.husky_se2m_table_border_color_pallet", this.elDropdownLayer);
|
|
this.elPanel_bgColorPallet = jindo.$$.getSingle("DIV.husky_se2m_table_bgcolor_pallet", this.elDropdownLayer);
|
|
this.elLayer_tableStyle = jindo.$$.getSingle("DIV.husky_se2m_table_style_layer", this.elDropdownLayer);
|
|
this.elPanel_tableStylePreview = jindo.$$.getSingle("SPAN.husky_se2m_table_style_preview", this.elDropdownLayer);
|
|
|
|
this.aElBtn_borderStyle = jindo.$$("BUTTON", this.elLayer_borderStyle);
|
|
this.aElBtn_tableStyle = jindo.$$("BUTTON", this.elLayer_tableStyle);
|
|
|
|
this.sNoBorderText = jindo.$$.getSingle("SPAN.se2m_no_border", this.elDropdownLayer).innerHTML;
|
|
|
|
this.rxLastDigits = RegExp("([0-9]+)$");
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(){
|
|
this._assignHTMLObjects(this.oApp.htOptions.elAppContainer);
|
|
|
|
this.oApp.registerBrowserEvent(this.elText_row, "change", "TABLE_SET_ROW_NUM", [null, 0]);
|
|
this.oApp.registerBrowserEvent(this.elText_col, "change", "TABLE_SET_COLUMN_NUM", [null, 0]);
|
|
this.oApp.registerBrowserEvent(this.elText_borderSize, "change", "TABLE_SET_BORDER_SIZE", [null, 0]);
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtn_rowInc, "click", "TABLE_INC_ROW");
|
|
this.oApp.registerBrowserEvent(this.elBtn_rowDec, "click", "TABLE_DEC_ROW");
|
|
jindo.$Fn(this._numRowKeydown, this).attach(this.elText_row.parentNode, "keydown");
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtn_colInc, "click", "TABLE_INC_COLUMN");
|
|
this.oApp.registerBrowserEvent(this.elBtn_colDec, "click", "TABLE_DEC_COLUMN");
|
|
jindo.$Fn(this._numColKeydown, this).attach(this.elText_col.parentNode, "keydown");
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtn_incBorderSize, "click", "TABLE_INC_BORDER_SIZE");
|
|
this.oApp.registerBrowserEvent(this.elBtn_decBorderSize, "click", "TABLE_DEC_BORDER_SIZE");
|
|
jindo.$Fn(this._borderSizeKeydown, this).attach(this.elText_borderSize.parentNode, "keydown");
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtn_borderStyle, "click", "TABLE_TOGGLE_BORDER_STYLE_LAYER");
|
|
this.oApp.registerBrowserEvent(this.elBtn_tableStyle, "click", "TABLE_TOGGLE_STYLE_LAYER");
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtn_borderColor, "click", "TABLE_TOGGLE_BORDER_COLOR_PALLET");
|
|
this.oApp.registerBrowserEvent(this.elBtn_BGColor, "click", "TABLE_TOGGLE_BGCOLOR_PALLET");
|
|
|
|
this.oApp.registerBrowserEvent(this.elRadio_manualStyle, "click", "TABLE_ENABLE_MANUAL_STYLE");
|
|
this.oApp.registerBrowserEvent(this.elRadio_templateStyle, "click", "TABLE_ENABLE_TEMPLATE_STYLE");
|
|
|
|
//this.oApp.registerBrowserEvent(this.elDropdownLayer, "click", "TABLE_LAYER_CLICKED");
|
|
//this.oApp.registerBrowserEvent(this.elLayer_borderStyle, "click", "TABLE_BORDER_STYLE_LAYER_CLICKED");
|
|
//this.oApp.registerBrowserEvent(this.elLayer_tableStyle, "click", "TABLE_STYLE_LAYER_CLICKED");
|
|
|
|
this.oApp.exec("SE2_ATTACH_HOVER_EVENTS", [this.aElBtn_borderStyle]);
|
|
this.oApp.exec("SE2_ATTACH_HOVER_EVENTS", [this.aElBtn_tableStyle]);
|
|
|
|
var i;
|
|
for(i=0; i<this.aElBtn_borderStyle.length; i++){
|
|
this.oApp.registerBrowserEvent(this.aElBtn_borderStyle[i], "click", "TABLE_SELECT_BORDER_STYLE");
|
|
}
|
|
|
|
for(i=0; i<this.aElBtn_tableStyle.length; i++){
|
|
this.oApp.registerBrowserEvent(this.aElBtn_tableStyle[i], "click", "TABLE_SELECT_STYLE");
|
|
}
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtn_apply, "click", "TABLE_INSERT");
|
|
this.oApp.registerBrowserEvent(this.elBtn_cancel, "click", "HIDE_ACTIVE_LAYER");
|
|
|
|
this.oApp.exec("TABLE_SET_BORDER_COLOR", [/#[0-9A-Fa-f]{6}/.test(this.elText_borderColor.value) ? this.elText_borderColor.value : "#cccccc"]);
|
|
this.oApp.exec("TABLE_SET_BGCOLOR", [/#[0-9A-Fa-f]{6}/.test(this.elText_BGColor.value) ? this.elText_BGColor.value : "#ffffff"]);
|
|
|
|
// 1: manual style
|
|
// 2: template style
|
|
this.nStyleMode = 1;
|
|
|
|
// add #BorderSize+x# if needed
|
|
//---
|
|
// [SMARTEDITORSUS-365] 테이블퀵에디터 > 속성 직접입력 > 테두리 스타일
|
|
// - 테두리 없음을 선택하는 경우 본문에 삽입하는 표에 가이드 라인을 표시해 줍니다. 보기 시에는 테두리가 보이지 않습니다.
|
|
this.aTableStyleByBorder = [
|
|
'',
|
|
'border="1" cellpadding="0" cellspacing="0" style="border:1px dashed #c7c7c7; border-left:0; border-bottom:0;"',
|
|
'border="1" cellpadding="0" cellspacing="0" style="border:#BorderSize#px dashed #BorderColor#; border-left:0; border-bottom:0;"',
|
|
'border="0" cellpadding="0" cellspacing="0" style="border:#BorderSize#px solid #BorderColor#; border-left:0; border-bottom:0;"',
|
|
'border="0" cellpadding="0" cellspacing="1" style="border:#BorderSize#px solid #BorderColor#;"',
|
|
'border="0" cellpadding="0" cellspacing="1" style="border:#BorderSize#px double #BorderColor#;"',
|
|
'border="0" cellpadding="0" cellspacing="1" style="border-width:#BorderSize*2#px #BorderSize#px #BorderSize#px #BorderSize*2#px; border-style:solid;border-color:#BorderColor#;"',
|
|
'border="0" cellpadding="0" cellspacing="1" style="border-width:#BorderSize#px #BorderSize*2#px #BorderSize*2#px #BorderSize#px; border-style:solid;border-color:#BorderColor#;"'
|
|
];
|
|
|
|
this.aTDStyleByBorder = [
|
|
'',
|
|
'style="border:1px dashed #c7c7c7; border-top:0; border-right:0; background-color:#BGColor#"',
|
|
'style="border:#BorderSize#px dashed #BorderColor#; border-top:0; border-right:0; background-color:#BGColor#"',
|
|
'style="border:#BorderSize#px solid #BorderColor#; border-top:0; border-right:0; background-color:#BGColor#"',
|
|
'style="border:#BorderSize#px solid #BorderColor#; background-color:#BGColor#"',
|
|
'style="border:#BorderSize+2#px double #BorderColor#; background-color:#BGColor#"',
|
|
'style="border-width:#BorderSize#px #BorderSize*2#px #BorderSize*2#px #BorderSize#px; border-style:solid;border-color:#BorderColor#; background-color:#BGColor#"',
|
|
'style="border-width:#BorderSize*2#px #BorderSize#px #BorderSize#px #BorderSize*2#px; border-style:solid;border-color:#BorderColor#; background-color:#BGColor#"'
|
|
];
|
|
this.oApp.registerBrowserEvent(this.elDropdownLayer, "keydown", "EVENT_TABLE_CREATE_KEYDOWN");
|
|
|
|
this._drawTableDropdownLayer();
|
|
},
|
|
|
|
$ON_TABLE_SELECT_BORDER_STYLE : function(weEvent){
|
|
var elButton = weEvent.currentElement;
|
|
// var aMatch = this.rxLastDigits.exec(weEvent.element.className);
|
|
var aMatch = this.rxLastDigits.exec(elButton.className);
|
|
this._selectBorderStyle(aMatch[1]);
|
|
},
|
|
|
|
$ON_TABLE_SELECT_STYLE : function(weEvent){
|
|
var aMatch = this.rxLastDigits.exec(weEvent.element.className);
|
|
this._selectTableStyle(aMatch[1]);
|
|
},
|
|
|
|
$ON_TOGGLE_TABLE_LAYER : function(){
|
|
// this.oSelection = this.oApp.getSelection();
|
|
this._showNewTable();
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.elDropdownLayer, null, "SELECT_UI", ["table"], "TABLE_CLOSE", []]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['table']);
|
|
},
|
|
|
|
// $ON_TABLE_BORDER_STYLE_LAYER_CLICKED : function(weEvent){
|
|
// top.document.title = weEvent.element.tagName;
|
|
// },
|
|
|
|
$ON_TABLE_CLOSE_ALL : function(){
|
|
this.oApp.exec("TABLE_HIDE_BORDER_COLOR_PALLET", []);
|
|
this.oApp.exec("TABLE_HIDE_BGCOLOR_PALLET", []);
|
|
this.oApp.exec("TABLE_HIDE_BORDER_STYLE_LAYER", []);
|
|
this.oApp.exec("TABLE_HIDE_STYLE_LAYER", []);
|
|
},
|
|
|
|
$ON_TABLE_INC_ROW : function(){
|
|
this.oApp.exec("TABLE_SET_ROW_NUM", [null, 1]);
|
|
},
|
|
|
|
$ON_TABLE_DEC_ROW : function(){
|
|
this.oApp.exec("TABLE_SET_ROW_NUM", [null, -1]);
|
|
},
|
|
|
|
$ON_TABLE_INC_COLUMN : function(){
|
|
this.oApp.exec("TABLE_SET_COLUMN_NUM", [null, 1]);
|
|
},
|
|
|
|
$ON_TABLE_DEC_COLUMN : function(){
|
|
this.oApp.exec("TABLE_SET_COLUMN_NUM", [null, -1]);
|
|
},
|
|
|
|
$ON_TABLE_SET_ROW_NUM : function(nRows, nRowDiff){
|
|
nRows = nRows || parseInt(this.elText_row.value, 10) || 0;
|
|
nRowDiff = nRowDiff || 0;
|
|
|
|
nRows += nRowDiff;
|
|
|
|
if(nRows < this.nMinRows){nRows = this.nMinRows;}
|
|
if(nRows > this.nMaxRows){nRows = this.nMaxRows;}
|
|
|
|
this.elText_row.value = nRows;
|
|
this._showNewTable();
|
|
},
|
|
|
|
$ON_TABLE_SET_COLUMN_NUM : function(nColumns, nColumnDiff){
|
|
nColumns = nColumns || parseInt(this.elText_col.value, 10) || 0;
|
|
nColumnDiff = nColumnDiff || 0;
|
|
|
|
nColumns += nColumnDiff;
|
|
|
|
if(nColumns < this.nMinColumns){nColumns = this.nMinColumns;}
|
|
if(nColumns > this.nMaxColumns){nColumns = this.nMaxColumns;}
|
|
|
|
this.elText_col.value = nColumns;
|
|
this._showNewTable();
|
|
},
|
|
|
|
_getTableString : function(){
|
|
var sTable;
|
|
if(this.nStyleMode == 1){
|
|
sTable = this._doGetTableString(this.nColumns, this.nRows, this.nBorderSize, this.sBorderColor, this.sBGColor, this.nBorderStyleIdx);
|
|
}else{
|
|
sTable = this._doGetTableString(this.nColumns, this.nRows, this.nBorderSize, this.sBorderColor, this.sBGColor, 0);
|
|
}
|
|
|
|
return sTable;
|
|
},
|
|
|
|
$ON_TABLE_INSERT : function(){
|
|
this.oApp.exec("IE_FOCUS", []); // [SMARTEDITORSUS-500] IE인 경우 명시적인 focus 추가
|
|
|
|
//[SMARTEDITORSUS-596]이벤트 발생이 안되는 경우,
|
|
//max 제한이 적용이 안되기 때문에 테이블 사입 시점에 다시한번 Max 값을 검사한다.
|
|
this.oApp.exec("TABLE_SET_COLUMN_NUM");
|
|
this.oApp.exec("TABLE_SET_ROW_NUM");
|
|
|
|
this._loadValuesFromHTML();
|
|
|
|
var sTable,
|
|
elLinebreak,
|
|
elBody,
|
|
welBody,
|
|
elTmpDiv,
|
|
elTable,
|
|
elFirstTD,
|
|
oSelection,
|
|
elTableHolder,
|
|
htBrowser;
|
|
|
|
elBody = this.oApp.getWYSIWYGDocument().body;
|
|
welBody = jindo.$Element(elBody);
|
|
htBrowser = jindo.$Agent().navigator();
|
|
|
|
this.nTableWidth = elBody.offsetWidth;
|
|
|
|
sTable = this._getTableString();
|
|
|
|
elTmpDiv = this.oApp.getWYSIWYGDocument().createElement("DIV");
|
|
elTmpDiv.innerHTML = sTable;
|
|
elTable = elTmpDiv.firstChild;
|
|
elTable.className = this._sSETblClass;
|
|
|
|
oSelection = this.oApp.getSelection();
|
|
oSelection = this._divideParagraph(oSelection); // [SMARTEDITORSUS-306]
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["INSERT TABLE", {sSaveTarget:"BODY"}]);
|
|
|
|
// If the table were inserted within a styled(strikethough & etc) paragraph, the table may inherit the style in IE.
|
|
elTableHolder = this.oApp.getWYSIWYGDocument().createElement("DIV");
|
|
// 영역을 잡았을 경우, 영역 지우고 테이블 삽입
|
|
oSelection.deleteContents();
|
|
oSelection.insertNode(elTableHolder);
|
|
oSelection.selectNode(elTableHolder);
|
|
this.oApp.exec("REMOVE_STYLE", [oSelection]);
|
|
|
|
if(htBrowser.ie && this.oApp.getWYSIWYGDocument().body.childNodes.length === 1 && this.oApp.getWYSIWYGDocument().body.firstChild === elTableHolder){
|
|
// IE에서 table이 body에 바로 붙어 있을 경우, 정렬등에서 문제가 발생 함으로 elTableHolder(DIV)를 남겨둠
|
|
elTableHolder.insertBefore(elTable, null);
|
|
}else{
|
|
elTableHolder.parentNode.insertBefore(elTable, elTableHolder);
|
|
elTableHolder.parentNode.removeChild(elTableHolder);
|
|
}
|
|
|
|
// FF : 테이블 하단에 BR이 없을 경우, 커서가 테이블 밑으로 이동할 수 없어 BR을 삽입 해 줌.
|
|
//[SMARTEDITORSUS-181][IE9] 표나 요약글 등의 테이블에서 > 테이블 외부로 커서 이동 불가
|
|
if(htBrowser.firefox){
|
|
elLinebreak = this.oApp.getWYSIWYGDocument().createElement("BR");
|
|
elTable.parentNode.insertBefore(elLinebreak, elTable.nextSibling);
|
|
}else if(htBrowser.ie){
|
|
elLinebreak = this.oApp.getWYSIWYGDocument().createElement("p");
|
|
if(document.documentMode == 11){ // [SMARTEDITORSUS-2064] IE11인 경우 빈 P태그 안에 커서가 안들어가기 때문에 br 추가
|
|
elLinebreak.innerHTML = "<br>";
|
|
}
|
|
elTable.parentNode.insertBefore(elLinebreak, elTable.nextSibling);
|
|
}
|
|
|
|
if(this.nStyleMode == 2){
|
|
this.oApp.exec("STYLE_TABLE", [elTable, this.nTableStyleIdx]);
|
|
}
|
|
|
|
elFirstTD = elTable.getElementsByTagName("TD")[0];
|
|
oSelection.selectNodeContents(elFirstTD.firstChild || elFirstTD);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
|
|
this.oApp.exec("FOCUS");
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["INSERT TABLE", {sSaveTarget:"BODY"}]);
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER", []);
|
|
this.oApp.exec('MSG_DISPLAY_REEDIT_MESSAGE_SHOW', [this.name, this.sReEditGuideMsg_table]);
|
|
},
|
|
|
|
/**
|
|
* P 안에 Table 이 추가되지 않도록 P 태그를 분리함
|
|
*
|
|
* [SMARTEDITORSUS-306]
|
|
* P 에 Table 을 추가한 경우, DOM 에서 비정상적인 P 를 생성하여 깨지는 경우가 발생함
|
|
* 테이블이 추가되는 부분에 P 가 있는 경우, P 를 분리시켜주는 처리
|
|
*/
|
|
_divideParagraph : function(oSelection){
|
|
var oParentP,
|
|
welParentP,
|
|
sNodeVaule,
|
|
sBM, oSWrapper, oEWrapper;
|
|
|
|
oSelection.fixCommonAncestorContainer(); // [SMARTEDITORSUS-423] 엔터에 의해 생성된 P 가 아닌 이전 P 가 선택되지 않도록 fix 하도록 처리
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1735]
|
|
* 기존 로직은 selection을 기준으로 위로 거슬러 올라가면서
|
|
* 첫 번째로 만나는 <P>를 분리 기준으로 했다.
|
|
*
|
|
* 하지만 selection이 cell 내부에 있는 상태에서
|
|
* 이 cell 내부에는 <P> 가 없고
|
|
* 해당 table을 <P>가 감싸고 있는 경우,
|
|
* selection 기준으로 바깥의 <P>가 두 개로 나뉜 뒤
|
|
* 해당 table이 양분되는 현상이 발생한다.
|
|
*
|
|
* 따라서 selection이 cell 내부에 있는 경우에는,
|
|
* 이 cell 내부에 <P>가 있는 경우에만 분리를 수행할 수 있도록 보정한다.
|
|
* */
|
|
//oParentP = oSelection.findAncestorByTagName("P");
|
|
var _elCommonAncestorContainer = oSelection.commonAncestorContainer;
|
|
var _htAncestor_P = nhn.husky.SE2M_Utils.findAncestorByTagNameWithCount("P", _elCommonAncestorContainer);
|
|
var _elAncestor_P = _htAncestor_P.elNode;
|
|
if(_elAncestor_P){
|
|
var _htAncestor_Cell = nhn.husky.SE2M_Utils.findClosestAncestorAmongTagNamesWithCount(["TD", "TH"], _elCommonAncestorContainer);
|
|
var _elAncestor_Cell = _htAncestor_Cell.elNode;
|
|
|
|
if(_elAncestor_Cell){
|
|
var _nRecursiveCount_P = _htAncestor_P.nRecursiveCount;
|
|
var _nRecursiveCount_Cell = _htAncestor_Cell.nRecursiveCount;
|
|
|
|
// cell 안에 있는 <P>일 때만 분할
|
|
if(_nRecursiveCount_P < _nRecursiveCount_Cell){
|
|
oParentP = _elAncestor_P;
|
|
}
|
|
}else{ // selection을 기준으로 cell이 발견되지 않으면 그대로 진행
|
|
oParentP = _elAncestor_P;
|
|
}
|
|
}
|
|
|
|
// --[SMARTEDITORSUS-1735]
|
|
|
|
if(!oParentP){
|
|
return oSelection;
|
|
}
|
|
|
|
if(!oParentP.firstChild || nhn.husky.SE2M_Utils.isBlankNode(oParentP)){
|
|
oSelection.selectNode(oParentP); // [SMARTEDITORSUS-423] 불필요한 개행이 일어나지 않도록 빈 P 를 선택하여 TABLE 로 대체하도록 처리
|
|
oSelection.select();
|
|
|
|
return oSelection;
|
|
}
|
|
|
|
sBM = oSelection.placeStringBookmark();
|
|
|
|
oSelection.moveToBookmark(sBM);
|
|
|
|
oSWrapper = this.oApp.getWYSIWYGDocument().createElement("P");
|
|
oSelection.setStartBefore(oParentP.firstChild);
|
|
oSelection.surroundContents(oSWrapper);
|
|
oSelection.collapseToEnd();
|
|
|
|
oEWrapper = this.oApp.getWYSIWYGDocument().createElement("P");
|
|
oSelection.setEndAfter(oParentP.lastChild);
|
|
oSelection.surroundContents(oEWrapper);
|
|
oSelection.collapseToStart();
|
|
|
|
oSelection.removeStringBookmark(sBM);
|
|
|
|
welParentP = jindo.$Element(oParentP);
|
|
welParentP.after(oEWrapper);
|
|
welParentP.after(oSWrapper);
|
|
|
|
welParentP.leave();
|
|
|
|
oSelection = this.oApp.getEmptySelection();
|
|
|
|
oSelection.setEndAfter(oSWrapper);
|
|
oSelection.setStartBefore(oEWrapper);
|
|
|
|
oSelection.select();
|
|
|
|
return oSelection;
|
|
},
|
|
|
|
$ON_TABLE_CLOSE : function(){
|
|
this.oApp.exec("TABLE_CLOSE_ALL", []);
|
|
this.oApp.exec("DESELECT_UI", ["table"]);
|
|
},
|
|
|
|
$ON_TABLE_SET_BORDER_SIZE : function(nBorderWidth, nBorderWidthDiff){
|
|
nBorderWidth = nBorderWidth || parseInt(this.elText_borderSize.value, 10) || 0;
|
|
nBorderWidthDiff = nBorderWidthDiff || 0;
|
|
|
|
nBorderWidth += nBorderWidthDiff;
|
|
|
|
if(nBorderWidth < this.nMinBorderWidth){nBorderWidth = this.nMinBorderWidth;}
|
|
if(nBorderWidth > this.nMaxBorderWidth){nBorderWidth = this.nMaxBorderWidth;}
|
|
|
|
this.elText_borderSize.value = nBorderWidth;
|
|
},
|
|
|
|
$ON_TABLE_INC_BORDER_SIZE : function(){
|
|
this.oApp.exec("TABLE_SET_BORDER_SIZE", [null, 1]);
|
|
},
|
|
|
|
$ON_TABLE_DEC_BORDER_SIZE : function(){
|
|
this.oApp.exec("TABLE_SET_BORDER_SIZE", [null, -1]);
|
|
},
|
|
|
|
$ON_TABLE_TOGGLE_BORDER_STYLE_LAYER : function(){
|
|
if(this.elLayer_borderStyle.style.display == "block"){
|
|
this.oApp.exec("TABLE_HIDE_BORDER_STYLE_LAYER", []);
|
|
}else{
|
|
this.oApp.exec("TABLE_SHOW_BORDER_STYLE_LAYER", []);
|
|
}
|
|
},
|
|
|
|
$ON_TABLE_SHOW_BORDER_STYLE_LAYER : function(){
|
|
this.oApp.exec("TABLE_CLOSE_ALL", []);
|
|
this.elBtn_borderStyle.className = "se2_view_more2";
|
|
this.elLayer_borderStyle.style.display = "block";
|
|
this._refresh();
|
|
},
|
|
|
|
$ON_TABLE_HIDE_BORDER_STYLE_LAYER : function(){
|
|
this.elBtn_borderStyle.className = "se2_view_more";
|
|
this.elLayer_borderStyle.style.display = "none";
|
|
this._refresh();
|
|
},
|
|
|
|
$ON_TABLE_TOGGLE_STYLE_LAYER : function(){
|
|
if(this.elLayer_tableStyle.style.display == "block"){
|
|
this.oApp.exec("TABLE_HIDE_STYLE_LAYER", []);
|
|
}else{
|
|
this.oApp.exec("TABLE_SHOW_STYLE_LAYER", []);
|
|
}
|
|
},
|
|
|
|
$ON_TABLE_SHOW_STYLE_LAYER : function(){
|
|
this.oApp.exec("TABLE_CLOSE_ALL", []);
|
|
this.elBtn_tableStyle.className = "se2_view_more2";
|
|
this.elLayer_tableStyle.style.display = "block";
|
|
this._refresh();
|
|
},
|
|
|
|
$ON_TABLE_HIDE_STYLE_LAYER : function(){
|
|
this.elBtn_tableStyle.className = "se2_view_more";
|
|
this.elLayer_tableStyle.style.display = "none";
|
|
this._refresh();
|
|
},
|
|
|
|
$ON_TABLE_TOGGLE_BORDER_COLOR_PALLET : function(){
|
|
if(this.welDropdownLayer.hasClass("p1")){
|
|
this.oApp.exec("TABLE_HIDE_BORDER_COLOR_PALLET", []);
|
|
}else{
|
|
this.oApp.exec("TABLE_SHOW_BORDER_COLOR_PALLET", []);
|
|
}
|
|
},
|
|
|
|
$ON_TABLE_SHOW_BORDER_COLOR_PALLET : function(){
|
|
this.oApp.exec("TABLE_CLOSE_ALL", []);
|
|
|
|
this.welDropdownLayer.addClass("p1");
|
|
this.welDropdownLayer.removeClass("p2");
|
|
|
|
this.oApp.exec("SHOW_COLOR_PALETTE", ["TABLE_SET_BORDER_COLOR_FROM_PALETTE", this.elPanel_borderColorPallet]);
|
|
this.elPanel_borderColorPallet.parentNode.style.display = "block";
|
|
},
|
|
|
|
$ON_TABLE_HIDE_BORDER_COLOR_PALLET : function(){
|
|
this.welDropdownLayer.removeClass("p1");
|
|
|
|
this.oApp.exec("HIDE_COLOR_PALETTE", []);
|
|
this.elPanel_borderColorPallet.parentNode.style.display = "none";
|
|
},
|
|
|
|
$ON_TABLE_TOGGLE_BGCOLOR_PALLET : function(){
|
|
if(this.welDropdownLayer.hasClass("p2")){
|
|
this.oApp.exec("TABLE_HIDE_BGCOLOR_PALLET", []);
|
|
}else{
|
|
this.oApp.exec("TABLE_SHOW_BGCOLOR_PALLET", []);
|
|
}
|
|
},
|
|
|
|
$ON_TABLE_SHOW_BGCOLOR_PALLET : function(){
|
|
this.oApp.exec("TABLE_CLOSE_ALL", []);
|
|
|
|
this.welDropdownLayer.removeClass("p1");
|
|
this.welDropdownLayer.addClass("p2");
|
|
|
|
this.oApp.exec("SHOW_COLOR_PALETTE", ["TABLE_SET_BGCOLOR_FROM_PALETTE", this.elPanel_bgColorPallet]);
|
|
this.elPanel_bgColorPallet.parentNode.style.display = "block";
|
|
},
|
|
|
|
$ON_TABLE_HIDE_BGCOLOR_PALLET : function(){
|
|
this.welDropdownLayer.removeClass("p2");
|
|
|
|
this.oApp.exec("HIDE_COLOR_PALETTE", []);
|
|
this.elPanel_bgColorPallet.parentNode.style.display = "none";
|
|
},
|
|
|
|
$ON_TABLE_SET_BORDER_COLOR_FROM_PALETTE : function(sColorCode){
|
|
this.oApp.exec("TABLE_SET_BORDER_COLOR", [sColorCode]);
|
|
this.oApp.exec("TABLE_HIDE_BORDER_COLOR_PALLET", []);
|
|
},
|
|
|
|
$ON_TABLE_SET_BORDER_COLOR : function(sColorCode){
|
|
this.elText_borderColor.value = sColorCode;
|
|
this.elBtn_borderColor.style.backgroundColor = sColorCode;
|
|
},
|
|
|
|
$ON_TABLE_SET_BGCOLOR_FROM_PALETTE : function(sColorCode){
|
|
this.oApp.exec("TABLE_SET_BGCOLOR", [sColorCode]);
|
|
this.oApp.exec("TABLE_HIDE_BGCOLOR_PALLET", []);
|
|
},
|
|
|
|
$ON_TABLE_SET_BGCOLOR : function(sColorCode){
|
|
this.elText_BGColor.value = sColorCode;
|
|
this.elBtn_BGColor.style.backgroundColor = sColorCode;
|
|
},
|
|
|
|
$ON_TABLE_ENABLE_MANUAL_STYLE : function(){
|
|
this.nStyleMode = 1;
|
|
this._drawTableDropdownLayer();
|
|
},
|
|
|
|
$ON_TABLE_ENABLE_TEMPLATE_STYLE : function(){
|
|
this.nStyleMode = 2;
|
|
this._drawTableDropdownLayer();
|
|
},
|
|
|
|
$ON_EVENT_TABLE_CREATE_KEYDOWN : function(oEvent){
|
|
if (oEvent.key().enter){
|
|
this.elBtn_apply.focus();
|
|
this.oApp.exec("TABLE_INSERT");
|
|
oEvent.stop();
|
|
}
|
|
},
|
|
|
|
_drawTableDropdownLayer : function(){
|
|
if(this.nBorderStyleIdx == 1){
|
|
this.elPanel_borderStylePreview.innerHTML = this.sNoBorderText;
|
|
this.elLayer_Dim1.className = "se2_t_dim2";
|
|
}else{
|
|
this.elPanel_borderStylePreview.innerHTML = "";
|
|
this.elLayer_Dim1.className = "se2_t_dim0";
|
|
}
|
|
|
|
if(this.nStyleMode == 1){
|
|
this.elRadio_manualStyle.checked = true;
|
|
this.elLayer_Dim2.className = "se2_t_dim3";
|
|
|
|
this.elText_borderSize.disabled = false;
|
|
this.elText_borderColor.disabled = false;
|
|
this.elText_BGColor.disabled = false;
|
|
}else{
|
|
this.elRadio_templateStyle.checked = true;
|
|
this.elLayer_Dim2.className = "se2_t_dim1";
|
|
|
|
this.elText_borderSize.disabled = true;
|
|
this.elText_borderColor.disabled = true;
|
|
this.elText_BGColor.disabled = true;
|
|
}
|
|
this.oApp.exec("TABLE_CLOSE_ALL", []);
|
|
},
|
|
|
|
_selectBorderStyle : function(nStyleNum){
|
|
this.elPanel_borderStylePreview.className = "se2_b_style"+nStyleNum;
|
|
this.nBorderStyleIdx = nStyleNum;
|
|
this._drawTableDropdownLayer();
|
|
},
|
|
|
|
_selectTableStyle : function(nStyleNum){
|
|
this.elPanel_tableStylePreview.className = "se2_t_style"+nStyleNum;
|
|
this.nTableStyleIdx = nStyleNum;
|
|
this._drawTableDropdownLayer();
|
|
},
|
|
|
|
_showNewTable : function(){
|
|
var oTmp = document.createElement("DIV");
|
|
this._loadValuesFromHTML();
|
|
|
|
oTmp.innerHTML = this._getPreviewTableString(this.nColumns, this.nRows);
|
|
|
|
//this.nTableWidth = 0;
|
|
//oTmp.innerHTML = this._getTableString();
|
|
var oNewTable = oTmp.firstChild;
|
|
this.elTable_preview.parentNode.insertBefore(oNewTable, this.elTable_preview);
|
|
this.elTable_preview.parentNode.removeChild(this.elTable_preview);
|
|
this.elTable_preview = oNewTable;
|
|
|
|
this._refresh();
|
|
},
|
|
|
|
_getPreviewTableString : function(nColumns, nRows){
|
|
var sTable = '<table border="0" cellspacing="1" class="se2_pre_table husky_se2m_table_preview">';
|
|
var sRow = '<tr>';
|
|
|
|
for(var i=0; i<nColumns; i++){
|
|
sRow += "<td><p> </p></td>\n";
|
|
}
|
|
sRow += "</tr>\n";
|
|
|
|
sTable += "<tbody>";
|
|
for(var i=0; i<nRows; i++){
|
|
sTable += sRow;
|
|
}
|
|
sTable += "</tbody>\n";
|
|
|
|
sTable += "</table>\n";
|
|
|
|
return sTable;
|
|
},
|
|
|
|
_loadValuesFromHTML : function(){
|
|
this.nColumns = parseInt(this.elText_col.value, 10) || 1;
|
|
this.nRows = parseInt(this.elText_row.value, 10) || 1;
|
|
|
|
this.nBorderSize = parseInt(this.elText_borderSize.value, 10) || 1;
|
|
this.sBorderColor = this.elText_borderColor.value;
|
|
this.sBGColor = this.elText_BGColor.value;
|
|
},
|
|
|
|
_doGetTableString : function(nColumns, nRows, nBorderSize, sBorderColor, sBGColor, nBorderStyleIdx){
|
|
var nTDWidth = parseInt(this.nTableWidth/nColumns, 10);
|
|
var nBorderSize = this.nBorderSize;
|
|
var sTableStyle = this.aTableStyleByBorder[nBorderStyleIdx].replace(/#BorderSize#/g, this.nBorderSize).replace(/#BorderSize\*([0-9]+)#/g, function(sAll, s1){return nBorderSize*parseInt(s1, 10);}).replace(/#BorderSize\+([0-9]+)#/g, function(sAll, s1){return nBorderSize+parseInt(s1, 10);}).replace("#BorderColor#", this.sBorderColor).replace("#BGColor#", this.sBGColor);
|
|
var sTDStyle = this.aTDStyleByBorder[nBorderStyleIdx].replace(/#BorderSize#/g, this.nBorderSize).replace(/#BorderSize\*([0-9]+)#/g, function(sAll, s1){return nBorderSize*parseInt(s1, 10);}).replace(/#BorderSize\+([0-9]+)#/g, function(sAll, s1){return nBorderSize+parseInt(s1, 10);}).replace("#BorderColor#", this.sBorderColor).replace("#BGColor#", this.sBGColor);
|
|
if(nTDWidth){
|
|
sTDStyle += " width="+nTDWidth;
|
|
}else{
|
|
//sTableStyle += " width=100%";
|
|
sTableStyle += "class=se2_pre_table";
|
|
}
|
|
|
|
// [SMARTEDITORSUS-365] 테이블퀵에디터 > 속성 직접입력 > 테두리 스타일
|
|
// - 테두리 없음을 선택하는 경우 본문에 삽입하는 표에 가이드 라인을 표시해 줍니다. 보기 시에는 테두리가 보이지 않습니다.
|
|
// - 글 저장 시에는 글 작성 시에 적용하였던 style 을 제거합니다. 이를 위해서 임의의 속성(attr_no_border_tbl)을 추가하였다가 저장 시점에서 제거해 주도록 합니다.
|
|
var sTempNoBorderClass = (nBorderStyleIdx == 1) ? 'attr_no_border_tbl="1"' : '';
|
|
|
|
var sTable = "<table "+sTableStyle+" "+sTempNoBorderClass+">";
|
|
var sRow = "<tr>";
|
|
for(var i=0; i<nColumns; i++){
|
|
sRow += "<td "+sTDStyle+"><p> </p></td>\n";
|
|
}
|
|
sRow += "</tr>\n";
|
|
|
|
sTable += "<tbody>\n";
|
|
for(var i=0; i<nRows; i++){
|
|
sTable += sRow;
|
|
}
|
|
sTable += "</tbody>\n";
|
|
|
|
sTable += "</table>\n<br>";
|
|
|
|
return sTable;
|
|
},
|
|
|
|
_numRowKeydown : function(weEvent){
|
|
var oKeyInfo = weEvent.key();
|
|
|
|
// up
|
|
if(oKeyInfo.keyCode == 38){
|
|
this.oApp.exec("TABLE_INC_ROW", []);
|
|
}
|
|
|
|
// down
|
|
if(oKeyInfo.keyCode == 40){
|
|
this.oApp.exec("TABLE_DEC_ROW", []);
|
|
}
|
|
},
|
|
|
|
_numColKeydown : function(weEvent){
|
|
var oKeyInfo = weEvent.key();
|
|
|
|
// up
|
|
if(oKeyInfo.keyCode == 38){
|
|
this.oApp.exec("TABLE_INC_COLUMN", []);
|
|
}
|
|
|
|
// down
|
|
if(oKeyInfo.keyCode == 40){
|
|
this.oApp.exec("TABLE_DEC_COLUMN", []);
|
|
}
|
|
},
|
|
|
|
_borderSizeKeydown : function(weEvent){
|
|
var oKeyInfo = weEvent.key();
|
|
|
|
// up
|
|
if(oKeyInfo.keyCode == 38){
|
|
this.oApp.exec("TABLE_INC_BORDER_SIZE", []);
|
|
}
|
|
|
|
// down
|
|
if(oKeyInfo.keyCode == 40){
|
|
this.oApp.exec("TABLE_DEC_BORDER_SIZE", []);
|
|
}
|
|
},
|
|
|
|
_refresh : function(){
|
|
// the dropdown layer breaks without this line in IE 6 when modifying the preview table
|
|
this.elDropdownLayer.style.zoom=0;
|
|
this.elDropdownLayer.style.zoom="";
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_TableEditor$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_TableEditor
|
|
* this.oApp.registerLazyMessage(["EVENT_EDITING_AREA_MOUSEMOVE", "STYLE_TABLE"], ["hp_SE2M_TableEditor$Lazy.js","SE2M_TableTemplate.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_TableEditor, {
|
|
// [SMARTEDITORSUS-1672]
|
|
_aCellName : ["TD", "TH"],
|
|
// --[SMARTEDITORSUS-1672]
|
|
|
|
//@lazyload_js EVENT_EDITING_AREA_MOUSEMOVE:SE2M_TableTemplate.js[
|
|
_assignHTMLObjects : function(){
|
|
this.oApp.exec("LOAD_HTML", ["qe_table"]);
|
|
|
|
this.elQELayer = jindo.$$.getSingle("DIV.q_table_wrap", this.oApp.htOptions.elAppContainer);
|
|
this.elQELayer.style.zIndex = 150;
|
|
this.elBtnAddRowBelow = jindo.$$.getSingle("BUTTON.se2_addrow", this.elQELayer);
|
|
this.elBtnAddColumnRight = jindo.$$.getSingle("BUTTON.se2_addcol", this.elQELayer);
|
|
this.elBtnSplitRow = jindo.$$.getSingle("BUTTON.se2_seprow", this.elQELayer);
|
|
this.elBtnSplitColumn = jindo.$$.getSingle("BUTTON.se2_sepcol", this.elQELayer);
|
|
this.elBtnDeleteRow = jindo.$$.getSingle("BUTTON.se2_delrow", this.elQELayer);
|
|
this.elBtnDeleteColumn = jindo.$$.getSingle("BUTTON.se2_delcol", this.elQELayer);
|
|
this.elBtnMergeCell = jindo.$$.getSingle("BUTTON.se2_merrow", this.elQELayer);
|
|
this.elBtnBGPalette = jindo.$$.getSingle("BUTTON.husky_se2m_table_qe_bgcolor_btn", this.elQELayer);
|
|
this.elBtnBGIMGPalette = jindo.$$.getSingle("BUTTON.husky_se2m_table_qe_bgimage_btn", this.elQELayer);
|
|
|
|
this.elPanelBGPaletteHolder = jindo.$$.getSingle("DIV.husky_se2m_tbl_qe_bg_paletteHolder", this.elQELayer);
|
|
this.elPanelBGIMGPaletteHolder = jindo.$$.getSingle("DIV.husky_se2m_tbl_qe_bg_img_paletteHolder", this.elQELayer);
|
|
|
|
this.elPanelTableBGArea = jindo.$$.getSingle("DIV.se2_qe2", this.elQELayer);
|
|
this.elPanelTableTemplateArea = jindo.$$.getSingle("DL.se2_qe3", this.elQELayer);
|
|
this.elPanelReviewBGArea = jindo.$$.getSingle("DL.husky_se2m_tbl_qe_review_bg", this.elQELayer);
|
|
|
|
this.elPanelBGImg = jindo.$$.getSingle("DD", this.elPanelReviewBGArea);
|
|
|
|
this.welPanelTableBGArea = jindo.$Element(this.elPanelTableBGArea);
|
|
this.welPanelTableTemplateArea = jindo.$Element(this.elPanelTableTemplateArea);
|
|
this.welPanelReviewBGArea = jindo.$Element(this.elPanelReviewBGArea);
|
|
|
|
// this.elPanelReviewBtnArea = jindo.$$.getSingle("DIV.se2_btn_area", this.elQELayer); //My리뷰 버튼 레이어
|
|
this.elPanelDim1 = jindo.$$.getSingle("DIV.husky_se2m_tbl_qe_dim1", this.elQELayer);
|
|
this.elPanelDim2 = jindo.$$.getSingle("DIV.husky_se2m_tbl_qe_dim2", this.elQELayer);
|
|
this.elPanelDimDelCol = jindo.$$.getSingle("DIV.husky_se2m_tbl_qe_dim_del_col", this.elQELayer);
|
|
this.elPanelDimDelRow = jindo.$$.getSingle("DIV.husky_se2m_tbl_qe_dim_del_row", this.elQELayer);
|
|
|
|
this.elInputRadioBGColor = jindo.$$.getSingle("INPUT.husky_se2m_radio_bgc", this.elQELayer);
|
|
this.elInputRadioBGImg = jindo.$$.getSingle("INPUT.husky_se2m_radio_bgimg", this.elQELayer);
|
|
|
|
this.elSelectBoxTemplate = jindo.$$.getSingle("DIV.se2_select_ty2", this.elQELayer);
|
|
this.elInputRadioTemplate = jindo.$$.getSingle("INPUT.husky_se2m_radio_template", this.elQELayer);
|
|
this.elPanelQETemplate = jindo.$$.getSingle("DIV.se2_layer_t_style", this.elQELayer);
|
|
this.elBtnQETemplate = jindo.$$.getSingle("BUTTON.husky_se2m_template_more", this.elQELayer);
|
|
this.elPanelQETemplatePreview = jindo.$$.getSingle("SPAN.se2_t_style1", this.elQELayer);
|
|
|
|
this.aElBtn_tableStyle = jindo.$$("BUTTON", this.elPanelQETemplate);
|
|
for(i = 0; i < this.aElBtn_tableStyle.length; i++){
|
|
this.oApp.registerBrowserEvent(this.aElBtn_tableStyle[i], "click", "TABLE_QE_SELECT_TEMPLATE");
|
|
}
|
|
},
|
|
|
|
_attachEvents : function(){
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
|
|
this.oApp.exec("SE2_ATTACH_HOVER_EVENTS", [this.aElBtn_tableStyle]);
|
|
|
|
this._wfnOnMouseDownResizeCover = jindo.$Fn(this._fnOnMouseDownResizeCover, this);
|
|
this._wfnOnMouseMoveResizeCover = jindo.$Fn(this._fnOnMouseMoveResizeCover, this);
|
|
this._wfnOnMouseUpResizeCover = jindo.$Fn(this._fnOnMouseUpResizeCover, this);
|
|
this._wfnOnMouseDownResizeCover.attach(this.elResizeCover, "mousedown");
|
|
|
|
if((htBrowser.ie) && (htBrowser.version > 8)){
|
|
this._wfnOnResizeEndTable = jindo.$Fn(this._fnOnResizeEndTable, this).bind();
|
|
}
|
|
|
|
// this.oApp.registerBrowserEvent(doc, "click", "EVENT_EDITING_AREA_CLICK");
|
|
this.oApp.registerBrowserEvent(this.elBtnMergeCell, "click", "TE_MERGE_CELLS");
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtnSplitColumn, "click", "TE_SPLIT_COLUMN");
|
|
this.oApp.registerBrowserEvent(this.elBtnSplitRow, "click", "TE_SPLIT_ROW");
|
|
|
|
// this.oApp.registerBrowserEvent(this.elBtnAddColumnLeft, "click", "TE_INSERT_COLUMN_LEFT");
|
|
this.oApp.registerBrowserEvent(this.elBtnAddColumnRight, "click", "TE_INSERT_COLUMN_RIGHT");
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtnAddRowBelow, "click", "TE_INSERT_ROW_BELOW");
|
|
// this.oApp.registerBrowserEvent(this.elBtnAddRowAbove, "click", "TE_INSERT_ROW_ABOVE");
|
|
|
|
this.oApp.registerBrowserEvent(this.elBtnDeleteColumn, "click", "TE_DELETE_COLUMN");
|
|
this.oApp.registerBrowserEvent(this.elBtnDeleteRow, "click", "TE_DELETE_ROW");
|
|
|
|
this.oApp.registerBrowserEvent(this.elInputRadioBGColor, "click", "DRAW_QE_RADIO_OPTION", [2]);
|
|
this.oApp.registerBrowserEvent(this.elInputRadioBGImg, "click", "DRAW_QE_RADIO_OPTION", [3]);
|
|
this.oApp.registerBrowserEvent(this.elInputRadioTemplate, "click", "DRAW_QE_RADIO_OPTION", [4]);
|
|
this.oApp.registerBrowserEvent(this.elBtnBGPalette, "click", "TABLE_QE_TOGGLE_BGC_PALETTE");
|
|
// this.oApp.registerBrowserEvent(this.elPanelReviewBtnArea, "click", "SAVE_QE_MY_REVIEW_ITEM"); //My리뷰 버튼 레이어
|
|
this.oApp.registerBrowserEvent(this.elBtnBGIMGPalette, "click", "TABLE_QE_TOGGLE_IMG_PALETTE");
|
|
this.oApp.registerBrowserEvent(this.elPanelBGIMGPaletteHolder, "click", "TABLE_QE_SET_IMG_FROM_PALETTE");
|
|
//this.elPanelQETemplate
|
|
//this.elBtnQETemplate
|
|
this.oApp.registerBrowserEvent(this.elBtnQETemplate, "click", "TABLE_QE_TOGGLE_TEMPLATE");
|
|
|
|
this.oApp.registerBrowserEvent(document.body, "mouseup", "EVENT_OUTER_DOC_MOUSEUP");
|
|
this.oApp.registerBrowserEvent(document.body, "mousemove", "EVENT_OUTER_DOC_MOUSEMOVE");
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(sMsg){
|
|
if(sMsg.indexOf("REGISTER_CONVERTERS") > -1){
|
|
this.oApp.acceptLocalBeforeFirstAgain(this, true);
|
|
return true;
|
|
}
|
|
this.htResizing = {};
|
|
this.nDraggableCellEdge = 2;
|
|
|
|
var elBody = jindo.$Element(document.body);
|
|
this.nPageLeftRightMargin = parseInt(elBody.css("marginLeft"), 10) + parseInt(elBody.css("marginRight"), 10);
|
|
this.nPageTopBottomMargin = parseInt(elBody.css("marginTop"), 10) + parseInt(elBody.css("marginBottom"), 10);
|
|
|
|
//this.nPageLeftRightMargin = parseInt(elBody.css("marginLeft"), 10)+parseInt(elBody.css("marginRight"), 10) + parseInt(elBody.css("paddingLeft"), 10)+parseInt(elBody.css("paddingRight"), 10);
|
|
//this.nPageTopBottomMargin = parseInt(elBody.css("marginTop"), 10)+parseInt(elBody.css("marginBottom"), 10) + parseInt(elBody.css("paddingTop"), 10)+parseInt(elBody.css("paddingBottom"), 10);
|
|
|
|
this.QE_DIM_MERGE_BTN = 1;
|
|
this.QE_DIM_BG_COLOR = 2;
|
|
this.QE_DIM_REVIEW_BG_IMG = 3;
|
|
this.QE_DIM_TABLE_TEMPLATE = 4;
|
|
|
|
this.rxLastDigits = RegExp("([0-9]+)$");
|
|
|
|
this._assignHTMLObjects();
|
|
|
|
this.addCSSClass(this.CELL_SELECTION_CLASS, "background-color:#B4C9E9;");
|
|
|
|
this._createCellResizeGrip();
|
|
|
|
this.elIFrame = this.oApp.getWYSIWYGWindow().frameElement;
|
|
//[SMARTEDITORSUS-1625] iframe의 offset을 수행하면 iOS에서 처음 탭시 포커싱이 안되는데 정확한 이유를 모르겠음
|
|
//일단 사용안하기 때문에 코멘트처리해서 회피함
|
|
//TODO: 다른 부분에서 동일한 이슈가 발생할 경우 디버깅이 어렵기 때문에 정확한 원인파악이 필요함
|
|
//this.htFrameOffset = jindo.$Element(this.elIFrame).offset();
|
|
|
|
this.sEmptyTDSrc = "";
|
|
if(this.oApp.oNavigator.firefox){
|
|
this.sEmptyTDSrc = "<p><br/></p>";
|
|
}else{
|
|
this.sEmptyTDSrc = "<p> </p>";
|
|
}
|
|
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
|
|
this._attachEvents();
|
|
|
|
// [SMARTEDITORSUS-1672]
|
|
this._rxCellNames = new RegExp("^(" + this._aCellName.join("|") + ")$", "i");
|
|
// --[SMARTEDITORSUS-1672]
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_KEYUP : function(oEvent){
|
|
// for undo/redo and other hotkey functions
|
|
var oKeyInfo = oEvent.key();
|
|
// 229: Korean/Eng, 33, 34: page up/down, 35,36: end/home, 37,38,39,40: left, up, right, down, 16: shift
|
|
if(oKeyInfo.keyCode == 229 || oKeyInfo.alt || oKeyInfo.ctrl || oKeyInfo.keyCode == 16){
|
|
return;
|
|
}else if(oKeyInfo.keyCode == 8 || oKeyInfo.keyCode == 46){
|
|
this.oApp.exec("DELETE_BLOCK_CONTENTS");
|
|
oEvent.stop();
|
|
}
|
|
|
|
switch(this.nStatus){
|
|
case this.STATUS.CELL_SELECTED:
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
break;
|
|
}
|
|
},
|
|
|
|
$ON_TABLE_QE_SELECT_TEMPLATE : function(weEvent){
|
|
var aMatch = this.rxLastDigits.exec(weEvent.element.className);
|
|
var elCurrentTable = this.elSelectionStartTable;
|
|
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
this.oApp.exec("STYLE_TABLE", [elCurrentTable, aMatch[1]]);
|
|
//this._selectTableStyle(aMatch[1]);
|
|
|
|
var elSaveTarget = !!elCurrentTable && elCurrentTable.parentNode ? elCurrentTable.parentNode : null;
|
|
var sSaveTarget = !elCurrentTable ? "BODY" : null;
|
|
|
|
this.oApp.exec("RECORD_UNDO_ACTION", ["CHANGE_TABLE_STYLE", {elSaveTarget:elSaveTarget, sSaveTarget : sSaveTarget, bDontSaveSelection:true}]);
|
|
},
|
|
|
|
$BEFORE_CHANGE_EDITING_MODE : function(sMode, bNoFocus){
|
|
if(sMode !== "WYSIWYG" && this.nStatus !== this.STATUS.S_0){
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
}
|
|
},
|
|
|
|
// [Undo/Redo] Table Selection 처리와 관련된 부분 주석 처리
|
|
// $AFTER_DO_RECORD_UNDO_HISTORY : function(){
|
|
// if(this.nStatus != this.STATUS.CELL_SELECTED){
|
|
// return;
|
|
// }
|
|
//
|
|
// if(this.aSelectedCells.length < 1){
|
|
// return;
|
|
// }
|
|
//
|
|
// var aTables = this.oApp.getWYSIWYGDocument().getElementsByTagName("TABLE");
|
|
// for(var nTableIdx = 0, nLen = aTables.length; nTableIdx < nLen; nTableIdx++){
|
|
// if(aTables[nTableIdx] === this.elSelectionStartTable){
|
|
// break;
|
|
// }
|
|
// }
|
|
//
|
|
// var aUndoHistory = this.oApp.getUndoHistory();
|
|
// var oUndoStateIdx = this.oApp.getUndoStateIdx();
|
|
// if(!aUndoHistory[oUndoStateIdx.nIdx].htTableSelection){
|
|
// aUndoHistory[oUndoStateIdx.nIdx].htTableSelection = [];
|
|
// }
|
|
// aUndoHistory[oUndoStateIdx.nIdx].htTableSelection[oUndoStateIdx.nStep] = {
|
|
// nTableIdx : nTableIdx,
|
|
// nSX : this.htSelectionSPos.x,
|
|
// nSY : this.htSelectionSPos.y,
|
|
// nEX : this.htSelectionEPos.x,
|
|
// nEY : this.htSelectionEPos.y
|
|
// };
|
|
// },
|
|
//
|
|
// $BEFORE_RESTORE_UNDO_HISTORY : function(){
|
|
// if(this.nStatus == this.STATUS.CELL_SELECTED){
|
|
// var oSelection = this.oApp.getEmptySelection();
|
|
// oSelection.selectNode(this.elSelectionStartTable);
|
|
// oSelection.collapseToEnd();
|
|
// oSelection.select();
|
|
// }
|
|
// },
|
|
//
|
|
// $AFTER_RESTORE_UNDO_HISTORY : function(){
|
|
// var aUndoHistory = this.oApp.getUndoHistory();
|
|
// var oUndoStateIdx = this.oApp.getUndoStateIdx();
|
|
//
|
|
// if(aUndoHistory[oUndoStateIdx.nIdx].htTableSelection && aUndoHistory[oUndoStateIdx.nIdx].htTableSelection[oUndoStateIdx.nStep]){
|
|
// var htTableSelection = aUndoHistory[oUndoStateIdx.nIdx].htTableSelection[oUndoStateIdx.nStep];
|
|
// this.elSelectionStartTable = this.oApp.getWYSIWYGDocument().getElementsByTagName("TABLE")[htTableSelection.nTableIdx];
|
|
// this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
//
|
|
// this.htSelectionSPos.x = htTableSelection.nSX;
|
|
// this.htSelectionSPos.y = htTableSelection.nSY;
|
|
// this.htSelectionEPos.x = htTableSelection.nEX;
|
|
// this.htSelectionEPos.y = htTableSelection.nEY;
|
|
// this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
//
|
|
// this._startCellSelection();
|
|
// this._changeTableEditorStatus(this.STATUS.CELL_SELECTED);
|
|
// }else{
|
|
// this._changeTableEditorStatus(this.STATUS.S_0);
|
|
// }
|
|
// },
|
|
|
|
/**
|
|
* 테이블 셀 배경색 셋팅
|
|
*/
|
|
$ON_TABLE_QE_TOGGLE_BGC_PALETTE : function(){
|
|
if(this.elPanelBGPaletteHolder.parentNode.style.display == "block"){
|
|
this.oApp.exec("HIDE_TABLE_QE_BGC_PALETTE", []);
|
|
}else{
|
|
this.oApp.exec("SHOW_TABLE_QE_BGC_PALETTE", []);
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_TABLE_QE_BGC_PALETTE : function(){
|
|
this.elPanelBGPaletteHolder.parentNode.style.display = "block";
|
|
this.oApp.exec("SHOW_COLOR_PALETTE", ["TABLE_QE_SET_BGC_FROM_PALETTE", this.elPanelBGPaletteHolder]);
|
|
},
|
|
|
|
$ON_HIDE_TABLE_QE_BGC_PALETTE : function(){
|
|
this.elPanelBGPaletteHolder.parentNode.style.display = "none";
|
|
this.oApp.exec("HIDE_COLOR_PALETTE", []);
|
|
},
|
|
|
|
$ON_TABLE_QE_SET_BGC_FROM_PALETTE : function(sColorCode){
|
|
this.oApp.exec("TABLE_QE_SET_BGC", [sColorCode]);
|
|
if(this.oSelection){
|
|
this.oSelection.select();
|
|
}
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
},
|
|
|
|
$ON_TABLE_QE_SET_BGC : function(sColorCode){
|
|
this.elBtnBGPalette.style.backgroundColor = sColorCode;
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
this.aSelectedCells[i].setAttribute(this.TMP_BGC_ATTR, sColorCode);
|
|
this.aSelectedCells[i].removeAttribute(this.TMP_BGIMG_ATTR);
|
|
}
|
|
this.sQEAction = "TABLE_SET_BGCOLOR";
|
|
},
|
|
|
|
/**
|
|
* 테이블 리뷰 테이블 배경 이미지 셋팅
|
|
*/
|
|
$ON_TABLE_QE_TOGGLE_IMG_PALETTE : function(){
|
|
if(this.elPanelBGIMGPaletteHolder.parentNode.style.display == "block"){
|
|
this.oApp.exec("HIDE_TABLE_QE_IMG_PALETTE", []);
|
|
}else{
|
|
this.oApp.exec("SHOW_TABLE_QE_IMG_PALETTE", []);
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_TABLE_QE_IMG_PALETTE : function(){
|
|
this.elPanelBGIMGPaletteHolder.parentNode.style.display = "block";
|
|
},
|
|
|
|
$ON_HIDE_TABLE_QE_IMG_PALETTE : function(){
|
|
this.elPanelBGIMGPaletteHolder.parentNode.style.display = "none";
|
|
},
|
|
|
|
$ON_TABLE_QE_SET_IMG_FROM_PALETTE : function(elEvt){
|
|
this.oApp.exec("TABLE_QE_SET_IMG", [elEvt.element]);
|
|
if(this.oSelection){
|
|
this.oSelection.select();
|
|
}
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
},
|
|
|
|
$ON_TABLE_QE_SET_IMG : function(elSelected){
|
|
var sClassName = jindo.$Element(elSelected).className();
|
|
var welBtnBGIMGPalette = jindo.$Element(this.elBtnBGIMGPalette);
|
|
var aBtnClassNames = welBtnBGIMGPalette.className().split(" ");
|
|
for(var i = 0, nLen = aBtnClassNames.length; i < nLen; i++){
|
|
if(aBtnClassNames[i].indexOf("cellimg") > 0){
|
|
welBtnBGIMGPalette.removeClass(aBtnClassNames[i]);
|
|
}
|
|
}
|
|
jindo.$Element(this.elBtnBGIMGPalette).addClass(sClassName);
|
|
|
|
var n = sClassName.substring(11, sClassName.length); //se2_cellimg11
|
|
var sImageName = "pattern_";
|
|
|
|
if(n === "0"){
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
jindo.$Element(this.aSelectedCells[i]).css("backgroundImage", "");
|
|
this.aSelectedCells[i].removeAttribute(this.TMP_BGC_ATTR);
|
|
this.aSelectedCells[i].removeAttribute(this.TMP_BGIMG_ATTR);
|
|
}
|
|
}else{
|
|
if(n == 19 || n == 20 || n == 21 || n == 22 || n == 25 || n == 26){ //파일 사이즈때문에 jpg
|
|
sImageName = sImageName + n + ".jpg";
|
|
}else{
|
|
sImageName = sImageName + n + ".gif";
|
|
}
|
|
|
|
for(var j = 0, nLen = this.aSelectedCells.length; j < nLen ; j++){
|
|
jindo.$Element(this.aSelectedCells[j]).css("backgroundImage", "url("+"http://static.se2.naver.com/static/img/"+sImageName+")");
|
|
this.aSelectedCells[j].removeAttribute(this.TMP_BGC_ATTR);
|
|
this.aSelectedCells[j].setAttribute(this.TMP_BGIMG_ATTR, "url("+"http://static.se2.naver.com/static/img/"+sImageName+")");
|
|
}
|
|
}
|
|
this.sQEAction = "TABLE_SET_BGIMAGE";
|
|
},
|
|
|
|
$ON_SAVE_QE_MY_REVIEW_ITEM : function(){
|
|
this.oApp.exec("SAVE_MY_REVIEW_ITEM");
|
|
this.oApp.exec("CLOSE_QE_LAYER");
|
|
},
|
|
|
|
/**
|
|
* 테이블 퀵 에디터 Show
|
|
*/
|
|
$ON_SHOW_COMMON_QE : function(){
|
|
if(jindo.$Element(this.elSelectionStartTable).hasClass(this._sSETblClass)){
|
|
this.oApp.exec("SHOW_TABLE_QE");
|
|
}else{
|
|
if(jindo.$Element(this.elSelectionStartTable).hasClass(this._sSEReviewTblClass)){
|
|
this.oApp.exec("SHOW_REVIEW_QE");
|
|
}
|
|
}
|
|
},
|
|
|
|
$ON_SHOW_TABLE_QE : function(){
|
|
this.oApp.exec("HIDE_TABLE_QE_BGC_PALETTE", []);
|
|
this.oApp.exec("TABLE_QE_HIDE_TEMPLATE", []);
|
|
this.oApp.exec("SETUP_TABLE_QE_MODE", [0]);
|
|
this.oApp.exec("OPEN_QE_LAYER", [this.htMap[this.htSelectionEPos.x][this.htSelectionEPos.y], this.elQELayer, "table"]);
|
|
//this.oApp.exec("FOCUS");
|
|
},
|
|
|
|
$ON_SHOW_REVIEW_QE : function(){
|
|
this.oApp.exec("SETUP_TABLE_QE_MODE", [1]);
|
|
this.oApp.exec("OPEN_QE_LAYER", [this.htMap[this.htSelectionEPos.x][this.htSelectionEPos.y], this.elQELayer, "review"]);
|
|
},
|
|
|
|
$ON_CLOSE_SUB_LAYER_QE : function(){
|
|
if(typeof this.elPanelBGPaletteHolder != 'undefined'){
|
|
this.elPanelBGPaletteHolder.parentNode.style.display = "none";
|
|
}
|
|
if(typeof this.elPanelBGIMGPaletteHolder != 'undefined'){
|
|
this.elPanelBGIMGPaletteHolder.parentNode.style.display = "none";
|
|
}
|
|
},
|
|
|
|
// 0: table
|
|
// 1: review
|
|
$ON_SETUP_TABLE_QE_MODE : function(nMode){
|
|
var bEnableMerge = true;
|
|
|
|
if(typeof nMode == "number"){
|
|
this.nQEMode = nMode;
|
|
}
|
|
|
|
if(this.aSelectedCells.length < 2){
|
|
bEnableMerge = false;
|
|
}
|
|
|
|
this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_MERGE_BTN, bEnableMerge]);
|
|
|
|
//null인경우를 대비해서 default값을 지정해준다.
|
|
var sBackgroundColor = this.aSelectedCells[0].getAttribute(this.TMP_BGC_ATTR) || "rgb(255,255,255)";
|
|
|
|
var bAllMatched = true;
|
|
for(var i = 1, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
// [SMARTEDITORSUS-1552] 드래그로 셀을 선택하는 중 elCell이 없는 경우 오류 발생
|
|
if(this.aSelectedCells[i]){
|
|
if(sBackgroundColor != this.aSelectedCells[i].getAttribute(this.TMP_BGC_ATTR)){
|
|
bAllMatched = false;
|
|
break;
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1552]
|
|
}
|
|
if(bAllMatched){
|
|
this.elBtnBGPalette.style.backgroundColor = sBackgroundColor;
|
|
}else{
|
|
this.elBtnBGPalette.style.backgroundColor = "#FFFFFF";
|
|
}
|
|
|
|
var sBackgroundImage = this.aSelectedCells[0].getAttribute(this.TMP_BGIMG_ATTR) || "";
|
|
var bAllMatchedImage = true;
|
|
var sPatternInfo, nPatternImage = 0;
|
|
var welBtnBGIMGPalette = jindo.$Element(this.elBtnBGIMGPalette);
|
|
|
|
if(!!sBackgroundImage){
|
|
var aPattern = sBackgroundImage.match(/\_[0-9]*/);
|
|
sPatternInfo = (!!aPattern)?aPattern[0] : "_0";
|
|
nPatternImage = sPatternInfo.substring(1, sPatternInfo.length);
|
|
for(var i = 1, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
if(sBackgroundImage != this.aSelectedCells[i].getAttribute(this.TMP_BGIMG_ATTR)){
|
|
bAllMatchedImage = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
var aBtnClassNames = welBtnBGIMGPalette.className().split(/\s/);
|
|
for(var j = 0, nLen = aBtnClassNames.length; j < nLen; j++){
|
|
if(aBtnClassNames[j].indexOf("cellimg") > 0){
|
|
welBtnBGIMGPalette.removeClass(aBtnClassNames[j]);
|
|
}
|
|
}
|
|
|
|
if(bAllMatchedImage && nPatternImage > 0){
|
|
welBtnBGIMGPalette.addClass("se2_cellimg" + nPatternImage);
|
|
}else{
|
|
welBtnBGIMGPalette.addClass("se2_cellimg0");
|
|
}
|
|
|
|
if(this.nQEMode === 0){ //table
|
|
this.elPanelTableTemplateArea.style.display = "block";
|
|
// this.elSelectBoxTemplate.style.display = "block";
|
|
this.elPanelReviewBGArea.style.display = "none";
|
|
|
|
// this.elSelectBoxTemplate.style.position = "";
|
|
|
|
//this.elPanelReviewBtnArea.style.display = "none"; //My리뷰 버튼 레이어
|
|
|
|
// 배경Area에서 css를 제거해야함
|
|
jindo.$Element(this.elPanelTableBGArea).className("se2_qe2");
|
|
|
|
var nTpl = this.parseIntOr0(this.elSelectionStartTable.getAttribute(this.ATTR_TBL_TEMPLATE));
|
|
if(nTpl){
|
|
//this.elInputRadioTemplate.checked = "true";
|
|
}else{
|
|
this.elInputRadioBGColor.checked = "true";
|
|
nTpl = 1;
|
|
}
|
|
|
|
this.elPanelQETemplatePreview.className = "se2_t_style" + nTpl;
|
|
|
|
this.elPanelBGImg.style.position = "";
|
|
}else if(this.nQEMode == 1){ //review
|
|
this.elPanelTableTemplateArea.style.display = "none";
|
|
// this.elSelectBoxTemplate.style.display = "none";
|
|
this.elPanelReviewBGArea.style.display = "block";
|
|
|
|
// this.elSelectBoxTemplate.style.position = "static";
|
|
|
|
// this.elPanelReviewBtnArea.style.display = "block"; //My리뷰 버튼 레이어
|
|
var nTpl = this.parseIntOr0(this.elSelectionStartTable.getAttribute(this.ATTR_REVIEW_TEMPLATE));
|
|
|
|
this.elPanelBGImg.style.position = "relative";
|
|
}else{
|
|
this.elPanelTableTemplateArea.style.display = "none";
|
|
this.elPanelReviewBGArea.style.display = "none";
|
|
// this.elPanelReviewBtnArea.style.display = "none"; //My리뷰 버튼 레이어
|
|
}
|
|
|
|
this.oApp.exec("DRAW_QE_RADIO_OPTION", [0]);
|
|
},
|
|
|
|
// nClickedIdx
|
|
// 0: none
|
|
// 2: bg color
|
|
// 3: bg img
|
|
// 4: template
|
|
$ON_DRAW_QE_RADIO_OPTION : function(nClickedIdx){
|
|
if(nClickedIdx !== 0 && nClickedIdx != 2){
|
|
this.oApp.exec("HIDE_TABLE_QE_BGC_PALETTE", []);
|
|
}
|
|
if(nClickedIdx !== 0 && nClickedIdx != 3){
|
|
this.oApp.exec("HIDE_TABLE_QE_IMG_PALETTE", []);
|
|
}
|
|
if(nClickedIdx !== 0 && nClickedIdx != 4){
|
|
this.oApp.exec("TABLE_QE_HIDE_TEMPLATE", []);
|
|
}
|
|
|
|
if(this.nQEMode === 0){
|
|
// bg image option does not exist in table mode. so select the bgcolor option
|
|
if(this.elInputRadioBGImg.checked){
|
|
this.elInputRadioBGColor.checked = "true";
|
|
}
|
|
if(this.elInputRadioBGColor.checked){
|
|
// one dimming layer is being shared so only need to dim once and the rest will be undimmed automatically
|
|
//this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_BG_COLOR, true]);
|
|
this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_TABLE_TEMPLATE, false]);
|
|
}else{
|
|
this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_BG_COLOR, false]);
|
|
//this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_TABLE_TEMPLATE, true]);
|
|
}
|
|
}else{
|
|
// template option does not exist in review mode. so select the bgcolor optio
|
|
if(this.elInputRadioTemplate.checked){
|
|
this.elInputRadioBGColor.checked = "true";
|
|
}
|
|
if(this.elInputRadioBGColor.checked){
|
|
//this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_BG_COLOR, true]);
|
|
this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_REVIEW_BG_IMG, false]);
|
|
}else{
|
|
this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_BG_COLOR, false]);
|
|
//this.oApp.exec("TABLE_QE_DIM", [this.QE_DIM_REVIEW_BG_IMG, true]);
|
|
}
|
|
}
|
|
},
|
|
|
|
// nPart
|
|
// 1: Merge cell btn
|
|
// 2: Cell bg color
|
|
// 3: Review - bg image
|
|
// 4: Table - Template
|
|
//
|
|
// bUndim
|
|
// true: Undim
|
|
// false(default): Dim
|
|
$ON_TABLE_QE_DIM : function(nPart, bUndim){
|
|
var elPanelDim;
|
|
var sDimClassPrefix = "se2_qdim";
|
|
if(nPart == 1){
|
|
elPanelDim = this.elPanelDim1;
|
|
}else{
|
|
elPanelDim = this.elPanelDim2;
|
|
}
|
|
|
|
if(bUndim){
|
|
nPart = 0;
|
|
}
|
|
elPanelDim.className = sDimClassPrefix + nPart;
|
|
},
|
|
|
|
$ON_TE_SELECT_TABLE : function(elTable){
|
|
this.elSelectionStartTable = elTable;
|
|
this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
},
|
|
|
|
$ON_TE_SELECT_CELLS : function(htSPos, htEPos){
|
|
this._selectCells(htSPos, htEPos);
|
|
},
|
|
|
|
$ON_TE_MERGE_CELLS : function(){
|
|
if(this.aSelectedCells.length === 0 || this.aSelectedCells.length == 1){
|
|
return;
|
|
}
|
|
this._removeClassFromSelection();
|
|
|
|
var i, elFirstTD, elTD;
|
|
|
|
elFirstTD = this.aSelectedCells[0];
|
|
var elTable = nhn.husky.SE2M_Utils.findAncestorByTagName("TABLE", elFirstTD);
|
|
|
|
var nHeight, nWidth;
|
|
var elCurTD, elLastTD = this.aSelectedCells[0];
|
|
nHeight = parseInt(elLastTD.style.height || elLastTD.getAttribute("height"), 10);
|
|
nWidth = parseInt(elLastTD.style.width || elLastTD.getAttribute("width"), 10);
|
|
//nHeight = elLastTD.offsetHeight;
|
|
//nWidth = elLastTD.offsetWidth;
|
|
|
|
for(i = this.htSelectionSPos.x + 1; i < this.htSelectionEPos.x + 1; i++){
|
|
curTD = this.htMap[i][this.htSelectionSPos.y];
|
|
if(curTD == elLastTD){
|
|
continue;
|
|
}
|
|
elLastTD = curTD;
|
|
nWidth += parseInt(curTD.style.width || curTD.getAttribute("width"), 10);
|
|
//nWidth += curTD.offsetWidth;
|
|
}
|
|
|
|
elLastTD = this.aSelectedCells[0];
|
|
for(i = this.htSelectionSPos.y + 1; i < this.htSelectionEPos.y + 1; i++){
|
|
curTD = this.htMap[this.htSelectionSPos.x][i];
|
|
if(curTD == elLastTD){
|
|
continue;
|
|
}
|
|
elLastTD = curTD;
|
|
nHeight += parseInt(curTD.style.height || curTD.getAttribute("height"), 10);
|
|
//nHeight += curTD.offsetHeight;
|
|
}
|
|
|
|
if(nWidth){
|
|
elFirstTD.style.width = nWidth + "px";
|
|
}
|
|
if(nHeight){
|
|
elFirstTD.style.height = nHeight + "px";
|
|
}
|
|
|
|
elFirstTD.setAttribute("colSpan", this.htSelectionEPos.x - this.htSelectionSPos.x + 1);
|
|
elFirstTD.setAttribute("rowSpan", this.htSelectionEPos.y - this.htSelectionSPos.y + 1);
|
|
|
|
for(i = 1; i < this.aSelectedCells.length; i++){
|
|
elTD = this.aSelectedCells[i];
|
|
|
|
if(elTD.parentNode){
|
|
if(!nhn.husky.SE2M_Utils.isBlankNode(elTD)){
|
|
elFirstTD.innerHTML += elTD.innerHTML;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1533] 병합되는 셀 바로 뒤에 포함된 빈 텍스트 노드도 함께 제거하여 DOM 트리 일관성 유지
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if(htBrowser.ie && (htBrowser.nativeVersion == 9 || htBrowser.nativeVersion == 10) && (htBrowser.version == 9 || htBrowser.version == 10)){
|
|
this._removeEmptyTextNode_IE(elTD);
|
|
}
|
|
// --[SMARTEDITORSUS-1533]
|
|
|
|
elTD.parentNode.removeChild(elTD);
|
|
}
|
|
}
|
|
// this._updateSelection();
|
|
|
|
this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
this._showTableTemplate(this.elSelectionStartTable);
|
|
this._addClassToSelection();
|
|
|
|
this.sQEAction = "TABLE_CELL_MERGE";
|
|
|
|
this.oApp.exec("SHOW_COMMON_QE");
|
|
},
|
|
|
|
|
|
$ON_TABLE_QE_TOGGLE_TEMPLATE : function(){
|
|
if(this.elPanelQETemplate.style.display == "block"){
|
|
this.oApp.exec("TABLE_QE_HIDE_TEMPLATE");
|
|
}else{
|
|
this.oApp.exec("TABLE_QE_SHOW_TEMPLATE");
|
|
}
|
|
},
|
|
|
|
$ON_TABLE_QE_SHOW_TEMPLATE : function(){
|
|
this.elPanelQETemplate.style.display = "block";
|
|
this.oApp.exec("POSITION_TOOLBAR_LAYER", [this.elPanelQETemplate]);
|
|
},
|
|
|
|
$ON_TABLE_QE_HIDE_TEMPLATE : function(){
|
|
this.elPanelQETemplate.style.display = "none";
|
|
},
|
|
|
|
$ON_STYLE_TABLE : function(elTable, nTableStyleIdx){
|
|
if(!elTable){
|
|
if(!this._t){
|
|
this._t = 1;
|
|
}
|
|
elTable = this.elSelectionStartTable;
|
|
nTableStyleIdx = (this._t++) % 20 + 1;
|
|
}
|
|
|
|
if(this.oSelection){
|
|
this.oSelection.select();
|
|
}
|
|
this._applyTableTemplate(elTable, nTableStyleIdx);
|
|
},
|
|
|
|
$ON_TE_DELETE_COLUMN : function(){
|
|
// [SMARTEDITORSUS-1784] [SMARTEDITORSUS-555] 처리 후 발생한 사이트이펙트
|
|
if(this.aSelectedCells.length === 0) {
|
|
return;
|
|
}
|
|
/*if(this.aSelectedCells.length === 0 || this.aSelectedCells.length == 1) {
|
|
return;
|
|
}*/
|
|
// --[SMARTEDITORSUS-1784]
|
|
|
|
this._selectAll_Column();
|
|
this._deleteSelectedCells();
|
|
this.sQEAction = "DELETE_TABLE_COLUMN";
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
},
|
|
|
|
$ON_TE_DELETE_ROW : function(){
|
|
// [SMARTEDITORSUS-1784] [SMARTEDITORSUS-555] 처리 후 발생한 사이트이펙트
|
|
if(this.aSelectedCells.length === 0) {
|
|
return;
|
|
}
|
|
/*if(this.aSelectedCells.length === 0 || this.aSelectedCells.length == 1) {
|
|
return;
|
|
}*/
|
|
// --[SMARTEDITORSUS-1784]
|
|
|
|
this._selectAll_Row();
|
|
this._deleteSelectedCells();
|
|
this.sQEAction = "DELETE_TABLE_ROW";
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
},
|
|
|
|
$ON_TE_INSERT_COLUMN_RIGHT : function(){
|
|
if(this.aSelectedCells.length === 0) {
|
|
return;
|
|
}
|
|
|
|
this._selectAll_Column();
|
|
this._insertColumnAfter(this.htSelectionEPos.x);
|
|
},
|
|
|
|
$ON_TE_INSERT_COLUMN_LEFT : function(){
|
|
this._selectAll_Column();
|
|
this._insertColumnAfter(this.htSelectionSPos.x - 1);
|
|
},
|
|
|
|
$ON_TE_INSERT_ROW_BELOW : function(){
|
|
if(this.aSelectedCells.length === 0) {
|
|
return;
|
|
}
|
|
|
|
this._insertRowBelow(this.htSelectionEPos.y);
|
|
},
|
|
|
|
$ON_TE_INSERT_ROW_ABOVE : function(){
|
|
this._insertRowBelow(this.htSelectionSPos.y - 1);
|
|
},
|
|
|
|
$ON_TE_SPLIT_COLUMN : function(){
|
|
var nSpan, nNewSpan, nWidth, nNewWidth;
|
|
var elCurCell, elNewTD;
|
|
|
|
if(this.aSelectedCells.length === 0) {
|
|
return;
|
|
}
|
|
|
|
this._removeClassFromSelection();
|
|
|
|
var elLastCell = this.aSelectedCells[0];
|
|
// Assign colSpan>1 to all selected cells.
|
|
// If current colSpan == 1 then increase the colSpan of the cell and all the vertically adjacent cells.
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
elCurCell = this.aSelectedCells[i];
|
|
nSpan = parseInt(elCurCell.getAttribute("colSpan"), 10) || 1;
|
|
if(nSpan > 1){
|
|
continue;
|
|
}
|
|
|
|
var htPos = this._getBasisCellPosition(elCurCell);
|
|
for(var y = 0; y < this.htMap[0].length;){
|
|
elCurCell = this.htMap[htPos.x][y];
|
|
nSpan = parseInt(elCurCell.getAttribute("colSpan"), 10) || 1;
|
|
elCurCell.setAttribute("colSpan", nSpan+1);
|
|
y += parseInt(elCurCell.getAttribute("rowSpan"), 10) || 1;
|
|
}
|
|
}
|
|
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
elCurCell = this.aSelectedCells[i];
|
|
nSpan = parseInt(elCurCell.getAttribute("colSpan"), 10) || 1;
|
|
nNewSpan = (nSpan/2).toFixed(0);
|
|
|
|
elCurCell.setAttribute("colSpan", nNewSpan);
|
|
|
|
elNewTD = this._shallowCloneTD(elCurCell);
|
|
elNewTD.setAttribute("colSpan", nSpan-nNewSpan);
|
|
elLastCell = elNewTD;
|
|
|
|
nSpan = parseInt(elCurCell.getAttribute("rowSpan"), 10) || 1;
|
|
elNewTD.setAttribute("rowSpan", nSpan);
|
|
elNewTD.innerHTML = " ";
|
|
|
|
nWidth = elCurCell.width || elCurCell.style.width;
|
|
if(nWidth){
|
|
nWidth = this.parseIntOr0(nWidth);
|
|
elCurCell.removeAttribute("width");
|
|
nNewWidth = (nWidth/2).toFixed();
|
|
elCurCell.style.width = nNewWidth + "px";
|
|
elNewTD.style.width = (nWidth - nNewWidth) + "px";
|
|
}
|
|
|
|
elCurCell.parentNode.insertBefore(elNewTD, elCurCell.nextSibling);
|
|
|
|
// [SMARTEDITORSUS-1745][SMARTEDITORSUS-1842] 배경색이 바로 반영되지 않는 버그로 인해 명시
|
|
// [SMARTEDITORSUS-2155] Win10 Spartan
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if((htBrowser.edge && (Math.floor(htBrowser.version) === 12)) || (htBrowser.ie && (htBrowser.nativeVersion >= 9 || htBrowser.nativeVersion <= 11) && (htBrowser.version >= 9 || htBrowser.version <= 11))){
|
|
elNewTD.style.cssText = elCurCell.style.cssText;
|
|
}
|
|
// --[SMARTEDITORSUS-1745][SMARTEDITORSUS-1842][SMARTEDITORSUS-2155]
|
|
}
|
|
|
|
this._reassignCellSizes(this.elSelectionStartTable);
|
|
|
|
this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
|
|
var htPos = this._getBasisCellPosition(elLastCell);
|
|
this.htSelectionEPos.x = htPos.x;
|
|
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
this.sQEAction = "SPLIT_TABLE_COLUMN";
|
|
|
|
this.oApp.exec("SHOW_COMMON_QE");
|
|
},
|
|
|
|
$ON_TE_SPLIT_ROW : function(){
|
|
var nSpan, nNewSpan, nHeight, nHeight;
|
|
var elCurCell, elNewTD, htPos, elNewTR;
|
|
|
|
if(this.aSelectedCells.length === 0) {
|
|
return;
|
|
}
|
|
|
|
var aTR = jindo.$$(">TBODY>TR", this.elSelectionStartTable, {oneTimeOffCache:true});
|
|
this._removeClassFromSelection();
|
|
//top.document.title = this.htSelectionSPos.x+","+this.htSelectionSPos.y+"::"+this.htSelectionEPos.x+","+this.htSelectionEPos.y;
|
|
|
|
var nNewRows = 0;
|
|
// Assign rowSpan>1 to all selected cells.
|
|
// If current rowSpan == 1 then increase the rowSpan of the cell and all the horizontally adjacent cells.
|
|
var elNextTRInsertionPoint;
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
elCurCell = this.aSelectedCells[i];
|
|
nSpan = parseInt(elCurCell.getAttribute("rowSpan"), 10) || 1;
|
|
if(nSpan > 1){
|
|
continue;
|
|
}
|
|
|
|
htPos = this._getBasisCellPosition(elCurCell);
|
|
elNextTRInsertionPoint = aTR[htPos.y];
|
|
|
|
// a new TR has to be inserted when there's an increase in rowSpan
|
|
elNewTR = this.oApp.getWYSIWYGDocument().createElement("TR");
|
|
elNextTRInsertionPoint.parentNode.insertBefore(elNewTR, elNextTRInsertionPoint.nextSibling);
|
|
nNewRows++;
|
|
|
|
// loop through horizontally adjacent cells and increase their rowSpan
|
|
for(var x = 0; x < this.htMap.length;){
|
|
elCurCell = this.htMap[x][htPos.y];
|
|
nSpan = parseInt(elCurCell.getAttribute("rowSpan"), 10) || 1;
|
|
elCurCell.setAttribute("rowSpan", nSpan + 1);
|
|
x += parseInt(elCurCell.getAttribute("colSpan"), 10) || 1;
|
|
}
|
|
}
|
|
|
|
aTR = jindo.$$(">TBODY>TR", this.elSelectionStartTable, {oneTimeOffCache:true});
|
|
|
|
var htPos1, htPos2;
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
elCurCell = this.aSelectedCells[i];
|
|
nSpan = parseInt(elCurCell.getAttribute("rowSpan"), 10) || 1;
|
|
nNewSpan = (nSpan/2).toFixed(0);
|
|
|
|
elCurCell.setAttribute("rowSpan", nNewSpan);
|
|
|
|
elNewTD = this._shallowCloneTD(elCurCell);
|
|
elNewTD.setAttribute("rowSpan", nSpan - nNewSpan);
|
|
|
|
nSpan = parseInt(elCurCell.getAttribute("colSpan"), 10) || 1;
|
|
elNewTD.setAttribute("colSpan", nSpan);
|
|
elNewTD.innerHTML = " ";
|
|
|
|
nHeight = elCurCell.height || elCurCell.style.height;
|
|
if(nHeight){
|
|
nHeight = this.parseIntOr0(nHeight);
|
|
elCurCell.removeAttribute("height");
|
|
nNewHeight = (nHeight/2).toFixed();
|
|
elCurCell.style.height = nNewHeight + "px";
|
|
elNewTD.style.height = (nHeight - nNewHeight) + "px";
|
|
}
|
|
|
|
//var elTRInsertTo = elCurCell.parentNode;
|
|
//for(var ii=0; ii<nNewSpan; ii++) elTRInsertTo = elTRInsertTo.nextSibling;
|
|
var nTRIdx = jindo.$A(aTR).indexOf(elCurCell.parentNode);
|
|
var nNextTRIdx = parseInt(nTRIdx, 10)+parseInt(nNewSpan, 10);
|
|
var elTRInsertTo = aTR[nNextTRIdx];
|
|
|
|
var oSiblingTDs = elTRInsertTo.childNodes;
|
|
var elInsertionPt = null;
|
|
var tmp;
|
|
htPos1 = this._getBasisCellPosition(elCurCell);
|
|
for(var ii = 0, nNumTDs = oSiblingTDs.length; ii < nNumTDs; ii++){
|
|
tmp = oSiblingTDs[ii];
|
|
// [SMARTEDITORSUS-1672]
|
|
//if(!tmp.tagName || tmp.tagName != "TD"){
|
|
if(!tmp.tagName || !this._rxCellNames.test(tmp.tagName)){
|
|
// --[SMARTEDITORSUS-1672]
|
|
continue;
|
|
}
|
|
|
|
htPos2 = this._getBasisCellPosition(tmp);
|
|
if(htPos1.x < htPos2.x){
|
|
elInsertionPt = tmp;
|
|
break;
|
|
}
|
|
}
|
|
elTRInsertTo.insertBefore(elNewTD, elInsertionPt);
|
|
|
|
// [SMARTEDITORSUS-1745][SMARTEDITORSUS-1842] 배경색이 바로 반영되지 않는 버그로 인해 명시
|
|
// [SMARTEDITORSUS-2155] Win10 Spartan
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if((htBrowser.edge && (Math.floor(htBrowser.version) === 12)) || (htBrowser.ie && (htBrowser.nativeVersion >= 9 || htBrowser.nativeVersion <= 11) && (htBrowser.version >= 9 || htBrowser.version <= 11))){
|
|
elNewTD.style.cssText = elNewTD.style.cssText;
|
|
}
|
|
// --[SMARTEDITORSUS-1745][SMARTEDITORSUS-1842][SMARTEDITORSUS-2155]
|
|
}
|
|
|
|
this._reassignCellSizes(this.elSelectionStartTable);
|
|
|
|
this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
this.htSelectionEPos.y += nNewRows;
|
|
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
this.sQEAction = "SPLIT_TABLE_ROW";
|
|
|
|
this.oApp.exec("SHOW_COMMON_QE");
|
|
},
|
|
|
|
$ON_MSG_CELL_SELECTED : function(){
|
|
// disable row/col delete btn
|
|
this.elPanelDimDelCol.className = "se2_qdim6r";
|
|
this.elPanelDimDelRow.className = "se2_qdim6c";
|
|
|
|
if(this.htSelectionSPos.x === 0 && this.htSelectionEPos.x === this.htMap.length - 1){
|
|
this.oApp.exec("MSG_ROW_SELECTED");
|
|
}
|
|
|
|
if(this.htSelectionSPos.y === 0 && this.htSelectionEPos.y === this.htMap[0].length - 1){
|
|
this.oApp.exec("MSG_COL_SELECTED");
|
|
}
|
|
|
|
this.oApp.exec("SHOW_COMMON_QE");
|
|
},
|
|
|
|
$ON_MSG_ROW_SELECTED : function(){
|
|
this.elPanelDimDelRow.className = "";
|
|
},
|
|
|
|
$ON_MSG_COL_SELECTED : function(){
|
|
this.elPanelDimDelCol.className = "";
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_MOUSEDOWN : function(wevE){
|
|
if(!this.oApp.isWYSIWYGEnabled()){
|
|
return;
|
|
}
|
|
|
|
switch(this.nStatus){
|
|
case this.STATUS.S_0:
|
|
// the user may just want to resize the image
|
|
if(!wevE.element){return;}
|
|
if(wevE.element.tagName == "IMG"){return;}
|
|
if(this.oApp.getEditingMode() !== "WYSIWYG"){return;}
|
|
|
|
// change the status to MOUSEDOWN_CELL if the mouse is over a table cell
|
|
// [SMARTEDITORSUS-1672]
|
|
/*var elTD = nhn.husky.SE2M_Utils.findAncestorByTagName("TD", wevE.element);
|
|
|
|
if(elTD && elTD.tagName == "TD"){*/
|
|
var elTD = nhn.husky.SE2M_Utils.findClosestAncestorAmongTagNames(this._aCellName, wevE.element);
|
|
|
|
if(elTD && this._rxCellNames.test(elTD.tagName)){
|
|
// --[SMARTEDITORSUS-1672]
|
|
var elTBL = nhn.husky.SE2M_Utils.findAncestorByTagName("TABLE", elTD);
|
|
|
|
if(!jindo.$Element(elTBL).hasClass(this._sSETblClass) && !jindo.$Element(elTBL).hasClass(this._sSEReviewTblClass)){return;}
|
|
if(!this._isValidTable(elTBL)){
|
|
jindo.$Element(elTBL).removeClass(this._sSETblClass);
|
|
jindo.$Element(elTBL).removeClass(this._sSEReviewTblClass);
|
|
return;
|
|
}
|
|
|
|
if(elTBL){
|
|
this.elSelectionStartTD = elTD;
|
|
this.elSelectionStartTable = elTBL;
|
|
this._changeTableEditorStatus(this.STATUS.MOUSEDOWN_CELL);
|
|
}
|
|
}
|
|
break;
|
|
case this.STATUS.MOUSEDOWN_CELL:
|
|
break;
|
|
case this.STATUS.CELL_SELECTING:
|
|
break;
|
|
case this.STATUS.CELL_SELECTED:
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
break;
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_MOUSEMOVE : function(wevE){
|
|
if(this.oApp.getEditingMode() != "WYSIWYG"){return;}
|
|
|
|
// [SMARTEDITORSUS-2136] [IE] <table>에 resizeend 핸들러 부여
|
|
var elTable = wevE.element,
|
|
htBrowser = jindo.$Agent().navigator(),
|
|
sPointerUpEvent = 'onpointerup',
|
|
sResizeEndEvent = 'onresizeend';
|
|
if(htBrowser.ie && elTable && elTable.tagName && (elTable.tagName.toUpperCase() === 'TABLE')){ // [IE]
|
|
if(sPointerUpEvent in elTable){ // [IE 11] resizeend 이벤트 deprecated
|
|
elTable[sPointerUpEvent] = this._wfnOnResizeEndTable;
|
|
}else if(sResizeEndEvent in elTable){
|
|
if(htBrowser.version > 8){
|
|
elTable[sResizeEndEvent] = this._wfnOnResizeEndTable;
|
|
}else{
|
|
// [IE 8-] event 객체가 handler로 전달되지 않는 문제가 있어서 별도의 handler 사용
|
|
elTable[sResizeEndEvent] = this._getTableResizeEndHandler(elTable);
|
|
}
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-2136]
|
|
|
|
switch(this.nStatus){
|
|
case this.STATUS.S_0:
|
|
if(this._isOnBorder(wevE)){
|
|
//this._changeTableEditorStatus(this.MOUSEOVER_BORDER);
|
|
this._showCellResizeGrip(wevE);
|
|
}else{
|
|
this._hideResizer();
|
|
}
|
|
break;
|
|
case this.STATUS.MOUSEDOWN_CELL:
|
|
// change the status to CELL_SELECTING if the mouse moved out of the inital TD
|
|
// [SMARTEDITORSUS-1672]
|
|
//var elTD = nhn.husky.SE2M_Utils.findAncestorByTagName("TD", wevE.element);
|
|
var elTD = nhn.husky.SE2M_Utils.findClosestAncestorAmongTagNames(this._aCellName, wevE.element);
|
|
// --[SMARTEDITORSUS-1672]
|
|
if((elTD && elTD !== this.elSelectionStartTD) || !elTD){
|
|
if(!elTD){elTD = this.elSelectionStartTD;}
|
|
|
|
this._reassignCellSizes(this.elSelectionStartTable);
|
|
|
|
this._startCellSelection();
|
|
this._selectBetweenCells(this.elSelectionStartTD, elTD);
|
|
}
|
|
break;
|
|
case this.STATUS.CELL_SELECTING:
|
|
// show selection
|
|
// [SMARTEDITORSUS-1672]
|
|
//var elTD = nhn.husky.SE2M_Utils.findAncestorByTagName("TD", wevE.element);
|
|
var elTD = nhn.husky.SE2M_Utils.findClosestAncestorAmongTagNames(this._aCellName, wevE.element);
|
|
// --[SMARTEDITORSUS-1672]
|
|
if(!elTD || elTD === this.elLastSelectedTD){return;}
|
|
|
|
var elTBL = nhn.husky.SE2M_Utils.findAncestorByTagName("TABLE", elTD);
|
|
if(elTBL !== this.elSelectionStartTable){return;}
|
|
|
|
this.elLastSelectedTD = elTD;
|
|
|
|
this._selectBetweenCells(this.elSelectionStartTD, elTD);
|
|
|
|
break;
|
|
case this.STATUS.CELL_SELECTED:
|
|
break;
|
|
}
|
|
},
|
|
|
|
// 셀 선택 상태에서 문서영역을 상/하로 벗어날 경우, 벗어난 방향으로 선택 셀을 늘려가며 문서의 스크롤을 해줌
|
|
$ON_EVENT_OUTER_DOC_MOUSEMOVE : function(wevE){
|
|
switch(this.nStatus){
|
|
case this.STATUS.CELL_SELECTING:
|
|
var htPos = wevE.pos();
|
|
var nYPos = htPos.pageY;
|
|
var nXPos = htPos.pageX;
|
|
if(nYPos < this.htEditingAreaPos.top){
|
|
var y = this.htSelectionSPos.y;
|
|
if(y > 0){
|
|
this.htSelectionSPos.y--;
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
oSelection.selectNodeContents(this.aSelectedCells[0]);
|
|
oSelection.select();
|
|
oSelection.oBrowserSelection.selectNone();
|
|
}
|
|
}else{
|
|
if(nYPos > this.htEditingAreaPos.bottom){
|
|
var y = this.htSelectionEPos.y;
|
|
if(y < this.htMap[0].length - 1){
|
|
this.htSelectionEPos.y++;
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
oSelection.selectNodeContents(this.htMap[this.htSelectionEPos.x][this.htSelectionEPos.y]);
|
|
oSelection.select();
|
|
oSelection.oBrowserSelection.selectNone();
|
|
}
|
|
}
|
|
}
|
|
|
|
if(nXPos < this.htEditingAreaPos.left){
|
|
var x = this.htSelectionSPos.x;
|
|
if(x > 0){
|
|
this.htSelectionSPos.x--;
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
oSelection.selectNodeContents(this.aSelectedCells[0]);
|
|
oSelection.select();
|
|
oSelection.oBrowserSelection.selectNone();
|
|
}
|
|
}else{
|
|
if(nXPos > this.htEditingAreaPos.right){
|
|
var x = this.htSelectionEPos.x;
|
|
if(x < this.htMap.length - 1){
|
|
this.htSelectionEPos.x++;
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
var oSelection = this.oApp.getSelection();
|
|
oSelection.selectNodeContents(this.htMap[this.htSelectionEPos.x][this.htSelectionEPos.y]);
|
|
oSelection.select();
|
|
oSelection.oBrowserSelection.selectNone();
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
},
|
|
|
|
$ON_EVENT_OUTER_DOC_MOUSEUP : function(wevE){
|
|
this._eventEditingAreaMouseup(wevE);
|
|
},
|
|
|
|
$ON_EVENT_EDITING_AREA_MOUSEUP : function(wevE){
|
|
this._eventEditingAreaMouseup(wevE);
|
|
},
|
|
|
|
_eventEditingAreaMouseup : function(wevE){
|
|
if(this.oApp.getEditingMode() != "WYSIWYG"){return;}
|
|
|
|
switch(this.nStatus){
|
|
case this.STATUS.S_0:
|
|
break;
|
|
case this.STATUS.MOUSEDOWN_CELL:
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
break;
|
|
case this.STATUS.CELL_SELECTING:
|
|
this._changeTableEditorStatus(this.STATUS.CELL_SELECTED);
|
|
break;
|
|
case this.STATUS.CELL_SELECTED:
|
|
break;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Table의 block으로 잡힌 영역을 넘겨준다.
|
|
* @see hp_SE2M_TableBlockStyler.js
|
|
*/
|
|
$ON_GET_SELECTED_CELLS : function(sAttr,oReturn){
|
|
if(!!this.aSelectedCells){
|
|
oReturn[sAttr] = this.aSelectedCells;
|
|
}
|
|
},
|
|
|
|
_coverResizeLayer : function(){
|
|
// [SMARTEDITORSUS-1504] 에디터 전체 크기보다 창이 작아졌을 때 elResizeGrid가 최대화된 elResizeCover으로부터 벗어나는 이슈가 있음
|
|
//this.elResizeCover.style.position = "absolute";
|
|
this.elResizeCover.style.position = "fixed";
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
var size = jindo.$Document().clientSize();
|
|
this.elResizeCover.style.width = size.width - this.nPageLeftRightMargin + "px";
|
|
this.elResizeCover.style.height = size.height - this.nPageTopBottomMargin + "px";
|
|
//this.elResizeCover.style.width = size.width + "px";
|
|
//this.elResizeCover.style.height = size.height + "px";
|
|
//document.body.insertBefore(this.elResizeCover, document.body.firstChild);
|
|
document.body.appendChild(this.elResizeCover);
|
|
},
|
|
|
|
_uncoverResizeLayer : function(){
|
|
this.elResizeGrid.appendChild(this.elResizeCover);
|
|
this.elResizeCover.style.position = "";
|
|
this.elResizeCover.style.width = "100%";
|
|
this.elResizeCover.style.height = "100%";
|
|
},
|
|
|
|
_reassignCellSizes : function(elTable){
|
|
var allCells = new Array(2);
|
|
allCells[0] = jindo.$$(">TBODY>TR>TD", elTable, {oneTimeOffCache:true});
|
|
allCells[1] = jindo.$$(">TBODY>TR>TH", elTable, {oneTimeOffCache:true});
|
|
|
|
var aAllCellsWithSizeInfo = new Array(allCells[0].length + allCells[1].length);
|
|
var numCells = 0;
|
|
|
|
var nTblBorderPadding = this.parseIntOr0(elTable.border);
|
|
var nTblCellPadding = this.parseIntOr0(elTable.cellPadding);
|
|
|
|
// remember all the dimensions first and then assign later.
|
|
// this is done this way because if the table/cell size were set in %, setting one cell size would change size of other cells, which are still yet in %.
|
|
// 1 for TD and 1 for TH
|
|
for(var n = 0; n < 2; n++){
|
|
for(var i = 0; i < allCells[n].length; i++){
|
|
var elCell = allCells[n][i];
|
|
var welCell = jindo.$Element(elCell);
|
|
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
|
|
// [SMARTEDITORSUS-1427][SMARTEDITORSUS-1431][SMARTEDITORSUS-1491][SMARTEDITORSUS-1504] IE9, 10에서 Jindo.$Element#css 가 빈 속성값을 1px로 가져오는 문제점이 있어 대체
|
|
/*var nPaddingLeft = this.parseIntOr0(welCell.css("paddingLeft"));
|
|
var nPaddingRight = this.parseIntOr0(welCell.css("paddingRight"));
|
|
var nPaddingTop = this.parseIntOr0(welCell.css("paddingTop"));
|
|
var nPaddingBottom = this.parseIntOr0(welCell.css("paddingBottom"));
|
|
var nBorderLeft = this.parseBorder(welCell.css("borderLeftWidth"), welCell.css("borderLeftStyle"));
|
|
var nBorderRight = this.parseBorder(welCell.css("borderRightWidth"), welCell.css("borderRightStyle"));
|
|
var nBorderTop = this.parseBorder(welCell.css("borderTopWidth"), welCell.css("borderTopStyle"));
|
|
var nBorderBottom = this.parseBorder(welCell.css("borderBottomWidth"), welCell.css("borderBottomStyle"));*/
|
|
var nPaddingLeft, nPaddingRight, nPaddingTop, nPaddingBottom;
|
|
var nBorderLeft, nBorderRight, nBorderTop, nBorderBottom;
|
|
// --[SMARTEDITORSUS-1427][SMARTEDITORSUS-1431][SMARTEDITORSUS-1491][SMARTEDITORSUS-1504]
|
|
|
|
var nOffsetWidth, nOffsetHeight;
|
|
|
|
// [SMARTEDITORSUS-1571] IE 10 이상임에도 불구하고, 문서 모드가 8 이하로 설정되어 있는 경우가 있어 메서드 기반 분기로 변경
|
|
if(elCell.getComputedStyle){
|
|
// --[SMARTEDITORSUS-1571]
|
|
// getComputedStyle()로 inherit된 스타일을 획득. IE 8 이하에서는 지원되지 않는다.
|
|
nPaddingLeft = parseFloat(getComputedStyle(elCell).paddingLeft, 10);
|
|
nPaddingRight = parseFloat(getComputedStyle(elCell).paddingRight, 10);
|
|
nPaddingTop = parseFloat(getComputedStyle(elCell).paddingTop, 10);
|
|
nPaddingBottom = parseFloat(getComputedStyle(elCell).paddingBottom, 10);
|
|
|
|
// 최초 리사이징 직전 width attribute에서 style의 width로 이행하는 과정에서 미세보정이 있기 때문에 크기가 조금 변한다.
|
|
nBorderLeft = parseFloat(getComputedStyle(elCell).borderLeftWidth, 10);
|
|
nBorderRight = parseFloat(getComputedStyle(elCell).borderRightWidth, 10);
|
|
nBorderTop = parseFloat(getComputedStyle(elCell).borderTopWidth, 10);
|
|
nBorderBottom = parseFloat(getComputedStyle(elCell).borderBottomWidth, 10);
|
|
}else{ // 이 방식은 inherit된 스타일을 가져오지 못하는 문제와 함께, 일부 브라우저의 소수점 값을 버림하는 문제가 있다.
|
|
// [SMARTEDITORSUS-1427][SMARTEDITORSUS-1431][SMARTEDITORSUS-1491]
|
|
nPaddingLeft = this.parseIntOr0(elCell.style.paddingLeft);
|
|
nPaddingRight = this.parseIntOr0(elCell.style.paddingRight);
|
|
nPaddingTop = this.parseIntOr0(elCell.style.paddingTop);
|
|
nPaddingBottom = this.parseIntOr0(elCell.style.paddingBottom);
|
|
// --[SMARTEDITORSUS-1427][SMARTEDITORSUS-1431][SMARTEDITORSUS-1491]
|
|
|
|
// 기존 로직을 사용. IE의 경우 bug로 분류하여 1px를 획득하도록 설정되어 있다.
|
|
nBorderLeft = this.parseBorder(welCell.css("borderLeftWidth"), welCell.css("borderLeftStyle"));
|
|
nBorderRight = this.parseBorder(welCell.css("borderRightWidth"), welCell.css("borderRightStyle"));
|
|
nBorderTop = this.parseBorder(welCell.css("borderTopWidth"), welCell.css("borderTopStyle"));
|
|
nBorderBottom = this.parseBorder(welCell.css("borderBottomWidth"), welCell.css("borderBottomStyle"));
|
|
}
|
|
|
|
/**
|
|
* 매번 발생하는 리사이징 오차를 최소하기 위하여, 2회차부터는 1회차에 적용되는 style 값을 가져온다.
|
|
*
|
|
* width와 height attribute는 최초 1회에 제거된다.
|
|
* 즉 2회차부터는, 동적으로 변하는 style의 width, height 값을 그대로 사용한다.
|
|
* */
|
|
/*nOffsetWidth = elCell.offsetWidth - (nPaddingLeft + nPaddingRight + nBorderLeft + nBorderRight) + "px";
|
|
nOffsetHeight = elCell.offsetHeight - (nPaddingTop + nPaddingBottom + nBorderTop + nBorderBottom) + "px";*/
|
|
var nWidth = jindo.$Element(elCell).attr("width");
|
|
var nHeight = jindo.$Element(elCell).attr("height");
|
|
if(!nWidth && !nHeight){
|
|
nOffsetWidth = elCell.style.width;
|
|
nOffsetHeight = elCell.style.height;
|
|
}else{
|
|
nOffsetWidth = elCell.offsetWidth - (nPaddingLeft + nPaddingRight + nBorderLeft + nBorderRight) + "px";
|
|
nOffsetHeight = elCell.offsetHeight - (nPaddingTop + nPaddingBottom + nBorderTop + nBorderBottom) + "px";
|
|
}
|
|
|
|
/*if(htBrowser.ie && (htBrowser.nativeVersion >= 9 && htBrowser.nativeVersion <= 10)){
|
|
// IE9, IE10
|
|
// [SMARTEDITORSUS-1427][SMARTEDITORSUS-1431][SMARTEDITORSUS-1491] IE9, 10에서 Jindo.$Element#css 관련 문제에 대응하면 다른 브라우저와 동일한 수식 적용 가능
|
|
//nOffsetWidth = elCell.offsetWidth + "px";
|
|
//nOffsetHeight = elCell.offsetHeight - (nPaddingTop + nPaddingBottom + nBorderTop + nBorderBottom) + "px";
|
|
nOffsetWidth = elCell.offsetWidth - (nPaddingLeft + nPaddingRight + nBorderLeft + nBorderRight) + "px";
|
|
nOffsetHeight = elCell.offsetHeight - (nPaddingTop + nPaddingBottom + nBorderTop + nBorderBottom) + "px";
|
|
// --[SMARTEDITORSUS-1427][SMARTEDITORSUS-1431][SMARTEDITORSUS-1491]
|
|
}else{
|
|
// Firefox, Chrome, IE7, IE8
|
|
nOffsetWidth = elCell.offsetWidth - (nPaddingLeft + nPaddingRight + nBorderLeft + nBorderRight) + "px";
|
|
nOffsetHeight = elCell.offsetHeight - (nPaddingTop + nPaddingBottom + nBorderTop + nBorderBottom) + "px";
|
|
}*/
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
aAllCellsWithSizeInfo[numCells++] = [elCell, nOffsetWidth, nOffsetHeight];
|
|
}
|
|
}
|
|
|
|
// [SMARTEDITORSUS-2136]
|
|
var nTableWidth = elTable._nWidth, nTableHeight = elTable._nHeight,
|
|
nResizedTableWidth = elTable._nResizedWidth, nResizedTableHeight = elTable._nResizedHeight, // [IE] resizeend 이벤트가 발생했다면 존재
|
|
nTableWidthAdjustRatio = 1, nTableHeightAdjustRatio = 1; // 셀 크기 변경 시 사용되는 보정계수
|
|
|
|
if(nhn.husky.SE2M_Utils.isNumber(nTableWidth) && nhn.husky.SE2M_Utils.isNumber(nTableHeight)
|
|
&& nhn.husky.SE2M_Utils.isNumber(nResizedTableWidth) && nhn.husky.SE2M_Utils.isNumber(nResizedTableHeight)){
|
|
// [IE] resizeend 정보가 기록되어 있다면, 셀 크기 변경 시 보정계수 조정
|
|
nTableWidthAdjustRatio = nResizedTableWidth / nTableWidth,
|
|
nTableHeightAdjustRatio = nResizedTableHeight / nTableHeight;
|
|
}
|
|
|
|
// [IE] 기록해 둔 resizeend 크기는 일회성으로 사용한다.
|
|
if(elTable._nResizedWidth){
|
|
nhn.husky.SE2M_Utils.deleteProperty(elTable, '_nResizedWidth');
|
|
}
|
|
if(elTable._nResizedHeight){
|
|
nhn.husky.SE2M_Utils.deleteProperty(elTable, '_nResizedHeight');
|
|
}
|
|
// --[SMARTEDITORSUS-2136]
|
|
|
|
for(var i = 0; i < numCells; i++){
|
|
var aCellInfo = aAllCellsWithSizeInfo[i];
|
|
|
|
aCellInfo[0].removeAttribute("width");
|
|
aCellInfo[0].removeAttribute("height");
|
|
|
|
// [SMARTEDITORSUS-2136] 보정계수를 적용하여 셀의 크기 변경
|
|
aCellInfo[0].style.width = (parseFloat(aCellInfo[1], 10) * nTableWidthAdjustRatio) + 'px';
|
|
aCellInfo[0].style.height = (parseFloat(aCellInfo[2], 10) * nTableHeightAdjustRatio) + 'px';
|
|
// --[SMARTEDITORSUS-2136]
|
|
|
|
// jindo.$Element(aCellInfo[0]).css("width", aCellInfo[1]);
|
|
// jindo.$Element(aCellInfo[0]).css("height", aCellInfo[2]);
|
|
}
|
|
|
|
// [SMARTEDITORSUS-2136] HTMLElement의 속성으로 <table>의 크기 기록
|
|
var welTable = jindo.$Element(elTable);
|
|
elTable._nWidth = welTable.width(),
|
|
elTable._nHeight = welTable.height();
|
|
// --[SMARTEDITORSUS-2136]
|
|
|
|
elTable.removeAttribute("width");
|
|
elTable.removeAttribute("height");
|
|
elTable.style.width = "";
|
|
elTable.style.height = "";
|
|
},
|
|
|
|
_fnOnMouseDownResizeCover : function(oEvent){
|
|
this.bResizing = true;
|
|
this.nStartHeight = oEvent.pos().clientY;
|
|
|
|
// [SMARTEDITORSUS-1504] 표 테두리를 누를 때마다 글양식이 2px씩 세로로 길어지는 문제가 있는데, 이를 위한 flag
|
|
this.bResizingCover = true;
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
this._wfnOnMouseMoveResizeCover.attach(this.elResizeCover, "mousemove");
|
|
this._wfnOnMouseUpResizeCover.attach(document, "mouseup");
|
|
|
|
this._coverResizeLayer();
|
|
this.elResizeGrid.style.border = "1px dotted black";
|
|
|
|
this.nStartHeight = oEvent.pos().clientY;
|
|
this.nStartWidth = oEvent.pos().clientX;
|
|
|
|
// [SMARTEDITORSUS-1504] 표 리사이즈용 gripper의 배치를 WYSIWYG 편집 영역 위치 기반으로 개선
|
|
this.nClientXDiff = this.nStartWidth - this.htResizing.htEPos.clientX;
|
|
this.nClientYDiff = this.nStartHeight - this.htResizing.htEPos.clientY;
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
this._reassignCellSizes(this.htResizing.elTable);
|
|
|
|
this.htMap = this._getCellMapping(this.htResizing.elTable);
|
|
var htPosition = this._getBasisCellPosition(this.htResizing.elCell);
|
|
|
|
var nOffsetX = (parseInt(this.htResizing.elCell.getAttribute("colspan")) || 1) - 1;
|
|
var nOffsetY = (parseInt(this.htResizing.elCell.getAttribute("rowspan")) || 1) - 1;
|
|
var x = htPosition.x + nOffsetX + this.htResizing.nHA;
|
|
var y = htPosition.y + nOffsetY + this.htResizing.nVA;
|
|
|
|
if(x < 0 || y < 0){return;}
|
|
|
|
this.htAllAffectedCells = this._getAllAffectedCells(x, y, this.htResizing.nResizeMode, this.htResizing.elTable);
|
|
},
|
|
|
|
_fnOnMouseMoveResizeCover : function(oEvent){
|
|
// [SMARTEDITORSUS-1504] 표 모서리 Drag 사용성 개선
|
|
// - 최초 리사이징 후 해당 위치에서 바로 마우스를 눌러 Drag 가능
|
|
if(jindo.$Agent().navigator().chrome || jindo.$Agent().navigator().safari){
|
|
if(this.htResizing.nPreviousResizeMode != undefined && this.htResizing.nPreviousResizeMode != 0){
|
|
if(this.htResizing.nResizeMode != this.htResizing.nPreviousResizeMode){
|
|
this.htResizing.nResizeMode = this.htResizing.nPreviousResizeMode;
|
|
this._showResizer();
|
|
}
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
var nHeightChange = oEvent.pos().clientY - this.nStartHeight;
|
|
var nWidthChange = oEvent.pos().clientX - this.nStartWidth;
|
|
|
|
var oEventPos = oEvent.pos();
|
|
|
|
// [SMARTEDITORSUS-1504] 표 리사이즈용 gripper의 배치를 WYSIWYG 편집 영역 위치 기반으로 개선
|
|
/*if(this.htResizing.nResizeMode == 1){
|
|
this.elResizeGrid.style.left = oEventPos.pageX - this.parseIntOr0(this.elResizeGrid.style.width)/2 + "px";
|
|
}else{
|
|
this.elResizeGrid.style.top = oEventPos.pageY - this.parseIntOr0(this.elResizeGrid.style.height)/2 + "px";
|
|
}*/
|
|
if(this.htResizing.nResizeMode == 1){
|
|
this.elResizeGrid.style.left = oEvent.pos().clientX - this.nClientXDiff - this.parseIntOr0(this.elResizeGrid.style.width)/2 + "px";
|
|
}else{
|
|
this.elResizeGrid.style.top = oEvent.pos().clientY - this.nClientYDiff - this.parseIntOr0(this.elResizeGrid.style.height)/2 + "px";
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
},
|
|
|
|
_fnOnMouseUpResizeCover : function(oEvent){
|
|
this.bResizing = false;
|
|
this._hideResizer();
|
|
this._uncoverResizeLayer();
|
|
this.elResizeGrid.style.border = "";
|
|
|
|
this._wfnOnMouseMoveResizeCover.detach(this.elResizeCover, "mousemove");
|
|
this._wfnOnMouseUpResizeCover.detach(document, "mouseup");
|
|
|
|
var nHeightChange = 0;
|
|
var nWidthChange = 0;
|
|
|
|
if(this.htResizing.nResizeMode == 2){
|
|
nHeightChange = oEvent.pos().clientY - this.nStartHeight;
|
|
}
|
|
if(this.htResizing.nResizeMode == 1){
|
|
nWidthChange = oEvent.pos().clientX - this.nStartWidth;
|
|
|
|
if(this.htAllAffectedCells.nMinBefore != -1 && nWidthChange < -1*this.htAllAffectedCells.nMinBefore){
|
|
nWidthChange = -1 * this.htAllAffectedCells.nMinBefore + this.MIN_CELL_WIDTH;
|
|
}
|
|
if(this.htAllAffectedCells.nMinAfter != -1 && nWidthChange > this.htAllAffectedCells.nMinAfter){
|
|
nWidthChange = this.htAllAffectedCells.nMinAfter - this.MIN_CELL_WIDTH;
|
|
}
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1504] FireFox는 소수점으로 size가 나오는데, parseInt는 소수점 이하를 버림
|
|
// [SMARTEDITORSUS-1655] 메서드, 프로퍼티 기반 리팩토링
|
|
var width, height;
|
|
// --[SMARTEDITORSUS-1655]
|
|
|
|
var aCellsBefore = this.htAllAffectedCells.aCellsBefore;
|
|
for(var i = 0; i < aCellsBefore.length; i++){
|
|
var elCell = aCellsBefore[i];
|
|
|
|
// [SMARTEDITORSUS-1655]
|
|
width = 0, height = 0;
|
|
|
|
width = elCell.style.width;
|
|
if(isNaN(parseFloat(width, 10))){ // 값이 없거나 "auto"인 경우
|
|
width = 0;
|
|
}else{
|
|
width = parseFloat(width, 10);
|
|
}
|
|
width += nWidthChange;
|
|
|
|
height = elCell.style.height;
|
|
if(isNaN(parseFloat(height, 10))){ // 값이 없거나 "auto"인 경우
|
|
height = 0;
|
|
}else{
|
|
height = parseFloat(height, 10);
|
|
}
|
|
height += nHeightChange;
|
|
// --[SMARTEDITORSUS-1655]
|
|
|
|
//var width = this.parseIntOr0(elCell.style.width) + nWidthChange;
|
|
elCell.style.width = Math.max(width, this.MIN_CELL_WIDTH) + "px";
|
|
|
|
//var height = this.parseIntOr0(elCell.style.height) + nHeightChange;
|
|
elCell.style.height = Math.max(height, this.MIN_CELL_HEIGHT) + "px";
|
|
}
|
|
|
|
var aCellsAfter = this.htAllAffectedCells.aCellsAfter;
|
|
for(var i = 0; i < aCellsAfter.length; i++){
|
|
var elCell = aCellsAfter[i];
|
|
|
|
// [SMARTEDITORSUS-1655]
|
|
width = 0, height = 0;
|
|
|
|
width = elCell.style.width;
|
|
if(isNaN(parseFloat(width, 10))){ // 값이 없거나 "auto"인 경우
|
|
width = 0;
|
|
}else{
|
|
width = parseFloat(width, 10);
|
|
}
|
|
width -= nWidthChange;
|
|
|
|
height = elCell.style.height;
|
|
if(isNaN(parseFloat(height, 10))){ // 값이 없거나 "auto"인 경우
|
|
height = 0;
|
|
}else{
|
|
height = parseFloat(height, 10);
|
|
}
|
|
height -= nHeightChange;
|
|
// --[SMARTEDITORSUS-1655]
|
|
|
|
//var width = this.parseIntOr0(elCell.style.width) - nWidthChange;
|
|
elCell.style.width = Math.max(width, this.MIN_CELL_WIDTH) + "px";
|
|
|
|
//var height = this.parseIntOr0(elCell.style.height) - nHeightChange;
|
|
elCell.style.height = Math.max(height, this.MIN_CELL_HEIGHT) + "px";
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
// [SMARTEDITORSUS-1504] 표 테두리를 누를 때마다 글양식이 2px씩 세로로 길어지는 문제가 있는데, 이를 위한 flag
|
|
this.bResizingCover = false;
|
|
// --[SMARTEDITORSUS-1504]
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2136] [IE 9, 10] resizeend 이벤트 handler
|
|
* [IE 11] resizeend 이벤트 deprecated : pointerup 이벤트로 갈음한다.
|
|
* */
|
|
_fnOnResizeEndTable : function(oEvent){
|
|
var wev = jindo.$Event(oEvent);
|
|
var elTable = wev.element;
|
|
|
|
this._markResizedMetric(elTable);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2136] [IE 8-] resizeend 이벤트 handler
|
|
* */
|
|
_getTableResizeEndHandler : function(elTable){
|
|
return jindo.$Fn(this._markResizedMetric, this).bind(elTable);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-2136] [IE] resizeend 이벤트 발생 시
|
|
* 이 경우에 변화하는 표의 크기를 기록해 두고
|
|
* 셀 크기 변경 시 보정에 사용한다.
|
|
* */
|
|
_markResizedMetric : function(elTable){
|
|
if(!elTable || (elTable.tagName.toUpperCase() !== 'TABLE')){
|
|
return;
|
|
}
|
|
|
|
// HTMLElement의 속성으로 resize된 <table>의 크기 기록
|
|
var welCurrentTable = jindo.$Element(elTable);
|
|
elTable._nResizedWidth = welCurrentTable.width(),
|
|
elTable._nResizedHeight = welCurrentTable.height();
|
|
},
|
|
|
|
$ON_CLOSE_QE_LAYER : function(oEvent){
|
|
// [SMARTEDITORSUS-2137] 퀵에디터의 접기/펼치기 클릭 시에는, cell의 상태를 변경하지 않는다.
|
|
var elTarget = oEvent ? oEvent.element : null,
|
|
welTarget = elTarget ? jindo.$Element(elTarget) : null,
|
|
aQEToggleClassName = ['q_open_table_fold', 'q_open_table_full'], sQEToggleClassName,
|
|
i, len = aQEToggleClassName.length,
|
|
isFromQEToggle = false;
|
|
|
|
if(welTarget){
|
|
for(i = len; i--;){
|
|
sQEToggleClassName = aQEToggleClassName[i];
|
|
if(welTarget.hasClass(sQEToggleClassName)){
|
|
isFromQEToggle = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(isFromQEToggle){
|
|
return;
|
|
}
|
|
// --[SMARTEDITORSUS-2137]
|
|
|
|
this._changeTableEditorStatus(this.STATUS.S_0);
|
|
},
|
|
|
|
_changeTableEditorStatus : function(nNewStatus){
|
|
if(this.nStatus == nNewStatus){return;}
|
|
this.nStatus = nNewStatus;
|
|
|
|
switch(nNewStatus){
|
|
case this.STATUS.S_0:
|
|
if(this.nStatus == this.STATUS.MOUSEDOWN_CELL){
|
|
break;
|
|
}
|
|
|
|
this._deselectCells();
|
|
|
|
// 히스토리 저장 (선택 위치는 저장하지 않음)
|
|
if(!!this.sQEAction){
|
|
this.oApp.exec("RECORD_UNDO_ACTION", [this.sQEAction, {elSaveTarget:this.elSelectionStartTable, bDontSaveSelection:true}]);
|
|
this.sQEAction = "";
|
|
}
|
|
|
|
if(this.oApp.oNavigator["safari"] || this.oApp.oNavigator["chrome"]){
|
|
this.oApp.getWYSIWYGDocument().onselectstart = null;
|
|
}
|
|
|
|
this.oApp.exec("ENABLE_WYSIWYG", []);
|
|
this.oApp.exec("CLOSE_QE_LAYER");
|
|
|
|
this.elSelectionStartTable = null;
|
|
break;
|
|
case this.STATUS.CELL_SELECTING:
|
|
// [SMARTEDITORSUS-2155] Legacy Event Methods (https://msdn.microsoft.com/en-us/library/ms536742(v=vs.85).aspx)
|
|
if(this.oApp.oNavigator.ie && (typeof(document.body.setCapture) === 'function')){
|
|
document.body.setCapture(false);
|
|
}
|
|
break;
|
|
case this.STATUS.CELL_SELECTED:
|
|
this.oApp.delayedExec("MSG_CELL_SELECTED", [], 0);
|
|
// [SMARTEDITORSUS-2155] Legacy Event Methods (https://msdn.microsoft.com/en-us/library/ms536742(v=vs.85).aspx)
|
|
if(this.oApp.oNavigator.ie && (typeof(document.body.releaseCapture) === 'function')){
|
|
document.body.releaseCapture();
|
|
}
|
|
break;
|
|
}
|
|
|
|
this.oApp.exec("TABLE_EDITOR_STATUS_CHANGED", [this.nStatus]);
|
|
},
|
|
|
|
_isOnBorder : function(wevE){
|
|
// ===========================[Start: Set/init global resizing info]===========================
|
|
// 0: not resizing
|
|
// 1: horizontal resizing
|
|
// 2: vertical resizing
|
|
this.htResizing.nResizeMode = 0;
|
|
this.htResizing.elCell = wevE.element;
|
|
// [SMARTEDITORSUS-1672]
|
|
//if(wevE.element.tagName != "TD" && wevE.element.tagName != "TH"){return false;}
|
|
if(!this._rxCellNames.test(wevE.element.tagName)){return false;}
|
|
// --[SMARTEDITORSUS-1672]
|
|
|
|
this.htResizing.elTable = nhn.husky.SE2M_Utils.findAncestorByTagName("TABLE", this.htResizing.elCell);
|
|
if(!this.htResizing.elTable){return;}
|
|
|
|
if(!jindo.$Element(this.htResizing.elTable).hasClass(this._sSETblClass) && !jindo.$Element(this.htResizing.elTable).hasClass(this._sSEReviewTblClass)){return;}
|
|
|
|
// Adjustment variables: to be used to map the x, y position of the resizing point relative to elCell
|
|
// eg) When left border of a cell at 2,2 is selected, the actual cell that has to be resized is the one at 1,2. So, set the horizontal adjustment to -1.
|
|
// Vertical Adjustment
|
|
this.htResizing.nVA = 0;
|
|
// Horizontal Adjustment
|
|
this.htResizing.nHA = 0;
|
|
|
|
this.htResizing.nBorderLeftPos = 0;
|
|
this.htResizing.nBorderTopPos = -1;
|
|
this.htResizing.htEPos = wevE.pos(true);
|
|
this.htResizing.nBorderSize = this.parseIntOr0(this.htResizing.elTable.border);
|
|
// ===========================[E N D: Set/init global resizing info]===========================
|
|
|
|
// Separate info is required as the offsetX/Y are different in IE and FF
|
|
// For IE, (0, 0) is top left corner of the cell including the border.
|
|
// For FF, (0, 0) is top left corner of the cell excluding the border.
|
|
var nAdjustedDraggableCellEdge1;
|
|
var nAdjustedDraggableCellEdge2;
|
|
if(jindo.$Agent().navigator().ie || jindo.$Agent().navigator().safari){
|
|
nAdjustedDraggableCellEdge1 = this.htResizing.nBorderSize + this.nDraggableCellEdge;
|
|
nAdjustedDraggableCellEdge2 = this.nDraggableCellEdge;
|
|
}else{
|
|
nAdjustedDraggableCellEdge1 = this.nDraggableCellEdge;
|
|
nAdjustedDraggableCellEdge2 = this.htResizing.nBorderSize + this.nDraggableCellEdge;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1504] 경계선 판별에 사용
|
|
var elCellWidth = this.htResizing.elCell.clientWidth,
|
|
elCellHeight = this.htResizing.elCell.clientHeight;
|
|
|
|
nRightBorderCriteria = elCellWidth - this.htResizing.htEPos.offsetX,
|
|
nBottomBorderCriteria = elCellHeight - this.htResizing.htEPos.offsetY;
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
// top border of the cell is selected
|
|
if(this.htResizing.htEPos.offsetY <= nAdjustedDraggableCellEdge1){
|
|
// [SMARTEDITORSUS-2136] resizeend 이후 표의 상단 테두리에 resize grip이 생성되면 스크립트 오류를 발생시킨다.
|
|
var _isFirstRow = false,
|
|
_elRow = this.htResizing.elCell.parentNode;
|
|
var _elTbody = _elRow.parentNode;
|
|
var _aRow = jindo.$$('tr', _elTbody, {oneTimeOffCache : true});
|
|
if(_aRow[0] == _elRow){
|
|
_isFirstRow = true;
|
|
}
|
|
// --[SMARTEDITORSUS-2136]
|
|
// top border of the first cell can't be dragged
|
|
if(_isFirstRow){
|
|
this.htResizing.nVA = -1;
|
|
|
|
// [SMARTEDITORSUS-1504] 표 리사이즈용 gripper 배치 개선
|
|
//this.htResizing.nResizeMode = 2;
|
|
this.htResizing.nResizeMode = 4;
|
|
// --[SMARTEDITORSUS-1504]
|
|
}
|
|
}
|
|
// bottom border of the cell is selected
|
|
// [SMARTEDITORSUS-1504] 표 모서리 Drag 사용성 개선
|
|
//if(this.htResizing.elCell.offsetHeight-nAdjustedDraggableCellEdge2 <= this.htResizing.htEPos.offsetY){
|
|
if(nBottomBorderCriteria <= nAdjustedDraggableCellEdge2){
|
|
this.htResizing.nBorderTopPos = this.htResizing.elCell.offsetHeight + nAdjustedDraggableCellEdge1 - 1;
|
|
this.htResizing.nResizeMode = 2;
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
// left border of the cell is selected
|
|
if(this.htResizing.htEPos.offsetX <= nAdjustedDraggableCellEdge1){
|
|
// left border of the first cell can't be dragged
|
|
// [SMARTEDITORSUS-1504] 표 리사이즈용 gripper 배치 개선
|
|
// 일반 표는 아래 if문을 거치지 않지만, 글양식의 경우 가장 좌측 cell의 previousSibling이 textNode이기 때문에 if문에 부합하는 문제가 있었다.
|
|
/*if(this.htResizing.elCell.previousSibling){
|
|
this.htResizing.nHA = -1;
|
|
|
|
this.htResizing.nResizeMode = 0;
|
|
}*/
|
|
|
|
// 가장 좌측의 cell은 그 offsetLeft가 table의 scrollLeft와 같다.
|
|
if(this.htResizing.elTable.scrollLeft != this.htResizing.elCell.offsetLeft){
|
|
this.htResizing.nHA = -1;
|
|
|
|
this.htResizing.nResizeMode = 3;
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
}
|
|
// right border of the cell is selected
|
|
// [SMARTEDITORSUS-1504] 표 모서리 Drag 사용성 개선
|
|
//if(this.htResizing.elCell.offsetWidth - nAdjustedDraggableCellEdge2 <= this.htResizing.htEPos.offsetX){
|
|
if(nRightBorderCriteria <= nAdjustedDraggableCellEdge1){
|
|
this.htResizing.nBorderLeftPos = this.htResizing.elCell.offsetWidth + nAdjustedDraggableCellEdge1 - 1;
|
|
this.htResizing.nResizeMode = 1;
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
// [SMARTEDITORSUS-1504] 표 모서리 Drag 사용성 개선
|
|
if(jindo.$Agent().navigator().chrome || jindo.$Agent().navigator().safari){
|
|
if(!this.htResizing.elPreviousCell){
|
|
this.htResizing.elPreviousCell = this.htResizing.elCell;
|
|
}else{
|
|
if(this.htResizing.elCell != this.htResizing.elPreviousCell){
|
|
this.htResizing.elPreviousCell = this.htResizing.elCell;
|
|
}
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
|
|
if(this.htResizing.nResizeMode === 0){return false;}
|
|
|
|
return true;
|
|
},
|
|
|
|
_showCellResizeGrip : function(){
|
|
// [SMARTEDITORSUS-2136] 표 상단 테두리에는 resize grip을 표시하지 않는다.
|
|
if(this.htResizing.nResizeMode == 4){
|
|
return;
|
|
}
|
|
// --[SMARTEDITORSUS-2136]
|
|
|
|
// [SMARTEDITORSUS-1504] gripper가 WYSIWYG 편집영역 위치정보에 기반하여 배치되도록 변경
|
|
// 만약 iframe 내부에 gripper를 생성한다면, 커서 위치를 기반으로 생성해 주면 됨
|
|
if(this.htResizing.nResizeMode == 1 || this.htResizing.nResizeMode == 3){
|
|
this.elResizeCover.style.cursor = "col-resize";
|
|
}else if(this.htResizing.nResizeMode == 2){
|
|
this.elResizeCover.style.cursor = "row-resize";
|
|
}
|
|
|
|
this._showResizer();
|
|
|
|
// gripper는 대상 셀에서 어느 경계 위에 커서가 위치했느냐에 기반하여 배치
|
|
if(this.htResizing.nResizeMode == 1){ // 오른쪽 경계
|
|
this._setResizerSize((this.htResizing.nBorderSize + this.nDraggableCellEdge) * 2, this.parseIntOr0(jindo.$Element(this.elIFrame).css("height")));
|
|
this.elResizeGrid.style.top = "0px";
|
|
this.elResizeGrid.style.left = this.htResizing.elCell.clientWidth + this.htResizing.htEPos.clientX - this.htResizing.htEPos.offsetX - this.parseIntOr0(this.elResizeGrid.style.width)/2 + "px";
|
|
}else if(this.htResizing.nResizeMode == 2){ // 아래쪽 경계
|
|
//가변폭을 지원하기 때문에 매번 현재 Container의 크기를 구해와서 Grip을 생성해야 한다.
|
|
var elIFrameWidth = this.oApp.elEditingAreaContainer.offsetWidth + "px";
|
|
this._setResizerSize(this.parseIntOr0(elIFrameWidth), (this.htResizing.nBorderSize + this.nDraggableCellEdge) * 2);
|
|
this.elResizeGrid.style.top = this.htResizing.elCell.clientHeight + this.htResizing.htEPos.clientY - this.htResizing.htEPos.offsetY - this.parseIntOr0(this.elResizeGrid.style.height)/2 + "px";
|
|
this.elResizeGrid.style.left = "0px";
|
|
}else if(this.htResizing.nResizeMode == 3){ // 왼쪽 경계
|
|
this._setResizerSize((this.htResizing.nBorderSize + this.nDraggableCellEdge) * 2, this.parseIntOr0(jindo.$Element(this.elIFrame).css("height")));
|
|
this.elResizeGrid.style.top = "0px";
|
|
this.elResizeGrid.style.left = + this.htResizing.htEPos.clientX - this.htResizing.htEPos.offsetX - this.parseIntOr0(this.elResizeGrid.style.width)/2 + "px";
|
|
|
|
// 이후 작업들은 오른쪽 경계를 기준으로 일괄처리
|
|
this.htResizing.nResizeMode = 1;
|
|
}
|
|
// --[SMARTEDITORSUS-1504]
|
|
},
|
|
|
|
_getAllAffectedCells : function(basis_x, basis_y, iResizeMode, oTable){
|
|
if(!oTable){return [];}
|
|
|
|
var oTbl = this._getCellMapping(oTable);
|
|
var iTblX = oTbl.length;
|
|
var iTblY = oTbl[0].length;
|
|
|
|
// 선택 테두리의 앞쪽 셀
|
|
var aCellsBefore = [];
|
|
// 선택 테두리의 뒤쪽 셀
|
|
var aCellsAfter = [];
|
|
|
|
var htResult;
|
|
|
|
var nMinBefore = -1, nMinAfter = -1;
|
|
// horizontal resizing -> need to get vertical rows
|
|
if(iResizeMode == 1){
|
|
for(var y = 0; y < iTblY; y++){
|
|
if(aCellsBefore.length>0 && aCellsBefore[aCellsBefore.length-1] == oTbl[basis_x][y]){continue;}
|
|
aCellsBefore[aCellsBefore.length] = oTbl[basis_x][y];
|
|
|
|
var nWidth = parseInt(oTbl[basis_x][y].style.width);
|
|
if(nMinBefore == -1 || nMinBefore > nWidth){
|
|
nMinBefore = nWidth;
|
|
}
|
|
}
|
|
|
|
if(oTbl.length > basis_x+1){
|
|
for(var y = 0; y < iTblY; y++){
|
|
if(aCellsAfter.length>0 && aCellsAfter[aCellsAfter.length-1] == oTbl[basis_x+1][y]){continue;}
|
|
aCellsAfter[aCellsAfter.length] = oTbl[basis_x+1][y];
|
|
|
|
var nWidth = parseInt(oTbl[basis_x + 1][y].style.width);
|
|
if(nMinAfter == -1 || nMinAfter > nWidth){
|
|
nMinAfter = nWidth;
|
|
}
|
|
}
|
|
}
|
|
htResult = {aCellsBefore: aCellsBefore, aCellsAfter: aCellsAfter, nMinBefore: nMinBefore, nMinAfter: nMinAfter};
|
|
}else{
|
|
for(var x = 0; x < iTblX; x++){
|
|
if(aCellsBefore.length>0 && aCellsBefore[aCellsBefore.length - 1] == oTbl[x][basis_y]){continue;}
|
|
aCellsBefore[aCellsBefore.length] = oTbl[x][basis_y];
|
|
|
|
if(nMinBefore == -1 || nMinBefore > oTbl[x][basis_y].style.height){
|
|
nMinBefore = oTbl[x][basis_y].style.height;
|
|
}
|
|
}
|
|
// 높이 리사이징 시에는 선택 테두리 앞쪽 셀만 조절 함으로 아래쪽 셀은 생성 할 필요 없음
|
|
|
|
htResult = {aCellsBefore: aCellsBefore, aCellsAfter: aCellsAfter, nMinBefore: nMinBefore, nMinAfter: nMinAfter};
|
|
}
|
|
|
|
return htResult;
|
|
},
|
|
|
|
_createCellResizeGrip : function(){
|
|
this.elTmp = document.createElement("DIV");
|
|
try{
|
|
this.elTmp.innerHTML = '<div style="position:absolute; overflow:hidden; z-index: 99; "><div onmousedown="return false" style="background-color:#000000;filter:alpha(opacity=0);opacity:0.0;-moz-opacity:0.0;-khtml-opacity:0.0;cursor: col-resize; left: 0px; top: 0px; width: 100%; height: 100%;font-size:1px;z-index: 999; "></div></div>';
|
|
this.elResizeGrid = this.elTmp.firstChild;
|
|
this.elResizeCover = this.elResizeGrid.firstChild;
|
|
}catch(e){}
|
|
|
|
// [SMARTEDITORSUS-1504] gripper를 WYSIWYG 편집 영역 위치 정보에 기반하여 배치하도록 개선
|
|
//document.body.appendChild(this.elResizeGrid);
|
|
// document.body 대신 WYSIWYG 편집 영역을 둘러싼 container div에 추가
|
|
var oContainer = jindo.$$.getSingle(".husky_seditor_editing_area_container");
|
|
oContainer.appendChild(this.elResizeGrid);
|
|
// --[SMARTEDITORSUS-1504]
|
|
},
|
|
|
|
_selectAll_Row : function(){
|
|
this.htSelectionSPos.x = 0;
|
|
this.htSelectionEPos.x = this.htMap.length - 1;
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
},
|
|
|
|
_selectAll_Column : function(){
|
|
this.htSelectionSPos.y = 0;
|
|
this.htSelectionEPos.y = this.htMap[0].length - 1;
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
},
|
|
|
|
_deleteSelectedCells : function(){
|
|
var elTmp;
|
|
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
elTmp = this.aSelectedCells[i];
|
|
|
|
// [SMARTEDITORSUS-1533] 삭제되는 열 바로 뒤에 인접한 빈 텍스트 노드도 함께 삭제하여 DOM 트리 일관성 유지
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if(htBrowser.ie && (htBrowser.nativeVersion == 9 || htBrowser.nativeVersion == 10) && (htBrowser.version == 9 || htBrowser.version == 10)){
|
|
this._removeEmptyTextNode_IE(elTmp);
|
|
}
|
|
// --[SMARTEDITORSUS-1533]
|
|
|
|
elTmp.parentNode.removeChild(elTmp);
|
|
}
|
|
|
|
var aTR = jindo.$$(">TBODY>TR", this.elSelectionStartTable, {oneTimeOffCache:true});
|
|
var nSelectionWidth = this.htSelectionEPos.x - this.htSelectionSPos.x + 1;
|
|
var nWidth = this.htMap.length;
|
|
if(nSelectionWidth == nWidth){
|
|
for(var i = 0, nLen = aTR.length; i < nLen; i++){
|
|
elTmp = aTR[i];
|
|
|
|
// There can be empty but necessary TR's because of Rowspan
|
|
if(!this.htMap[0][i] || !this.htMap[0][i].parentNode || this.htMap[0][i].parentNode.tagName !== "TR"){
|
|
// [SMARTEDITORSUS-1533] 삭제되는 행 바로 뒤에 인접한 빈 텍스트 노드도 함께 삭제하여 DOM 트리 일관성 유지
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if(htBrowser.ie && (htBrowser.nativeVersion == 9 || htBrowser.nativeVersion == 10) && (htBrowser.version == 9 || htBrowser.version == 10)){
|
|
this._removeEmptyTextNode_IE(elTmp);
|
|
}
|
|
// --[SMARTEDITORSUS-1533]
|
|
|
|
elTmp.parentNode.removeChild(elTmp);
|
|
}
|
|
}
|
|
|
|
aTR = jindo.$$(">TBODY>TR", this.elSelectionStartTable, {oneTimeOffCache:true});
|
|
}
|
|
|
|
if(aTR.length < 1){
|
|
this.elSelectionStartTable.parentNode.removeChild(this.elSelectionStartTable);
|
|
}
|
|
|
|
this._updateSelection();
|
|
},
|
|
|
|
_insertColumnAfter : function(){
|
|
this._removeClassFromSelection();
|
|
this._hideTableTemplate(this.elSelectionStartTable);
|
|
|
|
var aTR = jindo.$$(">TBODY>TR", this.elSelectionStartTable, {oneTimeOffCache:true});
|
|
var sInserted;
|
|
var sTmpAttr_Inserted = "_tmp_inserted";
|
|
var elCell, elCellClone, elCurTR, elInsertionPt;
|
|
// copy each cells in the following order: top->down, right->left
|
|
// +---+---+---+---+
|
|
// |...|.2.|.1.|...|
|
|
// |---+---+.1.|...|
|
|
// |...|.3.|.1.|...|
|
|
// |...|.3.+---+...|
|
|
// |...|.3.|.4.+...|
|
|
// |...+---+---+...|
|
|
// |...|.6.|.5.|...|
|
|
// +---+---+---+---+
|
|
|
|
// [SMARTEDITORSUS-991] IE는 insertionPt의 previousSibling에도 배경색을 적용해줘야 할 필요가 있음.
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
// --[SMARTEDITORSUS-991]
|
|
|
|
for(var y = 0, nYLen = this.htMap[0].length; y < nYLen; y++){
|
|
elCurTR = aTR[y];
|
|
for(var x = this.htSelectionEPos.x; x >= this.htSelectionSPos.x; x--){
|
|
elCell = this.htMap[x][y];
|
|
//sInserted = elCell.getAttribute(sTmpAttr_Inserted);
|
|
//if(sInserted){continue;}
|
|
|
|
//elCell.setAttribute(sTmpAttr_Inserted, "o");
|
|
elCellClone = this._shallowCloneTD(elCell);
|
|
|
|
// elCellClone의 outerHTML에 정상적인 rowSpan이 있더라도 IE에서는 이 위치에서 항상 1을 반환. (elCellClone.rowSpan & elCellClone.getAttribute("rowSpan")).
|
|
//var nSpan = parseInt(elCellClone.getAttribute("rowSpan"));
|
|
var nSpan = parseInt(elCell.getAttribute("rowSpan"));
|
|
|
|
if(nSpan > 1){
|
|
elCellClone.setAttribute("rowSpan", 1);
|
|
elCellClone.style.height = "";
|
|
}
|
|
nSpan = parseInt(elCell.getAttribute("colSpan"));
|
|
|
|
if(nSpan > 1){
|
|
elCellClone.setAttribute("colSpan", 1);
|
|
elCellClone.style.width = "";
|
|
}
|
|
|
|
// 현재 줄(TR)에 속한 셀(TD)을 찾아서 그 앞에 append 한다.
|
|
elInsertionPt = null;
|
|
for(var xx = this.htSelectionEPos.x; xx >= this.htSelectionSPos.x; xx--){
|
|
if(this.htMap[xx][y].parentNode == elCurTR){
|
|
elInsertionPt = this.htMap[xx][y].nextSibling;
|
|
break;
|
|
}
|
|
}
|
|
elCurTR.insertBefore(elCellClone, elInsertionPt);
|
|
|
|
// [SMARTEDITORSUS-1742][SMARTEDITORSUS-1842] 배경색이 바로 반영되지 않는 버그로 인해 명시
|
|
// [SMARTEDITORSUS-2155] Win10 Spartan
|
|
if((htBrowser.edge && (Math.floor(htBrowser.version) === 12)) || (htBrowser.ie && (htBrowser.nativeVersion >= 9 || htBrowser.nativeVersion <= 11) && (htBrowser.version >= 9 || htBrowser.version <= 11))){
|
|
elCellClone.style.cssText = elCellClone.style.cssText;
|
|
}
|
|
// --[SMARTEDITORSUS-1742][SMARTEDITORSUS-1842][SMARTEDITORSUS-2155]
|
|
}
|
|
}
|
|
// remove the insertion marker from the original cells
|
|
for(var i = 0, nLen = this.aSelectedCells.length; i < nLen; i++){
|
|
this.aSelectedCells[i].removeAttribute(sTmpAttr_Inserted);
|
|
}
|
|
|
|
var nSelectionWidth = this.htSelectionEPos.x - this.htSelectionSPos.x + 1;
|
|
var nSelectionHeight = this.htSelectionEPos.y - this.htSelectionSPos.y + 1;
|
|
this.htSelectionSPos.x += nSelectionWidth;
|
|
this.htSelectionEPos.x += nSelectionWidth;
|
|
|
|
this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
this._showTableTemplate(this.elSelectionStartTable);
|
|
this._addClassToSelection();
|
|
|
|
this.sQEAction = "INSERT_TABLE_COLUMN";
|
|
|
|
this.oApp.exec("SHOW_COMMON_QE");
|
|
},
|
|
|
|
_insertRowBelow : function(){
|
|
this._selectAll_Row();
|
|
|
|
this._removeClassFromSelection();
|
|
this._hideTableTemplate(this.elSelectionStartTable);
|
|
|
|
var elRowClone;
|
|
var elTBody = this.htMap[0][0].parentNode.parentNode;
|
|
|
|
var aTRs = jindo.$$(">TR", elTBody, {oneTimeOffCache:true});
|
|
var elInsertionPt = aTRs[this.htSelectionEPos.y + 1] || null;
|
|
|
|
// [SMARTEDITORSUS-991] IE는 insertionPt의 previousSibling에도 배경색을 적용해줘야 할 필요가 있음.
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
// --[SMARTEDITORSUS-991]
|
|
|
|
for(var y = this.htSelectionSPos.y; y <= this.htSelectionEPos.y; y++){
|
|
elRowClone = this._getTRCloneWithAllTD(y);
|
|
elTBody.insertBefore(elRowClone, elInsertionPt);
|
|
|
|
// [SMARTEDITORSUS-991] IE는 insertionPt의 previousSibling에도 추가로 배경색을 적용해줘야 할 필요가 있음.
|
|
//if(htBrowser.ie && htBrowser.nativeVersion >= 9){
|
|
// [SMARTEDITORSUS-1533] 문서모드가 9~10인 IE 9~10에서 빈 텍스트 노드가 인식된다.
|
|
// [SMARTEDITORSUS-1842]
|
|
// [SMARTEDITORSUS-2155] Win10 Spartan
|
|
if((htBrowser.edge && (Math.floor(htBrowser.version) === 12)) || (htBrowser.ie && (htBrowser.nativeVersion >= 9 || htBrowser.nativeVersion <= 11) && (htBrowser.version >= 9 || htBrowser.version <= 11))){
|
|
// --[SMARTEDITORSUS-1842][SMARTEDITORSUS-2155]
|
|
// 스타일을 적용시켜 줄 대상 tr
|
|
var elPreviousSiblingParent = this.htMap[0][y].parentNode;
|
|
|
|
var aOriginalPreviousSibling = elPreviousSiblingParent.childNodes;
|
|
var aPreviousSibling = [];
|
|
for(var i = 0, len = aOriginalPreviousSibling.length; i < len; i++){
|
|
// [SMARTEDITORSUS-1742]
|
|
//aPreviousSibling.push(aOriginalPreviousSibling[i]);
|
|
if(this._rxCellNames.test(aOriginalPreviousSibling[i].nodeName)){
|
|
aPreviousSibling.push(aOriginalPreviousSibling[i].cloneNode());
|
|
}
|
|
// --[SMARTEDITORSUS-1742]
|
|
}
|
|
|
|
// 배경색을 복사하기 위해 준비
|
|
var aCellClone = elRowClone.childNodes;
|
|
|
|
for(var i = 0, len = aCellClone.length; i < len; i++){
|
|
var elCloneTD = aCellClone[i];
|
|
|
|
var elPreviousTD = aPreviousSibling[i];
|
|
// [SMARTEDITORSUS-1639] 병합 후 추가시 JS 오류가 발생하는 문제가 있어 nodeName 확인
|
|
// [SMARTEDITORSUS-1672]
|
|
//if(elCloneTD.nodeName == "TD" && elPreviousTD && elPreviousTD.nodeName == "TD"){
|
|
if(this._rxCellNames.test(elCloneTD.nodeName) && elPreviousTD && this._rxCellNames.test(elPreviousTD.nodeName)){
|
|
// --[SMARTEDITORSUS-1672]
|
|
// [SMARTEDITORSUS-1742]
|
|
//elPreviousTD.style.backgroundColor = elCloneTD.style.backgroundColor;
|
|
elCloneTD.style.cssText = elPreviousTD.style.cssText;
|
|
// --[SMARTEDITORSUS-1742]
|
|
}
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-991]
|
|
}
|
|
|
|
var nSelectionWidth = this.htSelectionEPos.x - this.htSelectionSPos.x + 1;
|
|
var nSelectionHeight = this.htSelectionEPos.y - this.htSelectionSPos.y + 1;
|
|
this.htSelectionSPos.y += nSelectionHeight;
|
|
this.htSelectionEPos.y += nSelectionHeight;
|
|
|
|
this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
this._selectCells(this.htSelectionSPos, this.htSelectionEPos);
|
|
|
|
this._showTableTemplate(this.elSelectionStartTable);
|
|
this._addClassToSelection();
|
|
|
|
this.sQEAction = "INSERT_TABLE_ROW";
|
|
|
|
this.oApp.exec("SHOW_COMMON_QE");
|
|
},
|
|
|
|
_updateSelection : function(){
|
|
this.aSelectedCells = jindo.$A(this.aSelectedCells).filter(function(v){return (v.parentNode!==null && v.parentNode.parentNode!==null);}).$value();
|
|
},
|
|
|
|
_startCellSelection : function(){
|
|
this.htMap = this._getCellMapping(this.elSelectionStartTable);
|
|
|
|
// De-select the default selection
|
|
this.oApp.getEmptySelection().oBrowserSelection.selectNone();
|
|
|
|
if(this.oApp.oNavigator["safari"] || this.oApp.oNavigator["chrome"]){
|
|
this.oApp.getWYSIWYGDocument().onselectstart = function(){return false;};
|
|
}
|
|
|
|
var elIFrame = this.oApp.getWYSIWYGWindow().frameElement;
|
|
this.htEditingAreaPos = jindo.$Element(elIFrame).offset();
|
|
this.htEditingAreaPos.height = elIFrame.offsetHeight;
|
|
this.htEditingAreaPos.bottom = this.htEditingAreaPos.top + this.htEditingAreaPos.height;
|
|
this.htEditingAreaPos.width = elIFrame.offsetWidth;
|
|
this.htEditingAreaPos.right = this.htEditingAreaPos.left + this.htEditingAreaPos.width;
|
|
|
|
/*
|
|
if(!this.oNavigatorInfo["firefox"]){
|
|
this.oApp.exec("DISABLE_WYSIWYG", []);
|
|
}
|
|
*/
|
|
this._changeTableEditorStatus(this.STATUS.CELL_SELECTING);
|
|
},
|
|
|
|
_selectBetweenCells : function(elCell1, elCell2){
|
|
this._deselectCells();
|
|
|
|
var oP1 = this._getBasisCellPosition(elCell1);
|
|
var oP2 = this._getBasisCellPosition(elCell2);
|
|
this._setEndPos(oP1);
|
|
this._setEndPos(oP2);
|
|
|
|
var oStartPos = {}, oEndPos = {};
|
|
|
|
oStartPos.x = Math.min(oP1.x, oP1.ex, oP2.x, oP2.ex);
|
|
oStartPos.y = Math.min(oP1.y, oP1.ey, oP2.y, oP2.ey);
|
|
|
|
oEndPos.x = Math.max(oP1.x, oP1.ex, oP2.x, oP2.ex);
|
|
oEndPos.y = Math.max(oP1.y, oP1.ey, oP2.y, oP2.ey);
|
|
|
|
this._selectCells(oStartPos, oEndPos);
|
|
},
|
|
|
|
_getNextCell : function(elCell){
|
|
while(elCell){
|
|
elCell = elCell.nextSibling;
|
|
if(elCell && elCell.tagName && elCell.tagName.match(/^TD|TH$/)){return elCell;}
|
|
}
|
|
|
|
return null;
|
|
},
|
|
|
|
_getCellMapping : function(elTable){
|
|
var aTR = jindo.$$(">TBODY>TR", elTable, {oneTimeOffCache:true});
|
|
var nTD = 0;
|
|
var aTD_FirstRow = aTR[0].childNodes;
|
|
/*
|
|
// remove empty TR's from the bottom of the table
|
|
for(var i=aTR.length-1; i>0; i--){
|
|
if(!aTR[i].childNodes || aTR[i].childNodes.length === 0){
|
|
aTR[i].parentNode.removeChild(aTR[i]);
|
|
aTR = aTR.slice(0, i);
|
|
|
|
if(this.htSelectionSPos.y>=i) this.htSelectionSPos.y--;
|
|
if(this.htSelectionEPos.y>=i) this.htSelectionEPos.y--;
|
|
}else{
|
|
break;
|
|
}
|
|
}
|
|
*/
|
|
// count the number of columns
|
|
for(var i = 0; i < aTD_FirstRow.length; i++){
|
|
var elTmp = aTD_FirstRow[i];
|
|
|
|
if(!elTmp.tagName || !elTmp.tagName.match(/^TD|TH$/)){continue;}
|
|
|
|
if(elTmp.getAttribute("colSpan")){
|
|
nTD += this.parseIntOr0(elTmp.getAttribute("colSpan"));
|
|
}else{
|
|
nTD ++;
|
|
}
|
|
}
|
|
|
|
var nTblX = nTD;
|
|
var nTblY = aTR.length;
|
|
|
|
var aCellMapping = new Array(nTblX);
|
|
for(var x = 0; x < nTblX; x++){
|
|
aCellMapping[x] = new Array(nTblY);
|
|
}
|
|
|
|
for(var y = 0; y < nTblY; y++){
|
|
var elCell = aTR[y].childNodes[0];
|
|
|
|
if(!elCell){continue;}
|
|
if(!elCell.tagName || !elCell.tagName.match(/^TD|TH$/)){elCell = this._getNextCell(elCell);}
|
|
|
|
var x = -1;
|
|
while(elCell){
|
|
x++;
|
|
if(!aCellMapping[x]){aCellMapping[x] = [];}
|
|
if(aCellMapping[x][y]){continue;}
|
|
var colSpan = parseInt(elCell.getAttribute("colSpan"), 10) || 1;
|
|
var rowSpan = parseInt(elCell.getAttribute("rowSpan"), 10) || 1;
|
|
/*
|
|
if(y+rowSpan >= nTblY){
|
|
rowSpan = nTblY-y;
|
|
elCell.setAttribute("rowSpan", rowSpan);
|
|
}
|
|
*/
|
|
for(var yy = 0; yy < rowSpan; yy++){
|
|
for(var xx = 0; xx < colSpan; xx++){
|
|
if(!aCellMapping[x+xx]){
|
|
aCellMapping[x+xx] = [];
|
|
}
|
|
aCellMapping[x+xx][y+yy] = elCell;
|
|
}
|
|
}
|
|
|
|
elCell = this._getNextCell(elCell);
|
|
}
|
|
}
|
|
|
|
// remove empty TR's
|
|
// (상단 TD의 rowspan만으로 지탱되는) 빈 TR이 있을 경우 IE7 이하에서 랜더링 오류가 발생 할 수 있어 빈 TR을 지워 줌
|
|
var bRowRemoved = false;
|
|
var elLastCell = null;
|
|
for(var y = 0, nRealY = 0, nYLen = aCellMapping[0].length; y < nYLen; y++, nRealY++){
|
|
elLastCell = null;
|
|
if(!aTR[y].innerHTML.match(/TD|TH/i)){
|
|
for(var x = 0, nXLen = aCellMapping.length; x < nXLen; x++){
|
|
elCell = aCellMapping[x][y];
|
|
if(!elCell || elCell === elLastCell){
|
|
continue;
|
|
}
|
|
elLastCell = elCell;
|
|
var rowSpan = parseInt(elCell.getAttribute("rowSpan"), 10) || 1;
|
|
|
|
if(rowSpan > 1){
|
|
elCell.setAttribute("rowSpan", rowSpan - 1);
|
|
}
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1533]
|
|
var htBrowser = jindo.$Agent().navigator();
|
|
if(htBrowser.ie && (htBrowser.nativeVersion == 9 || htBrowser.nativeVersion == 10) && (htBrowser.version == 9 || htBrowser.version == 10)){
|
|
this._removeEmptyTextNode_IE(aTR[y]);
|
|
}
|
|
// --[SMARTEDITORSUS-1533]
|
|
|
|
aTR[y].parentNode.removeChild(aTR[y]);
|
|
|
|
if(this.htSelectionEPos.y >= nRealY){
|
|
nRealY--;
|
|
this.htSelectionEPos.y--;
|
|
}
|
|
|
|
bRowRemoved = true;
|
|
}
|
|
}
|
|
if(bRowRemoved){
|
|
return this._getCellMapping(elTable);
|
|
}
|
|
|
|
return aCellMapping;
|
|
},
|
|
|
|
_selectCells : function(htSPos, htEPos){
|
|
this.aSelectedCells = this._getSelectedCells(htSPos, htEPos);
|
|
this._addClassToSelection();
|
|
},
|
|
|
|
_deselectCells : function(){
|
|
this._removeClassFromSelection();
|
|
|
|
this.aSelectedCells = [];
|
|
this.htSelectionSPos = {x:-1, y:-1};
|
|
this.htSelectionEPos = {x:-1, y:-1};
|
|
},
|
|
|
|
_addClassToSelection : function(){
|
|
var welCell, elCell;
|
|
for(var i = 0; i < this.aSelectedCells.length; i++){
|
|
elCell = this.aSelectedCells[i];
|
|
|
|
// [SMARTEDITORSUS-1552] 드래그로 셀을 선택하는 중 elCell이 없는 경우 오류 발생
|
|
if(elCell){
|
|
// [SMARTEDITORSUS-1498][SMARTEDITORSUS-1549] 선택된 모든 셀에서 드래그가 발생하지 못하게 방지(FF, Chrome)
|
|
if(elCell.ondragstart == null){
|
|
elCell.ondragstart = function(){
|
|
return false;
|
|
};
|
|
}
|
|
// --[SMARTEDITORSUS-1498][SMARTEDITORSUS-1549]
|
|
|
|
welCell = jindo.$Element(elCell);
|
|
welCell.addClass(this.CELL_SELECTION_CLASS);
|
|
|
|
// [SMARTEDITORSUS-1498][SMARTEDITORSUS-1549] 선택된 모든 셀에서 드래그가 발생하지 못하게 방지(FF, Chrome)
|
|
welCell.addClass("undraggable");
|
|
// --[SMARTEDITORSUS-1498][SMARTEDITORSUS-1549]
|
|
|
|
if(elCell.style.backgroundColor){
|
|
elCell.setAttribute(this.TMP_BGC_ATTR, elCell.style.backgroundColor);
|
|
welCell.css("backgroundColor", "");
|
|
}
|
|
|
|
if(elCell.style.backgroundImage) {
|
|
elCell.setAttribute(this.TMP_BGIMG_ATTR, elCell.style.backgroundImage);
|
|
welCell.css("backgroundImage", "");
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1552]
|
|
}
|
|
},
|
|
|
|
_removeClassFromSelection : function(){
|
|
var welCell, elCell;
|
|
|
|
for(var i = 0; i < this.aSelectedCells.length; i++){
|
|
elCell = this.aSelectedCells[i];
|
|
|
|
// [SMARTEDITORSUS-1552] 드래그로 셀을 선택하는 중 elCell이 없는 경우 오류 발생
|
|
if(elCell){
|
|
welCell = jindo.$Element(elCell);
|
|
welCell.removeClass(this.CELL_SELECTION_CLASS);
|
|
|
|
// [SMARTEDITORSUS-1498][SMARTEDITORSUS-1549] 선택된 모든 셀에서 드래그가 발생하지 못하게 방지(FF, Chrome)
|
|
welCell.removeClass("undraggable");
|
|
// --[SMARTEDITORSUS-1498][SMARTEDITORSUS-1549]
|
|
|
|
//배경색
|
|
if(elCell.getAttribute(this.TMP_BGC_ATTR)){
|
|
elCell.style.backgroundColor = elCell.getAttribute(this.TMP_BGC_ATTR);
|
|
elCell.removeAttribute(this.TMP_BGC_ATTR);
|
|
}
|
|
//배경이미지
|
|
if(elCell.getAttribute(this.TMP_BGIMG_ATTR)) {
|
|
welCell.css("backgroundImage",elCell.getAttribute(this.TMP_BGIMG_ATTR));
|
|
elCell.removeAttribute(this.TMP_BGIMG_ATTR);
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1552]
|
|
}
|
|
},
|
|
|
|
_expandAndSelect : function(htPos1, htPos2){
|
|
var x, y, elTD, nTmp, i;
|
|
|
|
// expand top
|
|
if(htPos1.y > 0){
|
|
for(x = htPos1.x; x <= htPos2.x; x++){
|
|
elTD = this.htMap[x][htPos1.y];
|
|
if(this.htMap[x][htPos1.y - 1] == elTD){
|
|
nTmp = htPos1.y - 2;
|
|
while(nTmp >= 0 && this.htMap[x][nTmp] == elTD){
|
|
nTmp--;
|
|
}
|
|
htPos1.y = nTmp + 1;
|
|
this._expandAndSelect(htPos1, htPos2);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// expand left
|
|
if(htPos1.x > 0){
|
|
for(y = htPos1.y; y <= htPos2.y; y++){
|
|
elTD = this.htMap[htPos1.x][y];
|
|
if(this.htMap[htPos1.x - 1][y] == elTD){
|
|
nTmp = htPos1.x - 2;
|
|
while(nTmp >= 0 && this.htMap[nTmp][y] == elTD){
|
|
nTmp--;
|
|
}
|
|
htPos1.x = nTmp + 1;
|
|
this._expandAndSelect(htPos1, htPos2);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// expand bottom
|
|
if(htPos2.y < this.htMap[0].length - 1){
|
|
for(x = htPos1.x; x <= htPos2.x; x++){
|
|
elTD = this.htMap[x][htPos2.y];
|
|
if(this.htMap[x][htPos2.y + 1] == elTD){
|
|
nTmp = htPos2.y + 2;
|
|
while(nTmp < this.htMap[0].length && this.htMap[x][nTmp] == elTD){
|
|
nTmp++;
|
|
}
|
|
htPos2.y = nTmp - 1;
|
|
this._expandAndSelect(htPos1, htPos2);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// expand right
|
|
if(htPos2.x < this.htMap.length - 1){
|
|
for(y = htPos1.y; y <= htPos2.y; y++){
|
|
elTD = this.htMap[htPos2.x][y];
|
|
if(this.htMap[htPos2.x + 1][y] == elTD){
|
|
nTmp = htPos2.x + 2;
|
|
while(nTmp < this.htMap.length && this.htMap[nTmp][y] == elTD){
|
|
nTmp++;
|
|
}
|
|
htPos2.x = nTmp - 1;
|
|
this._expandAndSelect(htPos1, htPos2);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
_getSelectedCells : function(htPos1, htPos2){
|
|
this._expandAndSelect(htPos1, htPos2);
|
|
var x1 = htPos1.x;
|
|
var y1 = htPos1.y;
|
|
|
|
var x2 = htPos2.x;
|
|
var y2 = htPos2.y;
|
|
|
|
this.htSelectionSPos = htPos1;
|
|
this.htSelectionEPos = htPos2;
|
|
|
|
var aResult = [];
|
|
|
|
for(var y = y1; y <= y2; y++){
|
|
for(var x = x1; x <= x2; x++){
|
|
if(jindo.$A(aResult).has(this.htMap[x][y])){
|
|
continue;
|
|
}
|
|
aResult[aResult.length] = this.htMap[x][y];
|
|
}
|
|
}
|
|
return aResult;
|
|
},
|
|
|
|
_setEndPos : function(htPos){
|
|
var nColspan, nRowspan;
|
|
|
|
nColspan = parseInt(htPos.elCell.getAttribute("colSpan"), 10) || 1;
|
|
nRowspan = parseInt(htPos.elCell.getAttribute("rowSpan"), 10) || 1;
|
|
htPos.ex = htPos.x + nColspan - 1;
|
|
htPos.ey = htPos.y + nRowspan - 1;
|
|
},
|
|
|
|
_getBasisCellPosition : function(elCell){
|
|
var x = 0, y = 0;
|
|
for(x = 0; x < this.htMap.length; x++){
|
|
for(y = 0; y < this.htMap[x].length; y++){
|
|
if(this.htMap[x][y] == elCell){
|
|
return {'x': x, 'y': y, elCell: elCell};
|
|
}
|
|
}
|
|
}
|
|
return {'x': 0, 'y': 0, elCell: elCell};
|
|
},
|
|
|
|
_applyTableTemplate : function(elTable, nTemplateIdx){
|
|
// clear style first if already exists
|
|
/*
|
|
if(elTable.getAttribute(this.ATTR_TBL_TEMPLATE)){
|
|
this._doApplyTableTemplate(elTable, nhn.husky.SE2M_TableTemplate[this.parseIntOr0(elTable.getAttribute(this.ATTR_TBL_TEMPLATE))], true);
|
|
}else{
|
|
this._clearAllTableStyles(elTable);
|
|
}
|
|
*/
|
|
|
|
if (!elTable) {
|
|
return;
|
|
}
|
|
|
|
// 사용자가 지정한 스타일 무시하고 새 템플릿 적용
|
|
// http://bts.nhncorp.com/nhnbts/browse/COM-871
|
|
this._clearAllTableStyles(elTable);
|
|
|
|
this._doApplyTableTemplate(elTable, nhn.husky.SE2M_TableTemplate[nTemplateIdx], false);
|
|
elTable.setAttribute(this.ATTR_TBL_TEMPLATE, nTemplateIdx);
|
|
},
|
|
|
|
_clearAllTableStyles : function(elTable){
|
|
elTable.removeAttribute("border");
|
|
elTable.removeAttribute("cellPadding");
|
|
elTable.removeAttribute("cellSpacing");
|
|
elTable.style.padding = "";
|
|
elTable.style.border = "";
|
|
elTable.style.backgroundColor = "";
|
|
elTable.style.color = "";
|
|
|
|
var aTD = jindo.$$(">TBODY>TR>TD", elTable, {oneTimeOffCache:true});
|
|
for(var i = 0, nLen = aTD.length; i < nLen; i++){
|
|
aTD[i].style.padding = "";
|
|
aTD[i].style.border = "";
|
|
aTD[i].style.backgroundColor = "";
|
|
aTD[i].style.color = "";
|
|
}
|
|
// [SMARTEDITORSUS-1672]
|
|
var aTH = jindo.$$(">TBODY>TR>TH", elTable, {oneTimeOffCache:true});
|
|
for(var i = 0, nLen = aTH.length; i < nLen; i++){
|
|
aTH[i].style.padding = "";
|
|
aTH[i].style.border = "";
|
|
aTH[i].style.backgroundColor = "";
|
|
aTH[i].style.color = "";
|
|
}
|
|
// --[SMARTEDITORSUS-1672]
|
|
},
|
|
|
|
_hideTableTemplate : function(elTable){
|
|
if(elTable.getAttribute(this.ATTR_TBL_TEMPLATE)){
|
|
this._doApplyTableTemplate(elTable, nhn.husky.SE2M_TableTemplate[this.parseIntOr0(elTable.getAttribute(this.ATTR_TBL_TEMPLATE))], true);
|
|
}
|
|
},
|
|
|
|
_showTableTemplate : function(elTable){
|
|
if(elTable.getAttribute(this.ATTR_TBL_TEMPLATE)){
|
|
this._doApplyTableTemplate(elTable, nhn.husky.SE2M_TableTemplate[this.parseIntOr0(elTable.getAttribute(this.ATTR_TBL_TEMPLATE))], false);
|
|
}
|
|
},
|
|
|
|
_doApplyTableTemplate : function(elTable, htTableTemplate, bClearStyle){
|
|
var htTableProperty = htTableTemplate.htTableProperty;
|
|
var htTableStyle = htTableTemplate.htTableStyle;
|
|
var ht1stRowStyle = htTableTemplate.ht1stRowStyle;
|
|
var ht1stColumnStyle = htTableTemplate.ht1stColumnStyle;
|
|
var aRowStyle = htTableTemplate.aRowStyle;
|
|
var elTmp;
|
|
|
|
// replace all TH's with TD's
|
|
|
|
if(htTableProperty){
|
|
this._copyAttributesTo(elTable, htTableProperty, bClearStyle);
|
|
}
|
|
if(htTableStyle){
|
|
this._copyStylesTo(elTable, htTableStyle, bClearStyle);
|
|
}
|
|
|
|
var aTR = jindo.$$(">TBODY>TR", elTable, {oneTimeOffCache:true});
|
|
|
|
var nStartRowNum = 0;
|
|
if(ht1stRowStyle){
|
|
var nStartRowNum = 1;
|
|
|
|
for(var ii = 0, nNumCells = aTR[0].childNodes.length; ii < nNumCells; ii++){
|
|
elTmp = aTR[0].childNodes[ii];
|
|
if(!elTmp.tagName || !elTmp.tagName.match(/^TD|TH$/)){continue;}
|
|
this._copyStylesTo(elTmp, ht1stRowStyle, bClearStyle);
|
|
}
|
|
}
|
|
|
|
var nRowSpan;
|
|
var elFirstEl;
|
|
if(ht1stColumnStyle){
|
|
// if the style's got a row heading, skip the 1st row. (it was taken care above)
|
|
var nRowStart = ht1stRowStyle ? 1 : 0;
|
|
|
|
for(var i = nRowStart, nLen = aTR.length; i < nLen;){
|
|
elFirstEl = aTR[i].firstChild;
|
|
|
|
nRowSpan = 1;
|
|
|
|
if(elFirstEl && elFirstEl.tagName.match(/^TD|TH$/)){
|
|
nRowSpan = parseInt(elFirstEl.getAttribute("rowSpan"), 10) || 1;
|
|
this._copyStylesTo(elFirstEl, ht1stColumnStyle, bClearStyle);
|
|
}
|
|
|
|
i += nRowSpan;
|
|
}
|
|
}
|
|
|
|
if(aRowStyle){
|
|
var nNumStyles = aRowStyle.length;
|
|
for(var i = nStartRowNum, nLen = aTR.length; i < nLen; i++){
|
|
for(var ii = 0, nNumCells = aTR[i].childNodes.length; ii < nNumCells; ii++){
|
|
var elTmp = aTR[i].childNodes[ii];
|
|
if(!elTmp.tagName || !elTmp.tagName.match(/^TD|TH$/)){continue;}
|
|
this._copyStylesTo(elTmp, aRowStyle[(i+nStartRowNum)%nNumStyles], bClearStyle);
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
_copyAttributesTo : function(oTarget, htProperties, bClearStyle){
|
|
var elTmp;
|
|
for(var x in htProperties){
|
|
if(htProperties.hasOwnProperty(x)){
|
|
if(bClearStyle){
|
|
if(oTarget[x]){
|
|
elTmp = document.createElement(oTarget.tagName);
|
|
elTmp[x] = htProperties[x];
|
|
if(elTmp[x] == oTarget[x]){
|
|
oTarget.removeAttribute(x);
|
|
}
|
|
}
|
|
}else{
|
|
elTmp = document.createElement(oTarget.tagName);
|
|
elTmp.style[x] = "";
|
|
if(!oTarget[x] || oTarget.style[x] == elTmp.style[x]){oTarget.setAttribute(x, htProperties[x]);}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
_copyStylesTo : function(oTarget, htProperties, bClearStyle){
|
|
var elTmp;
|
|
for(var x in htProperties){
|
|
if(htProperties.hasOwnProperty(x)){
|
|
if(bClearStyle){
|
|
if(oTarget.style[x]){
|
|
elTmp = document.createElement(oTarget.tagName);
|
|
elTmp.style[x] = htProperties[x];
|
|
if(elTmp.style[x] == oTarget.style[x]){
|
|
oTarget.style[x] = "";
|
|
}
|
|
}
|
|
}else{
|
|
elTmp = document.createElement(oTarget.tagName);
|
|
elTmp.style[x] = "";
|
|
if(!oTarget.style[x] || oTarget.style[x] == elTmp.style[x] || x.match(/^border/)){oTarget.style[x] = htProperties[x];}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
_hideResizer : function(){
|
|
this.elResizeGrid.style.display = "none";
|
|
},
|
|
|
|
_showResizer : function(){
|
|
this.elResizeGrid.style.display = "block";
|
|
},
|
|
|
|
_setResizerSize : function(width, height){
|
|
this.elResizeGrid.style.width = width + "px";
|
|
this.elResizeGrid.style.height = height + "px";
|
|
},
|
|
|
|
parseBorder : function(vBorder, sBorderStyle){
|
|
if(sBorderStyle == "none"){return 0;}
|
|
|
|
var num = parseInt(vBorder, 10);
|
|
if(isNaN(num)){
|
|
if(typeof(vBorder) == "string"){
|
|
// IE Bug
|
|
return 1;
|
|
/*
|
|
switch(vBorder){
|
|
case "thin":
|
|
return 1;
|
|
case "medium":
|
|
return 3;
|
|
case "thick":
|
|
return 5;
|
|
}
|
|
*/
|
|
}
|
|
}
|
|
return num;
|
|
},
|
|
|
|
parseIntOr0 : function(num){
|
|
num = parseInt(num, 10);
|
|
if(isNaN(num)){return 0;}
|
|
return num;
|
|
},
|
|
|
|
_getTRCloneWithAllTD : function(nRow){
|
|
var elResult = this.htMap[0][nRow].parentNode.cloneNode(false);
|
|
|
|
var elCurTD, elCurTDClone;
|
|
for(var i = 0, nLen = this.htMap.length; i < nLen; i++){
|
|
elCurTD = this.htMap[i][nRow];
|
|
// [SMARTEDITORSUS-1672]
|
|
//if(elCurTD.tagName == "TD"){
|
|
if(this._rxCellNames.test(elCurTD.tagName)){
|
|
// --[SMARTEDITORSUS-1672]
|
|
elCurTDClone = this._shallowCloneTD(elCurTD);
|
|
elCurTDClone.setAttribute("rowSpan", 1);
|
|
elCurTDClone.setAttribute("colSpan", 1);
|
|
elCurTDClone.style.width = "";
|
|
elCurTDClone.style.height = "";
|
|
elResult.insertBefore(elCurTDClone, null);
|
|
}
|
|
}
|
|
|
|
return elResult;
|
|
},
|
|
|
|
_shallowCloneTD : function(elTD){
|
|
var elResult = elTD.cloneNode(false);
|
|
|
|
elResult.innerHTML = this.sEmptyTDSrc;
|
|
|
|
return elResult;
|
|
},
|
|
|
|
// elTbl이 꽉 찬 직사각형 형태의 테이블인지 확인
|
|
_isValidTable : function(elTbl){
|
|
if(!elTbl || !elTbl.tagName || elTbl.tagName != "TABLE"){
|
|
return false;
|
|
}
|
|
|
|
this.htMap = this._getCellMapping(elTbl);
|
|
var nXSize = this.htMap.length;
|
|
if(nXSize < 1){return false;}
|
|
|
|
var nYSize = this.htMap[0].length;
|
|
if(nYSize < 1){return false;}
|
|
|
|
for(var i = 1; i < nXSize; i++){
|
|
// 첫번째 열과 길이가 다른 열이 하나라도 있다면 직사각형이 아님
|
|
if(this.htMap[i].length != nYSize || !this.htMap[i][nYSize - 1]){
|
|
return false;
|
|
}
|
|
|
|
// 빈칸이 하나라도 있다면 꽉 찬 직사각형이 아님
|
|
for(var j = 0; j < nYSize; j++){
|
|
if(!this.htMap[i] || !this.htMap[i][j]){
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
},
|
|
|
|
addCSSClass : function(sClassName, sClassRule){
|
|
var oDoc = this.oApp.getWYSIWYGDocument();
|
|
if(oDoc.styleSheets[0] && oDoc.styleSheets[0].addRule){
|
|
// IE
|
|
oDoc.styleSheets[0].addRule("." + sClassName, sClassRule);
|
|
}else{
|
|
// FF
|
|
var elHead = oDoc.getElementsByTagName("HEAD")[0];
|
|
var elStyle = oDoc.createElement ("STYLE");
|
|
//styleElement.type = "text / css";
|
|
elHead.appendChild (elStyle);
|
|
|
|
elStyle.sheet.insertRule("." + sClassName + " { "+sClassRule+" }", 0);
|
|
}
|
|
},
|
|
|
|
// [SMARTEDITORSUS-1533]
|
|
_removeEmptyTextNode_IE : function(elPair){
|
|
var elEmptyTextNode = elPair.nextSibling;
|
|
if(elEmptyTextNode && elEmptyTextNode.nodeType == 3 && !/\S/.test(elEmptyTextNode.nodeValue)){
|
|
elPair.parentNode.removeChild(elEmptyTextNode);
|
|
}
|
|
}
|
|
// --[SMARTEDITORSUS-1533]
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_BGColor$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_BGColor
|
|
* this.oApp.registerLazyMessage(["APPLY_LAST_USED_BGCOLOR", "TOGGLE_BGCOLOR_LAYER"], ["hp_SE2M_BGColor$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_BGColor, {
|
|
//@lazyload_js APPLY_LAST_USED_BGCOLOR,TOGGLE_BGCOLOR_LAYER[
|
|
$ON_TOGGLE_BGCOLOR_LAYER : function(){
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.elDropdownLayer, null, "BGCOLOR_LAYER_SHOWN", [], "BGCOLOR_LAYER_HIDDEN", []]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['bgcolor']);
|
|
},
|
|
|
|
$ON_BGCOLOR_LAYER_SHOWN : function(){
|
|
this.oApp.exec("SELECT_UI", ["BGColorB"]);
|
|
this.oApp.exec("SHOW_COLOR_PALETTE", ["APPLY_BGCOLOR", this.elPaletteHolder]);
|
|
},
|
|
|
|
$ON_BGCOLOR_LAYER_HIDDEN : function(){
|
|
this.oApp.exec("DESELECT_UI", ["BGColorB"]);
|
|
this.oApp.exec("RESET_COLOR_PALETTE", []);
|
|
},
|
|
|
|
$ON_EVENT_APPLY_BGCOLOR : function(weEvent){
|
|
var elButton = weEvent.element;
|
|
|
|
// Safari/Chrome/Opera may capture the event on Span
|
|
while(elButton.tagName == "SPAN"){elButton = elButton.parentNode;}
|
|
if(elButton.tagName != "BUTTON"){return;}
|
|
|
|
var sBGColor, sFontColor;
|
|
|
|
sBGColor = elButton.style.backgroundColor;
|
|
sFontColor = elButton.style.color;
|
|
|
|
this.oApp.exec("APPLY_BGCOLOR", [sBGColor, sFontColor]);
|
|
},
|
|
|
|
$ON_APPLY_LAST_USED_BGCOLOR : function(){
|
|
this.oApp.exec("APPLY_BGCOLOR", [this.sLastUsedColor]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['bgcolor']);
|
|
},
|
|
|
|
$ON_APPLY_BGCOLOR : function(sBGColor, sFontColor){
|
|
if(!this.rxColorPattern.test(sBGColor)){
|
|
alert(this.oApp.$MSG("SE_Color.invalidColorCode"));
|
|
return;
|
|
}
|
|
this._setLastUsedBGColor(sBGColor);
|
|
|
|
var oStyle = {"backgroundColor": sBGColor};
|
|
if(sFontColor){oStyle.color = sFontColor;}
|
|
|
|
this.oApp.exec("SET_WYSIWYG_STYLE", [oStyle]);
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_FontColor$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_FontColor
|
|
* this.oApp.registerLazyMessage(["APPLY_LAST_USED_FONTCOLOR", "TOGGLE_FONTCOLOR_LAYER"], ["hp_SE2M_FontColor$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_FontColor, {
|
|
//@lazyload_js APPLY_LAST_USED_FONTCOLOR,TOGGLE_FONTCOLOR_LAYER[
|
|
$ON_TOGGLE_FONTCOLOR_LAYER : function(){
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.elDropdownLayer, null, "FONTCOLOR_LAYER_SHOWN", [], "FONTCOLOR_LAYER_HIDDEN", []]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['fontcolor']);
|
|
},
|
|
|
|
$ON_FONTCOLOR_LAYER_SHOWN : function(){
|
|
this.oApp.exec("SELECT_UI", ["fontColorB"]);
|
|
this.oApp.exec("SHOW_COLOR_PALETTE", ["APPLY_FONTCOLOR", this.elPaletteHolder]);
|
|
},
|
|
|
|
$ON_FONTCOLOR_LAYER_HIDDEN : function(){
|
|
this.oApp.exec("DESELECT_UI", ["fontColorB"]);
|
|
this.oApp.exec("RESET_COLOR_PALETTE", []);
|
|
},
|
|
|
|
$ON_APPLY_LAST_USED_FONTCOLOR : function(){
|
|
this.oApp.exec("APPLY_FONTCOLOR", [this.sLastUsedColor]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['fontcolor']);
|
|
},
|
|
|
|
$ON_APPLY_FONTCOLOR : function(sFontColor){
|
|
if(!this.rxColorPattern.test(sFontColor)){
|
|
alert(this.oApp.$MSG("SE_FontColor.invalidColorCode"));
|
|
return;
|
|
}
|
|
|
|
this._setLastUsedFontColor(sFontColor);
|
|
|
|
this.oApp.exec("SET_WYSIWYG_STYLE", [{"color":sFontColor}]);
|
|
|
|
// [SMARTEDITORSUS-907] 모든 브라우저에서 SET_WYSIWYG_STYLE로 색상을 설정하도록 변경
|
|
// var oAgent = jindo.$Agent().navigator();
|
|
// if( oAgent.ie || oAgent.firefox ){ // [SMARTEDITORSUS-658] Firefox 추가
|
|
// this.oApp.exec("SET_WYSIWYG_STYLE", [{"color":sFontColor}]);
|
|
// } else {
|
|
// var bDontAddUndoHistory = false;
|
|
|
|
// if(this.oApp.getSelection().collapsed){
|
|
// bDontAddUndoHistory = true;
|
|
// }
|
|
|
|
// this.oApp.exec("EXECCOMMAND", ["ForeColor", false, sFontColor, { "bDontAddUndoHistory" : bDontAddUndoHistory }]);
|
|
|
|
// if(bDontAddUndoHistory){
|
|
// this.oApp.exec("RECORD_UNDO_ACTION", ["FONT COLOR", {bMustBlockElement : true}]);
|
|
// }
|
|
// }
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_Hyperlink$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_Hyperlink
|
|
* this.oApp.registerLazyMessage(["TOGGLE_HYPERLINK_LAYER", "APPLY_HYPERLINK"], ["hp_SE2M_Hyperlink$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_Hyperlink, {
|
|
//@lazyload_js TOGGLE_HYPERLINK_LAYER,APPLY_HYPERLINK[
|
|
$ON_TOGGLE_HYPERLINK_LAYER : function(){
|
|
if(!this.bLayerShown){
|
|
this.oApp.exec("IE_FOCUS", []);
|
|
this.oSelection = this.oApp.getSelection();
|
|
}
|
|
|
|
// hotkey may close the layer right away so delay here
|
|
this.oApp.delayedExec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.oHyperlinkLayer, null, "MSG_HYPERLINK_LAYER_SHOWN", [], "MSG_HYPERLINK_LAYER_HIDDEN", [""]], 0);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['hyperlink']);
|
|
},
|
|
|
|
$ON_MSG_HYPERLINK_LAYER_SHOWN : function(){
|
|
this.bLayerShown = true;
|
|
var oAnchor = this.oSelection.findAncestorByTagName("A");
|
|
|
|
if (!oAnchor) {
|
|
oAnchor = this._getSelectedNode();
|
|
}
|
|
//this.oCbNewWin.checked = false;
|
|
|
|
if(oAnchor && !this.oSelection.collapsed){
|
|
this.oSelection.selectNode(oAnchor);
|
|
this.oSelection.select();
|
|
|
|
var sTarget = oAnchor.target;
|
|
//if(sTarget && sTarget == "_blank"){this.oCbNewWin.checked = true;}
|
|
|
|
// href속성에 문제가 있을 경우, 예: href="http://na ver.com", IE에서 oAnchor.href 접근 시에 알수 없는 오류를 발생시킴
|
|
try{
|
|
var sHref = oAnchor.getAttribute("href");
|
|
this.oLinkInput.value = sHref && sHref.indexOf("#") == -1 ? sHref : "http://";
|
|
}catch(e){
|
|
this.oLinkInput.value = "http://";
|
|
}
|
|
|
|
this.bModify = true;
|
|
}else{
|
|
this.oLinkInput.value = "http://";
|
|
this.bModify = false;
|
|
}
|
|
this.oApp.delayedExec("SELECT_UI", ["hyperlink"], 0);
|
|
this.oLinkInput.focus();
|
|
|
|
this.oLinkInput.value = this.oLinkInput.value;
|
|
this.oLinkInput.select();
|
|
},
|
|
|
|
$ON_MSG_HYPERLINK_LAYER_HIDDEN : function(){
|
|
this.bLayerShown = false;
|
|
|
|
this.oApp.exec("DESELECT_UI", ["hyperlink"]);
|
|
},
|
|
|
|
_validateTarget : function() {
|
|
var oNavigator = jindo.$Agent().navigator(),
|
|
bReturn = true;
|
|
|
|
if(oNavigator.ie) {
|
|
jindo.$A(this.oSelection.getNodes(true)).forEach(function(elNode, index, array){
|
|
if(!!elNode && elNode.nodeType == 1 && elNode.tagName.toLowerCase() == "iframe" && elNode.getAttribute('s_type').toLowerCase() == "db") {
|
|
bReturn = false;
|
|
jindo.$A.Break();
|
|
}
|
|
jindo.$A.Continue();
|
|
}, this);
|
|
}
|
|
|
|
return bReturn;
|
|
},
|
|
|
|
$ON_APPLY_HYPERLINK : function(){
|
|
|
|
// [SMARTEDITORSUS-1451] 글감에 링크를 적용하지 않도록 처리
|
|
if(!this._validateTarget()){
|
|
alert(this.oApp.$MSG("SE_Hyperlink.invalidTarget"));
|
|
return;
|
|
}
|
|
|
|
var sURL = this.oLinkInput.value;
|
|
if(!/^((http|https|ftp|mailto):(?:\/\/)?)/.test(sURL)){
|
|
sURL = "http://"+sURL;
|
|
}
|
|
sURL = sURL.replace(/\s+$/, "");
|
|
|
|
var oAgent = jindo.$Agent().navigator();
|
|
var sBlank = "";
|
|
|
|
this.oApp.exec("IE_FOCUS", []);
|
|
|
|
if(oAgent.ie){sBlank = "<span style=\"text-decoration:none;\"> </span>";}
|
|
|
|
if(this._validateURL(sURL)){
|
|
//if(this.oCbNewWin.checked){
|
|
// if(false){
|
|
// sTarget = "_blank";
|
|
// }else{
|
|
sTarget = "_self";
|
|
//}
|
|
|
|
this.oApp.exec("RECORD_UNDO_BEFORE_ACTION", ["HYPERLINK", {sSaveTarget:(this.bModify ? "A" : null)}]);
|
|
|
|
var sBM;
|
|
if(this.oSelection.collapsed){
|
|
var str = "<a href='" + sURL + "' target="+sTarget+">" + sURL + "</a>" + sBlank;
|
|
this.oSelection.pasteHTML(str);
|
|
sBM = this.oSelection.placeStringBookmark();
|
|
}else{
|
|
// 브라우저에서 제공하는 execcommand에 createLink로는 타겟을 지정할 수가 없다.
|
|
// 그렇기 때문에, 더미 URL을 createLink에 넘겨서 링크를 먼저 걸고, 이후에 loop을 돌면서 더미 URL을 가진 A태그를 찾아서 정상 URL 및 타겟을 세팅 해 준다.
|
|
sBM = this.oSelection.placeStringBookmark();
|
|
this.oSelection.select();
|
|
|
|
// [SMARTEDITORSUS-61] TD 안에 있는 텍스트를 전체 선택하여 URL 변경하면 수정되지 않음 (only IE8)
|
|
// SE_EditingArea_WYSIWYG 에서는 IE인 경우, beforedeactivate 이벤트가 발생하면 현재의 Range를 저장하고, RESTORE_IE_SELECTION 메시지가 발생하면 저장된 Range를 적용한다.
|
|
// IE8 또는 IE7 호환모드이고 TD 안의 텍스트 전체를 선택한 경우 Bookmark 생성 후의 select()를 처리할 때
|
|
// HuskyRange 에서 호출되는 this._oSelection.empty(); 에서 beforedeactivate 가 발생하여 empty 처리된 selection 이 저장되는 문제가 있어 링크가 적용되지 않음.
|
|
// 올바른 selection 이 저장되어 EXECCOMMAND에서 링크가 적용될 수 있도록 함
|
|
if(oAgent.ie && (oAgent.version === 8 || oAgent.nativeVersion === 8)){ // nativeVersion 으로 IE7 호환모드인 경우 확인
|
|
this.oApp.exec("IE_FOCUS", []);
|
|
this.oSelection.moveToBookmark(sBM);
|
|
this.oSelection.select();
|
|
}
|
|
|
|
// createLink 이후에 이번에 생성된 A 태그를 찾을 수 있도록 nSession을 포함하는 더미 링크를 만든다.
|
|
var nSession = Math.ceil(Math.random()*10000);
|
|
|
|
if(sURL == ""){ // unlink
|
|
this.oApp.exec("EXECCOMMAND", ["unlink"]);
|
|
}else{ // createLink
|
|
if(this._isExceptional()){
|
|
this.oApp.exec("EXECCOMMAND", ["unlink", false, "", {bDontAddUndoHistory: true}]);
|
|
|
|
var sTempUrl = "<a href='" + sURL + "' target="+sTarget+">";
|
|
|
|
jindo.$A(this.oSelection.getNodes(true)).forEach(function(value, index, array){
|
|
var oEmptySelection = this.oApp.getEmptySelection();
|
|
|
|
if(value.nodeType === 3){
|
|
oEmptySelection.selectNode(value);
|
|
oEmptySelection.pasteHTML(sTempUrl + value.nodeValue + "</a>");
|
|
}else if(value.nodeType === 1 && value.tagName === "IMG"){
|
|
oEmptySelection.selectNode(value);
|
|
oEmptySelection.pasteHTML(sTempUrl + jindo.$Element(value).outerHTML() + "</a>");
|
|
}
|
|
}, this);
|
|
}else{
|
|
this.oApp.exec("EXECCOMMAND", ["createLink", false, this.sATagMarker+nSession+encodeURIComponent(sURL), {bDontAddUndoHistory: true}]);
|
|
}
|
|
}
|
|
|
|
var oDoc = this.oApp.getWYSIWYGDocument();
|
|
var aATags = oDoc.body.getElementsByTagName("A");
|
|
var nLen = aATags.length;
|
|
|
|
var rxMarker = new RegExp(this.sRXATagMarker+nSession, "gi");
|
|
var elATag;
|
|
|
|
for(var i=0; i<nLen; i++){
|
|
elATag = aATags[i];
|
|
|
|
var sHref = "";
|
|
try{
|
|
sHref = elATag.getAttribute("href");
|
|
}catch(e){}
|
|
if (sHref && sHref.match(rxMarker)) {
|
|
var sNewHref = sHref.replace(rxMarker, "");
|
|
var sDecodeHref = decodeURIComponent(sNewHref);
|
|
if(oAgent.ie){
|
|
jindo.$Element(elATag).attr({
|
|
"href" : sDecodeHref,
|
|
"target" : sTarget
|
|
});
|
|
//}else if(oAgent.firefox){
|
|
}else{
|
|
var sAContent = jindo.$Element(elATag).html();
|
|
jindo.$Element(elATag).attr({
|
|
"href" : sDecodeHref,
|
|
"target" : sTarget
|
|
});
|
|
if(this._validateURL(sAContent)){
|
|
jindo.$Element(elATag).html(jindo.$Element(elATag).attr("href"));
|
|
}
|
|
}
|
|
/*else{
|
|
elATag.href = sDecodeHref;
|
|
}
|
|
*/
|
|
}
|
|
}
|
|
}
|
|
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
setTimeout(jindo.$Fn(function(){
|
|
var oSelection = this.oApp.getEmptySelection();
|
|
oSelection.moveToBookmark(sBM);
|
|
oSelection.collapseToEnd();
|
|
oSelection.select();
|
|
oSelection.removeStringBookmark(sBM);
|
|
|
|
this.oApp.exec("FOCUS");
|
|
this.oApp.exec("RECORD_UNDO_AFTER_ACTION", ["HYPERLINK", {sSaveTarget:(this.bModify ? "A" : null)}]);
|
|
}, this).bind(), 17);
|
|
}else{
|
|
alert(this.oApp.$MSG("SE_Hyperlink.invalidURL"));
|
|
this.oLinkInput.focus();
|
|
}
|
|
},
|
|
|
|
_isExceptional : function(){
|
|
var oNavigator = jindo.$Agent().navigator(),
|
|
bImg = false, bEmail = false;
|
|
|
|
if(!oNavigator.ie){
|
|
return false;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-612] 이미지 선택 후 링크 추가했을 때 링크가 걸리지 않는 문제
|
|
if(this.oApp.getWYSIWYGDocument().selection && this.oApp.getWYSIWYGDocument().selection.type === "None"){
|
|
bImg = jindo.$A(this.oSelection.getNodes()).some(function(value, index, array){
|
|
if(value.nodeType === 1 && value.tagName === "IMG"){
|
|
return true;
|
|
}
|
|
}, this);
|
|
|
|
if(bImg){
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if(oNavigator.nativeVersion > 8){ // version? nativeVersion?
|
|
return false;
|
|
}
|
|
|
|
// [SMARTEDITORSUS-579] IE8 이하에서 E-mail 패턴 문자열에 URL 링크 못거는 이슈
|
|
bEmail = jindo.$A(this.oSelection.getTextNodes()).some(function(value, index, array){
|
|
if(value.nodeValue.indexOf("@") >= 1){
|
|
return true;
|
|
}
|
|
}, this);
|
|
|
|
if(bEmail){
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
_getSelectedNode : function(){
|
|
var aNodes = this.oSelection.getNodes();
|
|
|
|
for (var i = 0; i < aNodes.length; i++) {
|
|
if (aNodes[i].tagName && aNodes[i].tagName == "A") {
|
|
return aNodes[i];
|
|
}
|
|
}
|
|
},
|
|
_validateURL : function(sURL){
|
|
if(!sURL){return false;}
|
|
|
|
// escape 불가능한 %가 들어있나 확인
|
|
try{
|
|
var aURLParts = sURL.split("?");
|
|
aURLParts[0] = aURLParts[0].replace(/%[a-z0-9]{2}/gi, "U");
|
|
decodeURIComponent(aURLParts[0]);
|
|
}catch(e){
|
|
return false;
|
|
}
|
|
return /^(http|https|ftp|mailto):(\/\/)?(([-가-힣]|\w)+(?:[\/\.:@]([-가-힣]|\w)+)+)\/?(.*)?\s*$/i.test(sURL);
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_LineHeightWithLayerUI$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_LineHeightWithLayerUI
|
|
* this.oApp.registerLazyMessage(["SE2M_TOGGLE_LINEHEIGHT_LAYER"], ["hp_SE2M_LineHeightWithLayerUI$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_LineHeightWithLayerUI, {
|
|
//@lazyload_js SE2M_TOGGLE_LINEHEIGHT_LAYER[
|
|
_assignHTMLObjects : function(elAppContainer) {
|
|
//this.elLineHeightSelect = jindo.$$.getSingle("SELECT.husky_seditor_ui_lineHeight_select", elAppContainer);
|
|
this.oDropdownLayer = jindo.$$.getSingle("DIV.husky_se2m_lineHeight_layer", elAppContainer);
|
|
this.aLIOptions = jindo.$A(jindo.$$("LI", this.oDropdownLayer)).filter(function(v,i,a){return (v.firstChild !== null);})._array;
|
|
|
|
this.oInput = jindo.$$.getSingle("INPUT", this.oDropdownLayer);
|
|
|
|
var tmp = jindo.$$.getSingle(".husky_se2m_lineHeight_direct_input", this.oDropdownLayer);
|
|
tmp = jindo.$$("BUTTON", tmp);
|
|
this.oBtn_up = tmp[0];
|
|
this.oBtn_down = tmp[1];
|
|
this.oBtn_ok = tmp[2];
|
|
this.oBtn_cancel = tmp[3];
|
|
},
|
|
|
|
$LOCAL_BEFORE_FIRST : function(){
|
|
this._assignHTMLObjects(this.oApp.htOptions.elAppContainer);
|
|
|
|
this.oApp.exec("SE2_ATTACH_HOVER_EVENTS", [this.aLIOptions]);
|
|
|
|
for(var i=0; i<this.aLIOptions.length; i++){
|
|
this.oApp.registerBrowserEvent(this.aLIOptions[i], "click", "SET_LINEHEIGHT_FROM_LAYER_UI", [this._getLineHeightFromLI(this.aLIOptions[i])]);
|
|
}
|
|
|
|
this.oApp.registerBrowserEvent(this.oBtn_up, "click", "SE2M_INC_LINEHEIGHT", []);
|
|
this.oApp.registerBrowserEvent(this.oBtn_down, "click", "SE2M_DEC_LINEHEIGHT", []);
|
|
this.oApp.registerBrowserEvent(this.oBtn_ok, "click", "SE2M_SET_LINEHEIGHT_FROM_DIRECT_INPUT", []);
|
|
this.oApp.registerBrowserEvent(this.oBtn_cancel, "click", "SE2M_CANCEL_LINEHEIGHT", []);
|
|
|
|
this.oApp.registerBrowserEvent(this.oInput, "keydown", "EVENT_SE2M_LINEHEIGHT_KEYDOWN");
|
|
},
|
|
|
|
$ON_EVENT_SE2M_LINEHEIGHT_KEYDOWN : function(oEvent){
|
|
if (oEvent.key().enter){
|
|
this.oApp.exec("SE2M_SET_LINEHEIGHT_FROM_DIRECT_INPUT");
|
|
oEvent.stop();
|
|
}
|
|
},
|
|
|
|
$ON_SE2M_TOGGLE_LINEHEIGHT_LAYER : function(){
|
|
this.oApp.exec("TOGGLE_TOOLBAR_ACTIVE_LAYER", [this.oDropdownLayer, null, "LINEHEIGHT_LAYER_SHOWN", [], "LINEHEIGHT_LAYER_HIDDEN", []]);
|
|
this.oApp.exec('MSG_NOTIFY_CLICKCR', ['lineheight']);
|
|
},
|
|
|
|
$ON_SE2M_INC_LINEHEIGHT : function(){
|
|
this.oInput.value = parseInt(this.oInput.value, 10) || this.MIN_LINE_HEIGHT;
|
|
this.oInput.value++;
|
|
},
|
|
|
|
$ON_SE2M_DEC_LINEHEIGHT : function(){
|
|
this.oInput.value = parseInt(this.oInput.value, 10) || this.MIN_LINE_HEIGHT;
|
|
if(this.oInput.value > this.MIN_LINE_HEIGHT){this.oInput.value--;}
|
|
},
|
|
|
|
$ON_LINEHEIGHT_LAYER_SHOWN : function(){
|
|
this.oApp.exec("SELECT_UI", ["lineHeight"]);
|
|
this.oInitialSelection = this.oApp.getSelection();
|
|
|
|
var nLineHeight = this.oApp.getLineStyle("lineHeight");
|
|
|
|
if(nLineHeight != null && nLineHeight !== 0){
|
|
this.oInput.value = (nLineHeight*100).toFixed(0);
|
|
var elLi = this._getMatchingLI(this.oInput.value+"%");
|
|
if(elLi){jindo.$Element(elLi.firstChild).addClass("active");}
|
|
}else{
|
|
this.oInput.value = "";
|
|
}
|
|
},
|
|
|
|
$ON_LINEHEIGHT_LAYER_HIDDEN : function(){
|
|
this.oApp.exec("DESELECT_UI", ["lineHeight"]);
|
|
this._clearOptionSelection();
|
|
},
|
|
|
|
$ON_SE2M_SET_LINEHEIGHT_FROM_DIRECT_INPUT : function(){
|
|
var nInputValue = parseInt(this.oInput.value, 10);
|
|
var sValue = (nInputValue < this.MIN_LINE_HEIGHT) ? this.MIN_LINE_HEIGHT : nInputValue;
|
|
this._setLineHeightAndCloseLayer(sValue);
|
|
},
|
|
|
|
$ON_SET_LINEHEIGHT_FROM_LAYER_UI : function(sValue){
|
|
this._setLineHeightAndCloseLayer(sValue);
|
|
},
|
|
|
|
$ON_SE2M_CANCEL_LINEHEIGHT : function(){
|
|
this.oInitialSelection.select();
|
|
this.oApp.exec("HIDE_ACTIVE_LAYER");
|
|
},
|
|
|
|
_setLineHeightAndCloseLayer : function(sValue){
|
|
var nLineHeight = parseInt(sValue, 10)/100;
|
|
if(nLineHeight>0){
|
|
this.oApp.exec("SET_LINE_STYLE", ["lineHeight", nLineHeight]);
|
|
}else{
|
|
alert(this.oApp.$MSG("SE_LineHeight.invalidLineHeight"));
|
|
}
|
|
this.oApp.exec("SE2M_TOGGLE_LINEHEIGHT_LAYER", []);
|
|
|
|
var oNavigator = jindo.$Agent().navigator();
|
|
if(oNavigator.chrome || oNavigator.safari){
|
|
this.oApp.exec("FOCUS"); // [SMARTEDITORSUS-654]
|
|
}
|
|
},
|
|
|
|
_getMatchingLI : function(sValue){
|
|
var elLi;
|
|
|
|
sValue = sValue.toLowerCase();
|
|
for(var i=0; i<this.aLIOptions.length; i++){
|
|
elLi = this.aLIOptions[i];
|
|
if(this._getLineHeightFromLI(elLi).toLowerCase() == sValue){return elLi;}
|
|
}
|
|
|
|
return null;
|
|
},
|
|
|
|
_getLineHeightFromLI : function(elLi){
|
|
return elLi.firstChild.firstChild.innerHTML;
|
|
},
|
|
|
|
_clearOptionSelection : function(elLi){
|
|
for(var i=0; i<this.aLIOptions.length; i++){
|
|
jindo.$Element(this.aLIOptions[i].firstChild).removeClass("active");
|
|
}
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_SE2M_QuickEditor_Common$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.SE2M_QuickEditor_Common
|
|
* this.oApp.registerLazyMessage(["OPEN_QE_LAYER"], ["hp_SE2M_QuickEditor_Common$Lazy.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.SE2M_QuickEditor_Common, {
|
|
//@lazyload_js OPEN_QE_LAYER[
|
|
/**
|
|
* openType을 저장하는 함수.
|
|
* @param {String} sType
|
|
* @param {Boolean} bBol
|
|
*/
|
|
setOpenType : function(sType,bBol){
|
|
// [SMARTEDITORSUS-1213] 작성된 컨텐츠 수정 화면에서 사진이 로드되자마자 바로 사진을 클릭하면 QuickEditor를 띄우는 데 문제가 있음
|
|
if(typeof(this._environmentData) == "undefined" || this._environmentData == null){
|
|
this._environmentData = {};
|
|
}
|
|
if(typeof(this._environmentData[sType]) == "undefined" || this._environmentData[sType] == null){
|
|
this._environmentData[sType] = {};
|
|
}
|
|
if(typeof(this._environmentData[sType].isOpen) == "undefined" || this._environmentData[sType].isOpen == null){
|
|
this._environmentData[sType].isOpen = true;
|
|
}
|
|
// --[SMARTEDITORSUS-1213]
|
|
|
|
this._environmentData[sType].isOpen = bBol;
|
|
},
|
|
/**
|
|
* 레이어가 오픈 할 때 실행되는 이벤트.
|
|
* 레이어가 처음 뜰 때,
|
|
* 저장된 단축키 리스트를 레이어에 등록하고 (레이어가 떠 있을때도 단축키가 먹도록 하기 위해)
|
|
* 레이어에 대한 키보드/마우스 이벤트를 등록한다.
|
|
* @param {Element} oEle
|
|
* @param {Element} oLayer
|
|
* @param {String} sType(img|table|review)
|
|
*/
|
|
$ON_OPEN_QE_LAYER : function(oEle,oLayer,sType){
|
|
if(this.waHotkeys.length() > 0 && !this.waHotkeyLayers.has(oLayer)){
|
|
this.waHotkeyLayers.push(oLayer);
|
|
|
|
var aParam;
|
|
for(var i=0, nLen=this.waHotkeys.length(); i<nLen; i++){
|
|
aParam = this.waHotkeys.get(i);
|
|
this.oApp.exec("ADD_HOTKEY", [aParam[0], aParam[1], aParam[2], oLayer]);
|
|
}
|
|
}
|
|
|
|
var type = sType;//?sType:"table";//this.get_type(oEle);
|
|
if(type){
|
|
this.targetEle = oEle;
|
|
this.currentEle = oLayer;
|
|
this.layer_show(type,oEle);
|
|
}
|
|
},
|
|
/**
|
|
* 레이어가 닫혔을때 실행되는 이벤트.
|
|
* @param {jindo.$Event} weEvent
|
|
*/
|
|
$ON_CLOSE_QE_LAYER : function(weEvent){
|
|
if(!this.currentEle){return;}
|
|
// this.oApp.exec("HIDE_EDITING_AREA_COVER");
|
|
// this.oApp.exec("ENABLE_ALL_UI");
|
|
this.oApp.exec("CLOSE_SUB_LAYER_QE");
|
|
|
|
this.layer_hide(weEvent);
|
|
},
|
|
|
|
/**
|
|
* 어플리케이션이 준비단계일때 실행되는 이벤트
|
|
*/
|
|
$LOCAL_BEFORE_FIRST : function(sMsg) {
|
|
if (!sMsg.match(/OPEN_QE_LAYER/)) { // (sMsg == "$ON_CLOSE_QE_LAYER" && !this.currentEle)
|
|
this.oApp.acceptLocalBeforeFirstAgain(this, true);
|
|
if(sMsg.match(/REGISTER_HOTKEY/)){
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
this.woEditor = jindo.$Element(this.oApp.elEditingAreaContainer);
|
|
this.woStandard = jindo.$Element(this.oApp.htOptions.elAppContainer).offset();
|
|
this._qe_wrap = jindo.$$.getSingle("DIV.quick_wrap", this.oApp.htOptions.elAppContainer);
|
|
|
|
var that = this;
|
|
|
|
new jindo.DragArea(this._qe_wrap, {
|
|
sClassName : 'q_dragable',
|
|
bFlowOut : false,
|
|
nThreshold : 1
|
|
}).attach({
|
|
beforeDrag : function(oCustomEvent) {
|
|
oCustomEvent.elFlowOut = oCustomEvent.elArea.parentNode;
|
|
},
|
|
dragStart: function(oCustomEvent){
|
|
if(!jindo.$Element(oCustomEvent.elDrag).hasClass('se2_qmax')){
|
|
oCustomEvent.elDrag = oCustomEvent.elDrag.parentNode;
|
|
}
|
|
that.oApp.exec("SHOW_EDITING_AREA_COVER");
|
|
},
|
|
dragEnd : function(oCustomEvent){
|
|
that.changeFixedMode();
|
|
that._in_event = false;
|
|
//if(that._currentType=="review"||that._currentType=="table"){ // [SMARTEDITORSUS-153] 이미지 퀵 에디터도 같은 로직으로 처리하도록 수정
|
|
var richEle = jindo.$Element(oCustomEvent.elDrag);
|
|
that._environmentData[that._currentType].position = [richEle.css("top"),richEle.css("left")];
|
|
//}
|
|
that.oApp.exec("HIDE_EDITING_AREA_COVER");
|
|
}
|
|
});
|
|
|
|
var imgFn = jindo.$Fn(this.toggle,this).bind("img");
|
|
var tableFn = jindo.$Fn(this.toggle,this).bind("table");
|
|
|
|
jindo.$Fn(imgFn,this).attach(jindo.$$.getSingle(".q_open_img_fold", this.oApp.htOptions.elAppContainer),"click");
|
|
jindo.$Fn(imgFn,this).attach(jindo.$$.getSingle(".q_open_img_full", this.oApp.htOptions.elAppContainer),"click");
|
|
|
|
jindo.$Fn(tableFn,this).attach(jindo.$$.getSingle(".q_open_table_fold", this.oApp.htOptions.elAppContainer),"click");
|
|
jindo.$Fn(tableFn,this).attach(jindo.$$.getSingle(".q_open_table_full", this.oApp.htOptions.elAppContainer),"click");
|
|
},
|
|
/**
|
|
* 레이어의 최대화/최소화를 토글링 하는 함수.
|
|
* @param {String} sType(table|img)
|
|
* @param {jindo.$Event} weEvent
|
|
*/
|
|
toggle : function(sType,weEvent){
|
|
sType = this._currentType;
|
|
// var oBefore = jindo.$Element(jindo.$$.getSingle("._"+this._environmentData[sType].type,this.currentEle));
|
|
// var beforeX = oBefore.css("left");
|
|
// var beforeY = oBefore.css("top");
|
|
|
|
this.oApp.exec("CLOSE_QE_LAYER", [weEvent]);
|
|
|
|
if(this._environmentData[sType].type=="full"){
|
|
this._environmentData[sType].type = "fold";
|
|
}else{
|
|
this._environmentData[sType].type = "full";
|
|
}
|
|
|
|
// [SMARTEDITORSUS-1028][SMARTEDITORSUS-1517] QuickEditor 설정 API 개선으로, submit 이후 발생하게 되는 beforeunload 이벤트 대신 호출 시점 변경
|
|
// QuickEditor를 접고 펼칠 때마다 API 통신을 거치기 때문에 submit이나 beforeunload에 구애받지 않고 안정적인 데이터 저장 가능
|
|
if (this._environmentData && this._bUseConfig) {
|
|
// [SMARTEDITORSUS-1970] 사용 설정값이 있는 경우에만 Ajax를 호출하도록 한다.
|
|
jindo.$Ajax(this._sAddTextAjaxUrl,{
|
|
type : "jsonp",
|
|
onload: function(){}
|
|
}).request({
|
|
text_key :"qeditor_fold",
|
|
text_data : "{table:'"+this._environmentData["table"]["type"]+"',img:'"+this._environmentData["img"]["type"]+"',review:'"+this._environmentData["review"]["type"]+"'}"
|
|
});
|
|
}
|
|
// --[SMARTEDITORSUS-1028][SMARTEDITORSUS-1517]
|
|
|
|
// this.positionCopy(beforeX,beforeY,this._environmentData[sType].type);
|
|
|
|
this.oApp.exec("OPEN_QE_LAYER", [this.targetEle,this.currentEle,sType]);
|
|
this._in_event = false;
|
|
weEvent.stop(jindo.$Event.CANCEL_DEFAULT);
|
|
},
|
|
/**
|
|
* 토글링시 전에 엘리먼트에 위치를 카피하는 함수.
|
|
* @param {Number} beforeX
|
|
* @param {Number} beforeY
|
|
* @param {Element} sAfterEle
|
|
*/
|
|
positionCopy:function(beforeX, beforeY, sAfterEle){
|
|
jindo.$Element(jindo.$$.getSingle("._"+sAfterEle,this.currentEle)).css({
|
|
top : beforeY,
|
|
left : beforeX
|
|
});
|
|
},
|
|
/**
|
|
* 레이어를 고정으로 할때 실행되는 함수.
|
|
*/
|
|
changeFixedMode : function(){
|
|
this._environmentData[this._currentType].isFixed = true;
|
|
},
|
|
/**
|
|
* 에디팅 영역에서 keyup할때 실행되는 함수.
|
|
* @param {jindo.$Event} weEvent
|
|
*/
|
|
/*
|
|
$ON_EVENT_EDITING_AREA_KEYUP:function(weEvent){
|
|
if(this._currentType&&(!this._in_event)&&this._environmentData[this._currentType].isOpen){
|
|
this.oApp.exec("CLOSE_QE_LAYER", [weEvent]);
|
|
}
|
|
this._in_event = false;
|
|
},
|
|
*/
|
|
$ON_HIDE_ACTIVE_LAYER : function(){
|
|
this.oApp.exec("CLOSE_QE_LAYER");
|
|
},
|
|
|
|
/**
|
|
* 에디팅 영역에서 mousedown할때 실행되는 함수.
|
|
* @param {jindo.$Event} weEvent
|
|
*/
|
|
$ON_EVENT_EDITING_AREA_MOUSEDOWN:function(weEvent){
|
|
if(this._currentType&&(!this._in_event)&&this._environmentData[this._currentType].isOpen){
|
|
this.oApp.exec("CLOSE_QE_LAYER", [weEvent]);
|
|
}
|
|
this._in_event = false;
|
|
},
|
|
/**
|
|
* 에디팅 영역에서 mousewheel할때 실행되는 함수.
|
|
* @param {jindo.$Event} weEvent
|
|
*/
|
|
$ON_EVENT_EDITING_AREA_MOUSEWHEEL:function(weEvent){
|
|
if(this._currentType&&(!this._in_event)&&this._environmentData[this._currentType].isOpen){
|
|
this.oApp.exec("CLOSE_QE_LAYER", [weEvent]);
|
|
}
|
|
this._in_event = false;
|
|
},
|
|
/**
|
|
* 레이어를 띄우는데 레이어가 table(템플릿),img인지를 확인하여 id를 반환하는 함수.
|
|
* @param {Element} oEle
|
|
* @return {String} layer id
|
|
*/
|
|
get_type : function(oEle){
|
|
var tagName = oEle.tagName.toLowerCase();
|
|
|
|
if(this.waTableTagNames.has(tagName)){
|
|
return "table";
|
|
}else if(tagName=="img"){
|
|
return "img";
|
|
}
|
|
},
|
|
/**
|
|
* 퀵에디터에서 keyup시 실행되는 이벤트
|
|
*/
|
|
$ON_QE_IN_KEYUP : function(){
|
|
this._in_event = true;
|
|
},
|
|
/**
|
|
* 퀵에디터에서 mousedown시 실행되는 이벤트
|
|
*/
|
|
$ON_QE_IN_MOUSEDOWN : function(){
|
|
this._in_event = true;
|
|
},
|
|
/**
|
|
* 퀵에디터에서 mousewheel시 실행되는 이벤트
|
|
*/
|
|
$ON_QE_IN_MOUSEWHEEL : function(){
|
|
this._in_event = true;
|
|
},
|
|
/**
|
|
* 레이어를 숨기는 함수.
|
|
* @param {jindo.$Event} weEvent
|
|
*/
|
|
layer_hide : function(weEvent){
|
|
this.setOpenType(this._currentType,false);
|
|
|
|
jindo.$Element(jindo.$$.getSingle("._"+this._environmentData[this._currentType].type,this.currentEle)).hide();
|
|
},
|
|
/**
|
|
* 늦게 이벤트 바인딩 하는 함수.
|
|
* 레이어가 처음 뜰 때 이벤트를 등록한다.
|
|
*/
|
|
lazy_common : function(){
|
|
this.oApp.registerBrowserEvent(jindo.$(this._qe_wrap), "keyup", "QE_IN_KEYUP");
|
|
this.oApp.registerBrowserEvent(jindo.$(this._qe_wrap), "mousedown", "QE_IN_MOUSEDOWN");
|
|
this.oApp.registerBrowserEvent(jindo.$(this._qe_wrap), "mousewheel", "QE_IN_MOUSEWHEEL");
|
|
this.lazy_common = function(){};
|
|
},
|
|
/**
|
|
* 레이어를 보여주는 함수.
|
|
* @param {String} sType
|
|
* @param {Element} oEle
|
|
*/
|
|
layer_show : function(sType,oEle){
|
|
this._currentType = sType;
|
|
this.setOpenType(this._currentType,true);
|
|
var layer = jindo.$$.getSingle("._"+this._environmentData[this._currentType].type,this.currentEle);
|
|
jindo.$Element(layer)
|
|
.show()
|
|
.css( this.get_position_layer(oEle , layer) );
|
|
|
|
|
|
this.lazy_common();
|
|
},
|
|
/**
|
|
* 레이어의 위치를 반환 하는 함수
|
|
* 고정 상태가 아니거나 최소화 상태이면 엘리먼트 위치에 퀵에디터를 띄우고
|
|
* 고정 상태이고 최대화 상태이면 표나 글 양식은 저장된 위치에 띄워주고, 이미지는...?
|
|
* @param {Element} oEle
|
|
* @param {Element} oLayer
|
|
*/
|
|
get_position_layer : function(oEle , oLayer){
|
|
if(!this.isCurrentFixed() || this._environmentData[this._currentType].type == "fold"){
|
|
return this.calculateLayer(oEle , oLayer);
|
|
}
|
|
|
|
//if(this._currentType == "review" || this._currentType == "table"){ // [SMARTEDITORSUS-153] 이미지 퀵 에디터도 같은 로직으로 처리하도록 수정
|
|
var position = this._environmentData[this._currentType].position;
|
|
var nTop = parseInt(position[0], 10);
|
|
var nAppHeight = this.getAppPosition().h;
|
|
var nLayerHeight = jindo.$Element(oLayer).height();
|
|
|
|
// [SMARTEDITORSUS-129] 편집 영역 높이를 줄였을 때 퀵에디터가 영역을 벗어나지 않도록 처리
|
|
if((nTop + nLayerHeight + this.nYGap) > nAppHeight){
|
|
nTop = nAppHeight - nLayerHeight;
|
|
this._environmentData[this._currentType].position[0] = nTop;
|
|
}
|
|
|
|
return {
|
|
top : nTop + "px",
|
|
left :position[1]
|
|
};
|
|
//}
|
|
//return this.calculateLayer(null , oLayer);
|
|
},
|
|
/**
|
|
* 현재 레이어가 고정형태인지 반환하는 함수.
|
|
*/
|
|
isCurrentFixed : function(){
|
|
return this._environmentData[this._currentType].isFixed;
|
|
},
|
|
/**
|
|
* 레이어를 띄울 위치를 계산하는 함수.
|
|
* @param {Element} oEle
|
|
* @param {Element} oLayer
|
|
*/
|
|
calculateLayer : function(oEle, oLayer){
|
|
/*
|
|
* 기준을 한군데로 만들어야 함.
|
|
* 1. 에디터는 페이지
|
|
* 2. 엘리먼트는 안에 에디팅 영역
|
|
* 3. 레이어는 에디팅 영역
|
|
*
|
|
* 기준은 페이지로 함.
|
|
*/
|
|
var positionInfo = this.getPositionInfo(oEle, oLayer);
|
|
|
|
return {
|
|
top : positionInfo.y + "px",
|
|
left : positionInfo.x + "px"
|
|
};
|
|
},
|
|
/**
|
|
* 위치를 반환 하는 함수.
|
|
* @param {Element} oEle
|
|
* @param {Element} oLayer
|
|
*/
|
|
getPositionInfo : function(oEle, oLayer){
|
|
this.nYGap = jindo.$Agent().navigator().ie? -16 : -18;
|
|
this.nXGap = 1;
|
|
|
|
var oRevisePosition = {};
|
|
|
|
var eleInfo = this.getElementPosition(oEle, oLayer);
|
|
var appInfo = this.getAppPosition();
|
|
var layerInfo = {
|
|
w : jindo.$Element(oLayer).width(),
|
|
h : jindo.$Element(oLayer).height()
|
|
};
|
|
|
|
if((eleInfo.x + layerInfo.w + this.nXGap) > appInfo.w){
|
|
oRevisePosition.x = appInfo.w - layerInfo.w ;
|
|
}else{
|
|
oRevisePosition.x = eleInfo.x + this.nXGap;
|
|
}
|
|
|
|
if((eleInfo.y + layerInfo.h + this.nYGap) > appInfo.h){
|
|
oRevisePosition.y = appInfo.h - layerInfo.h - 2;
|
|
}else{
|
|
oRevisePosition.y = eleInfo.y + this.nYGap;
|
|
}
|
|
|
|
return {
|
|
x : oRevisePosition.x ,
|
|
y : oRevisePosition.y
|
|
};
|
|
},
|
|
/**
|
|
* 기준 엘리먼트의 위치를 반환하는 함수
|
|
* 엘리먼트가 있는 경우
|
|
* @param {Element} eEle
|
|
*/
|
|
getElementPosition : function(eEle, oLayer){
|
|
var wEle, oOffset, nEleWidth, nEleHeight, nScrollX, nScrollY;
|
|
|
|
if(eEle){
|
|
wEle = jindo.$Element(eEle);
|
|
oOffset = wEle.offset();
|
|
nEleWidth = wEle.width();
|
|
nEleHeight = wEle.height();
|
|
}else{
|
|
oOffset = {
|
|
top : parseInt(oLayer.style.top, 10) - this.nYGap,
|
|
left : parseInt(oLayer.style.left, 10) - this.nXGap
|
|
};
|
|
nEleWidth = 0;
|
|
nEleHeight = 0;
|
|
}
|
|
|
|
var oAppWindow = this.oApp.getWYSIWYGWindow();
|
|
|
|
if(typeof oAppWindow.scrollX == "undefined"){
|
|
nScrollX = oAppWindow.document.documentElement.scrollLeft;
|
|
nScrollY = oAppWindow.document.documentElement.scrollTop;
|
|
}else{
|
|
nScrollX = oAppWindow.scrollX;
|
|
nScrollY = oAppWindow.scrollY;
|
|
}
|
|
|
|
var oEditotOffset = this.woEditor.offset();
|
|
return {
|
|
x : oOffset.left - nScrollX + nEleWidth,
|
|
y : oOffset.top - nScrollY + nEleHeight
|
|
};
|
|
},
|
|
/**
|
|
* 에디터의 크기 계산하는 함수.
|
|
*/
|
|
getAppPosition : function(){
|
|
return {
|
|
w : this.woEditor.width(),
|
|
h : this.woEditor.height()
|
|
};
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("hp_DialogLayerManager$Lazy.js");
|
|
/**
|
|
* @depends nhn.husky.DialogLayerManager
|
|
* this.oApp.registerLazyMessage(["SHOW_DIALOG_LAYER","TOGGLE_DIALOG_LAYER"], ["hp_DialogLayerManager$Lazy.js", "N_DraggableLayer.js"]);
|
|
*/
|
|
nhn.husky.HuskyCore.mixin(nhn.husky.DialogLayerManager, {
|
|
//@lazyload_js SHOW_DIALOG_LAYER,TOGGLE_DIALOG_LAYER:N_DraggableLayer.js[
|
|
$ON_SHOW_DIALOG_LAYER : function(elLayer, htOptions){
|
|
elLayer = jindo.$(elLayer);
|
|
htOptions = htOptions || {};
|
|
|
|
if(!elLayer){return;}
|
|
|
|
if(jindo.$A(this.aOpenedLayers).has(elLayer)){return;}
|
|
|
|
this.oApp.exec("POSITION_DIALOG_LAYER", [elLayer]);
|
|
|
|
this.aOpenedLayers[this.aOpenedLayers.length] = elLayer;
|
|
|
|
var oDraggableLayer;
|
|
var nIdx = jindo.$A(this.aMadeDraggable).indexOf(elLayer);
|
|
|
|
if(nIdx == -1){
|
|
oDraggableLayer = new nhn.DraggableLayer(elLayer, htOptions);
|
|
this.aMadeDraggable[this.aMadeDraggable.length] = elLayer;
|
|
this.aDraggableLayer[this.aDraggableLayer.length] = oDraggableLayer;
|
|
}else{
|
|
if(htOptions){
|
|
oDraggableLayer = this.aDraggableLayer[nIdx];
|
|
oDraggableLayer.setOptions(htOptions);
|
|
}
|
|
elLayer.style.display = "block";
|
|
}
|
|
|
|
if(htOptions.sOnShowMsg){
|
|
this.oApp.exec(htOptions.sOnShowMsg, htOptions.sOnShowParam);
|
|
}
|
|
},
|
|
|
|
$ON_HIDE_LAST_DIALOG_LAYER : function(){
|
|
this.oApp.exec("HIDE_DIALOG_LAYER", [this.aOpenedLayers[this.aOpenedLayers.length-1]]);
|
|
},
|
|
|
|
$ON_HIDE_ALL_DIALOG_LAYER : function(){
|
|
for(var i=this.aOpenedLayers.length-1; i>=0; i--){
|
|
this.oApp.exec("HIDE_DIALOG_LAYER", [this.aOpenedLayers[i]]);
|
|
}
|
|
},
|
|
|
|
$ON_HIDE_DIALOG_LAYER : function(elLayer){
|
|
elLayer = jindo.$(elLayer);
|
|
|
|
if(elLayer){elLayer.style.display = "none";}
|
|
this.aOpenedLayers = jindo.$A(this.aOpenedLayers).refuse(elLayer).$value();
|
|
},
|
|
|
|
$ON_TOGGLE_DIALOG_LAYER : function(elLayer, htOptions){
|
|
if(jindo.$A(this.aOpenedLayers).indexOf(elLayer)){
|
|
this.oApp.exec("SHOW_DIALOG_LAYER", [elLayer, htOptions]);
|
|
}else{
|
|
this.oApp.exec("HIDE_DIALOG_LAYER", [elLayer]);
|
|
}
|
|
},
|
|
|
|
$ON_SET_DIALOG_LAYER_POSITION : function(elLayer, nTop, nLeft){
|
|
elLayer.style.top = nTop;
|
|
elLayer.style.left = nLeft;
|
|
}
|
|
//@lazyload_js]
|
|
});
|
|
nhn.husky.HuskyCore.addLoadedFile("N_FindReplace.js");
|
|
|
|
/**
|
|
* @fileOverview This file contains a function that takes care of various operations related to find and replace
|
|
* @name N_FindReplace.js
|
|
*/
|
|
nhn.FindReplace = jindo.$Class({
|
|
sKeyword : "",
|
|
window : null,
|
|
document : null,
|
|
bBrowserSupported : false,
|
|
_bLGDevice : false,
|
|
|
|
// true if End Of Contents is reached during last execution of find
|
|
bEOC : false,
|
|
|
|
$init : function(win){
|
|
this.sInlineContainer = "SPAN|B|U|I|S|STRIKE";
|
|
this.rxInlineContainer = new RegExp("^("+this.sInlineContainer+")$");
|
|
|
|
this.window = win;
|
|
this.document = this.window.document;
|
|
|
|
if(this.document.domain != this.document.location.hostname){
|
|
var oAgentInfo = jindo.$Agent();
|
|
var oNavigatorInfo = oAgentInfo.navigator();
|
|
|
|
if(oNavigatorInfo.firefox && oNavigatorInfo.version < 3){
|
|
this.bBrowserSupported = false;
|
|
this.find = function(){return 3;};
|
|
return;
|
|
}
|
|
}
|
|
|
|
this._bLGDevice = (navigator.userAgent.indexOf("LG-") > -1); // [SMARTEDITORSUS-1814] LG기기 여부 판단
|
|
this.bBrowserSupported = true;
|
|
},
|
|
|
|
// 0: found
|
|
// 1: not found
|
|
// 2: keyword required
|
|
// 3: browser not supported
|
|
find : function(sKeyword, bCaseMatch, bBackwards, bWholeWord){
|
|
var bSearchResult, bFreshSearch;
|
|
|
|
// [SMARTEDITORSUS-1814] LG브라우저의 경우 focus를 주면 선택영역이 풀리는 문제가 있어서 LG기기가 아닌 경우만 focus를 실행하도록 수정
|
|
// TODO: this.window.focus() 가 꼭 필요한지 전체적으로 점검해 볼 필요가 있음
|
|
if(!this._bLGDevice){
|
|
this.window.focus();
|
|
}
|
|
if(!sKeyword) return 2;
|
|
|
|
// try find starting from current cursor position
|
|
this.bEOC = false;
|
|
bSearchResult = this.findNext(sKeyword, bCaseMatch, bBackwards, bWholeWord);
|
|
if(bSearchResult) return 0;
|
|
|
|
// end of the contents could have been reached so search again from the beginning
|
|
this.bEOC = true;
|
|
bSearchResult = this.findNew(sKeyword, bCaseMatch, bBackwards, bWholeWord);
|
|
|
|
if(bSearchResult) return 0;
|
|
|
|
return 1;
|
|
},
|
|
|
|
findNew : function (sKeyword, bCaseMatch, bBackwards, bWholeWord){
|
|
this.findReset();
|
|
return this.findNext(sKeyword, bCaseMatch, bBackwards, bWholeWord);
|
|
},
|
|
|
|
findNext : function(sKeyword, bCaseMatch, bBackwards, bWholeWord){
|
|
var bSearchResult;
|
|
bCaseMatch = bCaseMatch || false;
|
|
bWholeWord = bWholeWord || false;
|
|
bBackwards = bBackwards || false;
|
|
|
|
if(this.window.find){
|
|
var bWrapAround = false;
|
|
if(this.document.body.contentEditable === "false"){ // [SMARTEDITORSUS-2086] 크롬에서 맞춤법검사후 단어 선택시 스크롤이 튀는 문제에 대한 workaround
|
|
return window.find(sKeyword, bCaseMatch, bBackwards, bWrapAround, bWholeWord);
|
|
}else{
|
|
return this.window.find(sKeyword, bCaseMatch, bBackwards, bWrapAround, bWholeWord);
|
|
}
|
|
}
|
|
|
|
// IE solution
|
|
if(this.document.body.createTextRange){
|
|
try{
|
|
var iOption = 0;
|
|
if(bBackwards) iOption += 1;
|
|
if(bWholeWord) iOption += 2;
|
|
if(bCaseMatch) iOption += 4;
|
|
|
|
this.window.focus();
|
|
if(this.document.selection){ // document.selection 이 있으면 selection 에서 TextRange 생성
|
|
this._range = this.document.selection.createRangeCollection().item(0);
|
|
this._range.collapse(false);
|
|
}else if(!this._range){ // [SMARTEDITORSUS-1528] IE11인 경우 createTextRange 로 TextRange 생성
|
|
this._range = this.document.body.createTextRange();
|
|
}else{ // [SMARTEDITORSUS-1837] 이미 생성되어 있는 TextRange를 이용해 collapseEnd 하면 다음 문자를 찾을 수 있다.
|
|
this._range.collapse(false);
|
|
}
|
|
bSearchResult = this._range.findText(sKeyword, 1, iOption);
|
|
|
|
this._range.select();
|
|
|
|
return bSearchResult;
|
|
}catch(e){
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
findReset : function() {
|
|
if (this.window.find){
|
|
this.window.getSelection().removeAllRanges();
|
|
return;
|
|
}
|
|
|
|
// IE solution
|
|
if(this.document.body.createTextRange){
|
|
this._range = this.document.body.createTextRange();
|
|
this._range.collapse(true);
|
|
this._range.select();
|
|
}
|
|
},
|
|
|
|
// 0: replaced & next word found
|
|
// 1: replaced & next word not found
|
|
// 2: not replaced & next word found
|
|
// 3: not replaced & next word not found
|
|
// 4: sOriginalWord required
|
|
replace : function(sOriginalWord, Replacement, bCaseMatch, bBackwards, bWholeWord){
|
|
return this._replace(sOriginalWord, Replacement, bCaseMatch, bBackwards, bWholeWord);
|
|
},
|
|
|
|
/**
|
|
* [SMARTEDITORSUS-1591] 크롬에서 replaceAll 시 selection 을 새로 만들면 첫번째 단어가 삭제되지 않고 남는 문제가 있어서
|
|
* selection 객체를 받아서 사용할 수 있도록 private 메서드 추가
|
|
* TODO: 근본적으로 HuskyRange 를 리팩토링할 필요가 있음
|
|
*/
|
|
_replace : function(sOriginalWord, Replacement, bCaseMatch, bBackwards, bWholeWord, oSelection){
|
|
if(!sOriginalWord) return 4;
|
|
|
|
oSelection = oSelection || new nhn.HuskyRange(this.window);
|
|
oSelection.setFromSelection();
|
|
|
|
bCaseMatch = bCaseMatch || false;
|
|
var bMatch, selectedText = oSelection.toString();
|
|
if(bCaseMatch)
|
|
bMatch = (selectedText == sOriginalWord);
|
|
else
|
|
bMatch = (selectedText.toLowerCase() == sOriginalWord.toLowerCase());
|
|
|
|
if(!bMatch)
|
|
return this.find(sOriginalWord, bCaseMatch, bBackwards, bWholeWord)+2;
|
|
|
|
if(typeof Replacement == "function"){
|
|
// the returned oSelection must contain the replacement
|
|
oSelection = Replacement(oSelection);
|
|
}else{
|
|
oSelection.pasteText(Replacement);
|
|
}
|
|
|
|
// force it to find the NEXT occurance of sOriginalWord
|
|
oSelection.select();
|
|
|
|
return this.find(sOriginalWord, bCaseMatch, bBackwards, bWholeWord);
|
|
},
|
|
|
|
// returns number of replaced words
|
|
// -1 : if original word is not given
|
|
replaceAll : function(sOriginalWord, Replacement, bCaseMatch, bWholeWord){
|
|
if(!sOriginalWord) return -1;
|
|
|
|
var bBackwards = false;
|
|
|
|
var iReplaceResult;
|
|
var iResult = 0;
|
|
var win = this.window;
|
|
|
|
if(this.find(sOriginalWord, bCaseMatch, bBackwards, bWholeWord) !== 0){
|
|
return iResult;
|
|
}
|
|
|
|
var oSelection = new nhn.HuskyRange(this.window);
|
|
oSelection.setFromSelection();
|
|
|
|
// 시작점의 북마크가 지워지면서 시작점을 지나서 replace가 되는 현상 방지용
|
|
// 첫 단어 앞쪽에 특수 문자 삽입 해서, replace와 함께 북마크가 사라지는 것 방지
|
|
oSelection.collapseToStart();
|
|
var oTmpNode = this.window.document.createElement("SPAN");
|
|
oTmpNode.innerHTML = unescape("%uFEFF");
|
|
oSelection.insertNode(oTmpNode);
|
|
oSelection.select();
|
|
var sBookmark = oSelection.placeStringBookmark();
|
|
|
|
this.bEOC = false;
|
|
while(!this.bEOC){
|
|
iReplaceResult = this._replace(sOriginalWord, Replacement, bCaseMatch, bBackwards, bWholeWord, oSelection);
|
|
if(iReplaceResult == 0 || iReplaceResult == 1){
|
|
iResult++;
|
|
}
|
|
}
|
|
|
|
var startingPointReached = function(){
|
|
var oCurSelection = new nhn.HuskyRange(win);
|
|
oCurSelection.setFromSelection();
|
|
|
|
oSelection.moveToBookmark(sBookmark);
|
|
var pos = oSelection.compareBoundaryPoints(nhn.W3CDOMRange.START_TO_END, oCurSelection);
|
|
|
|
if(pos == 1) return false;
|
|
return true;
|
|
};
|
|
|
|
iReplaceResult = 0;
|
|
this.bEOC = false;
|
|
while(!startingPointReached() && iReplaceResult == 0 && !this.bEOC){
|
|
iReplaceResult = this._replace(sOriginalWord, Replacement, bCaseMatch, bBackwards, bWholeWord, oSelection);
|
|
if(iReplaceResult == 0 || iReplaceResult == 1){
|
|
iResult++;
|
|
}
|
|
}
|
|
|
|
oSelection.moveToBookmark(sBookmark);
|
|
oSelection.deleteContents(); // [SMARTEDITORSUS-1591] 크롬에서 첫번째 단어가 삭제되지 않는 경우가 있으므로 select()메서드대신 deleteContents() 메서드를 호출한다.
|
|
oSelection.removeStringBookmark(sBookmark);
|
|
|
|
// setTimeout 없이 바로 지우면 IE8 브라우저가 빈번하게 죽어버림
|
|
setTimeout(function(){
|
|
if(oTmpNode && oTmpNode.parentNode){
|
|
oTmpNode.parentNode.removeChild(oTmpNode);
|
|
}
|
|
}, 0);
|
|
|
|
return iResult;
|
|
},
|
|
|
|
_isBlankTextNode : function(oNode){
|
|
if(oNode.nodeType == 3 && oNode.nodeValue == ""){return true;}
|
|
return false;
|
|
},
|
|
|
|
_getNextNode : function(elNode, bDisconnected){
|
|
if(!elNode || elNode.tagName == "BODY"){
|
|
return {elNextNode: null, bDisconnected: false};
|
|
}
|
|
|
|
if(elNode.nextSibling){
|
|
elNode = elNode.nextSibling;
|
|
while(elNode.firstChild){
|
|
if(elNode.tagName && !this.rxInlineContainer.test(elNode.tagName)){
|
|
bDisconnected = true;
|
|
}
|
|
elNode = elNode.firstChild;
|
|
}
|
|
return {elNextNode: elNode, bDisconnected: bDisconnected};
|
|
}
|
|
|
|
return this._getNextNode(nhn.DOMFix.parentNode(elNode), bDisconnected);
|
|
},
|
|
|
|
_getNextTextNode : function(elNode, bDisconnected){
|
|
var htNextNode, elNode;
|
|
while(true){
|
|
htNextNode = this._getNextNode(elNode, bDisconnected);
|
|
elNode = htNextNode.elNextNode;
|
|
bDisconnected = htNextNode.bDisconnected;
|
|
|
|
if(elNode && elNode.nodeType != 3 && !this.rxInlineContainer.test(elNode.tagName)){
|
|
bDisconnected = true;
|
|
}
|
|
|
|
if(!elNode || (elNode.nodeType==3 && !this._isBlankTextNode(elNode))){
|
|
break;
|
|
}
|
|
}
|
|
|
|
return {elNextText: elNode, bDisconnected: bDisconnected};
|
|
},
|
|
|
|
_getFirstTextNode : function(){
|
|
// 문서에서 제일 앞쪽에 위치한 아무 노드 찾기
|
|
var elFirstNode = this.document.body.firstChild;
|
|
while(!!elFirstNode && elFirstNode.firstChild){
|
|
elFirstNode = elFirstNode.firstChild;
|
|
}
|
|
|
|
// 문서에 아무 노드도 없음
|
|
if(!elFirstNode){
|
|
return null;
|
|
}
|
|
|
|
// 처음 노드가 텍스트 노드가 아니거나 bogus 노드라면 다음 텍스트 노드를 찾음
|
|
if(elFirstNode.nodeType != 3 || this._isBlankTextNode(elFirstNode)){
|
|
var htTmp = this._getNextTextNode(elFirstNode, false);
|
|
elFirstNode = htTmp.elNextText;
|
|
}
|
|
|
|
return elFirstNode;
|
|
},
|
|
|
|
_addToTextMap : function(elNode, aTexts, aElTexts, nLen){
|
|
var nStartPos = aTexts[nLen].length;
|
|
for(var i=0, nTo=elNode.nodeValue.length; i<nTo; i++){
|
|
aElTexts[nLen][nStartPos+i] = [elNode, i];
|
|
}
|
|
aTexts[nLen] += elNode.nodeValue;
|
|
},
|
|
|
|
_createTextMap : function(){
|
|
var aTexts = [];
|
|
var aElTexts = [];
|
|
var nLen=-1;
|
|
|
|
var elNode = this._getFirstTextNode();
|
|
var htNextNode = {elNextText: elNode, bDisconnected: true};
|
|
while(elNode){
|
|
if(htNextNode.bDisconnected){
|
|
nLen++;
|
|
|
|
aTexts[nLen] = "";
|
|
aElTexts[nLen] = [];
|
|
}
|
|
this._addToTextMap(htNextNode.elNextText, aTexts, aElTexts, nLen);
|
|
|
|
htNextNode = this._getNextTextNode(elNode, false);
|
|
elNode = htNextNode.elNextText;
|
|
}
|
|
|
|
return {aTexts: aTexts, aElTexts: aElTexts};
|
|
},
|
|
|
|
replaceAll_js : function(sOriginalWord, Replacement, bCaseMatch, bWholeWord){
|
|
try{
|
|
var t0 = new Date();
|
|
|
|
var htTmp = this._createTextMap();
|
|
|
|
var t1 = new Date();
|
|
var aTexts = htTmp.aTexts;
|
|
var aElTexts = htTmp.aElTexts;
|
|
|
|
// console.log(sOriginalWord);
|
|
// console.log(aTexts);
|
|
// console.log(aElTexts);
|
|
|
|
var nMatchCnt = 0;
|
|
|
|
var nOriginLen = sOriginalWord.length;
|
|
|
|
// 단어 한개씩 비교
|
|
for(var i=0, niLen=aTexts.length; i<niLen; i++){
|
|
var sText = aTexts[i];
|
|
// 단어 안에 한글자씩 비교
|
|
//for(var j=0, njLen=sText.length - nOriginLen; j<njLen; j++){
|
|
for(var j=sText.length-nOriginLen; j>=0; j--){
|
|
var sTmp = sText.substring(j, j+nOriginLen);
|
|
if(bWholeWord &&
|
|
(j > 0 && sText.charAt(j-1).match(/[a-zA-Z가-힣]/))
|
|
){
|
|
continue;
|
|
}
|
|
|
|
if(sTmp == sOriginalWord){
|
|
nMatchCnt++;
|
|
|
|
var oSelection = new nhn.HuskyRange(this.window);
|
|
// 마지막 글자의 뒷부분 처리
|
|
var elContainer, nOffset;
|
|
if(j+nOriginLen < aElTexts[i].length){
|
|
elContainer = aElTexts[i][j+nOriginLen][0];
|
|
nOffset = aElTexts[i][j+nOriginLen][1];
|
|
}else{
|
|
elContainer = aElTexts[i][j+nOriginLen-1][0];
|
|
nOffset = aElTexts[i][j+nOriginLen-1][1]+1;
|
|
}
|
|
oSelection.setEnd(elContainer, nOffset, true, true);
|
|
oSelection.setStart(aElTexts[i][j][0], aElTexts[i][j][1], true);
|
|
|
|
if(typeof Replacement == "function"){
|
|
// the returned oSelection must contain the replacement
|
|
oSelection = Replacement(oSelection);
|
|
}else{
|
|
oSelection.pasteText(Replacement);
|
|
}
|
|
|
|
j -= nOriginLen;
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
/*
|
|
var t2 = new Date();
|
|
console.log("OK");
|
|
console.log(sOriginalWord);
|
|
console.log("MC:"+(t1-t0));
|
|
console.log("RP:"+(t2-t1));
|
|
*/
|
|
|
|
return nMatchCnt;
|
|
}catch(e){
|
|
/*
|
|
console.log("ERROR");
|
|
console.log(sOriginalWord);
|
|
console.log(new Date()-t0);
|
|
*/
|
|
return nMatchCnt;
|
|
}
|
|
}
|
|
});
|
|
|
|
nhn.husky.HuskyCore.addLoadedFile("SE2M_TableTemplate.js");
|
|
// "padding", "backgroundcolor", "border", "borderTop", "borderRight", "borderBottom", "borderLeft", "color", "textAlign", "fontWeight"
|
|
nhn.husky.SE2M_TableTemplate = [
|
|
{},
|
|
/*
|
|
// 0
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
htTableStyle : {
|
|
border : "1px dashed #666666",
|
|
borderRight : "0",
|
|
borderBottom : "0"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 0 2px 0",
|
|
border : "1px dashed #666666",
|
|
borderTop : "0",
|
|
borderLeft : "0"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 1
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
htTableStyle : {
|
|
border : "1px solid #c7c7c7",
|
|
borderRight : "0",
|
|
borderBottom : "0"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 0 2px 0",
|
|
border : "1px solid #c7c7c7",
|
|
borderTop : "0",
|
|
borderLeft : "0"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 2
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
border : "1px solid #c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "2px 0 1px 0",
|
|
border : "1px solid #c7c7c7"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 3
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
border : "1px double #c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "1px 0 0",
|
|
border : "3px double #c7c7c7"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 4
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
borderWidth : "2px 1px 1px 2px",
|
|
borderStyle : "solid",
|
|
borderColor : "#c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "2px 0 0",
|
|
borderWidth : "1px 2px 2px 1px",
|
|
borderStyle : "solid",
|
|
borderColor : "#c7c7c7"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 5
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
borderWidth : "1px 2px 2px 1px",
|
|
borderStyle : "solid",
|
|
borderColor : "#c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "1px 0 0",
|
|
borderWidth : "2px 1px 1px 2px",
|
|
borderStyle : "solid",
|
|
borderColor : "#c7c7c7"
|
|
}
|
|
]
|
|
},
|
|
*/
|
|
// Black theme ======================================================
|
|
|
|
// 6
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 7
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#666666"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#f3f3f3",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 8
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#ffffff",
|
|
borderTop : "1px solid #c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #c7c7c7",
|
|
backgroundColor : "#ffffff",
|
|
color : "#666666"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #c7c7c7",
|
|
backgroundColor : "#f3f3f3",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 9
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
htTableStyle : {
|
|
border : "1px solid #c7c7c7"
|
|
},
|
|
ht1stRowStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#f3f3f3",
|
|
color : "#666666",
|
|
borderRight : "1px solid #e7e7e7",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
borderTop : "1px solid #e7e7e7",
|
|
borderRight : "1px solid #e7e7e7",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 10
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#c7c7c7"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#f8f8f8",
|
|
color : "#666666"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ebebeb",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 11
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
ht1stRowStyle : {
|
|
padding : "3px 4px 2px",
|
|
borderTop : "1px solid #000000",
|
|
borderBottom : "1px solid #000000",
|
|
backgroundColor : "#333333",
|
|
color : "#ffffff",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #ebebeb",
|
|
backgroundColor : "#ffffff",
|
|
color : "#666666"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #ebebeb",
|
|
backgroundColor : "#f8f8f8",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 12
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#c7c7c7"
|
|
},
|
|
ht1stRowStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#333333",
|
|
color : "#ffffff",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
ht1stColumnStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#f8f8f8",
|
|
color : "#666666",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 13
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#c7c7c7"
|
|
},
|
|
ht1stColumnStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#333333",
|
|
color : "#ffffff",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#666666"
|
|
}
|
|
]
|
|
},
|
|
|
|
// Blue theme ======================================================
|
|
|
|
// 14
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#a6bcd1"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 15
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#a6bcd1"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#3d76ab"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#f6f8fa",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 16
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#ffffff",
|
|
borderTop : "1px solid #a6bcd1"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #a6bcd1",
|
|
backgroundColor : "#ffffff",
|
|
color : "#3d76ab"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #a6bcd1",
|
|
backgroundColor : "#f6f8fa",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 17
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
htTableStyle : {
|
|
border : "1px solid #a6bcd1"
|
|
},
|
|
ht1stRowStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#f6f8fa",
|
|
color : "#3d76ab",
|
|
borderRight : "1px solid #e1eef7",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
borderTop : "1px solid #e1eef7",
|
|
borderRight : "1px solid #e1eef7",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 18
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#a6bcd1"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#fafbfc",
|
|
color : "#3d76ab"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#e6ecf2",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 19
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "0"
|
|
},
|
|
ht1stRowStyle : {
|
|
padding : "3px 4px 2px",
|
|
borderTop : "1px solid #466997",
|
|
borderBottom : "1px solid #466997",
|
|
backgroundColor : "#6284ab",
|
|
color : "#ffffff",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #ebebeb",
|
|
backgroundColor : "#ffffff",
|
|
color : "#3d76ab"
|
|
},
|
|
{
|
|
padding : "3px 4px 2px",
|
|
borderBottom : "1px solid #ebebeb",
|
|
backgroundColor : "#f6f8fa",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 20
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#a6bcd1"
|
|
},
|
|
ht1stRowStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#6284ab",
|
|
color : "#ffffff",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
ht1stColumnStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#f6f8fa",
|
|
color : "#3d76ab",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
},
|
|
|
|
// 21
|
|
{
|
|
htTableProperty : {
|
|
border : "0",
|
|
cellPadding : "0",
|
|
cellSpacing : "1"
|
|
},
|
|
htTableStyle : {
|
|
backgroundColor : "#a6bcd1"
|
|
},
|
|
ht1stColumnStyle : {
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#6284ab",
|
|
color : "#ffffff",
|
|
textAlign : "left",
|
|
fontWeight : "normal"
|
|
},
|
|
aRowStyle : [
|
|
{
|
|
padding : "3px 4px 2px",
|
|
backgroundColor : "#ffffff",
|
|
color : "#3d76ab"
|
|
}
|
|
]
|
|
}
|
|
];
|
|
nhn.husky.HuskyCore.addLoadedFile("N_DraggableLayer.js");
|
|
/**
|
|
* @fileOverview This file contains a function that takes care of the draggable layers
|
|
* @name N_DraggableLayer.js
|
|
*/
|
|
nhn.DraggableLayer = jindo.$Class({
|
|
$init : function(elLayer, oOptions){
|
|
this.elLayer = elLayer;
|
|
|
|
this.setOptions(oOptions);
|
|
|
|
this.elHandle = this.oOptions.elHandle;
|
|
|
|
elLayer.style.display = "block";
|
|
elLayer.style.position = "absolute";
|
|
elLayer.style.zIndex = "9999";
|
|
|
|
this.aBasePosition = this.getBaseOffset(elLayer);
|
|
|
|
// "number-ize" the position and set it as inline style. (the position could've been set as "auto" or set by css, not inline style)
|
|
var nTop = (this.toInt(jindo.$Element(elLayer).offset().top) - this.aBasePosition.top);
|
|
var nLeft = (this.toInt(jindo.$Element(elLayer).offset().left) - this.aBasePosition.left);
|
|
|
|
var htXY = this._correctXY({x:nLeft, y:nTop});
|
|
|
|
elLayer.style.top = htXY.y+"px";
|
|
elLayer.style.left = htXY.x+"px";
|
|
|
|
this.$FnMouseDown = jindo.$Fn(jindo.$Fn(this._mousedown, this).bind(elLayer), this);
|
|
this.$FnMouseMove = jindo.$Fn(jindo.$Fn(this._mousemove, this).bind(elLayer), this);
|
|
this.$FnMouseUp = jindo.$Fn(jindo.$Fn(this._mouseup, this).bind(elLayer), this);
|
|
|
|
this.$FnMouseDown.attach(this.elHandle, "mousedown");
|
|
this.elHandle.ondragstart = new Function('return false');
|
|
this.elHandle.onselectstart = new Function('return false');
|
|
},
|
|
|
|
_mousedown : function(elLayer, oEvent){
|
|
if(oEvent.element.tagName == "INPUT") return;
|
|
|
|
this.oOptions.fnOnDragStart();
|
|
|
|
this.MouseOffsetY = (oEvent.pos().clientY-this.toInt(elLayer.style.top)-this.aBasePosition['top']);
|
|
this.MouseOffsetX = (oEvent.pos().clientX-this.toInt(elLayer.style.left)-this.aBasePosition['left']);
|
|
|
|
this.$FnMouseMove.attach(elLayer.ownerDocument, "mousemove");
|
|
this.$FnMouseUp.attach(elLayer.ownerDocument, "mouseup");
|
|
|
|
this.elHandle.style.cursor = "move";
|
|
},
|
|
|
|
_mousemove : function(elLayer, oEvent){
|
|
var nTop = (oEvent.pos().clientY-this.MouseOffsetY-this.aBasePosition['top']);
|
|
var nLeft = (oEvent.pos().clientX-this.MouseOffsetX-this.aBasePosition['left']);
|
|
|
|
var htXY = this._correctXY({x:nLeft, y:nTop});
|
|
|
|
elLayer.style.top = htXY.y + "px";
|
|
elLayer.style.left = htXY.x + "px";
|
|
},
|
|
|
|
_mouseup : function(elLayer, oEvent){
|
|
this.oOptions.fnOnDragEnd();
|
|
|
|
this.$FnMouseMove.detach(elLayer.ownerDocument, "mousemove");
|
|
this.$FnMouseUp.detach(elLayer.ownerDocument, "mouseup");
|
|
|
|
this.elHandle.style.cursor = "";
|
|
},
|
|
|
|
_correctXY : function(htXY){
|
|
var nLeft = htXY.x;
|
|
var nTop = htXY.y;
|
|
|
|
if(nTop<this.oOptions.nMinY) nTop = this.oOptions.nMinY;
|
|
if(nTop>this.oOptions.nMaxY) nTop = this.oOptions.nMaxY;
|
|
|
|
if(nLeft<this.oOptions.nMinX) nLeft = this.oOptions.nMinX;
|
|
if(nLeft>this.oOptions.nMaxX) nLeft = this.oOptions.nMaxX;
|
|
|
|
return {x:nLeft, y:nTop};
|
|
},
|
|
|
|
toInt : function(num){
|
|
var result = parseInt(num);
|
|
return result || 0;
|
|
},
|
|
|
|
findNonStatic : function(oEl){
|
|
if(!oEl) return null;
|
|
if(oEl.tagName == "BODY") return oEl;
|
|
|
|
if(jindo.$Element(oEl).css("position").match(/absolute|relative/i)) return oEl;
|
|
|
|
return this.findNonStatic(oEl.offsetParent);
|
|
},
|
|
|
|
getBaseOffset : function(oEl){
|
|
var oBase = this.findNonStatic(oEl.offsetParent) || oEl.ownerDocument.body;
|
|
var tmp = jindo.$Element(oBase).offset();
|
|
|
|
return {top: tmp.top, left: tmp.left};
|
|
},
|
|
|
|
setOptions : function(htOptions){
|
|
this.oOptions = htOptions || {};
|
|
this.oOptions.bModal = this.oOptions.bModal || false;
|
|
this.oOptions.elHandle = this.oOptions.elHandle || this.elLayer;
|
|
this.oOptions.nMinX = this.oOptions.nMinX || -999999;
|
|
this.oOptions.nMinY = this.oOptions.nMinY || -999999;
|
|
this.oOptions.nMaxX = this.oOptions.nMaxX || 999999;
|
|
this.oOptions.nMaxY = this.oOptions.nMaxY || 999999;
|
|
this.oOptions.fnOnDragStart = this.oOptions.fnOnDragStart || function(){};
|
|
this.oOptions.fnOnDragEnd = this.oOptions.fnOnDragEnd || function(){};
|
|
}
|
|
});
|