fs.html 52 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156
  1. <html>
  2. <head>
  3. <title>File Manager</title>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0 user-scalable=no">
  6. <script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
  7. <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.4.1/semantic.min.css" />
  8. <script src="https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.4.1/semantic.min.js"></script>
  9. <script src="https://cdnjs.cloudflare.com/ajax/libs/jszip/3.6.0/jszip.min.js"></script>
  10. <link rel="stylesheet" href="fs.css">
  11. <script>
  12. </script>
  13. <style>
  14. body{
  15. background-color: white;
  16. }
  17. #uploadProgressBar{
  18. position: fixed;
  19. bottom: 0;
  20. left: 0;
  21. width: 100%;
  22. }
  23. .fileOprBtn.disabled{
  24. opacity: 0.5;
  25. user-select: none;
  26. pointer-events: none;
  27. }
  28. </style>
  29. </head>
  30. <body class="whiteTheme">
  31. <div id="navibar" class="navibar">
  32. <!-- File Opr Group-->
  33. <button class="fileOprBtn" title="Open" onclick="openViaButton(event);"><img class="opricon" src="img/opr/open.svg"><p class="oprtxt" locale="fileopr/Open">Open</p></button>
  34. <button id="copyButton" class="fileOprBtn" title="Copy" onclick="copy();"><img class="opricon" src="img/opr/copy.svg"><p class="oprtxt" locale="fileopr/Copy">Copy</p></button>
  35. <button id="pasteButton" class="fileOprBtn" title="Paste" onclick="paste();"><img class="opricon" src="img/opr/paste.svg"><p class="oprtxt" locale="fileopr/Paste">Paste</p></button>
  36. <div class="fileoprGroupDivider" style="display: inline-block; vertical-align: top;">
  37. <button class="fileoprSmallBtn" title="Refresh" onclick="refresh();"><i class="green refresh icon"></i> <span locale="fileopr/Refresh">Refresh</span></button><br>
  38. <button class="fileoprSmallBtn" title="Cut" onclick="cut();"><i class="blue cut icon"></i> <span locale="fileopr/Cut">Cut</span></button><br>
  39. <button class="fileoprSmallBtn" title="Rename" onclick="rename();"><i class="teal i cursor icon"></i> <span locale="fileopr/Rename">Rename</span></button>
  40. </div>
  41. <button class="fileOprBtn" title="Upload" onclick="upload(); "><img class="opricon" src="img/opr/upload.svg"><p class="oprtxt wideScreenOnly" locale="fileopr/Upload">Upload</p></button>
  42. <button class="fileOprBtn" title="Download" onclick="downloadFile(); "><img class="opricon" src="img/opr/download.svg"><p class="oprtxt wideScreenOnly" locale="fileopr/Download">Download</p></button>
  43. <div class="fileoprGroupDivider" style="display: inline-block; vertical-align: top;"></div>
  44. <div class="fileoprGroupDivider" style="display: inline-block; vertical-align: top;">
  45. <button class="fileoprSmallBtn" title="New File" onclick="newFile();"><i style="color: #c7c7c7 !important;" class="file outline icon"></i> <span locale="fileopr/New File">New File</span></button><br>
  46. <button class="fileoprSmallBtn" title="New Folder" onclick="newFolder();"><i style="color: #ffe79e !important;" class="yellow folder icon"></i> <span locale="fileopr/New Folder">New Folder</span></button><br>
  47. <button class="fileoprSmallBtn" title="Delete" onclick="deleteFile();"><i class="red times icon"></i> <span locale="fileopr/Delete">Delete</span></button><br>
  48. </div>
  49. <br>
  50. <!-- Directoy navigations -->
  51. <div class="addressBar">
  52. <button id="prevDir" class="navibarBtn" onclick="prevDir();" title="Back"><i class="left arrow icon"></i></button>
  53. <button id="ppbtn" class="navibarBtn" onclick="parentDir();" title="Parent Folder"><i class="up arrow icon"></i></button>
  54. <div id="pathDisplayField" class="ui breadcrumb addressText pathDisplay desktopOnly" >
  55. <div class="section selectable"><i class="folder icon"></i> user</div>
  56. <div class="divider">:/</div>
  57. <div class="section selectable">Desktop</div>
  58. </div>
  59. <button id="togglePropertiesViewBtn" style="margin-left: 0.4em;" class="ui icon tiny button videmode propbar" title="Show Properties" onclick="togglePropertiesView(this);"><i class="columns icon"></i></button>
  60. </div>
  61. <div class="msgbox" style="z-index:999; display:none; padding-bottom: 1em;">
  62. <i class="checkmark icon showicon"></i> <span style="word-break: break-all;">No Message</span>
  63. <div class="closeMsgButton" onclick="$(this).parent().stop().slideUp('fast');"><i class="caret down icon"></i></div>
  64. </div>
  65. </div>
  66. <div id="mainWindow">
  67. <div id="folderView" style="height: 100%;">
  68. <div id="folderList" class="fileviewList">
  69. </div>
  70. <div id="fileList" class="fileviewList">
  71. </div>
  72. <br>
  73. </div>
  74. <div id="propertiesView" class="small">
  75. <div class="preview" style="margin-top: 0.4em;" align="center">
  76. <img class="ui image" style="max-height: 300px;">
  77. <audio src="" controls style="display:none; width: 100%;"></audio>
  78. </div>
  79. <h3 class="ui header" style="margin-top: 0.4em;">
  80. <span class="filename" style="word-break: break-all;" locale="sidebar/default/nofileselected">No File Selected</span>
  81. <div class="sub header vpath" style="word-break: break-all;" locale="sidebar/default/instruction">Select a file to view file properties</div>
  82. </h3>
  83. <table class="ui very basic table">
  84. <tbody class="propertiesTable">
  85. </tbody>
  86. </table>
  87. <button id="loadPreviewButton" class="ui small fluid basic disabled button" onclick="loadPreview();">Load Preview</button>
  88. </div>
  89. <div id="uploadProgressBar">
  90. <div class="ui small indicating progress" style="margin-bottom: 0px !important; border-radius: 0 !important;">
  91. <div class="bar" style="background-color: #92cfe7 !important; min-width: 0; border-radius: 0 !important;">
  92. <div class="progress"></div>
  93. </div>
  94. <div class="label">Uploading Files</div>
  95. </div>
  96. </div>
  97. </div>
  98. <script>
  99. let currentPath = "/";
  100. let listDirInitTime = 0;
  101. let propertiesView = true;
  102. //Uploads
  103. let uploadPendingFiles = [];
  104. let currentlyUploading = false;
  105. //File operations
  106. let cutMode = false;
  107. let cutPendingFilepath = [];
  108. let copyBuffering = false;
  109. let copySrcFilename = "";
  110. let copyPendingFile = undefined;
  111. //History
  112. let dirHistory = [];
  113. $(window).on("resize", function(){
  114. updateElementSize();
  115. })
  116. $(document).ready(function(){
  117. if (window.location.hash.length > 1){
  118. let previousPath = window.location.hash.substr(1);
  119. listDir(previousPath);
  120. }else{
  121. listDir("/");
  122. }
  123. //Add drop events to file view
  124. var folderView = document.getElementById('folderView');
  125. // Add event listeners for dragover and drop events
  126. folderView.addEventListener('dragover', handleDragOver, false);
  127. folderView.addEventListener('drop', handleFileDrop, false);
  128. });
  129. // Event handler for dragover event
  130. function handleDragOver(event) {
  131. event.preventDefault();
  132. event.stopPropagation();
  133. }
  134. function handleFileDrop(event) {
  135. event.preventDefault();
  136. event.stopPropagation();
  137. // Get the File object from the dataTransfer object
  138. var files = event.dataTransfer.files;
  139. //Push the files into queue
  140. for (var i = 0; i < files.length; i++) {
  141. if (files[i].name.indexOf(".") < 0){
  142. msgbox("Folder upload is not supported", false);
  143. if (files.length == 1){
  144. return;
  145. }
  146. continue;
  147. }
  148. let pathToUpload = currentPath;
  149. uploadPendingFiles.push({
  150. "file": files[i],
  151. "dir": pathToUpload
  152. });
  153. }
  154. //Upload the first file
  155. msgbox("File Upload Started")
  156. if (!currentlyUploading){
  157. uploadFileQueue();
  158. }
  159. }
  160. function uploadFileQueue(){
  161. if (uploadPendingFiles.length > 0){
  162. currentlyUploading = true;
  163. let nextFileToUpload = uploadPendingFiles.pop();
  164. handleFile(nextFileToUpload.file, nextFileToUpload.dir, function(){
  165. msgbox(nextFileToUpload.file.name + " uploaded");
  166. setTimeout(function(){
  167. uploadFileQueue();
  168. }, 300);
  169. });
  170. }else{
  171. msgbox("Upload Queue Completed");
  172. currentlyUploading = false;
  173. }
  174. }
  175. function deleteFile(){
  176. if ($(".fileObject.selected").length == 0){
  177. return;
  178. }
  179. if (confirm("Confirm removing " + $(".fileObject.selected").length + " files?")){
  180. let counter = $(".fileObject.selected").length;
  181. $(".fileObject.selected").each(function(){
  182. let thisFilepath = $(this).attr("filepath");
  183. $.ajax({
  184. url: "/api/fs/del?target=" + thisFilepath,
  185. method: "POST",
  186. success: function(data){
  187. if (data.error != undefined){
  188. msgbox(data.error, false);
  189. }else{
  190. counter--;
  191. if (counter == 0){
  192. //All removed
  193. msgbox("File(s) Removed");
  194. refresh();
  195. }
  196. }
  197. }
  198. });
  199. });
  200. }
  201. }
  202. //Check if a extension is code file, remember to trim the first dot
  203. function isCodeFiles(ext){
  204. if (ext == "html" || ext == "htm"|| ext == "css"|| ext == "js"|| ext == "json"){
  205. return true;
  206. }
  207. return false;
  208. }
  209. function openViaButton(evt){
  210. if ($(".fileObject.selected").length == 0){
  211. return;
  212. }
  213. let editableCodeFiles = [];
  214. $(".fileObject.selected").each(function(){
  215. let ftype = $(this).attr('type');
  216. let filepath = $(this).attr("filepath");
  217. let filename = $(this).attr("filename");
  218. if (ftype != "folder"){
  219. let ext = filepath.split(".").pop();
  220. if (isCodeFiles(ext)){
  221. editableCodeFiles.push({
  222. "filename": filename,
  223. "filepath": filepath
  224. });
  225. }else{
  226. openthis($(this), evt);
  227. }
  228. }
  229. });
  230. if (editableCodeFiles.length > 0){
  231. let hash = encodeURIComponent(JSON.stringify(editableCodeFiles))
  232. window.open("notepad/index.html#" + hash);
  233. }
  234. }
  235. function refresh(){
  236. listDir(currentPath);
  237. }
  238. function updatePathDisplay(){
  239. $("#pathDisplayField").empty();
  240. $("#pathDisplayField").append(`<div class="section selectable" onclick="jumpToDir('/');"><i class="microchip icon"></i> ESP8266</div><div class="divider">:/</div>`);
  241. let pathSegments = currentPath;
  242. if (pathSegments.startsWith("/")){
  243. pathSegments = pathSegments.substr(1);
  244. }
  245. if (pathSegments.endsWith("/")){
  246. pathSegments = pathSegments.substr(0, pathSegments.length - 1);
  247. }
  248. pathSegments = pathSegments.split("/");
  249. let htmlSegments = [];
  250. let accumulativeDir = "/";
  251. pathSegments.forEach(function(segment){
  252. accumulativeDir = accumulativeDir + segment + "/"
  253. htmlSegments.push(`<div class="section selectable" onclick="jumpToDir('${accumulativeDir}');">${segment}</div>`);
  254. });
  255. $("#pathDisplayField").append(htmlSegments.join(`<div class="divider">/</div>`));
  256. }
  257. function cut(){
  258. if ($(".fileObject.selected").length == 0){
  259. msgbox("No file selected", false);
  260. return;
  261. }
  262. cutMode = true;
  263. cutPendingFilepath = [];
  264. $(".fileObject.selected").each(function(){
  265. cutPendingFilepath.push({
  266. filename: $(this).attr("filename"),
  267. filepath: $(this).attr("filepath")
  268. });
  269. });
  270. msgbox("File Ready to Paste");
  271. }
  272. function downloadFile(){
  273. let selectedFiles = [];
  274. let filenames = [];
  275. let containsDir = false;
  276. $(".fileObject.selected").each(function(){
  277. if ($(this).attr("type") == "folder"){
  278. containsDir = true;
  279. }
  280. });
  281. if (containsDir){
  282. msgbox("Folder download is not supported", false);
  283. return;
  284. };
  285. if ($(".fileObject.selected").length > 0){
  286. $(".fileObject.selected").each(function(){
  287. selectedFiles.push($(this).attr("filepath"));
  288. filenames.push($(this).attr("filename"))
  289. });
  290. }
  291. if (selectedFiles.length == 1){
  292. //Only one file
  293. //window.open("/api/fs/download?file=" + selectedFiles[0]);
  294. var file_path = "/api/fs/download?file=" + selectedFiles[0];
  295. var a = document.createElement('A');
  296. a.href = file_path;
  297. a.download = file_path.substr(file_path.lastIndexOf('/') + 1);
  298. document.body.appendChild(a);
  299. a.click();
  300. document.body.removeChild(a);
  301. }else{
  302. let urls = [];
  303. selectedFiles.forEach(function(thisFilepath){
  304. urls.push("/api/fs/download?file=" + thisFilepath);
  305. });
  306. msgbox("Zipping might take a few minutes...");
  307. compressFileToZip(urls, filenames);
  308. }
  309. }
  310. function compressFileToZip(urls, filenames) {
  311. var zip = new JSZip();
  312. // Create a function to fetch each image and add it to the zip
  313. var addFileToZip = function (url, filename) {
  314. return new Promise(function (resolve, reject) {
  315. var xhr = new XMLHttpRequest();
  316. xhr.open('GET', url);
  317. xhr.responseType = 'blob';
  318. xhr.onload = function () {
  319. if (xhr.status === 200) {
  320. zip.file(filename, xhr.response);
  321. resolve();
  322. } else {
  323. reject(Error('Failed to fetch file: ' + url));
  324. }
  325. };
  326. xhr.onerror = function () {
  327. reject(Error('Error fetching file: ' + url));
  328. };
  329. xhr.send();
  330. });
  331. };
  332. // Iterate over each image URL and add it to the zip
  333. var promises = urls.map(function (url, index) {
  334. var filename = filenames[index];
  335. return addFileToZip(url, filename);
  336. });
  337. // When all promises are resolved, generate the zip file
  338. Promise.all(promises).then(function () {
  339. zip.generateAsync({ type: 'blob' }).then(function (content) {
  340. // Save the zip file or do something with it
  341. msgbox("Download Zip Created");
  342. saveAs(content, 'dl_' + Math.floor(Date.now() / 1000) +'.zip');
  343. });
  344. }).catch(function (error) {
  345. console.error(error);
  346. });
  347. }
  348. function saveAs(blob, filename) {
  349. if (navigator.msSaveBlob) {
  350. // For IE and Edge browsers
  351. navigator.msSaveBlob(blob, filename);
  352. } else {
  353. // For other browsers
  354. var link = document.createElement('a');
  355. link.href = URL.createObjectURL(blob);
  356. link.download = filename;
  357. link.style.display = 'none';
  358. document.body.appendChild(link);
  359. link.click();
  360. document.body.removeChild(link);
  361. }
  362. }
  363. //Jump to new directory via path input field
  364. function jumpToDir(newDir){
  365. if (newDir == currentPath){
  366. return;
  367. }
  368. listDir(newDir);
  369. }
  370. function prevDir(){
  371. if (dirHistory.length > 0){
  372. let pathToGo = dirHistory.pop();
  373. if (pathToGo == currentPath && dirHistory.length > 0){
  374. //pop again
  375. pathToGo = dirHistory.pop();
  376. listDir(pathToGo);
  377. return;
  378. }
  379. listDir(pathToGo, false);
  380. }
  381. }
  382. function listDir(path, recordHistory = true){
  383. if (path.length > 0 && path.substr(0, 1) != "/"){
  384. path = "/" + path;
  385. }
  386. if (!path.endsWith("/")){
  387. path = path + "/";
  388. }
  389. //Update the current path
  390. currentPath = path;
  391. if (recordHistory && currentPath != dirHistory[dirHistory.length - 1]){
  392. dirHistory.push(currentPath);
  393. }
  394. window.location.hash = currentPath;
  395. updatePathDisplay();
  396. listDirInitTime = Date.now();
  397. let validationTimestamp = listDirInitTime;
  398. $("#folderList").html(`<div class="fileObject item" style="pointer-events: none;">
  399. <span style="display:inline-block !important;word-break: break-all; width:100%;" class="normal object">
  400. <i class="loading spinner icon" style="margin-right:12px; color:#grey;"></i> <span class="filename">Loading</span>
  401. </span>
  402. </div>`);
  403. $("#fileList").html("");
  404. $.ajax({
  405. url: "/api/fs/list?dir=" + path,
  406. success: function(data){
  407. if (validationTimestamp != listDirInitTime){
  408. //Another refresh is in progress. Skip render.
  409. return;
  410. }
  411. $("#folderList").html("");
  412. if (data.error != undefined){
  413. $("#folderList").append(`<div class="ui segment">
  414. <div class="ui header themed">
  415. <i class="remove icon" style="display: inline-block;"></i> <span>Error Opening Folder</span>
  416. <div class="sub header" style="margin-top:12px;">Server return the following error message: <br><code>${data.error.toUpperCase()}</code><br>
  417. ${new Date().toLocaleString()}</div>
  418. </div>
  419. </div>`);
  420. msgbox(data.error, false);
  421. }else{
  422. data.forEach(function(filedata){
  423. let isDir = filedata.IsDir;
  424. let filename = filedata.Filename;
  425. let filesize = filedata.Filesize;
  426. if (isDir){
  427. $("#folderList").append(`<div class="fileObject item" draggable="true" filename="${filename}" filepath="${path + filename}" ondblclick="openthis(this,event);" type="folder">
  428. <span style="display:inline-block !important;word-break: break-all; width:100%;" class="normal object">
  429. <i class="folder icon" style="margin-right:12px; color:#eab54e;"></i> <span class="filename">${filename}</span>
  430. </span>
  431. </div>`);
  432. }else{
  433. let extension = "." + filename.split(".").pop();
  434. let fileIcon = getFileIcon(extension);
  435. $("#fileList").append(`<div class="fileObject item" draggable="true" filename="${filename}" filepath="${path + filename}" ondblclick="openthis(this,event);" type="file">
  436. <span style="display:inline-block !important;word-break: break-all; width:100%;" class="normal object">
  437. <i class="${fileIcon} icon" style="margin-right:12px; color:grey;"></i> <span class="filename">${filename} (${humanFileSize(filesize)})</span>
  438. </span>
  439. </div>`);
  440. }
  441. });
  442. }
  443. $(".fileObject").off("click").on("click", function(e){
  444. if (!e.ctrlKey) {
  445. $(".fileObject.selected").removeClass("selected");
  446. getFileProperties( $(this).attr("filepath"));
  447. let fileType = $(this).attr('type');
  448. if (fileType == "folder"){
  449. $("#propertiesView").find(".preview").find("img").attr("src", "img/folder.svg");
  450. }else{
  451. $("#propertiesView").find(".preview").find("img").attr("src", "img/file.svg");
  452. }
  453. }
  454. $(this).addClass("selected");
  455. });
  456. sortFileList();
  457. }
  458. });
  459. }
  460. function openthis(target, event){
  461. let isDir = ($(target).attr("type") == "folder");
  462. if (isDir){
  463. let targetPath = $(target).attr("filepath");
  464. listDir(targetPath);
  465. }else{
  466. let ext = $(target).attr("filepath").split(".").pop();
  467. if (ext == "txt" || ext == "md"){
  468. //Open with markdown editor
  469. let hash = encodeURIComponent(JSON.stringify({
  470. "filename": $(target).attr("filename"),
  471. "filepath": $(target).attr("filepath")
  472. }))
  473. window.open("mde/index.html#" + hash);
  474. }else if (ext == "jpg" || ext == "jpeg" || ext == "png" || ext == "gif" || ext == "webp"){
  475. //Open with photo viewer
  476. let hash = encodeURIComponent(JSON.stringify({
  477. "filename": $(target).attr("filename"),
  478. "filepath": $(target).attr("filepath")
  479. }))
  480. window.open("photo.html#" + hash);
  481. }else if (ext == "mp3" || ext == "aac" || ext == "ogg"){
  482. //Open with music player
  483. let hash = encodeURIComponent(JSON.stringify({
  484. "filename": $(target).attr("filename"),
  485. "filepath": $(target).attr("filepath")
  486. }))
  487. window.open("music.html#" + hash);
  488. }else if (ext == "mp4" || ext == "webm"){
  489. //Open with video player
  490. let hash = encodeURIComponent(JSON.stringify({
  491. "filename": $(target).attr("filename"),
  492. "filepath": $(target).attr("filepath")
  493. }))
  494. window.open("video.html#" + hash);
  495. }else if (isCodeFiles(ext)){
  496. //Open with notepad
  497. //**Notes the array wrapper in JSON object**
  498. let hash = encodeURIComponent(JSON.stringify([{
  499. "filename": $(target).attr("filename"),
  500. "filepath": $(target).attr("filepath")
  501. }]))
  502. window.open("notepad/index.html#" + hash);
  503. }else{
  504. window.open("/api/fs/download?file=" + $(target).attr("filepath") + "&preview=true");
  505. }
  506. }
  507. }
  508. function isFilenameValid(filename) {
  509. // Split the filename into the name and extension parts
  510. var name = filename.slice(0, filename.lastIndexOf('.'));
  511. var extension = filename.slice(filename.lastIndexOf('.') + 1);
  512. // Check if the name and extension lengths are within the limits
  513. if (name.length <= 8 && extension.length <= 3) {
  514. return true;
  515. } else {
  516. return false;
  517. }
  518. }
  519. function newFile(){
  520. var fileName = window.prompt("Name for new file: ", "file.txt");
  521. if (fileName.indexOf("/") >= 0){
  522. //Contains /. Reject
  523. msgbox("File name cannot contain path seperator", false);
  524. return;
  525. }
  526. if (fileName.indexOf(".") == -1){
  527. msgbox("Missing file extension")
  528. return
  529. }
  530. let filenameOnly = fileName.split(".");
  531. let ext = filenameOnly.pop();
  532. filenameOnly = filenameOnly.join(".");
  533. /*
  534. //Skip checking as modern FAT32 file systems can store as LONG FILENAME
  535. if (filenameOnly.length > 8){
  536. msgbox("File name too long (8 char max)", false);
  537. return;
  538. }
  539. if (ext.length > 3){
  540. msgbox("File extension too long (3 char max)", false);
  541. return
  542. }
  543. */
  544. //OK! Create the file
  545. const blob = new Blob(["\n"], { type: 'text/plain' });
  546. const file = new File([blob], fileName);
  547. handleFile(file, currentPath, function(){
  548. msgbox("New File Created");
  549. });
  550. }
  551. function newFolder(){
  552. var folderName = window.prompt("Name for new folder (8 char max): ", "Folder");
  553. if (folderName.indexOf("/") >= 0){
  554. //Contains /. Reject
  555. msgbox("Folder name cannot contain path seperator", false);
  556. return;
  557. }
  558. if (folderName.length > 8){
  559. msgbox("Folder name too long (8 char max)", false);
  560. return;
  561. }
  562. $.post("/api/fs/newFolder?path=" + currentPath + folderName, function(data){
  563. if (data.error != undefined){
  564. msgbox(data.error, false);
  565. }else{
  566. msgbox("Folder Created");
  567. refresh();
  568. }
  569. });
  570. }
  571. function rename(){
  572. if ($(".fileObject.selected").length > 1){
  573. //Too many objects
  574. }else if ($(".fileObject.selected").length == 1){
  575. var oldName = $(".fileObject.selected").attr("filename");
  576. var oldPath = $(".fileObject.selected").attr("filepath");
  577. var newName = window.prompt("Rename " + oldName + " to: ", oldName);
  578. if (newName.indexOf("/") >= 0){
  579. //Contains /. Reject
  580. msgbox("File name cannot contain path seperator", false);
  581. return;
  582. }
  583. /*
  584. //FAT32 allows filename longer than 8.3
  585. if (!isFilenameValid(newName)){
  586. msgbox("File name too long (8 char max)", false);
  587. return;
  588. }
  589. */
  590. if (newName && newName != oldName) {
  591. // User entered a new name, perform renaming logic here
  592. console.log(oldPath, currentPath + newName);
  593. $.ajax({
  594. url: "/api/fs/move?src=" + oldPath + "&dest=" + currentPath + newName,
  595. method: "POST",
  596. success: function(data){
  597. if (data.error != undefined){
  598. msgbox(data.error, false);
  599. }else{
  600. msgbox("File renamed");
  601. refresh();
  602. }
  603. }
  604. })
  605. }
  606. }
  607. }
  608. function getFileIcon(extension) {
  609. const textExtensions = [".md", ".txt"];
  610. const codeExtensions = [".js", ".json", ".css", ".html", ".htm"];
  611. const musicExtensions = [".mp3", ".aac", ".ogg", ".wav"];
  612. const videoExtensions = [".mp4", ".m4v", ".webm"];
  613. const photoExtensions = [".png", ".gif", ".jpg", ".ico", ".svg"];
  614. if (textExtensions.includes(extension)) {
  615. return "file alternate outline";
  616. } else if (codeExtensions.includes(extension)) {
  617. return "black file code outline";
  618. } else if (musicExtensions.includes(extension)) {
  619. return "blue music";
  620. } else if (videoExtensions.includes(extension)) {
  621. return "red video";
  622. } else if (photoExtensions.includes(extension)) {
  623. return "green image outline";
  624. } else {
  625. return "file outline";
  626. }
  627. }
  628. function isPreviewable(ext){
  629. let previeableFiles = [".png", ".gif", ".jpg", ".ico", ".svg"];
  630. return previeableFiles.includes(ext);
  631. }
  632. function getFileProperties(filepath){
  633. $.get("/api/fs/properties?file=" + filepath, function(data){
  634. if (data.error != undefined){
  635. msgbox(data.error, false);
  636. return;
  637. }
  638. $("#propertiesView").find(".filename").text(data.filename);
  639. $("#propertiesView").find(".vpath").text(data.filepath);
  640. let propTable = $("#propertiesView").find(".propertiesTable");
  641. let styleOverwrite = `min-width: 4em;`;
  642. $(propTable).html("");
  643. $(propTable).append(`<tr>
  644. <td style="${styleOverwrite}">
  645. File Size
  646. </td>
  647. <td>
  648. ${bytesToSize(data.filesize)}
  649. </td>
  650. </tr><tr>
  651. <td style="${styleOverwrite}">
  652. Disk Path
  653. </td>
  654. <td style="word-break: break-all;">
  655. /www${data.filepath}
  656. </td>
  657. </tr><tr>
  658. <td style="${styleOverwrite}">
  659. Folder
  660. </td>
  661. <td style="word-break: break-all;">
  662. ${data.isDir?`<i class="ui green check icon"></i>`:`<i class="ui red times icon"></i>`}
  663. </td>
  664. </tr>`);
  665. if (data.isDir){
  666. $(propTable).append(`<tr>
  667. <td style="${styleOverwrite}">
  668. Files #
  669. </td>
  670. <td>
  671. ${data.fileCounts}
  672. </td>
  673. </tr><tr>
  674. <td style="${styleOverwrite}">
  675. Folders #
  676. </td>
  677. <td style="word-break: break-all;">
  678. ${data.folderCounts}
  679. </td>
  680. </tr>`);
  681. //Folder is not previewable
  682. $("#propertiesView").find(".preview").find("img").attr("xsrc", "");
  683. $("#loadPreviewButton").addClass("disabled");
  684. }else{
  685. let ext = data.filepath.split(".").pop();
  686. if (isPreviewable("." + ext)){
  687. $("#propertiesView").find(".preview").find("img").attr("xsrc", "/api/fs/download?preview=true&file=" + data.filepath);
  688. $("#loadPreviewButton").removeClass("disabled");
  689. }else{
  690. $("#propertiesView").find(".preview").find("img").attr("xsrc", "");
  691. $("#loadPreviewButton").addClass("disabled");
  692. }
  693. }
  694. })
  695. }
  696. function loadPreview(){
  697. let readyToLoadSrc = $("#propertiesView").find(".preview").find("img").attr("xsrc");
  698. if (readyToLoadSrc == undefined || readyToLoadSrc == "" ){
  699. }else{
  700. let ext = readyToLoadSrc.split(".").pop();
  701. $("#propertiesView").find(".preview").find("img").show();
  702. $("#propertiesView").find(".preview").find("audio").hide();
  703. $("#propertiesView").find(".preview").find("img").attr("src", readyToLoadSrc);
  704. }
  705. }
  706. function bytesToSize(bytes) {
  707. var sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB'];
  708. if (bytes == 0) return '0 Byte';
  709. var i = parseInt(Math.floor(Math.log(bytes) / Math.log(1024)));
  710. return Math.round(bytes / Math.pow(1024, i) * 100, 2) / 100 + ' ' + sizes[i];
  711. }
  712. function getParentDirectory(path) {
  713. // Remove any trailing slashes
  714. if (path.endsWith("/")){
  715. path = path.substr(0, path.length - 1);
  716. }
  717. // Find the last index of the slash character
  718. var lastIndex = path.lastIndexOf('/');
  719. // Extract the parent directory substring
  720. var parentDir = path.substring(0, lastIndex);
  721. return parentDir;
  722. }
  723. function parentDir(){
  724. if (currentPath.indexOf("/") >= 0 && currentPath != "/"){
  725. let parentPath = getParentDirectory(currentPath);
  726. listDir(parentPath);
  727. }else{
  728. //already top
  729. }
  730. }
  731. function humanFileSize(bytes, si=false, dp=1) {
  732. const thresh = si ? 1000 : 1024;
  733. if (Math.abs(bytes) < thresh) {
  734. return bytes + ' B';
  735. }
  736. const units = si
  737. ? ['kB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']
  738. : ['KiB', 'MiB', 'GiB', 'TiB', 'PiB', 'EiB', 'ZiB', 'YiB'];
  739. let u = -1;
  740. const r = 10**dp;
  741. do {
  742. bytes /= thresh;
  743. ++u;
  744. } while (Math.round(Math.abs(bytes) * r) / r >= thresh && u < units.length - 1);
  745. return bytes.toFixed(dp) + ' ' + units[u];
  746. }
  747. function updateElementSize(){
  748. $("#mainWindow").css("height", window.innerHeight - $("#navibar").height() + "px");
  749. }
  750. updateElementSize();
  751. function upload() {
  752. // Create a file input element
  753. var fileInput = $('<input>').attr('type', 'file');
  754. // Trigger the file selector dialog
  755. fileInput.trigger('click');
  756. // Handle the selected file using a callback event handler
  757. fileInput.change(function(e) {
  758. var file = e.target.files[0];
  759. if (currentlyUploading){
  760. //Already tasks uploading in the background. Add it to queue
  761. uploadPendingFiles.push({
  762. "file": file,
  763. "dir": currentPath
  764. });
  765. msgbox("File Added to Upload Queue");
  766. return;
  767. }
  768. // Pass the file object to the callback event handler
  769. msgbox("File Upload Started")
  770. handleFile(file, currentPath, function(){
  771. msgbox("Upload Completed");
  772. });
  773. });
  774. }
  775. function handleFile(file, dir=currentPath, callback=undefined) {
  776. // Perform actions with the selected file
  777. $("#pasteButton").addClass("disabled");
  778. console.log('Selected file:', file);
  779. var formdata = new FormData();
  780. formdata.append("file1", file);
  781. var ajax = new XMLHttpRequest();
  782. ajax.upload.addEventListener("progress", progressHandler, false);
  783. ajax.addEventListener("load", function(event){
  784. let responseText = event.target.responseText;
  785. try{
  786. responseText = JSON.parse(responseText);
  787. if (responseText.error != undefined){
  788. alert(responseText.error);
  789. }
  790. }catch(ex){
  791. }
  792. completeHandler(event, dir==currentPath);
  793. $("#pasteButton").removeClass("disabled");
  794. if (callback != undefined){
  795. callback();
  796. }
  797. }, false); // doesnt appear to ever get called even upon success
  798. ajax.addEventListener("error", errorHandler, false);
  799. ajax.addEventListener("abort", abortHandler, false);
  800. ajax.open("POST", "/upload?dir=" + dir);
  801. ajax.send(formdata);
  802. }
  803. function progressHandler(event) {
  804. //_("loaded_n_total").innerHTML = "Uploaded " + event.loaded + " bytes of " + event.total; // event.total doesnt show accurate total file size
  805. var percent = (event.loaded / event.total) * 100;
  806. $("#uploadProgressBar").find(".bar").css("width", Math.round(percent) + "%");
  807. console.log("Uploaded " + event.loaded + " bytes => " + percent +"%");
  808. if (percent >= 100) {
  809. $("#uploadProgressBar").find(".bar").css("width", "100%");
  810. //_("status").innerHTML = "Please wait, writing file to filesystem";
  811. }
  812. }
  813. function completeHandler(event, requireRefresh=true) {
  814. $("#uploadProgressBar").find(".bar").css("width", "0%");
  815. if(requireRefresh){
  816. refresh();
  817. }
  818. }
  819. function errorHandler(event) {
  820. msgbox("Upload Failed", false);
  821. $("#pasteButton").removeClass("disabled");
  822. }
  823. function abortHandler(event) {
  824. msgbox("Upload Aborted", false);
  825. $("#pasteButton").removeClass("disabled");
  826. }
  827. function msgbox(message, succ=true){
  828. function capitalizeFirstLetter(string) {
  829. return string.charAt(0).toUpperCase() + string.slice(1);
  830. }
  831. message = capitalizeFirstLetter(message);
  832. if (succ){
  833. $(".msgbox").find(".showicon").attr("class", "green circle check icon showicon");
  834. }else{
  835. $(".msgbox").find(".showicon").attr("class", "red circle times icon showicon");
  836. }
  837. $(".msgbox").find("span").text(message);
  838. $(".msgbox").stop().finish().slideDown("fast").delay(3000).slideUp("fast");
  839. }
  840. //Copy file
  841. function copy(){
  842. if ($(".fileObject.selected").length == 0){
  843. //No file selected
  844. msgbox("No file selected", false);
  845. return;
  846. }
  847. if ($(".fileObject.selected").length > 1){
  848. msgbox("WebStick can only support 1 file copy at a time", false);
  849. return;
  850. }
  851. if ($(".fileObject.selected").attr('type') == "folder"){
  852. msgbox("Folder copy is not supported", false);
  853. return;
  854. }
  855. let url = "/api/fs/download?file=" + $(".fileObject.selected").attr("filepath");
  856. let filename = $(".fileObject.selected").attr("filename");
  857. cutMode = false;
  858. copyBuffering = true;
  859. copySrcFilename = filename;
  860. console.log("Buffering " + filename + " from " + url);
  861. msgbox("Allocating memory for copy...")
  862. $("#pasteButton").addClass("disabled");
  863. $("#copyButton").addClass("disabled");
  864. // Fetch the content from the URL
  865. fetch(url).then(response => response.blob()).then(blob => {
  866. // Create a new File object from the Blob
  867. copyPendingFile = (blob);
  868. msgbox("File Ready to Paste");
  869. $("#pasteButton").removeClass("disabled");
  870. $("#copyButton").removeClass("disabled");
  871. copyBuffering = false;
  872. }).catch(error => {
  873. msgbox("Copy Allocation Failed", false);
  874. $("#pasteButton").removeClass("disabled");
  875. $("#copyButton").removeClass("disabled");
  876. copyBuffering = false;
  877. });
  878. }
  879. function fileExistsInThisFolder(filename){
  880. let exists = false;
  881. $(".fileObject").each(function(){
  882. if ($(this).attr("filename") == filename){
  883. exists = true;
  884. }
  885. });
  886. return exists;
  887. }
  888. function paste(){
  889. if (cutMode){
  890. let remainingFilesCounter = cutPendingFilepath.length;
  891. console.log("Moving " , cutPendingFilepath);
  892. cutPendingFilepath.forEach(fileToPaste => {
  893. let filename = fileToPaste.filename;
  894. let filepath = fileToPaste.filepath;
  895. $.ajax({
  896. url: "/api/fs/move?src=" + filepath + "&dest=" + currentPath + filename,
  897. method: "POST",
  898. success: function(data){
  899. if (data.error != undefined){
  900. msgbox(data.error)
  901. }else{
  902. remainingFilesCounter--;
  903. if (remainingFilesCounter == 0){
  904. msgbox("File Move Completed");
  905. refresh();
  906. }
  907. }
  908. }
  909. })
  910. });
  911. }else{
  912. //Copy and Paste
  913. if (copyBuffering){
  914. msgbox("Copy buffer allocation in progress", false);
  915. return;
  916. }
  917. let pasteFilename = copySrcFilename;
  918. let counter = 1;
  919. while(fileExistsInThisFolder(pasteFilename)){
  920. let nameOnly = copySrcFilename.split(".");
  921. let ext = nameOnly.pop();
  922. nameOnly = nameOnly.join(".");
  923. pasteFilename = nameOnly + "-" + counter + "." + ext;
  924. counter++;
  925. }
  926. //Change the name if there is name clash
  927. const file = new File([copyPendingFile], pasteFilename);
  928. //Upload the file
  929. msgbox("Writing file to SD card...");
  930. handleFile(file, currentPath, function(){
  931. msgbox("File Pasted");
  932. });
  933. }
  934. }
  935. function sortFileList(){
  936. sortFileObjects("folderList");
  937. sortFileObjects("fileList");
  938. }
  939. function sortFileObjects(listSelector) {
  940. const fileObjects = document.querySelectorAll(`#${listSelector} .fileObject`)
  941. // Convert the NodeList to an array for sorting
  942. const fileObjectsArray = Array.from(fileObjects);
  943. // Sort the elements based on their text content
  944. fileObjectsArray.sort((a, b) => {
  945. const textA = a.querySelector('.filename').textContent.toLowerCase();
  946. const textB = b.querySelector('.filename').textContent.toLowerCase();
  947. return textA.localeCompare(textB);
  948. });
  949. // Reorder the elements in the DOM
  950. const fileList = document.getElementById(listSelector);
  951. fileObjectsArray.forEach((fileObject) => {
  952. fileList.appendChild(fileObject);
  953. });
  954. }
  955. function togglePropertiesView(object){
  956. propertiesView = !propertiesView;
  957. if (propertiesView){
  958. $("#propertiesView").show();
  959. $(object).addClass('active');
  960. localStorage.setItem("file_explorer/viewProperties", "true");
  961. if ($(".fileObject.selected").length >= 1){
  962. //Load the file properties
  963. let targetFile = getFileObjectFromFID(lastClickedFileID);
  964. if (targetFile == null){
  965. targetFile = $(".fileObject.selected")[0];
  966. }
  967. let filepath = $(targetFile).attr("filepath");
  968. loadFileProperties(filepath);
  969. }
  970. }else{
  971. $("#propertiesView").hide();
  972. $(object).removeClass('active');
  973. localStorage.setItem("file_explorer/viewProperties", "false");
  974. }
  975. }
  976. // Bind the onDeleteKeyPress() function to the document's keydown event
  977. $(document).on("keydown", function(evt){
  978. if (event.ctrlKey) {
  979. // Check for Ctrl key combinations
  980. if (event.keyCode == 67) {
  981. // Ctrl + C
  982. evt.preventDefault();
  983. copy();
  984. } else if (event.keyCode == 86) {
  985. // Ctrl + V
  986. evt.preventDefault();
  987. paste();
  988. } else if (event.keyCode == 88) {
  989. // Ctrl + X
  990. evt.preventDefault();
  991. cut();
  992. }
  993. } else {
  994. if (event.keyCode == 46) {
  995. //Delete
  996. evt.preventDefault();
  997. deleteFile();
  998. }else if (event.keyCode == 13){
  999. //Enter
  1000. evt.preventDefault();
  1001. $(".fileObject.selected").each(function(e){
  1002. openthis($(this), e);
  1003. });
  1004. }
  1005. }
  1006. });
  1007. </script>
  1008. </body>
  1009. </html>