Track file downloads with Google's Universal Analytics

Introduction

I wanted to track which version of the Radial Menu mixin was more popular. Since I already use Google Analytics for the page views it was logical to use it to track the file downloads. It took a couple of tries and wading through some poor examples on various websites before I had something that worked for me. In this article I will explain how I am counting file downloads using Google Analytics events.

The basics

In theory it is a simple exercise, especially if you already have Google Analytics setup for your website. It should be as easy as adding some code in de the onclick handler of your anchor element and calling the ga method.


<a onclick="ga('send', 'event', 'category', 'download', 'label', 'value');" href="/static/articles/radial-menu/radial-menu-less.zip">LESS version of the mixin</a>
        
This doesn't work

That was my first attempt, it failed miserably. Part of it was my fault for not reading Google's track event documentation properly. The value of the last parameter has to be a number and I was passing a string.

The other problem was obvious once I opened the network tab of the Google Chrome debugger. My track event was canceled, according to the debugger so was the request for the zip file. But unlike the zip file which was waiting for me in the download folder, Google Analytics never received the event.

The Google Chrome debugger showing the canceled network request

The onclick handler would have worked if the anchor had had the attribute target="_blank". The file download would have taken place in the new tab or window and the Google Analytics event would've been sent from the existing tab. I think this is not a very nice solution, I don't want to burden my visitors with an empty tab they didn't ask for. Time to try something else.

The backwards solution

I started looking around what other people did to track file downloads on anchor elements that don't open a new tab or window. If you search on Google for "track file downloads", one of the first results is How To Track File Downloads & Outbound Links in Google Analytics by Ryan Chase. It has a pretty solid solution for a lot of different situations, I figured this would be a good place to get started.

One of the things I didn't like about it was the usage of jQuery. I don't use jQuery on my site and I wasn't about to use it just so I could track an event in Google Analytics. The other problem was I couldn't get it to work properly. I boiled Ryan's solution down to the following:


(function() {
    'use strict';

    // Get all anchor elements that link to a zip file, these are the links I want to track
    var links = document.querySelectorAll('[href$=".zip"]');
    // Loop over all the elements we've found
    for (var index=0,ubound=links.length; index<ubound; index++) {
        // Attach an event handler on the click event
        links[index].addEventListener('click', handleClickEvent.bind(this));
    }

    function handleClickEvent(event) {
        // Send the event to Google Analytics
        ga.send('send', 'event', 'link', 'download', 'label', 0);
        // Set a timeout of 400ms, after the time out has lapsed we will start the download of
        // the file
        setTimeout(function() {
            location.href = event.target.href;
        }, 400);
        // Prevent the default behaviour of the link, we prioritize sending the event to
        // Google.
        event.preventDefault();
    }
})();
            
Stripped down version of Ryan's code

The magic happens in the setTimout part of the code. By waiting 400ms before sending the event we give the system enough time to send out the Google Analytics event before we make the request to download the zip file. I am not really sure where the 400ms comes from, I haven't been able to find any articles that explain why this value should be used. Perhaps a short time out will work just as well.

Because I only want to track an event for my download links I could get away by just attaching click handlers to the links whose href ends on ".zip". The other big change I had to make was to replace the return false at the end of the event handler with event.preventDefault(). When returning the value false the zip file downloaded twice, it didn't cancel out the default behavior of the anchor element.

There are plenty more examples you can find on-line. All the articles I came across pretty much did the same thing. Attach an event handler to the link, send out the event, set a time out to download the file and cancel the default behavior of the anchor element. When you think about it for a while it is kind of crazy, you're preventing the default behavior of the anchor only to recreate it a little later.

Fixing it the right way round

I think downloading the file should be the default action of the anchor element, sending out the event is of secondary concern. By prioritizing the file download the code for sending out the event became a little simpler.


(function() {
    'use strict';

    // Get all anchor elements that link to a zip file, these are the links I want to track
    var links = document.querySelectorAll('[href$=".zip"]');
    // Loop over all the elements we've found
    for (var index=0,ubound=links.length; index<ubound; index++) {
        // Attach an event handler on the click event
        links[index].addEventListener('click', handleClickEvent.bind(this));
    }

    function handleClickEvent(event) {
        // Set a timeout of 400ms, after the time out has lapsed we will send the event
        // to Google
        setTimeout(function() {
            // Send the event to Google Analytics
            ga.send('send', 'event', 'link', 'download', 'label', 0);
        }, 400);
    }
})();
            
Download first, Google Analytics event second

This way we don't have to prevent the default behavior of the anchor element and just wait 400ms before sending out the event to Google.

Now we can see the completed Google Analytics event request

Make it configurable

The code above still has one major shortcoming, it sends out the same data for all anchor elements we've attached an event handler to. To make it usable we need to be able to specify which data should be send with the event. In order to do this we're going to add a custom attribute to the anchor elements that should trigger a Google Analytics event.


<a data-ga="{'action':'download','category':'radial-menu','label':'LESS version','value':'1'}" href="/static/articles/radial-menu/radial-menu-less.zip">LESS version of the mixin</a>
                
The custom data attribute

We will parse the value of the data-ga attribute as JSON. The value as it is above is not valid JSON, for that the property names and values should be wrapped with double quotes instead of single quotes. Instead of using single quotes for the HTML attribute or escaping the double quotes in the string. I prefer to replace the single quotes with double quotes in JavaScript, this way the HTML has a uniform way of using quotes which I think is a plus.

Once the string is parsed as JSON we will have an object with the properties action, category, label and value. We can use these properties in our JavaScript to replace the values we send with the event. What we need to do now is change the JavaScript to use the data-ga property.


(function() {
    'use strict';

    // Get all anchor elements that have information for a Google Analytics event in the data-ga atttibute
    var links = document.querySelectorAll('[data-ga]');
    // Loop over all the elements we've found
    for (var index=0,ubound=links.length; index<ubound; index++) {
        // Attach an event handler on the click event
        links[index].addEventListener('click', handleClickEvent.bind(this));
    }

    function handleClickEvent(event) {
        // Get the content from the data-ga attribute from the element that send the click event
        var data = event.currentTarget.getAttribute('data-ga');
        // We'll be parsing the content of the attribute as JSON, in order to be able
        // to do this we need to make sure all single quotes in the string are double quotes.
        // This regular expression replaces all instances of a single quote
        data = data.replace(/\'/g, '"');
        // Now we can parse the string to an object. If you're not sure this attribute will
        // always contain a string which can be parsed as JSON you should add a try...catch
        // around this so your code won't crash
        try {
            data = JSON.parse(data);
            // Set a timeout of 400ms, after the time out has lapsed we will send the event
            // to Google
            setTimeout(function() {
                // Send the event to Google Analytics. We use the properties from the object we've parsed
                // from the attribute. Be sure to pass the value parameter as a number
                ga.send('send', 'event', data.category, data.action, data.label, parseInt(data.value, 10));
            }, 400);
        } catch (exception) {
            console.log('Unable to parse the string <' + data + '> as JSON');
        }
    }
})();
                
Using the data-ga attribute

Now when a visitor clicks on the link to download a file the following information will show up in Google Analytics.

The event shows up in Google Analytics

The result

We now have a small, light weight, piece of JavaScript which we can use to send events to Google Analytics. It can be configured per anchor element and it prioritizes the file download over sending the event to Google.

The JavaScript code is compatible with Internet Explorer 9 and up due to the usage of addEventListener. The code could easily be made compatible with Internet Explorer 8 by loading something like the addEventListener polyfill on the Mozilla Developer Network.