从0搭建vue3组件库:Upload文件上传组件

简介: 从0搭建vue3组件库:Upload文件上传组件

image.png


这篇文章我们将一步步实现Upload文件上传组件,本次将采用setup语法糖的形式来实现这个组件,因为setup语法实在太好用啦。


创建目录结构



同样的我们和前面组件一样先创Upload组件的目录结构,以及其导出方式。目录结构如下图

image.png

其中style为样式,types.ts是我们组件需要接收的属性(props),index.ts将组件导出。而src/index.ts则是将我们所有组件集中导出的地方。

  • upload.vue
<template>
    <div class="k-upload">
        <input type="file">
    </div>
</template>
<script lang="ts">
import { defineComponent } from 'vue'
export default defineComponent({
    name: 'k-upload'
});
</script>

upload/index.ts

import upload from './upload.vue'
import { withInstall } from '@kitty-ui/utils'
export default withInstall(upload)

withInstall就是给组件赋予install方法的函数,便于app.use(xx)使用

import type { App, Plugin } from "vue"
type SFCWithInstall<T> = T & Plugin
export default <T>(comp: T) => {
    (comp as SFCWithInstall<T>).install = (app: App) => {
        //注册组件
        app.component((comp as any).name, comp)
    }
    return comp as SFCWithInstall<T>
}
  • src/index.ts
export { default as Button } from './button'
export { default as Icon } from './Icon'
export { default as Link } from './link'
export { default as Upload } from './upload'

目录搭建完毕我们就可以启动我们的vue3测试项目并引入Upload组件进行我们组件的开发啦

//app.vue
<template>
    <div class="upload-demo">
        <Upload></Upload>
    </div>
</template>
<script lang="ts" setup>
import { Upload } from 'kitty-ui'
</script>
<style lang="less">
.upload-demo {
    width: 400px;
}
</style>

原生input处理



我们的Upload组件肯定是不能用原生的input样式,所以我们要做的是将原生input隐藏(注意这个隐藏是display:none),所以vue中我们可以使用v-show。然后点击其它元素触发原生inputclick事件从而调起系统文件选择,选择完毕后我们可以在原生inputchange事件中获取到我们选中的文件。下面我们看一下在vue中如何实现

  • upload.vue
<template>
    <div class="k-upload">
         <input type="file" ref="kIpt" @change="getFiles" v-show="false">
        <div @click="fileUpload">
            <slot />
        </div>
    </div>
</template>
<script lang="ts">
import { defineComponent } from 'vue'
export default defineComponent({
    name: 'k-upload'
});
</script>
<script setup lang="ts">
import { ref, defineEmits } from 'vue'
import './style/index.less'
//获取dom元素
const kIpt = ref()
const emits = defineEmits(['getFilesList'])
const filesList = ref<File[]>([])
const fileUpload = () => {
    //触发input点击事件
    kIpt.value.click()
}
//获取文件列表
const getFiles = (e: Event) => {
    const files = (e.target as HTMLInputElement).files
    if (!files) return
    filesList.value.push(...Array.from(files))
    //将文件列表传给父组件
    emits('getFilesList', filesList.value)
}
</script>

然后在app.vue中使用

<template>
    <div class="upload-demo">
        <Upload>
            <Button type="primary" size="small">文件上传</Button>
        </Upload>
    </div>
</template>
<script lang="ts" setup>
import { Upload, Button } from 'kitty-ui'
</script>
<style lang="less">
.upload-demo {
    width: 400px;
}
</style>

image.png

此时点击按钮便可获取到文件

image.png

接下来我们要做的是要将我们选择的文件展示出来,并且可以删除


展示文件列表



其实很简单,只需遍历选中文件数组filesList即可。


<template>
    <div class="k-upload">
        <input type="file" ref="kIpt" @change="getFiles" v-show="false">
        <div @click="fileUpload">
            <slot />
        </div>
        <div class="k-upload-list">
            <div class="k-upload-list_item" v-for="(item, index) in filesList" :key="index">
                <div class="k-upload-list_item-name">
                    {{ item.name }}
                </div>
                <div class="k-upload-list_item-status-label">
                    <Icon name="ashbin" @click="delFile(index)" />
                </div>
            </div>
        </div>
    </div>
</template>
<script lang="ts">
...
//删除
const delFile = (index: number) => {
    filesList.value.splice(index, 1)
    emits('getFilesList', filesList.value)
}
</script>

其中样式如下style/index.less


.k-upload {
  .k-upload-list {
    .k-upload-list_item:first-child {
      margin-top: 10px;
    }
    .k-upload-list_item {
      transition: all 0.5s cubic-bezier(0.55, 0, 0.1, 1);
      font-size: 14px;
      color: #606266;
      line-height: 1.8;
      margin-top: 5px;
      position: relative;
      box-sizing: border-box;
      border-radius: 4px;
      width: 100%;
      .k-upload-list_item-name {
        color: #606266;
        display: block;
        margin-right: 40px;
        overflow: hidden;
        padding-left: 4px;
        text-overflow: ellipsis;
        transition: color 0.3s;
        white-space: nowrap;
      }
      .k-upload-list_item-status-label {
        position: absolute;
        right: 5px;
        top: 0;
        line-height: inherit;
      }
      &:hover {
        .k-upload-list_item-name {
          color: #409eff;
        }
        cursor: pointer;
        background: #f5f7fa;
      }
    }
  }
}

最终效果如下图

image.png


定义props



types.ts中我们暂时先接收multipleaccept属性来控制是否能多选以及接收文件类型

import { ExtractPropTypes } from 'vue'
export const uoloadType = {
    multiple: Boolean,
    accept: String
}
export type LinkProps = ExtractPropTypes<typeof uoloadType>

然后在组件中引入并使用这个类型upload.vue

<template>
    <div class="k-upload">
        <input type="file" :multiple="props.multiple" :accept="props.accept" ref="kIpt" @change="getFiles"
            v-show="false">
        <div @click="fileUpload">
            <slot />
        </div>
        <div class="k-upload-list">
            <div class="k-upload-list_item" v-for="(item, index) in filesList" :key="index">
                <div class="k-upload-list_item-name">
                    {{ item.name }}
                </div>
                <div class="k-upload-list_item-status-label">
                    <Icon name="ashbin" @click="delFile(index)" />
                </div>
            </div>
        </div>
    </div>
</template>
<script setup lang="ts">
...
//引入类型
import { uoloadType } from './types'
const props = defineProps(uoloadType)
...
</script>

最后我们在 app.vue中传入这两个属性

<template>
    <div class="upload-demo">
        <Upload @getFilesList="getFilesList" multiple accept="image/*">
            <Button type="primary" size="small">文件上传</Button>
        </Upload>
    </div>
</template>
<script lang="ts" setup>
import { Upload, Button } from 'kitty-ui'
const getFilesList = (files: File[]) => {
    console.log(files)
}
</script>
<style lang="less">
.upload-demo {
    width: 400px;
}
</style>

然后就会发现可以进行多选了并且只能选择图片类型的文件。

这里实现的Upload组件最终是将文件抛出给了使用者,至于如何使用这些文件(上传,预览等)则由开发者自己定义了。

下篇文章将介绍文件Upload组件如何实现拖拽上传的。欢迎关注 从零搭建Vue3组件库专栏 将持续更新一些组件的实现。


下一篇



实现Upload组件文件拖拽上传


写在最后



源码地址 kitty-ui,可以直接clone使用(顺手点个star呗),使用方法:

  • 安装 pnpm npm i pnpm -g
  • 安装所有依赖 pnpm install
  • 启动本地调试vue3项目 pnpm run exm:dev
  • 打包 pnpm run build
  • 启动文档 pnpm run docs:dev
  • 打包文档 pnpm run docs:build
  • 启动打包后文档服务 pnpm run docs:serve


目录
打赏
0
0
0
0
16
分享
相关文章
Vue 中 key 属性的深入解析:改变 key 导致组件销毁与重建
Vue 中 key 属性的深入解析:改变 key 导致组件销毁与重建
160 0
斩获开发者口碑!SnowAdmin:基于 Vue3 的高颜值后台管理系统,3 步极速上手!
SnowAdmin 是一款基于 Vue3/TypeScript/Arco Design 的开源后台管理框架,以“清新优雅、开箱即用”为核心设计理念。提供角色权限精细化管理、多主题与暗黑模式切换、动态路由与页面缓存等功能,支持代码规范自动化校验及丰富组件库。通过模块化设计与前沿技术栈(Vite5/Pinia),显著提升开发效率,适合团队协作与长期维护。项目地址:[GitHub](https://github.com/WANG-Fan0912/SnowAdmin)。
381 5
Vue 表情包输入组件的实现代码:支持自定义表情库、快捷键发送和输入框联动的聊天表情解决方案
本文详细介绍了在 Vue 项目中实现一个功能完善、交互友好的表情包输入组件的方法,并提供了具体的应用实例。组件设计包含表情分类展示、响应式布局、与输入框的交互及样式定制等功能。通过核心技术实现,如将表情插入输入框光标位置和点击外部关闭选择器,确保用户体验流畅。同时探讨了性能优化策略,如懒加载和虚拟滚动,以及扩展性方案,如自定义主题和国际化支持。最终,展示了如何在聊天界面中集成该组件,为用户提供丰富的表情输入体验。
109 8
Vue 表情包输入组件实现代码及详细开发流程解析
这是一篇关于 Vue 表情包输入组件的使用方法与封装指南的文章。通过安装依赖、全局注册和局部使用,可以快速集成表情包功能到 Vue 项目中。文章还详细介绍了组件的封装实现、高级配置(如自定义表情列表、主题定制、动画效果和懒加载)以及完整集成示例。开发者可根据需求扩展功能,例如 GIF 搜索或自定义表情上传,提升用户体验。资源链接提供进一步学习材料。
74 1
Vue 项目中如何自定义实用的进度条组件
本文介绍了如何使用Vue.js创建一个灵活多样的自定义进度条组件。该组件可接受进度段数据数组作为输入,动态渲染进度段,支持动画效果和内容展示。当进度超出总长时,超出部分将以红色填充。文章详细描述了组件的设计目标、实现步骤(包括props定义、宽度计算、模板渲染、动画处理及超出部分的显示),并提供了使用示例。通过此组件,开发者可根据项目需求灵活展示进度情况,优化用户体验。资源地址:[https://pan.quark.cn/s/35324205c62b](https://pan.quark.cn/s/35324205c62b)。
44 0
Vue 2 与 Vue 3 的区别:深度对比与迁移指南
Vue.js 是一个用于构建用户界面的渐进式 JavaScript 框架,在过去的几年里,Vue 2 一直是前端开发中的重要工具。而 Vue 3 作为其升级版本,带来了许多显著的改进和新特性。在本文中,我们将深入比较 Vue 2 和 Vue 3 的主要区别,帮助开发者更好地理解这两个版本之间的变化,并提供迁移建议。 1. Vue 3 的新特性概述 Vue 3 引入了许多新特性,使得开发体验更加流畅、灵活。以下是 Vue 3 的一些关键改进: 1.1 Composition API Composition API 是 Vue 3 的核心新特性之一。它改变了 Vue 组件的代码结构,使得逻辑组
424 0
|
3月前
|
vue实现任务周期cron表达式选择组件
vue实现任务周期cron表达式选择组件
341 4
基于 Vue 与 D3 的可拖拽拓扑图技术方案及应用案例解析
本文介绍了基于Vue和D3实现可拖拽拓扑图的技术方案与应用实例。通过Vue构建用户界面和交互逻辑,结合D3强大的数据可视化能力,实现了力导向布局、节点拖拽、交互事件等功能。文章详细讲解了数据模型设计、拖拽功能实现、组件封装及高级扩展(如节点类型定制、连接样式优化等),并提供了性能优化方案以应对大数据量场景。最终,展示了基础网络拓扑、实时更新拓扑等应用实例,为开发者提供了一套完整的实现思路和实践经验。
228 77
Vue 自定义进度条组件封装及使用方法详解
这是一篇关于自定义进度条组件的使用指南和开发文档。文章详细介绍了如何在Vue项目中引入、注册并使用该组件,包括基础与高级示例。组件支持分段配置(如颜色、文本)、动画效果及超出进度提示等功能。同时提供了完整的代码实现,支持全局注册,并提出了优化建议,如主题支持、响应式设计等,帮助开发者更灵活地集成和定制进度条组件。资源链接已提供,适合前端开发者参考学习。
137 17

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问