At Kijiji, we’ve been using Optimizely extensively for traditional A/B testing, building different UI variations and measuring which one performs better.Recently, we started using this tool in a different way, in order to gather qualitative feedback. The idea is extremely simple: ask directly to our users what they think of something on our site, or something that we could be adding to it.

Here is an example. We wanted to know whether our users would appreciate receiving real-time feedback about the quality of their ad, as they type it. We could have sent out a survey by email, or include this question into our user testing sessions. But what better way than asking people directly, precisely where and when it matters to them?

Title Quality Satisfaction Survey

All we had to do was build a little tooltip with our question and measure the interest with custom events.

We wanted to use smileys for users to click and share their satisfaction (happy, neutral or angry) but Optimizely doesn’t currently support uploading static assets. An easy work-around was to encode our pictures into base64 and insert them with javascript.

Here are our 3 little icons:

var angry = "";
var happy = "";
var neutral = "";

Using these icons to track votes is done by adding an onclick binding that records the custom event, like this:

window.satisfactionVote = function(e) {
  e.preventDefault();
  var eventName = 'titleQuality_' + e.target.id;
  window.optimizely.push(["trackEvent", eventName]);
};
var happyIcon = "<img id='happy' onclick='satisfactionVote(event)' style='cursor:pointer;' src='" + happy + "'/>";

While we’re not looking for scientific precision, it’s probably better to get each person’s vote only once. Optimizely shows results for unique visitors by default, but another consideration is to provide the best possible user experience and get the survey out of the way as soon as a user replied.

In order to do that, we can simply store a flag in the browser’s local storage. When a user votes, we set the flag to 1 and hide the survey. Afterward, we don’t show the survey again if the flag is already set.

Putting everything together, the code of our Optimizely variation looks something like this:

window.satisfactionVote = function(e) {
  e.preventDefault();
  var eventName = 'titleQuality_' + e.target.id;
  window.optimizely.push(["trackEvent", eventName]);
  localStorage.setItem('titleQuality', 1);
  // hide the survey
  document.getElementById("satisfaction").style.display = "none";
};

function buildSurvey() {
  var angry = "";
  var happy = "";
  var neutral = "";

  var survey = "<div id='satisfaction'>";
  survey += "<strong>We're working on providing feedback on the quality of your title.</strong><br/><br/>";
  survey += "Would you like this?";
  survey += "<img id='happy' onclick='satisfactionVote(event)' style='cursor:pointer;' src='" + happy + "'/>";
  survey += "<img id='neutral' onclick='satisfactionVote(event)' style='cursor:pointer;' src='" + neutral + "'/>";
  survey += "<img id='angry' onclick='satisfactionVote(event)' style='cursor:pointer;' src='" + angry + "'/>";
  survey += "</div>";
  return survey;
}

function initTitleQuality() {
  if(typeof(Storage) === "undefined") {
    // local storage unsupported, stop there
    return;
  }
  if (localStorage.getItem('titleQuality')) {
    // user already voted
    return;
  }
  // insert the survey on the page
  $('#title-field').before(buildSurvey());
}

initTitleQuality();

After this, we just have to run the experiment and the results look something like that:

Title Quality Satisfaction Survey

We could stop there, but leaving some clutter in our users browser doesn’t seem like a good habit. So, when we’re done with the experiment, we can reuse it to clean-up after ourselves and remove the flag from the local storage. Since any returning user who had voted will still be targeted, we can just change the code to remove the flag and keep the experiment running for a few more days. We can even track the progress of this clean-up process with an additional goal:

if (localStorage.getItem('titleQuality')) {
  localStorage.removeItem('titleQuality');
  window.optimizely.push(["trackEvent", 'titleQuality_remove']);
}

Happy discovery!