Skip to search Skip to main content

Does the <body> rule the mind or does the mind rule the <body>?

A styled accessible file upload

Posted

Updated

I could have been wild and I could have been free
But nature played this trick on me

The Smiths, “Pretty Girls Make Graves”

To style especially the file upload is a real complex issue and you might stumble upon things like the Shadow DOM or Web Components, stuff that a regular frontend guy like me doesn’t want to be bothered with. But in the world of HTML5 and CSS3 we have been given enough power to get that fixed.

TL;DR

See the final result here with the working code examples.

First Steps

To enable upload files to your application or website you basically need three HTML elements: a <form>, a <label> and an <input type="file">.

I would like to start out with some basic HTML code:

<form action="#" enctype="multipart/form-data">
  <label for="fileupload">upload file</label>
  <input id="fileupload" type="file" />
</form>

But as we all know, this looks quite lame in all browsers and it’s look is very dependent on what operating system you are using.

Screenshots of file upload fields on different browsers
The current state of native file uploads: screenshots of a basic input in different browsers of today

Styling the <input type="file"> is quite painful. There is a nice (but not too new) trick to get that done:

#fileupload {
  display: none;
}
.btn {
  font-family: Arial;
  font-size: 1rem;
  padding: 8px 8px;
  border-radius: 5px;
  border: 2px solid #265986;  
  color: #ffffff;
  cursor: pointer;
  background-color: #265986;
}
.btn:hover {
  box-shadow: 0 0 5px #595959;
  background-color: #183a57;
  border-color: #265986;  
}

What I did is just hide the <input> and style it’s corresponding <label> that it looks like a button. And the upload functionality still works pretty fine, maybe even in IE6 (not tested yet, though). That is, because it’s defined in the web standards, that a click on the <label> will activate the corresponding <input> element.

But is that accessible? Short answer: no. Especially for those reasons:

  1. Missing semantics: By just styling it you only visually make the label a button, but for users of assistive technologies it’s still a label. And it’s not obvious, that you can handle a hidden input element with that.
  2. No keyboard accessibility: Most users of screenreaders use the keyboard to navigate. But you can not focus the label by using your keyboard. You can only activate the input element with the mouse or with your fingers in touch devices.

So how do we make this a real button?

And by “real” I mean a button that is accessible for all users out there.

Let’s pour some semantic sugar on top

I put a <span> element into the <label> element and set role="button" on it. That’s just to provide correct valid HTML markup, because ARIA roles would not be valid on the <label> element.

Now a screenreader would read something like “button: upload file” here, just like on a regular button. With the addition of aria-controls="fileupload" we make sure, that assistive technologies know, what element is controlled by the button.

Add some web standards know how

To provide keyboard accessibility I set tabindex="0" on the <span> element. This makes the button focussable. Now you can tab on my fake button. Additionally I set the same behaviour as on :hover to the focus state to provide stronger visibility for keyboard users.

<label for="fileupload" id="buttonlabel">
  <span role="button" aria-controls="filename" tabindex="0">upload file</span>
</label>

But still, I can not use the button with the keyboard properly. A button usally reacts on special keys pressed when it’s focussed. Here we need to provide the same functionality for enter and space key.

Some Javascript (jQuery) magic

I bind the keypress and keyup events for the “enter” (13) and the “space” (32) key on it. To activate the hidden upload <input> element, I set a click on it, when one of the keys is pressed.

$('#buttonlabel span[role=button]').bind('keypress keyup', function(e) {
  if(e.which === 32 || e.which === 13){
    e.preventDefault();
    $('#fileupload').click();
  }    
});

Now the button reacts like a real button, for screenreader users and for “regular” users.

Pimping the user experience

The raw HTML example above had some very helpful usability advantages: after selecting a special file I am able to see (and read!) the file name of my selection. To make this styled component a good alternative for the raw HTML version we should provide this as well.

I need to add some more HTML (and some CSS of course):

<label for="filename" class="hide">uploaded file</label>
<input type="text" id="filename" autocomplete="off" readonly placeholder="no file uploaded">

I recommend setting the readonly attribute on the input, because the user should not be able to be entering stuff manually here, only by choosing a file to upload. To provide initial feedback I added a placeholder attribute that tells the user, that he has not chosen any files yet.

With some more jQuery on top, this works pretty fine:

$('#fileupload').change(function(e) {
  var filename = $('#fileupload').val().split('\\').pop();
  $('#filename').val(filename);
  $('#filename').attr('placeholder', filename);
  $('#filename').focus();
});

This little jQuery example just picks the value of the #fileupload element when it is changed, sets it as a value to our new #filename element and wipes out some useless path information. We update the placeholder attribute with the filename as well, because otherwise VoiceOver in Safari will read “no file uploaded” again after the upload.

Final result

Here you can see the complete thing in action:

See the Pen A styled accessible file upload by Accessabilly (@accessabilly) on CodePen

You can find the working example and the source code of it here.
If you want to download the code, refer to my Github repository page.

Further Reading

2 thoughts on “A styled accessible file upload”

  1. Wen

    Your example here had the attribute of type set to text for the input element. Why isn’t it set to file instead?

    1. Accessabilly

      Hi there,
      there is a input with type “file”, but it’s hidden. The text input is just for visually showing the filename.

Leave a Reply

Your email address will not be published. Required fields are marked *