Author: Yang Ye

This article is produced by a member of YFE, please respect the original, please contact the public account (ID: yuewen_YFE) for authorization to reprint, and indicate the author, source and link.

background

In the dialogue fiction project I helped develop, we found that creative activities were very helpful in pulling up and transforming data. Through investigation and research, this dialogue type novel user groups most of the products is younger after 90-95, so the final conclusion is hope to the industry popular young APP interaction forms — “slide card” to do a redesign, pushed the book activities can also hope that this page is combined with the product itself as a permanent function page, Let’s take a look at the final implementation:

Is it smooth? Next, I will talk about what I experienced in development according to the thinking and process of development at that time.

reference

After the designer delivered the design draft with great care, she specially used Flinto ⤵️ to make an interactive prototype to help me achieve the desired effect of the planning.

After seeing this thoughtful interactive draft, my first thought is to refer to the exploration page in Jike App and the interaction form of dating App Tantan. Their interaction effects are as follows:

The effect is very similar, right? 😏 But different from my demand this time: our page is embedded in the STARTING point reading App H5, and the above two effects are realized by the native App development, so I have a little worry about “whether it can be highly restored” and “how to ensure good performance” 🤔.

try

Before capturing real data and developing complex logic, I sketched out the implementation ideas:

  • The initial state is 3 cards stacked together, to have a 3D sense, when dragging can reveal the back two
  • When you drag the first card, you need to follow the sliding direction of your finger. When you let go of your finger after a certain distance, the card will fly out, and the card behind you will automatically advance one. Three cards should always be visible on the page.

According to the above thinking, since the 3D three-dimensional feeling and propulsion dynamic effect are needed, z-Index alone cannot meet the requirements, so I choose to use translateZ to complete the propulsion effect of this stacked card, because it can better display the 3D space depth of field. In this way, motion effects such as advancing cards and automatically flying out of thrown cards can be left entirely to the CSS3 animation transition.

Style code (main structural attributes)

.card_container { position: relative; Width: 6.86 rem; Height: 8.96 rem; perspective: 1000px; perspective-origin: 50% 150%; -webkit-perspective: 1000px; -webkit-perspective-origin: 50% 150%; } .card { transform-style: preserve-3d; width: 100%; height: 100%; position: absolute; opacity: 0; }Copy the code

Stacked cards need to have a parent container that gives all stacked cards a 3D perspective.

HTML and binding methods

<div class="card_container">
  <div
    v-for="(item,index) in dataArr"
    :key="item.id"
    ondragstart="return false"
    class="card"
    :style="[cardTransform(index),indexTransform(index)]"
    @touchstart.stop.capture="touchStart($event,index)"
    @touchmove.stop.capture="touchMove($event)"
    @touchend.stop.capture="touchEnd($event,index)"
    @mousedown.stop.capture="touchStart($event)"
    @mousemove.stop.capture="touchMove($event)"
    @mouseup.stop.capture="touchEnd"
    @transitionend="onTransitionEnd(index)"
  >
</div>
Copy the code

We also need some key variables to record some properties that may change in real time:

Position: {start: {x: 0, y: 0}, end: {x: 0, y: 0} {x: 0, y: 0}, direction: 1, // Sliding direction: 1, left is -1, right is 1false}, // record each drop direction directionArr: [], // display the stack number of pictures visible: 3, // viewport width: 0, // slide threshold slideWidth: 70, // Automatic offset when the threshold is exceeded offsetWidth: 120,Copy the code

Attach two initialization methods to the style. CardTransform is used to initialize the style of each card, and indexTransform is used to initialize the style of the first card.

CardTransform (index) {letStyle = {} //let offset = 0
    if (this.directionArr[index] === 1) {
      offset = 800
    } else if (this.directionArr[index] === -1) {
      offset = -800
    }
    
    style['z-index'] = this.currentIndex - index + this.visible 
    style['transform'] = ` translate3d (0, 0,${(this.currentIndex - index) * 60}Px) '// Make the hidden cards shrink and stack together transparently. Once one flies away, the next card automatically transitions forwardif (index - this.currentIndex < 0) {
    style['opacity'] = 0
    style['transform'] = `translate3d(${this.position.end.x + offset}px,${this.position.end.y}px,${(this.currentIndex - index) * 60}px) rotate(${this.position.direction * -65}Deg) '} // Add transition animation only for non-gesture sliding statesif(! this.position.swipping) { style['transitionTimingFunction'] = 'ease'
    style['transitionDuration'] = 300 + 'ms'
  }
  returnStyle}, // First card style indexTransform (index) {let style = {}
  if (index === this.currentIndex) {
    style['transform'] = `translate3d(${this.displacement.x}px,${this.displacement.y}px,${(this.currentIndex - index) * 60}px) rotate(${this.displacement.x / this.winWidth * -65}Deg) '} // Add transition animation only for non-gesture sliding statesif(! this.position.swipping) { style['transitionTimingFunction'] = 'ease'
    style['transitionDuration'] = 300 + 'ms'
  }

  return style
 }
Copy the code

After that, the touch event of the drag card is as simple as writing the drag DIV before, and the method of returning the details of the previous slide and the background transition will not do too much code demonstration here.

So far, four books of mock data have been used, and everything is going well and the animation is very smooth:

App Webview crash 😱

I then requested real data and made a series of optimizations, such as:

  1. All-model adapter card screen center.
  2. Record user operations, drag and drop the direction of throwing into localStorage (when the user opens it again, the first card he sees is still the one he left before, the experience is more like in the App)
  3. The optimization reduces the request to load 2 images on the first page and the next image on each card that flies away.

After optimization, everything looked so smooth in PC Chrome mobile mode, I thought there would be no problem, and finally released to the test environment and opened it with the App scanning code, I saw this scene:

I tried to open the App with the browser on the mobile end, but the crash didn’t happen, but the operation was not smooth. I tried it again on the PC, but I still couldn’t feel any lag. I think it may be because the hardware of the mobile phone is not as good as that of PC. The crash could be due to 3D rendering or performance issues. With that in mind, I’m going to do a statistical comparison to see what the key factors were that led to the crash.

The performance comparison

First of all, I recorded the page with Chrome’s Performance for 7 seconds, during which I operated the card crazily. Finally, the Performance graph is as follows:

It doesn’t prove anything serious except for a small caveat: Handler took. I decided to monitor Rendering performance again, and I pulled up the Rendering panel from Chrome’s more tools

Once all the boxes are checked, the cause of the problem is suddenly revealed!

OMG 😱, the frame rate is only 18 FPS and all the original cards are superimposed and rendered. I immediately realized something was wrong in the development: the hidden cards had transparency set to 0, but just because they weren’t visible didn’t mean they wouldn’t be rendered, and the hidden cards were being rendered in real time every time the card flew out of the animation, causing a significant performance loss.

Opacity and opacity also cause backflow, while any reflow must cause repaint. Only display: None can be used for lightning protection. Because it’s completely out of document flow, I’ve been optimizing for page reversion and motion efficiency since I developed this requirement, and I’ve forgotten this important point.

To optimize the

Opacity is also required to beautify the transition of animation-efficiency, while display alone would be very stiff. Class =”{display:item.display}” class=”{display:item.display}” class=”{display:item.display}” Then set the CSS card style to display: None

<div class="card_container">
  <div
    v-for="(item,index) in dataArr"
    :key="item.id"
    ondragstart="return false"
    class="card"
    :style="[cardTransform(index),indexTransform(index)]"
    @touchstart.stop.capture="touchStart($event,index)"
    @touchmove.stop.capture="touchMove($event)"
    @touchend.stop.capture="touchEnd($event,index)"
    @mousedown.stop.capture="touchStart($event)"
    @mousemove.stop.capture="touchMove($event)"
    @mouseup.stop.capture="touchEnd"
    @transitionend="onTransitionEnd(index)"
    :class="{display:item.display}"
  >
</div>
Copy the code

Set it to true when it needs to be displayed, and the style changes to block.

.card.display {
  display: block;
  opacity: 1;
}
Copy the code

For example, I have a card move method called moveNext in touchEnd.

touchEnd () {
  this.position.swipping = false
  this.position.end['x'] = this.displacement.x
  this.position.end['y'] = this.displacement. Y // If the slider distance exceeds the set value, it will automatically fly outif(this.displacement.x > this.slideWidth) {this.movenext (1) // to the right}else(this.displacement. X < -this.slideWidth) {this.movenext (-1) // to the left} this.$nextTick(() => {
    this.displacement.x = 0
    this.displacement.y = 0
    this.isDrag = false})}Copy the code

We can then operate on index in moveNext. In moveNext, you need to add a display to the first card that is currently displayed and to the next stack, and hide the cards that have disappeared, so that the loop works seamlessly. In addition, because the data is uncertain, in order to avoid some extreme situations (for example, there are no more cards after the first card goes forward or the last few countdown cards, so it is necessary to do details fault tolerance processing).

MoveNext (direction) {this.position.direction = direction try {this.dataarr [this.currentIndex + 3].display =true} catch (e) {if (this.currentIndex > 0) {
    try {
      this.dataArr[this.currentIndex - 1].display = false} catch (e) {}} this.currentIndex++ // Move the next card forward each time, or vice versa! direction ? this.position.end['x'] -= this.offsetWidth : this.position.end['x'] += this.offsetWidth
  this.position.end['y'] += this.offsetWidth / 2
 }
Copy the code

After some adjustments and optimization, I retuned the Rendering panel to see the results:

As expected, the normal 60 FPS was achieved, no matter what the operation was, only 3 cards were still visible (rendered), performance was greatly improved, and access to the App was returned without any crashes.

Scan code experience (better viewing effect with Starting APP)

conclusion

After the crash caused by App WebView, I have learned some experience and summary from it, which I hope will be helpful to you reading this article 😊.

  1. When simulating native App animations on the Web, especially on mobile, extreme caution is required when using higher-order attributes to dynamically change elements in real time.
  2. “Visual perception” is not accurate, nor can it be used as a basis for measurement, everything has to be proven against the performance data in the development tool.
  3. Reflow and Repaint don’t cause performance problems on the PC side, as long as you don’t write code with the mindset of “go easy”. However, mobile rendering capabilities are not as good as those on the PC side. CSS triggering reflow and Repaint can be a performance killer on mobile. Therefore, it is important to anticipate the one-off performance of your solution before completing requirements and performance. Reflow and Repaint can also be a starting point when considering page performance.

For more sharing, please pay attention to the front end team official account of China Literature Group: