// ==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();
});
})();