var CloudCmd, Util, DOM, CloudFunc, ace, DiffProto, diff_match_patch, Zip, MenuIO; (function(CloudCmd, Util, DOM, CloudFunc) { 'use strict'; CloudCmd.Edit = EditProto; function EditProto() { var Name = 'Edit', Loading = false, DIR = CloudCmd.LIBDIRCLIENT + 'edit/', LIBDIR = CloudCmd.LIBDIR, Info = DOM.CurrentInfo, Menu, Value, Edit = this, Diff, Emmet, Ace, Session, Modelist, Msg, Events = DOM.Events, RESTful = DOM.RESTful, Dialog = DOM.Dialog, Images = DOM.Images, Element, JSHintConfig, EditConfig, ConfigView = { beforeClose: function() { isChanged(); if (Menu) Menu.hide(); }, afterShow: function() { Ace.clearSelection(); Ace.moveCursorTo(0, 0); Ace.focus(); } }; function init() { Loading = true; Util.exec.series([ CloudCmd.View, load, Edit.show ]); } this.show = function(value) { var mode, htmlMode, jsMode, isHTML, isJS, modesByName, onMenu, isStr = Util.isString(value), name = Info.name, isDir = Info.isDir; if (!Loading) { Images.showLoad(); if (!Element) { Element = DOM.load({ name : 'div', style : 'width : 100%;' + 'height : 100%;' + 'font : 16px "Droid Sans Mono";' + 'position : absolute;', notAppend : true }); onMenu = Util.exec.with(setMenu, Element); initAce(); Events.addOnce('contextmenu', Element, onMenu) .add(Element, { drop : onDrop, dragover : DOM.preventDefault }); } modesByName = Modelist.modesByName; if (isDir) mode = modesByName.json.mode; else mode = Modelist.getModeForPath(name).mode; htmlMode = modesByName.html.mode; jsMode = modesByName.javascript.mode; isHTML = mode === htmlMode; isJS = mode === jsMode; Session.setMode(mode); setUseOfWorker(mode); setEmmet(isHTML); Util.exec.if(isStr, function() { var UndoManager = ace.require('ace/undomanager').UndoManager; Ace.setValue(Value); CloudCmd.View.show(Element, ConfigView); Session.setUndoManager(new UndoManager()); }, function(callback) { Info.getData(function(error, data) { Value = data; if (isJS && Session.getUseWorker()) setJsHintConfig(); callback(); }); }); } }; this.hide = function() { CloudCmd.View.hide(); }; this.goToLine = function() { var msg = 'Enter line number:', cursor = Ace.selection.getCursor(), number = cursor.row + 1, line = Dialog.prompt(msg, number); number = line - 0; if (number) Ace.gotoLine(number); }; function isChanged() { var is, value = Ace.getValue(), isEqual = value === Value, msg = 'Do you want to save changes to ' + name + '?'; if (!isEqual) { is = Dialog.confirm(msg); if (is) save(); } } function setEmmet(isHTML) { getEditConfig(function(config) { var extensions = config.extensions, isEmmet = extensions.emmet; if (isEmmet) Util.exec.if(Emmet, function() { if (Emmet) Ace.setOption('enableEmmet', isHTML); }, function(callback) { var url; if (!isHTML) Util.exec(callback); else { url = CloudFunc.getJoinURL([ DIR + 'emmet.js', DIR + 'ext-emmet.js' ]); DOM.load.js(url, function() { Emmet = ace.require('ace/ext/emmet'); Emmet.setCore(window.emmet); Util.exec(callback); }); } }); }); } function setUseOfWorker(mode) { var isMatch, isStr = Util.isString(mode), regStr = 'coffee|css|html|javascript|json|lua|php|xquery', regExp = new RegExp(regStr); if (isStr) isMatch = mode.match(regExp); Session.setUseWorker(isMatch); } function initAce() { Ace = ace.edit(Element); Session = Ace.getSession(); Ace.commands.addCommand({ name : 'hide', bindKey : { win: 'Esc', mac: 'Esc' }, exec : function () { Edit.hide(); } }); Ace.commands.addCommand({ name : 'goToLine', bindKey : { win: 'Ctrl-G', mac: 'Command-G' }, exec : function () { Edit.goToLine(); } }); Ace.commands.addCommand({ name : 'save', bindKey : { win: 'Ctrl-S', mac: 'Command-S' }, exec : save }); Ace.commands.addCommand({ name : 'saveMC', bindKey : { win: 'F2', mac: 'F2' }, exec : save }); ace.require('ace/ext/language_tools'); Modelist = ace.require('ace/ext/modelist'); getEditConfig(function(config) { var options = config.options; Ace.setOptions(options); }); } function save () { var path = Info.path, value = Ace.getValue(); CloudCmd.getConfig(function(error, config) { var isDiff = config.diff, isZip = config.zip; Util.exec.if(!isDiff, function(patch) { var query = '', MAX_SIZE = CloudFunc.MAX_FILE_SIZE, patchLength = patch.length, length = Value.length, isLessMaxLength = length < MAX_SIZE, isLessLength = isLessMaxLength && patchLength < length, isStr = Util.isString(patch); Value = value; if (isStr && patch && isLessLength) query = '?patch'; else patch = false; Util.exec.if(!isZip || query, function(equal, data) { var result = data || patch || Value, url = path + query; RESTful.write(url, result , onSave); }, function(func) { zip(value, function(error, data) { if (error) Util.log(error); query = '?unzip'; func(null, data); }); }); }, Util.exec.with(doDiff, path)); }); } function doDiff(path, callback) { var value = Ace.getValue(); diff(value, function(patch) { var isAllowed = DOM.Storage.isAllowed(); Util.exec.if(!isAllowed, callback, function(func) { DOM.checkStorageHash(path, function(error, equal) { if (!equal) patch = ''; func(patch); }); }); }); } function diff(newValue, callback) { var libs = [ LIBDIR + 'diff/diff-match-patch.js', LIBDIR + 'diff.js' ], url = CloudFunc.getJoinURL(libs); DOM.load.js(url, function() { var patch, isAllowed = DOM.Storage.isAllowed(); if (!Diff) Diff = new DiffProto(diff_match_patch); Util.exec.if(!isAllowed, function() { patch = Diff.createPatch(Value, newValue); Util.exec(callback, patch); }, function(func) { var path = Info.path; DOM.getDataFromStorage(path, function(error, data) { if (data) Value = data; func(); }); }); }); } function zip(value, callback) { var dir = CloudCmd.LIBDIRCLIENT, libs = [ dir + 'zip/dist/pako.js', dir + 'zip.js' ], url = CloudFunc.getJoinURL(libs); DOM.load.js(url, function() { Zip.pack(value, callback); }); } function setJsHintConfig(callback) { var JSHINT_PATH = '/.jshintrc', func = function() { var worker = Session.$worker; if (worker) worker.send('changeOptions', [JSHintConfig]); Util.exec(callback); }; Util.exec.if(JSHintConfig, func, function() { DOM.load.ajax({ url : JSHINT_PATH, success : function(data) { JSHintConfig = Util.parseJSON(data); func(); } }); }); } function getEditConfig(callback) { var CONFIG_PATH = '/json/edit.json', func = Util.exec.ret(callback); Util.exec.if(EditConfig, function() { func(EditConfig); }, function() { DOM.load.ajax({ url : CONFIG_PATH, success : function(data) { EditConfig = data; func(EditConfig); } }); }); } function setMenu(element) { DOM.loadMenu(function() { var position = CloudCmd.MousePosition, options = { beforeShow: function(params) { params.x -= 18; params.y -= 27; }, afterClick: function() { Ace.focus(); } }, menuData = { 'Save Ctrl+S' : save, 'Go To Line Ctrl+G' : function() { Edit.goToLine(); }, 'Select All Ctrl+A' : function() { Ace.selectAll(); }, 'Delete Del' : function() { Ace.remove('right'); }, 'Close Esc' : Edit.hide }; Menu = new MenuIO(element, options, menuData); Menu.show(position.x, position.y); }); } function load(callback) { var files = [ DIR + 'theme-tomorrow_night_blue.js', DIR + 'ext-language_tools.js', DIR + 'ext-searchbox.js', DIR + 'ext-modelist.js' ], ace = DIR + 'ace.js', url = CloudFunc.getJoinURL(files); Util.time(Name + ' load'); DOM.loadRemote('ace', function() { DOM.load.js([url], function() { Util.timeEnd(Name + ' load'); Loading = false; Util.exec(callback); }); }); } function onSave(text) { var ret, isError = Util.isContainStrAtBegin(text, 'error'), path = Info.path, msg = '\nShould I save file anyway?'; if (!isError) { Edit.showMessage(text); DOM.saveDataToStorage(path, Value); DOM.setCurrentSize(Value.length); } else { ret = Dialog.confirm(text + msg); if (ret) RESTful.write(path, Value, onSave); } } function onDrop(event) { var reader, files, onLoad = function(event) { var data = event.target.result; Ace.setValue(data); }; event.preventDefault(); files = event.dataTransfer.files; Util.forEach(files, function(file) { reader = new FileReader(); Events.addLoad(reader, onLoad); reader.readAsBinaryString(file); }); } this.showMessage = function(text) { var HIDE_TIME = 2000; if (Msg) { Msg.innerHTML = text; } else { DOM.load.style({ id : 'msg-css', inner : '#js-view .msg {' + 'z-index' + ': 1;' + 'background-color' + ': #7285B7;' + 'color' + ': #D1F1A9;' + 'position' + ': fixed;' + 'left' + ': 40%;' + 'top' + ': 25px;' + 'padding' + ': 5px;' + 'opacity' + ': 0.9;' + 'transition' + ': ease 0.5s;' + '}' }); Msg = DOM.load({ name : 'div', className : 'msg', parent : Element, inner : text }); } DOM.show(Msg); setTimeout(function() { DOM.hide(Msg); }, HIDE_TIME); }; init(); } })(CloudCmd, Util, DOM, CloudFunc);