s3manager-web/web/template/bucket.html.tmpl

343 lines
12 KiB
Cheetah
Raw Normal View History

2016-12-18 22:54:21 +01:00
{{ define "content" }}
2023-06-13 17:37:57 +02:00
<style>
.breadcrumb:before {
2023-06-20 07:18:56 +02:00
content: '/';
}
#notifications {
top: 20px;
right: 30px;
position: fixed;
z-index: 2
2023-06-13 17:37:57 +02:00
}
</style>
<nav class="nav-extended">
2017-05-08 23:07:07 +02:00
<div class="nav-wrapper container">
2023-06-13 17:37:57 +02:00
<a href="/buckets/{{$.BucketName}}" class="brand-logo center"><i class="material-icons">folder_open</i>{{ .BucketName }}</a>
2017-05-08 23:07:07 +02:00
{{ if not .Objects }}
2019-08-18 14:26:29 +02:00
<ul class="right">
<li>
<a class="waves-effect waves-light btn" href="#" onclick="deleteBucket({{ .BucketName }})">
Delete <i class="material-icons right">delete</i>
</a>
</li>
</ul>
2017-05-08 23:07:07 +02:00
{{ end }}
</div>
2016-12-21 00:32:40 +01:00
2023-06-13 17:37:57 +02:00
<div class="nav-wrapper container">
<a href="/buckets" class="breadcrumb"><i class="material-icons">arrow_back</i> buckets </a>
{{ $url := printf "/buckets/%s/" $.BucketName }}
<a href="{{ $url }}" class="breadcrumb">{{ $.BucketName }}</a>
{{ range $index, $path := .Paths }}
{{ $url = printf "%s%s/" $url $path }}
<a href="{{ $url }}" class="breadcrumb">{{ $path }}</a>
{{ end }}
</div>
</div>
</nav>
2016-12-21 00:32:40 +01:00
2023-06-20 07:18:56 +02:00
<div class="section" style="margin: 10px; position: relative;">
2017-05-08 23:07:07 +02:00
{{ if .Objects }}
2019-08-18 14:26:29 +02:00
<table class="striped">
2017-05-08 23:07:07 +02:00
<thead>
<tr>
<th>Key</th>
<th>Size</th>
<th>Owner</th>
<th>Last Modified</th>
2019-08-18 14:26:29 +02:00
<th style="min-width:165px;"></th>
2017-05-08 23:07:07 +02:00
</tr>
</thead>
2016-12-21 00:32:40 +01:00
2017-05-08 23:07:07 +02:00
<tbody>
{{ range $index, $object := .Objects }}
<tr>
2023-06-13 17:37:57 +02:00
<td
{{ if $object.IsFolder }}
onclick="location.href='/buckets/{{ $.BucketName }}/{{ $object.Key }}'"}
style="cursor:pointer;"
{{ end }}>
<i class="material-icons">{{ $object.Icon }}</i> {{ $object.DisplayName }}
</td>
<td>{{ $object.Size }} bytes</td>
<td>{{ $object.Owner }}</td>
<td>{{ $object.LastModified }}</td>
2017-05-08 23:07:07 +02:00
<td>
2023-06-20 07:18:56 +02:00
{{ if not $object.IsFolder }}
<button class="dropdown-trigger waves-effect waves-teal btn" data-target="actions-dropdown-{{ $index }}">
Actions <i class="material-icons right">arrow_drop_down</i>
</button>
<!-- Dropdown Structure -->
<ul id="actions-dropdown-{{ $index }}" class="dropdown-content">
<li><a target="_blank" href="/api/buckets/{{ $.BucketName }}/objects/{{ $object.Key }}">Download</a></li>
2023-08-20 14:03:02 +02:00
<li><a onclick="handleOpenDownloadLinkModal({{ $object.Key }})">Download link</a></li>
2023-06-20 07:18:56 +02:00
{{- if $.AllowDelete }}
<li><a href="#" onclick="deleteObject({{ $.BucketName }}, {{ $object.Key }})">Delete</a></li>
{{- end }}
</ul>
{{ end }}
2017-05-08 23:07:07 +02:00
</td>
</tr>
{{ end }}
</tbody>
</table>
{{ end }}
2016-12-21 00:32:40 +01:00
2017-05-08 23:07:07 +02:00
{{ if not .Objects }}
2023-06-20 07:18:56 +02:00
<p style="text-align:center;margin-top:2em;color:gray;">No objects in <strong>{{ .BucketName }}/{{ .CurrentPath }}</strong> yet</p>
2017-05-08 23:07:07 +02:00
{{ end }}
2016-12-21 00:32:40 +01:00
2023-06-20 07:18:56 +02:00
<div id="notifications">
<div id="notification-template" class="card" style="display: none;">
<div class="card-content" style="padding: 12px">
<span class="card-title">Uploading</span>
<p> I am a very simple card </p>
<div class="progress">
<div class="indeterminate"></div>
</div>
</div>
</div>
</div>
2016-12-21 00:32:40 +01:00
</div>
<div class="fixed-action-btn">
2023-07-16 15:09:13 +02:00
<button type="button" class="btn-floating btn-large red tooltipped" id="upload-file-btn" data-position="top" data-tooltip="Upload files">
2017-05-08 23:07:07 +02:00
<i class="large material-icons">add</i>
2019-08-18 14:26:29 +02:00
</button>
2023-06-20 07:18:56 +02:00
2023-07-16 15:09:13 +02:00
<button type="button" class="btn-floating btn-large red tooltipped" id="upload-folder-btn" data-position="top" data-tooltip="Upload folder">
2023-06-20 07:18:56 +02:00
<i class="large material-icons">create_new_folder</i>
</button>
2017-05-08 23:07:07 +02:00
2023-08-20 14:03:02 +02:00
<button type="button" class="btn-floating btn-large red modal-trigger tooltipped" data-target="modal-change-path" data-position="top" data-tooltip="Change path">
2023-07-16 15:09:13 +02:00
<i class="large material-icons">create</i>
</button>
2016-12-21 00:32:40 +01:00
</div>
2016-12-18 22:54:21 +01:00
2023-07-16 15:09:13 +02:00
<input type="file" id="upload-folder-input" webkitdirectory multiple style="display: none;">
<input type="file" id="upload-file-input" name="file" multiple style="display: none;">
2023-08-20 14:03:02 +02:00
<div id="modal-change-path" class="modal">
<form id="change-path-form" enctype="multipart/form-data">
2023-06-20 07:18:56 +02:00
<div class="modal-content">
2023-07-16 15:09:13 +02:00
<h4>Change directory path</h4>
2023-06-20 07:18:56 +02:00
<br>
<div class="row">
<div class="col s6">
<div class="input-field">
<input name="new-path" id="new-path" type="text">
<label for="new-path">Path</label>
</div>
</div>
</div>
</div>
<div class="modal-footer">
<button type="button" class="modal-close waves-effect waves-green btn-flat">Cancel</button>
2023-08-20 14:03:02 +02:00
<button type="submit" class="modal-close waves-effect waves-green btn">Change path</button>
2023-06-20 07:18:56 +02:00
</div>
2023-08-20 14:03:02 +02:00
</form>
</div>
2023-06-20 07:18:56 +02:00
2023-08-20 14:03:02 +02:00
<div id="modal-create-download-link" class="modal">
<form id="download-link-form">
<div class="modal-content">
<div class="row">
<h4>Create download link for </h4>
<input name="objectName" id="objectName" type="text" readonly>
</div>
<div class="row">
<div class="col s4">
<div class="input-field">
<input name="day" id="gen-link-day" value="0" type="text" pattern="^[\d]+$" class="validate">
<label for="gen-link-day">Day</label>
<span class="helper-text" data-error="Invalid day format"></span>
</div>
</div>
<div class="col s4">
<div class="input-field">
<input name="hour" id="gen-link-hour" value="1" type="text" pattern="^([0-9]|1\d|2[0-3])$" class="validate">
<label for="gen-link-hour">Hour</label>
<span class="helper-text" data-error="Invalid hour format"></span>
</div>
</div>
<div class="col s4">
<div class="input-field">
<input name="minute" id="gen-link-minute" value="0" type="text" pattern="^([0-9]|[1-5]\d)$" class="validate">
<label for="gen-link-minute">Minute</label>
<span class="helper-text" data-error="Invalid minute format"></span>
</div>
</div>
</div>
<div class="row">
<div class="col s3">
<button id="create-link-btn" class="waves-effect waves-green btn">Create link</button>
</div>
<div class="col s9 red-text text-darken-2" id="gen-url-error"></div>
</div>
<div class="row">
<div class="col s11">
<div class="input-field">
<i class="material-icons prefix" onclick="handleCopyLink()" style="cursor:pointer;">content_copy</i>
<input name="generated-link" id="generated-link" type="text" readonly>
</div>
</div>
</div>
</div>
2023-06-20 07:18:56 +02:00
</form>
</div>
2016-12-21 00:32:40 +01:00
<script>
2017-05-08 23:07:07 +02:00
function deleteObject(bucketName, objectName) {
2016-12-21 00:32:40 +01:00
$.ajax({
2017-05-08 23:07:07 +02:00
type: 'DELETE',
url: '/api/buckets/' + bucketName + '/objects/' + objectName,
success: function () { location.reload(); }
2016-12-21 00:32:40 +01:00
})
2017-05-08 23:07:07 +02:00
}
2023-08-20 14:03:02 +02:00
2017-05-08 23:07:07 +02:00
function deleteBucket(bucketName) {
2016-12-21 22:29:12 +01:00
$.ajax({
2017-05-08 23:07:07 +02:00
type: 'DELETE',
url: '/api/buckets/' + bucketName,
success: function () { window.location.replace('/buckets'); }
2016-12-21 22:29:12 +01:00
})
2017-05-08 23:07:07 +02:00
}
2023-06-20 07:18:56 +02:00
2023-07-16 15:09:13 +02:00
function handleUploadFiles(event) {
files = event.target.files
url = "/api/buckets/{{ .BucketName }}/objects"
uploadFiles(files, url);
}
2023-08-20 14:03:02 +02:00
function handleChangePath(event) {
2023-06-20 07:18:56 +02:00
event.preventDefault();
const form = event.target;
const formData = new FormData(form);
2023-08-20 14:03:02 +02:00
let appendPath = formData.get("new-path")
if(!appendPath.endsWith("/")) {
appendPath = appendPath + "/";
}
let currentPath = window.location.href
if(!currentPath.endsWith("/")) {
currentPath = currentPath + "/";
2023-06-20 07:18:56 +02:00
}
2023-08-20 14:03:02 +02:00
form.reset();
window.location.href = currentPath + appendPath;
2023-06-20 07:18:56 +02:00
}
2023-07-16 15:09:13 +02:00
function uploadFiles(files, url) {
uploadPromises = [];
for(file of files) {
uploadPromises.push(uploadFile(file, url));
}
Promise.all(uploadPromises).then(values => {
window.location.reload();
});
}
2023-06-20 07:18:56 +02:00
2023-07-16 15:09:13 +02:00
function uploadFile(file, url) {
const formData = new FormData();
formData.append('file', file);
if( !!file.webkitRelativePath ) {
formData.append('path', "{{ .CurrentPath }}" + file.webkitRelativePath );
} else {
formData.append('path', "{{ .CurrentPath }}" + file.name );
}
const notification = createNotification(file.name);
notifications = document.getElementById('notifications');
notifications.appendChild(notification);
return fetch(url, {
2023-06-20 07:18:56 +02:00
method: "POST",
body: formData
}).then(response => {
2023-07-16 15:09:13 +02:00
notifications.removeChild(notification);
2023-06-20 07:18:56 +02:00
})
}
function createNotification(fileName) {
notificationTemplate = document.getElementById('notification-template');
2023-07-16 15:09:13 +02:00
notification = notificationTemplate.cloneNode(true);
notification.getElementsByTagName('p')[0].setHTML(fileName);
notification.removeAttribute("id");
notification.removeAttribute("style");
return notification;
2023-06-20 07:18:56 +02:00
}
2023-08-20 14:03:02 +02:00
function handleOpenDownloadLinkModal(objectName) {
const downloadLinkForm = document.forms['download-link-form']
downloadLinkForm.elements['objectName'].value = objectName;
const createLinkModalElement = document.getElementById('modal-create-download-link')
document.getElementById('generated-link').setAttribute('value', "");
document.getElementById('gen-url-error').setHTML("");
const modalInstance = M.Modal.init(createLinkModalElement);
modalInstance.open()
}
function handleGenerateDownloadLink(event) {
event.preventDefault();
const form = event.target;
const formData = new FormData(form);
const objectName = formData.get('objectName');
const genUrlMessage = document.getElementById('gen-url-error');
const expiryTime = formData.get('day') * 24 * 60 * 60 + formData.get('hour') * 60 * 60 + formData.get('minute') * 60;
if(expiryTime > 7 * 24 * 60 * 60) {
genUrlMessage.setHTML("Expiry time must be less than 7 days");
return;
}
$.ajax({
type: 'GET',
url: '/api/buckets/' + {{ $.BucketName }}+ "/objects/" + objectName + "/url?expiry=" + expiryTime,
success: function (result) {
genUrlMessage.setHTML("")
document.getElementById("generated-link").setAttribute('value', JSON.parse(result).url);
},
error: function(request, status, error) {
genUrlMessage.setHTML("Error when generating url")
}
});
}
function handleCopyLink() {
const url = document.getElementById("generated-link").value;
if(!!url) {
navigator.clipboard.writeText(url).then(function() {
M.toast({html: 'Copied to clipboard!'});
}, function(err) {
console.error('Could not copy:', err);
});
}
}
2023-06-20 07:18:56 +02:00
window.onload = (event) => {
2023-08-20 14:03:02 +02:00
$('#change-path-form').submit(handleChangePath)
$('#download-link-form').submit(handleGenerateDownloadLink)
2023-07-16 15:09:13 +02:00
uploadFolderInput = $('#upload-folder-input');
$('#upload-folder-btn').click(event => uploadFolderInput.click());
uploadFolderInput.change(handleUploadFiles);
uploadFileInput = $('#upload-file-input');
$('#upload-file-btn').click(event => uploadFileInput.click());
uploadFileInput.change(handleUploadFiles);
$(document).ready(function(){
$('.tooltipped').tooltip();
});
2023-06-20 07:18:56 +02:00
};
2016-12-21 00:32:40 +01:00
</script>
2016-12-18 22:54:21 +01:00
{{ end }}