1. The introduction
πβοΈπ What is “elastic left slip to see more”?
As shown in β¬οΈ, add an indicator slider to the end of the horizontal sliding list. When the list slides to the end, the slider stretches elastically. After stretching to a certain extent, the event can be triggered ~ the effect is like βοΈβοΈ
This interaction is quite common in the App, there are many codes to refer to.
When I got this requirement to implement in H5, I looked for references, not to mention plugins, even code blocks
π€¦π»βοΈ maybe the search posture is wrong? Anyway, I couldn’t find the wheel and had to write it myself.
2. Plan
2.1 thinking
The first thing that comes to mind when you look at this elastic slide is whether swiper is possible. How about using the Translate transform to write a virtual scroll? Also do smooth scrolling animation, think a little more complicated. Is it possible to add a bottom stretching effect to the original scrolling?
2.2 Flex Layout horizontal scrolling
Start with a Flex layout to scroll the card list horizontally. At the bottom of the list, we insert a “See More” card. With Flex layout, “See More” automatically aligns height with other cards in the list, which is a benefit of using elastic boxes.
.list{
display: flex;
flex-wrap: nowrap; /* Elastic boxes do not fold */
overflow-x: auto;
}
.item{
/* Internal card style... * /
}
.item-more{
content: "See more";
writing-mode: vertical-lr; /* Line up text */
/ *... * /
}
Copy the code
2.3 The color block hidden on the right side of the list
Now that the list is done, how do I pull and drag? My idea is to position an element on the right side of the list and hide it from the screen. When the list scrolls to the far right and continues to slide left, the entire list translates left to pull out the hidden element.
To do this: listen to the list touch event. When the scrolling distance of the list has reached the maximum, the touch position pageX is recorded in real time, and the translateX of the list container is changed by calculating the finger moving distance to realize the displacement transformation. When touchend determines that the displacement reaches the specified distance, the “release” event can be triggered.
The code in this article is only sample code, variable declarations, syntax and logic have been simplified, you can understand the meaning
.list{
/ *... flex */
position: relative;
transition: all 0.1 s; /* Smooth the transition */
}
.after{
position: absolute;
width: 100%;
right: -100%;
top: 0;
height: 100%;
background: #f5f5f5; /* The same color as "See more" */
}
Copy the code
The effect is as follows:
2.4 Elastic drawing
At this point, the implementation completes the basic function: slide left to the bottom and stretch, releasing the trigger event. In the above code, the finger movement and the list displacement are set one to one, and the pull experience is poor. Next, consider interactive optimization and make an elastic stretch effect: as the finger slides, the pull becomes slower and slower until it locks.
To do this, modify the function correspondence between trans and dragDistance above. This is where the math comes in.
Fooplot online equation
As shown above, fooplot generates a parabolic equation, a top point: in the range 0<x<100, y increases more and more slowly as x increases. (62 and 100 in the function correspond to the maximum distance of list sliding and finger sliding respectively, and the values can be adjusted according to actual needs)
Corresponding code:
const DISTANCE_MAX = 100; // The maximum distance a finger can move
// Insert the calculation before line 18 of the above code
trans = DRAG_MAX - (DISTANCE_MAX / DRAG_MAX ** 2) * (trans - DRAG_MAX) ** 2;
Copy the code
Take a look at the final result β
3. Unexpected pit
It took three hours to complete the function, let out a big sigh,
Unexpectedly delivered to test students, immediately like to mention bugs.
3.1 Pit compatible with iPhone
“π you this have bug on iPhone, exposed white”.
????? I’ve been using Android for a long time, but on the iPhone, scrolling lists come with a stretch-back effect.
π€·π»βοΈ On Android and computer browser emulators, lists can no longer be dragged left to the far right, but on the iPhone, they can continue to be pulled and pulled, resulting in a gap between the “See more” and.after elements, revealing the bottom color of the list. (Because there is no machine is not very convenient to record pictures, understand by yourself)
So what are the features of rolling and stretching that come with the iPhone? Debugging found that when the list is stretched, the scrollDistance values in the above code appear 10, 20, 30, 100… In iPhone, the offsetWidth + scrollLeft of scrollable elements can be greater than the scrollWidth. π³ ashamed, development for so many years, just know this matter…
Instead of judging displacement on the iPhone, place a background element at the bottom of the list and use the system’s own list rebound to expose the background element for a “See more” stretch effect without triggering list deformation assignments.
After code modification:
// 11 lines of code
if(scrollDistance>0){
trans = 0;
draggingOk = scrollDistance>DRAG_LINE; // Use the scroll bar offset directly to determine whether the event can be triggered
} else if(scrollDistance===0) {/ / the same
} else {
/ / the same
}
Copy the code
3.2 pit continues android compatibility
After submitting the changes like a tiger…
“π₯Ί one of the Android test machine, after sliding to the right, how can not drag? !” .
New bug…
After taking the test machine for verification, it turns out that the scrollDistance of Android machine can also be greater than 0. The code went to the wrong path, resulting in no trigger trans change. And get new knowledge…
After debugging, it was found that the scrollbar overflow value of the test machine was 0.6666666px, and the value of -0.3333333px appeared when testing other machines, and the overflow value was less than 1px. Finally, the modified code was as follows:
if(scrollDistance>1) {// ...
} else if(scrollDistance>-1) {// ...
} else {
// ...
}
Copy the code
Note that when writing functions such as bottom loading in the future, it is not possible to judge the bottom state directly by offsetWidth + scrollLeft – scrollWidth === 0
4. Code sharing
I made a plug-in out of the component and made some configuration items to improve usability. Some case effects:
πClick to see the Github address
React+Typescript code is written in React+Typescript. If necessary, you can refer to it directly. If not, you can write your own.
5. Write at the end
The whole implementation process is still relatively rough, there are a lot of places can be optimized, welcome to put forward valuable advice ~
As well as other proposals welcome to exchange ~