Vue 仿QQ左滑删除组件功能


Posted in Javascript onMarch 12, 2018

前几天在做Vue项目开发的时候,因为之前别人写的代码有点小 bug,有人反应 IOS 上面的滑动点击有点问题,于是让我来帮忙解决,我看了看以前的代码实现比较繁琐,冗余,索性就直接自己重新写了一套,供大家参考,如有更好的方式,欢迎及时交流~

我们先看看效果图吧,毕竟无图无真相啊~

效果图

Vue 仿QQ左滑删除组件功能 

实现思路

具体实现思路如下:

  • 布局方面我采用的是 rem + flex 布局,具体如何结构和样式可以参考我的代码,值得注意的是后面的删除按钮是我通过定位放在了每一行的最后,超出隐藏了而已
  • 左滑和右滑是通过 touchstart 和 touchend 事件,通过判断滑动开始和结束,水平方向 x 的偏移量,如果大于一定得阈值认为是左滑动,小于一定的阈值则认为是右滑动
  • 左滑动和右滑动分别都是通过父级 li 元素的 translate 偏移量进行变化的,这里我的实现方式是提前声明好样式,通过改变当前父级 li 的 type 值,进行样式切换
  • 点击某一个滑块的时候,首先判断当前所有的滑块是否有处于 滑出状态的 ,如果有,则必须先将所有的滑块状态还原,如果没有,则点击生效,我这里只是弹出一个 alter ,具体业务可以根据实际填写
  • 删除相对简单,当滑块画出后,出现删除按钮,点击按钮,拿到当前的数组索引值,通过数组的 splice 方法,删除对应的数组值即可

具体实现

Html代码

<div class="container">
  <div class="page-title">滑动组件</div>
  <ul>
    <li class="list-item " v-for="(item,index) in list " data-type="0">
      <div class="list-box" @touchstart.capture="touchStart" @touchend.capture="touchEnd" @click="skip">
        <img class="list-img" :src="item.imgUrl" alt="">
        <div class="list-content">
          <p class="title">{{item.title}}</p>
          <p class="tips">{{item.tips}}</p>
          <p class="time">{{item.time}}</p>
        </div>
      </div>
      <div class="delete" @click="deleteItem" :data-index="index">删除</div>
    </li>
  </ul>
</div>

注意:我这里的数据全是本地 mock 的~

Css样式代码

.page-title{
  text-align: center;
  font-size: 17px;
  padding: 10px 15px;
  position: relative;
}
.page-title:after{
  content: " ";
  position: absolute;
  left: 0;
  bottom: 0;
  right: 0;
  height: 1px;
  border-bottom: 1px solid #ccc;
  color: #ccc;
  -webkit-transform-origin: 0 100%;
  transform-origin: 0 100%;
  -webkit-transform: scaleY(0.5);
  transform: scaleY(0.5);
  z-index: 2;
}
.list-item{
  position: relative;
  height: 1.6rem;
  -webkit-transition: all 0.2s;
  transition: all 0.2s;
}
.list-item[data-type="0"]{
  transform: translate3d(0,0,0);
}
.list-item[data-type="1"]{
  transform: translate3d(-2rem,0,0);
}
.list-item:after{
  content: " ";
  position: absolute;
  left: 0.2rem;
  bottom: 0;
  right: 0;
  height: 1px;
  border-bottom: 1px solid #ccc;
  color: #ccc;
  -webkit-transform-origin: 0 100%;
  transform-origin: 0 100%;
  -webkit-transform: scaleY(0.5);
  transform: scaleY(0.5);
  z-index: 2;
}
.list-box{
  padding: 0.2rem;
  background: #fff;
  display: flex;
  align-items: center;
  -webkit-box-sizing: border-box;
  box-sizing: border-box;
  justify-content: flex-end;
  position: absolute;
  top: 0;
  right: 0;
  bottom: 0;
  left: 0;
  font-size: 0;
}
.list-item .list-img{
  display: block;
  width: 1rem;
  height: 1rem;
}
.list-item .list-content{
  padding: 0.1rem 0 0.1rem 0.2rem;
  position: relative;
  flex: 1;
  flex-direction: column;
  align-items: flex-start;
  justify-content: center;
  overflow: hidden;
}
.list-item .title{
  display: block;
  color: #333;
  overflow: hidden;
  font-size: 15px;
  font-weight: bold;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.list-item .tips{
  display: block;
  overflow: hidden;
  font-size: 12px;
  color: #999;
  line-height: 20px;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.list-item .time{
  display: block;
  font-size: 12px;
  position: absolute;
  right: 0;
  top: 0.1rem;
  color: #666;
}
.list-item .delete{
  width: 2rem;
  height: 1.6rem;
  background: #ff4949;
  font-size: 17px;
  color: #fff;
  text-align: center;
  line-height: 1.6rem;
  position: absolute;
  top:0;
  right: -2rem;
}

这是核心的样式代码,还有部分样式重置代码放在了 App.vue 里,通过计算根节点 html 的字体大小的脚本我放在了 index.html 中~

js代码

export default{
 name: 'index',
 data () {
 return {
  list : [
  {
   title : 'Chrome更新了' ,
   imgUrl : './static/images/Chrome.png' ,
   tips : '不再支持Flash视频播放' ,
   time : '上午 8:30'
  },
  {
   title : '电影新资讯' ,
   imgUrl : './static/images/Sina.png' ,
   tips : '电影《红海行动》上映以来票房暴涨,很多国人表示对国产电影有了新的改观' ,
   time : '上午 12:00'
  },
        {
   title : '聚焦两会·共筑中国梦' ,
   imgUrl : './static/images/video.png' ,
   tips : '习近平代表的两会故事' ,
   time : '下午 17:45'
        },
        {
   title : '微信团队' ,
   imgUrl : './static/images/Wechat.png' ,
   tips : '您的帐号有异常登录,如非本人操作,请及时修改密码' ,
   time : '昨天'
        }
  ],
  startX : 0 ,
  endX : 0 ,
 }
 },
 methods : {
 //跳转
 skip(){
  if( this.checkSlide() ){
  this.restSlide();
      }else{
  alert('You click the slide!')
      }
 },
 //滑动开始
 touchStart(e){
   // 记录初始位置
  this.startX = e.touches[0].clientX;
 },
 //滑动结束
 touchEnd(e){
            // 当前滑动的父级元素
  let parentElement = e.currentTarget.parentElement;
  // 记录结束位置
  this.endX = e.changedTouches[0].clientX;
            // 左滑
  if( parentElement.dataset.type == 0 && this.startX - this.endX > 30 ){
  this.restSlide();
  parentElement.dataset.type = 1;
  }
            // 右滑
  if( parentElement.dataset.type == 1 && this.startX - this.endX < -30 ){
  this.restSlide();
  parentElement.dataset.type = 0;
  }
  this.startX = 0;
  this.endX = 0;
 },
    //判断当前是否有滑块处于滑动状态
    checkSlide(){
  let listItems = document.querySelectorAll('.list-item');
  for( let i = 0 ; i < listItems.length ; i++){
  if( listItems[i].dataset.type == 1 ) {
   return true;
        }
  }
  return false;
    },
 //复位滑动状态
 restSlide(){
  let listItems = document.querySelectorAll('.list-item');
             // 复位
  for( let i = 0 ; i < listItems.length ; i++){
  listItems[i].dataset.type = 0;
  }
 },
 //删除
 deleteItem(e){
   // 当前索引
  let index = e.currentTarget.dataset.index;
  // 复位
  this.restSlide();
  // 删除
  this.list.splice(index,1);
 }
 }
}

js代码就这么一些,每个函数都有注释说明, 相信都能看得懂, 就不多解释了。

结语

怎么样,是不是也米有想象中的那么难,只要理清思路即可,实现起来也并不是那么困难,当然我这里写的只是我的一个实现思路而已,可能你们自己会有更好的实现方式,为什么不拿出来分享呢?

完整代码我已经托管到 gitlub 上了,欢迎下载使用并留言!

Javascript 相关文章推荐
document.all还是document.getElementsByName?
Jul 21 Javascript
functional继承模式 摘自javascript:the good parts
Jun 20 Javascript
使用jQuery同时控制四张图片的伸缩实现代码
Apr 19 Javascript
从数据结构的角度分析 for each in 比 for in 快的多
Jul 07 Javascript
JS 获取select(多选下拉)中所选值的示例代码
Aug 02 Javascript
jquery实现兼容浏览器的图片上传本地预览功能
Oct 14 Javascript
JavaScript中的this关键字使用详解
Aug 14 Javascript
JS实现简单的右下角弹出提示窗口完整实例
Jun 21 Javascript
vue中使用protobuf的过程记录
Oct 26 Javascript
vue中img src 动态加载本地json的图片路径写法
Apr 25 Javascript
新手快速入门微信小程序组件库 iView Weapp
Jun 24 Javascript
原生js实现二级联动菜单
Nov 27 Javascript
JS中touchstart事件与click事件冲突的解决方法
Mar 12 #Javascript
Node.JS循环删除非空文件夹及子目录下的所有文件
Mar 12 #Javascript
Javascript中prototype与__proto__的关系详解
Mar 11 #Javascript
js中document.write和document.writeln的区别
Mar 11 #Javascript
Javascript 编码约定(编码规范)
Mar 11 #Javascript
JavaScript获取移动设备型号的实现代码(JS获取手机型号和系统)
Mar 10 #Javascript
js经验分享 JavaScript反调试技巧
Mar 10 #Javascript
You might like
PHP JSON 数据解析代码
2010/05/26 PHP
PHP中判断变量为空的几种方法分享
2013/08/26 PHP
Yii框架调试心得--在页面输出执行sql语句
2014/12/25 PHP
PHP中实现Bloom Filter算法
2015/03/30 PHP
PHP+jQuery+Ajax实现分页效果 jPaginate插件的应用
2015/10/09 PHP
Laravel 修改验证异常的响应格式实例代码详解
2020/05/25 PHP
起点页面传值js,有空研究学习下
2010/01/25 Javascript
jquery ajax请求实例深入解析
2012/11/26 Javascript
实现51Map地图接口(示例代码)
2013/11/22 Javascript
jQuery根据元素值删除数组元素的方法
2015/06/24 Javascript
AngularJS的一些基本样式初窥
2015/07/27 Javascript
JavaScript判断按钮被点击的方法
2015/12/13 Javascript
js利用正则表达式检验输入内容是否为网址
2016/07/05 Javascript
jQuery实现 上升、下降、删除、添加一行代码
2017/03/06 Javascript
详解Vue组件之间的数据通信实例
2017/06/17 Javascript
JS写谷歌浏览器chrome的外挂实例
2018/01/11 Javascript
利用angular、react和vue实现相同的面试题组件
2018/02/19 Javascript
解决vue多个路由共用一个页面的问题
2018/03/12 Javascript
微信小程序实现侧边栏分类
2019/10/21 Javascript
vue实现短信验证码输入框
2020/04/17 Javascript
Python实现二叉堆
2016/02/03 Python
Python进度条实时显示处理进度的示例代码
2018/01/30 Python
pytorch在fintune时将sequential中的层输出方法,以vgg为例
2019/08/20 Python
Python更换pip源方法过程解析
2020/05/19 Python
AJAX检测用户名是否存在的方法
2021/03/24 Javascript
大学生自我鉴定范文
2013/12/28 职场文书
事业单位绩效考核实施方案
2014/03/27 职场文书
委托书范本
2014/04/02 职场文书
租房协议书范本
2014/04/09 职场文书
有关九一八事变的演讲稿
2014/09/14 职场文书
销售简历自我评价怎么写
2014/09/26 职场文书
交通肇事罪辩护词
2015/05/21 职场文书
惊天动地观后感
2015/06/10 职场文书
辅导员学期工作总结
2015/08/14 职场文书
Django中session进行权限管理的使用
2021/07/09 Python
MySQL面试题讲解之如何设置Hash索引
2021/11/01 MySQL