mirror of
https://github.com/filegator/filegator.git
synced 2025-10-24 11:26:20 +02:00
595 lines
18 KiB
Vue
595 lines
18 KiB
Vue
<template>
|
|
<div id="dropzone" class="container"
|
|
@dragover="dropZone = can('upload') && ! isLoading ? true : false"
|
|
@dragleave="dropZone = false"
|
|
@drop="dropZone = false"
|
|
>
|
|
<div v-if="isLoading" id="loading" />
|
|
|
|
<Upload v-if="can('upload')" v-show="dropZone == false" :files="files" :drop-zone="dropZone" />
|
|
|
|
<b-upload v-if="dropZone && ! isLoading" multiple drag-drop @input="files = $event">
|
|
<b class="drop-info">{{ lang('Drop files to upload') }}</b>
|
|
</b-upload>
|
|
|
|
<div v-if="!dropZone" class="container">
|
|
<Menu />
|
|
|
|
<div id="browser">
|
|
<div v-if="can('read')" class="is-flex is-justify-between">
|
|
<div class="breadcrumb" aria-label="breadcrumbs">
|
|
<ul>
|
|
<li v-for="(item, index) in breadcrumbs" :key="index">
|
|
<a @click="goTo(item.path)">{{ item.name }}</a>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
<div>
|
|
<a id="search" class="search-btn" @click="search">
|
|
<b-icon icon="search" size="is-small" />
|
|
</a>
|
|
<a id="sitemap" class="is-paddingless" @click="selectDir">
|
|
<b-icon icon="sitemap" size="is-small" />
|
|
</a>
|
|
</div>
|
|
</div>
|
|
|
|
<section id="multi-actions" class="is-flex is-justify-between">
|
|
<div>
|
|
<b-field v-if="can('upload') && ! checked.length" class="file is-inline-block">
|
|
<b-upload multiple native @input="files = $event">
|
|
<a v-if="! checked.length" class="is-inline-block">
|
|
<b-icon icon="upload" size="is-small" /> {{ lang('Add files') }}
|
|
</a>
|
|
</b-upload>
|
|
</b-field>
|
|
<a v-if="can(['read', 'write']) && ! checked.length" class="add-new is-inline-block">
|
|
<b-dropdown :disabled="checked.length > 0" aria-role="list">
|
|
<span slot="trigger">
|
|
<b-icon icon="plus" size="is-small" /> {{ lang('New') }}
|
|
</span>
|
|
|
|
<b-dropdown-item aria-role="listitem" @click="create('dir')">
|
|
<b-icon icon="folder" size="is-small" /> {{ lang('Folder') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item aria-role="listitem" @click="create('file')">
|
|
<b-icon icon="file" size="is-small" /> {{ lang('File') }}
|
|
</b-dropdown-item>
|
|
|
|
</b-dropdown>
|
|
</a>
|
|
<a v-if="can('batchdownload') && checked.length" class="is-inline-block" @click="batchDownload">
|
|
<b-icon icon="download" size="is-small" /> {{ lang('Download') }}
|
|
</a>
|
|
<a v-if="can('write') && checked.length" class="is-inline-block" @click="copy">
|
|
<b-icon icon="copy" size="is-small" /> {{ lang('Copy') }}
|
|
</a>
|
|
<a v-if="can('write') && checked.length" class="is-inline-block" @click="move">
|
|
<b-icon icon="external-link-square-alt" size="is-small" /> {{ lang('Move') }}
|
|
</a>
|
|
<a v-if="can(['write', 'zip']) && checked.length" class="is-inline-block" @click="zip">
|
|
<b-icon icon="file-archive" size="is-small" /> {{ lang('Zip') }}
|
|
</a>
|
|
<a v-if="can('write') && checked.length" class="is-inline-block" @click="remove">
|
|
<b-icon icon="trash-alt" size="is-small" /> {{ lang('Delete') }}
|
|
</a>
|
|
</div>
|
|
<div id="pagination" v-if="can('read')">
|
|
<Pagination :perpage="perPage" @selected="perPage = $event" />
|
|
</div>
|
|
</section>
|
|
|
|
<b-table v-if="can('read')"
|
|
:data="content"
|
|
:default-sort="defaultSort"
|
|
:paginated="perPage > 0"
|
|
:per-page="perPage"
|
|
:current-page.sync="currentPage"
|
|
:hoverable="true"
|
|
:is-row-checkable="(row) => row.type != 'back'"
|
|
:row-class="(row) => 'file-row type-'+row.type"
|
|
:checked-rows.sync="checked"
|
|
:loading="isLoading"
|
|
:checkable="can('batchDownload') || can('write') || can('zip')"
|
|
>
|
|
<template slot-scope="props">
|
|
<b-table-column :label="lang('Name')" :custom-sort="sortByName" field="data.name" sortable>
|
|
<a class="is-block name" @click="itemClick(props.row)">
|
|
{{ props.row.name }}
|
|
</a>
|
|
</b-table-column>
|
|
|
|
<b-table-column :label="lang('Size')" :custom-sort="sortBySize" field="data.size" sortable numeric width="150">
|
|
{{ props.row.type == 'back' || props.row.type == 'dir' ? lang('Folder') : formatBytes(props.row.size) }}
|
|
</b-table-column>
|
|
|
|
<b-table-column :label="lang('Time')" :custom-sort="sortByTime" field="data.time" sortable numeric width="200">
|
|
{{ props.row.time ? formatDate(props.row.time) : '' }}
|
|
</b-table-column>
|
|
|
|
<b-table-column id="single-actions" width="51">
|
|
<b-dropdown v-if="props.row.type != 'back'" :disabled="checked.length > 0" aria-role="list" position="is-bottom-left">
|
|
<button slot="trigger" class="button is-small">
|
|
<b-icon icon="ellipsis-h" size="is-small" />
|
|
</button>
|
|
|
|
<b-dropdown-item v-if="props.row.type == 'file' && can('download')" aria-role="listitem" @click="download(props.row)">
|
|
<b-icon icon="download" size="is-small" /> {{ lang('Download') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="props.row.type == 'file' && can(['download']) && hasPreview(props.row.path)" aria-role="listitem" @click="preview(props.row)">
|
|
<b-icon icon="file-alt" size="is-small" /> {{ lang('View') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="can('write')" aria-role="listitem" @click="copy($event, props.row)">
|
|
<b-icon icon="copy" size="is-small" /> {{ lang('Copy') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="can('write')" aria-role="listitem" @click="move($event, props.row)">
|
|
<b-icon icon="external-link-square-alt" size="is-small" /> {{ lang('Move') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="can('write')" aria-role="listitem" @click="rename($event, props.row)">
|
|
<b-icon icon="file-signature" size="is-small" /> {{ lang('Rename') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="can(['write', 'zip']) && isArchive(props.row)" aria-role="listitem" @click="unzip($event, props.row)">
|
|
<b-icon icon="file-archive" size="is-small" /> {{ lang('Unzip') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="can(['write', 'zip']) && ! isArchive(props.row)" aria-role="listitem" @click="zip($event, props.row)">
|
|
<b-icon icon="file-archive" size="is-small" /> {{ lang('Zip') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="can('write')" aria-role="listitem" @click="remove($event, props.row)">
|
|
<b-icon icon="trash-alt" size="is-small" /> {{ lang('Delete') }}
|
|
</b-dropdown-item>
|
|
<b-dropdown-item v-if="props.row.type == 'file' && can('download')" v-clipboard:copy="getDownloadLink(props.row.path)" aria-role="listitem">
|
|
<b-icon icon="clipboard" size="is-small" /> {{ lang('Copy link') }}
|
|
</b-dropdown-item>
|
|
</b-dropdown>
|
|
</b-table-column>
|
|
</template>
|
|
|
|
<template slot="bottom-left">
|
|
<span>{{ lang('Selected', checked.length, totalCount) }}</span>
|
|
</template>
|
|
</b-table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import Vue from 'vue'
|
|
import Menu from './partials/Menu'
|
|
import Tree from './partials/Tree'
|
|
import Editor from './partials/Editor'
|
|
import Gallery from './partials/Gallery'
|
|
import Search from './partials/Search'
|
|
import Pagination from './partials/Pagination'
|
|
import Upload from './partials/Upload'
|
|
import api from '../api/api'
|
|
import VueClipboard from 'vue-clipboard2'
|
|
import _ from 'lodash'
|
|
|
|
Vue.use(VueClipboard)
|
|
|
|
export default {
|
|
name: 'Browser',
|
|
components: { Menu, Pagination, Upload },
|
|
data() {
|
|
return {
|
|
dropZone: false,
|
|
perPage: '',
|
|
currentPage: 1,
|
|
checked: [],
|
|
isLoading: false,
|
|
defaultSort: ['data.name', 'desc'],
|
|
files: [],
|
|
}
|
|
},
|
|
computed: {
|
|
breadcrumbs() {
|
|
let path = ''
|
|
let breadcrumbs = [{name: this.lang('Home'), path: '/'}]
|
|
|
|
_.forEach(_.split(this.$store.state.cwd.location, '/'), (dir) => {
|
|
path += dir + '/'
|
|
breadcrumbs.push({
|
|
name: dir,
|
|
path: path,
|
|
})
|
|
})
|
|
|
|
return _.filter(breadcrumbs, o => o.name)
|
|
},
|
|
content() {
|
|
return this.$store.state.cwd.content
|
|
},
|
|
totalCount() {
|
|
return Number(_.sumBy(this.$store.state.cwd.content, (o) => {
|
|
return o.type == 'file' || o.type == 'dir'
|
|
}))
|
|
},
|
|
},
|
|
watch: {
|
|
'$route' (to) {
|
|
this.isLoading = true
|
|
this.checked = []
|
|
this.currentPage = 1
|
|
api.changeDir({
|
|
to: to.query.cd
|
|
})
|
|
.then(ret => {
|
|
this.$store.commit('setCwd', {
|
|
content: ret.files,
|
|
location: ret.location,
|
|
})
|
|
this.isLoading = false
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
},
|
|
},
|
|
mounted() {
|
|
if (this.can('read')) {
|
|
this.loadFiles()
|
|
}
|
|
},
|
|
methods: {
|
|
loadFiles() {
|
|
api.getDir({
|
|
to: '',
|
|
})
|
|
.then(ret => {
|
|
this.$store.commit('setCwd', {
|
|
content: ret.files,
|
|
location: ret.location,
|
|
})
|
|
})
|
|
.catch(error => this.handleError(error))
|
|
},
|
|
goTo(path) {
|
|
this.$router.push({ name: 'browser', query: { 'cd': path }}).catch(() => {})
|
|
},
|
|
getSelected() {
|
|
return _.reduce(this.checked, function(result, value) {
|
|
result.push(value)
|
|
return result
|
|
}, [])
|
|
},
|
|
itemClick(item) {
|
|
if (item.type == 'dir' || item.type == 'back') {
|
|
this.goTo(item.path)
|
|
} else if (this.can(['download']) && this.hasPreview(item.path)) {
|
|
this.preview(item)
|
|
} else if (this.can(['download'])) {
|
|
this.download(item)
|
|
}
|
|
},
|
|
selectDir() {
|
|
this.$modal.open({
|
|
parent: this,
|
|
hasModalCard: true,
|
|
component: Tree,
|
|
events: {
|
|
selected: dir => {
|
|
this.goTo(dir.path)
|
|
}
|
|
},
|
|
})
|
|
},
|
|
copy(event, item) {
|
|
this.$modal.open({
|
|
parent: this,
|
|
hasModalCard: true,
|
|
component: Tree,
|
|
events: {
|
|
selected: dir => {
|
|
this.isLoading = true
|
|
api.copyItems({
|
|
destination: dir.path,
|
|
items: item ? [item] : this.getSelected(),
|
|
})
|
|
.then(() => {
|
|
this.isLoading = false
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
this.checked = []
|
|
}
|
|
},
|
|
})
|
|
},
|
|
move(event, item) {
|
|
this.$modal.open({
|
|
parent: this,
|
|
hasModalCard: true,
|
|
component: Tree,
|
|
events: {
|
|
selected: dir => {
|
|
this.isLoading = true
|
|
api.moveItems({
|
|
destination: dir.path,
|
|
items: item ? [item] : this.getSelected(),
|
|
})
|
|
.then(() => {
|
|
this.isLoading = false
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
this.checked = []
|
|
}
|
|
},
|
|
})
|
|
},
|
|
batchDownload() {
|
|
let items = this.getSelected()
|
|
|
|
this.isLoading = true
|
|
api.batchDownload({
|
|
items: items,
|
|
})
|
|
.then(ret => {
|
|
this.isLoading = false
|
|
this.$dialog.alert({
|
|
message: this.lang('Your file is ready'),
|
|
confirmText: this.lang('Download'),
|
|
onConfirm: () => {
|
|
window.open(Vue.config.baseURL+'/batchdownload&uniqid='+ret.uniqid, '_blank')
|
|
}
|
|
})
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
},
|
|
download(item) {
|
|
window.open(this.getDownloadLink(item.path), '_blank')
|
|
},
|
|
search() {
|
|
this.$modal.open({
|
|
parent: this,
|
|
hasModalCard: true,
|
|
component: Search,
|
|
events: {
|
|
selected: item => {
|
|
this.goTo(item.dir)
|
|
}
|
|
},
|
|
})
|
|
},
|
|
preview(item) {
|
|
let modal = null
|
|
if (this.isImage(item.path)) {
|
|
modal = Gallery
|
|
}
|
|
if (this.isText(item.path)) {
|
|
modal = Editor
|
|
}
|
|
this.$modal.open({
|
|
parent: this,
|
|
props: { item: item },
|
|
hasModalCard: true,
|
|
component: modal,
|
|
})
|
|
},
|
|
isArchive(item) {
|
|
return item.type == 'file' && item.name.split('.').pop() == 'zip'
|
|
},
|
|
unzip(event, item) {
|
|
this.$dialog.confirm({
|
|
message: this.lang('Are you sure you want to do this?'),
|
|
type: 'is-danger',
|
|
cancelText: this.lang('Cancel'),
|
|
confirmText: this.lang('Unzip'),
|
|
onConfirm: () => {
|
|
this.isLoading = true
|
|
api.unzipItem({
|
|
item: item.path,
|
|
destination: this.$store.state.cwd.location,
|
|
})
|
|
.then(() => {
|
|
this.isLoading = false
|
|
this.loadFiles()
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
this.checked = []
|
|
}
|
|
})
|
|
},
|
|
zip(event, item) {
|
|
this.$dialog.prompt({
|
|
message: this.lang('Name'),
|
|
cancelText: this.lang('Cancel'),
|
|
confirmText: this.lang('Create'),
|
|
inputAttrs: {
|
|
value: this.$store.state.config.default_archive_name,
|
|
placeholder: this.$store.state.config.default_archive_name,
|
|
maxlength: 100,
|
|
required: false,
|
|
},
|
|
onConfirm: (value) => {
|
|
if (! value) {
|
|
return
|
|
}
|
|
this.isLoading = true
|
|
api.zipItems({
|
|
name: value,
|
|
items: item ? [item] : this.getSelected(),
|
|
destination: this.$store.state.cwd.location,
|
|
})
|
|
.then(() => {
|
|
this.isLoading = false
|
|
this.loadFiles()
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
this.checked = []
|
|
}
|
|
})
|
|
},
|
|
rename(event, item) {
|
|
this.$dialog.prompt({
|
|
message: this.lang('New name'),
|
|
cancelText: this.lang('Cancel'),
|
|
confirmText: this.lang('Rename'),
|
|
inputAttrs: {
|
|
value: item ? item.name : this.getSelected()[0].name,
|
|
maxlength: 100,
|
|
required: false,
|
|
},
|
|
onConfirm: (value) => {
|
|
this.isLoading = true
|
|
api.renameItem({
|
|
from: item.name,
|
|
to: value,
|
|
destination: this.$store.state.cwd.location,
|
|
})
|
|
.then(() => {
|
|
this.isLoading = false
|
|
this.loadFiles()
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
this.checked = []
|
|
}
|
|
})
|
|
},
|
|
create(type) {
|
|
this.$dialog.prompt({
|
|
cancelText: this.lang('Cancel'),
|
|
confirmText: this.lang('Create'),
|
|
inputAttrs: {
|
|
placeholder: type == 'dir' ? 'MyFolder' : 'file.txt',
|
|
maxlength: 100,
|
|
required: false,
|
|
},
|
|
onConfirm: (value) => {
|
|
this.isLoading = true
|
|
api.createNew({
|
|
type: type,
|
|
name: value,
|
|
destination: this.$store.state.cwd.location,
|
|
})
|
|
// TODO: cors is triggering this too early?
|
|
.then(() => {
|
|
this.isLoading = false
|
|
this.loadFiles()
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
this.checked = []
|
|
}
|
|
})
|
|
},
|
|
remove(event, item) {
|
|
this.$dialog.confirm({
|
|
message: this.lang('Are you sure you want to do this?'),
|
|
type: 'is-danger',
|
|
cancelText: this.lang('Cancel'),
|
|
confirmText: this.lang('Delete'),
|
|
onConfirm: () => {
|
|
this.isLoading = true
|
|
api.removeItems({
|
|
items: item ? [item] : this.getSelected(),
|
|
})
|
|
.then(() => {
|
|
this.isLoading = false
|
|
this.loadFiles()
|
|
})
|
|
.catch(error => {
|
|
this.isLoading = false
|
|
this.handleError(error)
|
|
})
|
|
this.checked = []
|
|
}
|
|
})
|
|
},
|
|
sortByName(a, b, isAsc) {
|
|
return this.customSort(a, b, isAsc, 'name')
|
|
},
|
|
sortBySize(a, b, isAsc) {
|
|
return this.customSort(a, b, isAsc, 'size')
|
|
},
|
|
sortByTime(a, b, isAsc) {
|
|
return this.customSort(a, b, isAsc, 'time')
|
|
},
|
|
customSort(a, b, isAsc, param) {
|
|
// TODO: firefox is broken
|
|
if (b.type == 'back') return 1
|
|
if (a.type == 'back') return -1
|
|
if (b.type == 'dir' && a.type == 'dir') {
|
|
return (a[param] < b[param]) || isAsc ? -1 : 1
|
|
} else if (b.type == 'dir') {
|
|
return 1
|
|
} else if (a.type == 'dir') {
|
|
return -1
|
|
}
|
|
if (_.isString(a[param])) {
|
|
return (_.lowerCase(a[param]) < _.lowerCase(b[param])) || isAsc ? -1 : 1
|
|
} else {
|
|
return (a[param] < b[param]) || isAsc ? -1 : 1
|
|
}
|
|
},
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
#loading {
|
|
width: 100%;
|
|
height: 100%;
|
|
position: fixed;
|
|
z-index: 1000;
|
|
top: 0;
|
|
left: 0;
|
|
user-drag: none;
|
|
user-select: none;
|
|
-moz-user-select: none;
|
|
-webkit-user-drag: none;
|
|
-webkit-user-select: none;
|
|
-ms-user-select: none;
|
|
}
|
|
#dropzone {
|
|
padding: 0;
|
|
}
|
|
#browser {
|
|
margin: 50px auto 100px auto;
|
|
}
|
|
.breadcrumb a {
|
|
font-weight: bold;
|
|
}
|
|
#multi-actions {
|
|
min-height: 55px;
|
|
}
|
|
#multi-actions a {
|
|
margin: 0 15px 15px 0;
|
|
}
|
|
.file-row a {
|
|
color: #373737;
|
|
}
|
|
.file-row.type-dir a.name {
|
|
font-weight: bold
|
|
}
|
|
#single-actions {
|
|
padding: 6px 12px;
|
|
}
|
|
.drop-info {
|
|
margin: 20% auto;
|
|
}
|
|
.search-btn {
|
|
margin-right: 10px;
|
|
}
|
|
</style>
|