当前位置: 首页 > article >正文

图片叠加拖拽对比展示效果实现——Vue版

图片叠加拖拽对比展示效果实现——Vue版

项目中遇见一个需求:2张图片按竖线分割,左右两侧分别展示对应图片,通过滚动条拖动对应展示图片区域;;
网上搜索了下,没有找到直接可用的组件,这里自己封装了一个次功能组件;后面会附上完整代码。,希望可以帮助到由此需求的小伙伴;

文章目录

  • 图片叠加拖拽对比展示效果实现——Vue版
    • 一、实现效果预览
    • 二、HTML 部分
      • 1. 组件包含:头部插槽、底部插槽、切换比较图片组
      • 2. 代码示例
    • 三、JS 部分
      • 1. 包含组件传参,左侧图片地址、右侧图片地址、事件处理等
      • 2. JS 部分代码如下:
    • 四、 CSS 样式代码
    • 六、完整使用示例

一、实现效果预览

在这里插入图片描述

二、HTML 部分

1. 组件包含:头部插槽、底部插槽、切换比较图片组

2. 代码示例

<template>
  <div class="image-comparator">
    <!--  头部插槽  -->
    <slot name="header"></slot>

    <!--   切换对比图片  下一组、上一组 -->
    <div class="prev" @click="prevFun"><i class="el-icon-arrow-left"></i></div>
    <div class="next" @click="nextFun"><i class="el-icon-arrow-right"></i></div>

    <!-- 核心标签代码   -->
    <div class="container" @mouseenter="startDrag" @mousemove="onDrag" @mouseup="endDrag" @mouseleave="endDrag">
      <div class="image-wrapper"
           :style="{ clipPath: `inset(0 ${100 - (dividerPosition / containerWidth * 100)}% 0 0)` }">
        <img src="https://fuss10.elemecdn.com/9/bb/e27858e973f5d7d3904835f46abbdjpeg.jpeg" class="image"/>
      </div>
      <div class="divider" :style="{ left: `${dividerPosition}px` }"></div>
      <div class="image-wrapper" :style="{ clipPath: `inset(0 0 0 ${(dividerPosition / containerWidth * 100)}%)` }">
        <img src="https://fuss10.elemecdn.com/d/e6/c4d93a3805b3ce3f323f7974e6f78jpeg.jpeg" class="image"/>
      </div>
    </div>

    <!--  底部插槽  -->
    <slot name="footer"></slot>

  </div>
</template>

三、JS 部分

1. 包含组件传参,左侧图片地址、右侧图片地址、事件处理等

2. JS 部分代码如下:

<script>

export default {

  name: 'imageComparator',
  props: {
    leftImgUrl: {
      type: String,
    },
    rightImgUrl: {
      type: String,
    },

  },

  data() {
    return {
      containerWidth: 500,
      dividerPosition: 250,
      dragging: false,
      containerRect: null,
    };
  },
  methods: {
    updateContainerWidth() {
      this.containerWidth = this.$el.querySelector('.container').clientWidth;
    },
    startDrag(event) {

      this.dragging = true;
      this.containerRect = this.$el.querySelector('.container').getBoundingClientRect();
    },
    onDrag(event) {
      if (this.dragging && this.containerRect) {
        let newPosition = event.clientX - this.containerRect.left;
        if (newPosition < 0) newPosition = 0;
        if (newPosition > this.containerRect.width) newPosition = this.containerRect.width;
        this.dividerPosition = newPosition;
      }
    },
    endDrag() {
      this.dragging = false;
      this.containerRect = null;
    },
    prevFun() {
      this.$emit('prevFun');
    },
    nextFun() {
      this.$emit('nextFun');

    },
  },
  mounted() {
    this.updateContainerWidth();
    window.addEventListener('resize', this.updateContainerWidth);
  },
  beforeDestroy() {
    window.removeEventListener('resize', this.updateContainerWidth);
  },


};


</script>

四、 CSS 样式代码

<style scoped lang="less">
.image-comparator {
  width: 90%;
  /*max-width: 600px;*/
  margin: 0 auto;

  .container {
    position: relative;
    width: 100%;
    height: 600px;
    overflow: hidden;
  }

  .image-wrapper {
    position: absolute;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;
    overflow: hidden;
  }

  .image {
    width: 100%;
    height: 100%;
    object-fit: cover;
  }

  .divider {
    position: absolute;
    top: 0;
    bottom: 0;
    width: 2px;
    background-color: black;
    cursor: ew-resize;
    z-index: 10;
  }

  .prev, .next {
    position: absolute;
    top: 50%;
    transform: translateY(-50%);
    z-index: 100;
    cursor: pointer;
    font-size: 40px;
    color: white;
    background-color: rgba(0, 0, 0, 0.5);
    padding: 10px;
    border-radius: 50%;
    transition: all 0.3s ease;
    left: 10px;

    &:hover {
      background-color: rgba(0, 0, 0, 0.8);
      color: #0bf4cb;
    }

    &.next {
      right: 10px;
      left: auto;
    }
  }
}


</style>

## 五、总结

本文着重介绍了Vue2 封装的图片分割展示组件,如果小伙伴项目是V3 可以自行修改Vue 组件即可,还有写组件传参地方,需要的话自行改为动态参数即可,这里父组件不做过多介绍,以免浪费小伙伴看代码时间;

六、完整使用示例

以下是一个完整的 代码示例

<template>
  <div class="image-comparator">
    <!--  头部插槽  -->
    <slot name="header"></slot>

    <!--   切换对比图片  下一组、上一组 -->
    <div class="prev" @click="prevFun"><i class="el-icon-arrow-left"></i></div>
    <div class="next" @click="nextFun"><i class="el-icon-arrow-right"></i></div>

    <!-- 核心标签代码   -->
    <div class="container" @mouseenter="startDrag" @mousemove="onDrag" @mouseup="endDrag" @mouseleave="endDrag">
      <div class="image-wrapper"
           :style="{ clipPath: `inset(0 ${100 - (dividerPosition / containerWidth * 100)}% 0 0)` }">
        <img src="https://fuss10.elemecdn.com/9/bb/e27858e973f5d7d3904835f46abbdjpeg.jpeg" class="image"/>
      </div>
      <div class="divider" :style="{ left: `${dividerPosition}px` }"></div>
      <div class="image-wrapper" :style="{ clipPath: `inset(0 0 0 ${(dividerPosition / containerWidth * 100)}%)` }">
        <img src="https://fuss10.elemecdn.com/d/e6/c4d93a3805b3ce3f323f7974e6f78jpeg.jpeg" class="image"/>
      </div>
    </div>

    <!--  底部插槽  -->
    <slot name="footer"></slot>

  </div>
</template>


<script>
export default {

  name: 'imageComparator',
  props: {
    leftImgUrl: {
      type: String,
    },
    rightImgUrl: {
      type: String,
    },

  },

  data() {
    return {
      containerWidth: 500,
      dividerPosition: 250,
      dragging: false,
      containerRect: null,
    };
  },
  methods: {
    updateContainerWidth() {
      this.containerWidth = this.$el.querySelector('.container').clientWidth;
    },
    startDrag(event) {

      this.dragging = true;
      this.containerRect = this.$el.querySelector('.container').getBoundingClientRect();
    },
    onDrag(event) {
      if (this.dragging && this.containerRect) {
        let newPosition = event.clientX - this.containerRect.left;
        if (newPosition < 0) newPosition = 0;
        if (newPosition > this.containerRect.width) newPosition = this.containerRect.width;
        this.dividerPosition = newPosition;
      }
    },
    endDrag() {
      this.dragging = false;
      this.containerRect = null;
    },
    prevFun() {
      this.$emit('prevFun');
    },
    nextFun() {
      this.$emit('nextFun');

    },
  },
  mounted() {
    this.updateContainerWidth();
    window.addEventListener('resize', this.updateContainerWidth);
  },
  beforeDestroy() {
    window.removeEventListener('resize', this.updateContainerWidth);
  },


};
</script>
<style scoped lang="less">
.image-comparator {
  width: 90%;
  /*max-width: 600px;*/
  margin: 0 auto;

  .container {
    position: relative;
    width: 100%;
    height: 600px;
    overflow: hidden;
  }

  .image-wrapper {
    position: absolute;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;
    overflow: hidden;
  }

  .image {
    width: 100%;
    height: 100%;
    object-fit: cover;
  }

  .divider {
    position: absolute;
    top: 0;
    bottom: 0;
    width: 2px;
    background-color: black;
    cursor: ew-resize;
    z-index: 10;
  }

  .prev, .next {
    position: absolute;
    top: 50%;
    transform: translateY(-50%);
    z-index: 100;
    cursor: pointer;
    font-size: 40px;
    color: white;
    background-color: rgba(0, 0, 0, 0.5);
    padding: 10px;
    border-radius: 50%;
    transition: all 0.3s ease;
    left: 10px;

    &:hover {
      background-color: rgba(0, 0, 0, 0.8);
      color: #0bf4cb;
    }

    &.next {
      right: 10px;
      left: auto;
    }
  }
}


</style>




看到这里的小伙伴,欢迎点赞、评论,收藏!

如有前端相关疑问,请评论区留言,博主会在第一时间解答!!!


http://www.kler.cn/a/460714.html

相关文章:

  • GitHub 及 GitHub Desktop 详细使用教程(通俗易懂)
  • 2501d,d.110
  • 【Vue】获取el-select修改前后的数据
  • 【异常解决】生产环境 net :: ERR_INCOMPLETE_CHUNKED_ENCODING的问题修复
  • 图像概念与分类
  • OpenGL ES 04 图片数据是怎么写入到对应纹理单元的
  • 高斯核函数(深入浅出)
  • Java 21 优雅和安全地处理 null
  • 短视频矩阵系统贴牌流程全解析
  • java锁
  • FFmpeg 编码和解码
  • 《Java编程入门官方教程》第十六章练习答案
  • 【Spring MVC 核心概念】揭秘概念和整体架构
  • 图表控件Aspose.Diagram入门教程(7):在 C# 中删除 Visio 形状保护
  • OpenStack系列第三篇:CentOS7 上部署 OpenStack(Train版)集群教程 Ⅳ Dashboard Cinder 服务部署
  • 低代码/无代码开发平台下的电商API接口创新应用
  • Microsoft 365 Copilot模型多元化,降低对OpenAI依赖并降低成本
  • gitlab 还原合并请求
  • JVM调优(内存、GC、JVM参数)
  • 《庐山派从熟悉到...》Sensor 模块(摄像头)基础设置
  • html+css+js网页设计 美食 美食杰8个页面
  • AviatorScript
  • 数据结构-排序思想
  • 1月第一讲:WxPython跨平台开发框架之前后端结合实现附件信息的上传及管理
  • 【MyBatis】MyBatis项目的创建、配置和启动
  • 异步请求在TypeScript网络爬虫中的应用