分享一个看起来很酷的图片上传组件

cnblogs 2024-07-25 17:41:00 阅读 84

🧑‍💻 写在开头

点赞 + 收藏 === 学会🤣🤣🤣

可能有人觉得,这个组件很简单,没什么技术含量,其实确实也啥技术含量。但是,我是想借这个组件,来表达一种封装的思想在里面,希望可以帮助到一些朋友。

简单的描述下这个组件的功能:

  • 用户可以点击下面颜色比较绚丽的上传按钮,选择本地图片进行上传,也可以直接点击图片区域进行上传。
  • 上传过程中,会有一个上传中的进度条,上传完成后,会有一个上传成功的提示,如果失败了,会有一个上传失败的提示,而且支持重试。
  • 可以点击图片右上角的删除按钮,删除图片。
  • 并发控制,最多只能同时上传 N 张图片,也就是所谓的限频,这里是 2 张。

是不是看了这么多功能之后,就开始有点头皮发麻了?哈哈,不要怕,这就带你了解下,如何拆解这种功能,而且,学会了这种拆解的办法,后面你遇到更加复杂的,也可以得心应手。

拆解功能,逐步实现

首先,我们思考,我们该使用自底向上的思路,还是自顶向下的思路来拆解这个功能呢?我的建议自顶向下的思路去思考架构,然后自底向上的去实现功能。

因为我们这个图片上传组件是支持多长图片同时上传的,而且,我们还需要支持上传失败重试的功能,所以,为了让功能更加聚焦,我们把关注点放在 PhotoItem 上,没一个 PhotoItem 就是一个图片上传的单元。他可以独立的上传,独立的删除,独立的重试。

那么,为了让 PhotoItem 这个组件更加简洁,我们把上传逻辑放在hooks useUpload中,这样,PhotoItem 只需要关注自己的展示逻辑即可。

这样做的目的是做到关注点分离,通常来讲,也是符合单一职责原则的。写出来的组件维护性必定大大提升。

代码实现

我们先来看下 useUpload 的代码,因为PhotoItem 依赖他,我们先实现它。

"use client";

export const useUploader = (uploadAction) => {

const [isUploading, setIsUploading] = useState(false);

const [error, setError] = useState(null);

const upload = useCallback(async (file) => {

setIsUploading(true);

setError(null);

try {

return await uploadAction(file);

} catch (err) {

setError(err.message || 'Upload failed');

} finally {

setIsUploading(false);

}

}, [uploadAction]);

const reset = useCallback(() => {

setIsUploading(false);

setError(null);

}, []);

return { upload, isUploading, error, reset };

};

可以看到,我们的 hooks 非常之简单,就是暴露了一个实现图片上传的狗子 upload,然而,他替我们的组件管理了上传中,上传失败,的状态,因此,接下来看,我们的PhotoItem 组件将会有多清晰。

export const PhotoItem = ({

file,

onRemove,

onUploadComplete,

onUploadError,

}) => {

const { upload, isUploading, error, reset } = useUploader();

const startUpload = useCallback(async () => {

try {

const url = await upload(file);

onUploadComplete(url);

} catch (err) {

onUploadError();

}

}, [file, upload, onUploadComplete, onUploadError]);

useEffect(() => {

startUpload();

}, [queueUpload, startUpload]);

const handleRetry = () => {

reset();

startUpload();

};

return (

<div className="relative w-full h-20">

<img

src={URL.createObjectURL(file)}

/>

{!isUploading && !error(

Uploaded

)}

{isUploading && (

<Progress />

)}

{error && (

<span>Failed</span>

)}

</div>

);

};

OK,到目前为止,还是极其简单的,但是我们貌似忘记了一个很核心的功能,限制并发数。为什么要限制并发数,因为我们自己的服务器或者三方的服务器,可能会有并发数的限制,如果我们不限制并发数,可能会导致一次传多张图片是卡住。

思考,如何限制并发数

我们想一样,是谁触发了上传的呢?是不是 PhotoItem 组件呢?是的,我们可以在 PhotoItem 组件中,去控制并发数,但是,这样做,会导致 PhotoItem 组件的逻辑变得复杂,因为他不仅要关注自己的展示逻辑,还要关注并发数的控制逻辑。这就显的不太合适了。所以,我们应该把他丢出去对吧,截止到目前为止,我们的PhotoUploader 这个组件似乎并没有干任何事情,我们思考下,并发控制的逻辑是否应该是他来呢?

答案是显然的,我们应该把并发控制的逻辑放在 PhotoUploader 组件中,因为他是整个上传组件的入口,他应该关注并发控制,而不是 PhotoItem 组件,而且最本质的原因是,PhotoItem 也不关心是否有其他的 PhotoItem 。

那么,问题来了,并发控制怎么写呢?使用什么数据结构较为合适呢?不卖关子了,我们知道,队列是最合适的数据结构,因为他是先进先出的,我们可以把上传任务放在队列中,然后,每次上传完成,就从队列中取出一个任务,继续上传。

好,我们改造一下,我们的 PhotoItem 组件,让他不要直接执行上传逻辑,而是把他做成一个任务,然后,把任务放在队列中,然后,我们在 PhotoUploader 组件中,去控制并发数。

export const PhotoItem = ({

file,

onRemove,

...

queueUpload // 加一个队列操作器

}) => {

const { upload, isUploading, error, reset } = useUploader();

...

useEffect(() => {

queueUpload(startUpload); // 修改这里

}, [queueUpload, startUpload]);

const handleRetry = () => {

reset();

queueUpload(startUpload);//修改这里

};

// .... 其他几乎不变

在来看看我们的 PhotoUploader 组件,他是如何控制并发数的。很简单,我们只需要维护一个队列,然后,每次上传完成,就从队列中取出一个任务,继续上传。

const processQueue = useCallback(() => {

while (activeUploadsRef.current < MAX_CONCURRENT_UPLOADS && uploadQueueRef.current.length > 0) {

const nextUpload = uploadQueueRef.current.shift();

activeUploadsRef.current++;

nextUpload();

}

}, []);

const queueUpload = useCallback((startUpload) => {

if (activeUploadsRef.current < MAX_CONCURRENT_UPLOADS) {

activeUploadsRef.current++;

startUpload();

} else {

uploadQueueRef.current.push(startUpload);

}

}, []);

这里,只给出最最核心的逻辑,实际上就是维护的了一个任务队列,然后,每次上传完成,就判断下队列中是否还有任务,并且是否超过了并发数,如果没有超过,并且队列中还有任务,就继续上传。仅此而已。

总结一下

这个图片上传组件,看似简单,但是,他涉及到了很多的知识点,比如并发控制,上传失败重试,组件拆解,自顶向下的架构设计,自底向上的功能实现。我们在实现这个组件的过程中。有过很多的思考,比如:

  • 如何拆解功能,让组件更加聚焦,做到关注点分离。
  • 控制并发数,使用队列是最合适的数据结构。
  • 如何设计一个 hooks,让组件更加简洁。
  • 以及自顶向下的架构设计,自底向上的功能实现。

如果对您有所帮助,欢迎您点个关注,我会定时更新技术文档,大家一起讨论学习,一起进步。



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。