JS021. 原生JS拦截事件的显式处理与默认动作(Web API: event.preventDefault)

Web API - event.preventDefault( )

Event 接口的  preventDefault( ) 方法,告诉 user agent :如果此事件没有被显式处理,它默认的动作也不应该照常执行。此事件还是继续传播,除非碰到事件侦听器调用 stopPropagation( )stopImmediatePropagation( ) ,才停止传播。

- 语法

event.preventDefault();

- 参数

- 返回值

 undefined  

示例

- 阻止默认的点击事件执行

选中复选框是点击复选框的默认行为。下面这个例子说明了怎样阻止默认行为的发生:

JavaScript

document.querySelector("#id-checkbox").addEventListener("click", function(event) {
         document.getElementById("output-box").innerHTML += "Sorry! <code>preventDefault()</code> won't let you check this!<br>";
         event.preventDefault();
}, false);

HTML

<p>Please click on the checkbox control.</p>

<form>
  <label for="id-checkbox">Checkbox:</label>
  <input type="checkbox" id="id-checkbox"/>
</form>

<div id="output-box"></div>

结果

- 在编辑域中阻止按键

下面这个例子演示如何用  preventDefault( )  在文本编辑域中阻止有效的文本输入。如今我们通常可以使用 原生的HTML表单验证 来代替。

HTML

<div class="container">
  <p>Please enter your name using lowercase letters only.</p>

  <form>
    <input type="text" id="my-textbox">
  </form>
</div>

CSS

当用户按下一个有效按键时,我们就给这个 warning box 加上一些样式:

.warning {
  border: 2px solid #f39389;
  border-radius: 2px;
  padding: 10px;
  position: absolute;
  background-color: #fbd8d4;
  color: #3b3c40;
}

JavaScript

首先监听 keypress (en-US) 事件:

var myTextbox = document.getElementById('my-textbox');
myTextbox.addEventListener('keypress', checkName, false);

 checkName( )  方法可以监听按键并决定是否允许按键的默认行为发生。

function checkName(evt) {
  var charCode = evt.charCode;
  if (charCode != 0) {
    if (charCode < 97 || charCode > 122) {
      evt.preventDefault();
      displayWarning(
        "Please use lowercase letters only."
        + "\n" + "charCode: " + charCode + "\n"
      );
    }
  }
}

 dislpayWarning( )  方法显示了一个问题的通知。这不是一种优雅的方法,但确实可以达到我们的目的。

var warningTimeout;
var warningBox = document.createElement("div");
warningBox.className = "warning";

function displayWarning(msg) {
  warningBox.innerHTML = msg;

  if (document.body.contains(warningBox)) {
    window.clearTimeout(warningTimeout);
  } else {
    // insert warningBox after myTextbox
    myTextbox.parentNode.insertBefore(warningBox, myTextbox.nextSibling);
  }

  warningTimeout = window.setTimeout(function() {
      warningBox.parentNode.removeChild(warningBox);
      warningTimeout = -1;
    }, 2000);
}

结果

非小写a-z的键入均被拦截。

- 备注

在事件流(捕获 / 冒泡)的任何阶段调用  preventDefault( )  都会取消事件,这意味着任何通常被该实现触发并作为结果的默认行为都不会发生。

我们可以使用 Event.cancelable 来检查该事件是否支持取消。为一个不支持  cancelable  的事件调用  preventDefault( )  将没有效果。

应用场景

我们知道了该API能实现的效果,那么在设计项目过程中,除了上述的实例,还有什么时候应该想到此API呢?

这里举一个蚂蚁金服框架 Ant Design Vue 的组件 Button 封装过程中用到的实例:

可以看到,Ant Designbutton onClick 事件绑定了一个 handleClick函数,该函数定义了当按钮绑定的 loading 参数为 true disabeld 参数为 true 时,拦截了click事件的显式处理与默认动作,显式处理就是 Button 被点击的显示效果,默认动作就是click本身返回的点击Event,并 return 函数阻止事件向上抛出。

规范

posted @ 2021-08-25 14:50  97z4moon  阅读(614)  评论(0)    收藏  举报
Title