file_properties.html 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. <html>
  2. <head>
  3. <title>File Properties</title>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0 user-scalable=no">
  6. <link rel="stylesheet" href="../../script/semantic/semantic.css">
  7. <script type="text/javascript" src="../../script/jquery.min.js"></script>
  8. <script type="text/javascript" src="../../script/semantic/semantic.min.js"></script>
  9. <script type="text/javascript" src="../../script/ao_module.js"></script>
  10. <style>
  11. body{
  12. overflow:hidden;
  13. }
  14. </style>
  15. </head>
  16. <body id="filePropertiesWindow">
  17. <br>
  18. <div class="ui container">
  19. <h3 class="ui header">
  20. File Properties
  21. <div class="sub header">Basic File Information</div>
  22. </h3>
  23. <div class="ui divider"></div>
  24. <div id="properties">
  25. </div>
  26. <br>
  27. <button style="display:none;" class="ui small blue fluid button singleFileOnly" onclick="changeDefaultWebApp();">Change Default WebApp</button>
  28. <button style="margin-top: 4px;" class="ui small fluid button linuxonly" onclick="openFilePermissionPanel();">Change File Permissions</button>
  29. <br>
  30. </div>
  31. <div id="filesizeLoader" class="ui active dimmer">
  32. <div class="ui indeterminate text loader">Calculating File Size</div>
  33. </div>
  34. <script>
  35. //Initiate the view model
  36. var files = ao_module_loadInputFiles();
  37. var fileProperties = [];
  38. var fileInfo = {};
  39. function initFileProperties(){
  40. $("#properties").html("");
  41. if (files.length == 1){
  42. //There are only 1 file to be shown
  43. getFileProp(files[0], renderSingleObject);
  44. }else if (files.length > 1){
  45. for (var i =0; i < files.length; i++){
  46. getFileProp(files[i], function(data){
  47. fileProperties.push(data);
  48. if (fileProperties.length == files.length){
  49. renderMultipleObjects();
  50. }
  51. });
  52. }
  53. }
  54. }
  55. initFileProperties();
  56. //Hide windows / linux only operations
  57. $.get("/system/info/getArOZInfo", function(data){
  58. if (data.HostOS == "windows"){
  59. $(".linuxonly").hide();
  60. }else{
  61. $(".windowsonly").hide();
  62. }
  63. });
  64. function getFileProp(vpath, callback){
  65. $.ajax({
  66. url: "../../system/file_system/getProperties",
  67. data: {path: vpath},
  68. method: "POST",
  69. success: function(data){
  70. callback(data);
  71. fileInfo = data;
  72. //Initialize system theme
  73. fpw_loadPreference("file_explorer/theme",function(data){
  74. if (data.error === undefined){
  75. if (data == "darkTheme"){
  76. fpw_toggleDarkTheme();
  77. }else{
  78. //White theme. Do nothing
  79. }
  80. }
  81. });
  82. }
  83. })
  84. }
  85. function openFilePermissionPanel(){
  86. var hashPassthrough = encodeURIComponent(JSON.stringify(files));
  87. ao_module_newfw({
  88. url: "SystemAO/file_system/file_permission.html#" + hashPassthrough,
  89. width: 340,
  90. height: 480,
  91. appicon: "SystemAO/file_system/img/properties.png",
  92. title: "File Permissions",
  93. });
  94. }
  95. function renderMultipleObjects(){
  96. hideLoader();
  97. var filesizeSum = sumProperties(fileProperties, "Filesize");
  98. $("#properties").append(ui_getInput(fileProperties[0].VirtualDirname + "/", "Root Name"));
  99. var filecount = 0;
  100. var foldercount = 0;
  101. for (var i =0; i < fileProperties.length; i++){
  102. if (fileProperties[i].IsDirectory){
  103. foldercount++;
  104. }else{
  105. filecount++;
  106. }
  107. }
  108. $("#properties").append(ui_getText("Multiple selections"));
  109. $("#properties").append(ui_getText(filecount + " Files"));
  110. $("#properties").append(ui_getText(foldercount + " Folders"));
  111. //Append other properties as table
  112. $("#properties").append(ui_getTable(
  113. [],
  114. [
  115. ["Virtual Directory", fileProperties[0].VirtualDirname + "/"],
  116. ["Storage Directory", fileProperties[0].StorageDirname + "/"],
  117. ["Total Size", bytesToSize(filesizeSum) + ` (${filesizeSum} bytes)`],
  118. ]
  119. ));
  120. }
  121. function sumProperties(data, propName){
  122. var sum = 0;
  123. for (var i = 0; i < data.length; i++){
  124. sum += data[i][propName];
  125. }
  126. return sum;
  127. }
  128. //Render one object property to the ui element
  129. function renderSingleObject(data){
  130. hideLoader();
  131. if (data.error !== undefined){
  132. //Something went wrong
  133. $("#properties").append(`<h4 class="ui header">
  134. <i class="question icon"></i>
  135. <div class="content">
  136. File Properties Unknown
  137. <div class="sub header">The system were unable to read the selected file properties.</div>
  138. </div>
  139. </h4>
  140. <div class="ui divider"></div>
  141. <small>${data.error}</small>
  142. `);
  143. }else{
  144. //Append Filename
  145. var filesizeText = "File Size";
  146. if (data.IsDirectory){
  147. $("#properties").append(ui_getInput(data.Basename, "Folder Name"));
  148. filesizeText = "Folder Size";
  149. }else{
  150. $("#properties").append(ui_getInput(data.Basename, "File Name"));
  151. }
  152. //Append MIME Type
  153. $("#properties").append(ui_getText(data.MimeType));
  154. //Get the default opener
  155. if (!data.IsDirectory){
  156. //Check if this file is shortcut
  157. if ( data.Basename.split(".").pop() == "shortcut"){
  158. //This is shortcut file
  159. $("#properties").append(ui_getTable(
  160. [],
  161. [
  162. ["Virtual Path", data.VirtualPath],
  163. ["Storage Path", data.StoragePath],
  164. ["Permission", data.Permission],
  165. ["Last Modified", generateDisplayLastModTime(data.LastModTime)],
  166. ["File Type", "System Shortcut"],
  167. ["Owner",data.Owner],
  168. ]
  169. ));
  170. }else{
  171. //Normal Files
  172. $(".singleFileOnly").show();
  173. $.ajax({
  174. url: "../../system/modules/getDefault",
  175. method: "GET",
  176. data: {
  177. opr: "launch",
  178. ext: "." + data.Basename.split(".").pop(),
  179. mode: "launch"
  180. },
  181. success: function(openerinfo) {
  182. //Check if the module is set.
  183. var defaultWebAppField = ["Default WebApp",`<img class="ui mini spaced image" style="margin-left: 0px; padding-right: 8px;" src="../../${openerinfo.IconPath}">` + openerinfo.Name];
  184. if ( openerinfo.Name == undefined){
  185. //Not set.
  186. defaultWebAppField = ["Default WebApp", `<a href="#" onclick="changeDefaultWebApp();">Set Default WebApp</a>`];
  187. }
  188. //Append other properties as table
  189. $("#properties").append(ui_getTable(
  190. [],
  191. [
  192. defaultWebAppField,
  193. ["Virtual Path", data.VirtualPath],
  194. ["Storage Path", data.StoragePath],
  195. [filesizeText, bytesToSize(data.Filesize) + ` (${data.Filesize} bytes)`],
  196. ["Permission", data.Permission],
  197. ["Last Modified", generateDisplayLastModTime(data.LastModTime)],
  198. ["File Type", "File"],
  199. ["Owner",data.Owner],
  200. ]
  201. ));
  202. }
  203. });
  204. }
  205. }else{
  206. $("#properties").append(ui_getTable(
  207. [],
  208. [
  209. ["Virtual Path", data.VirtualPath],
  210. ["Storage Path", data.StoragePath],
  211. [filesizeText, bytesToSize(data.Filesize) + ` (${data.Filesize} bytes)`],
  212. ["Permission", data.Permission],
  213. ["Last Modified", generateDisplayLastModTime(data.LastModTime)],
  214. ["File Type", "Folder"],
  215. ["Owner",data.Owner],
  216. ]
  217. ));
  218. }
  219. }
  220. }
  221. function hideLoader(){
  222. $("#filesizeLoader").hide();
  223. $("body").css('overflow-y',"auto");
  224. }
  225. //Model rendering scripts
  226. function ui_getInput(value, placeholder="", type="text"){
  227. return `<div class="ui fluid small input">
  228. <input type="${type}" placeholder="${placeholder}" value="${value}" readonly="true">
  229. </div>`
  230. }
  231. function ui_getText(value, color="black"){
  232. return `<p style="color:${color}; margin-bottom:0px;">${value}</p>`;
  233. }
  234. function ui_getDivider(){
  235. return `<div class="ui divider"></div>`;
  236. }
  237. //head is a 1D array and table is 2D array
  238. function ui_getTable(heads, table){
  239. html = `<table class="ui very basic fluid table">`;
  240. if (heads.length > 0){
  241. html += `<thead><tr>`;
  242. for (var i =0; i < heads.length; i++){
  243. html += `<th>${heads[i]}</th>`;
  244. }
  245. html += `</tr></thead>`;
  246. }
  247. html += `<tbody>`;
  248. for (var i =0; i < table.length; i++){
  249. html += `<tr>`;
  250. for (var j =0; j < table[i].length; j++){
  251. html += `<td style="word-break: break-all;">${table[i][j]}</td>`
  252. }
  253. html += `</tr>`;
  254. }
  255. html += `</tbody>
  256. </table>`;
  257. return html
  258. }
  259. function bytesToSize(bytes) {
  260. var sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];
  261. if (bytes == 0) return '0 Byte';
  262. var i = parseInt(Math.floor(Math.log(bytes) / Math.log(1024)));
  263. return Math.round(bytes / Math.pow(1024, i) * 100) / 100 + ' ' + sizes[i];
  264. }
  265. /*
  266. Updates Oct 2020 - Matching File Explorer Theme on other file system tabs
  267. */
  268. function fpw_toggleDarkTheme(){
  269. $("#filePropertiesWindow").css({
  270. "background-color":"#242330",
  271. "color":"white",
  272. });
  273. $("#filePropertiesWindow td,.header,p,div").css({
  274. "color":"white",
  275. });
  276. $("#filePropertiesWindow .input").addClass("inverted transparent big")
  277. }
  278. function fpw_loadPreference(key, callback){
  279. $.get("../../system/file_system/preference?key=" + key,function(data){
  280. callback(data);
  281. });
  282. }
  283. /*
  284. Updates 30 Jan 2021: Added change of file opener
  285. */
  286. //Open Opener Selector for the given file
  287. function changeDefaultWebApp(){
  288. var ext = fileInfo.Ext;
  289. var openFileList = [];
  290. var openFileObject = {
  291. filepath: fileInfo.VirtualPath,
  292. filename: fileInfo.Basename,
  293. }
  294. openFileList.push(openFileObject);
  295. var openParamter = encodeURIComponent(JSON.stringify(openFileObject));
  296. ao_module_newfw({
  297. url: "SystemAO/file_system/defaultOpener.html#" + openParamter,
  298. width: 320,
  299. height: 510,
  300. appicon: "SystemAO/file_system/img/opener.png",
  301. title: "Default WebApp for " + ext,
  302. parent: ao_module_windowID,
  303. callback: "handleRefresh"
  304. });
  305. }
  306. function handleRefresh(){
  307. //Default opener changed. Update the display
  308. initFileProperties();
  309. }
  310. /*
  311. Updates 9 April 2021: Added day compare for last modification days
  312. */
  313. function generateDisplayLastModTime(lastModTime){
  314. console.log(lastModTime);
  315. //Try to split the date into js date format
  316. var dateInfo = (lastModTime.split(" ")[0]).split("-");
  317. var modTime = new Date(dateInfo[0],dateInfo[1],dateInfo[2]);
  318. var diff = calcDate(new Date(), modTime);
  319. var displayText = "Unknown";
  320. if (diff[2] > 0){
  321. //years
  322. displayText = diff[2] + " year";
  323. if (diff[2] > 1){
  324. displayText += "s"
  325. }
  326. displayText += " ago";
  327. }else if (diff[1] > 0){
  328. //months
  329. displayText = diff[1] + " month";
  330. if (diff[1] > 1){
  331. displayText += "s"
  332. }
  333. displayText += " ago";
  334. }else if (diff [0] > 0){
  335. //days
  336. displayText = diff[0] + " day";
  337. if (diff[0] > 1){
  338. displayText += "s"
  339. }
  340. displayText += " ago";
  341. }else{
  342. //just now
  343. displayText = "Today"
  344. }
  345. return displayText + " (" + lastModTime + ")";
  346. }
  347. function calcDate(date1 = new Date(),date2) {
  348. var diff = Math.floor(date1.getTime() - date2.getTime());
  349. var day = 1000 * 60 * 60 * 24;
  350. var days = Math.floor(diff/day);
  351. var months = Math.floor(days/31);
  352. var years = Math.floor(months/12);
  353. return [days, months, years];
  354. }
  355. </script>
  356. </body>
  357. </html>