Fragmented Thought

UX - Autocomplete indicators vs IE & Edge clear control

By

Published:

Lance Gliser

Heads up! This content is more than six months old. Take some time to verify everything still works as expected.

This came as a shock to me, likely it may to you as well. IE & Edge have produced a new non-standard feature! A little button to clear text entries inside fields. It's actually handy, I applaud them. Sadly it does cause me problems. As designers, our goal is to provide systematic, effective hints. If you're working in a large scale application, there may be some times when search is a wildcard partial text match, and other times when a real JavaScript autocomplete is required. Properly signaling which is which ahead of time reduces mental friction, and sets the user up to appropriately use the search. Normally, we like to do that using hint images. Maybe a 'search' icon, and one of many famous loading spinners during network requests.

Unfortunately, the combination of MS's 'Clear' and autocomplete controls results in... overlap. It's just plain ugly.

Other browsers have not brought that feature up, though they may someday. For now though, we need handling providing the base hint, the loading spinner, and a special rule or two to remove those in IE & Edge when the user is focused on the field. To that end, you'll need two pieces:

  • Javascript to add the proper classes (base, loading, and empty)
  • CSS to target the same states above

Providing an autocomplete script is beyond the scope of this post, but I'll provide the little bits you need to know about.

// Add the initial class element.classList.add("autocomplete"); // Read the empty state initially updateEmptyClass(element); // Monitor changes to the value - This requires use of the 'input' event, the only event that fires when IE uses the clear control element.addEventListener("input", function (event) { updateEmptyClass(event.target); }); // Add some other handling to manage adding .autocomplete-loading /** * @param {HTMLInputElement} element */ function updateEmptyClass(element) { var className = "autocomplete-empty"; if (!element.value) { element.classList.add(className); return; } element.classList.remove(className); }
.autocomplete { background-image: url(); background-position: right 1rem center; background-repeat: no-repeat; background-size: 1.25em; } .autocomplete-loading { background-image: url(); } /** * IE10+ CSS styles go here * Using both version of an IE specific selector to target the browser */ @media all and (-ms-high-contrast: none), (-ms-high-contrast: active) { /* Hide the search icon, IE shows the "clear" when focusing * / .autocomplete:not(.autocomplete-loading):not(.autocomplete-empty):focus { background-image: inherit; } .autocomplete-loading::-ms-clear { display: none; } } /* Edge CSS styles go here */ @supports (-ms-ime-align: auto) { /* Hide the search icon, IE shows the "clear" when focusing */ .autocomplete:not(.autocomplete-loading):not(.autocomplete-empty):focus { background-image: inherit; } .autocomplete-loading::-ms-clear { display: none; } }