All posts by Lachlan Hunt

DOM 2 Events

The Document Object Model Level 2 defines several interfaces for an event handling model, called DOM 2 Events, that is far superior to the traditional model used by most web developers around the world. In the traditional model, you assign an event handler to an object using the onevent properties. e.g. window.onload = fn;. This has many limitations, with the most annoying being that it only accepts one event handler.

In the past, many attempts have been made to work around these issues, with one of the most widely known being Scott Andrew LePera’s Crossbrowser DOM Scripting: Event Handlers. Recently, Peter-Paul Koch wrote an article entitled addEvent() Considered Harmful. Following this, he launched a competition for someone to rewrite the addEvent() function and write a new removeEvent() function to address the issues with the original.

A few days ago, after I heard about addEvent(), I began work on a new script to address many of the issues with it. The result is beyond anything that was expected for the competition. In fact, I’m pretty sure it’s beyond what most mere mortals even considered possible: I implemented much of the DOM 2 Events interfaces in IE!

What does that mean exactly? Well, it means that you can now use the W3C’s superior DOM2 model across browsers without any of the mess. For example, a script that used to be written like this:

var foo = document.getElementById("foo");
foo.onclick = go;

function go(evt) {
    evt = evt || window.event;
    target = evt.target || evt.srcElement;
    // Handle other cross-browser issues
    doSomething(target);
    ...
}

That can now be reduced to this much more sane method from the DOM2 interfaces:

var foo = document.getElementById("foo");
foo.addEventListener("click", go, false);

function go(evt) {
    doSomething(evt.target);
    ...
}

There’s no need to check for evt || window.event, nor any need to check for evt.target || evt.srcElement. Much of that cross-browser incompatibility mess is taken care of by my script. All that you, the authors, have to do is work with the standard DOM 2 interfaces! You can take a look at the heavily commented script and also my entry for PPK’s competition, which nicely demonstrates the script in action.

The following interfaces, properties and functions are supported from DOM 2 events:

Interface EventTarget
addEventListener
dispatchEvent
removeEventListener
Interface EventListener
handleEvent
Interface Event
PhaseType Constants (CAPTURE_PHASE, AT_TARGET and BUBBLE_PHASE)
bubbles
cancelable
currentTarget
eventPhase
target
timeStamp
type
initEvent() (not supported)
preventDefault()
stopPropagation()

Along with these event interfaces, there are a number of other interfaces I implemented including a fully conforming .getElementsByClassName() function, as defined in the Web Apps 1 for both the Document and Element interfaces, a prototype implementation of DOMTokenString() and many more, all availble in my DOM example script repositiory.

A Month in Review

Hi everybody. As this month draws to a close, I realise it’s been about 45 days since I posted last. Enough is enough; it’s time I get this blog going again. So, I figured I’d kick it off again with a review of my activities over the last month and a half, and hopefully explain where I’ve been all this time.

In the second last week of July, from the 18th to the 22nd, I enjoyed a wonderful week of skiing the slopes at Perisher. It’d been 2 years since I’d skied last, but after just a bit of a wobbly start, I was up and going again like a pro! Well, not quite; but I did manage to make my way down most of the blue and black runs, even if it did involve a fair bit of falling.

As well as skiing, I managed to build my first snowman. It was quite an achievement for a first effort, at just over 1/2 a metre tall (approx. 2 feet). This was later followed by a second attempt at the end of the week, over twice the height of the first. I’d post some photos to show you, but I don’t have them with me; maybe another time, if I remember.

Following the ski trip, I finally made an appointment to get my eyes tested. I figured, since I was having trouble reading the menu board at McDonalds, it was time to get them checked. Well, it turned out I needed glasses and I am really surprised at the improved eye sight they’ve given me. Watching TV no longer involves moving closer and/or squinting at the screen and even looking at a computer monitor all day is easier.

It’s been about 2 months since I started working full time at HotHouse as a web developer, primarily writing HTML, CSS and occasionally, a little bit of JavaScript. I am enjoying working there a lot and, in fact, the first site I built from scratch while working there will be launching very shortly; I’ll give more details when it does.

Over the past 2 weeks, I’ve been putting in a lot of effort toward a more personal project involving a lot of JavaScript. I’m actually helping out Eric Meyer by going through his world famous S5 script, cleaning it up, pulling it apart, rewriting and restructuring entire sections of it making it completely object-oriented.

You can take a look at my current work, browse through the scripts and see a functioning Slide() object that can show, hide, increment and decrement slides. It’s been tested in Firefox, Opera and IE. Let me know if you find any issues with any other browser, particularly Mac and Linux browsers. If you can suggest a fix for any problems, that would be even better.

One of the design goals of this project is to fix all the issues that prevent the script from functioning in Opera without having to fallback to OperaShow, as well as making it more modular and easier to extend with new features and functionality. Another design goal is to actually make the script work for real XHTML served as XML (not just tag-soup served as text/html), with namespace support (including mixed namespace documents).

In fact, the document.getElementsByClassName() and related functions I wrote for this already support XHTML, SVG and MathML namespaces. (There are some issues with the Element.addClassName() and Element.removeClassName() functions relating to the case-insensitivity of HTML and case-sensitivity of XML, but I’ll write and publish the fixes for those later – it’s easy to fix, I just have to find the time to do it).

table-layout

Not to be confused with layout tables, this little used, yet wonderful CSS property ‘table-layout’ could have saved me a lot of time recently, had I thought of it. You see, the challenge I had was that I needed all columns of a table (containing tabular data) to be presented with equal width columns.

My immediate thought was to use col and specify widths in percentages. However, the difficulty was that the number of columns varied from 3 to 5 so I couldn’t just specify one single percentage, since the percentage would vary from approximately 33.3% for 3 columns to 20% for 5 columns.

Being a little rushed and not actually stopping to think, I decided that my only solution would be to specify a presentational class for the table element that indicated the number of columns, and then use that to select and style the col elements with the appropriate width. It wasn’t till a couple of days later that the better solution presented itself, which is ironic since I have, for about a week beforehand, been going over the CSS 2.1 tables module with a fine tooth comb, writing test cases for it and contributing them to the CSS 2.1 test suite.

The fixed table layout algorithm (CSS 2.1, section 17.5.2.1) defines in step 3:

Any remaining columns equally divide the remaining horizontal table space (minus borders or cell spacing).

Which means, if I hadn’t specified widths for any columns, nor any cell in the first row (as described in the rest of the algorithm), all the columns would be set to equal width. Oh, and by the way: Yes, this property is (miraculously) supported by IE! Now that I know how this property works, I’m sure I’ll be making a lot more use of it.