customSwitch.vue 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  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.isShowSwitchText = false;
  81. this.triggerRequest();
  82. },
  83. onTouchStart(e) {
  84. this.isShowSwitchText = false;
  85. this.isSwiping = false;
  86. this.startX = e.touches[0].clientX;
  87. },
  88. onTouchMove(e) {
  89. let moveX = e.touches[0].clientX - this.startX;
  90. if (Math.abs(moveX) > 10) {
  91. this.isSwiping = true;
  92. }
  93. },
  94. onTouchEnd() {
  95. if (!this.isSwiping) return;
  96. this.triggerRequest();
  97. this.isSwiping = false;
  98. }
  99. }
  100. };
  101. </script>
  102. <style scoped>
  103. .switch-container {
  104. /* width: 400rpx;
  105. height: 96rpx; */
  106. background: #FFFFFF;
  107. border-radius: 48rpx;
  108. border: 4rpx solid #F1F3F4;
  109. display: flex;
  110. align-items: center;
  111. padding: 2px;
  112. position: relative;
  113. }
  114. .switch {
  115. width: 100%;
  116. height: 100%;
  117. border-radius: 15px;
  118. position: relative;
  119. transition: background 0.3s;
  120. }
  121. .switch.active {
  122. background: #ffffff;
  123. }
  124. /* 外层比滑块大 5px */
  125. .indicator-wrapper {
  126. width: 144rpx;
  127. height: 80rpx;
  128. background: rgba(0, 0, 0, 1);
  129. border-radius: 42rpx;
  130. display: flex;
  131. justify-content: center;
  132. align-items: center;
  133. position: absolute;
  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: 158rpx;
  153. height: 80rpx;
  154. font-family: PingFangSC, PingFang SC;
  155. font-weight: 500;
  156. font-size: 32rpx;
  157. color: #060809;
  158. line-height: 80rpx;
  159. text-align: left;
  160. font-style: normal;
  161. margin-left: 200rpx;
  162. }
  163. .switch_text_off {
  164. position: relative;
  165. width: 158rpx;
  166. height: 80rpx;
  167. font-family: PingFangSC, PingFang SC;
  168. font-weight: 500;
  169. font-size: 32rpx;
  170. color: #060809;
  171. line-height: 80rpx;
  172. text-align: left;
  173. font-style: normal;
  174. margin-left: 50rpx;
  175. }
  176. @keyframes shake {
  177. 0% {
  178. transform: translateX(110rpx);
  179. }
  180. 100% {
  181. transform: translateX(120rpx);
  182. }
  183. }
  184. </style>