On Android Chrome, "Desktop Mode" removes the "Android" string from the user agent. If you can use JavaScript, the following mostly detects Android Chrome Desktop Mode:
var webkitVer = parseInt(/WebKit\/([0-9]+)|$/.exec(navigator.appVersion)[1], 10); // also matches AppleWebKit
var isGoogle = webkitVer && navigator.vendor.indexOf('Google') === 0; // Also true for Opera Mobile and maybe others
var isAndroid = isGoogle && userAgent.indexOf('Android') > 0; // Careful - Firefox and Windows Mobile also have Android in user agent
var androidDesktopMode = !isAndroid && isGoogle && (navigator.platform.indexOf('Linux a') === 0) && 'ontouchstart' in document.documentElement;
It makes the assumption that Chrome with an ARM processor is Android. The assumption definitely fails for users running Linux on ARM, fails for Android on i686 or MIPS etc (and I haven't been able to test ChromeOS).
For Windows Mobile, you can detect desktop mode by checking for the string "WPDesktop;" in the user agent.
Edit: The code used to use window.chrome
and window.chrome.webstore
which was a reliable test, but somewhere around Chrome 65 you couldn't use those properties anymore to detect desktop mode. Thanks to @faks for the info.
Edit 2: I now highly recommend against treating "desktop mode" as "mobile mode" but, here are my updated opinions:
beware that code detecting desktop mode is really fragile and newer browser versions regularly break sniffing code techniques
unless you have a serious bug or critical usability issue, it totally isn't worth sniffing
never sniff if you are not actively maintaining the code and testing against beta versions of browsers
I use the following for iOS: navigator.vendor.indexOf('Apple') === 0 && 'ontouchstart' in document.body
. We need this to set the astonishingly shitty iOS inputMode correctly for iPadOS 13 (old navigator.platform technique now broken in iOS 13 Beta) and avoid other iOS usability bugs with other input types. I think you can check window.screen.width == 768
to sniff iPad (stays same even if orientation change). The sniff will break if Macbook comes out in a touch version.
I now use the following for detecting Android desktop mode: 'ontouchstart' in document.body && navigator.platform.indexOf('Linux a') === 0 && (window.chrome || (window.Intl && Intl.v8BreakIterator))
. Horrible unreliable sniff, but we really need it because android viewport and pinch zooming (not page zooming) is really broken on Android for our SPA (screen size is not enough as a desktop touch user can use a small window).