Greasy Fork is available in English.

AppleGuessr

Adds Apple Look Around to GeoGuessr

// ==UserScript==
// @name         AppleGuessr
// @namespace    https://greasyfork.org/en/users/946023-mistystar
// @version      2.1
// @description  Adds Apple Look Around to GeoGuessr
// @author       Mistystar (Mistystar#2205, https://github.com/kittenz) & stocc (stocc#2919, https://github.com/stocc)
// @match        https://www.geoguessr.com/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=geoguessr.com
// @grant        none
// @license      MIT
// @run-at       document-start
// @require 	 https://cdn.jsdelivr.net/npm/protobufjs@7.0.0/dist/protobuf.js
// @require		 https://cdn.jsdelivr.net/npm/long@5/umd/index.js
// @require 	 https://cdn.jsdelivr.net/gh/chebum/heic2any@2c517409ac73e86e92560312b58fcfd565ad7393/dist/heic2any.min.js

// ==/UserScript==



/******/ (() => { // webpackBootstrap
/******/ 	"use strict";
/******/ 	var __webpack_modules__ = ({

/***/ 297:
/***/ (function(__unused_webpack_module, exports, __webpack_require__) {


// Blatantly stolen from https://github.com/sk-zk/lookaround-map/blob/main/static/auth.js
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
    return new (P || (P = Promise))(function (resolve, reject) {
        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
        function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
        step((generator = generator.apply(thisArg, _arguments || [])).next());
    });
};
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
    if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
    if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
    return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
};
var _Authenticator_instances, _Authenticator_generateSessionId, _Authenticator_generateTokenP3;
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.Authenticator = void 0;
const options_1 = __webpack_require__(944);
const proto_1 = __webpack_require__(224);
const TOKEN_P1 = "4cjLaD4jGRwlQ9U";
const MANIFEST_URL = "https://gspe35-ssl.ls.apple.com/geo_manifest/dynamic/config?application=geod" +
    "&application_version=1&country_code=US&hardware=MacBookPro11,2&os=osx" +
    "&os_build=20B29&os_version=11.0.1";
var GLOBAL_TOKENP2 = undefined;
class Authenticator {
    constructor() {
        _Authenticator_instances.add(this);
        this.sessionId = null;
    }
    init() {
        return __awaiter(this, void 0, void 0, function* () {
            yield this.refreshCredentials();
        });
    }
    refreshCredentials() {
        return __awaiter(this, void 0, void 0, function* () {
            this.sessionId = __classPrivateFieldGet(this, _Authenticator_instances, "m", _Authenticator_generateSessionId).call(this);
        });
    }
    hasSession() {
        return this.sessionId != null;
    }
    getTokenP2() {
        return __awaiter(this, void 0, void 0, function* () {
            if (GLOBAL_TOKENP2 == undefined) {
                GLOBAL_TOKENP2 = (yield this.getResourceManifest()).tokenP2;
            }
            return GLOBAL_TOKENP2;
        });
    }
    authenticateUrl(url) {
        return __awaiter(this, void 0, void 0, function* () {
            const urlObj = new URL(url);
            let p2 = yield this.getTokenP2();
            const tokenP3 = __classPrivateFieldGet(this, _Authenticator_instances, "m", _Authenticator_generateTokenP3).call(this);
            const token = TOKEN_P1 + p2 + tokenP3;
            const timestamp = Math.floor(Date.now() / 1000) + 4200;
            const separator = urlObj.search ? "&" : "?";
            let urlPath = urlObj.pathname;
            if (urlObj.search) {
                urlPath += urlObj.search;
            }
            const plaintext = `${urlPath}${separator}sid=${this.sessionId}${timestamp}${tokenP3}`;
            const plaintextBytes = new TextEncoder().encode(plaintext);
            const key = yield sha256(token);
            const ciphertext = yield aes(key, plaintextBytes);
            const ciphertextB64 = btoa(String.fromCharCode(...new Uint8Array(ciphertext)));
            const ciphertextUrl = encodeURIComponent(ciphertextB64);
            const accessKey = `${timestamp}_${tokenP3}_${ciphertextUrl}`;
            const final = `${url}${separator}sid=${this.sessionId}&accessKey=${accessKey}`;
            return final;
        });
    }
    getResourceManifest() {
        return __awaiter(this, void 0, void 0, function* () {
            const response = yield fetch(options_1.CORS_PROXY + MANIFEST_URL);
            let pb = yield response.arrayBuffer();
            return yield proto_1.default.parseResourceManifest(pb);
        });
    }
}
exports.Authenticator = Authenticator;
_Authenticator_instances = new WeakSet(), _Authenticator_generateSessionId = function _Authenticator_generateSessionId() {
    let id = "";
    for (let i = 0; i < 40; i++) {
        const digit = (Math.random() * 10) | 0;
        id += digit.toString();
    }
    return id;
}, _Authenticator_generateTokenP3 = function _Authenticator_generateTokenP3() {
    const chars = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ".split("");
    let token = "";
    for (let i = 0; i < 16; i++) {
        const idx = (Math.random() * chars.length) | 0;
        token += chars[idx];
    }
    return token;
};
function sha256(message) {
    return __awaiter(this, void 0, void 0, function* () {
        const msgBuffer = new TextEncoder().encode(message);
        const hashBuffer = yield window.crypto.subtle.digest("SHA-256", msgBuffer);
        return hashBuffer;
    });
}
function aes(key, encodedMessage) {
    return __awaiter(this, void 0, void 0, function* () {
        const iv = new Uint8Array(16); // 16 zeroes
        const cryptoKey = yield window.crypto.subtle.importKey("raw", key, { name: "AES-CBC" }, true, ["encrypt"]);
        return yield window.crypto.subtle.encrypt({
            name: "AES-CBC",
            iv,
        }, cryptoKey, encodedMessage);
    });
}
//# sourceMappingURL=auth.js.map

/***/ }),

/***/ 97:
/***/ ((__unused_webpack_module, exports) => {


Object.defineProperty(exports, "__esModule", ({ value: true }));
const TILE_SIZE = 256;
class GeoUtils {
    static haversineDistance(coords1, coords2) {
        function toRad(x) {
            return x * Math.PI / 180;
        }
        var lon1 = coords1[0];
        var lat1 = coords1[1];
        var lon2 = coords2[0];
        var lat2 = coords2[1];
        var R = 6371; // km
        var x1 = lat2 - lat1;
        var dLat = toRad(x1);
        var x2 = lon2 - lon1;
        var dLon = toRad(x2);
        var a = Math.sin(dLat / 2) * Math.sin(dLat / 2) +
            Math.cos(toRad(lat1)) * Math.cos(toRad(lat2)) *
                Math.sin(dLon / 2) * Math.sin(dLon / 2);
        var c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a));
        var d = R * c;
        return d;
    }
    static radians_to_degrees(radians) {
        var pi = Math.PI;
        return radians * (180 / pi);
    }
    static headingFromUnknowns(unknown10, unknown11) {
        let westmin = 1;
        let westmax = 2159;
        let eastmin = 16383; // looking (north/south) and very slightly east
        let eastmax = 14318; // looking slightly (north/south) directly east
        let northmin = 8204; // this is likely lower
        let northmax = 6054;
        let southmin = 8204; // this is likely lower
        let southmax = 10173;
        var ew = 0;
        if (unknown10 < westmax) {
            ew = -((unknown10 - westmin) / (westmax - westmin));
        }
        else if (unknown10 > eastmax) {
            ew = ((unknown10 - eastmin) / (eastmax - eastmin));
        }
        var ns = 0;
        if (unknown11 <= northmin) {
            ns = ((unknown11 - northmin) / (northmax - northmin));
        }
        else {
            ns = -((unknown11 - southmin) / (southmax - southmin));
        }
        var r = GeoUtils.radians_to_degrees(Math.atan2(ew, ns));
        if (r < 0) {
            r += 360;
        }
        return r;
    }
    static mercator_to_wgs84(x, y) {
        let lat = (2 * Math.atan(Math.exp((y - 128) / -(256 / (2 * Math.PI)))) - Math.PI / 2) / (Math.PI / 180);
        let lon = (x - 128) / (256 / 360);
        return [lat, lon];
    }
    static tile_coord_to_wgs84(x, y, z) {
        let scale = 1 << z;
        let pixel_coord = [x * TILE_SIZE, y * TILE_SIZE];
        let world_coord = [pixel_coord[0] / scale, pixel_coord[1] / scale];
        let lat_lon = GeoUtils.mercator_to_wgs84(world_coord[0], world_coord[1]);
        return [lat_lon[0], lat_lon[1]];
    }
    static protobuf_tile_offset_to_wsg84(x_offset, y_offset, tile_x, tile_y) {
        let pano_x = tile_x + (x_offset / 64.0) / (TILE_SIZE - 1);
        let pano_y = tile_y + (255 - (y_offset / 64.0)) / (TILE_SIZE - 1);
        let coords = GeoUtils.tile_coord_to_wgs84(pano_x, pano_y, 17);
        return coords;
    }
    static wgs84_to_mercator(lat, lon) {
        var siny = Math.sin(lat * Math.PI / 180);
        siny = Math.min(Math.max(siny, -0.9999), 0.9999);
        return [
            TILE_SIZE * (0.5 + lon / 360),
            TILE_SIZE * (0.5 - Math.log((1 + siny) / (1 - siny)) / (4 * Math.PI))
        ];
    }
    static wgs84_to_tile_coord(lat, lon, zoom) {
        var scale = 1 << zoom;
        var world_coord = this.wgs84_to_mercator(lat, lon);
        var tile_coord = [
            Math.floor((world_coord[0] * scale) / TILE_SIZE),
            Math.floor((world_coord[1] * scale) / TILE_SIZE)
        ];
        return tile_coord;
    }
    static heading(coords1, coords2) {
        try {
            let c1 = new google.maps.LatLng(coords1[0], coords1[1]);
            let c2 = new google.maps.LatLng(coords2[0], coords2[1]);
            let result = google.maps.geometry.spherical.computeHeading(c1, c2);
            if (result < 0) {
                result += 360;
            }
            return result;
        }
        catch (e) {
            console.log(e);
        }
    }
}
exports["default"] = GeoUtils;
//# sourceMappingURL=geoutils.js.map

/***/ }),

/***/ 590:
/***/ (function(__unused_webpack_module, exports, __webpack_require__) {


// ==UserScript==
// @name         AppleGuessr
// @namespace    https://greasyfork.org/en/users/946023-mistystar
// @version      2.1
// @description  Adds Apple Look Around to GeoGuessr
// @author       Mistystar (Mistystar#2205, https://github.com/kittenz) & stocc (stocc#2919, https://github.com/stocc)
// @match        https://www.geoguessr.com/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=geoguessr.com
// @grant        none
// @license      MIT
// @run-at       document-start
// @require 	 https://cdn.jsdelivr.net/gh/chebum/heic2any@master/dist/heic2any.min.js
// @require 	 https://cdn.jsdelivr.net/npm/protobufjs@7.0.0/dist/protobuf.js
// @require		 https://cdn.jsdelivr.net/npm/long@5/umd/index.js
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
    return new (P || (P = Promise))(function (resolve, reject) {
        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
        function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
        step((generator = generator.apply(thisArg, _arguments || [])).next());
    });
};
Object.defineProperty(exports, "__esModule", ({ value: true }));
// ==/UserScript==
/*
CREDITS

Massive thank you to the following people:
    - skzk#8049 - Without https://github.com/sk-zk/lookaround-map this script would not have been possible to make
    - Jupaoqq#7742 - I learned a lot from looking at Unity Script's source code
    - mattisinthesky#1294 or kowalski - For hosting the lookaround-map in Heroku and helping with issues
    - efefury#0519 and Apfeloxid#1368 - For making the Take A Look Around Germany map
*/
// BEGIN CODE SECTION
const Options = __webpack_require__(944);
const Lookaround = __webpack_require__(215);
const lookaround_1 = __webpack_require__(215);
const geoutils_1 = __webpack_require__(97);
protobuf.util.Long = Long;
protobuf.configure();
const MENU_HTML = (/* unused pure expression or super */ null && (`
<div class="start-standard-game_settings__x94PU">
	<div class="game-settings_default__DIBgs">
		<div class="game-settings_toggleLabel__nipwm">
			<div class="label_sizeXSmall__mFnrR">Apple Look Around</div>
			<span></span>
		</div>
		<div>
			<input type="checkbox" class="apple-look-around-toggle" checked>
		</div>
	</div>
</div>
`));
const isGamePage = () => location.pathname.startsWith("/challenge/") || location.pathname.startsWith("/results/") ||
    location.pathname.startsWith("/game/") || location.pathname.startsWith("/battle-royale/") ||
    location.pathname.startsWith("/duels/") || location.pathname.startsWith("/team-duels/") ||
    location.pathname.startsWith("/bullseye/") ||
    location.pathname.startsWith("/live-challenge/");
function overrideOnLoad(googleScript, observer, overrider) {
    const oldOnload = googleScript.onload;
    googleScript.onload = (event) => {
        const google = window.google;
        if (google) {
            observer.disconnect();
            overrider(google);
        }
        if (oldOnload) {
            oldOnload.call(googleScript, event);
        }
    };
}
function grabGoogleScript(mutations) {
    for (const mutation of mutations) {
        for (const newNode of mutation.addedNodes /* Please shut up, it works in JS so it must work here as well */) {
            const asScript = newNode;
            if (asScript && asScript.src && asScript.src.startsWith("https://maps.googleapis.com/")) {
                return asScript;
            }
        }
    }
    return null;
}
function injecter(overrider) {
    if (document.documentElement) {
        injecterCallback(overrider);
    }
    else {
        alert("Script didn't load, refresh to try loading the script");
    }
}
function injecterCallback(overrider) {
    new MutationObserver((mutations, observer) => {
        const googleScript = grabGoogleScript(mutations);
        if (googleScript) {
            overrideOnLoad(googleScript, observer, overrider);
        }
    }).observe(document.documentElement, { childList: true, subtree: true });
}
// End Script injection --------------------------------------------------------------s
function injectMenu() {
    const inject = () => {
        if (document.querySelector(".apple-look-around-toggle") !== null)
            return;
        const settingsSection = document.querySelector('.section_sectionMedium__yXgE6');
        if (settingsSection === null)
            return;
        settingsSection.insertAdjacentHTML("beforeend", MENU_HTML);
        const checkbox = document.querySelector(".apple-look-around-toggle");
        if (checkbox) {
            let isChecked = localStorage.getItem("applelookaroundchecked");
            if (isChecked === null) {
                checkbox.checked = false;
                localStorage.setItem("applelookaroundchecked", "false");
            }
            else if (isChecked === "true") {
                checkbox.checked = true;
            }
            else {
                checkbox.checked = false;
            }
            checkbox.addEventListener("change", (event) => {
                if (event.currentTarget === null)
                    return;
                if (event.currentTarget.checked) {
                    localStorage.setItem("applelookaroundchecked", "true");
                }
                else {
                    localStorage.setItem("applelookaroundchecked", "false");
                }
            });
        }
    };
    // We want the page to be loaded before trying to inject anything
    let documentLoadedInterval = setInterval(function () {
        if (document.readyState === "complete") {
            clearInterval(documentLoadedInterval);
            inject();
        }
    }, 100);
}
// ----------------------------------------------------------------------------
// Sate vars
// TODO: Is there a better way to do this?
var loadingInProgress = false;
var currentPano = new lookaround_1.PanoInfo("", "", "", 0, 0, 0);
var currentlyLoadedPanoTiles = [];
var curNeighbors = [];
// When moving, this is used to keep the current viewport while loading the next pano
var oldHeading = 0;
// ----------------------------------------------------------------------------
// Google Maps API callbacks
// Return a pano image given the panoID.
const getCustomPanoramaTileUrl = (pano, zoom, tileX, tileY) => {
    // Currently loading first image in a round, return a blank image
    //if (pano.startsWith("r")){
    if (currentlyLoadedPanoTiles.length === 0) {
        return "";
    }
    return currentlyLoadedPanoTiles[tileX];
};
const getPano = (pano) => {
    let rp = Options.RESOLUTION_PROFILES[Options.RESOLUTION_SETTING];
    let fullWidth = 2 * rp.big.width + 2 * rp.small.width - 4 * rp.overlap;
    return {
        location: {
            pano: pano,
            description: "Apple Look Around",
            latLng: new google.maps.LatLng(currentPano.lat, currentPano.lon),
        },
        links: [],
        // The text for the copyright control.
        copyright: "(C) Apple",
        // The definition of the tiles for this panorama.
        tiles: {
            tileSize: new google.maps.Size(Math.round(fullWidth / 4), Math.round(Options.EXTENSION_FACTOR * rp.big.height)),
            worldSize: new google.maps.Size(fullWidth, Math.round(rp.big.height * Options.EXTENSION_FACTOR)),
            // The heading in degrees at the origin of the panorama
            // tile set.
            centerHeading: function () {
                // While loading: use the old heading so that when moving, you keep the same viewport while loading the next pano
                if (loadingInProgress) {
                    return oldHeading;
                }
                else {
                    var newHeading = (currentPano.heading + Options.HEADING_CALIBRATION) % 360;
                    oldHeading = newHeading;
                    return newHeading;
                }
            }(),
            getTileUrl: getCustomPanoramaTileUrl,
        },
    };
};
// ----------------------------------------------------------------------------
// Init
function initLookAround() {
    google.maps.StreetViewPanorama = class extends google.maps.StreetViewPanorama {
        constructor(...args) {
            super(...args);
            let isChecked = localStorage.getItem("applelookaroundchecked");
            if (isChecked === "true") {
                this.registerPanoProvider(getPano);
                // Position is being changed by GeoGuessr at the beginning of each round. this.getPosition() contains lat/lng of round.
                this.addListener("position_changed", () => {
                    console.log("Position changed " + this.getPosition());
                    try {
                        // Detect if this is a new round. Normally, currentPano is already updated if this is a move in the same round.
                        if ((this.getPosition().lat() === currentPano.lat && this.getPosition().lng() === currentPano.lon)) {
                            console.log("Position is currentPano => same round");
                            return;
                        }
                        console.warn("Position actually changed => new round; full reload");
                        currentlyLoadedPanoTiles = []; // Causes black screen again
                        this.getFirstPanoId();
                    }
                    catch (e) {
                        console.error(e);
                    }
                });
                // Called after setPano(). If the pano is "r<panoId>/<regioId>", then we load the tiles for that pano.
                // If it doesn't start with "r", then loading is done.
                this.addListener("pano_changed", () => {
                    console.log("Pano changed " + this.getPano());
                    if (this.getPano() != null && this.getPano() != currentPano.panoFullId() && this.getPano() != "" && this.getPano().startsWith("r")) {
                        console.log("New pano requested " + this.getPano());
                        try {
                            this.beginLoadingPanos(this, this.getPano().replace("r", ""));
                        }
                        catch (_a) { }
                    }
                });
                this.addListener("links_changed", () => {
                    console.log("Links changed " + this.getLinks());
                    if (!this.getPano().startsWith("r") && curNeighbors != null) {
                        //this.getLinks().push(curNeighbors[0])
                        let neighborLinks = curNeighbors.map(neighbor => {
                            return {
                                "descripton": "",
                                "pano": "r" + neighbor.panoFullId(),
                                "heading": Math.round(geoutils_1.default.heading([neighbor.lat, neighbor.lon], [currentPano.lat, currentPano.lon]) + 180) % 360,
                            };
                        });
                        console.log("Pushing Links " + neighborLinks.length);
                        for (const neighbor of neighborLinks) {
                            if (neighbor.pano != "") {
                                this.getLinks().push(neighbor);
                            }
                        }
                    }
                });
            }
        }
        getFirstPanoId() {
            return __awaiter(this, void 0, void 0, function* () {
                let isChecked = localStorage.getItem("applelookaroundchecked");
                if (isChecked !== "true")
                    return;
                try {
                    let lat = this.position.lat();
                    let lon = this.position.lng();
                    let lookAroundPanoId, regionId;
                    let closestObject = yield Lookaround.getClosestPanoAtCoords(lat, lon);
                    lookAroundPanoId = closestObject.panoId;
                    regionId = closestObject.regionId;
                    // Request pano to load
                    currentPano = closestObject;
                    this.setPano("r" + lookAroundPanoId + "/" + regionId);
                }
                catch (_a) { }
            });
        }
        // param panoFullId is "panoId/regionId"
        beginLoadingPanos(_t, panoFullId) {
            return __awaiter(this, void 0, void 0, function* () {
                if (loadingInProgress)
                    return;
                //console.warn("http://localhost:5000/#c=17/"+currentPano.lat+"/"+currentPano.lon+"&p="+currentPano.lat+"/"+currentPano.lon);
                // Moved. Find the selected neigbor from ID.
                if (curNeighbors.length > 0) {
                    let selectedNeighbor = curNeighbors.filter(n => n.panoFullId() == panoFullId)[0];
                    if (selectedNeighbor != null) {
                        currentPano = selectedNeighbor;
                    }
                }
                console.log("Start loading Panos");
                loadingInProgress = true;
                let pano0 = Lookaround.loadTileForPano(panoFullId, 0);
                let pano1 = Lookaround.loadTileForPano(panoFullId, 1);
                let pano2 = Lookaround.loadTileForPano(panoFullId, 2);
                let pano3 = Lookaround.loadTileForPano(panoFullId, 3);
                curNeighbors = yield (yield Lookaround.getNeighbors(currentPano));
                loadingInProgress = false;
                currentlyLoadedPanoTiles = [yield pano0, yield pano1, yield pano2, yield pano3];
                // Set another panoId to refresh the view
                this.setPano(panoFullId);
            });
        }
    };
}
function launchObserver() {
    initLookAround();
    //let observer3 = new MutationObserver((mutations) => {
    //	const PATH_NAME = window.location.pathname;
    //	if (PATH_NAME.startsWith("/maps/") && PATH_NAME.endsWith("/play")) { // Inject the options menu if the path name is /maps/XXXXXXX/play
    //		//injectMenu();
    //	}
    //});
    //observer3.observe(document.body, {childList: true, subtree: true, attributes: false, characterData: false});
}
function onLoad() {
    let isChecked = localStorage.getItem("applelookaroundchecked");
    if (isChecked === null) {
        localStorage.setItem("applelookaroundchecked", "true");
    }
    //const PATH_NAME = window.location.pathname;
    //if (PATH_NAME.startsWith("/maps/") && PATH_NAME.endsWith("/play")) { // Inject the options menu if the path name is /maps/XXXXXXX/play
    //	//injectMenu();
    //}
    injecter(() => {
        launchObserver();
    });
}
(function () {
    onLoad();
})();
window.onload = onLoad;
//# sourceMappingURL=index.js.map

/***/ }),

/***/ 215:
/***/ (function(__unused_webpack_module, exports, __webpack_require__) {


var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
    return new (P || (P = Promise))(function (resolve, reject) {
        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
        function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
        step((generator = generator.apply(thisArg, _arguments || [])).next());
    });
};
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.getNeighbors = exports.getClosestPanoAtCoords = exports.loadTileForPano = exports.PanoInfo = void 0;
const Options = __webpack_require__(944);
const auth_1 = __webpack_require__(297);
const geoutils_1 = __webpack_require__(97);
const proto_1 = __webpack_require__(224);
const auth = new auth_1.Authenticator();
var tileCache = {};
class PanoInfo {
    constructor(date, panoId, regionId, heading, lat, lon) {
        this.date = date;
        this.panoId = panoId;
        this.regionId = regionId;
        this.heading = heading;
        this.lat = lat;
        this.lon = lon;
    }
    panoFullId() {
        return this.panoId + "/" + this.regionId;
    }
}
exports.PanoInfo = PanoInfo;
function getCoverageTileRaw(tile_x, tile_y) {
    return __awaiter(this, void 0, void 0, function* () {
        let headers = new Headers({
            "maps-tile-style": "style=57&size=2&scale=0&v=0&preflight=2",
            "maps-tile-x": tile_x.toString(),
            "maps-tile-y": tile_y.toString(),
            "maps-tile-z": "17",
            "maps-auth-token": "w31CPGRO/n7BsFPh8X7kZnFG0LDj9pAuR8nTtH3xhH8=",
        });
        let response = yield (yield fetch(Options.CORS_PROXY + "https://gspe76-ssl.ls.apple.com/api/tile?", { headers: headers })).arrayBuffer();
        let tile = yield proto_1.default.parseMapTile(response);
        return tile;
    });
}
function getCoverageInMapTile(x, y) {
    return __awaiter(this, void 0, void 0, function* () {
        try {
            if (tileCache["" + x + "/" + y]) {
                return tileCache[x + "/" + y];
            }
            let response = yield getCoverageTileRaw(x, y);
            var coverage = [];
            for (let pano of response.pano) {
                let coords = geoutils_1.default.protobuf_tile_offset_to_wsg84(pano.unknown4.longitudeOffset, pano.unknown4.latitudeOffset, x, y);
                let p = new PanoInfo(pano.timestamp.toString(), pano.panoid.toString(), response.unknown13[pano.regionIdIdx].regionId.toString(), geoutils_1.default.headingFromUnknowns(pano.unknown4.unknown10, pano.unknown4.unknown11), coords[0], coords[1]);
                coverage.push(p);
            }
            tileCache["" + x + "/" + y] = coverage;
            return coverage;
        }
        catch (error) {
            console.log(error);
        }
    });
}
function getClosestPanoAtCoords(lat, lon) {
    return __awaiter(this, void 0, void 0, function* () {
        try {
            let tile = geoutils_1.default.wgs84_to_tile_coord(lat, lon, 17);
            let coverage = yield getCoverageInMapTile(tile[0], tile[1]);
            if (coverage.length == 0) {
                return null;
            }
            let smallestDistance = 9999999;
            let closest = null;
            for (let pano of coverage) {
                let distance = geoutils_1.default.haversineDistance([lat, lon], [pano.lat, pano.lon]);
                if (distance < smallestDistance) {
                    smallestDistance = distance;
                    closest = pano;
                }
            }
            return closest;
        }
        catch (error) {
            console.log(error);
            return null;
        }
    });
}
exports.getClosestPanoAtCoords = getClosestPanoAtCoords;
function getNeighbors(panoInfo) {
    return __awaiter(this, void 0, void 0, function* () {
        try {
            let tile = geoutils_1.default.wgs84_to_tile_coord(panoInfo.lat, panoInfo.lon, 17);
            var coverage = yield getCoverageInMapTile(tile[0], tile[1]);
            // TODO Only extend when needed (we're close to the edge of the tile)
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0] + 1, tile[1]));
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0] - 1, tile[1]));
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0], tile[1] + 1));
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0], tile[1] - 1));
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0] - 1, tile[1] - 1));
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0] + 1, tile[1] - 1));
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0] - 1, tile[1] + 1));
            coverage = coverage.concat(yield getCoverageInMapTile(tile[0] + 1, tile[1] + 1));
            coverage = coverage.sort((a, b) => Math.abs(geoutils_1.default.haversineDistance([panoInfo.lat, panoInfo.lon], [a.lat, a.lon])) - Math.abs(geoutils_1.default.haversineDistance([panoInfo.lat, panoInfo.lon], [b.lat, b.lon])));
            coverage = coverage.filter(pano => pano.panoFullId() != panoInfo.panoFullId());
            let minDist = 0.030; // 30 meters
            let maxDist = 0.300; // 300 meters
            coverage = coverage.filter(n => (minDist < Math.abs(geoutils_1.default.haversineDistance([panoInfo.lat, panoInfo.lon], [n.lat, n.lon])) &&
                Math.abs(geoutils_1.default.haversineDistance([panoInfo.lat, panoInfo.lon], [n.lat, n.lon])) < maxDist));
            return coverage.slice(0, 8);
        }
        catch (error) {
            console.log(error);
        }
    });
}
exports.getNeighbors = getNeighbors;
function getUrlForTile(panoFullId, x, resolution) {
    return __awaiter(this, void 0, void 0, function* () {
        try {
            //if (!auth.hasSession()) {
            yield auth.init();
            //}
            let segments = panoFullId.split("/");
            let panoId = segments[0];
            let regionId = segments[1];
            let panoid_padded = panoId.padStart(20, "0");
            let region_id_padded = regionId.padStart(10, "0");
            let panoid_split = panoid_padded.slice(0, 4) + "/" + panoid_padded.slice(4, 8) + "/" + panoid_padded.slice(8, 12) + "/" + panoid_padded.slice(12, 16) + "/" + panoid_padded.slice(16, 20);
            return auth.authenticateUrl(Options.APPLE_MAPS_TILE_ENDPOINT + panoid_split + "/" + region_id_padded + "/t/" + x + "/" + resolution);
        }
        catch (error) {
            console.log(error);
        }
    });
}
// param panoFullId is "panoId/regionId"
function loadTileForPano(panoFullId, x) {
    return __awaiter(this, void 0, void 0, function* () {
        try {
            var jpegblob;
            if (Options.CONVERT_LOCALLY) {
                // Step 1: Get the URL of the tile to load
                // New endpoint /panourl in the python server returns just the Apple URL for the pano
                var appleMapsPanoURL = yield getUrlForTile(panoFullId, x, Options.RESOLUTION_SETTING);
                appleMapsPanoURL = Options.CORS_PROXY + appleMapsPanoURL;
                // Step 2: Load the tile
                //console.log("Requesting tile " + [appleMapsPanoURL])
                var blobres = yield fetch(appleMapsPanoURL);
                var blob = yield blobres.blob();
                // Step 3: Convert from HEIC to JPEG with heic2any
                //console.log("Fetched tile, converting and resizing... " + [appleMapsPanoURL])
                //let startTime = Math.floor(Date.now() / 1000);
                jpegblob = heic2any({ "blob": blob, "type": "image/jpeg" });
            }
            else {
                jpegblob = yield (yield fetch(Options.BASE_URL + "pano/" + panoFullId + "/" + Options.RESOLUTION_SETTING + "/" + x + "/")).blob();
            }
            // Step 4: Process image
            // Cut off the overlap from the right of the tile using canvas
            // and add black bars on top and bottom because we don't have sky/ground tiles
            let rp = Options.RESOLUTION_PROFILES[Options.RESOLUTION_SETTING];
            // Putting the jpeg blob into a canvas to remove 256 px from the right (removes overlap)
            var w = rp.big.width;
            if (x == 1 || x == 3) {
                w = rp.small.width;
            }
            w = w - rp.overlap;
            var canvas = document.createElement('canvas');
            canvas.height = Math.round(Options.EXTENSION_FACTOR * rp.big.height);
            canvas.width = w;
            var ctx = canvas.getContext('2d');
            var img = new Image();
            var result = "";
            img.onload = function () {
                ctx.drawImage(img, 0, (canvas.height - rp.big.height) / 2);
                // This is a big data:image/jpeg;base64, URL
                result = canvas.toDataURL("image/jpeg");
            };
            img.src = URL.createObjectURL(yield jpegblob);
            //let endTime = Math.floor(Date.now() / 1000);
            //console.log("Time to convert: " + (endTime - startTime) + " seconds");
            // Wait for context to finish loading
            // TODO: Is there a better way?
            const delay = ms => new Promise(res => setTimeout(res, ms));
            yield delay(100);
            //let endTime2 = Math.floor(Date.now() / 1000);
            //console.log("Full time: " + (endTime - startTime) + " seconds");
            URL.revokeObjectURL(img.src);
            canvas.remove();
            img.remove();
            return result;
        }
        catch (error) {
            console.log(error);
        }
    });
}
exports.loadTileForPano = loadTileForPano;
//# sourceMappingURL=lookaround.js.map

/***/ }),

/***/ 944:
/***/ ((__unused_webpack_module, exports) => {


Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.CONVERT_LOCALLY = exports.BASE_URL = exports.RESOLUTION_PROFILES = exports.APPLE_MAPS_TILE_ENDPOINT = exports.CORS_PROXY = exports.EXTENSION_FACTOR = exports.HEADING_CALIBRATION = exports.RESOLUTION_SETTING = void 0;
// Determines the resolution of images requested from Apple
// Setting a higher resolution will make rounds load WAY slower, until browsers start to support HEIC
// 0 = highest resolution available, 4 = lowest resolution available.
// Default: 2
const RESOLUTION_SETTING = 2;
exports.RESOLUTION_SETTING = RESOLUTION_SETTING;
// Constant value added to calculated heading to calibrate the GeoGuessr compass
const HEADING_CALIBRATION = 45;
exports.HEADING_CALIBRATION = HEADING_CALIBRATION;
const EXTENSION_FACTOR = 2.12; // TODO Play around with this value for best results with image stretching
exports.EXTENSION_FACTOR = EXTENSION_FACTOR;
const BASE_URL = "https://lookaround.stocc.dev/";
exports.BASE_URL = BASE_URL;
const CONVERT_LOCALLY = true;
exports.CONVERT_LOCALLY = CONVERT_LOCALLY;
const CORS_PROXY = "https://nameless-bastion-28139.herokuapp.com/";
exports.CORS_PROXY = CORS_PROXY;
const APPLE_MAPS_TILE_ENDPOINT = "https://gspe72-ssl.ls.apple.com/mnn_us/";
exports.APPLE_MAPS_TILE_ENDPOINT = APPLE_MAPS_TILE_ENDPOINT;
const RESOLUTION_PROFILES = {
    0: {
        "overlap": 256,
        "big": {
            "width": 5632,
            "height": 4352,
        },
        "small": {
            "width": 3072,
            "height": 4352,
        }
    },
    1: {
        "overlap": 188,
        "big": {
            "width": 4128,
            "height": 3088,
        },
        "small": {
            "width": 2256,
            "height": 3088,
        },
    },
    2: {
        "overlap": 100,
        "big": {
            "width": 2208,
            "height": 1648,
        },
        "small": {
            "width": 1200,
            "height": 1648,
        }
    },
    3: {
        "overlap": 71,
        "big": {
            "width": 1568,
            "height": 1168,
        },
        "small": {
            "width": 848,
            "height": 1168,
        }
    },
    4: {
        "overlap": 50,
        "big": {
            "width": 1104,
            "height": 832,
        },
        "small": {
            "width": 608,
            "height": 832,
        }
    }
};
exports.RESOLUTION_PROFILES = RESOLUTION_PROFILES;
//# sourceMappingURL=options.js.map

/***/ }),

/***/ 224:
/***/ (function(__unused_webpack_module, exports) {


var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
    return new (P || (P = Promise))(function (resolve, reject) {
        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
        function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
        step((generator = generator.apply(thisArg, _arguments || [])).next());
    });
};
Object.defineProperty(exports, "__esModule", ({ value: true }));
let mapTile = `
syntax = "proto3";

message MapTile {
  repeated Pano pano = 1;
  repeated Unknown13 unknown13 = 4;
  repeated Unknown22 unknown22 = 5;
  TileCoordinate tileCoordinate = 6;

  message Pano {
    uint64 panoid = 1;
    int32 unknown1 = 4;
    int64 timestamp = 5; // time the pano was taken
    int32 region_id_idx = 7;
    repeated int32 unknown3 = 9; // goes from 0 to 5. available sizes maybe?
    Unknown4 unknown4 = 10;
    Unknown5 unknown5 = 12;

    message Unknown4 {
      int32 longitude_offset = 1;
      int32 latitude_offset = 2;
      int32 unknown8 = 3;
      int32 unknown9 = 4;
      int32 unknown10 = 5;
      int32 unknown11 = 6;
    }

    message Unknown5 {
      repeated int32 unknown12 = 1;
    }
  }

  message Unknown13 {
    int32 unknown14 = 1;
    // this is the param that appears in pano URLs after the pano ID.
    // no idea what this does exactly.
    int32 region_id = 3;
    int32 unknown15 = 4;
    int32 unknown16 = 5;
    int32 unknown17 = 6;
    int32 unknown18 = 9;
    int32 unknown19 = 10;
    int32 unknown20 = 11;
    int32 unknown21 = 12;
  }

  message Unknown22 {
    int32 unknown23 = 1;
    Unknown24 unknown24 = 4;
    Unknown25 unknown25 = 5;
    int32 unknown26 = 6;

    message Unknown24 {
      int32 unknown27 = 1;
      double unknown28 = 2;
      double unknown29 = 3;
      double unknown30 = 4;
      double unknown31 = 5;
      double unknown32 = 6;
      double unknown33 = 7;
      double unknown34 = 8;
      double unknown35 = 9;
      double unknown36 = 10;
    }

    message Unknown25 {
      double unknown37 = 1;
      double unknown38 = 2;
      double unknown39 = 3;
      double unknown40 = 4;
      double unknown41 = 5;
      double unknown42 = 6;
    }
  }

  message TileCoordinate {
    int32 x = 1;
    int32 y = 2;
    int32 z = 3;
  }

}`;
let resourceManifest = `
syntax = "proto3";

message ResourceManifest {
	repeated StyleConfig style_config = 2;
	string token_p2 = 30;
	string cache_base_url = 31;
	repeated CacheFile cache_file = 72;
	repeated string cache_file_2 = 9;

	message CacheFile {
		string file_name = 2;
	}

	message StyleConfig {
		string url_prefix_1 = 1;
		string url_prefix_2 = 9;
		StyleID style_id = 3;

		enum StyleID {
			_ = 0;
			C3MM_1 = 14;
			C3M = 15;
			DTM_1 = 16;
			DTM_2 = 17;
			C3MM_2 = 52;
		}
	}
}
`;
class Proto {
    static parseResourceManifest(payload) {
        return __awaiter(this, void 0, void 0, function* () {
            const array = new Uint8Array(payload);
            let manifest = protobuf.parse(resourceManifest).root.lookup("ResourceManifest");
            let message = manifest.decode(array);
            return message;
        });
    }
    static parseMapTile(payload) {
        return __awaiter(this, void 0, void 0, function* () {
            const array = new Uint8Array(payload);
            let manifest = protobuf.parse(mapTile).root.lookup("MapTile");
            let message = manifest.decode(array);
            return message;
        });
    }
}
exports["default"] = Proto;
//# sourceMappingURL=proto.js.map

/***/ })

/******/ 	});
/************************************************************************/
/******/ 	// The module cache
/******/ 	var __webpack_module_cache__ = {};
/******/
/******/ 	// The require function
/******/ 	function __webpack_require__(moduleId) {
/******/ 		// Check if module is in cache
/******/ 		var cachedModule = __webpack_module_cache__[moduleId];
/******/ 		if (cachedModule !== undefined) {
/******/ 			return cachedModule.exports;
/******/ 		}
/******/ 		// Create a new module (and put it into the cache)
/******/ 		var module = __webpack_module_cache__[moduleId] = {
/******/ 			// no module.id needed
/******/ 			// no module.loaded needed
/******/ 			exports: {}
/******/ 		};
/******/
/******/ 		// Execute the module function
/******/ 		__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/
/******/ 		// Return the exports of the module
/******/ 		return module.exports;
/******/ 	}
/******/
/************************************************************************/
/******/
/******/ 	// startup
/******/ 	// Load entry module and return exports
/******/ 	// This entry module is referenced by other modules so it can't be inlined
/******/ 	var __webpack_exports__ = __webpack_require__(590);
/******/
/******/ })()
;