586 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			586 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
| <!-- This file has been written by Me-No-Dev
 | |
| see https://github.com/me-no-dev/ and is part
 | |
| of the sample code of ESPAsyncWebServer
 | |
| see https://github.com/me-no-dev/ESPAsyncWebServer -->
 | |
| <!DOCTYPE html>
 | |
| <html lang="en">
 | |
| <head>
 | |
| <title>ESP8266 SPIFFS File Editor</title>
 | |
| <meta name="author" content="Me-No-Dev" />
 | |
| <meta name="Description" content="ESP8266 SPIFFS File Web Editor"/>
 | |
| <link rel="shortcut icon" href="favicon.ico"/>
 | |
| <style type="text/css" media="screen">
 | |
| .cm {
 | |
|   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;
 | |
| }
 | |
| .cm ul {
 | |
|   list-style: none;
 | |
|   top: 0;
 | |
|   left: 0;
 | |
|   margin: 0;
 | |
|   padding: 0;
 | |
| }
 | |
| .cm li {
 | |
|   position: relative;
 | |
|   min-width: 60px;
 | |
|   cursor: pointer;
 | |
| }
 | |
| .cm span {
 | |
|   color: #444;
 | |
|   display: inline-block;
 | |
|   padding: 6px;
 | |
| }
 | |
| .cm li:hover { background: #444; }
 | |
| .cm li:hover span { color: #EEE; }
 | |
| .tvu ul, .tvu li {
 | |
|   padding: 0;
 | |
|   margin: 0;
 | |
|   list-style: none;
 | |
| }
 | |
| .tvu input {
 | |
|   position: absolute;
 | |
|   opacity: 0;
 | |
| }
 | |
| .tvu {
 | |
|   font: normal 12px Verdana, Arial, Sans-serif;
 | |
|   -moz-user-select: none;
 | |
|   -webkit-user-select: none;
 | |
|   user-select: none;
 | |
|   color: #444;
 | |
|   line-height: 16px;
 | |
| }
 | |
| .tvu span {
 | |
|   margin-bottom:5px;
 | |
|   padding: 0 0 0 18px;
 | |
|   cursor: pointer;
 | |
|   display: inline-block;
 | |
|   height: 16px;
 | |
|   vertical-align: middle;
 | |
|   background: url('') no-repeat;
 | |
|   background-position: 0px 0px;
 | |
| }
 | |
| .tvu span:hover {
 | |
|   text-decoration: underline;
 | |
| }
 | |
| @media screen and (-webkit-min-device-pixel-ratio:0){
 | |
|   .tvu{
 | |
|     -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:160px;
 | |
|   padding: 8px;
 | |
|   background-color: #222;
 | |
| }
 | |
| #editor, #preview { 
 | |
|   position: absolute;
 | |
|   top: 28px;
 | |
|   right: 0;
 | |
|   bottom: 0;
 | |
|   left: 160px;
 | |
|   border-left:1px solid #EEE;
 | |
| }
 | |
| #preview {
 | |
|   background-color: #EEE;
 | |
|   padding:5px;
 | |
| }
 | |
| #loader { 
 | |
|   position: absolute;
 | |
|   top: 36%;
 | |
|   right: 40%;
 | |
| }
 | |
| .loader {
 | |
|     z-index: 10000;
 | |
|     border: 8px solid #b5b5b5; /* Grey */
 | |
|     border-top: 8px solid #3498db; /* Blue */
 | |
|     border-bottom: 8px solid #3498db; /* Blue */
 | |
|     border-radius: 50%;
 | |
|     width: 240px;
 | |
|     height: 240px;
 | |
|     animation: spin 2s linear infinite;
 | |
|     display:none;
 | |
| }
 | |
| 
 | |
| @keyframes spin {
 | |
|     0% { transform: rotate(0deg); }
 | |
|     100% { transform: rotate(360deg); }
 | |
| }
 | |
| </style>
 | |
| <script>
 | |
| function ge(a){
 | |
|   return document.getElementById(a);
 | |
| }
 | |
| function ce(a){
 | |
|   return document.createElement(a);
 | |
| }
 | |
| 
 | |
| var QueuedRequester = function () {
 | |
|   this.queue = [];
 | |
|   this.running = false;
 | |
|   this.xmlhttp = null;
 | |
| }
 | |
| QueuedRequester.prototype = {
 | |
|   _request: function(req){
 | |
|     this.running = true;
 | |
|     if(!req instanceof Object) return;
 | |
|     var that = this;
 | |
|     
 | |
|     function ajaxCb(x,d){ return function(){
 | |
|       if (x.readyState == 4){
 | |
|         ge("loader").style.display = "none";
 | |
|         d.callback(x.status, x.responseText);
 | |
|         if(that.queue.length === 0) that.running = false;
 | |
|         if(that.running) that._request(that.queue.shift());
 | |
|       }
 | |
|     }}
 | |
|     
 | |
|     ge("loader").style.display = "block";
 | |
|     
 | |
|     var p = "";
 | |
|     if(req.params instanceof FormData){
 | |
|       p = req.params;
 | |
|     } else if(req.params instanceof Object){
 | |
|       for (var key in req.params) {
 | |
|         if(p === "")
 | |
|           p += (req.method === "GET")?"?":"";
 | |
|         else
 | |
|           p += "&";
 | |
|         p += encodeURIComponent(key+"="+req.params[key]);
 | |
|       };
 | |
|     }
 | |
|     
 | |
|     this.xmlhttp = new XMLHttpRequest();
 | |
|     this.xmlhttp.onreadystatechange = ajaxCb(this.xmlhttp, req);
 | |
|     if(req.method === "GET"){
 | |
|       this.xmlhttp.open(req.method, req.url+p, true);
 | |
|       this.xmlhttp.send();
 | |
|     } else {
 | |
|       this.xmlhttp.open(req.method, req.url, true);
 | |
|       if(p instanceof String)
 | |
|         this.xmlhttp.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
 | |
|       this.xmlhttp.send(p);
 | |
|     }
 | |
|   },
 | |
|   stop: function(){
 | |
|     if(this.running) this.running = false;
 | |
|     if(this.xmlhttp && this.xmlhttp.readyState < 4){
 | |
|       this.xmlhttp.abort();
 | |
|     }
 | |
|   },
 | |
|   add: function(method, url, params, callback){
 | |
|     this.queue.push({url:url,method:method,params:params,callback:callback});
 | |
|     if(!this.running){
 | |
|       this._request(this.queue.shift());
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| var requests = new QueuedRequester();
 | |
| 
 | |
| function createFileUploader(element, tree, editor){
 | |
|   var xmlHttp;
 | |
|   var input = ce("input");
 | |
|   input.type = "file";
 | |
|   input.multiple = false;
 | |
|   input.name = "data";
 | |
|   ge(element).appendChild(input);
 | |
|   var path = ce("input");
 | |
|   path.id = "upload-path";
 | |
|   path.type = "text";
 | |
|   path.name = "path";
 | |
|   path.defaultValue = "/";
 | |
|   ge(element).appendChild(path);
 | |
|   var button = ce("button");
 | |
|   button.innerHTML = 'Upload';
 | |
|   ge(element).appendChild(button);
 | |
|   var mkfile = ce("button");
 | |
|   mkfile.innerHTML = 'Create';
 | |
|   ge(element).appendChild(mkfile);
 | |
| 
 | |
|   var savefile = ce("button");
 | |
|   savefile.innerHTML = ' Save ' ;
 | |
|   ge(element).appendChild(savefile);
 | |
| 
 | |
|   function httpPostProcessRequest(status, responseText){
 | |
|     if(status != 200)
 | |
|       alert("ERROR "+status+": "+responseText);
 | |
|     else
 | |
|       tree.refreshPath(path.value);
 | |
|   }
 | |
|   function createPath(p){
 | |
|     var formData = new FormData();
 | |
|     formData.append("path", p);
 | |
|     requests.add("PUT", "/edit", formData, httpPostProcessRequest);
 | |
|   }
 | |
|   
 | |
|   mkfile.onclick = function(e){
 | |
|     if(path.value.indexOf(".") === -1) return;
 | |
|     createPath(path.value);
 | |
|     editor.loadUrl(path.value);
 | |
|   };
 | |
| 
 | |
|   savefile.onclick = function(e){
 | |
|     editor.execCommand('saveCommand');
 | |
|   };
 | |
|   
 | |
|   button.onclick = function(e){
 | |
|     if(input.files.length === 0){
 | |
|       return;
 | |
|     }
 | |
|     var formData = new FormData();
 | |
|     formData.append("data", input.files[0], path.value);
 | |
|     requests.add("POST", "/edit", formData, httpPostProcessRequest);
 | |
|   };
 | |
|   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){
 | |
|       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 = ge("preview");
 | |
|   var treeRoot = ce("div");
 | |
|   treeRoot.className = "tvu";
 | |
|   ge(element).appendChild(treeRoot);
 | |
| 
 | |
|   function loadDownload(path){
 | |
|     ge('download-frame').src = "/edit?download="+path;
 | |
|   }
 | |
| 
 | |
|   function loadPreview(path){
 | |
|     ge("editor").style.display = "none";
 | |
|     preview.style.display = "block";
 | |
|     preview.innerHTML = '<img src="/edit?edit='+path+'&_cb='+Date.now()+'" style="max-width:100%; max-height:100%; margin:auto; display:block;" />';
 | |
|   }
 | |
| 
 | |
|   function fillFileMenu(el, path){
 | |
|     var list = ce("ul");
 | |
|     el.appendChild(list);
 | |
|     var action = ce("li");
 | |
|     list.appendChild(action);
 | |
|     if(isImageFile(path)){
 | |
|       action.innerHTML = "<span>Preview</span>";
 | |
|       action.onclick = function(e){
 | |
|         loadPreview(path);
 | |
|         if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el);
 | |
|       };
 | |
|     } else if(isTextFile(path)){
 | |
|       action.innerHTML = "<span>Edit</span>";
 | |
|       action.onclick = function(e){
 | |
|         editor.loadUrl(path);
 | |
|         if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el);
 | |
|       };
 | |
|     }
 | |
|     var download = ce("li");
 | |
|     list.appendChild(download);
 | |
|     download.innerHTML = "<span>Download</span>";
 | |
|     download.onclick = function(e){
 | |
|       loadDownload(path);
 | |
|       if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el);
 | |
|     };
 | |
|     var delFile = ce("li");
 | |
|     list.appendChild(delFile);
 | |
|     delFile.innerHTML = "<span>Delete</span>";
 | |
|     delFile.onclick = function(e){
 | |
|       httpDelete(path);
 | |
|       if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el);
 | |
|     };
 | |
|   }
 | |
| 
 | |
|   function showContextMenu(event, path, isfile){
 | |
|     var divContext = ce("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 = event.clientX + scrollLeft;
 | |
|     var top = event.clientY + scrollTop;
 | |
|     divContext.className = 'cm';
 | |
|     divContext.style.display = 'block';
 | |
|     divContext.style.left = left + 'px';
 | |
|     divContext.style.top = top + 'px';
 | |
|     fillFileMenu(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('cm').length > 0) document.body.removeChild(divContext);
 | |
|       }
 | |
|     };
 | |
|   }
 | |
| 
 | |
|   function createTreeLeaf(path, name, size){
 | |
|     var leaf = ce("li");
 | |
|     leaf.id = (((path == "/")?"":path)+"/"+name);
 | |
|     var label = ce("span");
 | |
|     label.innerHTML = name;
 | |
|     leaf.appendChild(label);
 | |
|     leaf.onclick = function(e){
 | |
|       if(isTextFile(leaf.id.toLowerCase())){
 | |
|         editor.loadUrl(leaf.id);
 | |
|       } else if(isImageFile(leaf.id.toLowerCase())){
 | |
|         loadPreview(leaf.id);
 | |
|       }
 | |
|     };
 | |
|     leaf.oncontextmenu = function(e){
 | |
|       e.preventDefault();
 | |
|       e.stopPropagation();
 | |
|       showContextMenu(e, leaf.id, true);
 | |
|     };
 | |
|     return leaf;
 | |
|   }
 | |
| 
 | |
|   function addList(parent, path, items){
 | |
|     var list = ce("ul");
 | |
|     parent.appendChild(list);
 | |
|     var ll = items.length;
 | |
|     for(var i = 0; i < ll; i++){
 | |
|       if(items[i].type === "file")
 | |
|         list.appendChild(createTreeLeaf(path, items[i].name, items[i].size));
 | |
|     }
 | |
| 
 | |
|   }
 | |
| 
 | |
|   function isTextFile(path){
 | |
|     var ext = /(?:\.([^.]+))?$/.exec(path)[1];
 | |
|     if(typeof ext !== undefined){
 | |
|       switch(ext){
 | |
|         case "txt":
 | |
|         case "htm":
 | |
|         case "html":
 | |
|         case "js":
 | |
|         case "css":
 | |
|         case "xml":
 | |
|         case "json":
 | |
|         case "conf":
 | |
|         case "ini":
 | |
|         case "h":
 | |
|         case "c":
 | |
|         case "cpp":
 | |
|         case "php":
 | |
|         case "hex":
 | |
|           return true;
 | |
|       }
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   function isImageFile(path){
 | |
|     var ext = /(?:\.([^.]+))?$/.exec(path)[1];
 | |
|     if(typeof ext !== undefined){
 | |
|       switch(ext){
 | |
|         case "png":
 | |
|         case "jpg":
 | |
|         case "gif":
 | |
|           return true;
 | |
|       }
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   this.refreshPath = function(path){
 | |
|     treeRoot.removeChild(treeRoot.childNodes[0]);
 | |
|     httpGet(treeRoot, "/");
 | |
|   };
 | |
| 
 | |
|   function delCb(path){
 | |
|     return function(status, responseText){
 | |
|       if(status != 200){
 | |
|         alert("ERROR "+status+": "+responseText);
 | |
|       } else {
 | |
|         treeRoot.removeChild(treeRoot.childNodes[0]);
 | |
|         httpGet(treeRoot, "/");
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   function httpDelete(filename){
 | |
|     var formData = new FormData();
 | |
|     formData.append("path", filename);
 | |
|     requests.add("DELETE", "/edit", formData, delCb(filename));
 | |
|   }
 | |
| 
 | |
|   function getCb(parent, path){
 | |
|     return function(status, responseText){
 | |
|       if(status == 200)
 | |
|         addList(parent, path, JSON.parse(responseText));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   function httpGet(parent, path){
 | |
|     requests.add("GET", "/edit", { list: path }, getCb(parent, path));
 | |
|   }
 | |
| 
 | |
|   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 "hex": lang = "plain"; break;
 | |
|         case "conf": lang = "plain"; break;
 | |
|         case "htm": lang = "html"; break;
 | |
|         case "js": lang = "javascript"; break;
 | |
|         case "h": lang = "c_cpp"; 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":
 | |
|         case "ini":
 | |
|           lang = ext;
 | |
|       }
 | |
|     }
 | |
|     return lang;
 | |
|   }
 | |
| 
 | |
|   if(typeof file === "undefined") file = "/index.htm";
 | |
| 
 | |
|   if(typeof lang === "undefined"){
 | |
|     lang = getLangFromFilename(file);
 | |
|   }
 | |
| 
 | |
|   if(typeof theme === "undefined") theme = "monokai";
 | |
| 
 | |
|   if(typeof type === "undefined"){
 | |
|     type = "text/"+lang;
 | |
|     if(lang === "c_cpp") type = "text/plain";
 | |
|   }
 | |
| 
 | |
|   var editor = ace.edit(element);
 | |
|   function httpPostProcessRequest(status, responseText){
 | |
|     if(status != 200) alert("ERROR "+status+": "+responseText);
 | |
|   }
 | |
|   function httpPost(filename, data, type){
 | |
|     var formData = new FormData();
 | |
|     formData.append("data", new Blob([data], { type: type }), filename);
 | |
|     requests.add("POST", "/edit", formData, httpPostProcessRequest);
 | |
|   }
 | |
|   function httpGetProcessRequest(status, responseText){
 | |
|       ge("preview").style.display = "none";
 | |
|       ge("editor").style.display = "block";
 | |
|       if(status == 200)
 | |
|         editor.setValue(responseText);
 | |
|       else
 | |
|         editor.setValue("");
 | |
|       editor.clearSelection();
 | |
|   }
 | |
|   function httpGet(theUrl){
 | |
|       requests.add("GET", "/edit", { edit: theUrl }, httpGetProcessRequest);
 | |
|   }
 | |
| 
 | |
|   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.2.3/ace.js" type="text/javascript" charset="utf-8"></script>
 | |
| </head>
 | |
| <body onload="onBodyLoad();">
 | |
|   <div id="loader" class="loader"></div>
 | |
|   <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> | 
