iOS Mobile Safari Hover Menu Fix

By  on  

One component I'm quite proud of creating is the Mozilla Developer Network's dropdown menu component.  It's elegant, flexible, accessible via keyboard, and relatively compact (though it does require jQuery).  One problem I did notice, however, was that tapping once opened the menu (good) but you'd have to double tap (tap twice) on a submenu link to trigger a move to the link URL.  Not a catastrophic issue but certainly annoying. After a bit of tinkering I came up with a solution which satisfied the annoyance!

The JavaScript

I found out the core issue:  the main menu item was coded to open the submenu upon mouseenter.  While I was grateful that iOS mobile Safari was using mouseenter as simply an open trigger, it was causing users the double tap pain to visit a link in the submenu.  So I took advantage of the touchstart event:

$menuItem.on('touchstart mouseenter focus', function(e) {
    if(e.type == 'touchstart') {
        // Don't trigger mouseenter even if they hold
        e.stopImmediatePropagation();
        // If $item is a link (<a>), don't go to said link on mobile, show menu instead
        e.preventDefault();
    }

    // Show the submenu here
});

Why not touchend?  Unfortunately if the user held their finger down for more than a quick tap, the mouseenter event would trigger.  A bit odd but since I have touchstart as the first event in the listener, I can use stopImmediatePropagation to prevent the mouseenter from ever firing.  What's also nice is I don't have to do any device detection to juggle touchstart and mouseenter, so this is an awesome solution...

...but not a perfect solution.  If you tap and hold the main menu item link, the mobile browser context menu doesn't appear because we've used preventDefault.  I'm still exploring and experimenting so I'll continue to update this post as I search for the holy grail!

Recent Features

  • By
    Send Text Messages with PHP

    Kids these days, I tell ya.  All they care about is the technology.  The video games.  The bottled water.  Oh, and the texting, always the texting.  Back in my day, all we had was...OK, I had all of these things too.  But I still don't get...

  • By
    From Webcam to Animated GIF: the Secret Behind chat.meatspac.es!

    My team mate Edna Piranha is not only an awesome hacker; she's also a fantastic philosopher! Communication and online interactions is a subject that has kept her mind busy for a long time, and it has also resulted in a bunch of interesting experimental projects...

Incredible Demos

  • By
    Multiple File Upload Input

    More often than not, I find myself wanting to upload more than one file at a time.  Having to use multiple "file" INPUT elements is annoying, slow, and inefficient.  And if I hate them, I can't imagine how annoyed my users would be.  Luckily Safari, Chrome...

  • By
    Create a Simple News Scroller Using Dojo

    My journey into Dojo JavaScript has been exciting and I'm continuing to learn more as I port MooTools scripts to Dojo. My latest experiment is porting a simple new scroller from MooTools to Dojo. The code is very similar! The HTML The news items...

Discussion

  1. James

    Best solution I’ve seen so far. Tried using touchend, and it was firing twice or more on Android devices. Did play around with a timeout, that worked, but felt wrong. This solution works on everything I’ve tested so far, bar the minor issue of holding down the touch you mention.

  2. Is there anyway to do that without jQuery? I’ve been trying but it doesn’t seem to work:

    menuItem.addEventListener( 'touchstart mouseenter focus', toggleMenu );
    

    And the toggleMenu would be like this:

    function toggleMenu() {
        if(e.type == 'touchstart') {
            // Don't trigger mouseenter even if they hold
            e.stopImmediatePropagation();
            // If menuItem is a link (), don't go to said link on mobile, show menu instead
            e.preventDefault();
        }
        // toggle code here
    }
    

    Sorry if this sound silly, but I’m very much new to javascript.

    Anyway, you’re blog is helping me a lot, thanks!

  3. dan

    any idea how I could trigger the click in this case?

    I’m wanting to show the menu (in my case a tooltip) on touchstart, but I actually follow the link on touchend

  4. THANK YOU SO MUCH FOR THIS changing my listener from mouseenter to touchstart immediately solved the problem for me.

    I literally spent an hour fiddling with the CSS because I thought it was something to do with pseudo elements. :/

  5. Dave

    Thank you. I have solved the Upload link need to double tap in iOS issue with your help.

Wrap your code in <pre class="{language}"></pre> tags, link to a GitHub gist, JSFiddle fiddle, or CodePen pen to embed!