Working on an idea for a simple HTMLElement
wrapper I stumbled upon the following for Internet Explorer and Chrome:
For a given HTMLElement
with an id
in the DOM tree, it is possible to retrieve the <div>
using its ID as a variable name or as a property of window
. So for a <div>
like
<div id="example">some text</div>
in Internet Explorer 8 and Chrome you can do:
alert(example.innerHTML); // Alerts "some text".
or
alert(window["example"].innerHTML); // Alerts "some text".
So, does this mean every element in the DOM tree is converted to a property on the global object? And does it also mean one can use this as a replacement for the getElementById
method in these browsers?
A variable that is global means it is accessible everywhere in the JavaScript code. Function Scope: Variables declared within a JavaScript function, becomes LOCAL to the function. A variable that is local can only be accessed from within the function.
OK, there is no ID associated to DOM element automatically. DOM has a hierarchycal structure of elements which is the main information. From this perspective, you can associate data to DOM elements with jQuery or jQLite. It can solve some issues when you have to bind custom data to elements.
In JavaScript, you can almost select any element from the DOM based on its unique ID by using the getElementById() method. It returns the first element that matches the given ID, or null if no matching element was found in the document.
domnodeelement. 10 Comments. The Document Object Model (DOM) is an interface that treats HTML or XML document as a tree structure, where each node is an object of the document. DOM also provides a set of methods to query the tree, alter the structure, style.
What is supposed to happen is that ‘named elements’ are added as apparent properties of the document
object. This is a really bad idea, as it allows element names to clash with real properties of document
.
IE made the situation worse by also adding named elements as properties of the window
object. This is doubly bad in that now you have to avoid naming your elements after any member of either the document
or the window
object you (or any other library code in your project) might want to use.
It also means that these elements are visible as global-like variables. Luckily in this case any real global var
or function
declarations in your code shadow them, so you don't need to worry so much about naming here, but if you try to do an assignment to a global variable with a clashing name and you forget to declare it var
, you'll get an error in IE as it tries to assign the value to the element itself.
It's generally considered bad practice to omit var
, as well as to rely on named elements being visible on window
or as globals. Stick to document.getElementById
, which is more widely-supported and less ambiguous. You can write a trivial wrapper function with a shorter name if you don't like the typing. Either way, there's no point in using an id-to-element lookup cache, because browsers typically optimise the getElementById
call to use a quick lookup anyway; all you get is problems when elements change id
or are added/removed from the document.
Opera copied IE, then WebKit joined in, and now both the previously-unstandardised practice of putting named elements on document
properties, and the previously-IE-only practice of putting them on window
are being standardised by HTML5, whose approach is to document and standardise every terrible practice inflicted on us by browser authors, making them part of the web forever. So Firefox 4 will also support this.
What are ‘named elements’? Anything with an id
, and anything with a name
being used for ‘identifying’ purposes: that is, forms, images, anchors and a few others, but not other unrelated instances of a name
attribute, like control-names in form input fields, parameter names in <param>
or metadata type in <meta>
. ‘Identifying’ name
s are the ones that should be avoided in favour of id
.
As mentioned in the earlier answer this behavior is known as named access on the window object. The value of the name
attribute for some elements and the value of the id
attribute for all elements are made available as properties of the global window
object. These are known as named elements. Since window
is the global object in the browser, each named element will be accessible as a global variable.
This was originally added by Internet Explorer and eventually was implemented by all other browsers simply for compatibility with sites that are dependent on this behavior. Interestingly, Gecko (Firefox's rendering engine) chose to implement this in quirks mode only, whereas other rendering engines left it on in standards mode.
However, as of Firefox 14, Firefox now supports named access on the window
object in standards mode as well. Why did they change this? Turns out there's still a lot of sites that rely on this functionality in standards mode. Microsoft even released a marketing demo that did, preventing the demo from working in Firefox.
Webkit has recently considered the opposite, relegating named access on the window
object to quirks mode only. They decided against it by the same reasoning as Gecko.
So… crazy as it seems this behavior is now technically safe to use in the latest version of all major browsers in standards mode. But while named access can seem somewhat convenient , it should not be used.
Why? A lot of the reasoning can be summed up in this article about why global variables are bad. Simply put, having a bunch of extra global variables leads to more bugs. Let's say you accidentally type the name of a var
and happen to type an id
of a DOM node, SURPRISE!
Additionally, despite being standardized there are still quite a few discrepancies in browser's implementations of named access.
name
attribute accessible for form elements (input, select, etc).<a>
tags accessible via their name
attribute.And I'm sure there's more if you try using named access on edge cases.
As mentioned in other answers use document.getElementById
to get a reference to a DOM node by its id
. If you need to get a reference to a node by its name
attribute use document.querySelectorAll
.
Please, please do not propagate this problem by using named access in your site. So many web developers have wasted time trying to track down this magical behavior. We really need to take action and get rendering engines to turn named access off in standards mode. In the short term it will break some sites doing bad things, but in the long run it'll help move the web forward.
If you're interested I talk about this in more detail on my blog - https://www.tjvantoll.com/2012/07/19/dom-element-references-as-global-variables/.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With