u-loading-icon.vue 11.2 KB
   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
 255
 256
 257
 258
 259
 260
 261
 262
 263
 264
 265
 266
 267
 268
 269
 270
 271
 272
 273
 274
 275
 276
 277
 278
 279
 280
 281
 282
 283
 284
 285
 286
 287
 288
 289
 290
 291
 292
 293
 294
 295
 296
 297
 298
 299
 300
 301
 302
 303
 304
 305
 306
 307
 308
 309
 310
 311
 312
 313
 314
 315
 316
 317
 318
 319
 320
 321
 322
 323
 324
 325
 326
 327
 328
 329
 330
 331
 332
 333
 334
 335
 336
 337
 338
 339
 340
 341
 342
 343
<template>
<view
class="u-loading-icon"
:style="[$u.addStyle(customStyle)]"
:class="[vertical && 'u-loading-icon--vertical']"
v-if="show"
>
<view
v-if="!webviewHide"
class="u-loading-icon__spinner"
:class="[`u-loading-icon__spinner--${mode}`]"
ref="ani"
:style="{
color: color,
width: $u.addUnit(size),
height: $u.addUnit(size),
borderTopColor: color,
borderBottomColor: otherBorderColor,
borderLeftColor: otherBorderColor,
borderRightColor: otherBorderColor,
'animation-duration': `${duration}ms`,
'animation-timing-function': mode === 'semicircle' || mode === 'circle' ? timingFunction : ''
}"
>
<block v-if="mode === 'spinner'">
<!-- #ifndef APP-NVUE -->
<view
v-for="(item, index) in array12"
:key="index"
class="u-loading-icon__dot"
>
</view>
<!-- #endif -->
<!-- #ifdef APP-NVUE -->
<!-- 此组件内部图标部分无法设置宽高,即使通过width和height配置了也无效 -->
<loading-indicator
v-if="!webviewHide"
class="u-loading-indicator"
:animating="true"
:style="{
color: color,
width: $u.addUnit(size),
height: $u.addUnit(size)
}"
/>
<!-- #endif -->
</block>
</view>
<text
v-if="text"
class="u-loading-icon__text"
:style="{
fontSize: $u.addUnit(textSize),
color: textColor,
}"
>{{text}}</text>
</view>
</template>

<script>
import props from './props.js';
// #ifdef APP-NVUE
const animation = weex.requireModule('animation');
// #endif
/**
* loading 加载动画
* @description 警此组件为一个小动画,目前用在uView的loadmore加载更多和switch开关等组件的正在加载状态场景。
* @tutorial https://www.uviewui.com/components/loading.html
* @property {Boolean} show 是否显示组件 (默认 true)
* @property {String} color 动画活动区域的颜色,只对 mode = flower 模式有效(默认color['u-tips-color'])
* @property {String} textColor 提示文本的颜色(默认color['u-tips-color'])
* @property {Boolean} vertical 文字和图标是否垂直排列 (默认 false )
* @property {String} mode 模式选择,见官网说明(默认 'circle' )
* @property {String | Number} size 加载图标的大小,单位px (默认 24 )
* @property {String | Number} textSize 文字大小(默认 15 )
* @property {String | Number} text 文字内容
* @property {String} timingFunction 动画模式 (默认 'ease-in-out' )
* @property {String | Number} duration 动画执行周期时间(默认 1200)
* @property {String} inactiveColor mode=circle时的暗边颜色
* @property {Object} customStyle 定义需要用到的外部样式
* @example <u-loading mode="circle"></u-loading>
*/
export default {
name: 'u-loading-icon',
mixins: [uni.$u.mpMixin, uni.$u.mixin, props],
data() {
return {
// Array.form可以通过一个伪数组对象创建指定长度的数组
// https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/from
array12: Array.from({
length: 12
}),
// 这里需要设置默认值为360,否则在安卓nvue上,会延迟一个duration周期后才执行
// 在iOS nvue上,则会一开始默认执行两个周期的动画
aniAngel: 360, // 动画旋转角度
webviewHide: false, // 监听webview的状态,如果隐藏了页面,则停止动画,以免性能消耗
loading: false, // 是否运行中,针对nvue使用
}
},
computed: {
// 当为circle类型时,给其另外三边设置一个更轻一些的颜色
// 之所以需要这么做的原因是,比如父组件传了color为红色,那么需要另外的三个边为浅红色
// 而不能是固定的某一个其他颜色(因为这个固定的颜色可能浅蓝,导致效果没有那么细腻良好)
otherBorderColor() {
const lightColor = uni.$u.colorGradient(this.color, '#ffffff', 100)[80]
if (this.mode === 'circle') {
return this.inactiveColor ? this.inactiveColor : lightColor
} else {
return 'transparent'
}
// return this.mode === 'circle' ? this.inactiveColor ? this.inactiveColor : lightColor : 'transparent'
}
},
watch: {
show(n) {
// nvue中,show为true,且为非loading状态,就重新执行动画模块
// #ifdef APP-NVUE
if (n && !this.loading) {
setTimeout(() => {
this.startAnimate()
}, 30)
}
// #endif
}
},
mounted() {
this.init()
},
methods: {
init() {
setTimeout(() => {
// #ifdef APP-NVUE
this.show && this.nvueAnimate()
// #endif
// #ifdef APP-PLUS
this.show && this.addEventListenerToWebview()
// #endif
}, 20)
},
// 监听webview的显示与隐藏
addEventListenerToWebview() {
// webview的堆栈
const pages = getCurrentPages()
// 当前页面
const page = pages[pages.length - 1]
// 当前页面的webview实例
const currentWebview = page.$getAppWebview()
// 监听webview的显示与隐藏,从而停止或者开始动画(为了性能)
currentWebview.addEventListener('hide', () => {
this.webviewHide = true
})
currentWebview.addEventListener('show', () => {
this.webviewHide = false
})
},
// #ifdef APP-NVUE
nvueAnimate() {
// nvue下,非spinner类型时才需要旋转,因为nvue的spinner类型,使用了weex的
// loading-indicator组件,自带旋转功能
this.mode !== 'spinner' && this.startAnimate()
},
// 执行nvue的animate模块动画
startAnimate() {
this.loading = true
const ani = this.$refs.ani
if (!ani) return
animation.transition(ani, {
// 进行角度旋转
styles: {
transform: `rotate(${this.aniAngel}deg)`,
transformOrigin: 'center center'
},
duration: this.duration,
timingFunction: this.timingFunction,
// delay: 10
}, () => {
// 每次增加360deg,为了让其重新旋转一周
this.aniAngel += 360
// 动画结束后,继续循环执行动画,需要同时判断webviewHide变量
// nvue安卓,页面隐藏后依然会继续执行startAnimate方法
this.show && !this.webviewHide ? this.startAnimate() : this.loading = false
})
}
// #endif
}
}
</script>

<style lang="scss" scoped>
@import "../../libs/css/components.scss";
$u-loading-icon-color: #c8c9cc !default;
$u-loading-icon-text-margin-left:4px !default;
$u-loading-icon-text-color:$u-content-color !default;
$u-loading-icon-text-font-size:14px !default;
$u-loading-icon-text-line-height:20px !default;
$u-loading-width:30px !default;
$u-loading-height:30px !default;
$u-loading-max-width:100% !default;
$u-loading-max-height:100% !default;
$u-loading-semicircle-border-width: 2px !default;
$u-loading-semicircle-border-color:transparent !default;
$u-loading-semicircle-border-top-right-radius: 100px !default;
$u-loading-semicircle-border-top-left-radius: 100px !default;
$u-loading-semicircle-border-bottom-left-radius: 100px !default;
$u-loading-semicircle-border-bottom-right-radiu: 100px !default;
$u-loading-semicircle-border-style: solid !default;
$u-loading-circle-border-top-right-radius: 100px !default;
$u-loading-circle-border-top-left-radius: 100px !default;
$u-loading-circle-border-bottom-left-radius: 100px !default;
$u-loading-circle-border-bottom-right-radiu: 100px !default;
$u-loading-circle-border-width:2px !default;
$u-loading-circle-border-top-color:#e5e5e5 !default;
$u-loading-circle-border-right-color:$u-loading-circle-border-top-color !default;
$u-loading-circle-border-bottom-color:$u-loading-circle-border-top-color !default;
$u-loading-circle-border-left-color:$u-loading-circle-border-top-color !default;
$u-loading-circle-border-style:solid !default;
$u-loading-icon-host-font-size:0px !default;
$u-loading-icon-host-line-height:1 !default;
$u-loading-icon-vertical-margin:6px 0 0 !default;
$u-loading-icon-dot-top:0 !default;
$u-loading-icon-dot-left:0 !default;
$u-loading-icon-dot-width:100% !default;
$u-loading-icon-dot-height:100% !default;
$u-loading-icon-dot-before-width:2px !default;
$u-loading-icon-dot-before-height:25% !default;
$u-loading-icon-dot-before-margin:0 auto !default;
$u-loading-icon-dot-before-background-color:currentColor !default;
$u-loading-icon-dot-before-border-radius:40% !default;

.u-loading-icon {
/* #ifndef APP-NVUE */
// display: inline-flex;
/* #endif */
flex-direction: row;
align-items: center;
justify-content: center;
color: $u-loading-icon-color;

&__text {
margin-left: $u-loading-icon-text-margin-left;
color: $u-loading-icon-text-color;
font-size: $u-loading-icon-text-font-size;
line-height: $u-loading-icon-text-line-height;
}

&__spinner {
width: $u-loading-width;
height: $u-loading-height;
position: relative;
/* #ifndef APP-NVUE */
box-sizing: border-box;
max-width: $u-loading-max-width;
max-height: $u-loading-max-height;
animation: u-rotate 1s linear infinite;
/* #endif */
}

&__spinner--semicircle {
border-width: $u-loading-semicircle-border-width;
border-color: $u-loading-semicircle-border-color;
border-top-right-radius: $u-loading-semicircle-border-top-right-radius;
border-top-left-radius: $u-loading-semicircle-border-top-left-radius;
border-bottom-left-radius: $u-loading-semicircle-border-bottom-left-radius;
border-bottom-right-radius: $u-loading-semicircle-border-bottom-right-radiu;
border-style: $u-loading-semicircle-border-style;
}

&__spinner--circle {
border-top-right-radius: $u-loading-circle-border-top-right-radius;
border-top-left-radius: $u-loading-circle-border-top-left-radius;
border-bottom-left-radius: $u-loading-circle-border-bottom-left-radius;
border-bottom-right-radius: $u-loading-circle-border-bottom-right-radiu;
border-width: $u-loading-circle-border-width;
border-top-color: $u-loading-circle-border-top-color;
border-right-color: $u-loading-circle-border-right-color;
border-bottom-color: $u-loading-circle-border-bottom-color;
border-left-color: $u-loading-circle-border-left-color;
border-style: $u-loading-circle-border-style;
}

&--vertical {
flex-direction: column
}
}

/* #ifndef APP-NVUE */
:host {
font-size: $u-loading-icon-host-font-size;
line-height: $u-loading-icon-host-line-height;
}

.u-loading-icon {
&__spinner--spinner {
animation-timing-function: steps(12)
}

&__text:empty {
display: none
}

&--vertical &__text {
margin: $u-loading-icon-vertical-margin;
color: $u-content-color;
}

&__dot {
position: absolute;
top: $u-loading-icon-dot-top;
left: $u-loading-icon-dot-left;
width: $u-loading-icon-dot-width;
height: $u-loading-icon-dot-height;

&:before {
display: block;
width: $u-loading-icon-dot-before-width;
height: $u-loading-icon-dot-before-height;
margin: $u-loading-icon-dot-before-margin;
background-color: $u-loading-icon-dot-before-background-color;
border-radius: $u-loading-icon-dot-before-border-radius;
content: " "
}
}
}

@for $i from 1 through 12 {
.u-loading-icon__dot:nth-of-type(#{$i}) {
transform: rotate($i * 30deg);
opacity: 1 - 0.0625 * ($i - 1);
}
}

@keyframes u-rotate {
0% {
transform: rotate(0deg)
}

to {
transform: rotate(1turn)
}
}

/* #endif */
</style>