Search code examples
csshovermedia-queriestouchinteraction-media-features

Media query for devices supporting hover


I'd like to provide separate behaviour for browsers supporting hover (e.g. desktop browsers) and ones which don't (e.g. touchscreen devices). Specifically I want to declare a hover state on browsers that support it, but not for browsers that don't, so as to avoid having mobile browsers emulate it with extra taps, as this breaks other interactions on the page - by not defining a hover state for those browsers this is avoided.

I've read up on the Interaction Media Queries feature and it looks like it should do the trick. I'd be able to do something like:

@media (hover: none) {
  /* behaviour for touch browsers */
}

According to CanIUse it is available on all the browsers I need to support except IE11 and Firefox.

So I wondered if I could do it the other way around - since the main touch devices all support it, then negate it:

@media not (hover: none) {
  /* behaviour for desktop browsers */
}

However, this doesn't seem to work at all.

Pseudocode example of what I'm trying to do:

.myelement {
  /* some styling */
  /* note: no hover state here */
}
@media(this device supports hover) {
  .myelement:hover {
    /* more styling */
  }
}

So, is there a way to make this work in the way intended, or am I down the wrong track?


Solution

  • Thanks to Dekel's comments I solved this by running the logic in JS and applying a class instead:

    e.g.

    const canHover = !(matchMedia('(hover: none)').matches);
    if(canHover) {
      document.body.classList.add('can-hover');
    }
    

    Then in the stylesheet:

    .myElement {
      background: blue;
    }
    .can-hover .myElement:hover {
      background: red;
    }
    

    I've tested this on desktop Chrome, Safari and Firefox, and iOS Safari and it works as expected.