diff options
Diffstat (limited to 'apps/files/src/components/DragAndDropNotice.vue')
-rw-r--r-- | apps/files/src/components/DragAndDropNotice.vue | 155 |
1 files changed, 155 insertions, 0 deletions
diff --git a/apps/files/src/components/DragAndDropNotice.vue b/apps/files/src/components/DragAndDropNotice.vue new file mode 100644 index 00000000000..d5f93dac256 --- /dev/null +++ b/apps/files/src/components/DragAndDropNotice.vue @@ -0,0 +1,155 @@ +<!-- + - @copyright Copyright (c) 2023 John Molakvoæ <skjnldsv@protonmail.com> + - + - @author John Molakvoæ <skjnldsv@protonmail.com> + - + - @license GNU AGPL version 3 or any later version + - + - This program is free software: you can redistribute it and/or modify + - it under the terms of the GNU Affero General Public License as + - published by the Free Software Foundation, either version 3 of the + - License, or (at your option) any later version. + - + - This program is distributed in the hope that it will be useful, + - but WITHOUT ANY WARRANTY; without even the implied warranty of + - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + - GNU Affero General Public License for more details. + - + - You should have received a copy of the GNU Affero General Public License + - along with this program. If not, see <http://www.gnu.org/licenses/>. + - + --> +<template> + <div class="files-list__drag-drop-notice" + :class="{ 'files-list__drag-drop-notice--dragover': dragover }" + @drop="onDrop"> + <div class="files-list__drag-drop-notice-wrapper"> + <TrayArrowDownIcon :size="48" /> + <h3 class="files-list-drag-drop-notice__title"> + {{ t('files', 'Drag and drop files here to upload') }} + </h3> + </div> + </div> +</template> + +<script lang="ts"> +import type { Upload } from '@nextcloud/upload' +import { join } from 'path' +import { showSuccess } from '@nextcloud/dialogs' +import { translate as t } from '@nextcloud/l10n' +import { getUploader } from '@nextcloud/upload' +import Vue from 'vue' + +import TrayArrowDownIcon from 'vue-material-design-icons/TrayArrowDown.vue' + +import logger from '../logger.js' + +export default Vue.extend({ + name: 'DragAndDropNotice', + + components: { + TrayArrowDownIcon, + }, + + props: { + currentFolder: { + type: Object, + required: true, + }, + dragover: { + type: Boolean, + default: false, + }, + }, + + methods: { + onDrop(event: DragEvent) { + this.$emit('update:dragover', false) + + if (this.$el.querySelector('tbody')?.contains(event.target as Node)) { + return + } + + event.preventDefault() + event.stopPropagation() + + if (event.dataTransfer && event.dataTransfer.files?.length > 0) { + const uploader = getUploader() + uploader.destination = this.currentFolder + + // Start upload + logger.debug(`Uploading files to ${this.currentFolder.path}`) + const promises = [...event.dataTransfer.files].map((file: File) => { + return uploader.upload(file.name, file) as Promise<Upload> + }) + + // Process finished uploads + Promise.all(promises).then((uploads) => { + logger.debug('Upload terminated', { uploads }) + showSuccess(t('files', 'Upload successful')) + + // Scroll to last upload if terminated + const lastUpload = uploads[uploads.length - 1] + if (lastUpload?.response?.headers?.['oc-fileid']) { + this.$router.push(Object.assign({}, this.$route, { + params: { + // Remove instanceid from header response + fileid: parseInt(lastUpload.response?.headers?.['oc-fileid']), + }, + })) + } + }) + } + }, + t, + }, +}) +</script> + +<style lang="scss" scoped> +.files-list__drag-drop-notice { + position: absolute; + z-index: 9999; + top: 0; + right: 0; + left: 0; + display: none; + align-items: center; + justify-content: center; + width: 100%; + // Breadcrumbs height + row thead height + min-height: calc(58px + 55px); + margin: 0; + user-select: none; + color: var(--color-text-maxcontrast); + background-color: var(--color-main-background); + + &--dragover { + display: flex; + border-color: black; + } + + h3 { + margin-left: 16px; + color: inherit; + } + + &-wrapper { + display: flex; + align-items: center; + justify-content: center; + height: 15vh; + max-height: 70%; + padding: 0 5vw; + border: 2px var(--color-border-dark) dashed; + border-radius: var(--border-radius-large); + } + + &__close { + position: absolute !important; + top: 10px; + right: 10px; + } +} + +</style> |