Preface Most mini programs will have such a requirement. The page has a long list and needs to request background data when scrolling down to the bottom. The data is rendered continuously. When the data list is long, obvious lag will be found and the page will flash white screen. analyze
Initial Rendering Method/* * @Descripttion: * @version: * @Author: shijuwang * @Date: 2021-07-14 16:40:22 * @LastEditors: shijuwang * @LastEditTime: 2021-07-14 17:10:13 */ //Page Object Page({ data: { list: [], // all data}, //options(Object) onLoad: function (options) { this.index = 0 const arr = [ { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, ] this.setData({ list: arr }) }, onReachBottom: function () { const arr = [ { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, ] let { list } = this.data this.setData({ list: [...list, ...arr] }) }, }); // wxml <view class="container"> <view class="item-list" wx:for="{{list}}"> <text class=""> {{item.idx}} </text> </view> </view> Every time the bottom is touched, the data is requested again, the array is merged and the setData is re-set. When the list is responsible, a white screen will appear. The data of setData becomes larger and larger each time, which increases the communication time and renders too many DOM elements, as shown in the following figure: Initial Optimization1. Change the one-dimensional array to a two-dimensional array
// wxml <view class="container"> <block wx:for="{{list}}" wx:for-index="pageNuma"> <view class="item-list" wx:for="{{item}}"> <text class="">{{item.idx}}</text> </view> </block> </view> // wx.js Page({ data: { list: [], // all data}, //options(Object) onLoad: function (options) { this.index = 0; this.currentIndex = 0; // Current page number pageNuma const arr = [ { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, ] this.setData({ [`list[${this.currentIndex}]`]: arr }) }, onReachBottom: function () { this.currentIndex++; // bottom out +1 const arr = [ { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, ] this.setData({ [`list[${this.currentIndex}]`]: arr }) }, }); In this way, we can see that the entire rendering is rendered in pages based on the screen. If several pageNums are requested, several screens will be rendered, and each list will be rendered in each screen. Further optimization 2 We can render only one screen of the visible area, or render several screens near the visible area, and use views to occupy other areas without specific rendering, thereby reducing DOM rendering and reducing problems caused by too many nodes. To do this, you need to split the following steps:
is.currentIndex = 0; // Current page number pageNum this.pageHeight = []; //Store each screen height this.allList = []; //Get all the data this.systemHeight = 0; //Screen height this.visualIndex = []; //Store visible area pageNum When entering the page, mount the relevant data: // wxml <view wx:for="{{list}}" wx:for-index="pageNum" id="item{{pageNum}}"> <view class="item-list" wx:for="{{item}}"> <text class="">{{item.idx}}</text> </view> </view> onLoad: function (options) { this.index = 0; this.currentIndex = 0; // Current page number pageNum this.pageHeight = []; // Each screen height is stored this.allList = []; // All data obtained this.systemHeight = 0; // Screen height const arr = [ { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ }, { idx: this.index++ } ] this.setData({ [`list[${this.currentIndex}]`]: arr }, () => { this.setPageHeight(); }); this.getSystemInfo(); }, Dynamically set the ID name for the view of each screen to facilitate the acquisition and storage of the height of each screen during rendering. Use the loop pageHeight to set the height for subsequent non-visible area placement. Add each page height and the current scroll distance for comparison to obtain the currently rendered pageNum, and then put the currently selected rendered pageNum -+1, the previous screen and the next screen into the array. The current three page data are displayed, and the data of other screens are used for view placement, and the height is the stored height. // Scroll distance calculation onPageScroll: throttle(function (e) { let pageScrollTop = e[0].scrollTop; let that = this; // Scroll to calculate the current page screen let scrollTop = 0; let currentIndex = this.currentIndex; for (var i = 0; i < this.pageHeight.length; i++) { scrollTop = scrollTop + this.pageHeight[i]; if (scrollTop > pageScrollTop + this.systemHeight - 50) { this.currentIndex = i; this.visualIndex = [i - 1, i, i + 1]; that.setData({ visualIndex: this.visualIndex }) break; } } },200) Real-time monitoring of scrolling and throttling optimization const throttle = (fn, interval) => { var enterTime = 0; //Trigger time var gapTime = interval || 300; //Interval time, if interval is not passed, the default is 300ms return function () { var that = this; var backTime = new Date(); //The time when the function returns for the first time is triggered if (backTime - enterTime > gapTime) { fn.call(that, arguments); enterTime = backTime; //Assign the time of the first trigger to save the time of the second trigger} }; } Get the visible area array, and then determine whether the current pageNum is in the visualIndex. If it is, render it. If not, the view takes up space and the height is obtained and stored. <wxs module='filter'> var includesList = function(list,currentIndex){ if(list){ return list.indexOf(currentIndex) > -1 } } module.exports.includesList = includesList; </wxs> <view class="container"> <view wx:for="{{list}}" wx:for-index="pageNum" id="item{{pageNum}}" wx:key="pageNum"> <block wx:if="{{filter.includesList(visualIndex,pageNum)}}"> <view class="item-list" wx:for="{{item}}"> <text class="">{{item.idx}}</text> </view> </block> <block wx:else> <view class="item-visible" style="height:{{pageHeight[pageNum]}}px"></view> </block> </view> </view> If the pageNum of the array in the visible area is not in the current array, the height placeholder is set. The rendering effect is as follows: This method does the job! Method 2 Using WeChat Mini Program API //View monitor observePage: function (pageNum) { const that = this; const observerView = wx.createIntersectionObserver(this).relativeToViewport({ top: 0, bottom: 0}); observerView.observe(`#item${pageNum}`, (res) => { console.log(res,'res'); if (res.intersectionRatio > 0) { that.setData({ visualIndex: [pageNum - 1, pageNum, pageNum + 1] }) } }) } Use oberserve to monitor whether the current page is in the visible area. If so, put the current pageNum -+1, and put pageNum into the visible area array visualIndex. Use wxs to determine whether the current pageNum exists in the visible area array. If so, render it. If not, use view to occupy the place. Solution 1: Scrolling calculation >>>Code snippet: (scrolling virtual list) Solution 2 IntersectionObserver api >>> Code snippet: intersectionObserver This is the end of this article about the implementation example of the WeChat mini-program virtual list. For more relevant mini-program virtual list content, please search for previous articles on 123WORDPRESS.COM or continue to browse the following related articles. I hope you will support 123WORDPRESS.COM in the future! You may also be interested in:
|
<<: How to use port 80 in Tomcat under Linux system
>>: In-depth explanation of the impact of NULL on indexes in MySQL
This article shares the specific code for JavaScr...
Table of contents 1. Encapsulate complex page dat...
Table of contents 1- Error details 2-Single Solut...
1. Problem introduction Assume a scenario where a...
Vue2+elementui's hover prompts are divided in...
Table of contents 8. CSS3 click button circular p...
Table of contents 1. Pre-analysis 1. Variable pre...
The process of completely uninstalling the MySQL ...
1. Download the MySQL installation package First ...
This article mainly introduces three methods of i...
Table of contents Preface The principle of browse...
method: Take less in the actual project as an exa...
Table of contents Concept Introduction Logical ru...
MySQL CURDATE Function Introduction If used in a ...
RULES can be used to control the style of the int...