Free javascript Hosting


compass.js

Uploaded on Dec 05 2021 19:16 by keyurjain21

(function webpackUniversalModuleDefinition(root, factory) {
if(typeof exports === 'object' && typeof module === 'object')
module.exports = factory();
else if(typeof define === 'function' && define.amd)
define([], factory);
else if(typeof exports === 'object')
exports["Compass"] = factory();
else
root["Compass"] = factory();
})(window, function() {
return /******/ (function(modules) { // webpackBootstrap
/******/ // The module cache
/******/ var installedModules = {};
/******/
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/
/******/ // Check if module is in cache
/******/ if(installedModules[moduleId]) {
/******/ return installedModules[moduleId].exports;
/******/ }
/******/ // Create a new module (and put it into the cache)
/******/ var module = installedModules[moduleId] = {
/******/ i: moduleId,
/******/ l: false,
/******/ exports: {}
/******/ };
/******/
/******/ // Execute the module function
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/
/******/ // Flag the module as loaded
/******/ module.l = true;
/******/
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/
/******/
/******/ // expose the modules object (__webpack_modules__)
/******/ __webpack_require__.m = modules;
/******/
/******/ // expose the module cache
/******/ __webpack_require__.c = installedModules;
/******/
/******/ // define getter function for harmony exports
/******/ __webpack_require__.d = function(exports, name, getter) {
/******/ if(!__webpack_require__.o(exports, name)) {
/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
/******/ }
/******/ };
/******/
/******/ // define __esModule on exports
/******/ __webpack_require__.r = function(exports) {
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
/******/ }
/******/ Object.defineProperty(exports, '__esModule', { value: true });
/******/ };
/******/
/******/ // create a fake namespace object
/******/ // mode & 1: value is a module id, require it
/******/ // mode & 2: merge all properties of value into the ns
/******/ // mode & 4: return value when already ns object
/******/ // mode & 8|1: behave like require
/******/ __webpack_require__.t = function(value, mode) {
/******/ if(mode & 1) value = __webpack_require__(value);
/******/ if(mode & 8) return value;
/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
/******/ var ns = Object.create(null);
/******/ __webpack_require__.r(ns);
/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
/******/ return ns;
/******/ };
/******/
/******/ // getDefaultExport function for compatibility with non-harmony modules
/******/ __webpack_require__.n = function(module) {
/******/ var getter = module && module.__esModule ?
/******/ function getDefault() { return module['default']; } :
/******/ function getModuleExports() { return module; };
/******/ __webpack_require__.d(getter, 'a', getter);
/******/ return getter;
/******/ };
/******/
/******/ // Object.prototype.hasOwnProperty.call
/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
/******/
/******/ // __webpack_public_path__
/******/ __webpack_require__.p = "";
/******/
/******/
/******/ // Load entry module and return exports
/******/ return __webpack_require__(__webpack_require__.s = "./src/index.js");
/******/ })
/************************************************************************/
/******/ ({

/***/ "./node_modules/css-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js!./src/Compass/Compass.scss":
/*!***************************************************************************************************************!*\
!*** ./node_modules/css-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js!./src/Compass/Compass.scss ***!
\***************************************************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {

// Imports
var ___CSS_LOADER_API_IMPORT___ = __webpack_require__(/*! ../../node_modules/css-loader/dist/runtime/api.js */ "./node_modules/css-loader/dist/runtime/api.js");
exports = ___CSS_LOADER_API_IMPORT___(false);
// Module
exports.push([module.i, "#emoji__compass {\n width: 32px;\n height: 32px;\n position: fixed;\n top: 0.5rem;\n left: 0.5rem;\n background-color: white;\n border-radius: 2px;\n box-shadow: 1px 1px 8px rgba(0, 0, 0, 0.25); }\n", ""]);
// Exports
module.exports = exports;


/***/ }),

/***/ "./node_modules/css-loader/dist/runtime/api.js":
/*!*****************************************************!*\
!*** ./node_modules/css-loader/dist/runtime/api.js ***!
\*****************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
// css base code, injected by the css-loader
// eslint-disable-next-line func-names
module.exports = function (useSourceMap) {
var list = []; // return the list of modules as css string

list.toString = function toString() {
return this.map(function (item) {
var content = cssWithMappingToString(item, useSourceMap);

if (item[2]) {
return "@media ".concat(item[2], " {").concat(content, "}");
}

return content;
}).join('');
}; // import a list of modules into the list
// eslint-disable-next-line func-names


list.i = function (modules, mediaQuery, dedupe) {
if (typeof modules === 'string') {
// eslint-disable-next-line no-param-reassign
modules = [[null, modules, '']];
}

var alreadyImportedModules = {};

if (dedupe) {
for (var i = 0; i < this.length; i++) {
// eslint-disable-next-line prefer-destructuring
var id = this[i][0];

if (id != null) {
alreadyImportedModules[id] = true;
}
}
}

for (var _i = 0; _i < modules.length; _i++) {
var item = [].concat(modules[_i]);

if (dedupe && alreadyImportedModules[item[0]]) {
// eslint-disable-next-line no-continue
continue;
}

if (mediaQuery) {
if (!item[2]) {
item[2] = mediaQuery;
} else {
item[2] = "".concat(mediaQuery, " and ").concat(item[2]);
}
}

list.push(item);
}
};

return list;
};

function cssWithMappingToString(item, useSourceMap) {
var content = item[1] || ''; // eslint-disable-next-line prefer-destructuring

var cssMapping = item[3];

if (!cssMapping) {
return content;
}

if (useSourceMap && typeof btoa === 'function') {
var sourceMapping = toComment(cssMapping);
var sourceURLs = cssMapping.sources.map(function (source) {
return "/*# sourceURL=".concat(cssMapping.sourceRoot || '').concat(source, " */");
});
return [content].concat(sourceURLs).concat([sourceMapping]).join('\n');
}

return [content].join('\n');
} // Adapted from convert-source-map (MIT)


function toComment(sourceMap) {
// eslint-disable-next-line no-undef
var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap))));
var data = "sourceMappingURL=data:application/json;charset=utf-8;base64,".concat(base64);
return "/*# ".concat(data, " */");
}

/***/ }),

/***/ "./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js":
/*!****************************************************************************!*\
!*** ./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js ***!
\****************************************************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


var isOldIE = function isOldIE() {
var memo;
return function memorize() {
if (typeof memo === 'undefined') {
// Test for IE <= 9 as proposed by Browserhacks
// @see http://browserhacks.com/#hack-e71d8692f65334173fee715c222cb805
// Tests for existence of standard globals is to allow style-loader
// to operate correctly into non-standard environments
// @see https://github.com/webpack-contrib/style-loader/issues/177
memo = Boolean(window && document && document.all && !window.atob);
}

return memo;
};
}();

var getTarget = function getTarget() {
var memo = {};
return function memorize(target) {
if (typeof memo[target] === 'undefined') {
var styleTarget = document.querySelector(target); // Special case to return head of iframe instead of iframe itself

if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {
try {
// This will throw an exception if access to iframe is blocked
// due to cross-origin restrictions
styleTarget = styleTarget.contentDocument.head;
} catch (e) {
// istanbul ignore next
styleTarget = null;
}
}

memo[target] = styleTarget;
}

return memo[target];
};
}();

var stylesInDom = [];

function getIndexByIdentifier(identifier) {
var result = -1;

for (var i = 0; i < stylesInDom.length; i++) {
if (stylesInDom[i].identifier === identifier) {
result = i;
break;
}
}

return result;
}

function modulesToDom(list, options) {
var idCountMap = {};
var identifiers = [];

for (var i = 0; i < list.length; i++) {
var item = list[i];
var id = options.base ? item[0] + options.base : item[0];
var count = idCountMap[id] || 0;
var identifier = "".concat(id, " ").concat(count);
idCountMap[id] = count + 1;
var index = getIndexByIdentifier(identifier);
var obj = {
css: item[1],
media: item[2],
sourceMap: item[3]
};

if (index !== -1) {
stylesInDom[index].references++;
stylesInDom[index].updater(obj);
} else {
stylesInDom.push({
identifier: identifier,
updater: addStyle(obj, options),
references: 1
});
}

identifiers.push(identifier);
}

return identifiers;
}

function insertStyleElement(options) {
var style = document.createElement('style');
var attributes = options.attributes || {};

if (typeof attributes.nonce === 'undefined') {
var nonce = true ? __webpack_require__.nc : undefined;

if (nonce) {
attributes.nonce = nonce;
}
}

Object.keys(attributes).forEach(function (key) {
style.setAttribute(key, attributes[key]);
});

if (typeof options.insert === 'function') {
options.insert(style);
} else {
var target = getTarget(options.insert || 'head');

if (!target) {
throw new Error("Couldn't find a style target. This probably means that the value for the 'insert' parameter is invalid.");
}

target.appendChild(style);
}

return style;
}

function removeStyleElement(style) {
// istanbul ignore if
if (style.parentNode === null) {
return false;
}

style.parentNode.removeChild(style);
}
/* istanbul ignore next */


var replaceText = function replaceText() {
var textStore = [];
return function replace(index, replacement) {
textStore[index] = replacement;
return textStore.filter(Boolean).join('\n');
};
}();

function applyToSingletonTag(style, index, remove, obj) {
var css = remove ? '' : obj.media ? "@media ".concat(obj.media, " {").concat(obj.css, "}") : obj.css; // For old IE

/* istanbul ignore if */

if (style.styleSheet) {
style.styleSheet.cssText = replaceText(index, css);
} else {
var cssNode = document.createTextNode(css);
var childNodes = style.childNodes;

if (childNodes[index]) {
style.removeChild(childNodes[index]);
}

if (childNodes.length) {
style.insertBefore(cssNode, childNodes[index]);
} else {
style.appendChild(cssNode);
}
}
}

function applyToTag(style, options, obj) {
var css = obj.css;
var media = obj.media;
var sourceMap = obj.sourceMap;

if (media) {
style.setAttribute('media', media);
} else {
style.removeAttribute('media');
}

if (sourceMap && btoa) {
css += "\n/*# sourceMappingURL=data:application/json;base64,".concat(btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))), " */");
} // For old IE

/* istanbul ignore if */


if (style.styleSheet) {
style.styleSheet.cssText = css;
} else {
while (style.firstChild) {
style.removeChild(style.firstChild);
}

style.appendChild(document.createTextNode(css));
}
}

var singleton = null;
var singletonCounter = 0;

function addStyle(obj, options) {
var style;
var update;
var remove;

if (options.singleton) {
var styleIndex = singletonCounter++;
style = singleton || (singleton = insertStyleElement(options));
update = applyToSingletonTag.bind(null, style, styleIndex, false);
remove = applyToSingletonTag.bind(null, style, styleIndex, true);
} else {
style = insertStyleElement(options);
update = applyToTag.bind(null, style, options);

remove = function remove() {
removeStyleElement(style);
};
}

update(obj);
return function updateStyle(newObj) {
if (newObj) {
if (newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap) {
return;
}

update(obj = newObj);
} else {
remove();
}
};
}

module.exports = function (list, options) {
options = options || {}; // Force single-tag solution on IE6-9, which has a hard limit on the # of <style>
// tags it will allow on a page

if (!options.singleton && typeof options.singleton !== 'boolean') {
options.singleton = isOldIE();
}

list = list || [];
var lastIdentifiers = modulesToDom(list, options);
return function update(newList) {
newList = newList || [];

if (Object.prototype.toString.call(newList) !== '[object Array]') {
return;
}

for (var i = 0; i < lastIdentifiers.length; i++) {
var identifier = lastIdentifiers[i];
var index = getIndexByIdentifier(identifier);
stylesInDom[index].references--;
}

var newLastIdentifiers = modulesToDom(newList, options);

for (var _i = 0; _i < lastIdentifiers.length; _i++) {
var _identifier = lastIdentifiers[_i];

var _index = getIndexByIdentifier(_identifier);

if (stylesInDom[_index].references === 0) {
stylesInDom[_index].updater();

stylesInDom.splice(_index, 1);
}
}

lastIdentifiers = newLastIdentifiers;
};
};

/***/ }),

/***/ "./src/Compass/Compass.js":
/*!********************************!*\
!*** ./src/Compass/Compass.js ***!
\********************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {

const emojiCompass = __webpack_require__(/*! ../assets/1F9ED.svg */ "./src/assets/1F9ED.svg");
__webpack_require__(/*! ./Compass.scss */ "./src/Compass/Compass.scss");

const DEFAULTS = {
emojiCompass: emojiCompass,
};

class Compass {
/**
* The compass class is a utility for creating a javascript compass
* with no external dependencies. You can create a compass that
* points north from your location or points towards a specified
* latitude and longitude coordinate. Regardless of where you turn
* the compass will point towards the specified direction.
*
* @class
*/
constructor() {
this.heading = 0;
this.deviceAngleDelta = 0;
this.position = null;
this.button = null;
this.debug = true;

this.geolocationID = null;
this.permissionGranted = false;
}

/**
* initialized the compass - returns a promise or can invoke a callback
* @param {callback} callback - callback to be called after the .start() function is done
*/
init(callback = undefined) {
if (callback) {
this.callCallback(this.start(), callback);
} else {
return this.start();
}
}

/**
* Initializes the device orientation and watches the user position by default
* @async
* @function start
*
*/
async start() {
try {
await this.watchPosition();

if (this.debug === true) {
this.createButton();
this.button.addEventListener(
"click",
this.triggerDeviceOrientationPermissions()
);
document.body.appendChild(this.button);
}
// NOTE: if debug is false, the expectation is that you handle the
// the device orientation handling permissions on your own.

return true;
} catch (err) {
alert(err);
}
}

/**
* creates a button to trigger the device orientation permissions
*
* function
* @name createButton
*/
createButton() {
const div = document.createElement("div");
div.innerHTML += emojiCompass;
this.button = div.querySelector("#emoji__compass");
}

/**
* triggers the device orientation permissions
* @function
* @name triggerDeviceOrientationPermissions
*/
triggerDeviceOrientationPermissions() {
return async (evt) => {
try {
await this.allowOrientationPermissions();
if (this.debug === true) {
this.button.style.display = "none";
}
} catch (err) {
alert(err);
}
};
}

/**
* Asks the user to allow permissions to get orientation
*
* @async
* @name allowOrientationPermissions
*/
async allowOrientationPermissions() {
if (typeof window.DeviceOrientationEvent.requestPermission === "function") {
const permission = await window.DeviceOrientationEvent.requestPermission();
alert(permission);
if (permission == "granted") {
this.permissionGranted = true;
window.addEventListener(
"deviceorientation",
this.deviceOrientationHandler.bind(this),
true
);
return true;
} else {
throw new Error("no device orientation permissions!");
}
} else {
if (window.DeviceOrientationEvent) {
this.permissionGranted = true;
alert("device orientation permission granted");
window.addEventListener(
"deviceorientation",
this.deviceOrientationHandler.bind(this),
true
);
return true;
} else {
alert("no device orientation support");
}
}
}

/**
* This is where my nose points - and seeing as my nose
* is attached to my head, this is where my head
* (and thus my machine) is pointing relative to North.
* NOTE: requires that this.position is set
*
* @function getHeading
*/
getHeading(
origin = {
lat: this.position.coords.latitude,
lng: this.position.coords.longitude,
},
north = { lat: 90, lng: this.position.coords.longitude }
) {
this.heading = 360 - this.getBearingToNorth(origin, north);
return this.heading;
}

/**
* This is the angle between the location of an object,
* machine or destination and my heading.
*
* @param {Object} origin - {lat, lng}
* @param {Object} destination - {lat, lng}
*/
getBearing(origin, destination) {
return (
this.calculateAngle(
origin.lat,
origin.lng,
destination.lat,
destination.lng
) *
(180 / Math.PI)
);
}

/**
* get the angle between your heading and north
* the default is true north vs. magnetic north
*
* @function
* @param {object} origin - {lat, lng}
* @param {object} north - {lat, lng}
*/
getBearingToNorth(
origin = {
lat: this.position.coords.latitude,
lng: this.position.coords.longitude,
},
north = { lat: 90, lng: this.position.coords.longitude }
) {
return this.getBearingToDestination(origin, north);
}

/**
* Get the bearings towards the destination
*
* @param {object} origin - {lat, lng}
* @param {object} destination - {lat, lng}
*/
getBearingToDestination(
origin = {
lat: this.position.coords.latitude,
lng: this.position.coords.longitude,
},
destination
) {
const angleToDestination = this.getBearing(origin, destination);
return this.deviceAngleDelta + angleToDestination;
}

/**
* Handles changes created by the device orientation changes
* assumes that the phone is in a portrait mode, with the display up
* towards the sky as if you were holding an actual compass
*
* @callback
* @param {object} evt - the event object of the device orientation
*/
deviceOrientationHandler(evt) {
if (evt.webkitCompassHeading)
//iphone
this.deviceAngleDelta = 360 - evt.webkitCompassHeading;
else if (evt.alpha)
//android
this.deviceAngleDelta = evt.alpha;
else {
console.log("compass direction not found");
}

// console.log(this.deviceAngleDelta)
this.deviceAngleDelta = Math.round(this.deviceAngleDelta);
}

/**
* get the position of the user
*
* @async
* @function
*/
getPosition() {
return new Promise((resolve, reject) => {
navigator.geolocation.getCurrentPosition((position) => {
if (position) {
resolve(position);
} else {
reject("no position found");
}
});
});
}

/**
* watches the geolocation of the user
*
* @async
* @function
*/
watchPosition() {
return new Promise((resolve, reject) => {
this.geolocationID = navigator.geolocation.watchPosition((position) => {
if (position) {
this.position = position;
// console.log(
// this.position.coords.latitude,
// this.position.coords.longitude
// );
resolve(position);
} else {
reject("no position found");
}
});
});
}

/**
* Stops watching the user location
*
* @function
*/
stopTracking() {
navigator.geolocation.clearWatch(this.geolocationID);
console.log("stopped tracking location");
}

/**
* Calculates the angle given a latitude and longitude position
*
* @function
* @param {number} userLat - user latitude
* @param {number} userLon - user longitude
* @param {number} desiredLat - desired latitude
* @param {number} desiredLon - desired longitude
*/
calculateAngle(userLat, userLon, desiredLat, desiredLon) {
return Math.atan2(desiredLon - userLon, desiredLat - userLat);
}

/**
* Helper function that allows calling a callback from an promise function
*
* @param {promise} promise
* @param {callback} callback
*/
callCallback(promise, callback) {
if (callback) {
promise
.then((result) => {
callback(undefined, result);
return result;
})
.catch((error) => {
callback(error);
return error;
});
}
return promise;
}
}

module.exports = Compass;


/***/ }),

/***/ "./src/Compass/Compass.scss":
/*!**********************************!*\
!*** ./src/Compass/Compass.scss ***!
\**********************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {

var api = __webpack_require__(/*! ../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js */ "./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js");
var content = __webpack_require__(/*! !../../node_modules/css-loader/dist/cjs.js!../../node_modules/sass-loader/dist/cjs.js!./Compass.scss */ "./node_modules/css-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js!./src/Compass/Compass.scss");

content = content.__esModule ? content.default : content;

if (typeof content === 'string') {
content = [[module.i, content, '']];
}

var options = {};

options.insert = "head";
options.singleton = false;

var update = api(content, options);



module.exports = content.locals || {};

/***/ }),

/***/ "./src/assets/1F9ED.svg":
/*!******************************!*\
!*** ./src/assets/1F9ED.svg ***!
\******************************/
/*! no static exports found */
/***/ (function(module, exports) {

module.exports = "<svg id=\"emoji__compass\" viewBox=\"0 0 72 72\" xmlns=\"http://www.w3.org/2000/svg\"><g id=\"line-supplement\"><polyline points=\"33.2 33.2 48 24 38.8 38.8\"></polyline></g><g id=\"line\"><circle cx=\"36\" cy=\"36\" r=\"24\" fill=\"none\" stroke=\"#000\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"></circle><polyline fill=\"none\" stroke=\"#000\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" points=\"33.2 33.2 24 48 38.8 38.8\"></polyline><polyline fill=\"none\" stroke=\"#000\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" points=\"33.2 33.2 48 24 38.8 38.8\"></polyline><line x1=\"36\" x2=\"36\" y1=\"21\" y2=\"16\" fill=\"none\" stroke=\"#000\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"></line><line x1=\"36\" x2=\"36\" y1=\"56\" y2=\"51\" fill=\"none\" stroke=\"#000\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"></line><line x1=\"51\" x2=\"56\" y1=\"36\" y2=\"36\" fill=\"none\" stroke=\"#000\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"></line><line x1=\"16\" x2=\"21\" y1=\"36\" y2=\"36\" fill=\"none\" stroke=\"#000\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"></line></g></svg>"

/***/ }),

/***/ "./src/index.js":
/*!**********************!*\
!*** ./src/index.js ***!
\**********************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {

const c = __webpack_require__(/*! ./Compass/Compass.js */ "./src/Compass/Compass.js");
module.exports = c;

/***/ })

/******/ });
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Compass/webpack/universalModuleDefinition","webpack://Compass/webpack/bootstrap","webpack://Compass/./src/Compass/Compass.scss","webpack://Compass/./node_modules/css-loader/dist/runtime/api.js","webpack://Compass/./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js","webpack://Compass/./src/Compass/Compass.js","webpack://Compass/./src/Compass/Compass.scss?f4ab","webpack://Compass/./src/assets/1F9ED.svg","webpack://Compass/./src/index.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD,O;QCVA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;;QAEA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;;;QAGA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA,0CAA0C,gCAAgC;QAC1E;QACA;;QAEA;QACA;QACA;QACA,wDAAwD,kBAAkB;QAC1E;QACA,iDAAiD,cAAc;QAC/D;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA,yCAAyC,iCAAiC;QAC1E,gHAAgH,mBAAmB,EAAE;QACrI;QACA;;QAEA;QACA;QACA;QACA,2BAA2B,0BAA0B,EAAE;QACvD,iCAAiC,eAAe;QAChD;QACA;QACA;;QAEA;QACA,sDAAsD,+DAA+D;;QAErH;QACA;;;QAGA;QACA;;;;;;;;;;;;AClFA;AACA,kCAAkC,mBAAO,CAAC,wGAAmD;AAC7F;AACA;AACA,cAAc,QAAS,oBAAoB,gBAAgB,iBAAiB,oBAAoB,gBAAgB,iBAAiB,4BAA4B,uBAAuB,gDAAgD,EAAE;AACtO;AACA;;;;;;;;;;;;;ACNa;;AAEb;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB;;AAEhB;AACA;AACA;;AAEA;AACA,4CAA4C,qBAAqB;AACjE;;AAEA;AACA,KAAK;AACL,IAAI;AACJ;;;AAGA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA,qBAAqB,iBAAiB;AACtC;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,oBAAoB,qBAAqB;AACzC;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,8BAA8B;;AAE9B;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEA;AACA,CAAC;;;AAGD;AACA;AACA;AACA,qDAAqD,cAAc;AACnE;AACA,C;;;;;;;;;;;;AC7Fa;;AAEb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,CAAC;;AAED;AACA;AACA;AACA;AACA,uDAAuD;;AAEvD;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,CAAC;;AAED;;AAEA;AACA;;AAEA,iBAAiB,wBAAwB;AACzC;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,iBAAiB,iBAAiB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,OAAO;AACP;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,gBAAgB,KAAwC,GAAG,sBAAiB,GAAG,SAAI;;AAEnF;AACA;AACA;AACA;;AAEA;AACA;AACA,GAAG;;AAEH;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA,qEAAqE,qBAAqB,aAAa;;AAEvG;;AAEA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA,yDAAyD;AACzD,GAAG;;AAEH;;;AAGA;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,KAAK;AACL;AACA;AACA;AACA;;AAEA;AACA,0BAA0B;AAC1B;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,mBAAmB,4BAA4B;AAC/C;AACA;AACA;AACA;;AAEA;;AAEA,oBAAoB,6BAA6B;AACjD;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,E;;;;;;;;;;;AC5QA,qBAAqB,mBAAO,CAAC,mDAAqB;AAClD,mBAAO,CAAC,kDAAgB;;AAExB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,aAAa,SAAS;AACtB;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL,aAAa;AACb;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,aAAa,OAAO,WAAW;AAC/B,aAAa,OAAO,gBAAgB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,aAAa,OAAO,WAAW;AAC/B,aAAa,OAAO,UAAU;AAC9B;AACA;AACA;AACA;AACA;AACA,KAAK;AACL,aAAa;AACb;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa,OAAO,WAAW;AAC/B,aAAa,OAAO,gBAAgB;AACpC;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,OAAO;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,OAAO;AACP,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,OAAO;AACP,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,aAAa,OAAO;AACpB,aAAa,OAAO;AACpB,aAAa,OAAO;AACpB,aAAa,OAAO;AACpB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa,QAAQ;AACrB,aAAa,SAAS;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEA;;;;;;;;;;;;AC5TA,UAAU,mBAAO,CAAC,sJAA2E;AAC7F,0BAA0B,mBAAO,CAAC,qNAAuG;;AAEzI;;AAEA;AACA,0BAA0B,QAAS;AACnC;;AAEA;;AAEA;AACA;;AAEA;;;;AAIA,sC;;;;;;;;;;;AClBA,4wC;;;;;;;;;;;ACAA,UAAU,mBAAO,CAAC,sDAAsB;AACxC,mB","file":"compass.js","sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"Compass\"] = factory();\n\telse\n\t\troot[\"Compass\"] = factory();\n})(window, function() {\nreturn "," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = \"./src/index.js\");\n","// Imports\nvar ___CSS_LOADER_API_IMPORT___ = require(\"../../node_modules/css-loader/dist/runtime/api.js\");\nexports = ___CSS_LOADER_API_IMPORT___(false);\n// Module\nexports.push([module.id, \"#emoji__compass {\\n  width: 32px;\\n  height: 32px;\\n  position: fixed;\\n  top: 0.5rem;\\n  left: 0.5rem;\\n  background-color: white;\\n  border-radius: 2px;\\n  box-shadow: 1px 1px 8px rgba(0, 0, 0, 0.25); }\\n\", \"\"]);\n// Exports\nmodule.exports = exports;\n","\"use strict\";\n\n/*\n  MIT License http://www.opensource.org/licenses/mit-license.php\n  Author Tobias Koppers @sokra\n*/\n// css base code, injected by the css-loader\n// eslint-disable-next-line func-names\nmodule.exports = function (useSourceMap) {\n  var list = []; // return the list of modules as css string\n\n  list.toString = function toString() {\n    return this.map(function (item) {\n      var content = cssWithMappingToString(item, useSourceMap);\n\n      if (item[2]) {\n        return \"@media \".concat(item[2], \" {\").concat(content, \"}\");\n      }\n\n      return content;\n    }).join('');\n  }; // import a list of modules into the list\n  // eslint-disable-next-line func-names\n\n\n  list.i = function (modules, mediaQuery, dedupe) {\n    if (typeof modules === 'string') {\n      // eslint-disable-next-line no-param-reassign\n      modules = [[null, modules, '']];\n    }\n\n    var alreadyImportedModules = {};\n\n    if (dedupe) {\n      for (var i = 0; i < this.length; i++) {\n        // eslint-disable-next-line prefer-destructuring\n        var id = this[i][0];\n\n        if (id != null) {\n          alreadyImportedModules[id] = true;\n        }\n      }\n    }\n\n    for (var _i = 0; _i < modules.length; _i++) {\n      var item = [].concat(modules[_i]);\n\n      if (dedupe && alreadyImportedModules[item[0]]) {\n        // eslint-disable-next-line no-continue\n        continue;\n      }\n\n      if (mediaQuery) {\n        if (!item[2]) {\n          item[2] = mediaQuery;\n        } else {\n          item[2] = \"\".concat(mediaQuery, \" and \").concat(item[2]);\n        }\n      }\n\n      list.push(item);\n    }\n  };\n\n  return list;\n};\n\nfunction cssWithMappingToString(item, useSourceMap) {\n  var content = item[1] || ''; // eslint-disable-next-line prefer-destructuring\n\n  var cssMapping = item[3];\n\n  if (!cssMapping) {\n    return content;\n  }\n\n  if (useSourceMap && typeof btoa === 'function') {\n    var sourceMapping = toComment(cssMapping);\n    var sourceURLs = cssMapping.sources.map(function (source) {\n      return \"/*# sourceURL=\".concat(cssMapping.sourceRoot || '').concat(source, \" */\");\n    });\n    return [content].concat(sourceURLs).concat([sourceMapping]).join('\\n');\n  }\n\n  return [content].join('\\n');\n} // Adapted from convert-source-map (MIT)\n\n\nfunction toComment(sourceMap) {\n  // eslint-disable-next-line no-undef\n  var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap))));\n  var data = \"sourceMappingURL=data:application/json;charset=utf-8;base64,\".concat(base64);\n  return \"/*# \".concat(data, \" */\");\n}","\"use strict\";\n\nvar isOldIE = function isOldIE() {\n  var memo;\n  return function memorize() {\n    if (typeof memo === 'undefined') {\n      // Test for IE <= 9 as proposed by Browserhacks\n      // @see http://browserhacks.com/#hack-e71d8692f65334173fee715c222cb805\n      // Tests for existence of standard globals is to allow style-loader\n      // to operate correctly into non-standard environments\n      // @see https://github.com/webpack-contrib/style-loader/issues/177\n      memo = Boolean(window && document && document.all && !window.atob);\n    }\n\n    return memo;\n  };\n}();\n\nvar getTarget = function getTarget() {\n  var memo = {};\n  return function memorize(target) {\n    if (typeof memo[target] === 'undefined') {\n      var styleTarget = document.querySelector(target); // Special case to return head of iframe instead of iframe itself\n\n      if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {\n        try {\n          // This will throw an exception if access to iframe is blocked\n          // due to cross-origin restrictions\n          styleTarget = styleTarget.contentDocument.head;\n        } catch (e) {\n          // istanbul ignore next\n          styleTarget = null;\n        }\n      }\n\n      memo[target] = styleTarget;\n    }\n\n    return memo[target];\n  };\n}();\n\nvar stylesInDom = [];\n\nfunction getIndexByIdentifier(identifier) {\n  var result = -1;\n\n  for (var i = 0; i < stylesInDom.length; i++) {\n    if (stylesInDom[i].identifier === identifier) {\n      result = i;\n      break;\n    }\n  }\n\n  return result;\n}\n\nfunction modulesToDom(list, options) {\n  var idCountMap = {};\n  var identifiers = [];\n\n  for (var i = 0; i < list.length; i++) {\n    var item = list[i];\n    var id = options.base ? item[0] + options.base : item[0];\n    var count = idCountMap[id] || 0;\n    var identifier = \"\".concat(id, \" \").concat(count);\n    idCountMap[id] = count + 1;\n    var index = getIndexByIdentifier(identifier);\n    var obj = {\n      css: item[1],\n      media: item[2],\n      sourceMap: item[3]\n    };\n\n    if (index !== -1) {\n      stylesInDom[index].references++;\n      stylesInDom[index].updater(obj);\n    } else {\n      stylesInDom.push({\n        identifier: identifier,\n        updater: addStyle(obj, options),\n        references: 1\n      });\n    }\n\n    identifiers.push(identifier);\n  }\n\n  return identifiers;\n}\n\nfunction insertStyleElement(options) {\n  var style = document.createElement('style');\n  var attributes = options.attributes || {};\n\n  if (typeof attributes.nonce === 'undefined') {\n    var nonce = typeof __webpack_nonce__ !== 'undefined' ? __webpack_nonce__ : null;\n\n    if (nonce) {\n      attributes.nonce = nonce;\n    }\n  }\n\n  Object.keys(attributes).forEach(function (key) {\n    style.setAttribute(key, attributes[key]);\n  });\n\n  if (typeof options.insert === 'function') {\n    options.insert(style);\n  } else {\n    var target = getTarget(options.insert || 'head');\n\n    if (!target) {\n      throw new Error(\"Couldn't find a style target. This probably means that the value for the 'insert' parameter is invalid.\");\n    }\n\n    target.appendChild(style);\n  }\n\n  return style;\n}\n\nfunction removeStyleElement(style) {\n  // istanbul ignore if\n  if (style.parentNode === null) {\n    return false;\n  }\n\n  style.parentNode.removeChild(style);\n}\n/* istanbul ignore next  */\n\n\nvar replaceText = function replaceText() {\n  var textStore = [];\n  return function replace(index, replacement) {\n    textStore[index] = replacement;\n    return textStore.filter(Boolean).join('\\n');\n  };\n}();\n\nfunction applyToSingletonTag(style, index, remove, obj) {\n  var css = remove ? '' : obj.media ? \"@media \".concat(obj.media, \" {\").concat(obj.css, \"}\") : obj.css; // For old IE\n\n  /* istanbul ignore if  */\n\n  if (style.styleSheet) {\n    style.styleSheet.cssText = replaceText(index, css);\n  } else {\n    var cssNode = document.createTextNode(css);\n    var childNodes = style.childNodes;\n\n    if (childNodes[index]) {\n      style.removeChild(childNodes[index]);\n    }\n\n    if (childNodes.length) {\n      style.insertBefore(cssNode, childNodes[index]);\n    } else {\n      style.appendChild(cssNode);\n    }\n  }\n}\n\nfunction applyToTag(style, options, obj) {\n  var css = obj.css;\n  var media = obj.media;\n  var sourceMap = obj.sourceMap;\n\n  if (media) {\n    style.setAttribute('media', media);\n  } else {\n    style.removeAttribute('media');\n  }\n\n  if (sourceMap && btoa) {\n    css += \"\\n/*# sourceMappingURL=data:application/json;base64,\".concat(btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))), \" */\");\n  } // For old IE\n\n  /* istanbul ignore if  */\n\n\n  if (style.styleSheet) {\n    style.styleSheet.cssText = css;\n  } else {\n    while (style.firstChild) {\n      style.removeChild(style.firstChild);\n    }\n\n    style.appendChild(document.createTextNode(css));\n  }\n}\n\nvar singleton = null;\nvar singletonCounter = 0;\n\nfunction addStyle(obj, options) {\n  var style;\n  var update;\n  var remove;\n\n  if (options.singleton) {\n    var styleIndex = singletonCounter++;\n    style = singleton || (singleton = insertStyleElement(options));\n    update = applyToSingletonTag.bind(null, style, styleIndex, false);\n    remove = applyToSingletonTag.bind(null, style, styleIndex, true);\n  } else {\n    style = insertStyleElement(options);\n    update = applyToTag.bind(null, style, options);\n\n    remove = function remove() {\n      removeStyleElement(style);\n    };\n  }\n\n  update(obj);\n  return function updateStyle(newObj) {\n    if (newObj) {\n      if (newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap) {\n        return;\n      }\n\n      update(obj = newObj);\n    } else {\n      remove();\n    }\n  };\n}\n\nmodule.exports = function (list, options) {\n  options = options || {}; // Force single-tag solution on IE6-9, which has a hard limit on the # of <style>\n  // tags it will allow on a page\n\n  if (!options.singleton && typeof options.singleton !== 'boolean') {\n    options.singleton = isOldIE();\n  }\n\n  list = list || [];\n  var lastIdentifiers = modulesToDom(list, options);\n  return function update(newList) {\n    newList = newList || [];\n\n    if (Object.prototype.toString.call(newList) !== '[object Array]') {\n      return;\n    }\n\n    for (var i = 0; i < lastIdentifiers.length; i++) {\n      var identifier = lastIdentifiers[i];\n      var index = getIndexByIdentifier(identifier);\n      stylesInDom[index].references--;\n    }\n\n    var newLastIdentifiers = modulesToDom(newList, options);\n\n    for (var _i = 0; _i < lastIdentifiers.length; _i++) {\n      var _identifier = lastIdentifiers[_i];\n\n      var _index = getIndexByIdentifier(_identifier);\n\n      if (stylesInDom[_index].references === 0) {\n        stylesInDom[_index].updater();\n\n        stylesInDom.splice(_index, 1);\n      }\n    }\n\n    lastIdentifiers = newLastIdentifiers;\n  };\n};","const emojiCompass = require(\"../assets/1F9ED.svg\");\nrequire(\"./Compass.scss\");\n\nconst DEFAULTS = {\n  emojiCompass: emojiCompass,\n};\n\nclass Compass {\n  /**\n   * The compass class is a utility for creating a javascript compass\n   * with no external dependencies. You can create a compass that\n   * points north from your location or points towards a specified\n   * latitude and longitude coordinate. Regardless of where you turn\n   * the compass will point towards the specified direction.\n   *\n   * @class\n   */\n  constructor() {\n    this.heading = 0;\n    this.deviceAngleDelta = 0;\n    this.position = null;\n    this.button = null;\n    this.debug = true;\n\n    this.geolocationID = null;\n    this.permissionGranted = false;\n  }\n\n  /**\n   * initialized the compass - returns a promise or can invoke a callback\n   * @param {callback} callback - callback to be called after the .start() function is done\n   */\n  init(callback = undefined) {\n    if (callback) {\n      this.callCallback(this.start(), callback);\n    } else {\n      return this.start();\n    }\n  }\n\n  /**\n   * Initializes the device orientation and watches the user position by default\n   * @async\n   * @function start\n   *\n   */\n  async start() {\n    try {\n      await this.watchPosition();\n\n      if (this.debug === true) {\n        this.createButton();\n        this.button.addEventListener(\n          \"click\",\n          this.triggerDeviceOrientationPermissions()\n        );\n        document.body.appendChild(this.button);\n      }\n      // NOTE: if debug is false, the expectation is that you handle the\n      // the device orientation handling permissions on your own.\n\n      return true;\n    } catch (err) {\n      alert(err);\n    }\n  }\n\n  /**\n   * creates a button to trigger the device orientation permissions\n   *\n   * function\n   * @name createButton\n   */\n  createButton() {\n    const div = document.createElement(\"div\");\n    div.innerHTML += emojiCompass;\n    this.button = div.querySelector(\"#emoji__compass\");\n  }\n\n  /**\n   * triggers the device orientation permissions\n   * @function\n   * @name triggerDeviceOrientationPermissions\n   */\n  triggerDeviceOrientationPermissions() {\n    return async (evt) => {\n      try {\n        await this.allowOrientationPermissions();\n        if (this.debug === true) {\n          this.button.style.display = \"none\";\n        }\n      } catch (err) {\n        alert(err);\n      }\n    };\n  }\n\n  /**\n   * Asks the user to allow permissions to get orientation\n   *\n   * @async\n   * @name allowOrientationPermissions\n   */\n  async allowOrientationPermissions() {\n    if (typeof window.DeviceOrientationEvent.requestPermission === \"function\") {\n      const permission = await window.DeviceOrientationEvent.requestPermission();\n      alert(permission);\n      if (permission == \"granted\") {\n        this.permissionGranted = true;\n        window.addEventListener(\n          \"deviceorientation\",\n          this.deviceOrientationHandler.bind(this),\n          true\n        );\n        return true;\n      } else {\n        throw new Error(\"no device orientation permissions!\");\n      }\n    } else {\n      if (window.DeviceOrientationEvent) {\n        this.permissionGranted = true;\n        alert(\"device orientation permission granted\");\n        window.addEventListener(\n          \"deviceorientation\",\n          this.deviceOrientationHandler.bind(this),\n          true\n        );\n        return true;\n      } else {\n        alert(\"no device orientation support\");\n      }\n    }\n  }\n\n  /**\n   * This is where my nose points - and seeing as my nose\n   * is attached to my head, this is where my head\n   * (and thus my machine) is pointing relative to North.\n   * NOTE: requires that this.position is set\n   *\n   * @function getHeading\n   */\n  getHeading(\n    origin = {\n      lat: this.position.coords.latitude,\n      lng: this.position.coords.longitude,\n    },\n    north = { lat: 90, lng: this.position.coords.longitude }\n  ) {\n    this.heading = 360 - this.getBearingToNorth(origin, north);\n    return this.heading;\n  }\n\n  /**\n   * This is the angle between the location of an object,\n   * machine or destination and my heading.\n   *\n   * @param {Object} origin - {lat, lng}\n   * @param {Object} destination - {lat, lng}\n   */\n  getBearing(origin, destination) {\n    return (\n      this.calculateAngle(\n        origin.lat,\n        origin.lng,\n        destination.lat,\n        destination.lng\n      ) *\n      (180 / Math.PI)\n    );\n  }\n\n  /**\n   * get the angle between your heading and north\n   * the default is true north vs. magnetic north\n   *\n   * @function\n   * @param {object} origin - {lat, lng}\n   * @param {object} north - {lat, lng}\n   */\n  getBearingToNorth(\n    origin = {\n      lat: this.position.coords.latitude,\n      lng: this.position.coords.longitude,\n    },\n    north = { lat: 90, lng: this.position.coords.longitude }\n  ) {\n    return this.getBearingToDestination(origin, north);\n  }\n\n  /**\n   * Get the bearings towards the destination\n   *\n   * @param {object} origin - {lat, lng}\n   * @param {object} destination - {lat, lng}\n   */\n  getBearingToDestination(\n    origin = {\n      lat: this.position.coords.latitude,\n      lng: this.position.coords.longitude,\n    },\n    destination\n  ) {\n    const angleToDestination = this.getBearing(origin, destination);\n    return this.deviceAngleDelta + angleToDestination;\n  }\n\n  /**\n   * Handles changes created by the device orientation changes\n   * assumes that the phone is in a portrait mode, with the display up\n   * towards the sky as if you were holding an actual compass\n   *\n   * @callback\n   * @param {object} evt - the event object of the device orientation\n   */\n  deviceOrientationHandler(evt) {\n    if (evt.webkitCompassHeading)\n      //iphone\n      this.deviceAngleDelta = 360 - evt.webkitCompassHeading;\n    else if (evt.alpha)\n      //android\n      this.deviceAngleDelta = evt.alpha;\n    else {\n      console.log(\"compass direction not found\");\n    }\n\n    // console.log(this.deviceAngleDelta)\n    this.deviceAngleDelta = Math.round(this.deviceAngleDelta);\n  }\n\n  /**\n   * get the position of the user\n   *\n   * @async\n   * @function\n   */\n  getPosition() {\n    return new Promise((resolve, reject) => {\n      navigator.geolocation.getCurrentPosition((position) => {\n        if (position) {\n          resolve(position);\n        } else {\n          reject(\"no position found\");\n        }\n      });\n    });\n  }\n\n  /**\n   * watches the geolocation of the user\n   *\n   * @async\n   * @function\n   */\n  watchPosition() {\n    return new Promise((resolve, reject) => {\n      this.geolocationID = navigator.geolocation.watchPosition((position) => {\n        if (position) {\n          this.position = position;\n          // console.log(\n          //   this.position.coords.latitude,\n          //   this.position.coords.longitude\n          // );\n          resolve(position);\n        } else {\n          reject(\"no position found\");\n        }\n      });\n    });\n  }\n\n  /**\n   * Stops watching the user location\n   *\n   * @function\n   */\n  stopTracking() {\n    navigator.geolocation.clearWatch(this.geolocationID);\n    console.log(\"stopped tracking location\");\n  }\n\n  /**\n   * Calculates the angle given a latitude and longitude position\n   *\n   * @function\n   * @param {number} userLat - user latitude\n   * @param {number} userLon - user longitude\n   * @param {number} desiredLat - desired latitude\n   * @param {number} desiredLon - desired longitude\n   */\n  calculateAngle(userLat, userLon, desiredLat, desiredLon) {\n    return Math.atan2(desiredLon - userLon, desiredLat - userLat);\n  }\n\n  /**\n   * Helper function that allows calling a callback from an promise function\n   *\n   * @param {promise} promise\n   * @param {callback} callback\n   */\n  callCallback(promise, callback) {\n    if (callback) {\n      promise\n        .then((result) => {\n          callback(undefined, result);\n          return result;\n        })\n        .catch((error) => {\n          callback(error);\n          return error;\n        });\n    }\n    return promise;\n  }\n}\n\nmodule.exports = Compass;\n","var api = require(\"!../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\");\n            var content = require(\"!!../../node_modules/css-loader/dist/cjs.js!../../node_modules/sass-loader/dist/cjs.js!./Compass.scss\");\n\n            content = content.__esModule ? content.default : content;\n\n            if (typeof content === 'string') {\n              content = [[module.id, content, '']];\n            }\n\nvar options = {};\n\noptions.insert = \"head\";\noptions.singleton = false;\n\nvar update = api(content, options);\n\n\n\nmodule.exports = content.locals || {};","module.exports = \"<svg id=\\\"emoji__compass\\\" viewBox=\\\"0 0 72 72\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><g id=\\\"line-supplement\\\"><polyline points=\\\"33.2 33.2 48 24 38.8 38.8\\\"></polyline></g><g id=\\\"line\\\"><circle cx=\\\"36\\\" cy=\\\"36\\\" r=\\\"24\\\" fill=\\\"none\\\" stroke=\\\"#000\\\" stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" stroke-width=\\\"2\\\"></circle><polyline fill=\\\"none\\\" stroke=\\\"#000\\\" stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" stroke-width=\\\"2\\\" points=\\\"33.2 33.2 24 48 38.8 38.8\\\"></polyline><polyline fill=\\\"none\\\" stroke=\\\"#000\\\" stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" stroke-width=\\\"2\\\" points=\\\"33.2 33.2 48 24 38.8 38.8\\\"></polyline><line x1=\\\"36\\\" x2=\\\"36\\\" y1=\\\"21\\\" y2=\\\"16\\\" fill=\\\"none\\\" stroke=\\\"#000\\\" stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" stroke-width=\\\"2\\\"></line><line x1=\\\"36\\\" x2=\\\"36\\\" y1=\\\"56\\\" y2=\\\"51\\\" fill=\\\"none\\\" stroke=\\\"#000\\\" stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" stroke-width=\\\"2\\\"></line><line x1=\\\"51\\\" x2=\\\"56\\\" y1=\\\"36\\\" y2=\\\"36\\\" fill=\\\"none\\\" stroke=\\\"#000\\\" stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" stroke-width=\\\"2\\\"></line><line x1=\\\"16\\\" x2=\\\"21\\\" y1=\\\"36\\\" y2=\\\"36\\\" fill=\\\"none\\\" stroke=\\\"#000\\\" stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" stroke-width=\\\"2\\\"></line></g></svg>\"","const c = require('./Compass/Compass.js');\nmodule.exports = c;"],"sourceRoot":""}

Back to list