时间轴组件 by Vue.js
在繁重的业务工作中,如何提升自己的技术能力,而不是变为特定业务领域的熟练工,方法之一就是在业务工作当中提炼、抽象出通用的技术内容,总结归纳,并进一步思考是否有更好的解决办法。
在公司的项目开发中,涉及到了移动端H5页面的时间轴展示效果。现有的轮子比如ElementUI、iView中,都没有专门的时间轴组件,于是就萌生了自己封装一个的想法。
说干就干,杜绝拖延症,正好今天周末,就封装了一个时间轴的组件上传到了npm,大家有需求可以安装试一下。
npm install uni-time-line -S
效果如下图:
实现思路
开发一个vue组件,首先要确定好三要素props
、slot
和event
。因为第一版功能比较简单,所以只使用了props
。
要实现一个时间轴,可以让用户自定义的内容包括,icon图标的样式、标题的文字、内容体的文字,还有两个条目之间的距离。
于是我们props的定义就出来了:
uni_data: {//要使用时间轴的内容列表,包含标题和内容
type: Array,
default() {
return [
{
title: "这是标题",
content: "这是内容"
},
{
title: "标题过长会显示省略号",
content: "内容过长会隐藏"
},
{
title: "更多效果可以自定义",
content: "更多效果可以自定义"
}
];
}
},
space: {//条目之间的距离,默认100px
type: String,
default: "100px"
},
uni_icon: {//icon的地址,默认提供一个我觉得还算好看的
type: String,
default: "http://qiniu.iborge.cn/dian.png"
}
上面的定义中,稍微不太好理解的是,如何将条目之间的距离动态的设定到style
属性上呢?在这里我们使用了computed,即:
computed: {
uni_space() {
return "height:" + this.space;
}
},
这样,用户传入100px,设置到属性上就是style="height:100px"
.
下一步就是如何实现时间轴的效果,在这里我想到的一个简单的思路是使用div
的左边框。
其实大家看到的每条时间轴的竖线,都是一个box的左边框。将其相对定位一下,将包含icon和标题的p
绝对定位在竖线顶端,就实现了时间轴的效果。代码如下:
.uni_listbox {
box-sizing: border-box;
margin: 20px;
height: 60px;
position: relative;
}
.uni_listbox p {
position: absolute;
top: 0;
left: 0;
margin-left: -9px;
margin-top: -18px;
vertical-align: middle;
height: 16px;
line-height: 16px;
}
这么一来,基本上完成了时间轴的效果,但还有一点小小的问题。就是列表的最后一项,会有一条竖线,但是下面已经没有了新的内容。这样看上去就会很难看,应该如何处理呢?其实只需要简单的一句表达式就可以:
:class="[uni_data.length==index+1?uni_listbox_last:uni_listbox_notlast,uni_listbox]"
上面这句话的意思是,如果判断当前为数据的最后一项,那么就使用uni_listbox_last
的样式,否则就用uni_listbox_notlast
,而uni_listbox
的样式,则不管条件如何都会生效。
这样,我们只要把uni_listbox_last
的边框去掉,高度降低,就实现了文章开头中图片中的效果。
至此,时间轴的组件基本上就开发完了。
还可以根据自己的需求,添加标题行内容溢出显示省略号等各种效果。
下面贴上完整的源码:
<template>
<div id="uni_timeline">
<div
:style="uni_space"
v-for="(item,index) in uni_data"
:key="index"
:class="[uni_data.length==index+1?uni_listbox_last:uni_listbox_notlast,uni_listbox]"
>
<p>
<img :src="uni_icon" :class="uni_timeline_icon">
<span class="uni_timeline_title">{{item.title}}</span>
</p>
<div class="uni_timeline_content">{{item.content}}</div>
</div>
</div>
</template>
<script>
export default {
data() {
return {
uni_timeline_icon: "uni_timeline_icon",
uni_listbox_last: "uni_listbox_last",
uni_listbox_notlast: "uni_listbox_notlast",
uni_listbox: "uni_listbox"
};
},
computed: {
uni_space() {
return "height:" + this.space;
}
},
props: {
uni_data: {
type: Array,
default() {
return [
{
title: "这是标题",
content: "这是内容"
},
{
title: "标题过长会显示省略号",
content: "内容过长会隐藏"
},
{
title: "更多效果可以自定义",
content: "更多效果可以自定义"
}
];
}
},
space: {
type: String,
default: "100px"
},
uni_icon: {
type: String,
default: "http://qiniu.iborge.cn/dian.png"
}
}
};
</script>
<style>
* {
margin: 0;
padding: 0;
}
.uni_listbox {
box-sizing: border-box;
margin: 20px;
height: 60px;
position: relative;
}
.uni_listbox_last {
height: 20px;
}
.uni_listbox_notlast {
height: 60px;
margin-left: 18px;
border-left: 3px solid rgb(228, 228, 228);
}
.uni_listbox p {
position: absolute;
top: 0;
left: 0;
/* transform: translateY(-50%); */
margin-left: -9px;
margin-top: -18px;
vertical-align: middle;
height: 16px;
line-height: 16px;
}
.uni_timeline_icon {
position: absolute;
top: 0;
left: 0;
display: inline-block;
width: 16px;
height: 16px;
vertical-align: middle;
}
.uni_timeline_title {
font-weight: bold;
text-align: left;
display: inline-block;
width: 300px;
margin: 0 30px;
color: rgb(100, 100, 100);
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.uni_timeline_content {
text-align: left;
height: 70%;
padding: 10px 21px;
font-size: 14px;
color: rgb(100, 100, 100);
overflow: hidden;
}
</style>
- linux学习第三十九篇:Apache用户认证,域名跳转,Apache访问日志
- Python sorted()函数
- Python filter()函数
- Python reduce()函数
- Python map()函数
- Python iterator迭代器
- How To Implement The Decision Tree Algorithm From Scratch In Python (从零开始在Python中实现决策树算法)
- 『教程』微信小程序webview的使用
- How to Save an ARIMA Time Series Forecasting Model in Python (如何在Python中保存ARIMA时间序列预测模型)
- Decision Trees in Apache Spark (Apache Spark中的决策树)
- Feature Selection For Machine Learning in Python (Python机器学习中的特征选择)
- 简约的JAVA版本MapReduce和日常No.25
- 根据职位说明使用机器学习来检索相关简历
- 微信小游戏初体验
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- IDEA奇淫小技巧
- [PHP框架] ThinkPHP6 介绍、安装及配置
- 【翻译】withoutboats 的 io-uring 笔记
- [Python]随机生成大量的虚拟信息测试数据(姓名,手机号,ID,家庭住址等)
- Java核心技术之动态代理
- 开源verilog仿真工具iverilog+GTKWave初体验
- [算法] 数组排序 - 冒泡排序法与直接选择排序法
- TS 设计模式01 - 工厂模式
- Spring与Mybatis的整合
- Python中的计数 - Counter类
- vue 记账本
- c/c++补完计划(三): 素数统计
- Python正则表达式(上)
- 附001.Nginx location语法规则
- 016.Nginx HTTPS