Created
February 20, 2020 23:18
-
-
Save OlegShchavelev/ad93b1e118b0827ef8fdcf358fd6039b to your computer and use it in GitHub Desktop.
Моя не удачная попытка с рендерингом modx.texteditor.js - файл отвечающий за рендеринг JS и tvace.tpl - мой шаблон кастомного типа ТВ с текстовой областью и редактором ACE
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Ext.ux.Ace = Ext.extend(Ext.form.TextField, { | |
growMin : 60, | |
growMax: 1000, | |
mode : 'text', | |
theme : 'textmate', | |
showInvisibles : false, | |
selectionStyle : 'line', | |
scrollSpeed : 3, | |
showFoldWidgets : true, | |
useSoftTabs : true, | |
tabSize : 4, | |
useWrapMode : false, | |
fontSize : '13px', | |
value : '', | |
style: 'padding:0', | |
initEvents : function(){ | |
Ext.ux.Ace.superclass.initEvents.call(this); | |
this.editor.on('focus', this.onFocus.bind(this)); | |
this.editor.on('blur', this.onBlur.bind(this)); | |
}, | |
initComponent : function(){ | |
this.valueHolder = document.createElement('input'); | |
this.valueHolder.type = 'hidden'; | |
this.valueHolder.name = this.name; | |
this.valueHolder.value = this.value; | |
}, | |
onRender : function(ct, position){ | |
if(!this.el){ | |
this.defaultAutoCreate = { | |
tag: "div", | |
cls: "x-form-textarea", | |
style:"width:100%;height:60px" | |
}; | |
} | |
Ext.ux.Ace.superclass.onRender.call(this, ct, position); | |
var useragent = ace.require('ace/lib/useragent'); | |
if(this.grow){ | |
this.el.setHeight(this.growMin); | |
} | |
this.editor = ace.edit(this.el.dom); | |
this.editor.$blockScrolling = Infinity; | |
this.el.appendChild(this.valueHolder); | |
this.el.dom.removeAttribute('name'); | |
this.el.focus = this.focus.bind(this); | |
this.editor.getSession().setValue(this.valueHolder.value); | |
this.editor.setShowPrintMargin(false); | |
this.editor.getSession().setTabSize(this.tabSize); | |
this.editor.setAutoScrollEditorIntoView(true); | |
if (!useragent.isMac) | |
this.editor.setDragDelay(0); | |
this.editor.setFontSize(this.fontSize); | |
this.editor.setFadeFoldWidgets(false); | |
this.setShowInvisibles(this.showInvisibles); | |
this.setSelectionStyle(this.selectionStyle); | |
this.setScrollSpeed(this.scrollSpeed); | |
this.setShowFoldWidgets(this.showFoldWidgets); | |
this.setUseSoftTabs(this.useSoftTabs); | |
this.setUseWrapMode(this.useWrapMode); | |
ace.require("ace/ext/language_tools"); | |
this.editor.setOptions({ | |
enableBasicAutocompletion: true | |
}); | |
this.setTheme(this.theme); | |
this.setMode(this.mode); | |
this.editor.getSession().on('change', (function(){ | |
setTimeout(function(){ | |
this.valueHolder.value = this.editor.getSession().getValue(); | |
}.bind(this), 10); | |
}).bind(this)); | |
// TODO: attach autoSize to according event (?) | |
this.autoSize(); | |
}, | |
onDestroy : function(){ | |
this.editor.destroy(); | |
Ext.ux.Ace.superclass.onDestroy.call(this); | |
}, | |
validate : function(){ | |
return true; | |
}, | |
getErrors : function(value){ | |
return null; | |
}, | |
onResize : function(){ | |
this.editor.resize(true); | |
}, | |
doAutoSize : function(e){ | |
return !e.isNavKeyPress() || e.getKey() == e.ENTER; | |
}, | |
autoSize: function(){ | |
var linesCount = this.editor.getSession().getScreenLength(); | |
var lineHeight = this.editor.renderer.lineHeight; | |
var scrollBar = this.editor.renderer.scrollBar.getWidth(); | |
var bordersWidth = this.el.getBorderWidth('tb'); | |
var bottomOffset = lineHeight*5+scrollBar; | |
var h = Math.min(this.growMax, Math.max(linesCount * lineHeight + bordersWidth + bottomOffset, this.growMin)); | |
var heightChanged = h!=this.lastHeight; | |
if(this.grow && heightChanged){ | |
this.setHeight(h); | |
this.editor.resize(); | |
this.fireEvent("autosize", this, h); | |
} | |
if(!this.editor.searchBox || heightChanged){ | |
if(this.editor.searchBox)this.detectSearchBoxPosition(h); | |
else{ | |
var that = this; | |
ace.config.loadModule("ace/ext/searchbox",function(m){ | |
m.Search(that.editor); | |
that.editor.searchBox.hide(); | |
that.detectSearchBoxPosition(h); | |
}); | |
} | |
} | |
if(heightChanged)this.lastHeight = h; | |
}, | |
detectSearchBoxPosition : function(editorHeight){ | |
var triggerOffset = 150; | |
var defaultStyles={ | |
position:null | |
,bottom:null | |
,top:null | |
,borderRadius:null | |
}; | |
var fixedStyles={ | |
position:'fixed' | |
,bottom:'0' | |
,top:'initial' | |
,borderRadius:'5px 0px 0px 0' | |
}; | |
if(!this.isFullscreen&&editorHeight>=(window.innerHeight-triggerOffset))Ext.apply(this.editor.searchBox.element.style,fixedStyles); | |
else Ext.apply(this.editor.searchBox.element.style,defaultStyles); | |
}, | |
setSize : function(width, height){ | |
Ext.ux.Ace.superclass.setSize.apply(this, arguments); | |
this.editor.resize(true); | |
}, | |
getValue : function (){ | |
return this.valueHolder.value; | |
}, | |
setValue : function (value){ | |
if (this.editor) { | |
this.editor.getSession().setValue(value); | |
} else { | |
this.valueHolder.value = value; | |
} | |
this.value = value; | |
}, | |
setMode : function (mode){ | |
this.editor.getSession().setMode( 'ace/mode/' + mode ); | |
}, | |
setTheme : function(theme){ | |
this.editor.setTheme('ace/theme/' + theme); | |
}, | |
setFontSize : function(fontSize){ | |
this.editor.setFontSize(fontSize); | |
}, | |
setShowInvisibles : function(showInvisibles){ | |
this.editor.setShowInvisibles(showInvisibles); | |
}, | |
setSelectionStyle : function(selectionStyle){ | |
this.editor.setSelectionStyle(selectionStyle); | |
}, | |
setScrollSpeed : function(scrollSpeed){ | |
this.editor.setScrollSpeed(scrollSpeed); | |
}, | |
setShowFoldWidgets : function(showFoldWidgets){ | |
this.editor.setShowFoldWidgets(showFoldWidgets); | |
}, | |
setUseSoftTabs : function(useSoftTabs){ | |
this.editor.getSession().setUseSoftTabs(useSoftTabs); | |
}, | |
setUseWrapMode : function(useWrapMode){ | |
this.editor.getSession().setUseWrapMode(useWrapMode); | |
}, | |
insertAtCursor : function (value){ | |
return this.editor.insert(value); | |
}, | |
focus: function (){ | |
this.editor.focus(); | |
}, | |
blur: function (){ | |
this.editor.blur(); | |
} | |
}); | |
Ext.reg('ace', Ext.ux.Ace); | |
Ext.namespace('MODx.ux'); | |
MODx.ux.Ace = Ext.extend(Ext.ux.Ace, { | |
mimeType : 'text/plain', | |
theme : MODx.config['ace.theme'] || 'textmate', | |
fontSize : MODx.config['ace.font_size'] || '13px', | |
useWrapMode : MODx.config['ace.word_wrap'] == true, | |
useSoftTabs : MODx.config['ace.soft_tabs'] == true, | |
tabSize : MODx.config['ace.tab_size'] * 1 || 4, | |
showFoldWidgets : MODx.config['ace.fold_widgets'] == true, | |
showInvisibles : MODx.config['ace.show_invisibles'] == true, | |
modxTags : false, | |
initComponent : function() { | |
MODx.ux.Ace.superclass.initComponent.call(this); | |
var config = ace.require("ace/config"); | |
var acePath = MODx.config['assets_url'] + 'components/ace/ace'; | |
config.set('basePath', acePath); | |
config.set('modePath', acePath); | |
config.set('themePath', acePath); | |
config.set('workerPath', acePath); | |
if(MODx.config['ace.grow']!==undefined&&MODx.config['ace.grow']!==''){ | |
this.grow = true; | |
this.growMax = parseInt(MODx.config['ace.grow'])||Infinity; | |
this.growMin = this.height; | |
} | |
this.windows = []; | |
}, | |
onRender : function (ct, position) { | |
MODx.ux.Ace.superclass.onRender.call(this, ct, position); | |
var TokenIterator = ace.require("ace/token_iterator").TokenIterator; | |
var userAgent = ace.require("ace/lib/useragent"); | |
var shortcut = (userAgent.isMac ? 'Command + F12' : 'Ctrl + F11'); | |
this.maximizeTitle = _('ui_ace.maximize') + ' (' + shortcut + ')'; | |
this.minimizeTitle = _('ui_ace.minimize') + ' (' + shortcut + ')'; | |
this.maximizer = document.createElement('div'); | |
this.maximizer.title = this.maximizeTitle; | |
this.maximizer.className = 'ace_maximizer'; | |
this.maximizer.onmousedown = function(event) { | |
event.preventDefault(); | |
}; | |
this.maximizer.onclick = function(event) { | |
this.fullScreen(); | |
event.preventDefault(); | |
}.bind(this); | |
this.editor.renderer.scroller.appendChild(this.maximizer); | |
this.setMimeType(this.mimeType); | |
if (!MODx.ux.Ace.initialized) { | |
var style = "\ | |
.ace_maximized {position: fixed; border: none; top: 0; left: 0; right: 0; bottom: 0; width: auto !important; height: auto !important;z-index: 100}\ | |
.ace_maximizer {position: absolute; width: 16px; height: 16px; top: 3px; right: 3px; opacity: 0.7; z-index: 10; background: url()}\ | |
.ace_maximizer:hover {opacity: 1}\ | |
"; | |
new MODx.ux.Ace.CodeCompleter(); | |
var dom = ace.require("ace/lib/dom"); | |
dom.importCssString(style); | |
var snippetManager = ace.require("ace/snippets").snippetManager; | |
var snippets = MODx.config['ace.snippets'] || ''; | |
snippetManager.register(snippetManager.parseSnippetFile(snippets), "_"); | |
var HashHandler = ace.require("ace/keyboard/hash_handler").HashHandler; | |
var commands = new HashHandler(); | |
commands.addCommand({ | |
name: "insertsnippet", | |
bindKey: {win: "Tab", mac: "Tab"}, | |
exec: function(editor) { | |
return snippetManager.expandWithTab(editor); | |
} | |
}); | |
// to overwrite emmet | |
var onChangeMode = function(e, target) { | |
var editor = target; | |
editor.keyBinding.addKeyboardHandler(commands); | |
}; | |
onChangeMode({}, this.editor); | |
var Emmet = ace.require("ace/ext/emmet"); | |
Emmet.isSupportedMode = function(modeId) { | |
return modeId && /css|less|scss|sass|stylus|html|php|twig|ejs|handlebars|smarty/.test(modeId); | |
}; | |
var net = ace.require('ace/lib/net'); | |
net.loadScript(MODx.config['assets_url'] + 'components/ace/emmet/emmet.js', function() { | |
Emmet.setCore(window.emmet); | |
this.editor.setOption("enableEmmet", true); | |
this.editor.on("changeMode", onChangeMode); | |
onChangeMode({}, this.editor); | |
}.bind(this)); | |
ace.require('ace/ext/keybinding_menu').init(this.editor); | |
MODx.ux.Ace.initialized = true; | |
} | |
this.editor.commands.addCommand({ | |
name: "showKeyboardShortcuts", | |
bindKey: {win: "Ctrl-Alt-H", mac: "Command-Alt-H"}, | |
exec: function(editor) { | |
editor.showKeyboardShortcuts(); | |
}, | |
readOnly: true | |
}); | |
this.editor.commands.addCommand({ | |
name: "gotoline", | |
bindKey: {win: "Ctrl-L", mac: "Command-Option-L"}, | |
exec: this.showGotoLineWindow.bind(this), | |
readOnly: true | |
}); | |
this.editor.commands.addCommand({ | |
name: "fullscreen", | |
bindKey: {win: "Ctrl-F11", mac: "Command-F12"}, | |
exec: this.fullScreen.bind(this), | |
readOnly: true | |
}); | |
}, | |
fullScreen : function() { | |
if (this.isFullscreen){ | |
this.maximizer.title = this.maximizeTitle; | |
this.el.removeClass('ace_maximized'); | |
} else { | |
this.el.addClass('ace_maximized'); | |
this.maximizer.title = this.minimizeTitle; | |
} | |
this.isFullscreen = !this.isFullscreen; | |
this.onResize(); | |
}, | |
setMimeType : function (mimeType){ | |
this.setMode( MODx.ux.Ace.mimeTypes[mimeType] || 'text' ); | |
}, | |
showGotoLineWindow : function(){ | |
var window; | |
if (!this.windows.gotoLine){ | |
this.windows.gotoLine = this.createGotoLineWindow(); | |
} | |
window = this.windows.gotoLine; | |
window.show(); | |
}, | |
doGotoLine : function(){ | |
var window, line; | |
window = this.windows.gotoLine; | |
line = window.fp.getForm().getFieldValues('line')['line']; | |
if (!isNaN(line)){ | |
this.editor.gotoLine(line); | |
window.hide(); | |
} | |
}, | |
createGotoLineWindow: function () { | |
var window = MODx.load({ | |
xtype: 'modx-window', | |
title: _('ui_ace.goto_line') | |
,resizable: false | |
,maximizable: false | |
,allowDrop: false | |
,width: 300 | |
,buttons: [{ | |
text: _('ui_ace.go') | |
,scope: this | |
,handler: this.doGotoLine | |
},{ | |
text: _('ui_ace.close') | |
,scope: this | |
,handler: function() { window.hide(); } | |
}] | |
,keys: [{ | |
key: Ext.EventObject.ENTER | |
,fn: this.doGotoLine | |
,scope: this | |
}] | |
,action: 'gotoline' | |
,listeners: { | |
'hide': {fn: this.focus, scope: this} | |
} | |
,fields: [{ | |
xtype: 'textfield' | |
,validator: function (value) { | |
return !isNaN(value); | |
} | |
,fieldLabel: _('ui_ace.goto_line') | |
,name: 'line' | |
,anchor: '100%' | |
,value: '' | |
}] | |
}); | |
return window; | |
}, | |
setMode : function (mode){ | |
var editor = this.editor; | |
if (!this.modxTags) | |
return editor.session.setMode('ace/mode/' + mode); | |
var config = ace.require('ace/config'); | |
config.loadModule(["mode", 'ace/mode/' + mode], function(module) { | |
var mode = MODx.ux.Ace.createModxMixedMode(module.Mode); | |
editor.session.setMode(mode); | |
}.bind(this)); | |
} | |
}); | |
MODx.ux.Ace.replaceComponent = function(id, mimeType, modxTags) { | |
var textArea = Ext.getCmp(id); | |
if (!textArea) { | |
// Workaround for File Update panel (fix issue, caused by wrong event order) | |
return setTimeout(function() { | |
var textArea = Ext.getCmp(id); | |
if (textArea) | |
MODx.ux.Ace.replaceComponent(id, mimeType, modxTags); | |
}); | |
} | |
var textEditor = MODx.load({ | |
xtype: 'modx-texteditor', | |
enableKeyEvents: true, | |
anchor: textArea.anchor, | |
width: 'auto', | |
height: parseInt(MODx.config['ace.height']) || textArea.height, | |
name: textArea.name, | |
value: textArea.getValue(), | |
mimeType: mimeType, | |
modxTags: modxTags | |
}); | |
textArea.el.dom.removeAttribute('name'); | |
textArea.el.setStyle('display', 'none'); | |
textEditor.render(textArea.el.dom.parentNode); | |
textArea.setSize = function(){textEditor.setSize.apply(textEditor, arguments)}; | |
textEditor.editor.on('change', function(e){textArea.fireEvent('change', e);}); | |
textArea.on('destroy', function() {textEditor.destroy();}); | |
if (!modxTags) | |
return; | |
var dropTarget = MODx.load({ | |
xtype: 'modx-treedrop', | |
target: textEditor, | |
targetEl: textEditor.el, | |
onInsert: (function(s){ | |
this.insertAtCursor(s); | |
this.focus(); | |
return true; | |
}).bind(textEditor), | |
iframe: true | |
}); | |
textArea.on('destroy', function() {dropTarget.destroy();}); | |
}; | |
MODx.ux.Ace.replaceTextAreas = function(textAreas, mimeType) { | |
textAreas.forEach(function(textArea){ | |
var editor = MODx.load({ | |
xtype: 'modx-texteditor', | |
width: 'auto', | |
height: parseInt(textArea.style.height) || 200, | |
name: textArea.name, | |
value: textArea.value, | |
mimeType: mimeType || 'text/html', | |
modxTags: true | |
}); | |
textArea.name = ''; | |
textArea.style.display = 'none'; | |
editor.render(textArea.parentNode); | |
editor.editor.on('change', function(e){ MODx.fireResourceFormChange() }); | |
}); | |
}; | |
MODx.ux.Ace.createModxMixedMode = function(Mode) { | |
function ModxMixedMode() { | |
Mode.call(this); | |
var HighlightRules = this.HighlightRules; | |
function ModxMixedHighlightRules() { | |
HighlightRules.call(this); | |
this.$rules['modxtag-comment'] = [ | |
{ | |
token : "comment.modx", | |
regex : "[^\\[\\]]+", | |
merge : true | |
},{ | |
token : "comment.modx", | |
regex : "\\[\\[\\-.*?\\]\\]" | |
},{ | |
token : "comment.modx", | |
regex : "\\s+", | |
merge : true | |
}, | |
{ | |
token : "paren.rparen.comment.modx", | |
regex : "\\]\\]", | |
next: "pop" | |
} | |
]; | |
this.$rules['modxtag-start'] = [ | |
{ | |
token : ["cache-flag.variable.modx", "tag-token.variable.modx", "tag-name.variable.modx"], | |
regex : "(!)?([%|*|~|\\+|\\$]|(?:\\+\\+)|(?:\\*#))?([-_a-zA-Z0-9\\.]+)", | |
push : [ | |
{include: "modxtag-filter"}, | |
{ | |
token: "tag-delimiter.keyword.operator.modx", | |
regex: "\\?", | |
push: [ | |
{token : "text.modx", regex : "\\s+"}, | |
{include: 'modxtag-property-string'}, | |
{token: "", regex: "$"}, | |
{token: '', regex: '', next: 'pop'} | |
] | |
}, | |
{token : "text.modx", regex : "\\s+"}, | |
{token: "", regex: "$"}, | |
{token: '', regex: '', next: 'pop'} | |
] | |
}, | |
{ | |
token : "support.constant.paren.lparen.modx", // opening tag | |
regex : "\\[\\[", | |
push : 'modxtag-start' | |
}, | |
{ | |
token : "text", | |
regex : "\\s+" | |
}, | |
{ | |
token : "support.constant.paren.rparen.tag-brackets.modx", | |
regex : "\\]\\]", | |
next: "pop" | |
}, | |
{defaultToken: 'text.modx'} | |
]; | |
this.$rules['modxtag-propertyset'] = [ | |
{ | |
token : ['keyword.operator.modx', "support.class.modx"], | |
regex : "(@)([-_a-zA-Z0-9\\.]+|\\[\\[.*?\\]\\])", | |
next : 'modxtag-filter' | |
}, | |
{ | |
token : "text", | |
regex : "\\s+" | |
}, | |
{token: "", regex: "$"}, | |
{ | |
token: "empty", | |
regex: "", | |
next: "modxtag-filter" | |
} | |
]; | |
this.$rules['modxtag-filter'] = [ | |
{ | |
token : 'filter-delimiter.keyword.operator.modx', | |
regex : ":", | |
push : [ | |
{ | |
token: "filter-name.support.function.modx", | |
regex: "[-_a-zA-Z0-9]+|\\[\\[.*?\\]\\]", | |
push: "modxtag-filter-eq" | |
}, | |
{ | |
token: "empty", | |
regex: "", | |
next: "pop" | |
} | |
] | |
}, | |
{ | |
token : "text", | |
regex : "\\s+" | |
} | |
]; | |
this.$rules['modxtag-filter-eq'] = [ | |
{ | |
token : ["keyword.operator.modx"], | |
regex : "=" | |
},{ | |
token : 'string', | |
regex : '`', | |
push: "modxtag-filter-value" | |
}, | |
{ | |
token : "text", | |
regex : "\\s+" | |
}, | |
{ | |
token: "empty", | |
regex: "", | |
next: "pop" | |
} | |
]; | |
this.$rules["modxtag-property-string"] = [ | |
{ | |
token : "entity.other.attribute-name.modx", | |
regex: "&" | |
}, | |
{ | |
token: "entity.other.attribute-name.modx", | |
regex: "[-_a-zA-Z0-9]+" | |
}, | |
{ | |
token : "string.modx", | |
regex : '`', | |
push : "modxtag-attribute-value" | |
}, { | |
token : "keyword.operator.modx", | |
regex : "=" | |
}, { | |
token : "entity.other.attribute-name.modx", | |
regex : "[-_a-zA-Z0-9]+" | |
}, | |
{ | |
token : "comment.modx", | |
regex : "\\[\\[\\-.*?\\]\\]" | |
}, | |
{ | |
token : "property-string.text.modx", | |
regex : "\\s+" | |
} | |
]; | |
this.$rules["modxtag-attribute-value"] = [ | |
{ | |
token : "string.modx", | |
regex : "[^`\\[]+", | |
merge : true | |
},{ | |
token : "string.modx", | |
regex : "[^`]+", | |
merge : true | |
},/* { | |
token : "string", | |
regex : "\\\\$", | |
next : "modxtag-attribute-value", | |
merge : true | |
},*/ { | |
token : "string.modx", | |
regex : "`", | |
next : "pop", | |
merge : true | |
} | |
]; | |
this.$rules["modxtag-filter-value"] = [ | |
{ | |
token : "string.modx", | |
regex : "[^`\\[]+", | |
merge : true | |
},{ | |
token : "string.modx", | |
regex : "\\[\\[.*?\\]\\]", | |
merge : true | |
}, { | |
token : "string.modx", | |
regex : "\\\\$", | |
next : "pop", | |
merge : true | |
}, { | |
token : "string.modx", | |
regex : "`", | |
next : "pop", | |
merge : true | |
} | |
]; | |
// add twig start tags to the HTML start tags | |
for (var rule in this.$rules) { | |
this.$rules[rule].unshift({ | |
token : "paren.lparen.comment.modx", // opening tag | |
regex : "\\[\\[\\-", | |
push : 'modxtag-comment', | |
merge: true | |
}, { | |
token : "support.constant.paren.lparen.tag-brackets.modx", // opening tag | |
regex : "\\[\\[", | |
push : 'modxtag-start', | |
merge : false | |
}); | |
} | |
this.normalizeRules(); | |
} | |
ModxMixedHighlightRules.prototype = HighlightRules.prototype; | |
this.HighlightRules = ModxMixedHighlightRules; | |
if (typeof this.$behaviour == 'undefined') { | |
var Behaviour = ace.require("ace/mode/behaviour").Behaviour; | |
} | |
this.$behaviour = Object.create(this.$behaviour || new Behaviour()); | |
this.$behaviour.add("brackets", "insertion", function (state, action, editor, session, text) { | |
if (text == '[') { | |
var selection = editor.getSelectionRange(); | |
var selected = session.doc.getTextRange(selection); | |
if (selected !== "") { | |
return { | |
text: '[' + selected + ']', | |
selection: false | |
}; | |
} else { | |
return { | |
text: '[]', | |
selection: [1, 1] | |
}; | |
} | |
} else if (text == ']') { | |
var cursor = editor.getCursorPosition(); | |
var line = session.doc.getLine(cursor.row); | |
var rightChar = line.substring(cursor.column, cursor.column + 1); | |
if (rightChar == ']') { | |
var matching = session.$findOpeningBracket(']', {column: cursor.column + 1, row: cursor.row}); | |
if (matching !== null) { | |
return { | |
text: '', | |
selection: [1, 1] | |
}; | |
} | |
} | |
} | |
}); | |
this.$behaviour.add("brackets", "deletion", function (state, action, editor, session, range) { | |
var selected = session.doc.getTextRange(range); | |
if (!range.isMultiLine() && selected == '[') { | |
var line = session.doc.getLine(range.start.row); | |
var rightChar = line.substring(range.start.column + 1, range.start.column + 2); | |
if (rightChar == ']') { | |
range.end.column++; | |
return range; | |
} | |
} | |
}); | |
this.$behaviour.add("string_apostrophes", "insertion", function (state, action, editor, session, text) { | |
if (text == '`') { | |
var quote = "`"; | |
var selection = editor.getSelectionRange(); | |
var selected = session.doc.getTextRange(selection); | |
if (selected !== "") { | |
return { | |
text: quote + selected + quote, | |
selection: false | |
}; | |
} else { | |
var cursor = editor.getCursorPosition(); | |
var line = session.doc.getLine(cursor.row); | |
var leftChar = line.substring(cursor.column-1, cursor.column); | |
// Find what token we're inside. | |
var tokens = session.getTokens(selection.start.row); | |
var col = 0, token; | |
var quotepos = -1; // Track whether we're inside an open quote. | |
for (var x = 0; x < tokens.length; x++) { | |
token = tokens[x]; | |
if (token.type == "string.modx") { | |
quotepos = -1; | |
} else if (quotepos < 0) { | |
quotepos = token.value.indexOf(quote); | |
} | |
if ((token.value.length + col) > selection.start.column) { | |
break; | |
} | |
col += tokens[x].value.length; | |
} | |
// Try and be smart about when we auto insert. | |
if (!token || (quotepos < 0 && token.type !== "comment" && (token.type !== "string.modx" || ((selection.start.column !== token.value.length+col-1) && token.value.lastIndexOf(quote) === token.value.length-1)))) { | |
return { | |
text: quote + quote, | |
selection: [1,1] | |
}; | |
} else if (token && token.type === "string.modx") { | |
// Ignore input and move right one if we're typing over the closing quote. | |
var rightChar = line.substring(cursor.column, cursor.column + 1); | |
if (rightChar == quote) { | |
return { | |
text: '', | |
selection: [1, 1] | |
}; | |
} | |
} | |
} | |
} | |
}); | |
this.$behaviour.add("string_apostrophes", "deletion", function (state, action, editor, session, range) { | |
var selected = session.doc.getTextRange(range); | |
if (!range.isMultiLine() && (selected == '`')) { | |
var line = session.doc.getLine(range.start.row); | |
var rightChar = line.substring(range.start.column + 1, range.start.column + 2); | |
if (rightChar == '`') { | |
range.end.column++; | |
return range; | |
} | |
} | |
}); | |
} | |
ModxMixedMode.prototype = Object.create(Mode.prototype, { | |
constructor: {value: ModxMixedMode} | |
}); | |
return new ModxMixedMode(); | |
}; | |
MODx.ux.Ace.mimeTypes = { | |
'text/x-smarty' : 'smarty', | |
'text/html' : 'html', | |
'application/xhtml+xml' : 'html', | |
'text/css' : 'css', | |
'text/x-scss' : 'scss', | |
'text/x-less' : 'less', | |
'image/svg+xml' : 'svg', | |
'application/xml' : 'xml', | |
'text/xml' : 'xml', | |
'text/javascript' : 'javascript', | |
'application/javascript': 'javascript', | |
'application/json' : 'json', | |
'text/x-php' : 'php', | |
'application/x-php' : 'php', | |
'text/x-sql' : 'sql', | |
'text/x-markdown' : 'markdown', | |
'text/plain' : 'text', | |
'text/x-twig' : 'twig' | |
}; | |
MODx.ux.Ace.initialized = false; | |
MODx.ux.Ace.CodeCompleter = function() { | |
var TokenIterator = ace.require("ace/token_iterator").TokenIterator; | |
var langTools = ace.require("ace/ext/language_tools"); | |
var cache = {}; | |
function loadCompletions(params, callback) { | |
Ext.Ajax.request({ | |
url: MODx.config.assets_url + 'components/ace/completions.php', | |
params: params, | |
success: function(response) { | |
var completions = JSON.parse(response.responseText); | |
callback(completions); | |
} | |
}); | |
} | |
function gatherCompletions(completionParameters, callback) { | |
var wait = 0; | |
var completions = []; | |
completionParameters.forEach(function(parameters){ | |
var data = cache[parameters.cacheKey]; | |
if (!data) { | |
wait++; | |
loadCompletions(parameters.requestParams, function(data) { | |
wait--; | |
cache[parameters.cacheKey] = data; | |
completions = completions.concat(parameters.prepare(data)); | |
wait || callback(null, completions); | |
}); | |
return; | |
} | |
completions = completions.concat(parameters.prepare(data)); | |
}); | |
wait || callback(null, completions); | |
} | |
function prepareCompletions(completions, meta) { | |
return Object.keys(completions).map(function(completion){ | |
return { | |
value: meta == 'chunk' ? '$' + completion : completion, | |
caption: completion, | |
meta: meta == 'function' ? completions[completion] : meta, | |
description: completions[completion], | |
score: 1000 | |
}; | |
}); | |
} | |
function preparePropertyCompletions(completions) { | |
return Object.keys(completions).map(function(completion){ | |
return { | |
caption: completion, | |
snippet: completion + '=`$0`', | |
meta: 'property', | |
description: completions[completion], | |
score: 1000 | |
}; | |
}); | |
} | |
function hasType(token, type) { | |
var tokenTypes = token.type.split('.'); | |
return type.split('.').every(function(type){ | |
return (tokenTypes.indexOf(type) !== -1); | |
}); | |
} | |
function isWhitespace(string) { | |
for (var i = 0; i < string.length; i++) { | |
var c = string[i]; | |
if (!(c == ' ' || c == '\n' || c == '\r')) { | |
return false; | |
} | |
} | |
return true; | |
} | |
function parseTag(iterator) { | |
var token = iterator.getCurrentToken(); | |
if (!token) | |
return null; | |
if (token.type.substring(token.type.lastIndexOf('.') + 1) !== 'modx') | |
return null; | |
while(token && hasType(token, 'text.modx') && isWhitespace(token.value)) | |
{ | |
token = iterator.stepBackward(); | |
} | |
if (!token) | |
return null; | |
// we are in modx tag | |
var completionType = 'object'; | |
var objectName = ''; | |
var classKey = 'modSnippet'; | |
if (hasType(token, 'tag-name')) {// [[*tag|]] | |
objectName = token.value; | |
token = iterator.stepBackward(); | |
} | |
if (hasType(token, 'tag-brackets') && token.value == '[[') {// [[|]] | |
classKey = 'modSnippet'; | |
} else if (hasType(token, 'cache-flag')) { | |
// | |
} else if (hasType(token, 'tag-token') || hasType(token, 'text')) {// [[*|]] | |
switch (token.value) { | |
case '$': | |
classKey = 'modChunk'; | |
break; | |
case '*': | |
classKey = 'modTemplateVar'; | |
break; | |
case '++': | |
classKey = 'modSystemSetting'; | |
break; | |
default: | |
return null; | |
} | |
} else if (hasType(token, 'filter-name') || hasType(token, 'filter-delimiter')) {// [[*tag:filter|]], [[*tag:|]] | |
completionType = 'filter'; | |
} else if (hasType(token, 'attribute-name') || hasType(token, 'tag-delimiter')) {// [[*tag?|]] , [[*tag? &prop|]] | |
objectName = (function() { | |
do { | |
token = iterator.stepBackward(); | |
} while (token && !(hasType(token, 'tag-name') || hasType(token, 'modxtag-start'))); | |
if (token && hasType(token, 'tag-name')) | |
return token.value; | |
return null; | |
})(); | |
if (!objectName) | |
return null; | |
completionType = 'property'; | |
} else { | |
return null; | |
} | |
return { | |
completionType: completionType, | |
classKey: classKey, | |
objectName: objectName, | |
}; | |
} | |
langTools.addCompleter({ | |
getCompletions: function(editor, session, pos, prefix, callback) { | |
var iterator = new TokenIterator(session, pos.row, pos.column), | |
parsedInfo = parseTag(iterator), | |
completionType = 'function', | |
classKey, objectName; | |
if (parsedInfo) { | |
completionType = parsedInfo.completionType; | |
classKey = parsedInfo.classKey; | |
objectName = parsedInfo.objectName; | |
} | |
switch (completionType) { | |
case 'function' : | |
gatherCompletions([ | |
{ | |
cacheKey: 'function', | |
requestParams: {action: 'getFunctions'}, | |
prepare: function(completions) { | |
return prepareCompletions(completions, 'function'); | |
} | |
} | |
], callback); | |
break; | |
case 'propertyset': | |
break; | |
case 'lexiconentry': | |
break; | |
case 'property': | |
gatherCompletions([ | |
{ | |
cacheKey: classKey + '.' + objectName, | |
requestParams: {action: 'getProperties', classKey: classKey, key: objectName}, | |
prepare: function(completions) { | |
return preparePropertyCompletions(completions, 'property'); | |
} | |
} | |
], callback); | |
break; | |
case 'filter': | |
gatherCompletions([ | |
{ | |
cacheKey: 'filter', | |
requestParams: {action: 'getFilters'}, | |
prepare: function(completions) { | |
return prepareCompletions(completions, 'filter'); | |
} | |
}, { | |
cacheKey: 'modSnippet', | |
requestParams: {action: 'getObjects', classKey: 'modSnippet'}, | |
prepare: function(completions) { | |
return prepareCompletions(completions, 'snippet'); | |
} | |
}, | |
], callback); | |
break; | |
case 'object': | |
var aliases = { | |
'modSystemSetting': 'setting', | |
'modTemplateVar': 'tv', | |
'modSnippet': 'snippet', | |
'modChunk': 'chunk' | |
}; | |
alias = aliases[classKey]; | |
var completionParameters = []; | |
completionParameters[0] = { | |
cacheKey: classKey, | |
requestParams: {action: 'getObjects', classKey: classKey}, | |
prepare: function(completions) { | |
return prepareCompletions(completions, alias); | |
} | |
}; | |
if (classKey == 'modTemplateVar') { | |
completionParameters[1] = { | |
cacheKey: 'resourcefield', | |
requestParams: {action: 'getResourceFields'}, | |
prepare: function(completions) { | |
return prepareCompletions(completions, 'field'); | |
} | |
}; | |
} | |
gatherCompletions(completionParameters, callback); | |
break; | |
} | |
} | |
}); | |
}; | |
Ext.reg('modx-texteditor',MODx.ux.Ace); |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<div id="tv{$tv->id}"></div> | |
<script type="text/javascript"> | |
{literal} | |
Ext.onReady(function(tvace) { | |
var textArea = Ext.getCmp(tvace); | |
var TextEditor = MODx.load({ | |
xtype: 'modx-texteditor' | |
{/literal} | |
,anchor: '100%' | |
,name: 'tv{$tv->id}' | |
,renderTo: 'tv{$tv->id}' | |
,value: '{$tv->get('value')|escape:'javascript'}' | |
,height: {literal} parseInt(MODx.config['ace.height']) || 200 {/literal} | |
,width: 'auto' | |
,enableKeyEvents: true | |
,msgTarget: 'under' | |
,allowBlank: {if $params.allowBlank == 1 || $params.allowBlank == 'true'}true{else}false{/if} | |
,listeners: {literal} { 'keydown': { fn:MODx.fireResourceFormChange, scope:this}} {/literal} | |
,mimeType: 'text/x-smarty' | |
,modxTags: true | |
{literal} | |
}); | |
MODx.load({ | |
xtype: 'modx-treedrop', | |
target: TextEditor, | |
targetEl: TextEditor.el, | |
onInsert: (function(s){ | |
this.insertAtCursor(s); | |
this.focus(); | |
return true; | |
}).bind(TextEditor), | |
iframe: true | |
}); | |
textArea.name = ''; | |
textArea.style.display = 'none'; | |
TextEditor.render(textArea.parentNode); | |
}); | |
{/literal} | |
</script> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Uh oh!
There was an error while loading. Please reload this page.