JavaScript Ninja – NamedNodeMap, But Flat!

Given the following HTML-code, describing some sort of an element (a lazy-load image in that case..)

<img class="w" data-frz-src="/th_B2BB13CEF623A7F1D2F82B8D2014C030FFD3128A_4.jpg" src="" onload="lzld(this)" onerror="lzld(this)">

Serialising or looping through its attributes, although feasible, will be quite a pain, handling with NamedNodeMap unnesessary-complicated object structure.


This code:

NamedNodeMap.prototype.map = Array.prototype.map;

var a = document.querySelectorAll('img[src^="data"][src*="base64"]')[0];

a.attributes.map(function(attribute){
  return {name: attribute.nodeName, value:attribute.nodeValue};
});

Will lower-the-lovel of complexity, and will give you an array of name,value objects (much less pain in the ass to serialise, or loop through), but it is not very useful if you are looking for something specific..

[{
    "name": "class"
  , "value": "w"
}, {
    "name": "data-frz-src"
  , "value": "/th_B2BB13CEF623A7F1D2F82B8D2014C030FFD3128A_0.jpg"
}, {
    "name": "src"
  , "value": ""
}, {
    "name": "onload"
  , "value": "lzld(this)"
}, {
    "name": "onerror"
  , "value": "lzld(this)"
}]

This code, will flat the attributes to something more readable,
a plain key,value object, where (matching the HTML logic anyway..) each key
matches to a unique attribute-name – and its value.

NamedNodeMap.prototype.reduce = Array.prototype.reduce;

var a = document.querySelectorAll('img[src^="data"][src*="base64"]')[0];

a.attributes.reduce(function(structure, attribute){
  structure[attribute.nodeName] = attribute.nodeValue;
  return structure;
},{});
{
    "class": "w"
  , "data-frz-src": "/th_B2BB13CEF623A7F1D2F82B8D2014C030FFD3128A_0.jpg"
  , "src": ""
  , "onload": "lzld(this)"
  , "onerror": "lzld(this)"
}

An ultimate way of just allowing to get at any given time the key/value object (above),
is by adding a prototype method, into NodeHTMLElement:

HTMLElement.prototype.getAttributes = function(){
  var me = this;
  return Array.prototype.map.call(me.attributes, function(attribute){
    return {name: attribute.nodeName, value:attribute.nodeValue};
  });
})

So you could just do a ...querySelector(...).getAttributes(),
to get just the array of the keys, simply go with Object.keys( ...querySelector(...).getAttributes() ).