toggle-switch.vue 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. <template>
  2. <view class="switch-container" :style="containerStyle" @click="handleClick">
  3. <!-- 状态文字 -->
  4. <text class="status-text" :style="[statusTextStyle, leftTextStyle]">{{ activeText }}</text>
  5. <text class="status-text" :style="[statusTextStyle, rightTextStyle]">{{ inactiveText }}</text>
  6. <!-- 图片滑块 -->
  7. <image class="switch-thumb" :src="currentThumb" :style="thumbStyle" @touchstart="onTouchStart"
  8. @touchmove="onTouchMove" @touchend="onTouchEnd" />
  9. </view>
  10. </template>
  11. <script>
  12. export default {
  13. props: {
  14. value: Boolean,
  15. containerWidth: {
  16. type: Number,
  17. default: 120
  18. },
  19. thumbSize: {
  20. type: Number,
  21. default: 32
  22. },
  23. // 新增图片相关props
  24. activeThumb: {
  25. type: String,
  26. default: '' // 建议使用绝对路径
  27. },
  28. inactiveThumb: {
  29. type: String,
  30. default: ''
  31. },
  32. // 状态文字配置
  33. activeText: {
  34. type: String,
  35. default: 'ON'
  36. },
  37. inactiveText: {
  38. type: String,
  39. default: 'OFF'
  40. },
  41. activeColor: {
  42. type: String,
  43. default: 'ffffff'
  44. },
  45. inactiveColor: {
  46. type: String,
  47. default: 'ffffff'
  48. },
  49. textColor: {
  50. type: String,
  51. default: '#000000'
  52. }
  53. },
  54. data() {
  55. return {
  56. isDragging: false,
  57. startX: 0,
  58. currentX: 0,
  59. maxOffset: 0
  60. };
  61. },
  62. computed: {
  63. currentThumb() {
  64. // return this.value ? this.activeThumb : this.inactiveThumb;
  65. return 'https://qiniu.bms16.com/Fkovrpq1bexe-Unal_VJREbLUhdu';
  66. },
  67. containerStyle() {
  68. return {
  69. width: this.containerWidth + 'px',
  70. backgroundColor: this.value ? this.activeColor : this.inactiveColor,
  71. height: this.thumbSize + 8 + 'px',
  72. borderRadius: (this.thumbSize + 8) / 2 + 'px'
  73. }
  74. },
  75. thumbStyle() {
  76. const baseStyle = {
  77. width: this.thumbSize + 'px',
  78. height: this.thumbSize + 'px',
  79. borderRadius: this.thumbSize / 2 + 'px'
  80. };
  81. if (this.isDragging) {
  82. return {
  83. ...baseStyle,
  84. transform: `translateX(${this.currentX}px)`,
  85. transition: 'none'
  86. };
  87. }
  88. return {
  89. ...baseStyle,
  90. transform: `translateX(${this.value ? this.maxOffset : 0}px)`,
  91. transition: 'transform 0.3s cubic-bezier(0.3, 1, 0.3, 1)'
  92. };
  93. },
  94. statusTextStyle() {
  95. return {
  96. color: this.textColor,
  97. fontSize: this.thumbSize * 0.5 + 'px',
  98. opacity: this.isDragging ? 0.6 : 1
  99. }
  100. },
  101. leftTextStyle() {
  102. return {
  103. right: (this.thumbSize + 20) + 'px',
  104. opacity: this.value ? 1 : 0
  105. }
  106. },
  107. rightTextStyle() {
  108. return {
  109. left: (this.thumbSize + 20) + 'px',
  110. opacity: this.value ? 0 : 1
  111. }
  112. }
  113. },
  114. mounted() {
  115. //计算滑块(thumb)在滑动轨道(container)上的最大偏移量。
  116. this.maxOffset = this.containerWidth - this.thumbSize - 4;
  117. },
  118. methods: {
  119. handleClick() {
  120. //点击立即切换状态
  121. if (!this.isDragging) {
  122. this.toggleState();
  123. }
  124. },
  125. onTouchStart(e) {
  126. this.isDragging = true;
  127. this.startX = e.touches[0].pageX;
  128. this.currentX = this.value ? this.maxOffset : 0;
  129. },
  130. onTouchMove(e) {
  131. if (!this.isDragging) return;
  132. const deltaX = e.touches[0].pageX - this.startX;
  133. let newX = this.value ? this.maxOffset + deltaX : deltaX;
  134. // 限制滑动范围
  135. newX = Math.max(0, Math.min(newX, this.maxOffset));
  136. this.currentX = newX;
  137. },
  138. toggleState() {
  139. this.$emit('input', !this.value);
  140. this.$emit('change', !this.value);
  141. },
  142. onTouchEnd() {
  143. this.isDragging = false;
  144. // 判断滑动是否超过50%
  145. const threshold = this.containerWidth * 0.5;
  146. const newState = this.currentX > threshold - this.thumbSize / 2;
  147. if (newState !== this.value) {
  148. this.toggleState();
  149. } else {
  150. this.currentX = this.value ? this.maxOffset : 0;
  151. }
  152. }
  153. }
  154. };
  155. </script>
  156. <style scoped>
  157. .switch-container {
  158. position: relative;
  159. display: flex;
  160. align-items: center;
  161. transition: background-color 0.3s;
  162. box-shadow: 0 2px 6px rgba(0, 0, 0, 0.15);
  163. }
  164. .switch-thumb {
  165. position: absolute;
  166. top: 4px;
  167. left: 4px;
  168. z-index: 2;
  169. box-shadow: 0 2px 6px rgba(0, 0, 0, 0.15);
  170. }
  171. .status-text {
  172. position: absolute;
  173. z-index: 1;
  174. font-weight: bold;
  175. transition: opacity 0.3s;
  176. pointer-events: none;
  177. }
  178. </style>