273 lines
8.7 KiB
JavaScript
273 lines
8.7 KiB
JavaScript
import { ref, reactive, defineComponent, onBeforeUnmount, nextTick, mergeProps as _mergeProps, createVNode as _createVNode, vShow as _vShow, withDirectives as _withDirectives } from "vue";
|
|
import { pick, extend, toArray, isPromise, truthProp, getSizeStyle, makeArrayProp, makeStringProp, makeNumericProp } from "../utils/index.mjs";
|
|
import { bem, name, isOversize, filterFiles, isImageFile, readFileContent } from "./utils.mjs";
|
|
import { useCustomFieldValue } from "@vant/use";
|
|
import { useExpose } from "../composables/use-expose.mjs";
|
|
import { Icon } from "../icon/index.mjs";
|
|
import { showImagePreview } from "../image-preview/index.mjs";
|
|
import UploaderPreviewItem from "./UploaderPreviewItem.mjs";
|
|
const uploaderProps = {
|
|
name: makeNumericProp(""),
|
|
accept: makeStringProp("image/*"),
|
|
capture: String,
|
|
multiple: Boolean,
|
|
disabled: Boolean,
|
|
readonly: Boolean,
|
|
lazyLoad: Boolean,
|
|
maxCount: makeNumericProp(Infinity),
|
|
imageFit: makeStringProp("cover"),
|
|
resultType: makeStringProp("dataUrl"),
|
|
uploadIcon: makeStringProp("photograph"),
|
|
uploadText: String,
|
|
deletable: truthProp,
|
|
reupload: Boolean,
|
|
afterRead: Function,
|
|
showUpload: truthProp,
|
|
modelValue: makeArrayProp(),
|
|
beforeRead: Function,
|
|
beforeDelete: Function,
|
|
previewSize: [Number, String, Array],
|
|
previewImage: truthProp,
|
|
previewOptions: Object,
|
|
previewFullImage: truthProp,
|
|
maxSize: {
|
|
type: [Number, String, Function],
|
|
default: Infinity
|
|
}
|
|
};
|
|
var stdin_default = defineComponent({
|
|
name,
|
|
props: uploaderProps,
|
|
emits: ["delete", "oversize", "clickUpload", "closePreview", "clickPreview", "clickReupload", "update:modelValue"],
|
|
setup(props, {
|
|
emit,
|
|
slots
|
|
}) {
|
|
const inputRef = ref();
|
|
const urls = [];
|
|
const reuploadIndex = ref(-1);
|
|
const isReuploading = ref(false);
|
|
const getDetail = (index = props.modelValue.length) => ({
|
|
name: props.name,
|
|
index
|
|
});
|
|
const resetInput = () => {
|
|
if (inputRef.value) {
|
|
inputRef.value.value = "";
|
|
}
|
|
};
|
|
const onAfterRead = (items) => {
|
|
resetInput();
|
|
if (isOversize(items, props.maxSize)) {
|
|
if (Array.isArray(items)) {
|
|
const result = filterFiles(items, props.maxSize);
|
|
items = result.valid;
|
|
emit("oversize", result.invalid, getDetail());
|
|
if (!items.length) {
|
|
return;
|
|
}
|
|
} else {
|
|
emit("oversize", items, getDetail());
|
|
return;
|
|
}
|
|
}
|
|
items = reactive(items);
|
|
if (reuploadIndex.value > -1) {
|
|
const arr = [...props.modelValue];
|
|
arr.splice(reuploadIndex.value, 1, items);
|
|
emit("update:modelValue", arr);
|
|
reuploadIndex.value = -1;
|
|
} else {
|
|
emit("update:modelValue", [...props.modelValue, ...toArray(items)]);
|
|
}
|
|
if (props.afterRead) {
|
|
props.afterRead(items, getDetail());
|
|
}
|
|
};
|
|
const readFile = (files) => {
|
|
const {
|
|
maxCount,
|
|
modelValue,
|
|
resultType
|
|
} = props;
|
|
if (Array.isArray(files)) {
|
|
const remainCount = +maxCount - modelValue.length;
|
|
if (files.length > remainCount) {
|
|
files = files.slice(0, remainCount);
|
|
}
|
|
Promise.all(files.map((file) => readFileContent(file, resultType))).then((contents) => {
|
|
const fileList = files.map((file, index) => {
|
|
const result = {
|
|
file,
|
|
status: "",
|
|
message: "",
|
|
objectUrl: URL.createObjectURL(file)
|
|
};
|
|
if (contents[index]) {
|
|
result.content = contents[index];
|
|
}
|
|
return result;
|
|
});
|
|
onAfterRead(fileList);
|
|
});
|
|
} else {
|
|
readFileContent(files, resultType).then((content) => {
|
|
const result = {
|
|
file: files,
|
|
status: "",
|
|
message: "",
|
|
objectUrl: URL.createObjectURL(files)
|
|
};
|
|
if (content) {
|
|
result.content = content;
|
|
}
|
|
onAfterRead(result);
|
|
});
|
|
}
|
|
};
|
|
const onChange = (event) => {
|
|
const {
|
|
files
|
|
} = event.target;
|
|
if (props.disabled || !files || !files.length) {
|
|
return;
|
|
}
|
|
const file = files.length === 1 ? files[0] : [].slice.call(files);
|
|
if (props.beforeRead) {
|
|
const response = props.beforeRead(file, getDetail());
|
|
if (!response) {
|
|
resetInput();
|
|
return;
|
|
}
|
|
if (isPromise(response)) {
|
|
response.then((data) => {
|
|
if (data) {
|
|
readFile(data);
|
|
} else {
|
|
readFile(file);
|
|
}
|
|
}).catch(resetInput);
|
|
return;
|
|
}
|
|
}
|
|
readFile(file);
|
|
};
|
|
let imagePreview;
|
|
const onClosePreview = () => emit("closePreview");
|
|
const previewImage = (item) => {
|
|
if (props.previewFullImage) {
|
|
const imageFiles = props.modelValue.filter(isImageFile);
|
|
const images = imageFiles.map((item2) => {
|
|
if (item2.objectUrl && !item2.url && item2.status !== "failed") {
|
|
item2.url = item2.objectUrl;
|
|
urls.push(item2.url);
|
|
}
|
|
return item2.url;
|
|
}).filter(Boolean);
|
|
imagePreview = showImagePreview(extend({
|
|
images,
|
|
startPosition: imageFiles.indexOf(item),
|
|
onClose: onClosePreview
|
|
}, props.previewOptions));
|
|
}
|
|
};
|
|
const closeImagePreview = () => {
|
|
if (imagePreview) {
|
|
imagePreview.close();
|
|
}
|
|
};
|
|
const deleteFile = (item, index) => {
|
|
const fileList = props.modelValue.slice(0);
|
|
fileList.splice(index, 1);
|
|
emit("update:modelValue", fileList);
|
|
emit("delete", item, getDetail(index));
|
|
};
|
|
const reuploadFile = (index) => {
|
|
isReuploading.value = true;
|
|
reuploadIndex.value = index;
|
|
nextTick(() => chooseFile());
|
|
};
|
|
const onInputClick = () => {
|
|
if (!isReuploading.value) {
|
|
reuploadIndex.value = -1;
|
|
}
|
|
isReuploading.value = false;
|
|
};
|
|
const renderPreviewItem = (item, index) => {
|
|
const needPickData = ["imageFit", "deletable", "reupload", "previewSize", "beforeDelete"];
|
|
const previewData = extend(pick(props, needPickData), pick(item, needPickData, true));
|
|
return _createVNode(UploaderPreviewItem, _mergeProps({
|
|
"item": item,
|
|
"index": index,
|
|
"onClick": () => emit(props.reupload ? "clickReupload" : "clickPreview", item, getDetail(index)),
|
|
"onDelete": () => deleteFile(item, index),
|
|
"onPreview": () => previewImage(item),
|
|
"onReupload": () => reuploadFile(index)
|
|
}, pick(props, ["name", "lazyLoad"]), previewData), pick(slots, ["preview-cover", "preview-delete"]));
|
|
};
|
|
const renderPreviewList = () => {
|
|
if (props.previewImage) {
|
|
return props.modelValue.map(renderPreviewItem);
|
|
}
|
|
};
|
|
const onClickUpload = (event) => emit("clickUpload", event);
|
|
const renderUpload = () => {
|
|
const lessThanMax = props.modelValue.length < +props.maxCount;
|
|
const Input = props.readonly ? null : _createVNode("input", {
|
|
"ref": inputRef,
|
|
"type": "file",
|
|
"class": bem("input"),
|
|
"accept": props.accept,
|
|
"capture": props.capture,
|
|
"multiple": props.multiple && reuploadIndex.value === -1,
|
|
"disabled": props.disabled,
|
|
"onChange": onChange,
|
|
"onClick": onInputClick
|
|
}, null);
|
|
if (slots.default) {
|
|
return _withDirectives(_createVNode("div", {
|
|
"class": bem("input-wrapper"),
|
|
"onClick": onClickUpload
|
|
}, [slots.default(), Input]), [[_vShow, lessThanMax]]);
|
|
}
|
|
return _withDirectives(_createVNode("div", {
|
|
"class": bem("upload", {
|
|
readonly: props.readonly
|
|
}),
|
|
"style": getSizeStyle(props.previewSize),
|
|
"onClick": onClickUpload
|
|
}, [_createVNode(Icon, {
|
|
"name": props.uploadIcon,
|
|
"class": bem("upload-icon")
|
|
}, null), props.uploadText && _createVNode("span", {
|
|
"class": bem("upload-text")
|
|
}, [props.uploadText]), Input]), [[_vShow, props.showUpload && lessThanMax]]);
|
|
};
|
|
const chooseFile = () => {
|
|
if (inputRef.value && !props.disabled) {
|
|
inputRef.value.click();
|
|
}
|
|
};
|
|
onBeforeUnmount(() => {
|
|
urls.forEach((url) => URL.revokeObjectURL(url));
|
|
});
|
|
useExpose({
|
|
chooseFile,
|
|
reuploadFile,
|
|
closeImagePreview
|
|
});
|
|
useCustomFieldValue(() => props.modelValue);
|
|
return () => _createVNode("div", {
|
|
"class": bem()
|
|
}, [_createVNode("div", {
|
|
"class": bem("wrapper", {
|
|
disabled: props.disabled
|
|
})
|
|
}, [renderPreviewList(), renderUpload()])]);
|
|
}
|
|
});
|
|
export {
|
|
stdin_default as default,
|
|
uploaderProps
|
|
};
|