dengjunjie
5 天以前 4f39dcc195f28fa275fc2d065fbf1bf6a46c21b7
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
<template>
  <view class="tn-lazy-load-class tn-lazy-load">
    <view
      class="tn-lazy-load__item"
      :class="[`tn-lazy-load__item--${elIndex}`]"
      :style="[lazyLoadItemStyle]"
    >
      <view class="tn-lazy-load__item__content">
        <image
          v-if="!error"
          class="tn-lazy-load__item__image"
          :style="[imageStyle]"
          :src="show ? image : loadingImg"
          :mode="imgMode"
          @load="handleImgLoaded"
          @error="handleImgError"
          @tap="handleImgClick"
        ></image>
        <image
          v-else
          class="tn-lazy-load__item__image tn-lazy-load__item__image--error"
          :style="[imageStyle]"
          :src="errorImg"
          :mode="imgMode"
          @load="handleErrorImgLoaded"
          @tap="handleImgClick"
        ></image>
      </view>
    </view>
  </view>
</template>
 
<script>
  export default {
    name: 'tn-lazy-load',
    props: {
      // 组件标识
      index: {
        type: [String, Number],
        default: ''
      },
      // 待显示的图片地址
      image: {
        type: String,
        default: ''
      },
      // 图片裁剪模式
      imgMode: {
        type: String,
        default: 'scaleToFill'
      },
      // 占位图片路径
      loadingImg: {
          type: String,
          // default: ''
      },
      // 加载失败的错误占位图
      errorImg: {
          type: String,
          default: ''
      },
      // 图片进入可见区域前多少像素前,单位rpx,开始加载图片
      // 负数为图片超出屏幕底部多少像素后触发懒加载,正数为图片顶部距离屏幕底部多少距离时触发(图片还没出现在屏幕上)
      threshold: {
        type: [Number, String],
        default: 100
      },
      // 是否开启过渡效果
      isEffect: {
        type: Boolean,
        default: true
      },
      // 动画过渡时间
      duration: {
        type: [String, Number],
        default: 500
      },
      // 渡效果的速度曲线,各个之间差别不大,因为这是淡入淡出,且时间很短,不是那些变形或者移动的情况,会明显
      // linear|ease|ease-in|ease-out|ease-in-out|cubic-bezier(n,n,n,n);
      effect: {
          type: String,
          default: 'ease-in-out'
      },
      // 图片高度,单位rpx
      height: {
        type: [String, Number],
        default: 450
      },
      // 图片圆角
      borderRadius: {
        type: String,
        default: ''
      }
    },
    computed: {
      thresholdValue() {
        // 先取绝对值,因为threshold可能是负数,最后根据this.threshold是正数或者负数,重新还原
        let threshold = uni.upx2px(Math.abs(this.threshold))
        return this.threshold < 0 ? -threshold : threshold
      },
      lazyLoadItemStyle() {
        let style = {}
        style.opacity = Number(this.opacity)
        if (this.borderRadius) {
          style.borderRadius = this.borderRadius
        }
        // 因为time值需要改变,所以不直接用duration值(不能改变父组件prop传过来的值)
        style.transition = `opacity ${this.time / 1000}s ${this.effect}`
        style.height = this.$t.string.getLengthUnitValue(this.height)
        return style
      },
      imageStyle() {
        let style = {}
        if (typeof this.height === 'string' && this.height.indexOf('%') === -1) {
          style.height = this.$t.string.getLengthUnitValue(this.height)
        }
        return style
      }
    },
    watch: {
      show(val) {
        // 如果不开启过渡效果直接返回
        if (!this.effect) return
        this.time = 0
        // 原来opacity为1(不透明,是为了显示占位图),改成0(透明,意味着该元素显示的是背景颜色,默认的白色),再改成1,是为了获得过渡效果
        this.opacity = 0
        setTimeout(() => {
          this.time = this.duration
          this.opacity = 1
        }, 30)
      },
      image(val) {
        // 修改图片后重置部分变量
        if (!val) {
          // 如果传入null或者'',或者undefined,标记为错误状态
          this.error = true
        } else {
          this.init()
          this.error = false
        }
      }
    },
    data() {
      return {
        elIndex: this.$t.uuid(),
        // 显示图片
        show: false,
        // 图片透明度
        opacity: 1,
        // 动画时间
        time: this.duration,
        // 懒加载状态
        // loadlazy-懒加载中状态,loading-图片正在加载,loaded-图片加加载完成
        loadStatus: '',
        // 图片加载失败
        error: false
      }
    },
    created() {
      // 由于一些特殊原因,不能将此变量放到data中定义
      this.observer = {}
      this.observerName = 'lazyLoadContentObserver'
    },
    mounted() {
      // 在需要用到懒加载的页面,在触发底部的时候触发tOnLazyLoadReachBottom事件,保证所有图片进行加载
      this.$nextTick(() => {
        uni.$once('tOnLazyLoadReachBottom', () => {
          if (!this.show) this.show = true
        })
      })
      // mounted的时候,不一定挂载了这个元素,延时30ms,否则会报错或者不报错,但是也没有效果
      setTimeout(() => {
        this.disconnectObserver(this.observerName)
        const contentObserver = uni.createIntersectionObserver(this)
        contentObserver.relativeToViewport({
          bottom: this.thresholdValue
        }).observe(`.tn-lazy-load__item--${this.elIndex}`, (res) => {
          if (res.intersectionRatio > 0) {
            // 懒加载状态改变
            this.show = true
            // 如果图片已经加载,去掉监听,减少性能消耗
            this.disconnectObserver(this.observerName)
          }
        })
        this[this.observerName] = contentObserver
      }, 50)
    },
    methods: {
      // 初始化
      init() {
        this.error = false
        this.loadStatus = ''
      },
      // 处理图片点击事件
      handleImgClick() {
        let whichImg = ''
        // 如果show为false,则表示图片还没有开始加载,点击的是最开始占位图
        if (this.show === false) whichImg = 'lazyImg'
        // 如果error为true,则表示图片加载失败,点击的是错误占位图
        else if (this.error === true) whichImg = 'errorImg'
        // 点击了正常的图片
        else whichImg = 'realImg'
        
        this.$emit('click', {
          index: this.index,
          whichImg: whichImg
        })
      },
      // 处理图片加载完成事件,通过show来区分是占位图触发还是加载真正的图片触发
      handleImgLoaded() {
        if (this.loadStatus = '') {
          // 占位图加载完成
          this.loadStatus = 'lazyed'
        }
        else if (this.loadStatus == 'lazyed') {
          // 真正的图片加载完成
          this.loadStatus = 'loaded'
          this.$emit('loaded', this.index)
        }
      },
      // 处理错误图片加载完成
      handleErrorImgLoaded() {
        this.$emit('error', this.index)
      },
      // 处理图片加载失败
      handleImgError() {
        this.error = true
      },
      disconnectObserver(observerName) {
        const observer = this[observerName]
        observer && observer.disconnect()
      }
    }
  }
</script>
 
<style lang="scss" scoped>
  .tn-lazy-load {
    &__item {
      background-color: $tn-bg-gray-color;
      overflow: hidden;
      
      &__image {
        // 解决父容器会多出3px的问题
        display: block;
        width: 100%;
        // 骗系统开启硬件加速
        transform: transition3d(0, 0, 0);
        // 防止图片加载“闪一下”
        will-change: transform;
      }
    }
  }
</style>