customSwitch.vue 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  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">滑动关机</view>
  10. <view v-if="isShowSwitchText&&!isValue" class="switch_text_on">滑动开机</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. await this.fetchData(); // 触发后台请求
  68. } catch (error) {
  69. console.error("请求失败:", error);
  70. } finally {
  71. this.isShowSwitchText = true;
  72. this.isShaking = false; // 停止抖动
  73. this.isValue = !this.isValue
  74. }
  75. },
  76. handleClick() {
  77. if (this.isSwiping) return;
  78. this.isShowSwitchText = false;
  79. this.triggerRequest();
  80. },
  81. onTouchStart(e) {
  82. this.isShowSwitchText = false;
  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. }
  91. },
  92. onTouchEnd() {
  93. if (!this.isSwiping) return;
  94. this.triggerRequest();
  95. this.isSwiping = false;
  96. }
  97. }
  98. };
  99. </script>
  100. <style scoped>
  101. .switch-container {
  102. /* width: 400rpx;
  103. height: 96rpx; */
  104. background: #FFFFFF;
  105. border-radius: 48rpx;
  106. border: 4rpx solid #F1F3F4;
  107. display: flex;
  108. align-items: center;
  109. padding: 2px;
  110. position: relative;
  111. }
  112. .switch {
  113. width: 100%;
  114. height: 100%;
  115. border-radius: 15px;
  116. position: relative;
  117. transition: background 0.3s;
  118. }
  119. .switch.active {
  120. background: #ffffff;
  121. }
  122. /* 外层比滑块大 5px */
  123. .indicator-wrapper {
  124. width: 144rpx;
  125. height: 80rpx;
  126. background: rgba(0, 0, 0, 1);
  127. border-radius: 42rpx;
  128. display: flex;
  129. justify-content: center;
  130. align-items: center;
  131. position: absolute;
  132. /* top: 6rpx; */
  133. left: 6rpx;
  134. }
  135. .indicator {
  136. width: 26px;
  137. height: 26px;
  138. border-radius: 50%;
  139. position: relative;
  140. transition: transform 0.3s;
  141. }
  142. .switch.active .indicator-wrapper {
  143. transform: translateX(244rpx);
  144. }
  145. .indicator-wrapper.shaking {
  146. animation: shake 0.2s infinite alternate;
  147. }
  148. .switch_text_on {
  149. position: relative;
  150. width: 158rpx;
  151. height: 80rpx;
  152. font-family: PingFangSC, PingFang SC;
  153. font-weight: 500;
  154. font-size: 32rpx;
  155. color: #060809;
  156. line-height: 80rpx;
  157. text-align: left;
  158. font-style: normal;
  159. margin-left: 200rpx;
  160. }
  161. .switch_text_off {
  162. position: relative;
  163. width: 158rpx;
  164. height: 80rpx;
  165. font-family: PingFangSC, PingFang SC;
  166. font-weight: 500;
  167. font-size: 32rpx;
  168. color: #060809;
  169. line-height: 80rpx;
  170. text-align: left;
  171. font-style: normal;
  172. margin-left: 50rpx;
  173. }
  174. @keyframes shake {
  175. 0% {
  176. transform: translateX(110rpx);
  177. }
  178. 100% {
  179. transform: translateX(120rpx);
  180. }
  181. }
  182. </style>