Can I use Javascript in a cross-platform way to get the compass heading in iOS and Android (with Chrome), without using something like PhoneGap? I know iOS has DeviceOrientationEvent, but I can't find any equivalent on Chrome for Android.
-
1There is a [Web Compass API](http://dev.w3.org/2009/dap/system-info/compass.html). I don't know the browser support of it but you can try accessing `getCurrentOrientation` in global scope and see if it's there. You might also want to try it with `webkit` prefix. Good luck! – Mohsen Apr 17 '13 at 04:12
3 Answers
As a primer you should review this previous related StackOverflow answer and be familiar with the general practical considerations for using DeviceOrientation Events in web applications.
The simple solution I provided in my previous related StackOverflow answer only applies to browsers that implement absolute
deviceorientation events (i.e. browsers where the deviceorientation alpha
property is compass-oriented). That means the solution provided there currently only works in Android browsers and not iOS-based browsers or any other browser that does not provide absolute
-based deviceorientation event data.
To reliably obtain the current compass heading across both Android and iOS browsers today you need to handle both absolute
and non-absolute
implementations that provide the additional webkitCompassHeading
property and make sure to account for any current screen orientation changes as part of that. AFAIK the only library that currently does this is Full Tilt JS (disclaimer: I am the author of this library).
The following code will give you the same correct compass heading across both iOS and Android browsers, taking account of the differences in device orientation implementations and applying any necessary runtime screen orientation transforms too:
<!-- Include the Full Tilt JS library from https://github.com/richtr/Full-Tilt-JS -->
<script src="fulltilt-min.js"></script>
<script>
// Obtain a new *world-oriented* Full Tilt JS DeviceOrientation Promise
var promise = FULLTILT.getDeviceOrientation({ 'type': 'world' });
// Wait for Promise result
promise.then(function(deviceOrientation) { // Device Orientation Events are supported
// Register a callback to run every time a new
// deviceorientation event is fired by the browser.
deviceOrientation.listen(function() {
// Get the current *screen-adjusted* device orientation angles
var currentOrientation = deviceOrientation.getScreenAdjustedEuler();
// Calculate the current compass heading that the user is 'looking at' (in degrees)
var compassHeading = 360 - currentOrientation.alpha;
// Do something with `compassHeading` here...
});
}).catch(function(errorMessage) { // Device Orientation Events are not supported
console.log(errorMessage);
// Implement some fallback controls here...
});
</script>
Here is a demo that demonstrates this technique to obtain the compass heading the user is facing. It should work well on both iOS and Android browsers.
The implementation of the code in that demo is as shown above and can be viewed on Github at ./scripts/compass.js:L268-L272.
-
2hello Richt, I have tested your library, works perfectly, but I think it doesn't give me the real North right ? – Miguel May 17 '16 at 08:55
-
5none of these examples give the north on android. this is a total joke. – Average Joe Jun 21 '16 at 15:26
-
@richt after obtaining device orientation , how would i use it to get the compass heading ( direction to geo Point B)? I can successfully calculate bearing(degrees) from 2 geo-points(Point A & Point B). – jasan Oct 27 '16 at 23:56
-
-
2An issue on Chrome is waiting for a PR integration since february (may explain some previous comments): Issue https://github.com/adtile/Full-Tilt/issues/22 PR https://github.com/adtile/Full-Tilt/pull/25 :-/ – Maxime Pacary Dec 18 '17 at 14:43
-
Hi @Richt, I have checked the demo on two android devices. Both are pointing to different directions but none of them is correct, it's really +/- 180deg. I like your marine compass, but is there a fix to get real north? – kernification Mar 26 '20 at 13:19
-
@richt I get "No compass detected" on Chrome and Safari on my iPhone 7 ([screenshot](https://pasteboard.co/JfjqUJc.jpg)). I wonder if mobiles have added data-privacy security measures... – P i Jun 29 '20 at 09:59
-
Yes you can! Unfortunately the alpha
doesn't work on iPhones/iPads. With Mobile Safari, alpha
is based on the direction the device was pointing when device orientation was first requested. The included webkit offers you the compass heading. To make it work for all other browsers (which all supports alpha
as compassheading
) you can use the following Javascript code:
if (window.DeviceOrientationEvent) {
// Listen for the deviceorientation event and handle the raw data
window.addEventListener('deviceorientation', function(eventData) {
var compassdir;
if(event.webkitCompassHeading) {
// Apple works only with this, alpha doesn't work
compassdir = event.webkitCompassHeading;
}
else compassdir = event.alpha;
});
}
Android also supports Webkit, so would also use event.webkitCompassHeading, but that's OK.
BTW: "oncompassneedscalibration
" is also not supported for iPhones and iPads.

- 389
- 1
- 4
- 15

- 816
- 10
- 24
-
``event.alpha`` isn't enough to get a compass heading, ``alpha`` is just the z rotation of a device on its own axis (see http://w3c.github.io/deviceorientation/spec-source-orientation.html#rotations). It is possible to translate alpha, beta and gamma into a compass heading, though; see the answer from @richt below. – davidjb Jun 24 '15 at 21:50
-
3@davidjb: Your answer might confuse readers without any given explanations and richt example is also related to the alpha channel. Here is a visual example that shows that alpha channel is related to horizontal orientation from holder view: https://dev.opera.com/articles/w3c-device-orientation-api/ What you probably wanted to say: In case of using only alpha channel for compass heading you have to hold the phone horizontally and in portrait mode for proper use. Else the phone might got loose its orientation and not showing proper heading. – Jonny Jun 24 '15 at 23:07
I believe you can use the "heading" field of the location object, from navigator.geolocation, please see here:
https://developer.mozilla.org/en-US/docs/WebAPI/Using_geolocation
I know no other way.
Hope it helps, A.

- 626
- 4
- 12
-
1-1. Geolocation is not giving you Compass data. This is not a real answer. – Mohsen Apr 16 '13 at 22:56
-
The heading attribute denotes the direction of travel, not compass; and is insuitable for use indoors. – Gregor Eesmaa Sep 18 '19 at 09:42