不用ECharts!从0到1徒手撸一个Vue3柱状图

0. 前言

柱状图(Bar Chart),也称为条形图,是一种常用的数据可视化图表,用于展示不同类别的数据量。 它通过水平或垂直的条形来表示数据的大小,条形的长度或高度与它们所代表的值成比例。

有许多JavaScript库可以帮助你快速创建柱状图,其中一些流行的包括: Chart.jsD3.jsHighchartsECharts等。

今天,我们将从0开始,基于Vue3,仅使用CSS3动画来实现一个最基础的柱状图。

1. 数据生成

1-1. 类型定义

ts 复制代码
interface BarItem {
    id: string; // ID
    title: string; // 标题
    val: number; // 数值
}

interface BarChartItem extends BarItem {
    percent: string; // 整体占比
}

1-2. 计算占比

ts 复制代码
import { random, maxBy } from 'lodash-es';

// 数据列表
const barItems: Ref<BarItem[]> = ref([])

// 自动计算百分比
const barChartItems = computed<BarChartItem[]>(() => {
    // 最大值
    const maxItem = maxBy(barItems.value, item => item.val)
    if (!maxItem) return []
    return barItems.value.map(item => {
        return {
            percent: (item.val / maxItem?.val * 100).toFixed(1),
            ...item,
        }
    })
})

使用数据项中的最大值作为基准,计算所有项的百分比。此百分比将用于后续展示柱状图的高度。

1-3. 添加、删除数据

ts 复制代码
import { random, maxBy, times } from 'lodash-es';

// 添加一项数据
const addItem = () => {
    barItems.value.push({
        id: useId(),
        title: `${curIdx.value++}-${faker.commerce.department()}`,
        val: random(1, 200)
    })
}

// 删除一项
const delItem = () => {
    const idx = random(0, barItems.value.length - 1)
    barItems.value.splice(idx, 1)
}

// 随机增加数值
const addVal = () => {
    barItems.value.forEach((item) => {
        item.val = Math.max(1, item.val + random(-50, 50))
    })
}

onMounted(() => {
    // 初始化默认添加5项数据
    times(5, addItem)
})
  • addItem:添加一项数据项,并生成1-200的随机值;同时这里我们使用faker来生成一个公司部门名称作为标题。
  • delItem: 随机删除一条数据。
  • addVal:为每个数据项随机增加-50到50的数据值,来模拟数据值变化的效果。
  • 页面初始化:默认添加5项数据

2. 页面结构

2-1. 数据操作

html 复制代码
<n-button-group>
    <n-button @click="addItem">添加一项</n-button>
    <n-button @click="delItem">删除一项</n-button>
    <n-button @click="addVal">添加随机值</n-button>
</n-button-group>

添加操作按钮,手动调整柱状图数据。

2-2. 柱状图

html 复制代码
<TransitionGroup name="list" tag="section" class="flex flex-row h-500px self-center gap-x-5">
    <div v-for="item in barChartItems" :key="item.id" class="flex flex-col h-full w-60px">
        <section class="flex-1 relative flex flex-col-reverse items-center">
            <div class="w-full transition transition-height transition-duration-500 bg-[#5771c0]" :style="{ height: `${item.percent}%` }"></div>
            <span class="text-[#333] font-bold text-base text-center">{{ item.val }}</span>
        </section>
        <span class="mt-3 text-[#333] font-bold text-center text-base">{{ item.title }}</span>
    </div>
</TransitionGroup>

使用v-for循环生成柱状图数据项,并制定柱状图颜色; 这里使用了Unocsstransition transition-height transition-duration-500来设置柱状图高度变化时的transition动效。

2-3. transition

css 复制代码
.list-move, /* 对移动中的元素应用的过渡 */
.list-enter-active,
.list-leave-active {
  transition: all 0.5s ease;
}

.list-enter-from,
.list-leave-to {
  opacity: 0;
  transform: translateY(30px);
}

.list-leave-active {
  position: absolute;
}

使用Vue自带的组件TransitionGroup组件为柱状图元素在增加或删除时,添加过度动效。

3. 最终效果

最后我们运行一下,查看最终效果:


以上全部代码已经分享到Gitee上,如有所需,欢迎自取。

相关推荐
Odoo老杨6 分钟前
Odoo 免费开源 ERP:通过 JavaScript 创建对话框窗口的技术实践分享
javascript·odoo·数字化转型·erp·企业信息化
Kika写代码15 分钟前
【微信小程序】2|轮播图 | 我的咖啡店-综合实训
前端·微信小程序·小程序
red润21 分钟前
使用 HTML5 Canvas 实现动态蜈蚣动画
前端·html·html5
sg_knight29 分钟前
VSCode如何修改默认扩展路径和用户文件夹目录到D盘
前端·ide·vscode·编辑器·web
一个处女座的程序猿O(∩_∩)O38 分钟前
完成第一个 Vue3.2 项目后,这是我的技术总结
前端·vue.js
mubeibeinv39 分钟前
项目搭建+图片(添加+图片)
java·服务器·前端
逆旅行天涯1 小时前
【Threejs】从零开始(六)--GUI调试开发3D效果
前端·javascript·3d
m0_748255261 小时前
easyExcel导出大数据量EXCEL文件,前端实现进度条或者遮罩层
前端·excel
长风清留扬1 小时前
小程序毕业设计-音乐播放器+源码(可播放)下载即用
javascript·小程序·毕业设计·课程设计·毕设·音乐播放器
web147862107232 小时前
C# .Net Web 路由相关配置
前端·c#·.net