test-nut-ui
Version:
<p align="center"> <img alt="logo" src="https://img11.360buyimg.com/imagetools/jfs/t1/211965/25/7152/22022/61b16785E433119bb/aa41d7a9f7e823f3.png" width="150" style="margin-bottom: 10px;"> </p>
629 lines (628 loc) • 29.3 kB
JavaScript
import _toConsumableArray from "@babel/runtime/helpers/toConsumableArray";
import _objectWithoutProperties from "@babel/runtime/helpers/objectWithoutProperties";
import _slicedToArray from "@babel/runtime/helpers/slicedToArray";
import _createClass from "@babel/runtime/helpers/createClass";
import _classCallCheck from "@babel/runtime/helpers/classCallCheck";
import _defineProperty from "@babel/runtime/helpers/defineProperty";
var _excluded = ["children", "uploadIcon", "uploadLabel", "uploadTip", "loadingIcon", "loadingTip", "deleteIcon", "size", "style", "radius", "sourceType", "col", "dragable", "name", "accept", "defaultValue", "value", "previewType", "fit", "disabled", "disableReason", "multiple", "url", "previewUrl", "headers", "timeout", "method", "xhrState", "withCredentials", "data", "preview", "deletable", "maxCount", "capture", "maxFileSize", "className", "autoUpload", "clearInput", "onStart", "onDelete", "onChange", "onFileItemClick", "onProgress", "onSuccess", "onUpdate", "onFailure", "onOversize", "beforeUpload", "beforeXhrUpload", "beforeDelete"];
function ownKeys(object, enumerableOnly) {
var keys = Object.keys(object);
if (Object.getOwnPropertySymbols) {
var symbols = Object.getOwnPropertySymbols(object);
enumerableOnly && (symbols = symbols.filter(function(sym) {
return Object.getOwnPropertyDescriptor(object, sym).enumerable;
})), keys.push.apply(keys, symbols);
}
return keys;
}
function _objectSpread(target) {
for (var i = 1; i < arguments.length; i++) {
var source = null != arguments[i] ? arguments[i] : {};
i % 2 ? ownKeys(Object(source), true).forEach(function(key) {
_defineProperty(target, key, source[key]);
}) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function(key) {
Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key));
});
}
return target;
}
import React__default, { useState, useEffect, useImperativeHandle } from "react";
import classNames from "classnames";
import { Link, Del } from "@nutui/icons-react";
import Progress from "./Progress.js";
import { Y as Yr } from "./index.js";
import { useConfig } from "./ConfigProvider.js";
import { b as isPromise } from "./index2.js";
import Toast from "./Toast.js";
import { C as ComponentDefaults } from "./typings.js";
var PhotographIcon = "";
var UploaderIcon = "";
var DeleteIcon = "";
var UploadOptions = /* @__PURE__ */ _createClass(function UploadOptions2() {
_classCallCheck(this, UploadOptions2);
_defineProperty(this, "url", "");
_defineProperty(this, "name", "file");
_defineProperty(this, "fileType", "image");
_defineProperty(this, "formData", void 0);
_defineProperty(this, "sourceFile", void 0);
_defineProperty(this, "method", "post");
_defineProperty(this, "xhrState", 200);
_defineProperty(this, "timeout", 30 * 1e3);
_defineProperty(this, "headers", {});
_defineProperty(this, "withCredentials", false);
_defineProperty(this, "onStart", void 0);
_defineProperty(this, "taroFilePath", void 0);
_defineProperty(this, "onProgress", void 0);
_defineProperty(this, "onSuccess", void 0);
_defineProperty(this, "onFailure", void 0);
_defineProperty(this, "beforeXhrUpload", void 0);
});
var Upload = /* @__PURE__ */ function() {
function Upload2(options) {
_classCallCheck(this, Upload2);
_defineProperty(this, "options", void 0);
this.options = options;
}
_createClass(Upload2, [{
key: "upload",
value: function upload() {
var options = this.options;
var xhr = new XMLHttpRequest();
xhr.timeout = options.timeout;
if (xhr.upload) {
var _options$onStart;
xhr.upload.addEventListener("progress", function(e) {
var _options$onProgress;
(_options$onProgress = options.onProgress) === null || _options$onProgress === void 0 ? void 0 : _options$onProgress.call(options, e, options);
}, false);
xhr.onreadystatechange = function() {
if (xhr.readyState === 4) {
if (xhr.status === options.xhrState) {
var _options$onSuccess;
(_options$onSuccess = options.onSuccess) === null || _options$onSuccess === void 0 ? void 0 : _options$onSuccess.call(options, xhr.responseText, options);
} else {
var _options$onFailure;
(_options$onFailure = options.onFailure) === null || _options$onFailure === void 0 ? void 0 : _options$onFailure.call(options, xhr.responseText, options);
}
}
};
xhr.withCredentials = options.withCredentials;
xhr.open(options.method, options.url, true);
for (var _i = 0, _Object$entries = Object.entries(options.headers); _i < _Object$entries.length; _i++) {
var _Object$entries$_i = _slicedToArray(_Object$entries[_i], 2), key = _Object$entries$_i[0], value = _Object$entries$_i[1];
xhr.setRequestHeader(key, value);
}
(_options$onStart = options.onStart) === null || _options$onStart === void 0 ? void 0 : _options$onStart.call(options, options);
if (options.beforeXhrUpload) {
options.beforeXhrUpload(xhr, options);
} else {
xhr.send(options.formData);
}
} else {
console.warn("浏览器不支持 XMLHttpRequest");
}
}
}]);
return Upload2;
}();
var funcInterceptor = function funcInterceptor2(interceptor, _ref) {
var _ref$args = _ref.args, args = _ref$args === void 0 ? [] : _ref$args, done = _ref.done, canceled = _ref.canceled;
if (interceptor) {
var returnVal = interceptor.apply(null, args);
if (isPromise(returnVal)) {
returnVal.then(function(value) {
if (value) {
done(value);
} else if (canceled) {
canceled();
}
})["catch"](function() {
});
} else if (returnVal) {
done();
} else if (canceled) {
canceled();
}
} else {
done();
}
};
var defaultProps = _objectSpread(_objectSpread({}, ComponentDefaults), {}, {
url: "",
maxCount: 1,
size: "large",
radius: 2,
col: 3,
previewType: "picture",
fit: "cover",
name: "file",
dragable: false,
sourceType: ["album", "camera"],
accept: "*",
disabled: false,
autoUpload: true,
multiple: false,
maxFileSize: Number.MAX_VALUE,
data: {},
headers: {},
method: "post",
xhrState: 200,
timeout: 1e3 * 30,
withCredentials: false,
clearInput: true,
preview: true,
deletable: true,
capture: false,
beforeDelete: function beforeDelete(file, files) {
return true;
}
});
function generateRandomId() {
var characters = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
var uniqueId = "";
var idLength = 10;
for (var i = 0; i < idLength; i++) {
var randomIndex = Math.floor(Math.random() * characters.length);
uniqueId += characters.charAt(randomIndex);
}
return uniqueId;
}
var FileItem = /* @__PURE__ */ _createClass(function FileItem2() {
_classCallCheck(this, FileItem2);
_defineProperty(this, "status", "ready");
_defineProperty(this, "message", "准备中..");
_defineProperty(this, "uid", generateRandomId());
_defineProperty(this, "name", void 0);
_defineProperty(this, "url", void 0);
_defineProperty(this, "type", void 0);
_defineProperty(this, "path", void 0);
_defineProperty(this, "percentage", 0);
_defineProperty(this, "formData", new FormData());
_defineProperty(this, "loadingIcon", void 0);
});
var InternalUploader = function InternalUploader2(props, ref) {
var _useConfig = useConfig(), locale = _useConfig.locale;
var _defaultProps$props = _objectSpread(_objectSpread({}, defaultProps), props), children = _defaultProps$props.children, uploadIcon = _defaultProps$props.uploadIcon, uploadLabel = _defaultProps$props.uploadLabel, uploadTip = _defaultProps$props.uploadTip, loadingIcon = _defaultProps$props.loadingIcon, loadingTip = _defaultProps$props.loadingTip, deleteIcon = _defaultProps$props.deleteIcon, size = _defaultProps$props.size, style = _defaultProps$props.style, radius = _defaultProps$props.radius, sourceType = _defaultProps$props.sourceType, col = _defaultProps$props.col, dragable = _defaultProps$props.dragable, name = _defaultProps$props.name, accept = _defaultProps$props.accept, defaultValue = _defaultProps$props.defaultValue, value = _defaultProps$props.value, previewType = _defaultProps$props.previewType, fit = _defaultProps$props.fit, disabled = _defaultProps$props.disabled, disableReason = _defaultProps$props.disableReason, multiple = _defaultProps$props.multiple, url = _defaultProps$props.url, previewUrl = _defaultProps$props.previewUrl, headers = _defaultProps$props.headers, timeout = _defaultProps$props.timeout, method = _defaultProps$props.method, xhrState = _defaultProps$props.xhrState, withCredentials = _defaultProps$props.withCredentials, data = _defaultProps$props.data, preview = _defaultProps$props.preview, deletable = _defaultProps$props.deletable, maxCount = _defaultProps$props.maxCount, capture = _defaultProps$props.capture, maxFileSize = _defaultProps$props.maxFileSize, className = _defaultProps$props.className, autoUpload = _defaultProps$props.autoUpload, clearInput = _defaultProps$props.clearInput, onStart = _defaultProps$props.onStart, onDelete = _defaultProps$props.onDelete, onChange = _defaultProps$props.onChange, onFileItemClick = _defaultProps$props.onFileItemClick, onProgress = _defaultProps$props.onProgress, onSuccess = _defaultProps$props.onSuccess, onUpdate = _defaultProps$props.onUpdate, onFailure = _defaultProps$props.onFailure, onOversize = _defaultProps$props.onOversize, beforeUpload = _defaultProps$props.beforeUpload, beforeXhrUpload = _defaultProps$props.beforeXhrUpload, beforeDelete2 = _defaultProps$props.beforeDelete, restProps = _objectWithoutProperties(_defaultProps$props, _excluded);
var _useState = useState(defaultValue || []), _useState2 = _slicedToArray(_useState, 2), fileList = _useState2[0], setFileList = _useState2[1];
var _useState3 = useState([]), _useState4 = _slicedToArray(_useState3, 2), uploadQueue = _useState4[0], setUploadQueue = _useState4[1];
useEffect(function() {
if (value) {
setFileList(value);
}
}, [value]);
var classes = classNames(className, "nut-uploader");
useImperativeHandle(ref, function() {
return {
submit: function submit() {
Promise.all(uploadQueue).then(function(res) {
res.forEach(function(i) {
return i.upload();
});
});
},
clear: function clear() {
clearUploadQueue();
}
};
});
var clearUploadQueue = function clearUploadQueue2() {
var index = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : -1;
if (index > -1) {
uploadQueue.splice(index, 1);
setUploadQueue(uploadQueue);
} else {
setUploadQueue([]);
fileList.splice(0, fileList.length);
setFileList(_toConsumableArray(fileList));
}
};
var clearInputValue = function clearInputValue2(el) {
el.value = "";
};
var executeUpload = function executeUpload2(fileItem, index) {
var uploadOption = new UploadOptions();
uploadOption.url = url;
for (var _i2 = 0, _Object$entries2 = Object.entries(data); _i2 < _Object$entries2.length; _i2++) {
var _Object$entries2$_i = _slicedToArray(_Object$entries2[_i2], 2), key = _Object$entries2$_i[0], value2 = _Object$entries2$_i[1];
fileItem.formData.append(key, value2);
}
uploadOption.formData = fileItem.formData;
uploadOption.timeout = timeout * 1;
uploadOption.method = method;
uploadOption.xhrState = xhrState;
uploadOption.headers = headers;
uploadOption.withCredentials = withCredentials;
uploadOption.beforeXhrUpload = beforeXhrUpload;
try {
uploadOption.sourceFile = fileItem.formData.get(name);
} catch (error) {
}
uploadOption.onStart = function(option) {
clearUploadQueue(index);
setFileList(function(fileList2) {
fileList2.map(function(item) {
if (item.uid === fileItem.uid) {
item.status = "ready";
item.message = "";
}
return item;
});
return _toConsumableArray(fileList2);
});
onStart && onStart(option);
};
uploadOption.onProgress = function(e, option) {
setFileList(function(fileList2) {
fileList2.map(function(item) {
if (item.uid === fileItem.uid) {
item.status = "uploading";
item.message = "";
item.percentage = (e.loaded / e.total * 100).toFixed(0);
onProgress && onProgress({
e,
option,
percentage: item.percentage
});
}
return item;
});
return _toConsumableArray(fileList2);
});
};
uploadOption.onSuccess = function(responseText, option) {
setFileList(function(fileList2) {
onUpdate && onUpdate(fileList2);
fileList2.map(function(item) {
if (item.uid === fileItem.uid) {
item.status = "success";
item.message = locale.uploader.success;
}
return item;
});
return _toConsumableArray(fileList2);
});
onSuccess && onSuccess({
responseText,
option,
fileList
});
};
uploadOption.onFailure = function(responseText, option) {
setFileList(function(fileList2) {
fileList2.map(function(item) {
if (item.uid === fileItem.uid) {
item.status = "error";
item.message = locale.uploader.error;
}
return item;
});
return _toConsumableArray(fileList2);
});
onFailure && onFailure({
responseText,
option,
fileList
});
};
var task = new Upload(uploadOption);
if (props.autoUpload) {
task.upload();
} else {
uploadQueue.push(new Promise(function(resolve, reject) {
resolve(task);
}));
setUploadQueue(uploadQueue);
}
};
var readFile = function readFile2(files) {
files.forEach(function(file, index) {
var formData = new FormData();
formData.append(name, file);
var fileItem = new FileItem();
fileItem.name = file.name;
fileItem.status = "ready";
fileItem.type = file.type;
fileItem.formData = formData;
fileItem.uid = file.lastModified + fileItem.uid;
if (autoUpload) {
fileItem.message = "";
} else {
fileItem.message = locale.uploader.waitingUpload;
}
executeUpload(fileItem, index);
if (preview && file.type.includes("image")) {
var reader = new FileReader();
reader.onload = function(event) {
fileItem.url = event.target.result;
fileList.push(fileItem);
setFileList(_toConsumableArray(fileList));
};
reader.readAsDataURL(file);
} else {
fileList.push(fileItem);
setFileList(_toConsumableArray(fileList));
}
});
};
var filterFiles = function filterFiles2(files) {
var maximum = props.maxCount * 1;
var oversizes = new Array();
var filterFile = files.filter(function(file) {
if (file.size > maxFileSize) {
oversizes.push(file);
return false;
}
return true;
});
if (oversizes.length) {
onOversize && onOversize(files);
}
if (filterFile.length > maximum) {
filterFile.splice(maximum, filterFile.length - maximum);
}
if (fileList.length !== 0) {
var index = maximum - fileList.length;
filterFile.splice(index, filterFile.length - index);
}
return filterFile;
};
var deleted = function deleted2(file, index) {
fileList.splice(index, 1);
onDelete && onDelete(file, fileList);
setFileList(_toConsumableArray(fileList));
};
var onDeleteItem = function onDeleteItem2(file, index) {
clearUploadQueue(index);
funcInterceptor(beforeDelete2, {
args: [file, fileList],
done: function done() {
return deleted(file, index);
}
});
};
var fileChange = function fileChange2(event) {
if (disabled) {
return;
}
var $el = event.target;
var files = $el.files;
if (beforeUpload) {
beforeUpload(new Array().slice.call(files)).then(function(f) {
var _files2 = filterFiles(new Array().slice.call(f));
readFile(_files2);
});
} else {
var _files = filterFiles(new Array().slice.call(files));
readFile(_files);
}
onChange && onChange({
fileList,
event
});
if (clearInput) {
clearInputValue($el);
}
};
var handleItemClick = function handleItemClick2(file, index) {
if (file.status === "error") {
executeUpload(file, index);
}
onFileItemClick && onFileItemClick(file);
};
var handleDragstart = function handleDragstart2(event, index) {
if (!dragable)
return;
event.dataTransfer.setData("text/plain", index);
};
var handleDragOver = function handleDragOver2(event, index) {
event.preventDefault();
};
var handleDrop = function handleDrop2(event, index) {
if (!dragable)
return;
event.preventDefault();
var draggedIndex = event.dataTransfer.getData("text/plain");
var newFileListCopy = _toConsumableArray(fileList);
newFileListCopy[draggedIndex] = newFileListCopy.splice(index, 1, newFileListCopy[draggedIndex])[0];
setFileList(newFileListCopy);
};
var handleClickUploader = function handleClickUploader2() {
disabled && disableReason && Toast.show(disableReason);
};
return React__default.createElement("div", _objectSpread(_objectSpread({
className: classes
}, restProps), {}, {
style: _objectSpread({}, style),
onClick: handleClickUploader
}), uploadTip ? React__default.createElement("div", {
className: "nut-uploader__tip"
}, uploadTip) : React__default.createElement(React__default.Fragment, null), children && React__default.createElement("div", {
className: "nut-uploader__slot"
}, React__default.createElement(React__default.Fragment, null, children, maxCount > fileList.length && React__default.createElement(React__default.Fragment, null, capture ? React__default.createElement("input", {
onClick: function onClick() {
return handleClickUploader();
},
className: "nut-uploader__input",
type: "file",
capture: "user",
name,
accept,
disabled,
multiple,
onChange: fileChange
}) : React__default.createElement("input", {
onClick: function onClick() {
return handleClickUploader();
},
className: "nut-uploader__input",
type: "file",
name,
accept,
disabled,
multiple,
onChange: fileChange
})))), fileList.length !== 0 && fileList.map(function(item, index) {
return React__default.createElement("div", {
className: "nut-uploader__preview ".concat(previewType),
style: {
borderRadius: typeof radius === "number" ? radius + "px" : radius
},
key: item.uid
}, previewType === "picture" && !children && React__default.createElement("div", {
className: "nut-uploader__preview-img nut-uploader__preview-img--".concat(size)
}, deletable && React__default.createElement("div", {
className: "close",
onClick: function onClick() {
return onDeleteItem(item, index);
}
}, deleteIcon || React__default.createElement("img", {
src: DeleteIcon
})), item.status === "ready" ? React__default.createElement("div", {
className: "nut-uploader__preview__progress",
style: {
borderRadius: typeof radius === "number" ? radius + "px" : radius
}
}, React__default.createElement("div", {
className: "nut-uploader__preview__progress__msg"
}, item.message)) : item.status !== "success" && React__default.createElement("div", {
onClick: function onClick() {
return handleItemClick(item, index);
},
className: "nut-uploader__preview__progress",
style: {
borderRadius: typeof radius === "number" ? radius + "px" : radius
}
}, item.failIcon !== " " && loadingIcon !== " " && (item.status === "error" ? item.failIcon || React__default.createElement(Yr, {
color: "#fff",
size: 24
}) : loadingIcon || React__default.createElement(Progress, {
style: {
"--nutui-progress-border-radius": 0,
position: "absolute",
bottom: 0
},
percent: item.percentage,
showRightPercentText: false,
color: "#2C68FF",
strokeWidth: "4",
showText: false
})), React__default.createElement("div", {
className: "nut-uploader__preview__progress__msg"
}, item.status === "uploading" ? loadingTip : item.message)), item.type.includes("image") ? React__default.createElement(React__default.Fragment, null, item.url && React__default.createElement("img", {
className: "nut-uploader__preview-img__c",
style: {
objectFit: fit,
borderRadius: typeof radius === "number" ? radius + "px" : radius
},
src: item.url,
alt: "",
onClick: function onClick() {
return handleItemClick(item, index);
},
onDragStart: function onDragStart(e) {
return handleDragstart(e, index);
},
onDragOver: function onDragOver(e) {
return handleDragOver(e);
},
onDrop: function onDrop(e) {
return handleDrop(e, index);
},
draggable: true
})) : React__default.createElement(React__default.Fragment, null, previewUrl ? React__default.createElement("img", {
className: "nut-uploader__preview-img__c",
style: {
borderRadius: typeof radius === "number" ? radius + "px" : radius
},
src: previewUrl,
alt: "",
onClick: function onClick() {
return handleItemClick(item, index);
}
}) : React__default.createElement("div", {
className: "nut-uploader__preview-img__file"
}, React__default.createElement("div", {
onClick: function onClick() {
return handleItemClick(item, index);
},
className: "nut-uploader__preview-img__file__name"
}, React__default.createElement(Link, {
color: "#808080"
}), " ", item.name))), item.status === "success" && (uploadLabel || item.uploadLabel) ? React__default.createElement("div", {
className: "tips tips--".concat(size)
}, React__default.createElement("span", null, uploadLabel || item.uploadLabel)) : null), previewType === "list" && React__default.createElement("div", {
className: "nut-uploader__preview-list"
}, React__default.createElement("div", {
className: "nut-uploader__preview-img__file__name ".concat(item.status),
onClick: function onClick() {
return handleItemClick(item, index);
}
}, React__default.createElement(Link, null), " ", item.name), deletable && React__default.createElement(Del, {
color: "#808080",
className: "nut-uploader__preview-img__file__del",
onClick: function onClick() {
return onDeleteItem(item, index);
}
}), item.status === "uploading" && React__default.createElement(React__default.Fragment, null, React__default.createElement(Progress, {
showRightPercentText: false,
percent: item.percentage,
color: "linear-gradient(270deg, rgba(18,126,255,1) 0%,rgba(32,147,255,1) 32.815625%,rgba(13,242,204,1) 100%)",
showText: false
}))));
}), maxCount > fileList.length && previewType === "picture" && !children && React__default.createElement("div", {
className: "nut-uploader__upload nut-uploader__upload--".concat(size, " ").concat(previewType, " ").concat(disabled ? "nut-uploader__upload-disabled" : ""),
style: {
borderRadius: typeof radius === "number" ? radius + "px" : radius
},
onClick: function onClick() {
return handleClickUploader();
}
}, disabled && React__default.createElement("div", {
style: {
width: "100%",
height: "100%",
position: "absolute",
zIndex: 2
},
onClick: function onClick() {
return handleClickUploader();
}
}), React__default.createElement("div", {
className: "nut-uploader__icon"
}, uploadIcon || (sourceType.includes("camera") ? React__default.createElement("img", {
src: PhotographIcon
}) : React__default.createElement("img", {
src: UploaderIcon
})), React__default.createElement("span", {
className: "nut-uploader__icon-tip"
}, uploadLabel)), capture ? React__default.createElement("input", {
className: "nut-uploader__input",
type: "file",
capture: "user",
name,
accept,
disabled,
multiple,
onChange: fileChange,
onClick: function onClick() {
return handleClickUploader();
}
}) : React__default.createElement("input", {
className: "nut-uploader__input",
type: "file",
name,
accept,
disabled,
multiple,
onChange: fileChange,
onClick: function onClick() {
return handleClickUploader();
}
})));
};
var Uploader = React__default.forwardRef(InternalUploader);
Uploader.defaultProps = defaultProps;
Uploader.displayName = "NutUploader";
export {
Uploader as default
};