Infinite Craft Helper

A script that adds various useful features to Infinite Craft.

// ==UserScript==
// @name			Infinite Craft Helper
// @namespace		mikarific.com
// @match			https://neal.fun/infinite-craft/*
// @version			2.1.4
// @author			Mikarific
// @description		A script that adds various useful features to Infinite Craft.
// @icon			https://i.imgur.com/WlkWOkU.png
// @grant			GM.getValue
// @grant			GM.setValue
// @grant			GM.xmlHttpRequest
// @grant			unsafeWindow
// @run-at			document-end
// @noframes
// @inject-into		page
// @sandbox			raw
// @connect			*
// @supportURL		https://discord.gg/NSMut3Wx3Y
// @homepageURL		https://discord.gg/NSMut3Wx3Y
// @license			MIT
//
// Created with love using Gorilla
// ==/UserScript==

(function () {
	'use strict';

	const whiteFavicon = '';
	const closeIcon = '';
	const discoveriesIcon = '';
	const randomIcon = '';
	const uploadIcon = '';
	const downloadIcon = '';
	const settingsIcon = '';
	const contributeIcon = '';
	const dontContributeIcon = '';
	const shadowIcon = '';
	const logo = '';

	function semverCompare(current, latest) {
	    if (current.startsWith(latest + '-'))
	        return -1;
	    if (latest.startsWith(current + '-'))
	        return 1;
	    return current.localeCompare(latest, undefined, { numeric: true, sensitivity: 'case', caseFirst: 'upper' }) === -1;
	}
	function init$d(elements) {
	    GM.xmlHttpRequest({
	        method: 'GET',
	        url: `https://github.com/Mikarific/InfiniteCraftHelper/raw/main/gorilla.json`,
	        onload: (response) => {
	            if (response.status === 200) {
	                const responseJSON = JSON.parse(response.responseText);
	                if (semverCompare(GM.info.script.version, responseJSON.version)) {
	                    const outdatedModal = document.createElement('dialog');
	                    outdatedModal.classList.add('modal');
	                    elements.container.appendChild(outdatedModal);
	                    const outdatedHeader = document.createElement('div');
	                    outdatedHeader.classList.add('modal-header');
	                    const outdatedTitle = document.createElement('h1');
	                    outdatedTitle.classList.add('modal-title');
	                    outdatedTitle.appendChild(document.createTextNode('Infinite Craft Helper is out of date!'));
	                    outdatedHeader.appendChild(outdatedTitle);
	                    const closeButtonContainer = document.createElement('div');
	                    closeButtonContainer.classList.add('close-button-container');
	                    const closeButton = document.createElement('img');
	                    closeButton.src = closeIcon.trim();
	                    closeButton.classList.add('close-button');
	                    closeButtonContainer.appendChild(closeButton);
	                    outdatedHeader.appendChild(closeButtonContainer);
	                    outdatedModal.appendChild(outdatedHeader);
	                    const versionText = document.createElement('div');
	                    versionText.classList.add('modal-text');
	                    versionText.appendChild(document.createTextNode(`You are on v${GM.info.script.version}! The latest verion is v${responseJSON.version}!`));
	                    outdatedModal.appendChild(versionText);
	                    const buttonContainer = document.createElement('div');
	                    buttonContainer.classList.add('modal-button-container');
	                    const updateButton = document.createElement('a');
	                    updateButton.classList.add('item');
	                    updateButton.href = responseJSON.downloadURL;
	                    updateButton.appendChild(document.createTextNode('Update'));
	                    buttonContainer.appendChild(updateButton);
	                    const continueButton = document.createElement('button');
	                    continueButton.classList.add('item');
	                    continueButton.appendChild(document.createTextNode('Continue Anyways'));
	                    buttonContainer.appendChild(continueButton);
	                    outdatedModal.appendChild(buttonContainer);
	                    outdatedModal.showModal();
	                    closeButton.addEventListener('click', (e) => {
	                        outdatedModal.close();
	                    });
	                    continueButton.addEventListener('click', (e) => {
	                        outdatedModal.close();
	                    });
	                }
	            }
	        },
	    });
	}

	function init$c(elements) {
	    if (elements.favicon !== null) {
	        const whiteFaviconLink = elements.favicon.cloneNode();
	        elements.favicon.media = '(prefers-color-scheme:light)';
	        whiteFaviconLink.media = '(prefers-color-scheme:dark)';
	        whiteFaviconLink.href = whiteFavicon.trim();
	        elements.favicon.after(whiteFaviconLink);
	    }
	}

	const css = `
    .dark-mode .site-title, .dark-mode .instruction-icon, .dark-mode .settings-button, .dark-mode .setting > img, .dark-mode .close-button {
        filter: invert(1);
    }

    .dark-mode {
        scrollbar-color: var(--border-color) #262626;
    }

    .item {
        margin: 4px;
        cursor: pointer;
        padding: 8px 8px 7px;
        border-radius: 5px;
        display: inline-block;
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
        border: 1px solid var(--border-color);
        transition: background .15s linear;
        background: var(--item-bg);
        line-height: 1em;
        color: var(--text-color);
        white-space: nowrap;
    }

    .item:hover {
        background: var(--instance-bg-hover);
        border: 1px solid var(--instance-border-hover);
    }

    @media screen and (min-width: 1150px) {
        .item {
            font-size: 16.4px;
            padding: 9px 10px 8px;
        }
    }

    .pinned {
        max-width: 900px;
        margin-left: auto;
        margin-right: auto;
        padding: 9px;
        border: 0px;
        border-bottom: 1px;
        border-style: solid;
        border-color: var(--border-color);
    }

    .pinned-title {
        margin: 4px;
        font-size: 15px;
        font-family: Roboto, sans-serif;
        color: var(--text-color);
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
        pointer-events: none;
    }

    .sidebar {
        width: var(--sidebar-size) !important;
    }

    .resize-bar {
        position: absolute;
        height: 100%;
        width: 5px;
        right: calc(var(--sidebar-size) - 3px);
        z-index: 10;
        cursor: ew-resize;
    }

    .sidebar-header {
        display: flex;
        position: sticky;
        height: auto !important;
        top: 0px;
        background-color: var(--background-color) !important;
        max-width: 900px;
        margin-left: auto;
        margin-right: auto;
        padding: 9px;
        border: 0px;
        border-bottom: 1px;
        border-style: solid;
        border-color: var(--border-color);
        z-index: 1;
    }

    .sidebar-search {
        width: 100%;
    }

    .sidebar-input {
        height: 40px !important;
        margin: 4px;
        border-radius: 5px;
        border: 1px solid var(--border-color) !important;
        font-family: Roboto, sans-serif;
        background-size: 21px 21px !important;
        background-position: 10px 10px !important;
        color: var(--text-color);
    }

    .settings-details {
        display: flex;
        margin: 4px;
        height: 40px;
        border: 1px solid var(--border-color);
        border-radius: 5px;
    }

    .settings-summary {
        display: flex;
        list-style: none;
    }

    .settings-button {
        height: 40px;
        padding: 8px 8px 7px;
        cursor: pointer;
        opacity: .8;
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
        aspect-ratio: 1/1;
    }

    .settings-button:hover {
        transform: scale(1.05)
    }

    .settings-content {
        display: flex;
        flex-direction: column;
        gap: 5px;
        position: absolute;
        right: 13px;
        padding: 8px 8px 7px;
        border: 1px solid var(--border-color);
        border-radius: 5px;
        background-color: var(--background-color);
    }

    .setting {
        display: flex;
        gap: 5px;
        justify-content: flex-end;
        cursor: pointer;
        padding: 8px 8px 7px;
        border: 1px solid var(--border-color);
        border-radius: 5px;
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
        line-height: 1em;
        font-family: Roboto, sans-serif;
        font-size: 15.4px;
        color: var(--text-color);
    }

    .setting:hover {
        background: var(--instance-bg-hover);
        border: 1px solid var(--instance-border-hover);
    }

    .setting > img {
        height: 1em;
        opacity: .8;
        aspect-ratio: 1 / 1;
    }

    #import-save {
        display: none;
    }

    .logo {
        width: 85px !important;
        right: calc(var(--sidebar-size) + 15px) !important;
    }

    .version {
        position: fixed;
        top: 85px;
        right: calc(var(--sidebar-size) + 15px);
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
        pointer-events: none;
        color: var(--text-color);
        font-family: Roboto, sans-serif;
        font-size: 11px;
    }

    .side-controls {
        right: calc(var(--sidebar-size) + 9px) !important;
        z-index: 1;
    }

    .random, .discoveries-icon {
        width: 21px;
        cursor: pointer;
        opacity: .8;
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
    }

    .random:hover, .discoveries-icon:hover {
        transform: scale(1.05);
    }

    .modal {
        max-width: 75%;
        max-height: 75%;
        margin: auto;
        padding-top: 0px;
        border: 1px solid var(--border-color);
        border-radius: 5px;
        background-color: var(--background-color);
    }

    .modal::backdrop {
        background-color: rgb(0 0 0 / .5);
    }

    .modal-header {
        position: sticky;
        top: 0;
        display: flex;
        gap: 1rem;
        padding-top: 16px;
        padding-bottom: 16px;
        justify-content: space-between;
        background-color: var(--background-color);
    }

    .modal-title {
        font-size: 20px;
        font-family: Roboto, sans-serif;
        line-height: 35px;
        color: var(--text-color);
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
    }

    .modal-text {
        font-size: 15px;
        font-family: Roboto, sans-serif;
        color: var(--text-color);
        text-align: center;
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
        pointer-events: none;
    }

    .display-item {
        margin: 4px;
        padding: 8px 8px 7px;
        border-radius: 5px;
        display: inline-block;
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
        border: 1px solid var(--border-color);
        background: var(--item-bg);
        line-height: 1em;
        white-space: nowrap;
        color: var(--text-color);
    }

    @media screen and (min-width: 1150px) {
        .display-item {
            font-size: 16.4px;
            padding: 9px 10px 8px;
        }
    }

    .recipe {
        display: flex;
        align-items: center;
        color: var(--text-color);
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
    }

    .modal-button-container {
        display: flex;
        align-items: center;
        justify-content: center;
        padding-top: 16px;
        color: var(--text-color);
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
    }

    .modal-button-container > .item {
        font-family: Roboto, sans-serif;
        font-size: 15.4px;
        text-decoration: auto;
    }

    .close-button-container {
        display: flex;
        border: 1px solid var(--border-color);
        border-radius: 5px;
    }

    .close-button {
        height: 35px;
        padding: 8px 8px 7px;
        cursor: pointer;
        opacity: .8;
        -webkit-user-select: none;
        -moz-user-select: none;
        user-select: none;
    }

    .close-button:hover {
        transform: scale(1.05)
    }

    .instance-emoji {
        pointer-events: none;
    }
`;
	function init$b(elements) {
	    elements.styles.appendChild(document.createTextNode(css.trim()));
	    document.getElementsByTagName('head')[0].appendChild(elements.styles);
	}

	function setMiddleClickOnMutations(mutations, elements) {
	    for (const mutation of mutations) {
	        if (mutation.addedNodes.length > 0) {
	            for (const node of mutation.addedNodes) {
	                node.addEventListener('mousedown', (e) => {
	                    e.preventDefault();
	                    if (e instanceof MouseEvent &&
	                        e.button === 1 &&
	                        e.target instanceof HTMLElement &&
	                        (e.target.classList.contains('instance') ||
	                            e.target.classList.contains('instance-discovered-text') ||
	                            e.target.classList.contains('instance-discovered-emoji'))) {
	                        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].playInstanceSound();
	                        const targetElement = e.target.classList.contains('instance-discovered-emoji')
	                            ? e.target.parentElement?.parentElement
	                            : e.target.classList.contains('instance-discovered-text')
	                                ? e.target.parentElement
	                                : e.target;
	                        const { x, y, width, height } = targetElement.getBoundingClientRect();
	                        const data = {
	                            id: unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.instanceId++,
	                            text: targetElement.childNodes[1].textContent?.trim(),
	                            emoji: targetElement.childNodes[0].textContent?.trim(),
	                            discovered: targetElement.classList.contains('instance-discovered'),
	                            disabled: false,
	                            left: x,
	                            top: y,
	                            offsetX: 0.5,
	                            offsetY: 0.5,
	                            hasMoved: false,
	                            fromPanel: false,
	                        };
	                        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance = cloneInto(data, unsafeWindow);
	                        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.instances.push(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance);
	                        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].$nextTick(exportFunction(() => {
	                            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].setInstancePosition(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance, e.clientX - width / 2, e.clientY - height / 2);
	                            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].setInstanceZIndex(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance, data.id);
	                            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance.elem.addEventListener('mouseup', exportFunction((e) => {
	                                if (!unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance.hasMoved) {
	                                    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance.hasMoved =
	                                        true;
	                                    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].calcInstanceSize(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance);
	                                }
	                            }, unsafeWindow));
	                        }, unsafeWindow));
	                        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.mouseDown = true;
	                    }
	                });
	            }
	        }
	    }
	}

	const craftsModal = document.createElement('dialog');
	const craftsTitle = document.createElement('h1');
	const craftsContainer = document.createElement('div');
	let recipes = {};
	async function init$a(elements) {
	    craftsModal.classList.add('modal');
	    elements.container.appendChild(craftsModal);
	    const craftsHeader = document.createElement('div');
	    craftsHeader.classList.add('modal-header');
	    craftsTitle.classList.add('modal-title');
	    craftsTitle.appendChild(document.createTextNode('Crafts'));
	    craftsHeader.appendChild(craftsTitle);
	    const closeButtonContainer = document.createElement('div');
	    closeButtonContainer.classList.add('close-button-container');
	    const closeButton = document.createElement('img');
	    closeButton.src = closeIcon.trim();
	    closeButton.classList.add('close-button');
	    closeButtonContainer.appendChild(closeButton);
	    craftsHeader.appendChild(closeButtonContainer);
	    craftsModal.appendChild(craftsHeader);
	    craftsContainer.classList.add('crafts-container');
	    craftsModal.appendChild(craftsContainer);
	    recipes = JSON.parse((await GM.getValue('recipes')) ?? '{}');
	    delete recipes['Nothing'];
	    for (const recipeKey of Object.keys(recipes)) {
	        for (let i = recipes[recipeKey].length - 1; i >= 0; i--) {
	            if (recipes[recipeKey][i] === undefined ||
	                recipes[recipeKey][i] === null ||
	                recipes[recipeKey][i].length < 2 ||
	                recipes[recipeKey][i][0].text === recipeKey ||
	                recipes[recipeKey][i][1].text === recipeKey) {
	                recipes[recipeKey].splice(i, 1);
	            }
	        }
	    }
	    await GM.setValue('recipes', JSON.stringify(recipes));
	    closeButton.addEventListener('click', (e) => {
	        craftsModal.close();
	    });
	}
	async function addElementToCrafts(first, second, result, loading = false) {
	    const ingredients = [first, second].sort((a, b) => {
	        return a.text.localeCompare(b.text);
	    });
	    if (recipes[result] === undefined)
	        recipes[result] = [];
	    if (recipes[result].find((recipe) => recipe[0].text === ingredients[0].text && recipe[1].text === ingredients[1].text) !== undefined)
	        return;
	    recipes[result].push([
	        {
	            text: ingredients[0].text,
	            emoji: ingredients[0].emoji ?? '⬜',
	        },
	        {
	            text: ingredients[1].text,
	            emoji: ingredients[1].emoji ?? '⬜',
	        },
	    ]);
	    if (!loading)
	        await GM.setValue('recipes', JSON.stringify(recipes));
	}
	function openCraftsForElement(element) {
	    craftsTitle.innerHTML = '';
	    const titleEmoji = document.createElement('span');
	    titleEmoji.classList.add('display-item-emoji');
	    titleEmoji.appendChild(document.createTextNode(element.emoji ?? '⬜'));
	    craftsTitle.appendChild(titleEmoji);
	    craftsTitle.appendChild(document.createTextNode(` ${element.text} `));
	    craftsContainer.innerHTML = '';
	    const elementRecipes = recipes[element.text];
	    if (elementRecipes === undefined) {
	        const recipesEmpty = document.createElement('div');
	        recipesEmpty.classList.add('modal-text');
	        recipesEmpty.appendChild(document.createTextNode("I don't know how to craft this element!"));
	        craftsContainer.appendChild(recipesEmpty);
	    }
	    else {
	        const recipeKeys = Object.keys(recipes);
	        for (const elementRecipe of elementRecipes) {
	            const recipeDiv = document.createElement('div');
	            recipeDiv.classList.add('recipe');
	            const firstDiv = document.createElement('div');
	            recipeKeys.includes(elementRecipe[0].text)
	                ? firstDiv.classList.add('item')
	                : firstDiv.classList.add('display-item');
	            const firstEmoji = document.createElement('span');
	            recipeKeys.includes(elementRecipe[0].text)
	                ? firstEmoji.classList.add('item-emoji')
	                : firstEmoji.classList.add('display-item-emoji');
	            firstEmoji.appendChild(document.createTextNode(elementRecipe[0].emoji ?? '⬜'));
	            firstDiv.appendChild(firstEmoji);
	            firstDiv.appendChild(document.createTextNode(` ${elementRecipe[0].text} `));
	            if (recipeKeys.includes(elementRecipe[0].text)) {
	                firstDiv.addEventListener('click', () => {
	                    openCraftsForElement(elementRecipe[0]);
	                });
	            }
	            recipeDiv.appendChild(firstDiv);
	            recipeDiv.appendChild(document.createTextNode('+'));
	            const secondDiv = document.createElement('div');
	            recipeKeys.includes(elementRecipe[1].text)
	                ? secondDiv.classList.add('item')
	                : secondDiv.classList.add('display-item');
	            const secondEmoji = document.createElement('span');
	            recipeKeys.includes(elementRecipe[1].text)
	                ? secondEmoji.classList.add('item-emoji')
	                : secondEmoji.classList.add('display-item-emoji');
	            secondEmoji.appendChild(document.createTextNode(elementRecipe[1].emoji ?? '⬜'));
	            secondDiv.appendChild(secondEmoji);
	            secondDiv.appendChild(document.createTextNode(` ${elementRecipe[1].text} `));
	            if (recipeKeys.includes(elementRecipe[1].text)) {
	                secondDiv.addEventListener('click', () => {
	                    openCraftsForElement(elementRecipe[1]);
	                });
	            }
	            recipeDiv.appendChild(secondDiv);
	            recipeDiv.appendChild(document.createTextNode('='));
	            const resultDiv = document.createElement('div');
	            resultDiv.classList.add('display-item');
	            const resultEmoji = document.createElement('span');
	            resultEmoji.classList.add('display-item-emoji');
	            resultEmoji.appendChild(document.createTextNode(element.emoji ?? '⬜'));
	            resultDiv.appendChild(resultEmoji);
	            resultDiv.appendChild(document.createTextNode(` ${element.text} `));
	            recipeDiv.appendChild(resultDiv);
	            craftsContainer.appendChild(recipeDiv);
	        }
	    }
	    craftsModal.showModal();
	}
	async function resetCrafts() {
	    recipes = {};
	    await GM.setValue('recipes', '{}');
	}

	const discoveriesModal = document.createElement('dialog');
	const discoveriesHeader = document.createElement('div');
	const discoveriesEmpty = document.createElement('div');
	function init$9(elements) {
	    const discoveriesImage = document.createElement('img');
	    discoveriesImage.src = discoveriesIcon.trim();
	    discoveriesImage.classList.add('discoveries-icon');
	    elements.sideControls.appendChild(discoveriesImage);
	    discoveriesModal.classList.add('modal');
	    elements.container.appendChild(discoveriesModal);
	    discoveriesHeader.classList.add('modal-header');
	    const discoveriesTitle = document.createElement('h1');
	    discoveriesTitle.classList.add('modal-title');
	    discoveriesTitle.appendChild(document.createTextNode('Your First Discoveries'));
	    discoveriesHeader.appendChild(discoveriesTitle);
	    const closeButtonContainer = document.createElement('div');
	    closeButtonContainer.classList.add('close-button-container');
	    const closeButton = document.createElement('img');
	    closeButton.src = closeIcon.trim();
	    closeButton.classList.add('close-button');
	    closeButtonContainer.appendChild(closeButton);
	    discoveriesHeader.appendChild(closeButtonContainer);
	    discoveriesModal.appendChild(discoveriesHeader);
	    discoveriesEmpty.classList.add('modal-text');
	    discoveriesEmpty.appendChild(document.createTextNode("You don't have any first discoveries!"));
	    discoveriesModal.appendChild(discoveriesEmpty);
	    const discoveredElements = cloneInto(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements, unsafeWindow).filter((el) => el.discovered === true);
	    for (const discoveredElement of discoveredElements) {
	        addElementToDiscoveries(discoveredElement);
	    }
	    discoveriesImage.addEventListener('click', (e) => {
	        discoveriesModal.showModal();
	    });
	    closeButton.addEventListener('click', (e) => {
	        discoveriesModal.close();
	    });
	}
	function addElementToDiscoveries(element) {
	    const recipeKeys = Object.keys(recipes);
	    const elementDiv = document.createElement('div');
	    recipeKeys.includes(element.text) ? elementDiv.classList.add('item') : elementDiv.classList.add('display-item');
	    const elementEmoji = document.createElement('span');
	    recipeKeys.includes(element.text)
	        ? elementEmoji.classList.add('item-emoji')
	        : elementEmoji.classList.add('display-item-emoji');
	    elementEmoji.appendChild(document.createTextNode(element.emoji ?? '⬜'));
	    elementDiv.appendChild(elementEmoji);
	    elementDiv.appendChild(document.createTextNode(` ${element.text} `));
	    if (recipeKeys.includes(element.text)) {
	        elementDiv.addEventListener('click', () => {
	            openCraftsForElement(element);
	        });
	    }
	    discoveriesModal.appendChild(elementDiv);
	    discoveriesEmpty.style.display = 'none';
	}
	function resetDiscoveries() {
	    discoveriesModal.innerHTML = '';
	    discoveriesModal.appendChild(discoveriesHeader);
	    discoveriesModal.appendChild(discoveriesEmpty);
	    discoveriesEmpty.style.display = '';
	}

	let contributeToDatabase = false;
	function init$8(elements) {
	    const settingsDetails = document.createElement('details');
	    settingsDetails.classList.add('settings-details');
	    elements.sidebarHeader.appendChild(settingsDetails);
	    const settingsSummary = document.createElement('summary');
	    settingsSummary.classList.add('settings-summary');
	    settingsDetails.appendChild(settingsSummary);
	    const settingsButton = document.createElement('img');
	    settingsButton.src = settingsIcon.trim();
	    settingsButton.classList.add('settings-button');
	    settingsSummary.appendChild(settingsButton);
	    settingsDetails.appendChild(elements.settingsContent);
	    document.addEventListener('click', function (e) {
	        const target = e.target;
	        if (!settingsDetails.contains(target)) {
	            settingsDetails.removeAttribute('open');
	        }
	    });
	    contributeToDatabase = localStorage.getItem('contributeToDatabase') === 'false' ? false : true;
	    const contributeContainer = document.createElement('div');
	    contributeContainer.classList.add('setting');
	    const contributeText = document.createTextNode('Share Crafts?');
	    contributeContainer.appendChild(contributeText);
	    const contributeImage = document.createElement('img');
	    if (!contributeToDatabase) {
	        contributeImage.src = dontContributeIcon.trim();
	    }
	    else {
	        contributeImage.src = contributeIcon.trim();
	    }
	    contributeContainer.appendChild(contributeImage);
	    elements.settingsContent.appendChild(contributeContainer);
	    contributeContainer.addEventListener('click', (e) => {
	        if (!contributeToDatabase) {
	            contributeToDatabase = true;
	            contributeImage.src = contributeIcon.trim();
	            localStorage.setItem('contributeToDatabase', 'true');
	        }
	        else {
	            contributeToDatabase = false;
	            contributeImage.src = dontContributeIcon.trim();
	            localStorage.setItem('contributeToDatabase', 'false');
	        }
	    });
	}

	const pinnedContainer = document.createElement('div');
	const pinnedTitle = document.createElement('div');
	let pinnedElements = [];
	async function init$7(elements) {
	    pinnedContainer.classList.add('pinned');
	    pinnedTitle.classList.add('pinned-title');
	    pinnedTitle.appendChild(document.createTextNode('Pinned Elements'));
	    pinnedContainer.appendChild(pinnedTitle);
	    pinnedElements = JSON.parse((await GM.getValue('pinned')) ?? '[]');
	    if (pinnedElements.length === 0)
	        pinnedContainer.style.display = 'none';
	    for (const pinnedElement of pinnedElements) {
	        const elementDiv = document.createElement('div');
	        elementDiv.classList.add('item');
	        const elementEmoji = document.createElement('span');
	        elementEmoji.classList.add('item-emoji');
	        elementEmoji.appendChild(document.createTextNode(pinnedElement.emoji ?? '⬜'));
	        elementDiv.appendChild(elementEmoji);
	        elementDiv.appendChild(document.createTextNode(` ${pinnedElement.text} `));
	        elementDiv.addEventListener('mousedown', (e) => {
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].selectElement(e, cloneInto(pinnedElement, unsafeWindow));
	        });
	        pinnedContainer.appendChild(elementDiv);
	    }
	    elements.items.before(pinnedContainer);
	}
	async function pinElement(element, loading = false) {
	    if (pinnedElements.find((el) => el.text === element.text) === undefined) {
	        if (!loading)
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].playInstanceSound();
	        const elementDiv = document.createElement('div');
	        elementDiv.classList.add('item');
	        const elementEmoji = document.createElement('span');
	        elementEmoji.classList.add('item-emoji');
	        elementEmoji.appendChild(document.createTextNode(element.emoji ?? '⬜'));
	        elementDiv.appendChild(elementEmoji);
	        elementDiv.appendChild(document.createTextNode(` ${element.text} `));
	        elementDiv.addEventListener('mousedown', (e) => {
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].selectElement(e, element);
	        });
	        pinnedContainer.appendChild(elementDiv);
	        if (pinnedElements.length === 0)
	            pinnedContainer.style.display = '';
	        pinnedElements.push(element);
	        if (!loading)
	            await GM.setValue('pinned', JSON.stringify(pinnedElements));
	    }
	    else {
	        if (!loading)
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.deleteSound.play();
	        const elementDiv = Array.from(pinnedContainer.querySelectorAll('.item')).find((el) => el.childNodes[1].textContent?.trim() === element.text);
	        elementDiv?.remove();
	        if (pinnedElements.length === 1)
	            pinnedContainer.style.display = 'none';
	        pinnedElements = pinnedElements.filter((el) => el !== element);
	        if (!loading)
	            await GM.setValue('pinned', JSON.stringify(pinnedElements));
	    }
	}
	async function resetPinnedElements() {
	    pinnedContainer.innerHTML = '';
	    pinnedContainer.appendChild(pinnedTitle);
	    pinnedContainer.style.display = 'none';
	    pinnedElements = [];
	    await GM.setValue('pinned', '[]');
	}

	function init$6(elements) {
	    // Detect when fetch is monkeypatched because a certain someone made a tool to upload fake recipes.
	    const iframe = document.createElement('iframe');
	    document.body.appendChild(iframe);
	    const cleanFetch = iframe.contentWindow?.fetch?.toString() ?? '';
	    iframe.remove();
	    // New Element Crafted
	    const getCraftResponse = unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].getCraftResponse;
	    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].getCraftResponse = exportFunction((...args) => new window.Promise(async (resolve) => {
	        const response = await getCraftResponse(...args);
	        const args0 = args[0].wrappedJSObject === undefined ? args[0] : args[0].wrappedJSObject;
	        const args1 = args[1].wrappedJSObject === undefined ? args[1] : args[1].wrappedJSObject;
	        const ingredients = args0.text.localeCompare(args1.text, 'en') === -1 ? [args0, args1] : [args1, args0];
	        const first = ingredients[0];
	        const second = ingredients[1];
	        const result = {
	            text: response.result,
	            emoji: response.emoji,
	            discovered: response.isNew,
	        };
	        if (first.text === '')
	            return resolve(response);
	        if (second.text === '')
	            return resolve(response);
	        if (result.text === '' || result.text === 'Nothing')
	            return resolve(response);
	        if (contributeToDatabase && unsafeWindow.fetch.toString() === cleanFetch) {
	            GM.xmlHttpRequest({
	                method: 'POST',
	                url: `https://infinitecraft.mikarific.com/recipe`,
	                data: JSON.stringify({
	                    first: {
	                        text: first.text,
	                        emoji: first.emoji,
	                    },
	                    second: {
	                        text: second.text,
	                        emoji: second.emoji,
	                    },
	                    result: {
	                        text: result.text,
	                        emoji: result.emoji,
	                    },
	                }),
	                headers: {
	                    'Content-Type': 'application/json',
	                    Origin: 'https://neal.fun/infinite-craft/',
	                },
	            });
	        }
	        addElementToCrafts({
	            text: first.text,
	            emoji: first.emoji,
	        }, {
	            text: second.text,
	            emoji: second.emoji,
	        }, result.text);
	        if (result.discovered) {
	            addElementToDiscoveries(result);
	        }
	        console.log(`${first.text} + ${second.text} = ${result.text}`);
	        resolve(response);
	    }), unsafeWindow);
	    const selectElement = unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].selectElement;
	    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].selectElement = exportFunction((e, element) => {
	        element = element.wrappedJSObject === undefined ? element : element.wrappedJSObject;
	        if (e.button === 2) {
	            openCraftsForElement(element);
	            return;
	        }
	        if (e.altKey) {
	            pinElement(element);
	            return;
	        }
	        return selectElement(e, element);
	    }, unsafeWindow);
	    const instanceObserver = new MutationObserver((mutations) => {
	        setMiddleClickOnMutations(mutations);
	    });
	    instanceObserver.observe(elements.instances, { childList: true, subtree: true });
	    const oldResetButton = document.querySelector('.reset');
	    const resetButton = oldResetButton.cloneNode(true);
	    oldResetButton.parentNode?.replaceChild(resetButton, oldResetButton);
	    resetButton.addEventListener('click', async () => {
	        const confirmation = confirm('Are you sure? This will delete all your progress!');
	        if (confirmation) {
	            localStorage.removeItem('infinite-craft-data');
	            await resetPinnedElements();
	            await resetCrafts();
	            location.reload();
	        }
	    });
	}

	let sidebarSize = 0;
	let resizing = false;
	function onResize() {
	    sidebarSize =
	        window.innerWidth > 800 ? Math.min(Math.min(Math.max(sidebarSize, 305), 900), window.innerWidth - 100) : 0;
	    document.documentElement.style.setProperty('--sidebar-size', `${sidebarSize}px`);
	    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.sidebarSize = sidebarSize;
	    for (const instance of unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.instances) {
	        instance.width || unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].calcInstanceSize(instance),
	            instance.left + instance.width + 10 > window.innerWidth - sidebarSize &&
	                unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].setInstancePosition(instance, window.innerWidth - sidebarSize - instance.width - 10, instance.top),
	            instance.top + instance.height + 10 > window.innerHeight &&
	                unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].setInstancePosition(instance, instance.left, window.innerHeight - instance.height - 10);
	    }
	    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].checkControlsBlur();
	}
	function init$5(elements) {
	    sidebarSize =
	        window.innerWidth > 800
	            ? unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.sidebarSize > 310
	                ? 350
	                : 305
	            : 0;
	    document.documentElement.style.setProperty('--sidebar-size', `${sidebarSize}px`);
	    const resizeBar = document.createElement('div');
	    resizeBar.classList.add('resize-bar');
	    elements.sidebar.after(resizeBar);
	    resizeBar.addEventListener('mousedown', () => {
	        resizing = true;
	    });
	    window.addEventListener('mousemove', (e) => {
	        if (resizing) {
	            sidebarSize =
	                window.innerWidth > 800
	                    ? Math.min(Math.min(Math.max(window.innerWidth - e.clientX, 305), 900), window.innerWidth - 100)
	                    : 13;
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.sidebarSize = sidebarSize;
	            document.documentElement.style.setProperty('--sidebar-size', `${sidebarSize}px`);
	            onResize();
	        }
	    });
	    window.addEventListener('mouseup', () => {
	        resizing = false;
	    });
	    window.removeEventListener('resize', unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].onResize);
	    window.addEventListener('resize', onResize);
	}

	function init$4(elements) {
	    const uploadContainer = document.createElement('label');
	    uploadContainer.setAttribute('for', 'import-save');
	    uploadContainer.classList.add('setting');
	    const uploadInput = document.createElement('input');
	    uploadInput.type = 'file';
	    uploadInput.id = 'import-save';
	    uploadContainer.appendChild(uploadInput);
	    const uploadText = document.createTextNode('Import Save File');
	    uploadContainer.appendChild(uploadText);
	    const uploadImage = document.createElement('img');
	    uploadImage.src = uploadIcon.trim();
	    uploadContainer.appendChild(uploadImage);
	    elements.settingsContent.appendChild(uploadContainer);
	    uploadInput.addEventListener('change', async () => {
	        const file = uploadInput.files !== null ? uploadInput.files[0] : null;
	        if (file === null || file.type !== 'application/json')
	            return;
	        const fileContents = JSON.parse(await file.text());
	        if (!Object.keys(fileContents).includes('elements'))
	            return;
	        const saveFile = [];
	        for (const element of fileContents.elements) {
	            if (!Object.keys(element).includes('text'))
	                continue;
	            const toPush = {
	                text: element.text,
	                discovered: !Object.keys(element).includes('discovered')
	                    ? Object.keys(fileContents).includes('discoveries')
	                        ? fileContents.discoveries.includes(element.text)
	                        : false
	                    : element.discovered,
	            };
	            if (Object.keys(element).includes('emoji'))
	                toPush.emoji = element.emoji;
	            saveFile.push(toPush);
	        }
	        localStorage.setItem('infinite-craft-data', JSON.stringify({
	            elements: saveFile,
	        }));
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements = cloneInto(saveFile, unsafeWindow);
	        await resetCrafts();
	        if (Object.keys(fileContents).includes('recipes')) {
	            for (const recipeKey of Object.keys(fileContents.recipes)) {
	                if (recipeKey !== 'Nothing') {
	                    for (const recipe of fileContents.recipes[recipeKey]) {
	                        if (recipe[0].text !== recipeKey && recipe[1].text !== recipeKey) {
	                            addElementToCrafts({
	                                text: recipe[0].text,
	                                emoji: recipe[0].emoji,
	                            }, {
	                                text: recipe[1].text,
	                                emoji: recipe[1].emoji,
	                            }, recipeKey, true);
	                        }
	                    }
	                }
	            }
	            await GM.setValue('recipes', JSON.stringify(fileContents.recipes));
	        }
	        await resetDiscoveries();
	        const discoveredElements = cloneInto(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements, unsafeWindow).filter((el) => el.discovered === true);
	        for (const discoveredElement of discoveredElements) {
	            addElementToDiscoveries(discoveredElement);
	        }
	        await resetPinnedElements();
	        if (Object.keys(fileContents).includes('pinned')) {
	            for (let pinnedElement of fileContents.pinned) {
	                pinElement(cloneInto(pinnedElement, unsafeWindow), true);
	            }
	            await GM.setValue('pinned', JSON.stringify(fileContents.pinned));
	        }
	    });
	    const downloadContainer = document.createElement('div');
	    downloadContainer.classList.add('setting');
	    const downloadText = document.createTextNode('Export Save File');
	    downloadContainer.appendChild(downloadText);
	    const downloadImage = document.createElement('img');
	    downloadImage.src = downloadIcon.trim();
	    downloadContainer.appendChild(downloadImage);
	    elements.settingsContent.appendChild(downloadContainer);
	    downloadContainer.addEventListener('click', (e) => {
	        const saveFile = JSON.parse(localStorage.getItem('infinite-craft-data') ?? '');
	        saveFile.pinned = pinnedElements;
	        saveFile.recipes = recipes;
	        const downloadLink = document.createElement('a');
	        downloadLink.download = 'infinitecraft.json';
	        downloadLink.href = URL.createObjectURL(new Blob([JSON.stringify(saveFile, null, '\t')], {
	            type: 'application/json',
	        }));
	        downloadLink.dataset.downloadurl = ['application/json', downloadLink.download, downloadLink.href].join(':');
	        downloadLink.style.display = 'none';
	        document.body.appendChild(downloadLink);
	        downloadLink.click();
	        document.body.removeChild(downloadLink);
	        setTimeout(function () {
	            URL.revokeObjectURL(downloadLink.href);
	        }, 1500);
	    });
	}

	/**
	 * The MIT License (MIT)
	 *
	 * Copyright (c) 2015 Marin Atanasov
	 *
	 * Permission is hereby granted, free of charge, to any person obtaining a copy
	 * of this software and associated documentation files (the "Software"), to deal
	 * in the Software without restriction, including without limitation the rights
	 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
	 * copies of the Software, and to permit persons to whom the Software is
	 * furnished to do so, subject to the following conditions:
	 *
	 * The above copyright notice and this permission notice shall be included in all
	 * copies or substantial portions of the Software.
	 *
	 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
	 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
	 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
	 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
	 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
	 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
	 * SOFTWARE.
	 */
	const characterMap = {
	    À: 'A',
	    Á: 'A',
	    Â: 'A',
	    Ã: 'A',
	    Ä: 'A',
	    Å: 'A',
	    Ấ: 'A',
	    Ắ: 'A',
	    Ẳ: 'A',
	    Ẵ: 'A',
	    Ặ: 'A',
	    Æ: 'AE',
	    Ầ: 'A',
	    Ằ: 'A',
	    Ȃ: 'A',
	    Ả: 'A',
	    Ạ: 'A',
	    Ẩ: 'A',
	    Ẫ: 'A',
	    Ậ: 'A',
	    Ç: 'C',
	    Ḉ: 'C',
	    È: 'E',
	    É: 'E',
	    Ê: 'E',
	    Ë: 'E',
	    Ế: 'E',
	    Ḗ: 'E',
	    Ề: 'E',
	    Ḕ: 'E',
	    Ḝ: 'E',
	    Ȇ: 'E',
	    Ẻ: 'E',
	    Ẽ: 'E',
	    Ẹ: 'E',
	    Ể: 'E',
	    Ễ: 'E',
	    Ệ: 'E',
	    Ì: 'I',
	    Í: 'I',
	    Î: 'I',
	    Ï: 'I',
	    Ḯ: 'I',
	    Ȋ: 'I',
	    Ỉ: 'I',
	    Ị: 'I',
	    Ð: 'D',
	    Ñ: 'N',
	    Ò: 'O',
	    Ó: 'O',
	    Ô: 'O',
	    Õ: 'O',
	    Ö: 'O',
	    Ø: 'O',
	    Ố: 'O',
	    Ṍ: 'O',
	    Ṓ: 'O',
	    Ȏ: 'O',
	    Ỏ: 'O',
	    Ọ: 'O',
	    Ổ: 'O',
	    Ỗ: 'O',
	    Ộ: 'O',
	    Ờ: 'O',
	    Ở: 'O',
	    Ỡ: 'O',
	    Ớ: 'O',
	    Ợ: 'O',
	    Ù: 'U',
	    Ú: 'U',
	    Û: 'U',
	    Ü: 'U',
	    Ủ: 'U',
	    Ụ: 'U',
	    Ử: 'U',
	    Ữ: 'U',
	    Ự: 'U',
	    Ý: 'Y',
	    à: 'a',
	    á: 'a',
	    â: 'a',
	    ã: 'a',
	    ä: 'a',
	    å: 'a',
	    ấ: 'a',
	    ắ: 'a',
	    ẳ: 'a',
	    ẵ: 'a',
	    ặ: 'a',
	    æ: 'ae',
	    ầ: 'a',
	    ằ: 'a',
	    ȃ: 'a',
	    ả: 'a',
	    ạ: 'a',
	    ẩ: 'a',
	    ẫ: 'a',
	    ậ: 'a',
	    ç: 'c',
	    ḉ: 'c',
	    è: 'e',
	    é: 'e',
	    ê: 'e',
	    ë: 'e',
	    ế: 'e',
	    ḗ: 'e',
	    ề: 'e',
	    ḕ: 'e',
	    ḝ: 'e',
	    ȇ: 'e',
	    ẻ: 'e',
	    ẽ: 'e',
	    ẹ: 'e',
	    ể: 'e',
	    ễ: 'e',
	    ệ: 'e',
	    ì: 'i',
	    í: 'i',
	    î: 'i',
	    ï: 'i',
	    ḯ: 'i',
	    ȋ: 'i',
	    ỉ: 'i',
	    ị: 'i',
	    ð: 'd',
	    ñ: 'n',
	    ò: 'o',
	    ó: 'o',
	    ô: 'o',
	    õ: 'o',
	    ö: 'o',
	    ø: 'o',
	    ố: 'o',
	    ṍ: 'o',
	    ṓ: 'o',
	    ȏ: 'o',
	    ỏ: 'o',
	    ọ: 'o',
	    ổ: 'o',
	    ỗ: 'o',
	    ộ: 'o',
	    ờ: 'o',
	    ở: 'o',
	    ỡ: 'o',
	    ớ: 'o',
	    ợ: 'o',
	    ù: 'u',
	    ú: 'u',
	    û: 'u',
	    ü: 'u',
	    ủ: 'u',
	    ụ: 'u',
	    ử: 'u',
	    ữ: 'u',
	    ự: 'u',
	    ý: 'y',
	    ÿ: 'y',
	    Ā: 'A',
	    ā: 'a',
	    Ă: 'A',
	    ă: 'a',
	    Ą: 'A',
	    ą: 'a',
	    Ć: 'C',
	    ć: 'c',
	    Ĉ: 'C',
	    ĉ: 'c',
	    Ċ: 'C',
	    ċ: 'c',
	    Č: 'C',
	    č: 'c',
	    C̆: 'C',
	    c̆: 'c',
	    Ď: 'D',
	    ď: 'd',
	    Đ: 'D',
	    đ: 'd',
	    Ē: 'E',
	    ē: 'e',
	    Ĕ: 'E',
	    ĕ: 'e',
	    Ė: 'E',
	    ė: 'e',
	    Ę: 'E',
	    ę: 'e',
	    Ě: 'E',
	    ě: 'e',
	    Ĝ: 'G',
	    Ǵ: 'G',
	    ĝ: 'g',
	    ǵ: 'g',
	    Ğ: 'G',
	    ğ: 'g',
	    Ġ: 'G',
	    ġ: 'g',
	    Ģ: 'G',
	    ģ: 'g',
	    Ĥ: 'H',
	    ĥ: 'h',
	    Ħ: 'H',
	    ħ: 'h',
	    Ḫ: 'H',
	    ḫ: 'h',
	    Ĩ: 'I',
	    ĩ: 'i',
	    Ī: 'I',
	    ī: 'i',
	    Ĭ: 'I',
	    ĭ: 'i',
	    Į: 'I',
	    į: 'i',
	    İ: 'I',
	    ı: 'i',
	    IJ: 'IJ',
	    ij: 'ij',
	    Ĵ: 'J',
	    ĵ: 'j',
	    Ķ: 'K',
	    ķ: 'k',
	    Ḱ: 'K',
	    ḱ: 'k',
	    K̆: 'K',
	    k̆: 'k',
	    Ĺ: 'L',
	    ĺ: 'l',
	    Ļ: 'L',
	    ļ: 'l',
	    Ľ: 'L',
	    ľ: 'l',
	    Ŀ: 'L',
	    ŀ: 'l',
	    Ł: 'l',
	    ł: 'l',
	    Ḿ: 'M',
	    ḿ: 'm',
	    M̆: 'M',
	    m̆: 'm',
	    Ń: 'N',
	    ń: 'n',
	    Ņ: 'N',
	    ņ: 'n',
	    Ň: 'N',
	    ň: 'n',
	    ʼn: 'n',
	    N̆: 'N',
	    n̆: 'n',
	    Ō: 'O',
	    ō: 'o',
	    Ŏ: 'O',
	    ŏ: 'o',
	    Ő: 'O',
	    ő: 'o',
	    Œ: 'OE',
	    œ: 'oe',
	    P̆: 'P',
	    p̆: 'p',
	    Ŕ: 'R',
	    ŕ: 'r',
	    Ŗ: 'R',
	    ŗ: 'r',
	    Ř: 'R',
	    ř: 'r',
	    R̆: 'R',
	    r̆: 'r',
	    Ȓ: 'R',
	    ȓ: 'r',
	    Ś: 'S',
	    ś: 's',
	    Ŝ: 'S',
	    ŝ: 's',
	    Ş: 'S',
	    Ș: 'S',
	    ș: 's',
	    ş: 's',
	    Š: 'S',
	    š: 's',
	    Ţ: 'T',
	    ţ: 't',
	    ț: 't',
	    Ț: 'T',
	    Ť: 'T',
	    ť: 't',
	    Ŧ: 'T',
	    ŧ: 't',
	    T̆: 'T',
	    t̆: 't',
	    Ũ: 'U',
	    ũ: 'u',
	    Ū: 'U',
	    ū: 'u',
	    Ŭ: 'U',
	    ŭ: 'u',
	    Ů: 'U',
	    ů: 'u',
	    Ű: 'U',
	    ű: 'u',
	    Ų: 'U',
	    ų: 'u',
	    Ȗ: 'U',
	    ȗ: 'u',
	    V̆: 'V',
	    v̆: 'v',
	    Ŵ: 'W',
	    ŵ: 'w',
	    Ẃ: 'W',
	    ẃ: 'w',
	    X̆: 'X',
	    x̆: 'x',
	    Ŷ: 'Y',
	    ŷ: 'y',
	    Ÿ: 'Y',
	    Y̆: 'Y',
	    y̆: 'y',
	    Ź: 'Z',
	    ź: 'z',
	    Ż: 'Z',
	    ż: 'z',
	    Ž: 'Z',
	    ž: 'z',
	    ſ: 's',
	    ƒ: 'f',
	    Ơ: 'O',
	    ơ: 'o',
	    Ư: 'U',
	    ư: 'u',
	    Ǎ: 'A',
	    ǎ: 'a',
	    Ǐ: 'I',
	    ǐ: 'i',
	    Ǒ: 'O',
	    ǒ: 'o',
	    Ǔ: 'U',
	    ǔ: 'u',
	    Ǖ: 'U',
	    ǖ: 'u',
	    Ǘ: 'U',
	    ǘ: 'u',
	    Ǚ: 'U',
	    ǚ: 'u',
	    Ǜ: 'U',
	    ǜ: 'u',
	    Ứ: 'U',
	    ứ: 'u',
	    Ṹ: 'U',
	    ṹ: 'u',
	    Ǻ: 'A',
	    ǻ: 'a',
	    Ǽ: 'AE',
	    ǽ: 'ae',
	    Ǿ: 'O',
	    ǿ: 'o',
	    Þ: 'TH',
	    þ: 'th',
	    Ṕ: 'P',
	    ṕ: 'p',
	    Ṥ: 'S',
	    ṥ: 's',
	    X́: 'X',
	    x́: 'x',
	    Ѓ: 'Г',
	    ѓ: 'г',
	    Ќ: 'К',
	    ќ: 'к',
	    A̋: 'A',
	    a̋: 'a',
	    E̋: 'E',
	    e̋: 'e',
	    I̋: 'I',
	    i̋: 'i',
	    Ǹ: 'N',
	    ǹ: 'n',
	    Ồ: 'O',
	    ồ: 'o',
	    Ṑ: 'O',
	    ṑ: 'o',
	    Ừ: 'U',
	    ừ: 'u',
	    Ẁ: 'W',
	    ẁ: 'w',
	    Ỳ: 'Y',
	    ỳ: 'y',
	    Ȁ: 'A',
	    ȁ: 'a',
	    Ȅ: 'E',
	    ȅ: 'e',
	    Ȉ: 'I',
	    ȉ: 'i',
	    Ȍ: 'O',
	    ȍ: 'o',
	    Ȑ: 'R',
	    ȑ: 'r',
	    Ȕ: 'U',
	    ȕ: 'u',
	    B̌: 'B',
	    b̌: 'b',
	    Č̣: 'C',
	    č̣: 'c',
	    Ê̌: 'E',
	    ê̌: 'e',
	    F̌: 'F',
	    f̌: 'f',
	    Ǧ: 'G',
	    ǧ: 'g',
	    Ȟ: 'H',
	    ȟ: 'h',
	    J̌: 'J',
	    ǰ: 'j',
	    Ǩ: 'K',
	    ǩ: 'k',
	    M̌: 'M',
	    m̌: 'm',
	    P̌: 'P',
	    p̌: 'p',
	    Q̌: 'Q',
	    q̌: 'q',
	    Ř̩: 'R',
	    ř̩: 'r',
	    Ṧ: 'S',
	    ṧ: 's',
	    V̌: 'V',
	    v̌: 'v',
	    W̌: 'W',
	    w̌: 'w',
	    X̌: 'X',
	    x̌: 'x',
	    Y̌: 'Y',
	    y̌: 'y',
	    A̧: 'A',
	    a̧: 'a',
	    B̧: 'B',
	    b̧: 'b',
	    Ḑ: 'D',
	    ḑ: 'd',
	    Ȩ: 'E',
	    ȩ: 'e',
	    Ɛ̧: 'E',
	    ɛ̧: 'e',
	    Ḩ: 'H',
	    ḩ: 'h',
	    I̧: 'I',
	    i̧: 'i',
	    Ɨ̧: 'I',
	    ɨ̧: 'i',
	    M̧: 'M',
	    m̧: 'm',
	    O̧: 'O',
	    o̧: 'o',
	    Q̧: 'Q',
	    q̧: 'q',
	    U̧: 'U',
	    u̧: 'u',
	    X̧: 'X',
	    x̧: 'x',
	    Z̧: 'Z',
	    z̧: 'z',
	    й: 'и',
	    Й: 'И',
	    ё: 'е',
	    Ё: 'Е',
	};
	const removeAccentsRegex = new RegExp(Object.keys(characterMap).join('|'), 'g');
	function removeAccents(string) {
	    return string.replace(removeAccentsRegex, (char) => {
	        return characterMap[char];
	    });
	}
	const rankings = {
	    CASE_SENSITIVE_EQUAL: 7,
	    EQUAL: 6,
	    STARTS_WITH: 5,
	    WORD_STARTS_WITH: 4,
	    CONTAINS: 3,
	    ACRONYM: 2,
	    MATCHES: 1,
	    NO_MATCH: 0,
	};
	const defaultBaseSortFn = (a, b) => String(a.rankedValue).localeCompare(String(b.rankedValue));
	function matchSorter(items, value, options = {}) {
	    const { keys, threshold = rankings.MATCHES, baseSort = defaultBaseSortFn, sorter = (matchedItems) => matchedItems.sort((a, b) => sortRankedValues(a, b, baseSort)), } = options;
	    const matchedItems = items.reduce(reduceItemsToRanked, []);
	    return sorter(matchedItems).map(({ item }) => item);
	    function reduceItemsToRanked(matches, item, index) {
	        const rankingInfo = getHighestRanking(item, keys, value, options);
	        const { rank, keyThreshold = threshold } = rankingInfo;
	        if (rank >= keyThreshold) {
	            matches.push({ ...rankingInfo, item, index });
	        }
	        return matches;
	    }
	}
	matchSorter.rankings = rankings;
	function getHighestRanking(item, keys, value, options) {
	    if (!keys) {
	        const stringItem = item;
	        return {
	            rankedValue: stringItem,
	            rank: getMatchRanking(stringItem, value, options),
	            keyIndex: -1,
	            keyThreshold: options.threshold,
	        };
	    }
	    const valuesToRank = getAllValuesToRank(item, keys);
	    return valuesToRank.reduce(({ rank, rankedValue, keyIndex, keyThreshold }, { itemValue, attributes }, i) => {
	        let newRank = getMatchRanking(itemValue, value, options);
	        let newRankedValue = rankedValue;
	        const { minRanking, maxRanking, threshold } = attributes;
	        if (newRank < minRanking && newRank >= rankings.MATCHES) {
	            newRank = minRanking;
	        }
	        else if (newRank > maxRanking) {
	            newRank = maxRanking;
	        }
	        if (newRank > rank) {
	            rank = newRank;
	            keyIndex = i;
	            keyThreshold = threshold;
	            newRankedValue = itemValue;
	        }
	        return { rankedValue: newRankedValue, rank, keyIndex, keyThreshold };
	    }, {
	        rankedValue: item,
	        rank: rankings.NO_MATCH,
	        keyIndex: -1,
	        keyThreshold: options.threshold,
	    });
	}
	function getMatchRanking(testString, stringToRank, options) {
	    testString = prepareValueForComparison(testString, options);
	    stringToRank = prepareValueForComparison(stringToRank, options);
	    if (stringToRank.length > testString.length) {
	        return rankings.NO_MATCH;
	    }
	    if (testString === stringToRank) {
	        return rankings.CASE_SENSITIVE_EQUAL;
	    }
	    testString = testString.toLowerCase();
	    stringToRank = stringToRank.toLowerCase();
	    if (testString === stringToRank) {
	        return rankings.EQUAL;
	    }
	    if (testString.startsWith(stringToRank)) {
	        return rankings.STARTS_WITH;
	    }
	    if (testString.includes(` ${stringToRank}`)) {
	        return rankings.WORD_STARTS_WITH;
	    }
	    if (testString.includes(stringToRank)) {
	        return rankings.CONTAINS;
	    }
	    else if (stringToRank.length === 1) {
	        return rankings.NO_MATCH;
	    }
	    if (getAcronym(testString).includes(stringToRank)) {
	        return rankings.ACRONYM;
	    }
	    return getClosenessRanking(testString, stringToRank);
	}
	function getAcronym(string) {
	    let acronym = '';
	    const wordsInString = string.split(' ');
	    for (const wordInString of wordsInString) {
	        const splitByHyphenWords = wordInString.split('-');
	        for (const splitByHyphenWord of splitByHyphenWords) {
	            acronym += splitByHyphenWord.substr(0, 1);
	        }
	    }
	    return acronym;
	}
	function getClosenessRanking(testString, stringToRank) {
	    let matchingInOrderCharCount = 0;
	    let charNumber = 0;
	    function findMatchingCharacter(matchChar, string, index) {
	        for (let j = index, J = string.length; j < J; j++) {
	            const stringChar = string[j];
	            if (stringChar === matchChar) {
	                matchingInOrderCharCount += 1;
	                return j + 1;
	            }
	        }
	        return -1;
	    }
	    function getRanking(spread) {
	        const spreadPercentage = 1 / spread;
	        const inOrderPercentage = matchingInOrderCharCount / stringToRank.length;
	        const ranking = rankings.MATCHES + inOrderPercentage * spreadPercentage;
	        return ranking;
	    }
	    const firstIndex = findMatchingCharacter(stringToRank[0], testString, 0);
	    if (firstIndex < 0) {
	        return rankings.NO_MATCH;
	    }
	    charNumber = firstIndex;
	    for (let i = 1, I = stringToRank.length; i < I; i++) {
	        const matchChar = stringToRank[i];
	        charNumber = findMatchingCharacter(matchChar, testString, charNumber);
	        const found = charNumber > -1;
	        if (!found) {
	            return rankings.NO_MATCH;
	        }
	    }
	    const spread = charNumber - firstIndex;
	    return getRanking(spread);
	}
	function sortRankedValues(a, b, baseSort) {
	    const aFirst = -1;
	    const bFirst = 1;
	    const { rank: aRank, keyIndex: aKeyIndex } = a;
	    const { rank: bRank, keyIndex: bKeyIndex } = b;
	    const same = aRank === bRank;
	    if (same) {
	        if (aKeyIndex === bKeyIndex) {
	            return baseSort(a, b);
	        }
	        else {
	            return aKeyIndex < bKeyIndex ? aFirst : bFirst;
	        }
	    }
	    else {
	        return aRank > bRank ? aFirst : bFirst;
	    }
	}
	function prepareValueForComparison(value, { keepDiacritics }) {
	    value = `${value}`;
	    if (!keepDiacritics) {
	        value = removeAccents(value);
	    }
	    return value;
	}
	function getItemValues(item, key) {
	    if (typeof key === 'object') {
	        key = key.key;
	    }
	    let value;
	    if (typeof key === 'function') {
	        value = key(item);
	    }
	    else if (item == null) {
	        value = null;
	    }
	    else if (Object.hasOwn(item, key)) {
	        value = item[key];
	    }
	    else if (key.includes('.')) {
	        return getNestedValues(key, item);
	    }
	    else {
	        value = null;
	    }
	    if (value == null) {
	        return [];
	    }
	    if (Array.isArray(value)) {
	        return value;
	    }
	    return [String(value)];
	}
	function getNestedValues(path, item) {
	    const keys = path.split('.');
	    let values = [item];
	    for (let i = 0, I = keys.length; i < I; i++) {
	        const nestedKey = keys[i];
	        let nestedValues = [];
	        for (let j = 0, J = values.length; j < J; j++) {
	            const nestedItem = values[j];
	            if (nestedItem == null)
	                continue;
	            if (Object.hasOwn(nestedItem, nestedKey)) {
	                const nestedValue = nestedItem[nestedKey];
	                if (nestedValue != null) {
	                    nestedValues.push(nestedValue);
	                }
	            }
	            else if (nestedKey === '*') {
	                nestedValues = nestedValues.concat(nestedItem);
	            }
	        }
	        values = nestedValues;
	    }
	    if (Array.isArray(values[0])) {
	        const result = [];
	        return result.concat(...values);
	    }
	    return values;
	}
	function getAllValuesToRank(item, keys) {
	    const allValues = [];
	    for (let j = 0, J = keys.length; j < J; j++) {
	        const key = keys[j];
	        const attributes = getKeyAttributes(key);
	        const itemValues = getItemValues(item, key);
	        for (let i = 0, I = itemValues.length; i < I; i++) {
	            allValues.push({
	                itemValue: itemValues[i],
	                attributes,
	            });
	        }
	    }
	    return allValues;
	}
	const defaultKeyAttributes = {
	    maxRanking: Infinity,
	    minRanking: -Infinity,
	};
	function getKeyAttributes(key) {
	    if (typeof key === 'string') {
	        return defaultKeyAttributes;
	    }
	    return { ...defaultKeyAttributes, ...key };
	}

	function init$3(elements) {
	    elements.sidebarHeader.prepend(elements.searchBar);
	    window.addEventListener('keydown', () => {
	        if (document.activeElement !== elements.searchBar) {
	            elements.searchBar.focus();
	        }
	    });
	    elements.searchBar.addEventListener('input', (e) => {
	        if (e.inputType === 'insertText' &&
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.searchQuery.trim().length === 1) {
	            elements.sidebar.scrollTo(0, 0);
	        }
	    });
	    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._computedWatchers.sortedElements.getter =
	        exportFunction(() => {
	            const query = unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.searchQuery.trim();
	            if (query === '') {
	                const elements = [...unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements];
	                if (unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.showDiscoveredOnly) {
	                    return cloneInto(elements.filter((el) => el.discovered), unsafeWindow);
	                }
	                if (unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.sortBy === 'name') {
	                    return cloneInto(elements, unsafeWindow).sort((a, b) => a.text.localeCompare(b.text, undefined, { numeric: true }));
	                }
	                if (unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.sortBy === 'emoji') {
	                    return cloneInto(elements, unsafeWindow).sort((a, b) => {
	                        const emojiA = a.emoji ?? '⬜';
	                        const emojiB = b.emoji ?? '⬜';
	                        return emojiA.localeCompare(emojiB);
	                    });
	                }
	                return unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements;
	            }
	            else {
	                return cloneInto(matchSorter(cloneInto(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements, unsafeWindow), unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.searchQuery, {
	                    keys: ['text'],
	                }), unsafeWindow);
	            }
	        }, unsafeWindow);
	}

	function init$2(elements) {
	    const randomImage = document.createElement('img');
	    randomImage.src = randomIcon.trim();
	    randomImage.classList.add('random');
	    elements.sideControls.appendChild(randomImage);
	    randomImage.addEventListener('click', (e) => {
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].playInstanceSound();
	        const randomElement = unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements[Math.floor(Math.random() * unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.elements.length)];
	        const data = {
	            id: unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.instanceId++,
	            text: randomElement.text,
	            emoji: randomElement.emoji,
	            discovered: randomElement.discovered,
	            disabled: false,
	            left: 0,
	            top: 0,
	            offsetX: 0.5,
	            offsetY: 0.5,
	        };
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance = cloneInto(data, unsafeWindow);
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.instances.push(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance);
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].$nextTick(exportFunction(() => {
	            const randomPosition = Math.random() * Math.PI * 2;
	            const cos = 50 * Math.cos(randomPosition);
	            const sin = 50 * Math.sin(randomPosition);
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].setInstancePosition(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance, (window.innerWidth - unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.sidebarSize) / 2 +
	                cos, window.innerHeight / 2 - 40 + sin);
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].setInstanceZIndex(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance, data.id);
	            unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].calcInstanceSize(unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.selectedInstance);
	        }, unsafeWindow));
	    });
	}

	function init$1(elements) {
	    elements.logo.src = logo.trim();
	    const versionNumber = document.createElement('p');
	    versionNumber.appendChild(document.createTextNode(`v${GM.info.script.version}`));
	    versionNumber.classList.add('version');
	    elements.logo.after(versionNumber);
	}

	let theme = 'shadow';
	const shadowCSS = `
    .container.dark-mode {
		--border-color: #525252 !important;
        --item-bg: #18181b !important;
        --instance-bg: linear-gradient(180deg,#22252b,#18181b 80%) !important;
        --instance-bg-hover: linear-gradient(180deg,#3d4249,#18181b 80%) !important;
        --instance-border: #525252 !important;
        --instance-border-hover: #a3a3a3 !important;
        --sidebar-bg: #18181b !important;
        --background-color: #18181b !important;
        --discoveries-bg-active: #423a24 !important;
        --text-color: #fff !important;
	}

    .dark-mode {
        scrollbar-color: #525252 #262626 !important;
    }

    .dark-mode .sidebar-controls:after {
        background: linear-gradient(180deg, rgba(24,24,27,0), rgba(24,24,27,.9)) !important;
    }
`;
	const shadowStyles = document.createElement('style');
	shadowStyles.appendChild(document.createTextNode(shadowCSS.trim()));
	async function init(elements) {
	    const oldTheme = localStorage.getItem('theme');
	    if (oldTheme !== null && oldTheme === 'light')
	        theme = 'light';
	    localStorage.removeItem('theme');
	    const storedTheme = await GM.getValue('theme');
	    if (storedTheme === 'light')
	        theme = 'light';
	    if (storedTheme === 'dark')
	        theme = 'dark';
	    await GM.setValue('theme', theme);
	    if (theme === 'light' && unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.isDarkMode) {
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].toggleDarkMode();
	    }
	    else if (theme === 'shadow' && !unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.isDarkMode) {
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].toggleDarkMode();
	    }
	    else if (theme === 'dark' && !unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0]._data.isDarkMode) {
	        unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].toggleDarkMode();
	    }
	    if (theme === 'shadow') {
	        document.getElementsByTagName('head')[0].appendChild(shadowStyles);
	    }
	    const darkModeIcon = elements.darkModeIcon.cloneNode(true);
	    darkModeIcon.src = shadowIcon;
	    if (theme === 'light')
	        darkModeIcon.src = '/infinite-craft/dark-mode.svg';
	    if (theme === 'dark')
	        darkModeIcon.src = '/infinite-craft/dark-mode-on.svg';
	    elements.darkModeIcon.parentNode?.replaceChild(darkModeIcon, elements.darkModeIcon);
	    const toggleDarkMode = unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].toggleDarkMode;
	    unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].toggleDarkMode = exportFunction(async () => {
	        if (theme === 'light') {
	            toggleDarkMode();
	            darkModeIcon.src = shadowIcon;
	            document.getElementsByTagName('head')[0].appendChild(shadowStyles);
	            theme = 'shadow';
	        }
	        else if (theme === 'shadow') {
	            darkModeIcon.src = '/infinite-craft/dark-mode-on.svg';
	            shadowStyles.remove();
	            theme = 'dark';
	        }
	        else if (theme === 'dark') {
	            toggleDarkMode();
	            darkModeIcon.src = '/infinite-craft/dark-mode.svg';
	            theme = 'light';
	        }
	        await GM.setValue('theme', theme);
	    }, unsafeWindow);
	    darkModeIcon.addEventListener('click', unsafeWindow.$nuxt.$root.$children[1].$children[0].$children[0].toggleDarkMode);
	}

	window.addEventListener('load', async () => {
	    const sidebarHeader = document.createElement('div');
	    sidebarHeader.classList.add('sidebar-header');
	    const settingsContent = document.createElement('div');
	    settingsContent.classList.add('settings-content');
	    const elements = {
	        favicon: document.querySelector('link[rel="icon"]'),
	        container: document.querySelector('.container'),
	        instances: document.querySelector('.instances'),
	        styles: document.createElement('style'),
	        sideControls: document.querySelector('.side-controls'),
	        darkModeIcon: document.querySelector('.dark-mode-icon'),
	        sidebar: document.querySelector('.sidebar'),
	        sidebarHeader: sidebarHeader,
	        searchBar: document.querySelector('.sidebar-search'),
	        settingsContent: settingsContent,
	        items: document.querySelector('.items'),
	        getItems: () => {
	            return Array.from(document.querySelectorAll('.items div.item'));
	        },
	        instruction: document.querySelector('.instruction'),
	        sort: document.querySelector('.sort'),
	        particles: document.querySelector('.particles'),
	        logo: document.querySelector('.logo'),
	    };
	    elements.items.before(elements.sidebarHeader);
	    init$d(elements);
	    init$c(elements);
	    init$b(elements);
	    init$6(elements);
	    init$8(elements);
	    init$5(elements);
	    init$4(elements);
	    init$3(elements);
	    init$7(elements);
	    init$2(elements);
	    await init$a(elements);
	    init$9(elements);
	    init$1(elements);
	    init(elements);
	}, false);
	window.addEventListener('contextmenu', (e) => {
	    e.preventDefault();
	});

})();