在Vue2开发中,我遇到了根据用户行为来动态显示或隐藏删除图标的需求,而v-if
指令正是为此设计的。然而,使用同一变量同时控制两个不同的行为会有预期之外的效果。
问题描述
假设我们有一个自定义组件common-input
,其核心代码如下:
<script>
export default {
name: 'common-input',
props: {
inputObject: {
type: Object,
default: () => {
return {};
}
}
},
data() {
return {
isFocused: false,
clearVisible: false
};
},
watch: {
inputObject: {
immediate: true, //第一次就立即生效
deep: true, // 数组及对象要深度watch
handler(val) {
this.updateClearStatus();
}
},
},
methods: {
onFocus(){
this.isFocused = true;
this.updateClearStatus();
},
onBlur(){
this.isFocused = false;
this.updateClearStatus();
},
clickClearAction(){
console.log('clear input info');
this.inputObject.value = '';
},
updateClearStatus(){
setTimeout(() => {
if(this.inputObject.value.length > 0 && this.isFocused) {
this.clearVisible = true;
} else {
this.clearVisible = false;
}
}, 200);
}
}
在这个组件中,我们通过v-if="clearVisible"
来控制一个清空输入框内容的图标的显示。当输入框有内容且获得焦点时,clearVisible
会被设为true
,图标显示;否则,图标隐藏。
遇到的问题
问题出现在用户点击清空图标时。点击清空图标应该触发clickClearAction
方法,清空输入框内容。然而,由于点击图标会触发输入框的blur失焦
事件,导致isFocused
变为false
,从而使clearVisible
变为false
,图标消失。结果是click
事件不会被触发,输入框内容无法被清空。
这个可以理解为blur事件是先一步触发的,导致图标消失,进而无法触发其上面绑定的事件。
解决方案
为了避免blur
事件导致图标消失并阻止click
事件,我们需要调整逻辑。具体来说,我使用了不同的变量,然后还使用setTimeout
来延迟v-if
条件的变化,从而保证click
事件能被正常触发。
以下是修改后的代码:
<template>
<div>
<input v-model="inputObject.value" class="main-input" type="tel" @focus="onFocus" @blur="onBlur" />
<div class="icon-section" v-if="isClear" @click="clickClearAction">
<i class="icon" />
</div>
</div>
</template>
<script>
export default {
name: 'common-input',
props: {
inputObject: {
type: Object,
default: () => {
return {};
}
}
},
data() {
return {
isFocused: false,
isClear: false
};
},
watch: {
inputObject: {
immediate: true,
deep: true,
handler(val) {
this.updateClearStatus();
}
}
},
methods: {
onFocus() {
this.isFocused = true;
this.updateClearStatus();
},
onBlur() {
setTimeout(() => {
this.isFocused = false;
this.updateClearStatus();
}, 200); // 延迟处理
},
clickClearAction() {
console.log('clear input info');
this.inputObject.value = '';
},
updateClearStatus() {
if (this.inputObject.value.length > 0 && this.isFocused) {
this.isClear = true;
} else {
this.isClear = false;
}
}
}
};
</script>
在这个修改后的版本中,我们将blur
事件的处理逻辑放入一个setTimeout
中,延迟执行isFocused
的状态更新。这确保了点击清空图标时,click
事件能先被触发,再执行blur
事件逻辑,从而避免图标过早消失的问题。
总结
在使用Vue2开发的时候,使用v-if
来控制元素的显示和隐藏很常用。但是要注意的是,当v-if
条件变化会影响到用户交互事件时,我们可能需要采取一些措施来确保事件的正常触发。