You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
136 lines
5.0 KiB
136 lines
5.0 KiB
<template> |
|
<div class="w-full"> |
|
<el-upload :action="fileUploadUrl" |
|
:headers="{ ...getAuthHeaders(), ...getSiteHeaders() }" |
|
:accept="accept" |
|
:before-upload="beforeUpload" |
|
:on-success="(res) => fileList.push({ name: res.name, url: res.url, length: res.size })" |
|
:on-progress="(event, file) => (progressFile = file)" |
|
:show-file-list="false" |
|
multiple> |
|
<!-- |
|
action="https://jsonplaceholder.typicode.com/posts/" |
|
--> |
|
<el-button type="primary">{{ $t('clickToUpload') }}</el-button> |
|
</el-upload> |
|
<el-progress v-if="progressFile.status === 'uploading'" |
|
:percentage="parseInt(progressFile.percentage, 10)"></el-progress> |
|
<transition-group tag="ul" |
|
:class="['el-upload-list', 'el-upload-list--text', { 'is-disabled': disabled }]" |
|
name="el-list"> |
|
<li v-for="file in fileList" |
|
:key="file.url" |
|
class="el-upload-list__item is-success"> |
|
<a class="el-upload-list__item-name" |
|
@click="handlePreview(file)"> |
|
<el-icon class="el-icon--document"> |
|
<Document /> |
|
</el-icon>{{ file.name }} |
|
</a> |
|
<label class="el-upload-list__item-status-label"> |
|
<el-icon class="el-icon--upload-success el-icon--circle-check"> |
|
<CircleCheck /> |
|
</el-icon> |
|
</label> |
|
<el-icon v-if="!disabled" |
|
class="el-icon--close" |
|
@click="fileList.splice(fileList.indexOf(file), 1)"> |
|
<Close /> |
|
</el-icon> |
|
</li> |
|
</transition-group> |
|
<el-dialog :title="$t('article.fileList.attribute')" |
|
v-model="previewVisible" |
|
top="5vh" |
|
:width="768" |
|
append-to-body> |
|
<el-form ref="form" |
|
:model="previewFile" |
|
label-width="150px"> |
|
<el-form-item prop="name" |
|
:label="$t('name')" |
|
:rules="{ required: true, message: () => $t('v.required') }"> |
|
<el-input v-model="previewFile.name" |
|
maxlength="100"></el-input> |
|
</el-form-item> |
|
<el-form-item prop="length" |
|
:label="$t('size')" |
|
:rules="{ required: true, message: () => $t('v.required') }"> |
|
<el-input v-model="previewFile.length" |
|
maxlength="19"> |
|
<template #append>Byte</template> |
|
</el-input> |
|
</el-form-item> |
|
<el-form-item prop="url" |
|
label="URL" |
|
:rules="{ required: true, message: () => $t('v.required') }"> |
|
<el-input v-model="previewFile.url" |
|
maxlength="255"></el-input> |
|
</el-form-item> |
|
<el-button @click.prevent="handleSubmit()" |
|
type="primary" |
|
native-type="submit">{{ $t('submit') }}</el-button> |
|
</el-form> |
|
</el-dialog> |
|
</div> |
|
</template> |
|
|
|
<script setup lang="ts"> |
|
import { onMounted, ref, toRefs, computed } from 'vue'; |
|
import { ElMessage } from 'element-plus'; |
|
import { Close, Document, CircleCheck } from '@element-plus/icons-vue'; |
|
import { useI18n } from 'vue-i18n'; |
|
import { getAuthHeaders } from '@/utils/auth'; |
|
import { getSiteHeaders } from '@/utils/common'; |
|
import { fileUploadUrl, queryGlobalSettings } from '@/api/config'; |
|
|
|
const props = defineProps({ |
|
modelValue: { type: Array, default: () => [] }, |
|
fileAccept: { type: String }, |
|
fileMaxSize: { type: Number }, |
|
disabled: { type: Boolean, default: false }, |
|
}); |
|
const emit = defineEmits({ 'update:modelValue': null }); |
|
|
|
const { fileAccept, fileMaxSize } = toRefs(props); |
|
const { t } = useI18n(); |
|
const { modelValue } = toRefs(props); |
|
const progressFile = ref<any>({}); |
|
const fileList = computed({ |
|
get: (): any[] => modelValue.value, |
|
set: (val) => emit('update:modelValue', val), |
|
}); |
|
const previewVisible = ref<boolean>(false); |
|
const previewFile = ref<any>({}); |
|
const form = ref<any>(); |
|
|
|
const handlePreview = (file: any) => { |
|
previewFile.value = file; |
|
previewVisible.value = true; |
|
}; |
|
const handleSubmit = () => { |
|
form.value.validate(async (valid: boolean) => { |
|
if (!valid) return; |
|
previewVisible.value = false; |
|
}); |
|
}; |
|
const global = ref<any>(); |
|
const fetchGlobalSettings = async () => { |
|
global.value = await queryGlobalSettings(); |
|
}; |
|
onMounted(() => { |
|
fetchGlobalSettings(); |
|
}); |
|
const defaultAccept = '.zip,.7z,.gz,.bz2,.iso,.rar,.pdf,.doc,.docx,.xls,.xlsx,.ppt,.pptx,.jpg,.jpeg,.png,.gif,.mp4,.m3u8,.mp3,.ogg'; |
|
const accept = computed(() => fileAccept?.value ?? global?.value?.upload?.fileInputAccept ?? defaultAccept); |
|
const maxSize = computed(() => fileMaxSize?.value ?? global?.value?.upload?.fileLimitByte ?? 0); |
|
const beforeUpload = (file: any) => { |
|
if (maxSize.value > 0 && file.size > maxSize.value) { |
|
ElMessage.error(t('error.fileMaxSize', { size: `${maxSize.value / 1024 / 1024}MB` })); |
|
return false; |
|
} |
|
return true; |
|
}; |
|
</script> |
|
|
|
<style lang="scss" scoped></style>
|
|
|