The introduction

The interaction between JavaScript and HTML is achieved through events. Events are specific moments of interaction that occur in a document or browser window. The browser’s event system is relatively complex. Although all major browsers have implemented “DOM2 level events,” the specification itself does not cover all event types, and the DOM event API has become richer with the advent of DOM3 level. There are also events supported by the Browser Object Model (BOM), but the relationship between these events and the DOCUMENT Object Model (DOM) is not clear because BOM events have long had no specification to follow (HTML5 later explained). Browser DOM event system were introduced in this paper, including three phases flow of events, the event handler of the three ways of different (DOM0, DOM2, IE), considering the event handling IE the differences of the event object and how to do compatibility processing, event object attributes in the how to apply to the practical application and the differences between them, And the sequence of event capture and bubbling.

At the beginning of this article, I will briefly introduce several important knowledge points of this article:

  • DOMThere are three ways that event handlers work,DOM0theonType.IE9The followingattachEventwithdetachEvent.DOM2theaddEventListenerwithremoveEventListener.
  • DOM2The advantage of grade A is that it can passaddEventListenerTo specify whether to capture or bubble, and can be the sameDOMThe element registers multiple event handlers of the same type. whileDOM0Only one event handler is supported per event
  • DOM0andDOM2Event handlers are passed in automaticallyeventObject; In the IEeventObject depends on the method of the specified event handler, so there will be one in IEwindow.event,eventTwo cases;eventObjects have some useful properties, such as target,currentTarget.preventDefault.stopPropagation.stopImmediatePropagationEtc.
  • forDOM0theontypeBinding methods to the element’s event behavior are performed during the bubble phase (or target phase) of the current element’s event behavior. forDOM2theaddEventListenerIn most cases, event handlers are added to the event bubble phase for maximum compatibility. It is not particularly necessary and it is not recommended to register event handlers during the event capture phase.
  • Compatibility handling of event handlers should be consideredDOM0andIE9The following event handling methods, the compatibility handling of event objects and event object properties, are taken into accountIEThe different
  • event.stopPropagation()Method prevents events from bubbling to the parent element, preventing any parent event handlers from being executedstopPropagationIs used to prevent events from bubbling, but it can also prevent catching events.
  • event.targetPoints to the element that causes the triggering event, andevent.currentTargetIs the element bound to the event, only the target element that was clickedevent.targetWill be equal toevent.currentTarget

Most browsers that support DOM event streaming implement a specific behavior; Even though the “dom2-level events” specification explicitly requires that the capture phase does not involve event targets, IE9, Safari, Chrome, Firefox, Opera9.5 and later all fire events on event objects during the capture phase. As a result, there are two opportunities to manipulate events on the target object.

Flow of events

The event flow describes the order in which events are received from the page. Interestingly, the IE and Netscape development teams came up with two opposite concepts of event flow. The IE event flow is the event bubble flow, and the standard browser event flow is the event capture flow. However, the W3C takes a compromise approach to the standard: catch and bubble later (the third parameter given via addEventListener supports both bubble and capture). Specifically, the same DOM element can register multiple events of the same type, with addEventListener registering the event and removeEventListener removing the event.

Note that in order for registered events to be unregistered, the callback function must be saved or unregistered.

The DOM event flow is divided into three phases: capture phase, target phase, and bubble phase. Handlers for the capture phase are called first, followed by handlers for the target phase, and finally, handlers for the bubble phase. (There are no standard HTML tags in the figure below)

(1) Capture stage: the stage where the event is propagated from the Window object top-down to the target node;

(2) Target stage: the stage when the real target node is processing events;

(3) Bubbling stage: the stage in which the event propagates from the target node to the Window object from bottom to top.

Capturing works from top to bottom. The event goes from the Window object, then to the Document (object), then to the HTML tag (get the HTML tag from the Document.documentElement), then to the body tag (get the body tag from the document.body), Then follow the normal HTML structure of the layer down, and finally reach the target element.

The event bubbling process is the inverse of event capture. Here’s an example of an event bubbling up:

/ / example 3 < div id = "outer" > < div id = "inner" > < / div > < / div >... window.onclick = function() { console.log('window'); }; document.onclick = function() { console.log('document'); }; document.documentElement.onclick = function() { console.log('html'); }; document.body.onclick = function() { console.log('body'); } outer.onclick = function(ev) { console.log('outer'); }; inner.onclick = function(ev) { console.log('inner'); };Copy the code

As mentioned below, onclick’s event behavior bindings to elements are performed during the bubbling (or target) phase of the current element’s event behavior.

DOM Event level

DOM levels can be divided into four levels: DOM0, DOM1, DOM2, and DOM3. DOM events are divided into three levels: DOM 0 level event processing, DOM 2 level event processing and DOM 3 level event processing. There are no DOM 1 level events because there is no DOM 1 level event content. And because IE and other browsers in the DOM2 level event processing is different, so generally can be divided into three types of event processing, namely DOM0, DOM2, IE. The following is at the DOM level

DOM 0-level events

el.onclick=function(){}

var btn = document.getElementById('btn');
 btn.onclick = function(){
     alert(this.innerHTML);
 }
Copy the code

It is not allowed to bind multiple events of the same type to the same element/tag (for example, to bind three click events to the BTN element above). DOM0 event binding, binding methods to the element’s event behavior that are executed during the bubbling (or target) phase of the current element’s event behavior.

DOM level 2 events

el.addEventListener(event-name, callback, useCapture)

  • Event-name: indicates the event name, which can be a standard DOM event
  • Callback: A callback function that is injected with an argument to the current event object, event, when an event is triggered
  • UseCapture: The default is false, meaning that the event handle executes during the bubble phase (or registers the bubble event), and true meaning that the event handle executes during the capture phase (or registers the capture event).
var btn = document.getElementById('btn');
btn.addEventListener("click", test, false);
function test(e){
	e = e || window.event;
    alert((e.target || e.srcElement).innerHTML);
    btn.removeEventListener("click", test)
}
/ / ie 9 - : attachEvent () and detachEvent ().
/ / ie 9 + / chrom/FF: addEventListener () and removeEventListener ()
Copy the code

AddEventListener () and removeEventListener() are not supported in IE9. AttachEvent () and detachEvent() are used instead. The first event name is preceded by on. This can be done with compatibility:

DOM level 3 events

More event types have been added to the DOM 2 level events.

  • UI events that are triggered when the user interacts with elements on the page, such as Load and Scroll

  • Focus events, which are triggered when an element gains or loses focus, such as blur and focus

  • Mouse events, such as dblclick and mouseup, are triggered when the user performs actions on the page using the mouse

  • A wheel event that is triggered when a mousewheel or similar device is used, such as mousewheel

  • A text event that is triggered when text is entered into a document, such as textInput

  • Keyboard events, such as keyDown and keyPress, are triggered when users perform operations on the page using the keyboard

  • A composite event that is triggered when a character is entered for the IME, such as compositionStart

  • Change event, triggered when the underlying DOM structure changes, such as DOMsubtreeModified

  • DOM3 events also allow users to customize some events.

Conclusion:

  • The benefit of DOM2 level is that you can add multiple event handlers; DOM0 supports only one event handler per event;

  • Anonymous functions added via DOM2 cannot be removed, and handlers for addEventListener and removeEventListener must have the same name

  • Scope: DOM0 handlers will run in the scope of the element, IE handlers will run in the global scope, this === window

  • Trigger order: When adding multiple events, DOM2 will execute them in the order they were added and IE will execute them in the reverse order

Cross-browser event handlers

Compatible with Internet Explorer 9 and DOM0

var EventUtil = {
  // Element is the current element and can be obtained by getElementById(id)
  // Type is the event type, usually click, but may also be mouse, focus, scroll wheel, etc
  // handle the event handler function
  addHandler: (element, type, handler) = > {
    // Check if there is a dom2-level method, then check if there is a dom0-level method.
    if (element.addEventListener) {
      // The third argument false indicates the bubbling phase
      element.addEventListener(type, handler, false);
    } else if (element.attachEvent) {
      element.attachEvent(`on${type}`, handler)
    } else {
      element[`on${type}`] = handler; }},removeHandler: (element, type, handler) = > {
    if (element.removeEventListener) {
      // The third argument false indicates the bubbling phase
      element.removeEventListener(type, handler, false);
    } else if (element.detachEvent) {
      element.detachEvent(`on${type}`, handler)
    } else {
      element[`on${type}`] = null; }}}// Get the element
var btn = document.getElementById('btn');
/ / define a handler
var handler = function(e) {
  console.log('I got clicked.');
}
// Listen on events
EventUtil.addHandler(btn, 'click', handler);
// Remove event listening
// EventUtil.removeHandler(button1, 'click', clickEvent);
Copy the code

The event agent

Since events propagate up to the parent node during the bubbling phase, you can define the listener function of the child node on the parent node and let the listener function of the parent node process events of multiple child elements uniformly. This approach is called delegation of events, or event delegation. The event broker has two advantages:

  • Reduce memory consumption and improve performance

Suppose we have a list with a large number of list items, and we need to respond to an event when each list item is clicked

/ / case 4 < ul id = "list" > < li > item 1 < / li > < li > item 2 < / li > < li > < / li > item 3... <li>item n</li> </ul>Copy the code

If you bind a function to each list item, it is very expensive in memory and requires a lot of performance in terms of efficiency. Using event agent, we only need to give the parent container ul binding approach, so no matter which a descendant elements, click will be according to the transmission mechanism of transmission of bubbling, click behavior triggers the container, and then the corresponding method to carry out, according to the event source, we can know is who, click to do different things.

  • Dynamically bound events

By user operation in many cases, we need to dynamically add or delete a list item element, if give each child element binding events at the beginning, then in the list of changes, just need to give new elements of the binding event, to tie events is about to delete the elements of the solution, if use event agency will save a lot of trouble.

Delegate the li element from the parent element #list to its parent:

// Bind events to parent elements
document.getElementById('list').addEventListener('click'.function (e) {
  // Compatibility processing
  var event = e || window.event;
  var target = event.target || event.srcElement;
  // Determine if the target element matches
  if (target.nodeName.toLocaleLowerCase === 'li') {
    console.log('the content is: ', target.innerHTML); }});Copy the code

The event object

Both DOM0 and DOM2 event handlers automatically pass in event objects, that is, when an event is triggered on the DOM, an event object is generated that contains all the information related to the event. The Event object in IE depends on the method of the specified event handler

The IE handler will run in the global scope, this === window, so there will be window.event and event cases in IE

In addition, in IE, the attributes of the event object are also different, the corresponding relationship is as follows:

SrcElement => Target returnValue => preventDefault() cancelBubble => stopPropagation() IE does not support event capture. But stopPropagation can cancel event capture and bubbling at the same time

An event object exists only during the event handler and is destroyed once the event handler completes execution

1. event. preventDefault()

If this method is called, the default event behavior will no longer fire. What is the default event? For example, a form click submit button to jump to the page, a label default page jump or anchor point positioning, etc.

Most of the time, we just want to use the A tag as a normal button, click to achieve a function, do not want to jump to the page, do not want to anchor location.

<a href="javascript:;" > link < / a >Copy the code

We can also prevent this by using the JS method, binding the method to the click event. When we click on the A tag, the click event will be triggered first, and the default behavior will be executed second

/ / method 2: < a id = "test" href = "http://www.cnblogs.com" > link < / a > < script > test. The onclick = function (e) {e = e | | window. The event; return false; } </script> < a id = "test" href = "http://www.cnblogs.com" > link < / a > < script > test. The onclick = function (e) {e = e | | window. The event; e.preventDefault(); } </script>Copy the code

A maximum of six characters can be entered in the input field.

/ / case 5
 <input type="text" id='tempInp'>
 <script>
    tempInp.onkeydown = function(ev) {
        ev = ev || window.event;
        let val = this.value.trim() //trim removes first space from string (incompatible)
        . / / this value = this. Value. The replace (/ ^ + | + $/ g, ' ') compatible with writing
        let len = val.length
        if (len >= 6) {
            this.value = val.substr(0.6);
            // Prevents the default behavior from removing special keys (DELETE\ back-space \ arrow keys...)
            let code = ev.which || ev.keyCode;
            if (!/ ^ (46 8 37 | | | | | 39 38, 40) $/.test(code)) {
                ev.preventDefault()
            }
        }
    }
 </script>
Copy the code

2. event.stopPropagation() & event.stopImmediatePropagation()

The event.stopPropagation() method prevents events from bubbling to the parent element, and prevents any parent event handlers from being executed. The event bubble phase mentioned above is the phase in which the event propagates bottom-up from the target node to the Window object. By adding event.stopPropagation() to the inner click event in the example above, we prevent the parent event from executing and print only ‘inner’.

 inner.onclick = function(ev) {
    console.log('inner');
    ev.stopPropagation();
};
Copy the code

StopImmediatePropagation prevents events from bubbling to the parent element and also prevents elements from being fired with other listeners of the event type. StopPropagation can only achieve the former effect. Let’s look at an example:

<body>
  <button id="btn">click me to stop propagation</button>
</body>
......
const btn = document.querySelector('#btn');
btn.addEventListener('click', event => {
  console.log('btn click 1');
  event.stopImmediatePropagation();
});
btn.addEventListener('click', event => {
  console.log('btn click 2');
});
document.body.addEventListener('click', () => {
  console.log('body click');
});
// btn click 1
Copy the code

As shown above, with stopImmediatePropagation, when a button is clicked, not only the body binding event is not fired, but also another click event of the button is not fired.

3. event.target & event.currentTarget

To be honest, the difference between the two can’t be described in words. Let’s look at an example:

<div id="a">
  <div id="b">
    <div id="c">
      <div id="d"></div>
    </div>
  </div>
</div>
<script>
  document.getElementById('a').addEventListener('click', function(e) {
    console.log(
      'target:' + e.target.id + '&currentTarget:' + e.currentTarget.id
    )
  })
  document.getElementById('b').addEventListener('click', function(e) {
    console.log(
      'target:' + e.target.id + '&currentTarget:' + e.currentTarget.id
    )
  })
  document.getElementById('c').addEventListener('click', function(e) {
    console.log(
      'target:' + e.target.id + '&currentTarget:' + e.currentTarget.id
    )
  })
  document.getElementById('d').addEventListener('click', function(e) {
    console.log(
      'target:' + e.target.id + '&currentTarget:' + e.currentTarget.id
    )
  })
</script>
Copy the code

When we click on the innermost element D, it will output:

target:d&currentTarget:d
target:d&currentTarget:c
target:d&currentTarget:b
target:d&currentTarget:a
Copy the code

As you can see from the output, event.target refers to the element that triggered the event, and Event. currentTarget is the element bound to the event. That is, event.currenttarget is always the listener of the event, and event.target is the actual sender of the event.

4. Cross-browser event objects

var EventUtil = {
    addHandler: function (el, type, handler) {
        if (el.addEventListener) {
            el.addEventListener(type, handler, false);
        } else if (el.attachEvent) {
            el.attachEvent('on' + type, handler);
        } else {
            el['on'+ type] = handler; }},removeHandler: function (el, type, handler) {
        if (el.removeEventListener) {
            el.removeEventListerner(type, handler, false);
        } else if (el.detachEvent) {
            el.detachEvent('on' + type, handler);
        } else {
            el['on' + type] = null; }},getEvent: function (e) {
        return e ? e : window.event;
    },
    getTarget: function (e) {
        return e.target ? e.target : e.srcElement;
    },
    preventDefault: function (e) {
        if (e.preventDefault) {
            e.preventDefault();
        } else {
            e.returnValue = false; }},stopPropagation: function (e) {
        if (e.stopPropagation) {
            e.stopPropagation();
        } else {
            e.cancelBubble = true; }}};Copy the code

Capture and bubble order problems

When there are multiple layers of interaction nesting, the sequence of event capture and bubbling can seem difficult to determine. Below, we will discuss their order in five different ways, and how to avoid unexpected situations.

1. When an event is registered in the outer div and triggered by clicking on the inner div, the capture event is always triggered before the bubble event (regardless of code order).

Suppose we have an HTML structure like this:

<div id="test" class="test">
   <div id="testInner" class="test-inner"></div>
</div>
Copy the code

We then register two click events on the outer div, a capture event and a bubble event, as follows:

const btn = document.getElementById("test");
 
// Catch the event
btn.addEventListener("click".function(e){
    alert("capture is ok");
}, true);
 
// Bubbling event
btn.addEventListener("click".function(e){
    alert("bubble is ok");
}, false);
Copy the code

Click on the inner div to pop Capture is OK and then bubble is OK. Only when the DOM element that actually fires the event is the inner one, does the outer DOM element have a chance to simulate capturing and bubbling events.

2. When an event is registered on the DOM element that triggered the event, whichever is registered first is executed first

The HTML structure is the same as above, and the JS code is as follows:

const btnInner = document.getElementById("testInner");

// Bubbling event
btnInner.addEventListener("click".function(e){
    alert("bubble is ok");
}, false);
 
// Catch the event
btnInner.addEventListener("click".function(e){
    alert("capture is ok");
}, true);
Copy the code

In this case, the bubbling event is registered first, so it is executed first. So, click on the inner div and bubble is OK pops up, then Capture is OK pops up.

3. When both the inner div and the outer div are registered for capture events, the inner div’s event must fire first when clicked

const btn = document.getElementById("test");
const btnInner = document.getElementById("testInner");

btnInner.addEventListener("click".function(e){
    alert("inner capture is ok");
}, true);

btn.addEventListener("click".function(e){
    alert("outer capture is ok");
}, true);
Copy the code

Events for the outer divs are registered later, but fired first. So, the result is that outer capture is OK and then inner capture is OK.

4. Similarly, when both the outer div and the inner div register bubbling events at the same time, when the inner div is clicked, the inner div event must be triggered first.

const btn = document.getElementById("test");
const btnInner = document.getElementById("testInner");

btn.addEventListener("click".function(e){
    alert("outer bubble is ok");
}, false);

btnInner.addEventListener("click".function(e){
    alert("inner bubble is ok");
}, false);
Copy the code

So inner bubble is OK, outer bubble is OK.

5. Prevent distribution of events

In general, you want to click on a div to trigger only your own event callback. For example, if the inner div is clicked, but the event of the outer div is also triggered, this is not what we want. At this point, you need to prevent the dispatch of events.

When the event is triggered, an event object is passed in by default. This event object has a method: stopPropagation. Prevent further propagation of the current event in the capture and bubble phases. So, with this method, the outer div will not receive the event and will not fire.

btnInner.addEventListener("click".function(e){
    // Prevent bubbling
    e.stopPropagation();
    alert("inner bubble is ok");
}, false);
Copy the code

Refer to the article

  1. JavaScript Advanced Programming (3rd Edition)
  2. The difference between event.target and event.currentTarget
  3. JS events: capture and bubble, event handlers, event objects, cross-browser, event delegates
  4. Javascript event flow
  5. Event Mechanisms in JavaScript (from native to Framework)
  6. DOM event mechanism