Handling Events for Many Elements

Not exactly the same. clicker in your example is still being treated as an event handler - something specific to events, accepting a single argument that is from an event. It doesn’t work well in contexts other than being an event handler. updateFrom is more generic. It can be called anywhere at any time and doesn’t rely on an event. This makes it more versatile than any other, ordinary event handler.

That said, in onclick attributes on html, you’re just running arbitrary code that is wrapped in an event handler. So technically clicker doesn’t have to be an event handler. It can ignore the event argument and just be a generic function too. But most of the time when you’re writing attribute code like that, its in the form of a handler specific to that attribute (e.g. a click handler for the onclick event)

2 Likes

By the way, something interesting you can do is write an attribute handler and then check the handler property of that element in javascript. For example:

<a onclick="alert(this.nodeName)">Click Me</a>
var a = document.getElementsByTagName('a')[0];
console.log(a.onclick); // =>
/*
function onclick(event) {
  alert(this.nodeName)
}
*/

Here you can see the wrapper handler that goes around the code that was put in the onclick attribute. Its a single-argument function with a parameter named event. That function is also called in the context of that element so this in that call is the element itself.

1 Like

Nice to see you joined kforums @joe_hoeller, I remember when I would look @senocular code back in the 90s and think how stupid I was. Nothing has changed LOL

3 Likes

@grimdeath - one of my most favoritest people! :love: :love: :love:

The feelings mutual master :yoda:

Group hug!!! :hamburger:

I’m still confused. What if I have different buttons/links with different parents throughout my entire site that each have a different HTML class/id, but all use the same event listener. Would this code still work?

Example:
var button1 = document.querySelector(’#parent1 .button’).addEventListener…;
var button2 = document.querySelector(’#parent2 .button’).addEventListener…;
var button3 = document.querySelector(’#parent3 .button’).addEventListener…;

I don’t have access to modifying the HTML either.

@TallGlassOfMike That still works. The approach described in this tutorial doesn’t work as well in that case, so maybe you’ll want to assign them individually like that.

Otherwise it would mean a more complicated filter. In using a single event listener, you need to be able to tell that what you clicked was what you wanted to be clicked. This was what if (e.target !== e.currentTarget) { ... } was doing - making sure that clicks that were made on #theDude (e.currentTarget) but not any of the one, two, three, four, or five buttons (e.target) were getting ignored. If the background of #theDude was clicked, for example, then e.target === e.currentTarget would be true.

If you have multiple buttons in different hierarchies, then the single handler would have to be applied to the common parent of those buttons. In a worst case scenario, that would be the body element. But then you would need to check e.target for basically every click made in the entire document to see if it’s one of your buttons. The check for this wouldn’t actually be too hard if you had a list of selectors targeting your buttons. It could be as simple as:

document.body.addEventListener('click', function (e) {
    if (e.target.matches('#parent1 .button, #parent2 .button, #parent3 .button')) {
        // do something with your clicked button
    }
});

But there is the downside of this check happening for every click in the common parent. If you’re ok with that (does one if check really matter that much?) then this could work for you. One event handler is a lot easier to manage than multiple event handlers and it means the dom can be changed and have .button elements added and removed and re-added without affecting the click handler at all.

1 Like

Excellent post, thanks! How would this be done for contenteditable div tags? It looks like this:

<div id="theParent" contenteditable="true"></div>
    <div class="el"...></div>
    <div class="el"...></div>
    ...

With JavaScript:

var e = document.querySelector("#theParent");
e.addEventListener("input", doSomething, false);

When I run this, it returns e.target as “theParent”. I tried adding contenteditable to each of the inner div tags as well, which didn’t work. Would greatly appreciate any help!

If you replace the input event with click, then the appropriate element you clicked on works - even if the parent is a contenteditable div element:

<body>
  <div id="theParent" contenteditable="true">
    <div id="foo" class="el">A</div>
    <div id="bar" class="el">B</div>
  </div>

  <script>
    var theParent = document.querySelector("#theParent");
    theParent.addEventListener("click", doSomething, false);

    function doSomething(e) {
      if (e.target !== e.currentTarget) {
          var clickedItem = e.target.id;
          console.log("Hello " + clickedItem);
      }
      e.stopPropagation();
    }
  </script>

The input event is a bit bizarre in how it behaves. Can you share more on what you are trying to do?

Thanks,
Kirupa :nerd_face:

Thanks for the suggestion, although click won’t work for this use case. I’m adding paragraphs within a larger text window. Each paragraph is within a div tag, and the entire text window is inside the parent div. Every time the user enters new text, the event listener fires only on the specific div paragraph

Can you provide a link to your page or a smaller repro? This seems like a very reasonable scenario, so there has to be some magical approach that works. I’d love to fiddle with it for a bit :stuck_out_tongue:

1 Like

I am making a calculator pls how can i apply this to mine?Your explanation was legit but i learn slow

2018-05-26_21h22_04

Can you be a bit more specific in what you’d like help with? :slight_smile:

Yes sure. Sorry I was very frustrate at the time I sent my last message. A building a calculator and aim is quite simple. I want this to have keys from 1-9 including 0 of course. So when clicked, that is any of the buttons like 3 it display in a screen (a div with light background) so as my of picture shows i have gotten the keys in a variable with a query selector. So should I add an event listener one by one on each element or is there an easier way.

You can add an event listener to the parent of the keys like the article describes. Then you can then check for which key triggered the event. That would be the cleaner way to do it :slight_smile:

**the innerHTML value not saved whene i input value in another input lebel **
function formButtonClicked (){

var twoElement   = document.querySelector("#cm_name");
var threeElement = document.querySelector("#cm_phone");
var fourElement  = document.querySelector("#cm_adress");
var fiveElement  = document.querySelector("#car_type");
var sixElement   = document.querySelector("#car_color");
var sevenElement = document.querySelector("#car_plate");
var eightElement = document.querySelector("#car_shasy");


twoElement.addEventListener(  'input', doSomething, false);
threeElement.addEventListener('input', doSomething, false);
fourElement.addEventListener( 'input', doSomething, false);
fiveElement.addEventListener( 'input', doSomething, false);
sixElement.addEventListener(  'input', doSomething, false);
sevenElement.addEventListener('input', doSomething, false);
eightElement.addEventListener('input', doSomething, false);

function doSomething(e) {
 var clickedItem = e.target.value;
 var inputid=e.target.id;
switch(inputid ){
    case  "cm_name"  :  document.querySelector('.div_cm_name' ).innerHTML=clickedItem;   break;
    case  "cm_phone" :  document.querySelector('.div_cm_phone').innerHTML=clickedItem;   break;
    case  "cm_adress":  document.querySelector('.div_cm_adress').innerHTML=clickedItem;   break;
    case  "car_type" :  document.querySelector('.div_car_type').innerHTML=clickedItem;   break;
    case  "car_color":  document.querySelector('.div_car_color').innerHTML=clickedItem;   break;
    case  "car_plate":  document.querySelector('.div_car_plate').innerHTML=clickedItem;   break;
    case  "car_shasy":  document.querySelector('.div_car_shasy').innerHTML=clickedItem;   break;
  
     default:
console.log("the id of the form input is: " + inputid);
}

}}

What are you trying to do exactly? The code snippet you posted doesn’t seem to fully help :slight_smile:

6 posts were split to a new topic: Picking Random Item in a Multi-dimensional Array

@v_vendetta - I moved your question to the following thread, for it isn’t related to event handling and multiple events: Picking Random Item in a Multi-dimensional Array

:slight_smile: