u-switch.vue 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. <template>
  2. <view class="u-switch" :class="[value == true ? 'u-switch--on' : '', disabled ? 'u-switch--disabled' : '']" @tap="onClick" :style="[switchStyle]">
  3. <view
  4. class="u-switch__node node-class"
  5. :style="{
  6. width: $u.addUnit(this.size),
  7. height: $u.addUnit(this.size)
  8. }"
  9. ></view>
  10. </view>
  11. </template>
  12. <script>
  13. /**
  14. * switch 开关选择器
  15. * @description 选择开关一般用于只有两个选择,且只能选其一的场景。
  16. * @tutorial https://www.uviewui.com/components/switch.html
  17. * @property {Boolean} loading 是否处于加载中(默认false)
  18. * @property {Boolean} disabled 是否禁用(默认false)
  19. * @property {String Number} size 开关尺寸,单位rpx(默认50)
  20. * @property {String} active-color 打开时的背景色(默认#2979ff)
  21. * @property {Boolean} inactive-color 关闭时的背景色(默认#ffffff)
  22. * @property {Boolean | Number | String} active-value 打开选择器时通过change事件发出的值(默认true)
  23. * @property {Boolean | Number | String} inactive-value 关闭选择器时通过change事件发出的值(默认false)
  24. * @event {Function} change 在switch打开或关闭时触发
  25. * @example <u-switch v-model="checked" active-color="red" inactive-color="#eee"></u-switch>
  26. */
  27. export default {
  28. name: 'u-switch',
  29. props: {
  30. // 是否为加载中状态
  31. loading: {
  32. type: Boolean,
  33. default: false
  34. },
  35. // 是否为禁用装填
  36. disabled: {
  37. type: Boolean,
  38. default: false
  39. },
  40. // 开关尺寸,单位rpx
  41. size: {
  42. type: [Number, String],
  43. default: 50
  44. },
  45. // 打开时的背景颜色
  46. activeColor: {
  47. type: String,
  48. default: '#2979ff'
  49. },
  50. // 关闭时的背景颜色
  51. inactiveColor: {
  52. type: String,
  53. default: '#ffffff'
  54. },
  55. // 通过v-model双向绑定的值
  56. value: {
  57. type: Boolean,
  58. default: false
  59. },
  60. // 是否使手机发生短促震动,目前只在iOS的微信小程序有效(2020-05-06)
  61. vibrateShort: {
  62. type: Boolean,
  63. default: false
  64. },
  65. // 打开选择器时的值
  66. activeValue: {
  67. type: [Number, String, Boolean],
  68. default: true
  69. },
  70. // 关闭选择器时的值
  71. inactiveValue: {
  72. type: [Number, String, Boolean],
  73. default: false
  74. }
  75. },
  76. data() {
  77. return {};
  78. },
  79. computed: {
  80. switchStyle() {
  81. let style = {};
  82. style.fontSize = this.size + 'rpx';
  83. style.backgroundColor = this.value ? this.activeColor : this.inactiveColor;
  84. return style;
  85. },
  86. loadingColor() {
  87. return this.value ? this.activeColor : null;
  88. }
  89. },
  90. methods: {
  91. onClick() {
  92. if (!this.disabled && !this.loading) {
  93. // 使手机产生短促震动,微信小程序有效,APP(HX 2.6.8)和H5无效
  94. if (this.vibrateShort) uni.vibrateShort();
  95. this.$emit('input', !this.value);
  96. // 放到下一个生命周期,因为双向绑定的value修改父组件状态需要时间,且是异步的
  97. this.$nextTick(() => {
  98. this.$emit('change', this.value ? this.activeValue : this.inactiveValue);
  99. });
  100. }
  101. }
  102. }
  103. };
  104. </script>
  105. <style lang="scss" scoped>
  106. @import '../../libs/css/style.components.scss';
  107. .u-switch {
  108. position: relative;
  109. /* #ifndef APP-NVUE */
  110. display: inline-block;
  111. /* #endif */
  112. box-sizing: initial;
  113. width: 2em;
  114. height: 1em;
  115. background-color: #fff;
  116. border: 1px solid rgba(0, 0, 0, 0.1);
  117. border-radius: 1em;
  118. transition: background-color 0.3s;
  119. font-size: 50rpx;
  120. }
  121. .u-switch__node {
  122. @include vue-flex;
  123. align-items: center;
  124. justify-content: center;
  125. position: absolute;
  126. top: 0;
  127. left: 0;
  128. border-radius: 100%;
  129. z-index: 1;
  130. background-color: #fff;
  131. background-color: #fff;
  132. box-shadow: 0 3px 1px 0 rgba(0, 0, 0, 0.05), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 3px 3px 0 rgba(0, 0, 0, 0.05);
  133. box-shadow: 0 3px 1px 0 rgba(0, 0, 0, 0.05), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 3px 3px 0 rgba(0, 0, 0, 0.05);
  134. transition: transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05);
  135. transition: transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05), -webkit-transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05);
  136. transition: transform cubic-bezier(0.3, 1.05, 0.4, 1.05);
  137. transition: transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05);
  138. }
  139. .u-switch__loading {
  140. @include vue-flex;
  141. align-items: center;
  142. justify-content: center;
  143. }
  144. .u-switch--on {
  145. background-color: #1989fa;
  146. }
  147. .u-switch--on .u-switch__node {
  148. transform: translateX(100%);
  149. }
  150. .u-switch--disabled {
  151. opacity: 0.4;
  152. }
  153. </style>