Image enlargement preview is a very common scene and function, general mobile website home page of the rotation banner, product details page of the product picture and other positions will use this function

A common scenario feature like this must have been written by someone already, so there are three common steps to follow in this scenario:

  • Open the refrigeratorStart making
  • searchphoto,preview,carousel,photoSwipeKey words such as
  • Find the library you want,npm installthe

There is nothing wrong with this practice, have a ready-made wheels can be used to, of course, because the project USES the vue, so I’m on the Internet to find a circle amplification preview component library based on vue, the result makes me a little surprised, zoom preview library significantly less than the number of wheel component library, and what is more mentally out is, these little component library, More than half of them are repackaged based on the open source library PhotoSwipe, in addition to the image Gallery, which can be used in production… I don’t think so (or maybe I’m being short-sighted), and this is true not just for vUE libraries, but for other frameworks and even native libraries

While it’s not advisable to duplicate wheels, it doesn’t make sense that there are too few wheels to choose from. PhotoSwipe works well and is versatile enough to handle most situations in a real production environment

But at the same time, it also means that the code volume will be larger, the introduction of redundant code will be more, so, with the idea of simplifying the code and incidentally enrich the zoom preview plug-in family, decided to build their own wheel

Let’s take a look at the final implementation:

Or if you want to try it out for yourself, there’s also a written Demo

I have packaged this function into an NPM package, which can be downloaded, installed and used directly. The size of the code including the style is less than 22KB after compression, and the size of the code after Gzipped is less than 8KB. The source code has been uploaded

Sliding form

The selection of the sliding form is the same as in swiper, without further elaboration

The data processing

Data processing is the same as the first method in swiper without further elaboration:

<VueActivePreview :urlList="urlList" />
Copy the code

Touch events

The touch events of this component are complex and involve the interaction between different touch events, so it is a bit troublesome, but can be solved with clarity and consideration

Single refers to the sliding

The main logic for single-finger swiper is similar to that of swiper, which calculates the distance a finger slides and shifts it by constantly changing the translate value

Double refers to zoom

Support the zoom operation of a single picture, the principle is actually very simple, by calculating the proportion of the distance between the two fingers in the beginning and the sliding process, you can get the scale of the picture

To obtain the distance between two fingers:

getDistance (p1, p2) {
  return Math.sqrt(Math.pow(p2.clientX - p1.clientX, 2) + Math.pow(p2.clientY - p1.clientY, 2))}Copy the code

Get the image scale:

this.scaleValue = this.getDistance(targetTouch1, targetTouch2) / doubleTransferInfo.startDistance
Copy the code

By changing transform: Scale (scaleValue), the image can be scaled in real time

The default value is 50%, 50%, 0, which is the center of the element. Therefore, if you set the scale without setting transform-Origin, the image will also scale properly. But the result of scaling is not necessarily what you want

For example, the center coordinate of the two fingers is (10, 56). According to the normal habit, when zooming in, the whole picture should be zoomed in on this point instead of the central position of the picture

There are two ways to solve this problem

  • Dynamic settingtransform-origin

The easiest way to do this is to simply set the center coordinate between the two fingers to transform-Origin and scale

Therefore, transform-origin needs to be set dynamically

const targetTouch1 = e.touches[0]
const targetTouch2 = e.touches[1]
this.transOriginX = (targetTouch1.clientX + targetTouch2.clientX) / 2 - this.left
this.transOriginY = (targetTouch1.clientY + targetTouch2.clientY) / 2 - this.top
Copy the code
  • Dynamically set the position coordinates of the picture

The transform-origin change actually changes the position state of the image. There is no need to care what the transform-Origin should be, and the default center position of the image is the transform-Origin for every image scaling. Then, during the image scaling process, Dynamically correct the position of the image to offset the transform-Origin effect and maintain visual unity

For example, the default transform-Origin for an image is (100, 100). If the image is magnified twice from this center, the upper left corner of the image will be shifted 100 units to the left of the original image after magnification, but now the starting center of both fingers will be (0, 0) (just an assumption, For ease of calculation), and set this to transform-Rogin, when magnified twice, the upper left corner of the image will shift 0 units to the left from the original state, that is, there will be no shift at all

Therefore, when the zoom center is (0, 0), in order to maintain visual unity without changing transform-Origin, the image must be continuously moved right in the process of image enlargement to ensure that the distance moved in each frame can offset the gap caused by transform-Origin. Until you finally remove 100 units

I choose the first method because the position coordinates of the following images need to be used in other places, and it is easier and convenient to set transform-Origin directly

But I soon found that I was still thinking too simple

Suppose that after the fingers leave the screen, the picture is enlarged by 2 times with (10, 56) as the zoom center, and then the two fingers are placed on the screen again. At this time, the center coordinate of the two fingers is (70, 88). At this time, according to the above mentioned, You need to dynamically change transform-Origin from 10, 56 to (70, 88), but if you do, you’ll notice that the image immediately jumps

This is because before the second two-finger touch screen, the image magnification state was based on transform-Origin (10, 56). Now changing transform-Origin means that the magnification base point of the image is changed, and the image state is bound to change

Should it be the second way? However, there is something wrong with frequently changing the value of left/top, and the calculation method of this method is quite complicated, which may affect the performance

On second thought, it can be solved

The reason why the second zoom will produce a jump is to change the state after the first time, because this state is not fixed, at this time the scale and transform-Origin of the image are dynamically modified, as long as the state can be fixed, fixed to the default value, that is ok?

As for how to fix this state, it’s actually very simple, right

For an image with a size of 100*100, zoom in (10, 10) as transform-origin by 2 times, then the size of the enlarged image is 200*200, and the offset of the upper left corner is (-10, -10) SO after the first zoom, Immediately set the width and height of the image to 200 × 200 and give it a left: -10; Scale and transform-origin can then be restored to their default state, which is equivalent to not using any transform property

So the second time you zoom in, you’re going to start with the current 200 by 200 image instead of the original 100 by 100 image, and you don’t have to worry about the state, because every time you zoom in, it’s going to be a new state

Intuitive example:

transform: scale(2);  =>  width: 200; height: 200;
transform-origin: 10 10;  =>  left: -10; top: -10;
Copy the code

Code examples:

this.left = left
this.top = top
this.currentW = currentW
this.currentH = currentH
this.scaleValue = 1
Copy the code

Zoom during the slide view

After a single image is zoomed, it is allowed to slide the zoomed image in order to allow the user to view every detail of the image more freely

The main logic of this function is relatively simple, by listening to the touch event, calculate the move distance between each frame, dynamic displacement picture position can be

However, in order to get closer to the actual physical interaction and achieve a better user experience, an inertial sliding ability is added, that is, when the user finishes touch in the process of sliding the picture, the picture will continue to slide forward a certain distance

There are two solutions to this scenario

  • CSS animations

At the end of the touch, calculate how far the image should slide to stop at its current speed, and set a transition animation that slowly slows down

  • Js dynamic animation

Specify a speed decrement coefficient. Each frame’s speed decreases from the previous frame until it stops

After comprehensive consideration, the first method may save more performance, but it is not good to simulate the feeling of physical inertia, and the numerical value is not easy to calculate. Compared with the point of saving performance, the cost performance is not high

The second method is easier to control, so the second method is chosen

With the help of the requestAnimationFrame API (already degraded for devices that are not compatible with this API) :

rafHandler = raf((a)= > {
  speedX *= 0.9
  speedY *= 0.9
  // ...
  if (Math.abs(speedX) < 1) speedX = 0
  if (Math.abs(speedY) < 1) speedY = 0
  if(speedX ! = =0|| speedY ! = =0) {
    this.frictionMove(speedX, speedY)
  } else {
    // ...}})Copy the code

conclusion

Is the main body of the component logic is actually clear, no way way too much, but you need to consider too much, and there are three different interaction and judging cases touch events, all the things together is very nerve-racking, less than one 5 of the time to write the subject logic, for the rest of the consumption in the if… Else by the time I finish writing this wheel, I can see why there are so few wheels in this scene, because it really hurts, not the functional logic pain, after all, functional logic is interesting to write, but if… Else’s pain

Source code has been put on Github, code annotation is also more detailed, interested in can refer to the next, if there is any problem, welcome to mention issues