Skip to contentSkip to footer

If you want to be kept up to date with new articles, CSS resources and tools, join our newsletter.

Recently I needed a way to detect support for a media query in CSS and JavaScript. To detect if a browser supports a certain CSS feature, you can use @supports () { ... }, but that doesn't work for media queries. In this article I'll show you how you can.

Why I needed this

For a presentation I did on prefers-reduced-data I wanted to apply something in one of two situations:

  • There was no support for prefers-reduced-data at all
  • There was support for prefers-reduced-data and the value was "no-preference".

For this, I couldn't use just @media (prefers-reduced-data: no-preference) because that would be false if either there was no support (since the browser wouldn't understand the media query) or if it was supported but the user wanted to preserve data.

What I needed was a test for the media feature regardless of it's value. To do that, we can use the or notation.

Detecting media query support in CSS

To detect if a media query is supported in CSS at all, you can use the following CSS:

@media not all and (prefers-reduced-data), (prefers-reduced-data) {
  ...;
}

That looks like a bit of weird, so lets dissect what it actually says. Firstly, let's split the two media features:

(prefers-reduced-data)

This one looks straightforward but there's something weird: the media feature is missing a value! Usually, media features come with a value like "min-width: 400px", but this one doesn't.

That's because media features have a "shorthand" when they only have two options and prefers-reduced-data does, it only has "no-preference" (off) and "reduce" (on). When you omit the value, it tests for it being on.

So here's how this will resolve:

  • No preference: false
  • Reduce: true

But if the browser doesn't support a media feature, it will automatically interpret the media query as "not all", which is always false, so we end with this:

  • No support: false
  • No preference: false
  • Reduce: true

not all and (prefers-reduced-data)

The notable thing here is not all and. "all" is the default media type, and it applies to both screen and print. You can omit it but if you add it you need to add "and" in between it and the media feature (which is the part between parentheses).

not is how you can negate a media query. For example, @media not print {...} would apply everywhere except print.

It's worth noting that not applies to the entire media query, so we're really testing for `not (all and (prefers-reduced-data: reduce)).

With all being the default, what we're really checking here for is "not (prefers-reduced-data)" but that's invalid notation until supports for Media Queries level 4 lands. If you use not, for now you always need to add the media type as well.

Here's how this resolves:

  • no support: still false, since the browser doesn't understand it
  • support but off: true (its the negation of it being on)
  • support but on: false

Combined

When a browser interprets the entire media query only one of them has to be true for the entire media query to apply. So lets see the results for these in all three situations: no support, support but off and support and on.

No supportSupport & offSupport & on
not all and (prefers-reduced-data)falsetruefalse
(prefers-reduced-data)falsefalsetrue
Resultfalsetruetrue

For "No support", the browser does not understand either media query so it converts both to not all and you end up with "false" for both. For the other two situations it's either going to be turned on or off but in both situations one of them is going to resolve to true. In that case, CSS in that media query block will now be applied if the feature is supported, regardless of what its value is.

Detecting media query support in JavaScript

We can use the same media query in JavaScript using the window.matchMedia API:

const isSupported = window.matchMedia(`not all and (prefers-reduced-data), (prefers-reduced-data)`).matches;

window.matchMedia returns an object with a "matches" boolean property that is either true or false. For more on the API, check out the using media queries in JavaScript section of my guide on media queries.

After I shared the above out on Twitter, Mathias pointed out a different method.

const query = '(prefers-reduced-data)';
const resolvedMediaQuery = window.matchMedia(query).media;

const isSupported = query === resolvedMediaQuery;

The window.matchMedia API also returns a "media" property, which is the normalized and resolved string representation of the query you tested. If matchMedia encounters something it doesn't understand, that changed to "not all", and if it does support the query it will return that, regardless of if it matches (you can use the matches property for that).

So by comparing your input to the media, you either get:

No support: '(prefers-reduced-data)' === 'not all' which is false.

Support: '(prefers-reduced-data)' === '(prefers-reduced-data)' which is true.

Which one to use?

What I like about the first option, with the complex media query, is that all the logic happens inside CSS. I also like how you get a boolean, and don't have to do string comparison.

The second can be a little bit easier to understand at a glance, but you need to make sure that your query input is the same as the browser normalizes it.

For example, if you test (prefers-reduced-data ) (notice the space), that would resolve "matches" to true in supported browsers because the white space is not important, but comparing the normalized media query would return false, since that normalization has removed that extra space. So string comparison can be tricky depending on your input.

Edit: Thomas Steiner pointed out you can check for 'resolvedMediaQuery' !== 'not all' instead to avoid the issue of normalization. Clever!

When to use this?

We're set to get a whole lot of new media features in the coming years, like prefers-reduced-data, prefers-contrast, screen-spanning and more.

While transitioning to all browsers supporting this, you'll often want to turn on extra features for browsers that support it without causing issues in older browsers since the new default might not always be the best experience in older browsers. With this media feature you can split the behavior in older browsers without support for newer browsers with support.

Originally posted on Kilian's personal blog.

Build your next project with Polypane

  • Use all features on all plans
  • On Mac, Window and Linux
  • 14-day free trial – no credit card needed
Try for free
Polypane UI