| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675 |
- <!DOCTYPE html>
- <html lang="en">
- <head>
- <title>SD Editor</title>
- <style type="text/css" media="screen">
- .contextMenu {
- z-index: 300;
- position: absolute;
- left: 5px;
- border: 1px solid #444;
- background-color: #F5F5F5;
- display: none;
- box-shadow: 0 0 10px rgba( 0, 0, 0, .4 );
- font-size: 12px;
- font-family: sans-serif;
- font-weight:bold;
- }
- .contextMenu ul {
- list-style: none;
- top: 0;
- left: 0;
- margin: 0;
- padding: 0;
- }
- .contextMenu li {
- position: relative;
- min-width: 60px;
- cursor: pointer;
- }
- .contextMenu span {
- color: #444;
- display: inline-block;
- padding: 6px;
- }
- .contextMenu li:hover { background: #444; }
- .contextMenu li:hover span { color: #EEE; }
-
- .css-treeview ul, .css-treeview li {
- padding: 0;
- margin: 0;
- list-style: none;
- }
- .css-treeview input {
- position: absolute;
- opacity: 0;
- }
- .css-treeview {
- font: normal 11px Verdana, Arial, Sans-serif;
- -moz-user-select: none;
- -webkit-user-select: none;
- user-select: none;
- }
- .css-treeview span {
- color: #00f;
- cursor: pointer;
- }
- .css-treeview span:hover {
- text-decoration: underline;
- }
- .css-treeview input + label + ul {
- margin: 0 0 0 22px;
- }
- .css-treeview input ~ ul {
- display: none;
- }
- .css-treeview label, .css-treeview label::before {
- cursor: pointer;
- }
- .css-treeview input:disabled + label {
- cursor: default;
- opacity: .6;
- }
- .css-treeview input:checked:not(:disabled) ~ ul {
- display: block;
- }
- .css-treeview label, .css-treeview label::before {
- background: url("") no-repeat;
- }
- .css-treeview label, .css-treeview span, .css-treeview label::before {
- display: inline-block;
- height: 16px;
- line-height: 16px;
- vertical-align: middle;
- }
- .css-treeview label {
- background-position: 18px 0;
- }
- .css-treeview label::before {
- content: "";
- width: 16px;
- margin: 0 22px 0 0;
- vertical-align: middle;
- background-position: 0 -32px;
- }
- .css-treeview input:checked + label::before {
- background-position: 0 -16px;
- }
- /* webkit adjacent element selector bugfix */
- @media screen and (-webkit-min-device-pixel-ratio:0)
- {
- .css-treeview{
- -webkit-animation: webkit-adjacent-element-selector-bugfix infinite 1s;
- }
- @-webkit-keyframes webkit-adjacent-element-selector-bugfix
- {
- from {
- padding: 0;
- }
- to {
- padding: 0;
- }
- }
- }
- #uploader {
- position: absolute;
- top: 0;
- right: 0;
- left: 0;
- height:28px;
- line-height: 24px;
- padding-left: 10px;
- background-color: #444;
- color:#EEE;
- }
- #tree {
- position: absolute;
- top: 28px;
- bottom: 0;
- left: 0;
- width:200px;
- padding: 8px;
- }
- #editor, #preview {
- position: absolute;
- top: 28px;
- right: 0;
- bottom: 0;
- left: 200px;
- }
- #preview {
- background-color: #EEE;
- padding:5px;
- }
- </style>
- <script>
- function createFileUploader(element, tree, editor){
- var xmlHttp;
- var input = document.createElement("input");
- input.type = "file";
- input.multiple = false;
- input.name = "data";
- document.getElementById(element).appendChild(input);
- var path = document.createElement("input");
- path.id = "upload-path";
- path.type = "text";
- path.name = "path";
- path.defaultValue = "/";
- document.getElementById(element).appendChild(path);
- var button = document.createElement("button");
- button.innerHTML = 'Upload';
- document.getElementById(element).appendChild(button);
- var mkdir = document.createElement("button");
- mkdir.innerHTML = 'MkDir';
- document.getElementById(element).appendChild(mkdir);
- var mkfile = document.createElement("button");
- mkfile.innerHTML = 'MkFile';
- document.getElementById(element).appendChild(mkfile);
-
- function httpPostProcessRequest(){
- if (xmlHttp.readyState == 4){
- if(xmlHttp.status != 200) alert("ERROR["+xmlHttp.status+"]: "+xmlHttp.responseText);
- else {
- tree.refreshPath(path.value);
- }
- }
- }
- function createPath(p){
- xmlHttp = new XMLHttpRequest();
- xmlHttp.onreadystatechange = httpPostProcessRequest;
- var formData = new FormData();
- formData.append("path", p);
- xmlHttp.open("PUT", "/edit");
- xmlHttp.send(formData);
- }
-
- mkfile.onclick = function(e){
- if(path.value.indexOf(".") === -1) return;
- createPath(path.value);
- editor.loadUrl(path.value);
- };
- mkdir.onclick = function(e){
- if(path.value.length < 2) return;
- var dir = path.value
- if(dir.indexOf(".") !== -1){
- if(dir.lastIndexOf("/") === 0) return;
- dir = dir.substring(0, dir.lastIndexOf("/"));
- }
- createPath(dir);
- };
- button.onclick = function(e){
- if(input.files.length === 0){
- return;
- }
- xmlHttp = new XMLHttpRequest();
- xmlHttp.onreadystatechange = httpPostProcessRequest;
- var formData = new FormData();
- formData.append("data", input.files[0], path.value);
- xmlHttp.open("POST", "/edit");
- xmlHttp.send(formData);
- }
- input.onchange = function(e){
- if(input.files.length === 0) return;
- var filename = input.files[0].name;
- var ext = /(?:\.([^.]+))?$/.exec(filename)[1];
- var name = /(.*)\.[^.]+$/.exec(filename)[1];
- if(typeof name !== undefined){
- if(name.length > 8) name = name.substring(0, 8);
- filename = name;
- }
- if(typeof ext !== undefined){
- if(ext === "html") ext = "htm";
- else if(ext === "jpeg") ext = "jpg";
- filename = filename + "." + ext;
- }
- if(path.value === "/" || path.value.lastIndexOf("/") === 0){
- path.value = "/"+filename;
- } else {
- path.value = path.value.substring(0, path.value.lastIndexOf("/")+1)+filename;
- }
- }
- }
- function createTree(element, editor){
- var preview = document.getElementById("preview");
- var treeRoot = document.createElement("div");
- treeRoot.className = "css-treeview";
- document.getElementById(element).appendChild(treeRoot);
-
- function loadDownload(path){
- document.getElementById('download-frame').src = path+"?download=true";
- }
-
- function loadPreview(path){
- document.getElementById("editor").style.display = "none";
- preview.style.display = "block";
- preview.innerHTML = '<img src="'+path+'" style="max-width:100%; max-height:100%; margin:auto; display:block;" />';
- }
-
- function fillFolderMenu(el, path){
- var list = document.createElement("ul");
- el.appendChild(list);
- var action = document.createElement("li");
- list.appendChild(action);
- var isChecked = document.getElementById(path).checked;
- var expnd = document.createElement("li");
- list.appendChild(expnd);
- if(isChecked){
- expnd.innerHTML = "<span>Collapse</span>";
- expnd.onclick = function(e){
- document.getElementById(path).checked = false;
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- var refrsh = document.createElement("li");
- list.appendChild(refrsh);
- refrsh.innerHTML = "<span>Refresh</span>";
- refrsh.onclick = function(e){
- var leaf = document.getElementById(path).parentNode;
- if(leaf.childNodes.length == 3) leaf.removeChild(leaf.childNodes[2]);
- httpGet(leaf, path);
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- } else {
- expnd.innerHTML = "<span>Expand</span>";
- expnd.onclick = function(e){
- document.getElementById(path).checked = true;
- var leaf = document.getElementById(path).parentNode;
- if(leaf.childNodes.length == 3) leaf.removeChild(leaf.childNodes[2]);
- httpGet(leaf, path);
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- }
- var upload = document.createElement("li");
- list.appendChild(upload);
- upload.innerHTML = "<span>Upload</span>";
- upload.onclick = function(e){
- var pathEl = document.getElementById("upload-path");
- if(pathEl){
- var subPath = pathEl.value;
- if(subPath.lastIndexOf("/") < 1) pathEl.value = path+subPath;
- else pathEl.value = path.substring(subPath.lastIndexOf("/"))+subPath;
- }
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- var delFile = document.createElement("li");
- list.appendChild(delFile);
- delFile.innerHTML = "<span>Delete</span>";
- delFile.onclick = function(e){
- httpDelete(path);
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- }
-
- function fillFileMenu(el, path){
- var list = document.createElement("ul");
- el.appendChild(list);
- var action = document.createElement("li");
- list.appendChild(action);
- if(isTextFile(path)){
- action.innerHTML = "<span>Edit</span>";
- action.onclick = function(e){
- editor.loadUrl(path);
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- } else if(isImageFile(path)){
- action.innerHTML = "<span>Preview</span>";
- action.onclick = function(e){
- loadPreview(path);
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- }
- var download = document.createElement("li");
- list.appendChild(download);
- download.innerHTML = "<span>Download</span>";
- download.onclick = function(e){
- loadDownload(path);
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- var delFile = document.createElement("li");
- list.appendChild(delFile);
- delFile.innerHTML = "<span>Delete</span>";
- delFile.onclick = function(e){
- httpDelete(path);
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(el);
- };
- }
-
- function showContextMenu(e, path, isfile){
- var divContext = document.createElement("div");
- var scrollTop = document.body.scrollTop ? document.body.scrollTop : document.documentElement.scrollTop;
- var scrollLeft = document.body.scrollLeft ? document.body.scrollLeft : document.documentElement.scrollLeft;
- var left = e.clientX + scrollLeft;
- var top = e.clientY + scrollTop;
- divContext.className = 'contextMenu';
- divContext.style.display = 'block';
- divContext.style.left = left + 'px';
- divContext.style.top = top + 'px';
- if(isfile) fillFileMenu(divContext, path);
- else fillFolderMenu(divContext, path);
- document.body.appendChild(divContext);
- var width = divContext.offsetWidth;
- var height = divContext.offsetHeight;
- divContext.onmouseout = function(e){
- if(e.clientX < left || e.clientX > (left + width) || e.clientY < top || e.clientY > (top + height)){
- if(document.body.getElementsByClassName('contextMenu').length > 0) document.body.removeChild(divContext);
- }
- };
- }
-
- function createTreeLeaf(path, name, size){
- var leaf = document.createElement("li");
- leaf.id = (((path == "/")?"":path)+"/"+name).toLowerCase();
- var label = document.createElement("span");
- label.textContent = name.toLowerCase();
- leaf.appendChild(label);
- leaf.onclick = function(e){
- if(isTextFile(leaf.id)){
- editor.loadUrl(leaf.id);
- } else if(isImageFile(leaf.id)){
- loadPreview(leaf.id);
- }
- };
- leaf.oncontextmenu = function(e){
- e.preventDefault();
- e.stopPropagation();
- showContextMenu(e, leaf.id, true);
- };
- return leaf;
- }
-
- function createTreeBranch(path, name, disabled){
- var leaf = document.createElement("li");
- var check = document.createElement("input");
- check.type = "checkbox";
- check.id = (((path == "/")?"":path)+"/"+name).toLowerCase();
- if(typeof disabled !== "undefined" && disabled) check.disabled = "disabled";
- leaf.appendChild(check);
- var label = document.createElement("label");
- label.for = check.id;
- label.textContent = name.toLowerCase();
- leaf.appendChild(label);
- check.onchange = function(e){
- if(check.checked){
- if(leaf.childNodes.length == 3) leaf.removeChild(leaf.childNodes[2]);
- httpGet(leaf, check.id);
- }
- };
- label.onclick = function(e){
- if(!check.checked){
- check.checked = true;
- if(leaf.childNodes.length == 3) leaf.removeChild(leaf.childNodes[2]);
- httpGet(leaf, check.id);
- } else {
- check.checked = false;
- }
- };
- leaf.oncontextmenu = function(e){
- e.preventDefault();
- e.stopPropagation();
- showContextMenu(e, check.id, false);
- }
- return leaf;
- }
-
- function addList(parent, path, items){
- var list = document.createElement("ul");
- parent.appendChild(list);
- var ll = items.length;
- for(var i = 0; i < ll; i++){
- var item = items[i];
- var itemEl;
- if(item.type === "file"){
- itemEl = createTreeLeaf(path, item.name, item.size);
- } else {
- itemEl = createTreeBranch(path, item.name);
- }
- list.appendChild(itemEl);
- }
-
- }
-
- function isTextFile(path){
- var ext = /(?:\.([^.]+))?$/.exec(path)[1];
- if(typeof ext !== undefined){
- switch(ext){
- case "txt":
- case "htm":
- case "html":
- case "js":
- case "json":
- case "c":
- case "h":
- case "cpp":
- case "css":
- case "xml":
- return true;
- }
- }
- return false;
- }
-
- function isImageFile(path){
- var ext = /(?:\.([^.]+))?$/.exec(path)[1];
- if(typeof ext !== undefined){
- switch(ext){
- case "png":
- case "jpg":
- case "gif":
- case "ico":
- return true;
- }
- }
- return false;
- }
-
- this.refreshPath = function(path){
- if(path.lastIndexOf('/') < 1){
- path = '/';
- treeRoot.removeChild(treeRoot.childNodes[0]);
- httpGet(treeRoot, "/");
- } else {
- path = path.substring(0, path.lastIndexOf('/'));
- var leaf = document.getElementById(path).parentNode;
- if(leaf.childNodes.length == 3) leaf.removeChild(leaf.childNodes[2]);
- httpGet(leaf, path);
- }
- };
-
- function delCb(path){
- return function(){
- if (xmlHttp.readyState == 4){
- if(xmlHttp.status != 200){
- alert("ERROR["+xmlHttp.status+"]: "+xmlHttp.responseText);
- } else {
- if(path.lastIndexOf('/') < 1){
- path = '/';
- treeRoot.removeChild(treeRoot.childNodes[0]);
- httpGet(treeRoot, "/");
- } else {
- path = path.substring(0, path.lastIndexOf('/'));
- var leaf = document.getElementById(path).parentNode;
- if(leaf.childNodes.length == 3) leaf.removeChild(leaf.childNodes[2]);
- httpGet(leaf, path);
- }
- }
- }
- }
- }
-
- function httpDelete(filename){
- xmlHttp = new XMLHttpRequest();
- xmlHttp.onreadystatechange = delCb(filename);
- var formData = new FormData();
- formData.append("path", filename);
- xmlHttp.open("DELETE", "/edit");
- xmlHttp.send(formData);
- }
-
- function getCb(parent, path){
- return function(){
- if (xmlHttp.readyState == 4){
- //clear loading
- if(xmlHttp.status == 200) addList(parent, path, JSON.parse(xmlHttp.responseText));
- }
- }
- }
-
- function httpGet(parent, path){
- xmlHttp = new XMLHttpRequest(parent, path);
- xmlHttp.onreadystatechange = getCb(parent, path);
- xmlHttp.open("GET", "/list?dir="+path, true);
- xmlHttp.send(null);
- //start loading
- }
-
- httpGet(treeRoot, "/");
- return this;
- }
- function createEditor(element, file, lang, theme, type){
- function getLangFromFilename(filename){
- var lang = "plain";
- var ext = /(?:\.([^.]+))?$/.exec(filename)[1];
- if(typeof ext !== undefined){
- switch(ext){
- case "txt": lang = "plain"; break;
- case "htm": lang = "html"; break;
- case "js": lang = "javascript"; break;
- case "c": lang = "c_cpp"; break;
- case "cpp": lang = "c_cpp"; break;
- case "css":
- case "scss":
- case "php":
- case "html":
- case "json":
- case "xml":
- lang = ext;
- }
- }
- return lang;
- }
-
- if(typeof file === "undefined") file = "/index.htm";
-
- if(typeof lang === "undefined"){
- lang = getLangFromFilename(file);
- }
-
- if(typeof theme === "undefined") theme = "textmate";
-
- if(typeof type === "undefined"){
- type = "text/"+lang;
- if(lang === "c_cpp") type = "text/plain";
- }
-
- var xmlHttp = null;
- var editor = ace.edit(element);
-
- //post
- function httpPostProcessRequest(){
- if (xmlHttp.readyState == 4){
- if(xmlHttp.status != 200) alert("ERROR["+xmlHttp.status+"]: "+xmlHttp.responseText);
- }
- }
- function httpPost(filename, data, type){
- xmlHttp = new XMLHttpRequest();
- xmlHttp.onreadystatechange = httpPostProcessRequest;
- var formData = new FormData();
- formData.append("data", new Blob([data], { type: type }), filename);
- xmlHttp.open("POST", "/edit");
- xmlHttp.send(formData);
- }
- //get
- function httpGetProcessRequest(){
- if (xmlHttp.readyState == 4){
- document.getElementById("preview").style.display = "none";
- document.getElementById("editor").style.display = "block";
- if(xmlHttp.status == 200) editor.setValue(xmlHttp.responseText);
- else editor.setValue("");
- editor.clearSelection();
- }
- }
- function httpGet(theUrl){
- xmlHttp = new XMLHttpRequest();
- xmlHttp.onreadystatechange = httpGetProcessRequest;
- xmlHttp.open("GET", theUrl, true);
- xmlHttp.send(null);
- }
-
- if(lang !== "plain") editor.getSession().setMode("ace/mode/"+lang);
- editor.setTheme("ace/theme/"+theme);
- editor.$blockScrolling = Infinity;
- editor.getSession().setUseSoftTabs(true);
- editor.getSession().setTabSize(2);
- editor.setHighlightActiveLine(true);
- editor.setShowPrintMargin(false);
- editor.commands.addCommand({
- name: 'saveCommand',
- bindKey: {win: 'Ctrl-S', mac: 'Command-S'},
- exec: function(editor) {
- httpPost(file, editor.getValue()+"", type);
- },
- readOnly: false
- });
- editor.commands.addCommand({
- name: 'undoCommand',
- bindKey: {win: 'Ctrl-Z', mac: 'Command-Z'},
- exec: function(editor) {
- editor.getSession().getUndoManager().undo(false);
- },
- readOnly: false
- });
- editor.commands.addCommand({
- name: 'redoCommand',
- bindKey: {win: 'Ctrl-Shift-Z', mac: 'Command-Shift-Z'},
- exec: function(editor) {
- editor.getSession().getUndoManager().redo(false);
- },
- readOnly: false
- });
- httpGet(file);
- editor.loadUrl = function(filename){
- file = filename;
- lang = getLangFromFilename(file);
- type = "text/"+lang;
- if(lang !== "plain") editor.getSession().setMode("ace/mode/"+lang);
- httpGet(file);
- }
- return editor;
- }
- function onBodyLoad(){
- var vars = {};
- var parts = window.location.href.replace(/[?&]+([^=&]+)=([^&]*)/gi, function(m,key,value) { vars[key] = value; });
- var editor = createEditor("editor", vars.file, vars.lang, vars.theme);
- var tree = createTree("tree", editor);
- createFileUploader("uploader", tree, editor);
- };
- </script>
- <script src="https://cdnjs.cloudflare.com/ajax/libs/ace/1.1.9/ace.js" type="text/javascript" charset="utf-8"></script>
- </head>
- <body onload="onBodyLoad();">
- <div id="uploader"></div>
- <div id="tree"></div>
- <div id="editor"></div>
- <div id="preview" style="display:none;"></div>
- <iframe id=download-frame style='display:none;'></iframe>
- </body>
- </html>
|