Detailed explanation of CSS3+JS perfect implementation of magnifying glass mode

Detailed explanation of CSS3+JS perfect implementation of magnifying glass mode

About a year ago, I wrote an article: Analysis of the principles of several ways to imitate the magnifying glass effect. At that time, I felt that my skills were good enough and I was proud of it, so I gave it such a boastful title. In fact, it only introduced two animation methods in CSS: transform and animation. Of course, the effect achieved was also very simple... I am ashamed.

Although with the advancement of technology, we gradually realized a canvas magnifying glass and another "Taobao-style" model using pure JS, it was still unsatisfactory: because the implementation was too complicated and relied on most of the JS logic. The movement and display effects all relied on JS, and the offset was calculated through JS and then the style was rendered.

But the emergence of CSS3 custom variables has given me hope!

First look at the effect:

CSS3+JS implementation of magnifying glass

Its core implementation:

  • CSS functions, such as: calc() - dynamic calculation; var() - use custom variables
  • CSS pseudo-elements: ::before/after — easy to control, independent of the document flow, and easy to render
  • JS API: offsetX/offsetY : Position relative to the upper left corner of the parent node area

In fact, what we want to achieve specifically is: when the mouse moves in, a small circle is displayed (following the mouse), and wherever this small circle goes, the image area there will be enlarged by the corresponding multiple and displayed in the circle.

Why use the offset API?
In fact, according to the above description, we need to obtain the left offset and top offset of the mouse in real time, and these two offsets are relative to the parent node. The display position of the magnifying glass content relative to the parent node can be calculated by combining the left offset and the top offset with calc() .
It is not difficult to find that in the mouse event object, js provides us with the following API:

  • screenX/screenY : Position relative to the upper left corner of the screen area. If scrolling occurs, position relative to this area.
  • pageX/pageY : Position relative to the upper left corner of the web page area
  • clientX/clientY : Position relative to the upper left corner of the browser's visible area
  • offsetX/offsetY : Position relative to the upper left corner of the parent node area. If there is no parent node, position relative to <html> or <body>

But by comparison, the only one that meets the requirements is offset "relative to the parent element".

<div class="bruce">
    <div class="magnifier"></div>
</div>
let magnifier = document.querySelector(".magnifier");
magnifier.addEventListener("mousemove",e=>{
	//Control the movement of the small circle of the "mirror"});

The magnifying glass displays the content by magnifying the original image N times, and proportionally intercepting a certain area to display the content through the above offset.

First define the relevant css variables. We set the magnification to 2.5 times, so the width and height of the enlarged image will also be 2.5 times the original width and height. Declare two variables, divided into --x and --y :

:root{
    --ratio: 2.5;
    --box-w: 600px;
    --box-h: 400px;
    --outbox-w: calc(var(--box-w) * var(--ratio));
    --outbox-h: calc(var(--box-h) * var(--ratio));
}
.bruce{
    margin-top: 50px;
}
.magnifier{
    --x:0;
    --y:0;
    overflow: hidden;
    position: relative;
    width: var(--box-w);
    height: var(--box-h);
    background: url("img/nan.png") no-repeat center/100% 100%;
    cursor: grabbing;
}

The picture is displayed as a background image, which makes it easy to control the size.

Obviously, in this scenario, there is no need to insert a child node as a container for the magnifying glass. Just use ::before !

The magnifying glass is 100px wide and high when in use, and 0px wide and high when not in use. The position of the magnifying glass as the mouse moves is arranged through absolute positioning, that is, declaring left and top, and then filling the position of the magnifying glass by declaring transform:translate(-50%,-50%) so that the center of the magnifying glass is consistent with the position of the mouse cursor. Since left and top are declared to locate the position of the magnifying glass, will-change can also be declared to improve performance issues caused by changes in left and top!
Another benefit of using CSS to solve these problems is that with the help of pseudo-elements/pseudo-classes, we can use CSS to solve some of the more detailed things instead of relying on "heavy" JavaScript. For example: mouse enters the style hover:

.magnifier::before{
    --size: 0;
    position: absolute;
    left: var(--x);
    top: var(--y);
    border-radius: 100%;
    width: var(--size);
    height: var(--size);
    box-shadow: 1px 1px 3px rgba(0,0,0,.5);
    content: "";
    will-change:left,top;
    transform: translate(-50%,-50%);
}
.magnifier:hover::before{
    --size: 100px;
}

Next, use background to implement (display) the magnifying glass content. According to the magnification of 2.5 times, you can declare size: --outbox-w --outbox-h , and move the background through position-x and position-y. Finally, you can write it as background:#333 url(背景圖片) no-repeat var(--scale-x) var(--scale-y)/var(--outbox-w) var(--outbox-h) .
Among them, --scale-x and --scale-y correspond to position-x and position-y (ie background-position ), which are used to change the background position as the mouse moves.

--scale-x: calc(var(--size) / var(--ratio) - var(--ratio) * var(--x));
--scale-y: calc(var(--size) / var(--ratio) - var(--ratio) * var(--y));

Then the "position coordinates" of the mirror in the mousemove function above can be written like this:

e.target.style.setProperty("--x",`${e.offsetX}px`);
e.target.style.setProperty("--y",`${e.offsetY}px`);

so eazy~

The final CSS content is as follows:

:root{
    --ratio: 2.5;
    --box-w: 600px;
    --box-h: 400px;
    --outbox-w: calc(var(--box-w) * var(--ratio));
    --outbox-h: calc(var(--box-h) * var(--ratio));
}
.bruce{
    margin-top: 50px;
}
.magnifier{
    --x:0;
    --y:0;
    overflow: hidden;
    position: relative;
    width: var(--box-w);
    height: var(--box-h);
    background: url("img/nan.png") no-repeat center/100% 100%;
    cursor: grabbing;
}
.magnifier::before{
    --size: 0;
    --scale-x: calc(var(--size) / var(--ratio) - var(--ratio) * var(--x));
    --scale-y: calc(var(--size) / var(--ratio) - var(--ratio) * var(--y));
    position: absolute;
    left: var(--x);
    top: var(--y);
    border-radius: 100%;
    width: var(--size);
    height: var(--size);
    background: #333 url("img/nan.png") no-repeat var(--scale-x) var(--scale-y)/var(--outbox-w) var(--outbox-h);
    box-shadow: 1px 1px 3px rgba(0,0,0,.5);
    content: "";
    will-change:left,top;
    transform: translate(-50%,-50%);
}
.magnifier:hover::before{
    --size: 100px;
}

If you want to use an image that is twice the size in ::before , replace --outbox-w and --outbox-h with the original --box-w and --box-h in background and make appropriate adjustments.

Pay attention to the content in your magnifying glass, it shows that it is not just a simple enlargement of the image, so there is the var(--size) / var(--ratio) code;
Regarding modifying CSS3 custom variables in CSS: I still believe that it can only be modified and displayed successfully within the "same level and same belonging" scope.

This is the end of this article about how to perfectly implement the magnifying glass mode with CSS3+JS. For more relevant CSS3+JS magnifying glass content, please search for previous articles on 123WORDPRESS.COM or continue to browse the related articles below. I hope you will support 123WORDPRESS.COM in the future!

<<:  Count the list tags in HTML

>>:  How to disable web page styles using Firefox's web developer

Recommend

Native JS to achieve cool paging effect

This article uses an example to share with you a ...

A detailed explanation of the subtle differences between Readonly and Disabled

Readonly and Disabled both prevent users from chan...

Bundling non-JavaScript static resources details

Table of contents 1. Custom import in packaging t...

How to monitor Tomcat using LambdaProbe

Introduction: Lambda Probe (formerly known as Tom...

Detailed explanation of the use of Refs in React's three major attributes

Table of contents Class Component Functional Comp...

How to set and get the number of Mysql connections

Get the number of connections --- Get the maximum...

Ubuntu 20.04 sets a static IP address (including different versions)

Because Ubuntu 20.04 manages the network through ...

A brief analysis of how to use border and display attributes in CSS

Introduction to border properties border property...

Detailed explanation of filters and directives in Vue

Table of contents vue custom directive Global Dir...

How to automatically execute SQL statements when MySQL in Docker starts

When creating a MySQL container with Docker, some...

Tomcat parses XML and creates objects through reflection

The following example code introduces the princip...

Detailed introduction to CSS priority knowledge

Before talking about CSS priority, we need to und...

Detailed explanation of the use of bus in Vue

Vue bus mechanism (bus) In addition to using vuex...

IE8 Beta 1 has two areas that require your attention

<br />Related articles: Web skills: Multiple...