customSwitch.vue 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. <template>
  2. <view class="switch-container" :style="{ width: width, height: height }" @touchstart="onTouchStart"
  3. @touchmove="onTouchMove" @touchend="onTouchEnd" @click="handleClick">
  4. <view class="switch" :class="{ active: isValue }">
  5. <!-- 额外包裹一层 -->
  6. <view class="indicator-wrapper" :class="{ shaking: isShaking }">
  7. <image class="indicator" :src="imageSrc" />
  8. </view>
  9. <view v-if="isShowSwitchText&&isValue" class="switch_text_off">{{$t('滑动关机')}}</view>
  10. <view v-if="isShowSwitchText&&!isValue" class="switch_text_on">{{$t('滑动开机')}}</view>
  11. </view>
  12. </view>
  13. </template>
  14. <script>
  15. export default {
  16. props: {
  17. modelValue: String,
  18. imageSrc: {
  19. type: String,
  20. default: "https://qiniu.bms16.com/Fkovrpq1bexe-Unal_VJREbLUhdu" // 默认滑块图片
  21. },
  22. width: {
  23. type: String,
  24. default: "400rpx" // 默认宽度
  25. },
  26. height: {
  27. type: String,
  28. default: "96rpx" // 默认高度
  29. },
  30. defaultPosition: {
  31. type: String,
  32. default: "right" // 默认在左边
  33. },
  34. fetchData: Function // 后台请求函数
  35. },
  36. data() {
  37. return {
  38. isValue:false,
  39. isShowSwitchText: true,
  40. isSwiping: false, // 是否滑动
  41. isShaking: false, // 是否抖动
  42. startX: 0, // 触摸起始位置
  43. position: 2, // 滑块初始位置 (左侧)
  44. containerWidth: 400, // 容器宽度(px)
  45. indicatorSize: 36 // 滑块大小(px)
  46. };
  47. },
  48. computed: {
  49. maxPosition() {
  50. return this.containerWidth - this.indicatorSize - 4; // 计算滑块最右边的位移
  51. }
  52. },
  53. watch: {
  54. modelValue(newValue) {
  55. // console.log(newValue,"值变动")
  56. // this.isValue = JSON.parse(newValue).state
  57. this.position = this.isValue ? this.maxPosition : 2;
  58. }
  59. },
  60. mounted() {
  61. },
  62. methods: {
  63. async triggerRequest() {
  64. if (this.isShaking) return; // 避免重复触发
  65. this.isShaking = true; // 开始抖动
  66. try {
  67. let data = await this.fetchData(this.isValue); // 触发后台请求
  68. this.isValue = JSON.parse(data).state
  69. this.$emit('changEnd',data)
  70. } catch (error) {
  71. console.error("请求失败:", error);
  72. } finally {
  73. this.isShowSwitchText = true;
  74. this.isShaking = false; // 停止抖动
  75. // this.isValue = !this.isValue
  76. }
  77. },
  78. handleClick() {
  79. if (this.isSwiping) return;
  80. this.triggerRequest();
  81. },
  82. onTouchStart(e) {
  83. this.isSwiping = false;
  84. this.startX = e.touches[0].clientX;
  85. },
  86. onTouchMove(e) {
  87. let moveX = e.touches[0].clientX - this.startX;
  88. if (Math.abs(moveX) > 10) {
  89. this.isSwiping = true;
  90. // this.isShowSwitchText = false;
  91. }
  92. },
  93. onTouchEnd() {
  94. if (!this.isSwiping) return;
  95. this.triggerRequest();
  96. this.isSwiping = false;
  97. }
  98. }
  99. };
  100. </script>
  101. <style scoped>
  102. .switch-container {
  103. /* width: 400rpx;
  104. height: 96rpx; */
  105. background: #FFFFFF;
  106. border-radius: 48rpx;
  107. border: 4rpx solid #F1F3F4;
  108. display: flex;
  109. align-items: center;
  110. padding: 2px;
  111. position: relative;
  112. }
  113. .switch {
  114. width: 100%;
  115. height: 100%;
  116. border-radius: 15px;
  117. position: relative;
  118. transition: background 0.3s;
  119. }
  120. .switch.active {
  121. background: #ffffff;
  122. }
  123. /* 外层比滑块大 5px */
  124. .indicator-wrapper {
  125. width: 144rpx;
  126. height: 80rpx;
  127. background: rgba(0, 0, 0, 1);
  128. border-radius: 42rpx;
  129. display: flex;
  130. justify-content: center;
  131. align-items: center;
  132. position: absolute;
  133. z-index: 9;
  134. /* top: 6rpx; */
  135. left: 6rpx;
  136. }
  137. .indicator {
  138. width: 26px;
  139. height: 26px;
  140. border-radius: 50%;
  141. position: relative;
  142. transition: transform 0.3s;
  143. }
  144. .switch.active .indicator-wrapper {
  145. transform: translateX(244rpx);
  146. }
  147. .indicator-wrapper.shaking {
  148. animation: shake 0.2s infinite alternate;
  149. }
  150. .switch_text_on {
  151. position: relative;
  152. width: 200rpx;
  153. text-align: center;
  154. height: 80rpx;
  155. font-family: PingFangSC, PingFang SC;
  156. font-weight: 500;
  157. font-size: 32rpx;
  158. color: #060809;
  159. line-height: 80rpx;
  160. text-align: left;
  161. font-style: normal;
  162. margin-left: 170rpx;
  163. }
  164. .switch_text_off {
  165. position: relative;
  166. width: 158rpx;
  167. height: 80rpx;
  168. font-family: PingFangSC, PingFang SC;
  169. font-weight: 500;
  170. font-size: 32rpx;
  171. color: #060809;
  172. line-height: 80rpx;
  173. text-align: left;
  174. font-style: normal;
  175. margin-left: 50rpx;
  176. }
  177. @keyframes shake {
  178. 0% {
  179. transform: translateX(110rpx);
  180. }
  181. 100% {
  182. transform: translateX(120rpx);
  183. }
  184. }
  185. </style>