原生js实现放大镜组件

作者:蒲公英芽 时间:2024-05-11 09:06:05 

本文实例为大家分享了js实现放大镜组件开发的具体代码,供大家参考,具体内容如下

功能需求:

1、根据图片数组创建图标列表;
2、鼠标滑过图标时,当前图标增加红色边框;
3、鼠标滑过图标时,上方图片区域显示对应的图片,右侧显示放大后的图片内容;
4、鼠标在图片区域移动时,在右侧实现放大效果;
5、下方图标列表,点击左右按钮,实现翻页效果;
6、当图标内容不够一页时,只移动到最后一个图标的位置;

以京东的详情页为例,看一下效果:

原生js实现放大镜组件

放大镜内容写在 Zoom.js 文件里,下方的图标列表内容写在 IconList.js 文件里,当鼠标滑过下面的图标时,需要更改放大镜里div的背景图片,这里用到了事件抛发。

下面附上代码:

html结构 :


<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>zoom</title>
</head>
<body>
<script type="module">
 import Zoom from './js/Zoom.js';
 //图标数组
 let list=["a_icon.jpg","e_icon.jpg","f_icon.jpg","g_icon.jpg","h_icon.jpg","i_icon.jpg","j_icon.jpg",];
 init();
 function init(){
  let zoom=new Zoom(list,"./img/");
  zoom.appendTo("body");
 }
</script>
</body>
</html>

Zoom.js文件,创建放大镜组件:


import Utils from "./Utils.js";
import IconList from './IconList.js';
export default class Zoom{
static styles=false;
static small_width=450;
static mask_width=303.75;
static zoom_width=540;
static SET_BG_IMG="set_bg_img";
constructor(_list,_basePath){
 if(_basePath) _list=_list.map(item=>_basePath+item);
 //创建外层的div容器
 this.elem=this.createE();
 //监听事件,改变zoomSmall的背景图
 document.addEventListener(Zoom.SET_BG_IMG,e=>this.setBgImg(e));
 //创建下方的icon列表
 this.createIconList(_list,this.elem);
}
createE(){
 //创建外层div容器
 let div=Utils.createE("div");
 div.className="zoomContainer";
 div.innerHTML=`<div class="zoomSmall" id="zoomSmall"><div class="zoomMask" id="zoomMask"></div></div>
 <div class="zoomContent" id="zoomCont"></div>`;
 //设置样式
 Zoom.setStyle();
 //获取样式
 Utils.getIdElem(div,this);
 //监听鼠标滑入事件
 this.zoomSmall.addEventListener("mouseenter",e=>this.mouseHandler(e));
 return div;
}
appendTo(parent){
 Utils.appendTo(this.elem,parent);
}
setBgImg(e){
 //设置背景图片
 this.zoomSmall.style.backgroundImage=`url(${e.src})`;
 this.zoomCont.style.backgroundImage=`url(${e.src})`;
}
createIconList(list,parent){
 //创建下方icon图标列表
 let iconList=new IconList(list);
 Utils.appendTo(iconList.elem,parent);
}
mouseHandler(e){
 switch (e.type) {
  case "mouseenter":
   //鼠标滑入后,显示遮罩和右侧大图片
   this.zoomMask.style.display="block";
   this.zoomCont.style.display="block";
   //监听鼠标移动和滑出事件
   this.mouseHandlers=e=>this.mouseHandler(e);
   this.zoomSmall.addEventListener("mousemove",this.mouseHandlers);
   this.zoomSmall.addEventListener("mouseleave",this.mouseHandlers);
   break;
  case "mousemove":
   //遮罩移动
   this.zoomMaskMove(e);
   break;
  case "mouseleave":
   //鼠标滑出后,显示遮罩和右侧大图片
   this.zoomMask.style.display="none";
   this.zoomCont.style.display="none";
   //移除鼠标移动和滑出事件
   this.zoomSmall.removeEventListener("mousemove",this.mouseHandlers);
   this.zoomSmall.removeEventListener("mouseleave",this.mouseHandlers);
   break;
 }
}
zoomMaskMove(e){
 //遮罩移动
 let rect=this.elem.getBoundingClientRect();
 //计算let和top的值,等于鼠标的坐标-父容器的left值-遮罩的一半宽
 let x=e.clientX-rect.x-Zoom.mask_width/2;
 let y=e.clientY-rect.y-Zoom.mask_width/2;
 //判断left和top的范围
 if(x<0) x=0;
 if(x>Zoom.small_width-Zoom.mask_width) x=Zoom.small_width-Zoom.mask_width;
 if(y<0) y=0;
 if(y>Zoom.small_width-Zoom.mask_width) y=Zoom.small_width-Zoom.mask_width;
 this.zoomMask.style.left=x+"px";
 this.zoomMask.style.top=y+"px";
 //大图片移动
 this.zoomContMove(x,y);
}
zoomContMove(_x,_y){
 //计算大图片的背景定位,公式:zoom的宽/mask的宽=zoom的背景left值/mask的left值
 let x=-Zoom.zoom_width/Zoom.mask_width*_x;
 let y=-Zoom.zoom_width/Zoom.mask_width*_y;
 this.zoomCont.style.backgroundPosition=x+"px "+y+"px";
}
static setStyle(){
 //设置样式
 if(Zoom.styles) return;
 Zoom.styles=true;
 Utils.insertCss(".zoomContainer",{
  width:Zoom.small_width+"px",
  height:Zoom.small_width+"px",
  position:"relative"
 })
 Utils.insertCss(".zoomSmall",{
  width:Zoom.small_width+"px",
  height:Zoom.small_width+"px",
  border: "1px solid #000",
  backgroundSize: "100% 100%",
  position:"absolute",
  left:"0px",
  top:"0px"
 })
 Utils.insertCss(".zoomMask",{
  width: this.mask_width + "px",
  height: this.mask_width + "px",
  backgroundColor: "rgba(200,170,0,0.3)",
  position: "absolute",
  left: "0px",
  top: "0px",
  display: "none"
 })
 Utils.insertCss(".zoomContent",{
  width: this.zoom_width + "px",
  height: this.zoom_width + "px",
  border: "1px solid #ccc",
  position: "absolute",
  left: (this.small_width + 2) + "px",
  top: "0px",
  display: "none"
 })
}
}

IconList.js文件,创建下方图标列表,并完成翻页效果:


import Utils from "./Utils.js";
import Zoom from "./Zoom.js";
export default class IconList{
static styles=false;
static num=5;//每页显示的图标数
static gap=0;//表示li的左右间距
position=0;//当前显示的图标为第几页
x=0;//列表的left值
prepIcon;//上一个点击的图标
static SET_BG_IMG="set_bg_img";
constructor(list){
 this.list=list;
 this.elem=this.createE();
}
createE(){
 //创建外层容器
 let div=Utils.createE("div");
 div.className="iconContainer";
 div.innerHTML=`<img class="prevBtn" src="./img/prev.png"><div class="iconListCont">${this.createIcon()}</div><img class="nextBtn" src="./img/next.png">`;
 //设置css样式
 IconList.setStyles(this.list);
 //获取元素
 Utils.getIdElem(div,this);
 //外层容器监听点击事件
 div.addEventListener("click",e=>this.clickHandler(e));
 //图标列表监听鼠标滑过事件
 this.iconList.addEventListener("mouseover",e=>this.mouseHandler(e));
 //默认显示第一个图标的边框
 this.setIconState(this.iconList.firstElementChild);
 //默认显示第一个图片
 this.setBgImg(this.iconList.firstElementChild.firstElementChild);
 return div;
}
createIcon(){
 //创建图标列表
 let str=`<ul class="iconList clearfix" id="iconList">`;
 this.list.forEach(item=>{
  str+=`<li><img src="${item}"></li>`;
 })
 str+="</ul>";
 return str;
}
clickHandler(e){
 let src=e.target.src;
 //如果点击的不是左右按钮,直接跳出
 if(!/prev/.test(src)&&!/next/.test(src)) return;
 //每一个li的实际宽度,width+border+margin
 let liWidth=54+4+IconList.gap;
 //page为一共有几个整数页
 let page=Math.floor(this.list.length/IconList.num)-1;
 //remainder为最后不够一页的剩余图标数
 let remainder=this.list.length%IconList.num;
 if(/prev/.test(src)){
  //如果点击的是上一页按钮
  if(this.x===0) return;
  //移动到最后一页时
  if(this.position===0&&remainder>0){
   //移动的距离加等于li宽度*剩余图标数
   this.x+=liWidth*remainder;
  }
  else if(this.position<=page){
   this.position--;
   //移动的距离加等于li的宽度*每页显示的图标数(5个)
   this.x+=liWidth*IconList.num;
  }
 }else if(/next/.test(src)){
  //如果点击的是下一页按钮
  if(this.x===-(this.list.length-IconList.num)*liWidth) return;
  if(this.position===page&&remainder>0){
   //移动的距离减等于li宽度*剩余图标数
   this.x-=liWidth*remainder;
  }
  else if(this.position<page){
   this.position++;
   //移动的距离减等于li的宽度*每页显示的图标数(5个)
   this.x-=liWidth*IconList.num;
  }
 }
 //设置图标列表的left值
 this.iconList.style.left=this.x+"px";
}
mouseHandler(e){
 //如果滑过的不是Img标签,直接跳出
 if(e.target.constructor!==HTMLImageElement) return;
 //设置背景图片
 this.setBgImg(e.target);
 //设置当前滑过图标的样式
 this.setIconState(e.target.parentElement);
}
setIconState(target){
 //移除上一个滑过图标的active样式
 if(this.prepIcon) Utils.removeClass(this.prepIcon,"active");
 //将当前滑过的对象赋值给this.prepIcon
 this.prepIcon=target;
 //给当前滑过图标增加active样式
 Utils.addClass(this.prepIcon,"active");
}
setBgImg(target){
 //抛发事件,将当前图片的src传过去
 let src=target.src.replace("_icon","");
 let evt=new Event(IconList.SET_BG_IMG);
 evt.src=src;
 document.dispatchEvent(evt);
}
static setStyles(list){
 //设置样式
 if(IconList.styles) return;
 IconList.styles=true;
 Utils.insertCss(".iconContainer",{
  width:Zoom.small_width+2+"px",
  height: "58px",
  position: "absolute",
  top: Zoom.small_width+2+"px",
  left: "0px",
 })
 Utils.insertCss(".iconContainer>img",{
  width:"22px",
  height:"32px",
  cursor:"pointer",
  position:"absolute",
  top:"13px",
 })
 Utils.insertCss(".prevBtn",{
  left:"8px"
 })
 Utils.insertCss(".nextBtn",{
  right:"8px"
 })
 Utils.insertCss(".iconListCont",{
  width:Zoom.small_width-30*2+"px",
  height:"58px",
  position:"relative",
  left:"30px",
  overflow:"hidden"
 })
 IconList.gap=((Zoom.small_width-30*2)-(54+4)*IconList.num)/IconList.num;
 Utils.insertCss(".iconList",{
  width:(54+4+IconList.gap)*list.length+"px",
  listStyle:"none",
  padding:"0px",
  margin:"0px",
  position:"absolute",
  left:"0px",
  top:"0px",
  transition:"all .3s"
 })
 Utils.insertCss(".iconList li",{
  float:"left",
  width:"54px",
  height:"54px",
  margin:"0px "+IconList.gap/2+"px",
  cursor:"pointer",
  border:"2px solid transparent"
 })
 Utils.insertCss(".iconList li.active",{
  borderColor:"#f00"
 })
 Utils.insertCss(".iconList li>img",{
  width:"54px",
  height:"54px"
 })
 Utils.insertCss(".clearfix::after",{
  content:"\".\"",
  display:"block",
  height:"0px",
  clear:"both",
  overflow:"hidden",
  visibility:"hidden"
 })
}
}

Utils.js文件,是一个工具包:


export default class Utils{
static createE(elem,style,prep){
 elem=document.createElement(elem);
 if(style) for(let prop in style) elem.style[prop]=style[prop];
 if(prep) for(let prop in prep) elem[prop]=prep[prop];
 return elem;
}
static appendTo(elem,parent){
 if (parent.constructor === String) parent = document.querySelector(parent);
 parent.appendChild(elem);
}
static insertBefore(elem,parent){
 if(parent.constructor === String) parent=document.querySelector(parent);
 parent.insertBefore(elem,parent.firstElementChild);
}
static randomNum(min,max){
 return Math.floor(Math.random*(max-min)+min);
}
static randomColor(alpha){
 alpha=alpha||Math.random().toFixed(1);
 if(isNaN(alpha)) alpha=1;
 if(alpha>1) alpha=1;
 if(alpha<0) alpha=0;
 let col="rgba(";
 for(let i=0;i<3;i++){
  col+=Utils.randomNum(0,256)+",";
 }
 col+=alpha+")";
 return col;
}
static insertCss(select,styles){
 if(document.styleSheets.length===0){
  let styleS=Utils.createE("style");
  Utils.appendTo(styleS,document.head);
 }
 let styleSheet=document.styleSheets[document.styleSheets.length-1];
 let str=select+"{";
 for(var prop in styles){
  str+=prop.replace(/[A-Z]/g,function(item){
   return "-"+item.toLocaleLowerCase();
  })+":"+styles[prop]+";";
 }
 str+="}"
 styleSheet.insertRule(str,styleSheet.cssRules.length);
}
static getIdElem(elem,obj){
 if(elem.id) obj[elem.id]=elem;
 if(elem.children.length===0) return obj;
 for(let i=0;i<elem.children.length;i++){
  Utils.getIdElem(elem.children[i],obj);
 }
}
static addClass(elem,className){
 let arr=(elem.className+" "+className).match(/\S+/g);
 arr=arr.filter((item,index)=>arr.indexOf(item,index+1)<0)
 elem.className=arr.join(" ");
}
static removeClass(elem,className){
 if(!elem.className) return;
 let arr=elem.className.match(/\S+/g);
 let arr1=className.match(/\S+/g);
 arr1.forEach(item=>{
  arr=arr.filter(t=>t!==item)
 })
 elem.className=arr.join(" ");
}
static hasClass(elem,className){
 if(!elem.className) return false;
 let arr=elem.className.match(/\S+/g);
 let arr1=className.match(/\S+/g);
 let res;
 arr1.forEach(item=>{
  res= arr.some(it=>it===item)
 })
 return res;
}
static loadImg({list,basePath,callback}){
 if(!list || list.length===0) return;
 if(basePath) list=list.map(item=>basePath+item);
 let img=Utils.createE("img");
 img.data={
  list:list,
  callback:callback,
  resultList:[],
  num:0
 }
 img.addEventListener("load",Utils.loadImgHandler);
 img.src=list[img.data.num];
}
static loadImgHandler(e){
 let data=e.currentTarget.data;
 data.resultList.push(e.currentTarget.cloneNode(false));
 data.num++;
 if(data.num>data.list.length-1){
  e.currentTarget.removeEventListener("load",Utils.loadImgHandler);
  data.callback(data.resultList);
  data=null;
  return;
 }
 e.currentTarget.src=data.list[data.num];
}
}

来源:https://blog.csdn.net/Charissa2017/article/details/104159002

标签:js,放大镜
0
投稿

猜你喜欢

  • 一些让Python代码简洁的实用技巧总结

    2022-02-06 11:03:25
  • Keras设定GPU使用内存大小方式(Tensorflow backend)

    2021-12-20 22:15:12
  • MySQL的一级防范检查列表

    2011-12-14 18:39:22
  • bootstrap multiselect 多选功能实现方法

    2024-05-28 15:38:25
  • 浅析SQL语句中GROUP BY的用法

    2024-01-28 21:36:51
  • golang正则之命名分组方式

    2024-05-22 10:17:10
  • PHP基础之运算符的使用方法

    2023-11-20 17:33:55
  • Python导入数值型Excel数据并生成矩阵操作

    2023-05-15 16:59:24
  • Python运用于数据分析的简单教程

    2023-08-14 07:49:13
  • sqlserver数据库迁移后,孤立账号解决办法

    2011-10-24 20:01:40
  • pyqt5打包成exe可执行文件的方法

    2022-02-09 11:25:57
  • python人工智能tensorflow函数tf.get_collection使用方法

    2023-08-09 14:27:54
  • python基础之while循环语句的使用

    2021-03-16 16:54:20
  • Python之时间和日期使用小结

    2021-01-11 09:33:21
  • 简单代码实现可输入的下拉框功能(select)

    2008-10-20 19:52:00
  • 设计师的幸福

    2009-05-21 11:59:00
  • Python selenium模块实现定位过程解析

    2021-01-10 10:50:27
  • 实例讲解MySQL数据库中文问题的解决方法

    2008-12-31 15:15:00
  • sql语句优化之用EXISTS替代IN、用NOT EXISTS替代NOT IN的语句

    2024-01-22 14:34:06
  • 写出完美CSS代码的5个重要方面

    2009-12-30 16:44:00
  • asp之家 网络编程 m.aspxhome.com