sharefiles.html 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. <!--
  2. Copyright (C) 2019-2022 Nicola Murino
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU Affero General Public License as published
  5. by the Free Software Foundation, version 3.
  6. This program is distributed in the hope that it will be useful,
  7. but WITHOUT ANY WARRANTY; without even the implied warranty of
  8. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  9. GNU Affero General Public License for more details.
  10. You should have received a copy of the GNU Affero General Public License
  11. along with this program. If not, see <https://www.gnu.org/licenses/>.
  12. -->
  13. {{template "base" .}}
  14. {{define "title"}}{{.Title}}{{end}}
  15. {{define "extra_css"}}
  16. <link href="{{.StaticURL}}/vendor/datatables/dataTables.bootstrap4.min.css" rel="stylesheet">
  17. <link href="{{.StaticURL}}/vendor/datatables/buttons.bootstrap4.min.css" rel="stylesheet">
  18. <link href="{{.StaticURL}}/vendor/datatables/fixedHeader.bootstrap4.min.css" rel="stylesheet">
  19. <link href="{{.StaticURL}}/vendor/datatables/responsive.bootstrap4.min.css" rel="stylesheet">
  20. <link href="{{.StaticURL}}/vendor/datatables/dataTables.checkboxes.css" rel="stylesheet">
  21. <link href="{{.StaticURL}}/vendor/filepond/filepond.min.css" rel="stylesheet" />
  22. <style>
  23. div.dataTables_wrapper span.selected-info,
  24. div.dataTables_wrapper span.selected-item {
  25. margin-left: 0.5em;
  26. }
  27. </style>
  28. {{end}}
  29. {{define "page_body"}}
  30. <div class="card shadow my-4">
  31. <div class="card-header py-3">
  32. <h6 class="m-0 font-weight-bold"><a href="{{.FilesURL}}?path=%2F"><i class="fas fa-home"></i>&nbsp;Home</a>&nbsp;{{range .Paths}}{{if eq .Href ""}}/{{.DirName}}{{else}}<a href="{{.Href}}">/{{.DirName}}</a>{{end}}{{end}}</h6>
  33. </div>
  34. <div class="card-body">
  35. {{if .Error}}
  36. <div class="card mb-4 border-left-warning">
  37. <div class="card-body text-form-error">{{.Error}}</div>
  38. </div>
  39. {{end}}
  40. <div id="errorMsg" class="card mb-4 border-left-warning" style="display: none;">
  41. <div id="errorTxt" class="card-body text-form-error"></div>
  42. </div>
  43. <div id="tableContainer" class="table-responsive">
  44. <table class="table table-hover nowrap" id="dataTable" width="100%" cellspacing="0">
  45. <thead>
  46. <tr>
  47. <th></th>
  48. <th>Type</th>
  49. <th>Name</th>
  50. <th>Size</th>
  51. <th>Last modified</th>
  52. </tr>
  53. </thead>
  54. </table>
  55. </div>
  56. </div>
  57. </div>
  58. {{end}}
  59. {{define "dialog"}}
  60. <div class="modal fade" id="uploadFilesModal" tabindex="-1" role="dialog" aria-labelledby="uploadFilesModalLabel"
  61. aria-hidden="true">
  62. <div class="modal-dialog" role="document">
  63. <div class="modal-content">
  64. <div class="modal-header">
  65. <h5 class="modal-title" id="uploadFilesModalLabel">
  66. Upload one or more files
  67. </h5>
  68. <button class="close" type="button" data-dismiss="modal" aria-label="Close">
  69. <span aria-hidden="true">&times;</span>
  70. </button>
  71. </div>
  72. <form id="upload_files_form" action="{{.FilesURL}}?path={{.CurrentDir}}" method="POST" enctype="multipart/form-data">
  73. <div class="modal-body">
  74. <div id="uploadErrorMsg" class="card mb-4 border-left-warning" style="display: none;">
  75. <div id="uploadErrorTxt" class="card-body text-form-error"></div>
  76. </div>
  77. <input type="file" id="files_name" name="filenames" required multiple>
  78. </div>
  79. <div class="modal-footer">
  80. <button class="btn btn-secondary" type="button" data-dismiss="modal">Cancel</button>
  81. <button type="submit" class="btn btn-primary">Submit</button>
  82. </div>
  83. </form>
  84. </div>
  85. </div>
  86. </div>
  87. <div class="modal fade" id="spinnerModal" tabindex="-1" role="dialog" data-keyboard="false" data-backdrop="static">
  88. <div class="modal-dialog modal-dialog-centered justify-content-center" role="document">
  89. <span style="color: #333333;" class="fa fa-spinner fa-spin fa-3x"></span>
  90. </div>
  91. </div>
  92. {{end}}
  93. {{define "extra_js"}}
  94. <script src="{{.StaticURL}}/vendor/datatables/jquery.dataTables.min.js"></script>
  95. <script src="{{.StaticURL}}/vendor/datatables/dataTables.bootstrap4.min.js"></script>
  96. <script src="{{.StaticURL}}/vendor/datatables/dataTables.buttons.min.js"></script>
  97. <script src="{{.StaticURL}}/vendor/datatables/buttons.bootstrap4.min.js"></script>
  98. <script src="{{.StaticURL}}/vendor/datatables/dataTables.fixedHeader.min.js"></script>
  99. <script src="{{.StaticURL}}/vendor/datatables/dataTables.responsive.min.js"></script>
  100. <script src="{{.StaticURL}}/vendor/datatables/responsive.bootstrap4.min.js"></script>
  101. <script src="{{.StaticURL}}/vendor/datatables/dataTables.checkboxes.min.js"></script>
  102. <script src="{{.StaticURL}}/vendor/filepond/filepond.min.js"></script>
  103. <script type="text/javascript">
  104. var spinnerDone = false;
  105. function shortenData(d, cutoff) {
  106. if ( typeof d !== 'string' ) {
  107. return d;
  108. }
  109. if ( d.length <= cutoff ) {
  110. return escapeHTML(d);
  111. }
  112. var shortened = d.substr(0, cutoff-1);
  113. return escapeHTML(shortened)+'&#8230;';
  114. }
  115. function getIconForFile(filename) {
  116. var extension = filename.slice((filename.lastIndexOf(".") - 1 >>> 0) + 2).toLowerCase();
  117. switch (extension) {
  118. case "doc":
  119. case "docx":
  120. case "odt":
  121. case "wps":
  122. return "far fa-file-word";
  123. case "ppt":
  124. case "pptx":
  125. return "far fa-file-powerpoint";
  126. case "xls":
  127. case "xlsx":
  128. case "ods":
  129. return "far fa-file-excel";
  130. case "pdf":
  131. return "far fa-file-pdf";
  132. case "webm":
  133. case "mkv":
  134. case "flv":
  135. case "vob":
  136. case "ogv":
  137. case "ogg":
  138. case "avi":
  139. case "ts":
  140. case "mov":
  141. case "wmv":
  142. case "asf":
  143. case "mpeg":
  144. case "mpv":
  145. case "3gp":
  146. case "mp4":
  147. return "far fa-file-video";
  148. case "jpeg":
  149. case "jpg":
  150. case "png":
  151. case "gif":
  152. case "webp":
  153. case "tiff":
  154. case "psd":
  155. case "bmp":
  156. case "svg":
  157. case "jp2":
  158. return "far fa-file-image";
  159. case "go":
  160. case "sh":
  161. case "bat":
  162. case "java":
  163. case "php":
  164. case "cs":
  165. case "asp":
  166. case "aspx":
  167. case "css":
  168. case "html":
  169. case "xhtml":
  170. case "htm":
  171. case "js":
  172. case "jsp":
  173. case "py":
  174. case "rb":
  175. case "cgi":
  176. case "c":
  177. case "cpp":
  178. case "h":
  179. case "hpp":
  180. case "kt":
  181. case "ktm":
  182. case "kts":
  183. case "swift":
  184. case "r":
  185. return "far fa-file-code";
  186. case "zip":
  187. case "zipx":
  188. case "7z":
  189. case "rar":
  190. case "tar":
  191. case "gz":
  192. case "bz2":
  193. case "zstd":
  194. case "zst":
  195. case "sz":
  196. case "lz":
  197. case "lz4":
  198. case "xz":
  199. case "jar":
  200. return "far fa-file-archive";
  201. case "txt":
  202. case "rtf":
  203. case "json":
  204. case "xml":
  205. case "yaml":
  206. case "toml":
  207. case "log":
  208. case "csv":
  209. case "ini":
  210. case "cfg":
  211. return "far fa-file-alt";
  212. default:
  213. return "far fa-file";
  214. }
  215. }
  216. function getNameFromMeta(meta) {
  217. return meta.split('_').slice(1).join('_');
  218. }
  219. const isDirectoryEntry = item => isEntry(item) && (getAsEntry(item) || {}).isDirectory;
  220. const isEntry = item => 'webkitGetAsEntry' in item;
  221. const getAsEntry = item => item.webkitGetAsEntry();
  222. $(document).ready(function () {
  223. $('#spinnerModal').on('shown.bs.modal', function () {
  224. if (spinnerDone){
  225. $('#spinnerModal').modal('hide');
  226. }
  227. });
  228. {{if gt .Scope 1}}
  229. FilePond.create(document.getElementById("files_name"),{
  230. allowMultiple: true,
  231. name: 'filenames',
  232. maxFiles: 30,
  233. credits: false,
  234. required: true,
  235. onwarning: function(error){
  236. if (error.code == 0){
  237. $('#uploadErrorTxt').text('You can upload a maximum of 30 files');
  238. $('#uploadErrorMsg').show();
  239. setTimeout(function () {
  240. $('#uploadErrorMsg').hide();
  241. }, 10000);
  242. }
  243. },
  244. beforeAddFile: (fileItem) => new Promise(resolve => {
  245. let num = 0;
  246. FilePond.find(document.getElementById("files_name")).getFiles().forEach(function(val){
  247. if (val.filename == fileItem.filename){
  248. num++;
  249. }
  250. });
  251. resolve(num == 1);
  252. })
  253. });
  254. $('#tableContainer').on("dragover", function(ev){
  255. ev.preventDefault();
  256. $('#tableContainer').css('opacity','0.5');
  257. });
  258. $('#tableContainer').on("dragend dragleave", function(ev){
  259. ev.preventDefault();
  260. $('#tableContainer').css('opacity','1');
  261. });
  262. $('#tableContainer').on("drop", function(ev){
  263. ev.preventDefault();
  264. $('#tableContainer').css('opacity','1');
  265. let filesDropped = false;
  266. if (ev.originalEvent.dataTransfer.items) {
  267. [...ev.originalEvent.dataTransfer.items].forEach((item, i) => {
  268. if (item.kind === 'file') {
  269. // if this is a directory just open the upload dialog
  270. if (!isDirectoryEntry(item)){
  271. FilePond.find(document.getElementById("files_name")).addFile(item.getAsFile());
  272. filesDropped = true;
  273. }
  274. }
  275. });
  276. } else {
  277. [...ev.originalEvent.dataTransfer.files].forEach((file, i) => {
  278. FilePond.find(document.getElementById("files_name")).addFile(file);
  279. filesDropped = true;
  280. });
  281. }
  282. if (filesDropped && !$('#uploadFilesModal').hasClass('show')){
  283. $('#uploadFilesModal').modal('show');
  284. }
  285. });
  286. {{end}}
  287. $("#upload_files_form").submit(function (event){
  288. event.preventDefault();
  289. var files = FilePond.find(document.getElementById("files_name")).getFiles();
  290. var has_errors = false;
  291. var index = 0;
  292. var success = 0;
  293. spinnerDone = false;
  294. $('#uploadFilesModal').modal('hide');
  295. $('#spinnerModal').modal('show');
  296. function uploadFile() {
  297. if (index >= files.length || has_errors){
  298. $('#spinnerModal').modal('hide');
  299. spinnerDone = true;
  300. if (!has_errors){
  301. location.reload();
  302. }
  303. return;
  304. }
  305. async function saveFile() {
  306. var errorMessage = "Error uploading files";
  307. let response;
  308. try {
  309. var f = files[index].file;
  310. var uploadPath = '{{.UploadBaseURL}}'+fixedEncodeURIComponent("/"+escapeHTML(f.name));
  311. var lastModified;
  312. try {
  313. lastModified = f.lastModified;
  314. } catch (e) {
  315. console.log("unable to get last modified time from file: "+e.message);
  316. lastModified = "";
  317. }
  318. response = await fetch(uploadPath, {
  319. method: 'POST',
  320. headers: {
  321. 'X-SFTPGO-MTIME': lastModified
  322. },
  323. credentials: 'same-origin',
  324. redirect: 'error',
  325. body: f
  326. });
  327. } catch (e){
  328. throw Error(errorMessage+": " +e.message);
  329. }
  330. if (response.status == 201){
  331. index++;
  332. success++;
  333. uploadFile();
  334. } else {
  335. let jsonResponse;
  336. try {
  337. jsonResponse = await response.json();
  338. } catch(e){
  339. throw Error(errorMessage);
  340. }
  341. if (jsonResponse.message) {
  342. errorMessage = jsonResponse.message;
  343. }
  344. if (jsonResponse.error) {
  345. errorMessage += ": " + jsonResponse.error;
  346. }
  347. throw Error(errorMessage);
  348. }
  349. }
  350. saveFile().catch(function(error){
  351. index++;
  352. has_errors = true;
  353. $('#errorTxt').text(error.message);
  354. $('#errorMsg').show();
  355. setTimeout(function () {
  356. $('#errorMsg').hide();
  357. }, 10000);
  358. uploadFile();
  359. });
  360. }
  361. uploadFile();
  362. });
  363. $.fn.dataTable.ext.buttons.refresh = {
  364. text: '<i class="fas fa-sync-alt"></i>',
  365. name: 'refresh',
  366. titleAttr: "Refresh",
  367. action: function (e, dt, node, config) {
  368. location.reload();
  369. }
  370. };
  371. $.fn.dataTable.ext.buttons.download = {
  372. text: '<i class="fas fa-download"></i>',
  373. name: 'download',
  374. titleAttr: "Download zip",
  375. action: function (e, dt, node, config) {
  376. var filesArray = [];
  377. var selected = dt.column(0).checkboxes.selected();
  378. for (i = 0; i < selected.length; i++) {
  379. filesArray.push(getNameFromMeta(selected[i]));
  380. }
  381. var files = encodeURIComponent(JSON.stringify(filesArray));
  382. var downloadURL = '{{.DownloadURL}}';
  383. var currentDir = '{{.CurrentDir}}';
  384. var ts = new Date().getTime().toString();
  385. window.open(`${downloadURL}?path=${currentDir}&files=${files}&_=${ts}`);
  386. },
  387. enabled: false
  388. };
  389. $.fn.dataTable.ext.buttons.addFiles = {
  390. text: '<i class="fas fa-file-upload"></i>',
  391. name: 'addFiles',
  392. titleAttr: "Upload files",
  393. action: function (e, dt, node, config) {
  394. //FilePond.find(document.getElementById("files_name")).removeFiles();
  395. $('#uploadFilesModal').modal('show');
  396. },
  397. enabled: true
  398. };
  399. var table = $('#dataTable').DataTable({
  400. "ajax": {
  401. "url": "{{.DirsURL}}?path={{.CurrentDir}}",
  402. "dataSrc": "",
  403. "error": function ($xhr, textStatus, errorThrown) {
  404. $(".dataTables_processing").hide();
  405. var txt = "Failed to get directory listing";
  406. if ($xhr) {
  407. var json = $xhr.responseJSON;
  408. if (json) {
  409. if (json.message){
  410. txt += ": " + json.message;
  411. } else {
  412. txt += ": " + json.error;
  413. }
  414. }
  415. }
  416. $('#errorTxt').text(txt);
  417. $('#errorMsg').show();
  418. setTimeout(function () {
  419. $('#errorMsg').hide();
  420. }, 10000);
  421. }
  422. },
  423. "deferRender": true,
  424. "processing": true,
  425. "lengthMenu": [ 10, 25, 50, 100, 250, 500 ],
  426. "stateSave": true,
  427. "stateDuration": 0,
  428. "stateSaveParams": function (settings, data) {
  429. data.sftpgo_dir = '{{.CurrentDir}}';
  430. },
  431. "stateLoadParams": function (settings, data) {
  432. if (!data.sftpgo_dir || data.sftpgo_dir != '{{.CurrentDir}}'){
  433. data.start = 0;
  434. data.search.search = "";
  435. }
  436. data.checkboxes = [];
  437. },
  438. "columns": [
  439. { "data": "meta" },
  440. { "data": "type" },
  441. {
  442. "data": "name",
  443. "render": function (data, type, row) {
  444. if (type === 'display') {
  445. var title = "";
  446. var cssClass = "";
  447. var shortened = shortenData(data, 70);
  448. data = escapeHTML(data);
  449. if (shortened != data){
  450. title = escapeHTML(data);
  451. cssClass = "ellipsis";
  452. }
  453. if (row["type"] == "1") {
  454. return `<i class="fas fa-folder"></i>&nbsp;<a class="${cssClass}" href="${row['url']}" title="${title}">${shortened}</a>`;
  455. }
  456. if (row["size"] == "") {
  457. return `<i class="fas fa-external-link-alt"></i>&nbsp;<a class="${cssClass}" href="${row['url']}" title="${title}">${shortened}</a>`;
  458. }
  459. var icon = getIconForFile(data);
  460. return `<i class="${icon}"></i>&nbsp;<a class="${cssClass}" href="${row['url']}" title="${title}">${shortened}</a>`;
  461. }
  462. return data;
  463. }
  464. },
  465. { "data": "size" },
  466. { "data": "last_modified" }
  467. ],
  468. "buttons": [],
  469. "lengthChange": true,
  470. "columnDefs": [
  471. {
  472. "targets": [0],
  473. "checkboxes": {
  474. "selectCallback": function (nodes, selected) {
  475. var selectedItems = table.column(0).checkboxes.selected().length;
  476. var selectedText = "";
  477. if (selectedItems == 1) {
  478. selectedText = "1 item selected";
  479. } else if (selectedItems > 1) {
  480. selectedText = `${selectedItems} items selected`;
  481. }
  482. table.button('download:name').enable(selectedItems > 0);
  483. $('#dataTable_info').find('span').remove();
  484. $("#dataTable_info").append('<span class="selected-info"><span class="selected-item">' + selectedText + '</span></span>');
  485. }
  486. },
  487. "orderable": false,
  488. "searchable": false
  489. },
  490. {
  491. "targets": [1],
  492. "visible": false,
  493. "searchable": false
  494. },
  495. {
  496. "targets": [3, 4],
  497. "searchable": false
  498. }
  499. ],
  500. "scrollX": false,
  501. "scrollY": false,
  502. "responsive": true,
  503. "language": {
  504. "loadingRecords": "",
  505. "emptyTable": "No files or folders"
  506. },
  507. "initComplete": function (settings, json) {
  508. table.button().add(0, 'refresh');
  509. //table.button().add(0, 'pageLength');
  510. table.button().add(0, 'download');
  511. {{if gt .Scope 1}}
  512. table.button().add(0, 'addFiles');
  513. {{end}}
  514. table.buttons().container().appendTo('.col-md-6:eq(0)', table.table().container());
  515. },
  516. "orderFixed": [1, 'asc'],
  517. "order": [2, 'asc']
  518. });
  519. new $.fn.dataTable.FixedHeader(table);
  520. $.fn.dataTable.ext.errMode = 'none';
  521. });
  522. </script>
  523. {{end}}