vue.js @click绑定点击事件不生效?

本文解决Vue子组件点击事件需加native修饰符的问题,介绍如何在子组件内部处理并重新触发click事件,实现更优雅的组件调用方式。

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

这个是在组件开发中遇到的问题,当时我在编写button的组件,模板是这样的:

<template>
  <button class="disable-hover button ion-button"
          :class="[modeClass,typeClass,shapeClass,sizeClass,colorClass,roleClass,strongClass]">
    <span class="button-inner">
      <slot></slot>
    </span>
    <div class="button-effect"></div>
  </button>
</template>

使用是这样子的:

<ion-button @click.native="primary()" color="primary">primary</ion-button>

根据Vue2.0官方文档关于父子组件通讯的原则,父组件通过prop传递数据给子组件,子组件触发事件给父组件。但父组件想在子组件上监听自己的click的话,需要加上native修饰符,故写法就像上面这样。

好吧,处女座的毛病又来了。像button这样常用的组件如果加上native的话是感觉很突兀的。虽然组件设计有自身的考虑,因此我想将click的native去掉,思路如下:

  1. 子组件监听父组件给的click事件,
  2. 子组件内部处理click事件然后向外发送click事件:$emit("click".fn)

改造后的代码如下(亲测可用):

<template>
  <button class="disable-hover button ion-button" @click="_click"
          :class="[modeClass,typeClass,shapeClass,sizeClass,colorClass,roleClass,strongClass]">
    <span class="button-inner">
      <slot></slot>
    </span>
    <div class="button-effect"></div>
  </button>
</template>

<script type="text/babel">
export default{
    ....
    ....
    methods: {
      _click: function () {
        this.$emit('click', function () {
          alert('inner')
        })
      }
    }
}
</script>

父组件中这样使用:

<ion-button @click="primary()" color="primary">primary</ion-button>

 

### 解决Vue中`@click`事件在按钮上不触发的问题 当遇到 `@click` 事件无法正常工作的情况时,有几种常见原因可能导致此现象。以下是排查和解决问题的方法: #### 1. 检查模板语法错误 确保 HTML 结构正确无误,特别是标签闭合以及属性书写格式。任何未关闭的标签或拼写错误都可能阻止事件监听器生效。 #### 2. 验证组件注册与引入 确认当前使用的组件已经正确导入并全局/局部注册。如果是在子组件内使用,则需保证父级已声明该子组件[^1]。 #### 3. 确认事件修饰符配置适当 有时为了防止默认行为或其他特殊需求会添加一些修饰符(如 `.prevent`, `.stop`),这些可能会干扰到点击操作的实际效果。移除不必要的修饰符来测试是否恢复正常响应。 #### 4. 查看样式层面上的影响 CSS 样式也可能影响元素可点击区域大小甚至隐藏掉目标按钮,通过浏览器开发者工具查看渲染后的 DOM 节点状态,排除因视觉因素造成的误解触控范围问题。 #### 5. 排查其他 JavaScript 错误 页面中存在的 JS 报错会影响整个应用逻辑执行流程,进而导致部分功能失效。打开控制台检查是否有异常提示,并修复相应代码缺陷。 #### 6. 使用事件委托提高效率 对于动态生成的内容或者大量重复项中的交互操作,考虑采用事件委派的方式代替直接绑定多个独立事件处理器。这不仅有助于提升性能表现,在某些场景下也能更好地兼容不同版本浏览器特性[^2]。 ```html <!-- 示例:基本按钮点击 --> <template> <div id="app"> <!-- 正常情况下应如此定义 --> <button @click="handleClick">点击这里</button> <!-- 如果涉及条件渲染等情况,请注意 v-if/v-show 的优先级 --> <button :class="{ active: isActive }" @click.stop="doSomethingElse()">另一个动作</button> <!-- 动态列表示例 --> <ul ref="listContainer"> <li v-for="(item, index) in items" :key="index">{{ item.name }}</li> </ul> </div> </template> <script> export default { data() { return { clickCount: 0, isActive: false, items: [ { name: 'Item One' }, { name: 'Item Two' } ] }; }, methods: { handleClick(event) { console.log('Button clicked!', ++this.clickCount); }, doSomethingElse() { alert('Alternative action performed.'); } }, mounted() { // 对于频繁更新的数据集合推荐这样做 const container = this.$refs.listContainer; container.addEventListener('click', (e) => { if(e.target.tagName.toLowerCase() === 'li') { console.log(`List item ${e.target.textContent} was selected.`); } }); } }; </script> ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值