edu/admin/src/components/material-select/index.vue

261 lines
7.5 KiB
Vue
Raw Normal View History

2022-04-08 06:46:31 +00:00
2022-04-08 02:42:44 +00:00
<template>
<div class="material-select">
<popup
ref="popupRef"
width="950px"
custom-class="body-padding"
:title="`选择${tipsText}`"
@confirm="handleConfirm"
>
2022-04-08 06:46:31 +00:00
<template v-if="!hiddenUpload" #trigger>
2022-04-08 02:42:44 +00:00
<div class="material-select__trigger clearfix" @click.stop>
2022-04-08 06:46:31 +00:00
<draggable
class="draggable"
v-model="fileList"
animation="300"
item-key="id"
>
<template v-slot:item="{ element, index }">
2022-04-08 02:42:44 +00:00
<div
class="material-preview"
:class="{
'is-disabled': disabled,
2022-04-08 06:46:31 +00:00
'is-one': limit == 1,
2022-04-08 02:42:44 +00:00
}"
@click="showPopup(index)"
>
2022-04-08 06:46:31 +00:00
<file-item :uri="element" :file-size="size" @close="deleteImg(index)" />
2022-04-08 02:42:44 +00:00
</div>
</template>
</draggable>
<div
class="material-upload"
2022-04-08 06:46:31 +00:00
@click="showPopup(-1)"
v-show="showUpload"
2022-04-08 02:42:44 +00:00
:class="{
'is-disabled': disabled,
2022-04-08 06:46:31 +00:00
'is-one': limit == 1,
2022-04-08 02:42:44 +00:00
}"
>
<slot name="upload">
<div
class="upload-btn flex flex-col flex-center"
:style="{
width: size,
2022-04-08 06:46:31 +00:00
height: size,
2022-04-08 02:42:44 +00:00
}"
>
<el-icon :size="25"><plus /></el-icon>
<span>添加</span>
</div>
</slot>
</div>
</div>
</template>
<div class="material-wrap">
<material
ref="materialRefs"
:file-size="fileSize"
:limit="meterialLimit"
@change="selectChange"
/>
</div>
</popup>
</div>
</template>
2022-04-08 06:46:31 +00:00
2022-04-08 02:42:44 +00:00
<script lang="ts">
import {
provide,
reactive,
defineComponent,
computed,
ref,
Ref,
toRef,
toRefs,
watch,
2022-04-08 06:46:31 +00:00
nextTick,
2022-04-08 02:42:44 +00:00
} from 'vue'
import Draggable from 'vuedraggable'
import Popup from '@/components/popup/index.vue'
import FileItem from './file-item.vue'
import Material from './material.vue'
export default defineComponent({
components: {
Popup,
Draggable,
FileItem,
2022-04-08 06:46:31 +00:00
Material,
2022-04-08 02:42:44 +00:00
},
props: {
modelValue: {
type: [String, Array],
2022-04-08 06:46:31 +00:00
default: () => [],
2022-04-08 02:42:44 +00:00
},
// 文件类型
type: {
type: String,
2022-04-08 06:46:31 +00:00
default: 'image',
2022-04-08 02:42:44 +00:00
},
// 选择器尺寸
size: {
type: String,
2022-04-08 06:46:31 +00:00
default: '100px',
2022-04-08 02:42:44 +00:00
},
// 文件尺寸
fileSize: {
type: String,
2022-04-08 06:46:31 +00:00
default: '100px',
2022-04-08 02:42:44 +00:00
},
// 选择数量限制
limit: {
2022-04-08 06:46:31 +00:00
// type: Number,
default: 1,
2022-04-08 02:42:44 +00:00
},
// 禁用选择
disabled: {
type: Boolean,
2022-04-08 06:46:31 +00:00
default: false,
},
// 隐藏上传框*(目前在富文本中使用到)
hiddenUpload: {
type: Boolean,
default: false,
2022-04-08 02:42:44 +00:00
}
},
emits: ['change', 'update:modelValue'],
setup(props, { emit }) {
const popupRef: Ref<typeof Popup | null> = ref(null)
const materialRefs: Ref<typeof Material | null> = ref(null)
const fileList: Ref<any[]> = ref([])
const select: Ref<any[]> = ref([])
const isAdd = ref(true)
const currentIndex = ref(-1)
const { disabled, limit, modelValue } = toRefs(props)
const tipsText = computed(() => {
switch (props.type) {
case 'image':
return '图片'
case 'video':
return '视频'
}
})
const typeValue = computed(() => {
switch (props.type) {
case 'image':
return 10
case 'video':
return 20
case 'file':
return 30
}
})
const showUpload = computed(() => {
return props.limit - fileList.value.length > 0
})
2022-04-08 06:46:31 +00:00
const meterialLimit: any = computed(() => {
2022-04-08 02:42:44 +00:00
if (!isAdd.value) {
return 1
}
2022-04-08 06:46:31 +00:00
if (!limit.value) return null
2022-04-08 02:42:44 +00:00
return limit.value - fileList.value.length
})
const handleConfirm = () => {
2022-04-08 06:46:31 +00:00
const selectUri = select.value.map((item) => item.uri)
2022-04-08 02:42:44 +00:00
if (!isAdd.value) {
fileList.value.splice(currentIndex.value, 1, selectUri.shift())
} else {
2022-04-08 06:46:31 +00:00
fileList.value = [...fileList.value,...selectUri]
2022-04-08 02:42:44 +00:00
}
handleChange()
}
const showPopup = (index: number) => {
2022-04-08 06:46:31 +00:00
if (disabled.value) return
2022-04-08 02:42:44 +00:00
if (index >= 0) {
isAdd.value = false
currentIndex.value = index
} else {
isAdd.value = true
}
popupRef.value?.open()
}
const selectChange = (val: any[]) => {
select.value = val
}
const handleChange = () => {
2022-04-08 06:46:31 +00:00
const valueImg =
limit.value != 1 ? fileList.value : fileList.value[0] || ''
2022-04-08 02:42:44 +00:00
emit('update:modelValue', valueImg)
emit('change', valueImg)
nextTick(() => {
materialRefs.value?.clearSelect()
})
}
const deleteImg = (index: number) => {
fileList.value.splice(index, 1)
handleChange()
}
watch(modelValue, (val: any[] | string) => {
fileList.value = Array.isArray(val) ? val : val == '' ? [] : [val]
})
2022-04-08 06:46:31 +00:00
provide('type', props)
2022-04-08 02:42:44 +00:00
provide('fileSize', props.fileSize)
provide('limit', props.limit)
provide('typeValue', typeValue)
2022-04-08 06:46:31 +00:00
provide('hiddenUpload', props.hiddenUpload)
2022-04-08 02:42:44 +00:00
return {
popupRef,
materialRefs,
fileList,
tipsText,
handleConfirm,
meterialLimit,
showUpload,
showPopup,
selectChange,
deleteImg
}
2022-04-08 06:46:31 +00:00
},
2022-04-08 02:42:44 +00:00
})
</script>
<style scoped lang="scss">
.material-select {
.material-upload,
.material-preview {
border-radius: 4px;
cursor: pointer;
color: $color-text-secondary;
margin-right: 8px;
margin-bottom: 8px;
box-sizing: border-box;
float: left;
&.is-disabled {
cursor: not-allowed;
}
&.is-one {
margin-bottom: 0;
}
}
.material-upload {
.upload-btn {
box-sizing: border-box;
border-radius: 4px;
border: 1px dashed #d7d7d7;
}
}
}
.material-wrap {
height: 540px;
border-top: 1px solid $border-color-base;
border-bottom: 1px solid $border-color-base;
}
</style>