Practice 2: DPS (Draw-Page-Structure)
Premise Background:
Vue-skeleton-webpack-plugin generates the corresponding skeleton screen through pre-rendering manually written code, and renders the written Vue skeleton screen components during construction through vueSSR and webPack. The DOM nodes and related styles generated by pre-rendering are inserted into the final output HTML, and then the DOM nodes required by the skeleton screen are generated by pre-rendering, but this scheme is not flexible and controllable. Therefore, the scheme of skeleton screen injection (above) of Vue2. X project is optimized — automatic generation of webpage skeleton screen (DPS).
DPS procedure
1. Install DPS
npm i draw-page-structure -D
Copy the code
2. Add the page that needs to generate skeleton screen in a specific page:
mounted() {
// Note: after generating the skeleton screen, paste the skeleton screen code into the vue file, the following code needs to comment out, otherwise generate HTML insert page every time
setTimeout(() = > {
createSkeletonHTML({
// This code is actually invalid
background: 'red'.animation: 'opacity 1s linear infinite; '
}).then((skeletonHTML) = > {
// This output is the skeleton screen node, you can customize the skeleton screen node content
console.log(skeletonHTML);
}).catch((e) = > {
console.error(e);
});
}, 5000);
},
Copy the code
3,npm run dev
After running in the browser, you can output the skeleton screen node of the current page on the console, copy and add it to the application page, for example:
src/commercialActivity/index.vue
<template lang="html">
<div class="commercialActivity">
<! Transition -->
<transition name="skeleton">
<div v-if="showSkeleton">
<! -- Skeleton screen node -->
<div class="_ __" style="height:100%; z-index:990; background:#fff"></div><div class="_" style="Height: 2.099%; Top: 3.748%; Left: 6.667%; Width: 73.667%."></div><div class="_" style="Height: 2.099%; Top: 3.748%; Left: 81.067%; Width: 12.267%."></div><div class="_" style="Height: 1.799%; Top: 8.096%; Left: 6.667%; Width: 34.492%."></div><div class="_" style="Height: 1.799%; Top: 8.096%; Left: 74.133%; Width: 19.200%."></div><div class="_" style="Height: 4.498%; Top: 14.768%; Left: 9.333%; Width: 7.467%."></div><div class="_" style="Height: 2.249%; Top: 15.967%; Left: 26.975%; Width: 40.000%."></div><div class="_" style="Height: 2.249%; Top: 15.892%; Left: 69.333%; Width: 4.000%; border-radius:50%"></div><div class="_" style="Height: 2.099%; Top: 15.967%; Left: 76.000%; Width: 16.000%."></div><div class="_" style="Height: 2.249%; Top: 27.361%; Left: 1.867%; Width: 3.950%."></div><div class="_" style="Height: 10.195%; Top: 23.388%; Left: 8.483%; Width: 17.908%."></div><div class="_" style="Height: 2.249%; Top: 23.763%; Left: 29.058%; Width: 66.938%."></div><div class="_" style="Height: 2.399%; Top: 26.912%; Left: 30.125%; Width: 6.400%; border-radius:2px"></div><div class="_" style="Height: 1.799%; Top: 27.211%; Left: 74.246%; Width: 21.750%."></div><div class="_" style="Height: 2.249%; Top: 30.960%; Left: 29.058%; Width: 66.938%."></div><div class="_" style="Height: 2.399%; Top: 37.031%; Left: 40.858%; Width: 12.800%."></div><div class="_" style="Height: 1.150%; Top: 37.830%; Left: 55.942%; Width: 3.200%."></div><div class="_" style="Height: 2.849%; Top: 42.429%; Left: 35.067%; Width: 29.867%."></div><div class="_" style="Height: 2.249%; Top: 95.610%; Left: 1.867%; Width: 4.000%."></div><div class="_" style="Height: 6.447%; Top: 93.403%; Left: 6.975%; Width: 6.400%."></div><div class="_" style="Height: 6.447%; Top: 93.403%; Left: 73.333%; Width: 26.667%."></div>
</div>
</transition>.</div>
</template>
<style>. _ {position:fixed
z-index:999; <! -- Because in the createSkeletonHTML method,background: 'red'< span style = "box-sizing: border-box; color: RGB (51, 51, 51); line-height: 22px; font-size: 14px! Important; word-break: inherit! Important;"#efefef; Payable}. {top:0%;
left:0%;
width:100%;
}
</style>
Copy the code
app.styl
// Skeleton transition
.skeleton-enter-active
transition opacity .6s
.skeleton-enter
opacity 0
.skeleton-leave-active
transition opacity .6s
.skeleton-leave-to
opacity 0
Copy the code
The skeleton is displayed before the page is rendered, i.e. the interface is not returned, and the interface returns the skeleton hidden. Note the code that generates the skeleton screen in Mounted. Otherwise, a skeleton screen node is automatically generated after each page is run. Multiple skeleton screens overlap each other.
src/commercialActivity/commercialActivity.js
data() {
return {
showSkeleton: true.// Whether to display the skeleton}}methods: {
// Scroll down to load
infinite(val) {
return _.debounce(() = > {
// Where the interface is requested
this.getProductInfo({
activity_id: this.activityId,
page: this.listQery.page,
ser: val ? val.search : ' '
}).then(() = > {
// Interface returns skeleton hide
this.showSkeleton = false;
if (!this.errorType.type) {
this.$refs.infiniteLoading.stateChanger.loaded();
} else {
this.$refs.infiniteLoading.stateChanger.complete();
}
if (this.hasMore) {
this.distance = -Infinity;
} else {
this.$refs.infiniteLoading.stateChanger.complete();
}
}).catch((err) = > {
// Interface returns skeleton hide
this.showSkeleton = false;
console.log(err);
});
if (this.isAllChecked) {
this.isAllChecked = true; }},500); }},Copy the code
4, DPS is to generate the skeleton of the currently visible page. The rules of DPS design are as follows:
- Only the DOM nodes visible in the visible area are traversed, including: non-hidden elements, elements larger than 0, non-transparent elements, elements whose content is not a space, elements located in the visible area of the browsing window, etc.
- Generate color blocks for (background) images, text, form items, audio and video, Canvas, custom feature blocks and other areas;
- You can use getBoundingClientRect to obtain the absolute values of the node width, height, and distance from the viewport, and calculate the percentage corresponding to the width and height of the current device as the unit of the color block to adapt to different devices
5, Because the DPS generated page skeleton is run in the page node, rather than the first screen skeleton in index.html, the interaction is seen as seeing ‘.White screen -> Page skeleton -> Normal page
‘, the actual DPS is simply code that generates the skeleton of a page. So in the front screenindex.html
Page to joinloading
(For details, it depends on whether the project requires first-screen loading) :
<body>
<div id="app">
<! -- The style is written here so that when the page comes in, everything in the app will be replaced -->
<style>
#app {
height: 100%;
margin: 0px;
padding: 0px;
}
#loader-wrapper {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
}
#loader {
display: block;
position: relative;
left: 50%;
top: 50%;
width: 150px;
height: 150px;
margin: -75px 0 0 -75px;
border-radius: 50%;
border: 3px solid transparent;
border-top-color: red;
-webkit-animation: spin 2s linear infinite;
-ms-animation: spin 2s linear infinite;
-moz-animation: spin 2s linear infinite;
-o-animation: spin 2s linear infinite;
animation: spin 2s linear infinite;
}
#loader:before {
content: "";
position: absolute;
top: 5px;
left: 5px;
right: 5px;
bottom: 5px;
border-radius: 50%;
border: 3px solid transparent;
border-top-color: red;
-webkit-animation: spin 3s linear infinite;
-moz-animation: spin 3s linear infinite;
-o-animation: spin 3s linear infinite;
-ms-animation: spin 3s linear infinite;
animation: spin 3s linear infinite;
}
#loader:after {
content: "";
position: absolute;
top: 15px;
left: 15px;
right: 15px;
bottom: 15px;
border-radius: 50%;
border: 3px solid transparent;
border-top-color: red;
-moz-animation: spin 1.5 s linear infinite;
-o-animation: spin 1.5 s linear infinite;
-ms-animation: spin 1.5 s linear infinite;
-webkit-animation: spin 1.5 s linear infinite;
animation: spin 1.5 s linear infinite;
}
#loader-wrapper .fengche-logo {
display: block;
position: absolute;
left: 50%;
top: 50%;
width: 60px;
height: 60px;
margin: -30px 0 0 -30px;
}
#loader-wrapper .fengche-logo img {
width: 100%;
}
@-webkit-keyframes spin {
0% {
-webkit-transform: rotate(0deg);
-ms-transform: rotate(0deg);
transform: rotate(0deg);
}
100% {
-webkit-transform: rotate(360deg);
-ms-transform: rotate(360deg);
transform: rotate(360deg); }}@keyframes spin {
0% {
-webkit-transform: rotate(0deg);
-ms-transform: rotate(0deg);
transform: rotate(0deg);
}
100% {
-webkit-transform: rotate(360deg);
-ms-transform: rotate(360deg);
transform: rotate(360deg); }}</style>
<div id="loader-wrapper">
<div id="loader"></div>
<div class="fengche-logo">
<img src="" />
</div>
</div>
</div>
<! -- built files will be auto injected -->
</body>
Copy the code
After running, the page interaction is as follows: First screen loading-> Page skeleton screen -> Normal UI interaction after data request returns.
6, when used in the projectHand wash lib - flexible
orPx is processed by REM
And use someUI framework
, e.g.vant
, then using DPS, directly generated % skeleton screen, is not applicable, you need to customize the skeleton screen,Let's do % to rem
, so that the skeleton screen UI interaction of the page is normal
mounted() {
setTimeout(() = > {
/* This is a custom skeleton screen, processing % to REM, project root font size =37.5px */
createSkeletonHTML({}).then(skeletonHTML= > {
// It is better to customize a larger height ratio, compatible with a variety of longer screens, such as 375 * 1000 (iphoneX-375 * 812)
const skeletonWidth = document.body.clientWidth // Screen width
const skeletonHeight = document.body.clientHeight // Screen height
const skeletonTop = 0 // Width :375px, not including header 44px
let cacheHtml = JSON.parse(JSON.stringify(skeletonHTML))
cacheHtml = cacheHtml.replace(/<style>\S+<\/style>/.' ') // Remove the style and extract it to public
// Convert the percentage to rem: height(px) = val * clientHeight/100
cacheHtml = cacheHtml.replace(/height:\d*\.*\d*%/g.(val) = > {
return `height:${(skeletonHeight * val.slice(7, -1) / 3750).toFixed(6)}rem`
})
cacheHtml = cacheHtml.replace(/top:\d*\.*\d*%/g.(val) = > {
return `top:${(skeletonHeight * val.slice(4, -1) / 3750).toFixed(6)}rem`
})
cacheHtml = cacheHtml.replace(/left:\d*\.*\d*%/g.(val) = > {
return `left:${(skeletonWidth * val.slice(5, -1) / 3750).toFixed(6)}rem`
})
cacheHtml = cacheHtml.replace(/width:\d*\.*\d*%/g.(val) = > {
return `width:${(skeletonWidth * val.slice(6, -1) / 3750).toFixed(6)}rem`
})
console.log('First screen skeleton :', cacheHtml)
// skeleton screen capture, remove the header44px(375px below the screen width)+ custom skeletonTop
const skeletonCutTop = (skeletonTop + 44) / 37.5
cacheHtml = cacheHtml.replace(/<div class="_" style="\S+<\/div>/g.(val) = > {
let isCut = true
val = val.replace(/top:\d*\.*\d*rem/g.(topRem) = > {
if (topRem.slice(4, -3) < skeletonCutTop) {
isCut = false
return topRem
}
return `top:${(topRem.slice(4, -3) - (44 / 37.5)).toFixed(6)}rem`
})
return isCut ? val : ' '
})
// Frame screen background capture + scroll down
cacheHtml = cacheHtml.replace(/<div class="_ __" style="\S+<\/div>/g.(val) = > {
return `<div class="_ __" style="height:${((skeletonHeight - skeletonTop) / 37.5) toFixed (6)} rem, top:${(skeletonTop / 37.5).toFixed(6)}rem; z-index:990; background:#fff"></div>`
})
// Handle header hiding in wechat environment
cacheHtml = `<div class="skeleton-box"><div :class="[$store.getters.isWx||$store.getters.isApp?'':'not-wx','skeleton-content']">${cacheHtml}</div></div>`
console.log('Page skeleton :', cacheHtml)
}).catch(e= > {
console.error(e)
})
}, 8000)}Copy the code
Note 1:
Lib-flexible solution limit font size maximum 54px; Max-width :1280px; Word-wrap: break-word! Important; word-wrap: break-word! Important;
function refreshRem(){
var width = docEl.getBoundingClientRect().width;
if (width / dpr > 1280) {
width = 1280 * dpr;
}
var rem = width / 10;
docEl.style.fontSize = rem + 'px';
flexible.rem = win.rem = rem;
}
Copy the code
Note 2:
The vant component at the top, like a search box, is not covered by a skeleton
const skeletonTop = 48 // Where the skeleton screen should be cut from (width:375px)
Copy the code
Note 3:
The vant component also needs to go to rem, postcss.config.js instead:
module.exports = ({ file }) = > {
let remUnit
if (file && file.dirname && file.dirname.indexOf('vant') > -1) {
remUnit = 37.5
} else {
remUnit = 75
}
return {
plugins: {
'postcss-px2rem-exclude': {
remUnit,
}
}
}
}
Copy the code
Note 4:
General static page is not skeleton screen, because there is no interface request, its loading speed is very fast, there is no need for skeleton screen, and skeleton screen is also a static, just to have a request to improve the interactive experience!