Vue2单选框点击后可取消勾选的一种解决办法

在Vue2中,由于HTML5的单选框选中后无法直接取消,本文介绍了一种利用鼠标事件处理的解决办法。通过在`beforeclick`事件中判断并更改`checked`属性,结合事件修饰符`.stop`防止默认行为,实现了单选框的取消勾选。同时,为处理数据更新,需要手动更新Vue实例的数据。对于label与radio的交互问题,可通过添加事件处理和`v-bind:for`属性进行解决。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Vue2单选框点击后可取消勾选的一种解决办法

因为原生html5的单选框一旦选中后无法指定取消勾选,于是在实践中尝试出了以下方法

通过鼠标事件来取消勾选

待修改的原生html代码

<template>
	<input type="radio" name="gender"><input type="radio" name="gender"><template/>

思路是,在点击事件发生前判断radio的 checked 的值,如果为 false,将被点击的标签的 checked 改为 false
但是如果将方法绑定在 click 事件上,实际不会获得点击前的 checked 值,只会获得点击事件完成后的值

<template>
	<input type="radio" name="gender" @click="radioClick"><input type="radio" name="gender"><template/>
methods:{
	redioClick(e){
		console.log('checked-0', e.target.checked)
		// 控制台输出:checked-0 true
	}
}

于是改用 mousedown 事件,但是radio标签本身具有click事件,因此无论在何种状态下点击单选框,radio都会自动勾选上,所以为了阻碍click,用vue的事件修饰符 prevent 来阻止click发生

<template>
	<input type="radio" name="gender" @mousedown="radioClick" @click.prevent><input type="radio" name="gender"><template/>
methods:{
	redioClick(e){
		console.log('checked-0', e.target.checked)
		var i = e.target.checked
		e.target.checked = !i
		console.log('checked-1', e.target.checked)
		// 控制台输出: checked-0 true
		//             checked-1 false
					   
	}
}

至此click可以取消勾选,但是click事件取消后将无法把radio标签所带的信息传入vm
如需要在radio中加入value属性来更新vm存储的表单信息

<template>
	<input type="radio" name="gender" v-model="gender" value="male" @mousedown="radioClick" @click.prevent><input type="radio" name="gender" v-model="gender" value="female" @mousedown="radioClick" @click.prevent><template/>
data(){
	return{
		gender: null
	}
},
methods:{
	redioClick(e){
		console.log('checked-0', e.target.checked)
		var i = e.target.checked
		e.target.checked = !i
		console.log('checked-1', e.target.checked)
		console.log('gender', this.gender)
		// 控制台输出: checked-0 true
		//             checked-1 false
		//             gender null
					   
	}
}

这将导致vm接收不到DOM传来的数据,所以需要手动为vm更新

data(){
	return{
		gender: null
	}
},
methods:{
	redioClick(e){
		var i = e.target.checked
		e.target.checked = !i
		if(i === false){
			this.gender = e.target.value
		}
		else if(i === true){
			this.gender = '' // 取消勾选后将gender清空
		}
		console.log('gender', this.gender)
		// 点击 男 控制台输出 gender male
		// 再次点击 男 控制台输出 gender 			   
	}
}

到此可以简单实现radio标签的取消勾选和数据更新
但是如果业务中要求使用label来指定radio,如:

<template>
	<input type="radio" id="male" name="gender" v-model="gender" value="male" @mousedown="radioClick" @click.prevent><label for="male"><label/>
	<input type="radio" id="female" name="gender" v-model="gender" value="female" @mousedown="radioClick" @click.prevent><label for="female"><label/>
<template/>
data(){
	return{
		gender: null
	}
},
methods:{
	redioClick(e){
		var i = e.target.checked
		e.target.checked = !i
		if(i === false){
			this.gender = e.target.value
		}
		else if(i === true){
			this.gender = '' // 取消勾选后将gender清空
		}
		console.log('gender', this.gender)			   
	}
}

但是此时点击label后,将无法对响应的radio生效,原因大概就是因为前面 @click.prevent 了,为了解决这个问题,可以依照 radioClick 再配置一个方法,同时给radio添加上 ref 属性,之前的id属性在这里意义就不大了,当然for属性也可以用自定义属性

<template>
	<input type="radio" ref="male"  name="gender" v-model="gender" value="male" @mousedown="radioClick" @click.prevent><label for="male" @mousedown="labelClick"><label/>
	<input type="radio" ref="female" name="gender" v-model="gender" value="female" @mousedown="radioClick" @click.prevent><label for="female" @mousedown="labelClick"><label/>
<template/>
data(){
	return{
		gender: null
	}
},
methods:{
	redioClick(e){
		var i = e.target.checked
		e.target.checked = !i
		if(i === false){
			this.gender = e.target.value
		}
		else if(i === true){
			this.gender = '' // 取消勾选后将gender清空
		}
		console.log('gender', this.gender)			   
	},
	labelClick(e){
		// 逻辑同上一个方法类似
		var forRef = e.target.getAttribute('for')
		var i = this.$refs[forRef].checked
		this.$refs[forRef].checked = !i
		if(i === false){
			this.gender = this.$refs[forRef].value
		}
		else if(i === true){
			this.gender = ''
		}
		console.log('gender', this.gender)
		// 控制台输出同样有效
	}
}

本人菜鸟一枚~ 我这个思路并不难,但由于阻止了radio其原有的click事件,所以这种方法的逻辑可能并不是最简洁的,对于click和label也应该有更好的处理方式。如有更好的方法请多指教~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值