123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185 |
- <template>
- <view class="switch-container" :style="containerStyle" @click="handleClick">
- <!-- 状态文字 -->
- <text class="status-text" :style="[statusTextStyle, leftTextStyle]">{{ activeText }}</text>
- <text class="status-text" :style="[statusTextStyle, rightTextStyle]">{{ inactiveText }}</text>
- <!-- 图片滑块 -->
- <image class="switch-thumb" :src="currentThumb" :style="thumbStyle" @touchstart="onTouchStart"
- @touchmove="onTouchMove" @touchend="onTouchEnd" />
- </view>
- </template>
- <script>
- export default {
- props: {
- value: Boolean,
- containerWidth: {
- type: Number,
- default: 120
- },
- thumbSize: {
- type: Number,
- default: 32
- },
- // 新增图片相关props
- activeThumb: {
- type: String,
- default: '' // 建议使用绝对路径
- },
- inactiveThumb: {
- type: String,
- default: ''
- },
- // 状态文字配置
- activeText: {
- type: String,
- default: 'ON'
- },
- inactiveText: {
- type: String,
- default: 'OFF'
- },
- activeColor: {
- type: String,
- default: 'ffffff'
- },
- inactiveColor: {
- type: String,
- default: 'ffffff'
- },
- textColor: {
- type: String,
- default: '#000000'
- }
- },
- data() {
- return {
- isDragging: false,
- startX: 0,
- currentX: 0,
- maxOffset: 0
- };
- },
- computed: {
- currentThumb() {
- // return this.value ? this.activeThumb : this.inactiveThumb;
- return 'https://qiniu.bms16.com/Fkovrpq1bexe-Unal_VJREbLUhdu';
- },
- containerStyle() {
- return {
- width: this.containerWidth + 'px',
- backgroundColor: this.value ? this.activeColor : this.inactiveColor,
- height: this.thumbSize + 8 + 'px',
- borderRadius: (this.thumbSize + 8) / 2 + 'px'
- }
- },
- thumbStyle() {
- const baseStyle = {
- width: this.thumbSize + 'px',
- height: this.thumbSize + 'px',
- borderRadius: this.thumbSize / 2 + 'px'
- };
- if (this.isDragging) {
- return {
- ...baseStyle,
- transform: `translateX(${this.currentX}px)`,
- transition: 'none'
- };
- }
- return {
- ...baseStyle,
- transform: `translateX(${this.value ? this.maxOffset : 0}px)`,
- transition: 'transform 0.3s cubic-bezier(0.3, 1, 0.3, 1)'
- };
- },
- statusTextStyle() {
- return {
- color: this.textColor,
- fontSize: this.thumbSize * 0.5 + 'px',
- opacity: this.isDragging ? 0.6 : 1
- }
- },
- leftTextStyle() {
- return {
- right: (this.thumbSize + 20) + 'px',
- opacity: this.value ? 1 : 0
- }
- },
- rightTextStyle() {
- return {
- left: (this.thumbSize + 20) + 'px',
- opacity: this.value ? 0 : 1
- }
- }
- },
- mounted() {
- //计算滑块(thumb)在滑动轨道(container)上的最大偏移量。
- this.maxOffset = this.containerWidth - this.thumbSize - 4;
- },
- methods: {
- handleClick() {
- //点击立即切换状态
- if (!this.isDragging) {
- this.toggleState();
- }
- },
- onTouchStart(e) {
- this.isDragging = true;
- this.startX = e.touches[0].pageX;
- this.currentX = this.value ? this.maxOffset : 0;
- },
- onTouchMove(e) {
- if (!this.isDragging) return;
- const deltaX = e.touches[0].pageX - this.startX;
- let newX = this.value ? this.maxOffset + deltaX : deltaX;
- // 限制滑动范围
- newX = Math.max(0, Math.min(newX, this.maxOffset));
- this.currentX = newX;
- },
- toggleState() {
- this.$emit('input', !this.value);
- this.$emit('change', !this.value);
- },
- onTouchEnd() {
- this.isDragging = false;
- // 判断滑动是否超过50%
- const threshold = this.containerWidth * 0.5;
- const newState = this.currentX > threshold - this.thumbSize / 2;
- if (newState !== this.value) {
- this.toggleState();
- } else {
- this.currentX = this.value ? this.maxOffset : 0;
- }
- }
- }
- };
- </script>
- <style scoped>
- .switch-container {
- position: relative;
- display: flex;
- align-items: center;
- transition: background-color 0.3s;
- box-shadow: 0 2px 6px rgba(0, 0, 0, 0.15);
- }
- .switch-thumb {
- position: absolute;
- top: 4px;
- left: 4px;
- z-index: 2;
- box-shadow: 0 2px 6px rgba(0, 0, 0, 0.15);
- }
- .status-text {
- position: absolute;
- z-index: 1;
- font-weight: bold;
- transition: opacity 0.3s;
- pointer-events: none;
- }
- </style>
|