customSwitch.vue 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  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: Boolean,
  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 = newValue
  57. this.position = this.isValue ? this.maxPosition : 2;
  58. }
  59. },
  60. mounted() {
  61. //this.position = this.defaultPosition === "right" ? this.maxPosition : 2;
  62. // 默认根据 defaultPosition 设置 modelValue
  63. // this.$emit("update:modelValue", this.defaultPosition === "right");
  64. // console.log("modelValue", this.modelValue);
  65. },
  66. methods: {
  67. async triggerRequest() {
  68. if (this.isShaking) return; // 避免重复触发
  69. this.isShaking = true; // 开始抖动
  70. try {
  71. await this.fetchData(); // 触发后台请求
  72. } catch (error) {
  73. console.error("请求失败:", error);
  74. } finally {
  75. console.log(this.defaultPosition);
  76. this.isShowSwitchText = true;
  77. this.isShaking = false; // 停止抖动
  78. this.isValue = !this.isValue
  79. // this.$emit("changeValue", this.isValue); // 切换状态
  80. }
  81. },
  82. handleClick() {
  83. if (this.isSwiping) return;
  84. this.isShowSwitchText = false;
  85. this.triggerRequest();
  86. },
  87. onTouchStart(e) {
  88. this.isShowSwitchText = false;
  89. this.isSwiping = false;
  90. this.startX = e.touches[0].clientX;
  91. },
  92. onTouchMove(e) {
  93. let moveX = e.touches[0].clientX - this.startX;
  94. if (Math.abs(moveX) > 10) {
  95. this.isSwiping = true;
  96. }
  97. },
  98. onTouchEnd() {
  99. if (!this.isSwiping) return;
  100. this.triggerRequest();
  101. this.isSwiping = false;
  102. }
  103. }
  104. };
  105. </script>
  106. <style scoped>
  107. .switch-container {
  108. /* width: 400rpx;
  109. height: 96rpx; */
  110. background: #FFFFFF;
  111. border-radius: 48rpx;
  112. border: 4rpx solid #F1F3F4;
  113. display: flex;
  114. align-items: center;
  115. padding: 2px;
  116. position: relative;
  117. }
  118. .switch {
  119. width: 100%;
  120. height: 100%;
  121. border-radius: 15px;
  122. position: relative;
  123. transition: background 0.3s;
  124. }
  125. .switch.active {
  126. background: #ffffff;
  127. }
  128. /* 外层比滑块大 5px */
  129. .indicator-wrapper {
  130. width: 144rpx;
  131. height: 80rpx;
  132. background: rgba(0, 0, 0, 1);
  133. border-radius: 42rpx;
  134. display: flex;
  135. justify-content: center;
  136. align-items: center;
  137. position: absolute;
  138. /* top: 6rpx; */
  139. left: 6rpx;
  140. }
  141. .indicator {
  142. width: 26px;
  143. height: 26px;
  144. border-radius: 50%;
  145. position: relative;
  146. transition: transform 0.3s;
  147. }
  148. .switch.active .indicator-wrapper {
  149. transform: translateX(244rpx);
  150. }
  151. .indicator-wrapper.shaking {
  152. animation: shake 0.2s infinite alternate;
  153. }
  154. .switch_text_on {
  155. position: relative;
  156. width: 158rpx;
  157. height: 80rpx;
  158. font-family: PingFangSC, PingFang SC;
  159. font-weight: 500;
  160. font-size: 32rpx;
  161. color: #060809;
  162. line-height: 80rpx;
  163. text-align: left;
  164. font-style: normal;
  165. margin-left: 200rpx;
  166. }
  167. .switch_text_off {
  168. position: relative;
  169. width: 158rpx;
  170. height: 80rpx;
  171. font-family: PingFangSC, PingFang SC;
  172. font-weight: 500;
  173. font-size: 32rpx;
  174. color: #060809;
  175. line-height: 80rpx;
  176. text-align: left;
  177. font-style: normal;
  178. margin-left: 50rpx;
  179. }
  180. @keyframes shake {
  181. 0% {
  182. transform: translateX(110rpx);
  183. }
  184. 100% {
  185. transform: translateX(120rpx);
  186. }
  187. }
  188. </style>