forked from OpenWF/SpaceNinjaServer
		
	Compare commits
	
		
			10 Commits
		
	
	
		
			f099b64ef4
			...
			c4c1bc68ca
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c4c1bc68ca | |||
| 6a6683fb25 | |||
| e3b6accb5d | |||
| 7e437d75bf | |||
| 62570177b6 | |||
| d2aff211c6 | |||
| 791ae389d8 | |||
| d027e7f26e | |||
| cd6ce61b80 | |||
| a5be29159f | 
@ -11,8 +11,6 @@
 | 
			
		||||
  "administratorNames": [],
 | 
			
		||||
  "autoCreateAccount": true,
 | 
			
		||||
  "skipTutorial": false,
 | 
			
		||||
  "unlockAllShipDecorations": false,
 | 
			
		||||
  "unlockAllFlavourItems": false,
 | 
			
		||||
  "unlockAllSkins": false,
 | 
			
		||||
  "fullyStockedVendors": false,
 | 
			
		||||
  "skipClanKeyCrafting": false,
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										8
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										8
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							@ -17,7 +17,7 @@
 | 
			
		||||
        "morgan": "^1.10.0",
 | 
			
		||||
        "ncp": "^2.0.0",
 | 
			
		||||
        "undici": "^7.10.0",
 | 
			
		||||
        "warframe-public-export-plus": "^0.5.87",
 | 
			
		||||
        "warframe-public-export-plus": "^0.5.88",
 | 
			
		||||
        "warframe-riven-info": "^0.1.2",
 | 
			
		||||
        "winston": "^3.17.0",
 | 
			
		||||
        "winston-daily-rotate-file": "^5.0.0",
 | 
			
		||||
@ -5532,9 +5532,9 @@
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/warframe-public-export-plus": {
 | 
			
		||||
      "version": "0.5.87",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/warframe-public-export-plus/-/warframe-public-export-plus-0.5.87.tgz",
 | 
			
		||||
      "integrity": "sha512-pWDU3Df3fcEgYn42NNZb2XFOs5AdCIqFF/t9fU7VLpokBWjFzZgLz3O4gILssiUrwB4KCkvnjHi3BivgzJuv6g=="
 | 
			
		||||
      "version": "0.5.88",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/warframe-public-export-plus/-/warframe-public-export-plus-0.5.88.tgz",
 | 
			
		||||
      "integrity": "sha512-uX766+MYDY3pMncu/23Dp9VZvrUe8pdWRWMcxfUbXg29aYO2GqipimHaFtw+vfrY06YAE8nbFkCWhFL3oPDPGw=="
 | 
			
		||||
    },
 | 
			
		||||
    "node_modules/warframe-riven-info": {
 | 
			
		||||
      "version": "0.1.2",
 | 
			
		||||
 | 
			
		||||
@ -35,7 +35,7 @@
 | 
			
		||||
    "morgan": "^1.10.0",
 | 
			
		||||
    "ncp": "^2.0.0",
 | 
			
		||||
    "undici": "^7.10.0",
 | 
			
		||||
    "warframe-public-export-plus": "^0.5.87",
 | 
			
		||||
    "warframe-public-export-plus": "^0.5.88",
 | 
			
		||||
    "warframe-riven-info": "^0.1.2",
 | 
			
		||||
    "winston": "^3.17.0",
 | 
			
		||||
    "winston-daily-rotate-file": "^5.0.0",
 | 
			
		||||
 | 
			
		||||
@ -8,8 +8,8 @@ export const giveKeyChainTriggeredMessageController: RequestHandler = async (req
 | 
			
		||||
    const accountId = await getAccountIdForRequest(req);
 | 
			
		||||
    const keyChainInfo = JSON.parse((req.body as Buffer).toString()) as IKeyChainRequest;
 | 
			
		||||
 | 
			
		||||
    const inventory = await getInventory(accountId, "QuestKeys");
 | 
			
		||||
    await giveKeyChainMessage(inventory, accountId, keyChainInfo);
 | 
			
		||||
    const inventory = await getInventory(accountId, "QuestKeys accountOwnerId");
 | 
			
		||||
    await giveKeyChainMessage(inventory, keyChainInfo);
 | 
			
		||||
    await inventory.save();
 | 
			
		||||
 | 
			
		||||
    res.send(1);
 | 
			
		||||
 | 
			
		||||
@ -10,7 +10,7 @@ import { equipmentKeys } from "../../types/inventoryTypes/inventoryTypes.ts";
 | 
			
		||||
import type { IPolarity } from "../../types/inventoryTypes/commonInventoryTypes.ts";
 | 
			
		||||
import { ArtifactPolarity } from "../../types/inventoryTypes/commonInventoryTypes.ts";
 | 
			
		||||
import type { ICountedItem } from "warframe-public-export-plus";
 | 
			
		||||
import { ExportCustoms, ExportFlavour, ExportResources } from "warframe-public-export-plus";
 | 
			
		||||
import { ExportCustoms } from "warframe-public-export-plus";
 | 
			
		||||
import { applyCheatsToInfestedFoundry, handleSubsumeCompletion } from "../../services/infestedFoundryService.ts";
 | 
			
		||||
import {
 | 
			
		||||
    addEmailItem,
 | 
			
		||||
@ -321,21 +321,7 @@ export const getInventoryResponse = async (
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (config.unlockAllShipDecorations) {
 | 
			
		||||
        inventoryResponse.ShipDecorations = [];
 | 
			
		||||
        for (const [uniqueName, item] of Object.entries(ExportResources)) {
 | 
			
		||||
            if (item.productCategory == "ShipDecorations") {
 | 
			
		||||
                inventoryResponse.ShipDecorations.push({ ItemType: uniqueName, ItemCount: 999_999 });
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (config.unlockAllFlavourItems) {
 | 
			
		||||
        inventoryResponse.FlavourItems = [];
 | 
			
		||||
        for (const uniqueName in ExportFlavour) {
 | 
			
		||||
            inventoryResponse.FlavourItems.push({ ItemType: uniqueName });
 | 
			
		||||
        }
 | 
			
		||||
    } else if (config.worldState?.baroTennoConRelay) {
 | 
			
		||||
    if (config.worldState?.baroTennoConRelay) {
 | 
			
		||||
        [
 | 
			
		||||
            "/Lotus/Types/Items/Events/TennoConRelay2022EarlyAccess",
 | 
			
		||||
            "/Lotus/Types/Items/Events/TennoConRelay2023EarlyAccess",
 | 
			
		||||
 | 
			
		||||
@ -2,7 +2,11 @@ import type { RequestHandler } from "express";
 | 
			
		||||
import { getJSONfromString } from "../../helpers/stringHelpers.ts";
 | 
			
		||||
import { getAccountForRequest } from "../../services/loginService.ts";
 | 
			
		||||
import type { IMissionInventoryUpdateRequest } from "../../types/requestTypes.ts";
 | 
			
		||||
import { addMissionInventoryUpdates, addMissionRewards } from "../../services/missionInventoryUpdateService.ts";
 | 
			
		||||
import {
 | 
			
		||||
    addMissionInventoryUpdates,
 | 
			
		||||
    addMissionRewards,
 | 
			
		||||
    handleConservation
 | 
			
		||||
} from "../../services/missionInventoryUpdateService.ts";
 | 
			
		||||
import { getInventory } from "../../services/inventoryService.ts";
 | 
			
		||||
import { getInventoryResponse } from "./inventoryController.ts";
 | 
			
		||||
import { logger } from "../../utils/logger.ts";
 | 
			
		||||
@ -94,6 +98,7 @@ export const missionInventoryUpdateController: RequestHandler = async (req, res)
 | 
			
		||||
        SyndicateXPItemReward,
 | 
			
		||||
        ConquestCompletedMissionsCount
 | 
			
		||||
    } = await addMissionRewards(account, inventory, missionReport, firstCompletion);
 | 
			
		||||
    handleConservation(inventory, missionReport, AffiliationMods); // Conservation reports have GS_SUCCESS
 | 
			
		||||
 | 
			
		||||
    if (missionReport.EndOfMatchUpload) {
 | 
			
		||||
        inventory.RewardSeed = generateRewardSeed();
 | 
			
		||||
@ -111,8 +116,16 @@ export const missionInventoryUpdateController: RequestHandler = async (req, res)
 | 
			
		||||
        AffiliationMods,
 | 
			
		||||
        ConquestCompletedMissionsCount
 | 
			
		||||
    };
 | 
			
		||||
    if (missionReport.RJ) {
 | 
			
		||||
        logger.debug(`railjack interstitial request, sending only deltas`, deltas);
 | 
			
		||||
    if (
 | 
			
		||||
        missionReport.BMI ||
 | 
			
		||||
        missionReport.TNT ||
 | 
			
		||||
        missionReport.SSC ||
 | 
			
		||||
        missionReport.RJ ||
 | 
			
		||||
        missionReport.SS ||
 | 
			
		||||
        missionReport.CMI ||
 | 
			
		||||
        missionReport.EJC
 | 
			
		||||
    ) {
 | 
			
		||||
        logger.debug(`interstitial request, sending only deltas`, deltas);
 | 
			
		||||
        res.json(deltas);
 | 
			
		||||
    } else if (missionReport.RewardInfo) {
 | 
			
		||||
        logger.debug(`classic mission completion, sending everything`);
 | 
			
		||||
 | 
			
		||||
@ -10,6 +10,7 @@ import {
 | 
			
		||||
    ExportDojoRecipes,
 | 
			
		||||
    ExportDrones,
 | 
			
		||||
    ExportFactions,
 | 
			
		||||
    ExportFlavour,
 | 
			
		||||
    ExportGear,
 | 
			
		||||
    ExportKeys,
 | 
			
		||||
    ExportRailjackWeapons,
 | 
			
		||||
@ -35,6 +36,7 @@ interface ListedItem {
 | 
			
		||||
    partType?: string;
 | 
			
		||||
    chainLength?: number;
 | 
			
		||||
    parazon?: boolean;
 | 
			
		||||
    excludeFromAddMissing?: boolean;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
interface ItemLists {
 | 
			
		||||
@ -62,6 +64,8 @@ interface ItemLists {
 | 
			
		||||
    Abilities: ListedItem[];
 | 
			
		||||
    TechProjects: ListedItem[];
 | 
			
		||||
    VaultDecoRecipes: ListedItem[];
 | 
			
		||||
    FlavourItems: ListedItem[];
 | 
			
		||||
    ShipDecorations: ListedItem[];
 | 
			
		||||
    //circuitGameModes: ListedItem[];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -102,7 +106,9 @@ const getItemListsController: RequestHandler = (req, response) => {
 | 
			
		||||
        VarziaOffers: [],
 | 
			
		||||
        Abilities: [],
 | 
			
		||||
        TechProjects: [],
 | 
			
		||||
        VaultDecoRecipes: []
 | 
			
		||||
        VaultDecoRecipes: [],
 | 
			
		||||
        FlavourItems: [],
 | 
			
		||||
        ShipDecorations: []
 | 
			
		||||
        /*circuitGameModes: [
 | 
			
		||||
            {
 | 
			
		||||
                uniqueName: "Survival",
 | 
			
		||||
@ -230,7 +236,12 @@ const getItemListsController: RequestHandler = (req, response) => {
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if (
 | 
			
		||||
        if (item.productCategory == "ShipDecorations") {
 | 
			
		||||
            res.ShipDecorations.push({
 | 
			
		||||
                uniqueName: uniqueName,
 | 
			
		||||
                name: name
 | 
			
		||||
            });
 | 
			
		||||
        } else if (
 | 
			
		||||
            name &&
 | 
			
		||||
            uniqueName.substring(0, 30) != "/Lotus/Types/Game/Projections/" &&
 | 
			
		||||
            uniqueName != "/Lotus/Types/Gameplay/EntratiLab/Resources/EntratiLanthornBundle"
 | 
			
		||||
@ -443,6 +454,15 @@ const getItemListsController: RequestHandler = (req, response) => {
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    for (const [uniqueName, item] of Object.entries(ExportFlavour)) {
 | 
			
		||||
        const flavourItem: ListedItem = {
 | 
			
		||||
            uniqueName,
 | 
			
		||||
            name: getString(item.name, lang)
 | 
			
		||||
        };
 | 
			
		||||
        if (item.alwaysAvailable) flavourItem.excludeFromAddMissing = true;
 | 
			
		||||
        res.FlavourItems.push(flavourItem);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    response.json(res);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -102,8 +102,16 @@ export const manageQuestsController: RequestHandler = async (req, res) => {
 | 
			
		||||
                    questKey.Completed = false;
 | 
			
		||||
                    questKey.CompletionDate = undefined;
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                const run = questKey.Progress[0]?.c ?? 0;
 | 
			
		||||
                const stage = questKey.Progress.map(p => p.c).lastIndexOf(run);
 | 
			
		||||
 | 
			
		||||
                if (run > 0) {
 | 
			
		||||
                    questKey.Progress[stage].c = run - 1;
 | 
			
		||||
                } else {
 | 
			
		||||
                    questKey.Progress.pop();
 | 
			
		||||
                const stage = questKey.Progress.length - 1;
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                if (stage > 0) {
 | 
			
		||||
                    await giveKeyChainStageTriggered(inventory, {
 | 
			
		||||
                        KeyChain: questKey.ItemType,
 | 
			
		||||
@ -123,28 +131,28 @@ export const manageQuestsController: RequestHandler = async (req, res) => {
 | 
			
		||||
                }
 | 
			
		||||
                if (!questKey.Progress) break;
 | 
			
		||||
 | 
			
		||||
                const currentStage = questKey.Progress.length;
 | 
			
		||||
                const run = questKey.Progress[0]?.c ?? 0;
 | 
			
		||||
                const currentStage = questKey.Progress.map(p => p.c).lastIndexOf(run);
 | 
			
		||||
 | 
			
		||||
                if (currentStage + 1 == questManifest.chainStages?.length) {
 | 
			
		||||
                    logger.debug(`Trying to complete last stage with nextStage, calling completeQuest instead`);
 | 
			
		||||
                    await completeQuest(inventory, questKey.ItemType);
 | 
			
		||||
                } else {
 | 
			
		||||
                    const progress = {
 | 
			
		||||
                        c: 0,
 | 
			
		||||
                        i: false,
 | 
			
		||||
                        m: false,
 | 
			
		||||
                        b: []
 | 
			
		||||
                    };
 | 
			
		||||
                    questKey.Progress.push(progress);
 | 
			
		||||
                    if (run > 0) {
 | 
			
		||||
                        questKey.Progress[currentStage + 1].c = run;
 | 
			
		||||
                    } else {
 | 
			
		||||
                        questKey.Progress.push({ c: run, i: false, m: false, b: [] });
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    await giveKeyChainStageTriggered(inventory, {
 | 
			
		||||
                        KeyChain: questKey.ItemType,
 | 
			
		||||
                        ChainStage: currentStage
 | 
			
		||||
                        ChainStage: currentStage + 1
 | 
			
		||||
                    });
 | 
			
		||||
 | 
			
		||||
                    if (currentStage > 0) {
 | 
			
		||||
                        await giveKeyChainMissionReward(inventory, {
 | 
			
		||||
                            KeyChain: questKey.ItemType,
 | 
			
		||||
                            ChainStage: currentStage - 1
 | 
			
		||||
                            ChainStage: currentStage
 | 
			
		||||
                        });
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										14
									
								
								src/controllers/custom/removeCustomizationController.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/controllers/custom/removeCustomizationController.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,14 @@
 | 
			
		||||
import { getAccountIdForRequest } from "../../services/loginService.ts";
 | 
			
		||||
import { getInventory } from "../../services/inventoryService.ts";
 | 
			
		||||
import type { RequestHandler } from "express";
 | 
			
		||||
import { broadcastInventoryUpdate } from "../../services/wsService.ts";
 | 
			
		||||
 | 
			
		||||
export const removeCustomizationController: RequestHandler = async (req, res) => {
 | 
			
		||||
    const accountId = await getAccountIdForRequest(req);
 | 
			
		||||
    const ItemType = req.query.itemType as string;
 | 
			
		||||
    const inventory = await getInventory(accountId, "FlavourItems");
 | 
			
		||||
    inventory.FlavourItems.pull({ ItemType });
 | 
			
		||||
    await inventory.save();
 | 
			
		||||
    res.end();
 | 
			
		||||
    broadcastInventoryUpdate(req);
 | 
			
		||||
};
 | 
			
		||||
@ -21,6 +21,7 @@ import { unlockAllSimarisResearchEntriesController } from "../controllers/custom
 | 
			
		||||
import { unlockAllScansController } from "../controllers/custom/unlockAllScansController.ts";
 | 
			
		||||
import { unlockAllShipFeaturesController } from "../controllers/custom/unlockAllShipFeaturesController.ts";
 | 
			
		||||
import { unlockAllCapturaScenesController } from "../controllers/custom/unlockAllCapturaScenesController.ts";
 | 
			
		||||
import { removeCustomizationController } from "../controllers/custom/removeCustomizationController.ts";
 | 
			
		||||
 | 
			
		||||
import { abilityOverrideController } from "../controllers/custom/abilityOverrideController.ts";
 | 
			
		||||
import { createAccountController } from "../controllers/custom/createAccountController.ts";
 | 
			
		||||
@ -70,6 +71,7 @@ customRouter.get("/unlockAllSimarisResearchEntries", unlockAllSimarisResearchEnt
 | 
			
		||||
customRouter.get("/unlockAllScans", unlockAllScansController);
 | 
			
		||||
customRouter.get("/unlockAllShipFeatures", unlockAllShipFeaturesController);
 | 
			
		||||
customRouter.get("/unlockAllCapturaScenes", unlockAllCapturaScenesController);
 | 
			
		||||
customRouter.get("/removeCustomization", removeCustomizationController);
 | 
			
		||||
 | 
			
		||||
customRouter.post("/abilityOverride", abilityOverrideController);
 | 
			
		||||
customRouter.post("/createAccount", createAccountController);
 | 
			
		||||
 | 
			
		||||
@ -21,8 +21,6 @@ export interface IConfig {
 | 
			
		||||
    administratorNames?: string[];
 | 
			
		||||
    autoCreateAccount?: boolean;
 | 
			
		||||
    skipTutorial?: boolean;
 | 
			
		||||
    unlockAllShipDecorations?: boolean;
 | 
			
		||||
    unlockAllFlavourItems?: boolean;
 | 
			
		||||
    unlockAllSkins?: boolean;
 | 
			
		||||
    fullyStockedVendors?: boolean;
 | 
			
		||||
    skipClanKeyCrafting?: boolean;
 | 
			
		||||
@ -128,6 +126,8 @@ export const configRemovedOptionsKeys = [
 | 
			
		||||
    "noDojoResearchCosts",
 | 
			
		||||
    "noDojoResearchTime",
 | 
			
		||||
    "fastClanAscension",
 | 
			
		||||
    "unlockAllFlavourItems",
 | 
			
		||||
    "unlockAllShipDecorations",
 | 
			
		||||
    "unlockAllDecoRecipes"
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -5,6 +5,7 @@ import type {
 | 
			
		||||
    TMissionType
 | 
			
		||||
} from "warframe-public-export-plus";
 | 
			
		||||
import {
 | 
			
		||||
    ExportAnimals,
 | 
			
		||||
    ExportEnemies,
 | 
			
		||||
    ExportFusionBundles,
 | 
			
		||||
    ExportRegions,
 | 
			
		||||
@ -61,7 +62,6 @@ import { createMessage } from "./inboxService.ts";
 | 
			
		||||
import kuriaMessage50 from "../../static/fixed_responses/kuriaMessages/fiftyPercent.json" with { type: "json" };
 | 
			
		||||
import kuriaMessage75 from "../../static/fixed_responses/kuriaMessages/seventyFivePercent.json" with { type: "json" };
 | 
			
		||||
import kuriaMessage100 from "../../static/fixed_responses/kuriaMessages/oneHundredPercent.json" with { type: "json" };
 | 
			
		||||
import conservationAnimals from "../../static/fixed_responses/conservationAnimals.json" with { type: "json" };
 | 
			
		||||
import {
 | 
			
		||||
    generateNemesisProfile,
 | 
			
		||||
    getInfestedLichItemRewards,
 | 
			
		||||
@ -92,11 +92,6 @@ import { handleGuildGoalProgress } from "./guildService.ts";
 | 
			
		||||
import { importLoadOutConfig } from "./importService.ts";
 | 
			
		||||
 | 
			
		||||
const getRotations = (rewardInfo: IRewardInfo, tierOverride?: number): number[] => {
 | 
			
		||||
    // Disruption missions just tell us (https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2599)
 | 
			
		||||
    if (rewardInfo.rewardTierOverrides) {
 | 
			
		||||
        return rewardInfo.rewardTierOverrides;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // For Spy missions, e.g. 3 vaults cracked = A, B, C
 | 
			
		||||
    if (rewardInfo.VaultsCracked) {
 | 
			
		||||
        const rotations: number[] = [];
 | 
			
		||||
@ -109,16 +104,26 @@ const getRotations = (rewardInfo: IRewardInfo, tierOverride?: number): number[]
 | 
			
		||||
    const region = ExportRegions[rewardInfo.node] as IRegion | undefined;
 | 
			
		||||
    const missionType: TMissionType | undefined = region?.missionType;
 | 
			
		||||
 | 
			
		||||
    if (missionType == "MT_RESCUE" && rewardInfo.rewardTier) {
 | 
			
		||||
    // Disruption uses 'rewardTierOverrides' to tell us (https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2599)
 | 
			
		||||
    // Note that this may stick in lab conquest so we need to filter by mission type (https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2768)
 | 
			
		||||
    if (missionType == "MT_ARTIFACT") {
 | 
			
		||||
        return rewardInfo.rewardTierOverrides ?? [];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (missionType == "MT_RESCUE" && rewardInfo.rewardTier !== undefined) {
 | 
			
		||||
        return [rewardInfo.rewardTier];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 'rewardQualifications' is unreliable for non-endless railjack missions (https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2586, https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2612)
 | 
			
		||||
    // 'rewardQualifications' may stick from previous missions for non-endless missions done after them
 | 
			
		||||
    // - via railjack (https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2586, https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2612)
 | 
			
		||||
    // - via lab conquest (https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2768)
 | 
			
		||||
    switch (region?.missionName) {
 | 
			
		||||
        case "/Lotus/Language/Missions/MissionName_Railjack":
 | 
			
		||||
        case "/Lotus/Language/Missions/MissionName_RailjackVolatile":
 | 
			
		||||
        case "/Lotus/Language/Missions/MissionName_RailjackExterminate":
 | 
			
		||||
        case "/Lotus/Language/Missions/MissionName_RailjackAssassinate":
 | 
			
		||||
        case "/Lotus/Language/Missions/MissionName_Assassination":
 | 
			
		||||
        case "/Lotus/Language/Missions/MissionName_Exterminate":
 | 
			
		||||
            return [0];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@ -512,39 +517,6 @@ export const addMissionInventoryUpdates = async (
 | 
			
		||||
                }
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
            case "CapturedAnimals": {
 | 
			
		||||
                for (const capturedAnimal of value) {
 | 
			
		||||
                    const meta = conservationAnimals[capturedAnimal.AnimalType as keyof typeof conservationAnimals];
 | 
			
		||||
                    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
 | 
			
		||||
                    if (meta) {
 | 
			
		||||
                        if (capturedAnimal.NumTags) {
 | 
			
		||||
                            addMiscItems(inventory, [
 | 
			
		||||
                                {
 | 
			
		||||
                                    ItemType: meta.tag,
 | 
			
		||||
                                    ItemCount: capturedAnimal.NumTags
 | 
			
		||||
                                }
 | 
			
		||||
                            ]);
 | 
			
		||||
                        }
 | 
			
		||||
                        if (capturedAnimal.NumExtraRewards) {
 | 
			
		||||
                            if ("extraReward" in meta) {
 | 
			
		||||
                                addMiscItems(inventory, [
 | 
			
		||||
                                    {
 | 
			
		||||
                                        ItemType: meta.extraReward,
 | 
			
		||||
                                        ItemCount: capturedAnimal.NumExtraRewards
 | 
			
		||||
                                    }
 | 
			
		||||
                                ]);
 | 
			
		||||
                            } else {
 | 
			
		||||
                                logger.warn(
 | 
			
		||||
                                    `client attempted to claim unknown extra rewards for conservation of ${capturedAnimal.AnimalType}`
 | 
			
		||||
                                );
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    } else {
 | 
			
		||||
                        logger.warn(`ignoring conservation of unknown AnimalType: ${capturedAnimal.AnimalType}`);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
            case "KubrowPetEggs": {
 | 
			
		||||
                for (const egg of value) {
 | 
			
		||||
                    inventory.KubrowPetEggs.push({
 | 
			
		||||
@ -949,7 +921,7 @@ interface AddMissionRewardsReturnType {
 | 
			
		||||
    MissionRewards: IMissionReward[];
 | 
			
		||||
    inventoryChanges?: IInventoryChanges;
 | 
			
		||||
    credits?: IMissionCredits;
 | 
			
		||||
    AffiliationMods?: IAffiliationMods[];
 | 
			
		||||
    AffiliationMods: IAffiliationMods[];
 | 
			
		||||
    SyndicateXPItemReward?: number;
 | 
			
		||||
    ConquestCompletedMissionsCount?: number;
 | 
			
		||||
}
 | 
			
		||||
@ -1125,10 +1097,12 @@ export const addMissionRewards = async (
 | 
			
		||||
    }: IMissionInventoryUpdateRequest,
 | 
			
		||||
    firstCompletion: boolean
 | 
			
		||||
): Promise<AddMissionRewardsReturnType> => {
 | 
			
		||||
    AffiliationMods ??= [];
 | 
			
		||||
 | 
			
		||||
    if (!rewardInfo) {
 | 
			
		||||
        //TODO: if there is a case where you can have credits collected during a mission but no rewardInfo, add credits needs to be handled earlier
 | 
			
		||||
        logger.debug(`Mission ${missions!.Tag} did not have Reward Info `);
 | 
			
		||||
        return { MissionRewards: [] };
 | 
			
		||||
        return { MissionRewards: [], AffiliationMods };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    //TODO: check double reward merging
 | 
			
		||||
@ -1438,8 +1412,6 @@ export const addMissionRewards = async (
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    AffiliationMods ??= [];
 | 
			
		||||
 | 
			
		||||
    if (rewardInfo.JobStage != undefined && rewardInfo.jobId) {
 | 
			
		||||
        // eslint-disable-next-line @typescript-eslint/no-unused-vars
 | 
			
		||||
        const [jobType, unkIndex, hubNode, syndicateMissionId] = rewardInfo.jobId.split("_");
 | 
			
		||||
@ -2240,6 +2212,54 @@ function getRandomMissionDrops(
 | 
			
		||||
    return drops;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const handleConservation = (
 | 
			
		||||
    inventory: TInventoryDatabaseDocument,
 | 
			
		||||
    missionReport: IMissionInventoryUpdateRequest,
 | 
			
		||||
    AffiliationMods: IAffiliationMods[]
 | 
			
		||||
): void => {
 | 
			
		||||
    if (missionReport.CapturedAnimals) {
 | 
			
		||||
        for (const capturedAnimal of missionReport.CapturedAnimals) {
 | 
			
		||||
            // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
 | 
			
		||||
            const meta = ExportAnimals[capturedAnimal.AnimalType]?.conservation;
 | 
			
		||||
            // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
 | 
			
		||||
            if (meta) {
 | 
			
		||||
                if (capturedAnimal.NumTags) {
 | 
			
		||||
                    addMiscItems(inventory, [
 | 
			
		||||
                        {
 | 
			
		||||
                            ItemType: meta.itemReward,
 | 
			
		||||
                            ItemCount: capturedAnimal.NumTags * capturedAnimal.Count
 | 
			
		||||
                        }
 | 
			
		||||
                    ]);
 | 
			
		||||
                }
 | 
			
		||||
                if (capturedAnimal.NumExtraRewards) {
 | 
			
		||||
                    if (meta.woundedAnimalReward) {
 | 
			
		||||
                        addMiscItems(inventory, [
 | 
			
		||||
                            {
 | 
			
		||||
                                ItemType: meta.woundedAnimalReward,
 | 
			
		||||
                                ItemCount: capturedAnimal.NumExtraRewards * capturedAnimal.Count
 | 
			
		||||
                            }
 | 
			
		||||
                        ]);
 | 
			
		||||
                    } else {
 | 
			
		||||
                        logger.warn(
 | 
			
		||||
                            `client attempted to claim unknown extra rewards for conservation of ${capturedAnimal.AnimalType}`
 | 
			
		||||
                        );
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                if (meta.standingReward) {
 | 
			
		||||
                    addStanding(
 | 
			
		||||
                        inventory,
 | 
			
		||||
                        missionReport.Missions!.Tag == "SolNode129" ? "SolarisSyndicate" : "CetusSyndicate",
 | 
			
		||||
                        [2, 1.5, 1][capturedAnimal.CaptureRating] * meta.standingReward * capturedAnimal.Count,
 | 
			
		||||
                        AffiliationMods
 | 
			
		||||
                    );
 | 
			
		||||
                }
 | 
			
		||||
            } else {
 | 
			
		||||
                logger.warn(`ignoring conservation of unknown AnimalType: ${capturedAnimal.AnimalType}`);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const corruptedMods = [
 | 
			
		||||
    "/Lotus/StoreItems/Upgrades/Mods/Melee/DualStat/CorruptedHeavyDamageChargeSpeedMod", // Corrupt Charge
 | 
			
		||||
    "/Lotus/StoreItems/Upgrades/Mods/Pistol/DualStat/CorruptedCritDamagePistol", // Hollow Point
 | 
			
		||||
 | 
			
		||||
@ -6,7 +6,6 @@ import { addItem, addItems, addKeyChainItems, setupKahlSyndicate } from "./inven
 | 
			
		||||
import { fromStoreItem, getKeyChainMessage, getLevelKeyRewards } from "./itemDataService.ts";
 | 
			
		||||
import type { IQuestKeyClient, IQuestKeyDatabase, IQuestStage } from "../types/inventoryTypes/inventoryTypes.ts";
 | 
			
		||||
import { logger } from "../utils/logger.ts";
 | 
			
		||||
import type { Types } from "mongoose";
 | 
			
		||||
import { ExportKeys } from "warframe-public-export-plus";
 | 
			
		||||
import { addFixedLevelRewards } from "./missionInventoryUpdateService.ts";
 | 
			
		||||
import type { IInventoryChanges } from "../types/purchaseTypes.ts";
 | 
			
		||||
@ -44,7 +43,12 @@ export const updateQuestKey = async (
 | 
			
		||||
        inventory.QuestKeys[questKeyIndex].CompletionDate = new Date();
 | 
			
		||||
 | 
			
		||||
        const questKey = questKeyUpdate[0].ItemType;
 | 
			
		||||
        await handleQuestCompletion(inventory, questKey, inventoryChanges);
 | 
			
		||||
        await handleQuestCompletion(
 | 
			
		||||
            inventory,
 | 
			
		||||
            questKey,
 | 
			
		||||
            inventoryChanges,
 | 
			
		||||
            (questKeyUpdate[0].Progress?.[0]?.c ?? 0) > 0
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
    return inventoryChanges;
 | 
			
		||||
};
 | 
			
		||||
@ -52,7 +56,7 @@ export const updateQuestKey = async (
 | 
			
		||||
export const updateQuestStage = (
 | 
			
		||||
    inventory: TInventoryDatabaseDocument,
 | 
			
		||||
    { KeyChain, ChainStage }: IKeyChainRequest,
 | 
			
		||||
    questStageUpdate: IQuestStage
 | 
			
		||||
    questStageUpdate: Partial<IQuestStage>
 | 
			
		||||
): void => {
 | 
			
		||||
    const quest = inventory.QuestKeys.find(quest => quest.ItemType === KeyChain);
 | 
			
		||||
 | 
			
		||||
@ -68,14 +72,22 @@ export const updateQuestStage = (
 | 
			
		||||
 | 
			
		||||
    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
 | 
			
		||||
    if (!questStage) {
 | 
			
		||||
        const questStageIndex = quest.Progress.push(questStageUpdate) - 1;
 | 
			
		||||
        const questStageIndex =
 | 
			
		||||
            quest.Progress.push({
 | 
			
		||||
                c: questStageUpdate.c ?? 0,
 | 
			
		||||
                i: questStageUpdate.i ?? false,
 | 
			
		||||
                m: questStageUpdate.m ?? false,
 | 
			
		||||
                b: questStageUpdate.b ?? []
 | 
			
		||||
            }) - 1;
 | 
			
		||||
        if (questStageIndex !== ChainStage) {
 | 
			
		||||
            throw new Error(`Quest stage index mismatch: ${questStageIndex} !== ${ChainStage}`);
 | 
			
		||||
        }
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    Object.assign(questStage, questStageUpdate);
 | 
			
		||||
    for (const [key, value] of Object.entries(questStageUpdate) as [keyof IQuestStage, number | boolean | any[]][]) {
 | 
			
		||||
        (questStage[key] as any) = value;
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const addQuestKey = (
 | 
			
		||||
@ -112,58 +124,53 @@ export const completeQuest = async (inventory: TInventoryDatabaseDocument, quest
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const chainStageTotal = chainStages.length;
 | 
			
		||||
    let existingQuestKey = inventory.QuestKeys.find(qk => qk.ItemType === questKey);
 | 
			
		||||
 | 
			
		||||
    const existingQuestKey = inventory.QuestKeys.find(qk => qk.ItemType === questKey);
 | 
			
		||||
 | 
			
		||||
    const startingStage = Math.max((existingQuestKey?.Progress?.length ?? 0) - 1, 0);
 | 
			
		||||
 | 
			
		||||
    if (existingQuestKey?.Completed) {
 | 
			
		||||
    if (!existingQuestKey) {
 | 
			
		||||
        const completedQuestKey: IQuestKeyDatabase = {
 | 
			
		||||
            ItemType: questKey,
 | 
			
		||||
            Completed: false,
 | 
			
		||||
            unlock: true,
 | 
			
		||||
            Progress: Array.from({ length: chainStageTotal }, () => ({
 | 
			
		||||
                c: 0,
 | 
			
		||||
                i: false,
 | 
			
		||||
                m: false,
 | 
			
		||||
                b: []
 | 
			
		||||
            }))
 | 
			
		||||
        };
 | 
			
		||||
        addQuestKey(inventory, completedQuestKey);
 | 
			
		||||
        existingQuestKey = inventory.QuestKeys.find(qk => qk.ItemType === questKey)!;
 | 
			
		||||
    } else if (existingQuestKey.Completed) {
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
    if (existingQuestKey) {
 | 
			
		||||
 | 
			
		||||
    existingQuestKey.Progress = existingQuestKey.Progress ?? [];
 | 
			
		||||
 | 
			
		||||
        const existingProgressLength = existingQuestKey.Progress.length;
 | 
			
		||||
    const run = existingQuestKey.Progress[0]?.c ?? 0;
 | 
			
		||||
 | 
			
		||||
    const existingProgressLength = existingQuestKey.Progress.length;
 | 
			
		||||
    if (existingProgressLength < chainStageTotal) {
 | 
			
		||||
        const missingProgress: IQuestStage[] = Array.from(
 | 
			
		||||
            { length: chainStageTotal - existingProgressLength },
 | 
			
		||||
                () =>
 | 
			
		||||
                    ({
 | 
			
		||||
                        c: 0,
 | 
			
		||||
                        i: false,
 | 
			
		||||
                        m: false,
 | 
			
		||||
                        b: []
 | 
			
		||||
                    }) as IQuestStage
 | 
			
		||||
            () => ({ c: run, i: false, m: false, b: [] }) as IQuestStage
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        existingQuestKey.Progress.push(...missingProgress);
 | 
			
		||||
            existingQuestKey.CompletionDate = new Date();
 | 
			
		||||
            existingQuestKey.Completed = true;
 | 
			
		||||
        }
 | 
			
		||||
    } else {
 | 
			
		||||
        const completedQuestKey: IQuestKeyDatabase = {
 | 
			
		||||
            ItemType: questKey,
 | 
			
		||||
            Completed: true,
 | 
			
		||||
            unlock: true,
 | 
			
		||||
            Progress: Array(chainStageTotal).fill({
 | 
			
		||||
                c: 0,
 | 
			
		||||
                i: false,
 | 
			
		||||
                m: false,
 | 
			
		||||
                b: []
 | 
			
		||||
            } satisfies IQuestStage),
 | 
			
		||||
            CompletionDate: new Date()
 | 
			
		||||
        };
 | 
			
		||||
        addQuestKey(inventory, completedQuestKey);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    for (let i = startingStage; i < chainStageTotal; i++) {
 | 
			
		||||
    for (let i = 0; i < chainStageTotal; i++) {
 | 
			
		||||
        const stage = existingQuestKey.Progress[i];
 | 
			
		||||
        if (stage.c < run) {
 | 
			
		||||
            stage.c = run;
 | 
			
		||||
            await giveKeyChainStageTriggered(inventory, { KeyChain: questKey, ChainStage: i });
 | 
			
		||||
 | 
			
		||||
            await giveKeyChainMissionReward(inventory, { KeyChain: questKey, ChainStage: i });
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    await handleQuestCompletion(inventory, questKey);
 | 
			
		||||
    if (existingQuestKey.Progress.every(p => p.c == run)) {
 | 
			
		||||
        existingQuestKey.Completed = true;
 | 
			
		||||
        existingQuestKey.CompletionDate = new Date();
 | 
			
		||||
        await handleQuestCompletion(inventory, questKey, undefined, run > 0);
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const getQuestCompletionItems = (questKey: string): ITypeCount[] | undefined => {
 | 
			
		||||
@ -214,28 +221,35 @@ const doesQuestCompletionFinishSet = (
 | 
			
		||||
const handleQuestCompletion = async (
 | 
			
		||||
    inventory: TInventoryDatabaseDocument,
 | 
			
		||||
    questKey: string,
 | 
			
		||||
    inventoryChanges: IInventoryChanges = {}
 | 
			
		||||
    inventoryChanges: IInventoryChanges = {},
 | 
			
		||||
    isRerun: boolean = false
 | 
			
		||||
): Promise<void> => {
 | 
			
		||||
    logger.debug(`completed quest ${questKey}`);
 | 
			
		||||
 | 
			
		||||
    if (inventory.ActiveQuest == questKey) inventory.ActiveQuest = "";
 | 
			
		||||
    if (questKey == "/Lotus/Types/Keys/OrokinMoonQuest/OrokinMoonQuestKeyChain") {
 | 
			
		||||
        const att = isRerun
 | 
			
		||||
            ? []
 | 
			
		||||
            : [
 | 
			
		||||
                  "/Lotus/Weapons/Tenno/Melee/Swords/StalkerTwo/StalkerTwoSmallSword",
 | 
			
		||||
                  "/Lotus/Upgrades/Skins/Sigils/ScarSigil"
 | 
			
		||||
              ];
 | 
			
		||||
        await createMessage(inventory.accountOwnerId, [
 | 
			
		||||
            {
 | 
			
		||||
                sndr: "/Lotus/Language/Bosses/Ordis",
 | 
			
		||||
                msg: "/Lotus/Language/G1Quests/SecondDreamFinishInboxMessage",
 | 
			
		||||
                att: [
 | 
			
		||||
                    "/Lotus/Weapons/Tenno/Melee/Swords/StalkerTwo/StalkerTwoSmallSword",
 | 
			
		||||
                    "/Lotus/Upgrades/Skins/Sigils/ScarSigil"
 | 
			
		||||
                ],
 | 
			
		||||
                att,
 | 
			
		||||
                sub: "/Lotus/Language/G1Quests/SecondDreamFinishInboxTitle",
 | 
			
		||||
                icon: "/Lotus/Interface/Icons/Npcs/Ordis.png",
 | 
			
		||||
                highPriority: true
 | 
			
		||||
            }
 | 
			
		||||
        ]);
 | 
			
		||||
    } else if (questKey == "/Lotus/Types/Keys/NewWarQuest/NewWarQuestKeyChain") {
 | 
			
		||||
    } else if (questKey == "/Lotus/Types/Keys/NewWarQuest/NewWarQuestKeyChain" && !isRerun) {
 | 
			
		||||
        setupKahlSyndicate(inventory);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (isRerun) return;
 | 
			
		||||
 | 
			
		||||
    // Whispers in the Walls is unlocked once The New War + Heart of Deimos are completed.
 | 
			
		||||
    if (
 | 
			
		||||
        doesQuestCompletionFinishSet(inventory, questKey, [
 | 
			
		||||
@ -279,21 +293,24 @@ const handleQuestCompletion = async (
 | 
			
		||||
    if (questCompletionItems) {
 | 
			
		||||
        await addItems(inventory, questCompletionItems, inventoryChanges);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (inventory.ActiveQuest == questKey) inventory.ActiveQuest = "";
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const giveKeyChainItem = async (
 | 
			
		||||
    inventory: TInventoryDatabaseDocument,
 | 
			
		||||
    keyChainInfo: IKeyChainRequest
 | 
			
		||||
    keyChainInfo: IKeyChainRequest,
 | 
			
		||||
    isRerun: boolean = false
 | 
			
		||||
): Promise<IInventoryChanges> => {
 | 
			
		||||
    const inventoryChanges = await addKeyChainItems(inventory, keyChainInfo);
 | 
			
		||||
    let inventoryChanges: IInventoryChanges = {};
 | 
			
		||||
 | 
			
		||||
    if (!isRerun) {
 | 
			
		||||
        inventoryChanges = await addKeyChainItems(inventory, keyChainInfo);
 | 
			
		||||
 | 
			
		||||
        if (isEmptyObject(inventoryChanges)) {
 | 
			
		||||
            logger.warn("inventory changes was empty after getting keychain items: should not happen");
 | 
			
		||||
        }
 | 
			
		||||
        // items were added: update quest stage's i (item was given)
 | 
			
		||||
        updateQuestStage(inventory, keyChainInfo, { i: true });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return inventoryChanges;
 | 
			
		||||
 | 
			
		||||
@ -309,12 +326,17 @@ export const giveKeyChainItem = async (
 | 
			
		||||
 | 
			
		||||
export const giveKeyChainMessage = async (
 | 
			
		||||
    inventory: TInventoryDatabaseDocument,
 | 
			
		||||
    accountId: string | Types.ObjectId,
 | 
			
		||||
    keyChainInfo: IKeyChainRequest
 | 
			
		||||
    keyChainInfo: IKeyChainRequest,
 | 
			
		||||
    isRerun: boolean = false
 | 
			
		||||
): Promise<void> => {
 | 
			
		||||
    const keyChainMessage = getKeyChainMessage(keyChainInfo);
 | 
			
		||||
 | 
			
		||||
    await createMessage(accountId, [keyChainMessage]);
 | 
			
		||||
    if (!isRerun) {
 | 
			
		||||
        keyChainMessage.att = [];
 | 
			
		||||
        keyChainMessage.countedAtt = [];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    await createMessage(inventory.accountOwnerId, [keyChainMessage]);
 | 
			
		||||
 | 
			
		||||
    updateQuestStage(inventory, keyChainInfo, { m: true });
 | 
			
		||||
};
 | 
			
		||||
@ -328,8 +350,10 @@ export const giveKeyChainMissionReward = async (
 | 
			
		||||
 | 
			
		||||
    if (chainStages) {
 | 
			
		||||
        const missionName = chainStages[keyChainInfo.ChainStage].key;
 | 
			
		||||
        if (missionName) {
 | 
			
		||||
        const questKey = inventory.QuestKeys.find(q => q.ItemType === keyChainInfo.KeyChain);
 | 
			
		||||
        if (missionName && questKey) {
 | 
			
		||||
            const fixedLevelRewards = getLevelKeyRewards(missionName);
 | 
			
		||||
            const run = questKey.Progress?.[0]?.c ?? 0;
 | 
			
		||||
            if (fixedLevelRewards.levelKeyRewards) {
 | 
			
		||||
                const missionRewards: { StoreItem: string; ItemCount: number }[] = [];
 | 
			
		||||
                inventory.RegularCredits += addFixedLevelRewards(fixedLevelRewards.levelKeyRewards, missionRewards);
 | 
			
		||||
@ -338,7 +362,7 @@ export const giveKeyChainMissionReward = async (
 | 
			
		||||
                    await addItem(inventory, fromStoreItem(reward.StoreItem), reward.ItemCount);
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                updateQuestStage(inventory, keyChainInfo, { c: 0 });
 | 
			
		||||
                updateQuestStage(inventory, keyChainInfo, { c: run });
 | 
			
		||||
            } else if (fixedLevelRewards.levelKeyRewards2) {
 | 
			
		||||
                for (const reward of fixedLevelRewards.levelKeyRewards2) {
 | 
			
		||||
                    if (reward.rewardType == "RT_CREDITS") {
 | 
			
		||||
@ -352,7 +376,7 @@ export const giveKeyChainMissionReward = async (
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                updateQuestStage(inventory, keyChainInfo, { c: 0 });
 | 
			
		||||
                updateQuestStage(inventory, keyChainInfo, { c: run });
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
@ -364,14 +388,17 @@ export const giveKeyChainStageTriggered = async (
 | 
			
		||||
): Promise<void> => {
 | 
			
		||||
    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
 | 
			
		||||
    const chainStages = ExportKeys[keyChainInfo.KeyChain]?.chainStages;
 | 
			
		||||
    const questKey = inventory.QuestKeys.find(qk => qk.ItemType === keyChainInfo.KeyChain);
 | 
			
		||||
 | 
			
		||||
    if (chainStages && questKey) {
 | 
			
		||||
        const run = questKey.Progress?.[0]?.c ?? 0;
 | 
			
		||||
 | 
			
		||||
    if (chainStages) {
 | 
			
		||||
        if (chainStages[keyChainInfo.ChainStage].itemsToGiveWhenTriggered.length > 0) {
 | 
			
		||||
            await giveKeyChainItem(inventory, keyChainInfo);
 | 
			
		||||
            await giveKeyChainItem(inventory, keyChainInfo, run > 0);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (chainStages[keyChainInfo.ChainStage].messageToSendWhenTriggered) {
 | 
			
		||||
            await giveKeyChainMessage(inventory, inventory.accountOwnerId, keyChainInfo);
 | 
			
		||||
            await giveKeyChainMessage(inventory, keyChainInfo, run > 0);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
@ -14,7 +14,6 @@ import type {
 | 
			
		||||
import { logger } from "../utils/logger.ts";
 | 
			
		||||
import { Types } from "mongoose";
 | 
			
		||||
import { addFusionTreasures, addShipDecorations, getInventory } from "./inventoryService.ts";
 | 
			
		||||
import { config } from "./configService.ts";
 | 
			
		||||
import { Guild } from "../models/guildModel.ts";
 | 
			
		||||
import { hasGuildPermission } from "./guildService.ts";
 | 
			
		||||
import { GuildPermission } from "../types/guildTypes.ts";
 | 
			
		||||
@ -137,7 +136,6 @@ export const handleSetShipDecorations = async (
 | 
			
		||||
        roomToPlaceIn.MaxCapacity += meta.capacityCost;
 | 
			
		||||
        await personalRooms.save();
 | 
			
		||||
 | 
			
		||||
        if (!config.unlockAllShipDecorations) {
 | 
			
		||||
        const inventory = await getInventory(accountId);
 | 
			
		||||
        if (deco.Sockets !== undefined) {
 | 
			
		||||
            addFusionTreasures(inventory, [{ ItemType: itemType, Sockets: deco.Sockets, ItemCount: 1 }]);
 | 
			
		||||
@ -145,7 +143,6 @@ export const handleSetShipDecorations = async (
 | 
			
		||||
            addShipDecorations(inventory, [{ ItemType: itemType, ItemCount: 1 }]);
 | 
			
		||||
        }
 | 
			
		||||
        await inventory.save();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return {
 | 
			
		||||
            DecoId: placedDecoration.RemoveId,
 | 
			
		||||
@ -155,7 +152,6 @@ export const handleSetShipDecorations = async (
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (!config.unlockAllShipDecorations) {
 | 
			
		||||
    const inventory = await getInventory(accountId);
 | 
			
		||||
    if (placedDecoration.Sockets !== undefined) {
 | 
			
		||||
        addFusionTreasures(inventory, [{ ItemType: itemType, Sockets: placedDecoration.Sockets, ItemCount: -1 }]);
 | 
			
		||||
@ -163,7 +159,6 @@ export const handleSetShipDecorations = async (
 | 
			
		||||
        addShipDecorations(inventory, [{ ItemType: itemType, ItemCount: -1 }]);
 | 
			
		||||
    }
 | 
			
		||||
    await inventory.save();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    //place decoration
 | 
			
		||||
    const decoId = new Types.ObjectId();
 | 
			
		||||
@ -221,13 +216,11 @@ export const handleResetShipDecorations = async (
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // refund item
 | 
			
		||||
        if (!config.unlockAllShipDecorations) {
 | 
			
		||||
        if (deco.Sockets !== undefined) {
 | 
			
		||||
            addFusionTreasures(inventory, [{ ItemType: itemType, Sockets: deco.Sockets, ItemCount: 1 }]);
 | 
			
		||||
        } else {
 | 
			
		||||
            addShipDecorations(inventory, [{ ItemType: itemType, ItemCount: 1 }]);
 | 
			
		||||
        }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // refund capacity
 | 
			
		||||
        room.MaxCapacity += meta.capacityCost;
 | 
			
		||||
 | 
			
		||||
@ -2803,7 +2803,7 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
 | 
			
		||||
 | 
			
		||||
        const activeStartDay = day - ghoulsCycleDay + 17;
 | 
			
		||||
        const activeEndDay = activeStartDay + 5;
 | 
			
		||||
        const dayWithFraction = (timeMs - EPOCH) / 86400000;
 | 
			
		||||
        const dayWithFraction = (timeMs - EPOCH) / unixTimesInMs.day;
 | 
			
		||||
 | 
			
		||||
        const progress = (dayWithFraction - activeStartDay) / (activeEndDay - activeStartDay);
 | 
			
		||||
        const healthPct = 1 - Math.min(Math.max(progress, 0), 1);
 | 
			
		||||
@ -2814,22 +2814,14 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
 | 
			
		||||
                $date: {
 | 
			
		||||
                    $numberLong: config.worldState?.ghoulEmergenceOverride
 | 
			
		||||
                        ? "1753204900185"
 | 
			
		||||
                        : Date.UTC(
 | 
			
		||||
                              date.getUTCFullYear(),
 | 
			
		||||
                              date.getUTCMonth(),
 | 
			
		||||
                              date.getUTCDate() + activeStartDay
 | 
			
		||||
                          ).toString()
 | 
			
		||||
                        : (EPOCH + activeStartDay * unixTimesInMs.day).toString()
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            Expiry: {
 | 
			
		||||
                $date: {
 | 
			
		||||
                    $numberLong: config.worldState?.ghoulEmergenceOverride
 | 
			
		||||
                        ? "2000000000000"
 | 
			
		||||
                        : Date.UTC(
 | 
			
		||||
                              date.getUTCFullYear(),
 | 
			
		||||
                              date.getUTCMonth(),
 | 
			
		||||
                              date.getUTCDate() + activeEndDay
 | 
			
		||||
                          ).toString()
 | 
			
		||||
                        : (EPOCH + activeEndDay * unixTimesInMs.day).toString()
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            HealthPct: config.worldState?.ghoulEmergenceOverride ? 1 : healthPct,
 | 
			
		||||
 | 
			
		||||
@ -977,10 +977,10 @@ export interface IQuestKeyClient extends Omit<IQuestKeyDatabase, "CompletionDate
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface IQuestStage {
 | 
			
		||||
    c?: number;
 | 
			
		||||
    i?: boolean;
 | 
			
		||||
    m?: boolean;
 | 
			
		||||
    b?: any[];
 | 
			
		||||
    c: number;
 | 
			
		||||
    i: boolean;
 | 
			
		||||
    m: boolean;
 | 
			
		||||
    b: any[];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface IRawUpgrade {
 | 
			
		||||
 | 
			
		||||
@ -45,6 +45,15 @@ export type IMissionInventoryUpdateRequest = {
 | 
			
		||||
    EmailItems?: ITypeCount[];
 | 
			
		||||
    ShipDecorations?: ITypeCount[];
 | 
			
		||||
 | 
			
		||||
    // flags for interstitial requests
 | 
			
		||||
    BMI?: boolean;
 | 
			
		||||
    TNT?: boolean; // Conservation; definitely need to include AffiliationMods in this case, so a normal 'inventory sync' would not work here.
 | 
			
		||||
    SSC?: boolean; // K-Drive race?
 | 
			
		||||
    RJ?: boolean; // Railjack. InventoryJson should only be returned when going back to dojo.
 | 
			
		||||
    SS?: boolean;
 | 
			
		||||
    CMI?: boolean;
 | 
			
		||||
    EJC?: boolean;
 | 
			
		||||
 | 
			
		||||
    SyndicateId?: string;
 | 
			
		||||
    SortieId?: string;
 | 
			
		||||
    CalendarProgress?: { challenge: string }[];
 | 
			
		||||
@ -149,7 +158,6 @@ export type IMissionInventoryUpdateRequest = {
 | 
			
		||||
        MultiProgress: unknown[];
 | 
			
		||||
    }[];
 | 
			
		||||
    InvasionProgress?: IInvasionProgressClient[];
 | 
			
		||||
    RJ?: boolean;
 | 
			
		||||
    ConquestMissionsCompleted?: number;
 | 
			
		||||
    duviriSuitSelection?: string;
 | 
			
		||||
    duviriPistolSelection?: string;
 | 
			
		||||
 | 
			
		||||
@ -1,491 +0,0 @@
 | 
			
		||||
{
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/CommonBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/CommonFemaleBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/CommonMaleBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/RareBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/RareFemaleBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/RareMaleBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/UncommonBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/UncommonFemaleBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/BirdOfPrey/UncommonMaleBirdOfPreyAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagCondrocUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedCritter/BaseInfestedCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Conservation/WoundedAnimalRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedCritter/CommonInfestedCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedCritterCommon",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Deimos/WoundedInfestedCritterCommonRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedCritter/RareInfestedCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedCritterRare",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Deimos/WoundedInfestedCritterRareRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedCritter/UncommonInfestedCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedCritterUncommon",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Deimos/WoundedInfestedCritterUncommonRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedKDrive/GrottoInfKDriveAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedKdriveUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedKDrive/HighlandInfKDriveAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedKdriveRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedKDrive/SwampInfKDriveAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedKdriveCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedMaggot/CommonInfestedMaggotAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedMaggotCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedMaggot/RareInfestedMaggotAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedMaggotRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedMaggot/UncommonInfestedMaggotAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedMaggotUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedMergoo/CommonInfestedMergooAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedMergooCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedMergoo/RareInfestedMergooAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedMergooRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedMergoo/UncommonInfestedMergooAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedMergooUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedNexifera/BaseInfestedNexiferaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedNexiferaCommon",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Conservation/WoundedAnimalRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedNexifera/CommonInfestedNexiferaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedNexiferaCommon",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Conservation/WoundedAnimalRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedNexifera/RareInfestedNexiferaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedNexiferaRare",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Conservation/WoundedAnimalRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedNexifera/UncommonInfestedNexiferaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedNexiferaUncommon",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Conservation/WoundedAnimalRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedPredator/BaseInfestedPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Conservation/WoundedAnimalRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedPredator/CommonInfestedPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedPredatorCommon",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Deimos/WoundedInfestedPredatorCommonRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedPredator/RareInfestedPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedPredatorRare",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Deimos/WoundedInfestedPredatorRareRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedPredator/UncommonInfestedPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedPredatorUncommon",
 | 
			
		||||
    "extraReward": "/Lotus/Types/Items/Deimos/WoundedInfestedPredatorUncommonRewardItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedUndazoa/BaseUndazoaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedZongroCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedUndazoa/CommonUndazoaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedZongroCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedUndazoa/RareUndazoaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedZongroRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Deimos/InfestedUndazoa/UncommonUndazoaAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Deimos/AnimalTagInfestedZongroUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Duviri/Rabbit/BaseDuviriRabbitAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Duviri/Rabbit/TeshinRabbitAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Duviri/Rabbit/TeshinRabbitOnHandAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Duviri/Wolf/DuviriWolfAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/Duviri/Wolf/DuviriWolfConservationAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/CommonFemaleForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/CommonForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/CommonMaleForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/RareFemaleForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/RareForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/RareMaleForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/TutorialForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/UncommonFemaleForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/UncommonForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/ForestRodent/UncommonMaleForestRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagKuakaUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/BaseLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/CommonFemaleLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/CommonLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/CommonMaleLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/CommonPupLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/RareFemaleLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/RareLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/RareMaleLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/RarePupLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/UncommonFemaleLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/UncommonLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/UncommonMaleLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/LegendaryKubrow/UncommonPupLegendaryKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagKubrodonUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/BaseOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/CommonFemaleOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/CommonMaleOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/CommonOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/CommonPupOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/RareFemaleOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/RareMaleOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/RareOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/RarePupOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/UncommonFemaleOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/UncommonMaleOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/UncommonOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OrokinKubrow/UncommonPupOrokinKubrowAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagStoverUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/CommonFemaleOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/CommonMaleOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/CommonOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/RareFemaleOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/RareMaleOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/RareOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/UncommonFemaleOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/UncommonMaleOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/OstronSeaBird/UncommonOstronSeaBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagMergooUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/BaseSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/CommonFemaleSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/CommonMaleSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/CommonPupSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/CommonSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/RareFemaleSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/RareMaleSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/RarePupSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/RareSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/UncommonFemaleSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/UncommonMaleSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/UncommonPupSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowArmadillo/UncommonSnowArmadilloAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagBolarolaUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/BaseSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/CommonFemaleSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/CommonMaleSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/CommonPupSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/CommonSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/RareFemaleSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/RareMaleSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/RarePupSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/RareSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/UncommonFemaleSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/UncommonMaleSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/UncommonPupSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowBird/UncommonSnowBirdAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagSawgawUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/BaseSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/CommonFemaleSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/CommonMaleSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/CommonPupSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/CommonSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/RareFemaleSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/RareMaleSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/RarePupSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/RareSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/UncommonFemaleSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/UncommonMaleSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/UncommonPupSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowCritter/UncommonSnowCritterAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagVirminkUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/BaseSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/CommonFemaleSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/CommonMaleSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/CommonPupSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/CommonSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/RareFemaleSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/RareMaleSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/RarePupSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/RareSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/UncommonFemaleSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/UncommonMaleSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/UncommonPupSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowPredator/UncommonSnowPredatorAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagHorrasqueUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/BaseSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/CommonFemaleSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/CommonMaleSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/CommonSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/RareFemaleSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/RareMaleSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/RareSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/UncommonFemaleSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/UncommonMaleSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/SnowRodent/UncommonSnowRodentAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Solaris/AnimalTagPobbersUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/BaseVampireKavatAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Conservation/ConservationTagItem"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/CommonVampireKavatAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/CommonVampireKavatCubAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/CommonVampireKavatFemaleAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/CommonVampireKavatMaleAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatCommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/RareVampireKavatAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/RareVampireKavatCubAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/RareVampireKavatFemaleAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/RareVampireKavatMaleAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatRare"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/UncommonVampireKavatAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/UncommonVampireKavatCubAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/UncommonVampireKavatFemaleAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatUncommon"
 | 
			
		||||
  },
 | 
			
		||||
  "/Lotus/Types/NeutralCreatures/Conservation/VampireKavat/UncommonVampireKavatMaleAvatar": {
 | 
			
		||||
    "tag": "/Lotus/Types/Items/Eidolon/AnimalTagVampireKavatUncommon"
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@ -108,9 +108,9 @@
 | 
			
		||||
                    <div class="card-body">
 | 
			
		||||
                        <div class="tab-content">
 | 
			
		||||
                            <div class="tab-pane" id="miscItems-tab-content">
 | 
			
		||||
                                <form class="card-body input-group" onsubmit="doAcquireMiscItems();return false;">
 | 
			
		||||
                                <form class="card-body input-group" onsubmit="doAcquireCountItems('miscitem');return false;">
 | 
			
		||||
                                    <input class="form-control" id="miscitem-count" type="number" value="1" />
 | 
			
		||||
                                    <input class="form-control w-50" id="miscitem-type" list="datalist-miscitems" />
 | 
			
		||||
                                    <input class="form-control w-50" id="acquire-type-miscitem" list="datalist-miscitems" />
 | 
			
		||||
                                    <button class="btn btn-primary" type="submit" data-loc="general_addButton"></button>
 | 
			
		||||
                                </form>
 | 
			
		||||
                            </div>
 | 
			
		||||
@ -459,6 +459,37 @@
 | 
			
		||||
                        </div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="row g-3 mb-3">
 | 
			
		||||
                    <div class="col-lg-6">
 | 
			
		||||
                        <div class="card" style="height: 400px;">
 | 
			
		||||
                            <h5 class="card-header" data-loc="inventory_flavourItems"></h5>
 | 
			
		||||
                            <div class="card-body overflow-auto">
 | 
			
		||||
                                <form class="input-group mb-3" onsubmit="doAcquireEquipment('FlavourItems');return false;">
 | 
			
		||||
                                    <input class="form-control" id="acquire-type-FlavourItems" list="datalist-FlavourItems" />
 | 
			
		||||
                                    <button class="btn btn-primary" type="submit" data-loc="general_addButton"></button>
 | 
			
		||||
                                </form>
 | 
			
		||||
                                <table class="table table-hover w-100">
 | 
			
		||||
                                    <tbody id="FlavourItems-list"></tbody>
 | 
			
		||||
                                </table>
 | 
			
		||||
                            </div>
 | 
			
		||||
                        </div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <div class="col-lg-6">
 | 
			
		||||
                        <div class="card" style="height: 400px;">
 | 
			
		||||
                            <h5 class="card-header" data-loc="inventory_shipDecorations"></h5>
 | 
			
		||||
                            <div class="card-body overflow-auto">
 | 
			
		||||
                                <form class="card-body input-group" onsubmit="doAcquireCountItems('ShipDecorations');return false;">
 | 
			
		||||
                                    <input class="form-control" id="ShipDecorations-count" type="number" value="1" />
 | 
			
		||||
                                    <input class="form-control w-50" id="acquire-type-ShipDecorations" list="datalist-ShipDecorations" />
 | 
			
		||||
                                    <button class="btn btn-primary" type="submit" data-loc="general_addButton"></button>
 | 
			
		||||
                                </form>
 | 
			
		||||
                                <table class="table table-hover w-100">
 | 
			
		||||
                                    <tbody id="ShipDecorations-list"></tbody>
 | 
			
		||||
                                </table>
 | 
			
		||||
                            </div>
 | 
			
		||||
                        </div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="card">
 | 
			
		||||
                    <h5 class="card-header" data-loc="general_bulkActions"></h5>
 | 
			
		||||
                    <div class="card-body">
 | 
			
		||||
@ -469,6 +500,8 @@
 | 
			
		||||
                            <button class="btn btn-primary" onclick="debounce(addMissingEquipment, ['SpaceGuns', 'SpaceMelee']);" data-loc="inventory_bulkAddSpaceWeapons"></button>
 | 
			
		||||
                            <button class="btn btn-primary" onclick="debounce(addMissingEquipment, ['Sentinels']);" data-loc="inventory_bulkAddSentinels"></button>
 | 
			
		||||
                            <button class="btn btn-primary" onclick="debounce(addMissingEquipment, ['SentinelWeapons']);" data-loc="inventory_bulkAddSentinelWeapons"></button>
 | 
			
		||||
                            <button class="btn btn-primary" onclick="debounce(addMissingEquipment, ['FlavourItems']);" data-loc="inventory_bulkAddFlavourItems"></button>
 | 
			
		||||
                            <button class="btn btn-primary" onclick="debounce(addMissingEquipment, ['ShipDecorations']);" data-loc="inventory_bulkAddShipDecorations"></button>
 | 
			
		||||
                            <button class="btn btn-primary" onclick="debounce(addMissingEvolutionProgress);" data-loc="inventory_bulkAddEvolutionProgress"></button>
 | 
			
		||||
                        </div>
 | 
			
		||||
                        <div class="mb-2 d-flex flex-wrap gap-2">
 | 
			
		||||
@ -979,14 +1012,6 @@
 | 
			
		||||
                                        <input class="form-check-input" type="checkbox" id="skipTutorial" />
 | 
			
		||||
                                        <label class="form-check-label" for="skipTutorial" data-loc="cheats_skipTutorial"></label>
 | 
			
		||||
                                    </div>
 | 
			
		||||
                                    <div class="form-check">
 | 
			
		||||
                                        <input class="form-check-input" type="checkbox" id="unlockAllShipDecorations" />
 | 
			
		||||
                                        <label class="form-check-label" for="unlockAllShipDecorations" data-loc="cheats_unlockAllShipDecorations"></label>
 | 
			
		||||
                                    </div>
 | 
			
		||||
                                    <div class="form-check">
 | 
			
		||||
                                        <input class="form-check-input" type="checkbox" id="unlockAllFlavourItems" />
 | 
			
		||||
                                        <label class="form-check-label" for="unlockAllFlavourItems" data-loc="cheats_unlockAllFlavourItems"></label>
 | 
			
		||||
                                    </div>
 | 
			
		||||
                                    <div class="form-check">
 | 
			
		||||
                                        <input class="form-check-input" type="checkbox" id="unlockAllSkins" />
 | 
			
		||||
                                        <label class="form-check-label" for="unlockAllSkins" data-loc="cheats_unlockAllSkins"></label>
 | 
			
		||||
@ -1387,6 +1412,8 @@
 | 
			
		||||
    <datalist id="datalist-Abilities"></datalist>
 | 
			
		||||
    <datalist id="datalist-TechProjects"></datalist>
 | 
			
		||||
    <datalist id="datalist-VaultDecoRecipes"></datalist>
 | 
			
		||||
    <datalist id="datalist-FlavourItems"></datalist>
 | 
			
		||||
    <datalist id="datalist-ShipDecorations"></datalist>
 | 
			
		||||
    <datalist id="datalist-circuitGameModes">
 | 
			
		||||
        <option>Survival</option>
 | 
			
		||||
        <option>VoidFlood</option>
 | 
			
		||||
 | 
			
		||||
@ -505,6 +505,9 @@ function fetchItemList() {
 | 
			
		||||
                        i => i.uniqueName === "/Lotus/Types/Friendly/Pets/ZanukaPets/ZanukaPetParts/ZanukaPetPartHeadC"
 | 
			
		||||
                    ).name
 | 
			
		||||
                },
 | 
			
		||||
                "/Lotus/Powersuits/Stalker/Stalker": {
 | 
			
		||||
                    name: loc("code_stalker")
 | 
			
		||||
                },
 | 
			
		||||
                "/Lotus/Language/Game/Rank_Creator": {
 | 
			
		||||
                    name: loc("guildView_rank_creator")
 | 
			
		||||
                },
 | 
			
		||||
@ -570,6 +573,34 @@ function fetchItemList() {
 | 
			
		||||
                        option.textContent = item.name;
 | 
			
		||||
                        document.getElementById("changeSyndicate").appendChild(option);
 | 
			
		||||
                    });
 | 
			
		||||
                } else if (type == "FlavourItems") {
 | 
			
		||||
                    const cursorPrefixes = {
 | 
			
		||||
                        Controller: loc("code_controller"),
 | 
			
		||||
                        MouseGrey: loc("code_mouse"),
 | 
			
		||||
                        MouseLine: loc("code_mouseLine"),
 | 
			
		||||
                        Mouse: loc("code_mouse")
 | 
			
		||||
                    };
 | 
			
		||||
                    items.forEach(item => {
 | 
			
		||||
                        if (item.uniqueName.startsWith("/Lotus/Interface/Graphics/CustomUI/Cursors/")) {
 | 
			
		||||
                            let base = item.uniqueName.replace("/Lotus/Interface/Graphics/CustomUI/Cursors/", "");
 | 
			
		||||
                            for (const key in cursorPrefixes) {
 | 
			
		||||
                                if (base.startsWith(key)) {
 | 
			
		||||
                                    const prefix = cursorPrefixes[key];
 | 
			
		||||
                                    const suffix = base.slice(key.length);
 | 
			
		||||
                                    item.name = prefix + " " + suffix;
 | 
			
		||||
                                    break;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                        } else if (item.uniqueName.includes("ColourPicker")) {
 | 
			
		||||
                            item.name = loc("code_itemColorPalette").split("|ITEM|").join(item.name);
 | 
			
		||||
                        }
 | 
			
		||||
                        const option = document.createElement("option");
 | 
			
		||||
                        option.setAttribute("data-key", item.uniqueName);
 | 
			
		||||
                        option.value = item.name;
 | 
			
		||||
                        document.getElementById("datalist-" + type).appendChild(option);
 | 
			
		||||
                        if (item.excludeFromAddMissing) option.setAttribute("data-exclude-from-add-missing", "");
 | 
			
		||||
                        itemMap[item.uniqueName] = { ...item, type };
 | 
			
		||||
                    });
 | 
			
		||||
                } else {
 | 
			
		||||
                    const nameToItems = {};
 | 
			
		||||
                    items.forEach(item => {
 | 
			
		||||
@ -960,6 +991,46 @@ function updateInventory() {
 | 
			
		||||
                document.getElementById("EvolutionProgress-list").appendChild(tr);
 | 
			
		||||
            });
 | 
			
		||||
 | 
			
		||||
            document.getElementById("FlavourItems-list").innerHTML = "";
 | 
			
		||||
            data.FlavourItems.forEach(item => {
 | 
			
		||||
                const datalist = document.getElementById("datalist-FlavourItems");
 | 
			
		||||
                if (!data.FlavourItems.some(x => x.ItemType == item.ItemType)) {
 | 
			
		||||
                    if (!datalist.querySelector(`option[data-key="${item.ItemType}"]`)) {
 | 
			
		||||
                        reAddToItemList(itemMap, "FlavourItems", item.ItemType);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                const optionToRemove = datalist.querySelector(`option[data-key="${item.ItemType}"]`);
 | 
			
		||||
                optionToRemove?.remove();
 | 
			
		||||
 | 
			
		||||
                const tr = document.createElement("tr");
 | 
			
		||||
                tr.setAttribute("data-item-type", item.ItemType);
 | 
			
		||||
                {
 | 
			
		||||
                    const td = document.createElement("td");
 | 
			
		||||
                    td.textContent = itemMap[item.ItemType]?.name ?? item.ItemType;
 | 
			
		||||
                    tr.appendChild(td);
 | 
			
		||||
                }
 | 
			
		||||
                {
 | 
			
		||||
                    const td = document.createElement("td");
 | 
			
		||||
                    td.classList = "text-end text-nowrap";
 | 
			
		||||
 | 
			
		||||
                    {
 | 
			
		||||
                        const a = document.createElement("a");
 | 
			
		||||
                        a.href = "#";
 | 
			
		||||
                        a.onclick = function (event) {
 | 
			
		||||
                            event.preventDefault();
 | 
			
		||||
                            reAddToItemList(itemMap, "FlavourItems", item.ItemType);
 | 
			
		||||
                            removeCustomization(item.ItemType);
 | 
			
		||||
                        };
 | 
			
		||||
                        a.title = loc("code_remove");
 | 
			
		||||
                        a.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 448 512"><!--!Font Awesome Free 6.5.2 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2024 Fonticons, Inc.--><path d="M135.2 17.7L128 32H32C14.3 32 0 46.3 0 64S14.3 96 32 96H416c17.7 0 32-14.3 32-32s-14.3-32-32-32H320l-7.2-14.3C307.4 6.8 296.3 0 284.2 0H163.8c-12.1 0-23.2 6.8-28.6 17.7zM416 128H32L53.2 467c1.6 25.3 22.6 45 47.9 45H346.9c25.3 0 46.3-19.7 47.9-45L416 128z"/></svg>`;
 | 
			
		||||
                        td.appendChild(a);
 | 
			
		||||
                    }
 | 
			
		||||
                    tr.appendChild(td);
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                document.getElementById("FlavourItems-list").appendChild(tr);
 | 
			
		||||
            });
 | 
			
		||||
 | 
			
		||||
            const datalistEvolutionProgress = document.querySelectorAll("#datalist-EvolutionProgress option");
 | 
			
		||||
            const formEvolutionProgress = document.querySelector('form[onsubmit*="doAcquireEvolution()"]');
 | 
			
		||||
 | 
			
		||||
@ -976,6 +1047,40 @@ function updateInventory() {
 | 
			
		||||
                };
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            document.getElementById("ShipDecorations-list").innerHTML = "";
 | 
			
		||||
            data.ShipDecorations.forEach(item => {
 | 
			
		||||
                if (item.ItemCount > 0) {
 | 
			
		||||
                    const tr = document.createElement("tr");
 | 
			
		||||
                    tr.setAttribute("data-item-type", item.ItemType);
 | 
			
		||||
                    {
 | 
			
		||||
                        const td = document.createElement("td");
 | 
			
		||||
                        td.textContent = itemMap[item.ItemType]?.name ?? item.ItemType;
 | 
			
		||||
                        if (item.ItemCount > 1) {
 | 
			
		||||
                            td.innerHTML +=
 | 
			
		||||
                                " <span title='" + loc("code_count") + "'>🗍 " + parseInt(item.ItemCount) + "</span>";
 | 
			
		||||
                        }
 | 
			
		||||
                        tr.appendChild(td);
 | 
			
		||||
                    }
 | 
			
		||||
                    {
 | 
			
		||||
                        const td = document.createElement("td");
 | 
			
		||||
                        td.classList = "text-end text-nowrap";
 | 
			
		||||
                        {
 | 
			
		||||
                            const a = document.createElement("a");
 | 
			
		||||
                            a.href = "#";
 | 
			
		||||
                            a.onclick = function (event) {
 | 
			
		||||
                                event.preventDefault();
 | 
			
		||||
                                removeCountItems(item.ItemType, item.ItemCount);
 | 
			
		||||
                            };
 | 
			
		||||
                            a.title = loc("code_remove");
 | 
			
		||||
                            a.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 448 512"><!--!Font Awesome Free 6.5.2 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2024 Fonticons, Inc.--><path d="M135.2 17.7L128 32H32C14.3 32 0 46.3 0 64S14.3 96 32 96H416c17.7 0 32-14.3 32-32s-14.3-32-32-32H320l-7.2-14.3C307.4 6.8 296.3 0 284.2 0H163.8c-12.1 0-23.2 6.8-28.6 17.7zM416 128H32L53.2 467c1.6 25.3 22.6 45 47.9 45H346.9c25.3 0 46.3-19.7 47.9-45L416 128z"/></svg>`;
 | 
			
		||||
                            td.appendChild(a);
 | 
			
		||||
                        }
 | 
			
		||||
                        tr.appendChild(td);
 | 
			
		||||
                    }
 | 
			
		||||
                    document.getElementById("ShipDecorations-list").appendChild(tr);
 | 
			
		||||
                }
 | 
			
		||||
            });
 | 
			
		||||
 | 
			
		||||
            // Populate quests route
 | 
			
		||||
            document.getElementById("QuestKeys-list").innerHTML = "";
 | 
			
		||||
            window.allQuestKeys.forEach(questKey => {
 | 
			
		||||
@ -989,7 +1094,8 @@ function updateInventory() {
 | 
			
		||||
            data.QuestKeys.forEach(item => {
 | 
			
		||||
                const tr = document.createElement("tr");
 | 
			
		||||
                tr.setAttribute("data-item-type", item.ItemType);
 | 
			
		||||
                const stage = item.Progress?.length ?? 0;
 | 
			
		||||
                const run = item.Progress[0]?.c ?? 0;
 | 
			
		||||
                const stage = run == 0 ? item.Progress.length : item.Progress.map(p => p.c ?? 0).lastIndexOf(run);
 | 
			
		||||
 | 
			
		||||
                const datalist = document.getElementById("datalist-QuestKeys");
 | 
			
		||||
                const optionToRemove = datalist.querySelector(`option[data-key="${item.ItemType}"]`);
 | 
			
		||||
@ -1007,6 +1113,10 @@ function updateInventory() {
 | 
			
		||||
                        td.textContent += " | " + loc("code_completed");
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    if (run > 0) {
 | 
			
		||||
                        td.textContent += " | " + loc("code_replays") + ": " + (run + 1);
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    if (data.ActiveQuest == item.ItemType) td.textContent += " | " + loc("code_active");
 | 
			
		||||
                    tr.appendChild(td);
 | 
			
		||||
                }
 | 
			
		||||
@ -1232,7 +1342,8 @@ function updateInventory() {
 | 
			
		||||
                        td.textContent = itemMap[item.ItemType]?.name ?? item.ItemType;
 | 
			
		||||
                        td.innerHTML += " <span title='" + loc("code_rank") + "'>★ 0/" + maxRank + "</span>";
 | 
			
		||||
                        if (item.ItemCount > 1) {
 | 
			
		||||
                            td.innerHTML += " <span title='Count'>🗍 " + parseInt(item.ItemCount) + "</span>";
 | 
			
		||||
                            td.innerHTML +=
 | 
			
		||||
                                " <span title='" + loc("code_count") + "'>🗍 " + parseInt(item.ItemCount) + "</span>";
 | 
			
		||||
                        }
 | 
			
		||||
                        tr.appendChild(td);
 | 
			
		||||
                    }
 | 
			
		||||
@ -1966,6 +2077,15 @@ function doAcquireEquipment(category) {
 | 
			
		||||
    });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function removeCustomization(uniqueName) {
 | 
			
		||||
    revalidateAuthz().then(() => {
 | 
			
		||||
        const req = $.get("/custom/removeCustomization?" + window.authz + "&itemType=" + uniqueName);
 | 
			
		||||
        req.done(() => {
 | 
			
		||||
            updateInventory();
 | 
			
		||||
        });
 | 
			
		||||
    });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function getRequiredParts(category, WeaponType) {
 | 
			
		||||
    switch (category) {
 | 
			
		||||
        case "Hoverboards":
 | 
			
		||||
@ -2146,8 +2266,13 @@ function addMissingEquipment(categories) {
 | 
			
		||||
                    "#" + category + "-list [data-item-type='" + elm.getAttribute("data-key") + "']"
 | 
			
		||||
                )
 | 
			
		||||
            ) {
 | 
			
		||||
                if (!webUiModularWeapons.includes(elm.getAttribute("data-key"))) {
 | 
			
		||||
                    requests.push({ ItemType: elm.getAttribute("data-key"), ItemCount: 1 });
 | 
			
		||||
                if (
 | 
			
		||||
                    !webUiModularWeapons.includes(elm.getAttribute("data-key")) &&
 | 
			
		||||
                    !elm.hasAttribute("data-exclude-from-add-missing")
 | 
			
		||||
                ) {
 | 
			
		||||
                    let ItemCount = 1;
 | 
			
		||||
                    if (category == "ShipDecorations") ItemCount = 100;
 | 
			
		||||
                    requests.push({ ItemType: elm.getAttribute("data-key"), ItemCount });
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        });
 | 
			
		||||
@ -2667,13 +2792,13 @@ function setEvolutionProgress(requests) {
 | 
			
		||||
    });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function doAcquireMiscItems() {
 | 
			
		||||
    const uniqueName = getKey(document.getElementById("miscitem-type"));
 | 
			
		||||
function doAcquireCountItems(category) {
 | 
			
		||||
    const uniqueName = getKey(document.getElementById(category + "-type"));
 | 
			
		||||
    if (!uniqueName) {
 | 
			
		||||
        $("#miscitem-type").addClass("is-invalid").focus();
 | 
			
		||||
        $(`#acquire-type-${category}`).addClass("is-invalid").focus();
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
    const count = parseInt($("#miscitem-count").val());
 | 
			
		||||
    const count = parseInt($(`#${category}-count`).val());
 | 
			
		||||
    if (count != 0) {
 | 
			
		||||
        revalidateAuthz().then(() => {
 | 
			
		||||
            $.post({
 | 
			
		||||
@ -2691,11 +2816,30 @@ function doAcquireMiscItems() {
 | 
			
		||||
                } else {
 | 
			
		||||
                    toast(loc("code_succRemoved"));
 | 
			
		||||
                }
 | 
			
		||||
                if (category != "miscitem") updateInventory();
 | 
			
		||||
            });
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function removeCountItems(uniqueName, count) {
 | 
			
		||||
    revalidateAuthz().then(() => {
 | 
			
		||||
        $.post({
 | 
			
		||||
            url: "/custom/addItems?" + window.authz,
 | 
			
		||||
            contentType: "application/json",
 | 
			
		||||
            data: JSON.stringify([
 | 
			
		||||
                {
 | 
			
		||||
                    ItemType: uniqueName,
 | 
			
		||||
                    ItemCount: count * -1
 | 
			
		||||
                }
 | 
			
		||||
            ])
 | 
			
		||||
        }).done(function () {
 | 
			
		||||
            toast(loc("code_succRemoved"));
 | 
			
		||||
            updateInventory();
 | 
			
		||||
        });
 | 
			
		||||
    });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function addItemByItemType() {
 | 
			
		||||
    const ItemType = document.getElementById("typeName-type").value;
 | 
			
		||||
    // Must start with "/Lotus/", contain only A–Z letters, no "//", and not end with "/"
 | 
			
		||||
@ -3271,6 +3415,7 @@ function reAddToItemList(itemMap, datalist, itemType) {
 | 
			
		||||
    const option = document.createElement("option");
 | 
			
		||||
    option.setAttribute("data-key", itemType);
 | 
			
		||||
    option.value = itemMap[itemType]?.name ?? itemType;
 | 
			
		||||
    if (itemMap[itemType]?.excludeFromAddMissing) option.setAttribute("data-exclude-from-add-missing", "");
 | 
			
		||||
    document.getElementById("datalist-" + datalist).appendChild(option);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -64,10 +64,16 @@ dict = {
 | 
			
		||||
    code_completed: `Abgeschlossen`,
 | 
			
		||||
    code_active: `Aktiv`,
 | 
			
		||||
    code_pigment: `Pigment`,
 | 
			
		||||
    code_controller: `[UNTRANSLATED] Controller cursor`,
 | 
			
		||||
    code_mouseLine: `[UNTRANSLATED] Line cursor`,
 | 
			
		||||
    code_mouse: `[UNTRANSLATED] Cursor`,
 | 
			
		||||
    code_itemColorPalette: `|ITEM| Farbpalette`,
 | 
			
		||||
    code_mature: `Für den Kampf auswachsen lassen`,
 | 
			
		||||
    code_unmature: `Genetisches Altern zurücksetzen`,
 | 
			
		||||
    code_fund: `[UNTRANSLATED] Fund`,
 | 
			
		||||
    code_funded: `[UNTRANSLATED] Funded`,
 | 
			
		||||
    code_replays: `[UNTRANSLATED] Replays`,
 | 
			
		||||
    code_stalker: `Stalker`,
 | 
			
		||||
    code_succChange: `Erfolgreich geändert.`,
 | 
			
		||||
    code_requiredInvigorationUpgrade: `Du musst sowohl ein offensives & defensives Upgrade auswählen.`,
 | 
			
		||||
    login_description: `Melde dich mit deinem OpenWF-Account an (denselben Angaben wie im Spiel, wenn du dich mit diesem Server verbindest).`,
 | 
			
		||||
@ -103,12 +109,16 @@ dict = {
 | 
			
		||||
    inventory_kubrowPets: `Bestien`,
 | 
			
		||||
    inventory_evolutionProgress: `Incarnon-Entwicklungsfortschritte`,
 | 
			
		||||
    inventory_Boosters: `Booster`,
 | 
			
		||||
    inventory_flavourItems: `<abbr title="Animationssets, Glyphen, Farbpaletten usw.">Sammlerstücke</abbr>`,
 | 
			
		||||
    inventory_shipDecorations: `Schiffsdekorationen`,
 | 
			
		||||
    inventory_bulkAddSuits: `Fehlende Warframes hinzufügen`,
 | 
			
		||||
    inventory_bulkAddWeapons: `Fehlende Waffen hinzufügen`,
 | 
			
		||||
    inventory_bulkAddSpaceSuits: `Fehlende Archwings hinzufügen`,
 | 
			
		||||
    inventory_bulkAddSpaceWeapons: `Fehlende Archwing-Waffen hinzufügen`,
 | 
			
		||||
    inventory_bulkAddSentinels: `Fehlende Wächter hinzufügen`,
 | 
			
		||||
    inventory_bulkAddSentinelWeapons: `Fehlende Wächter-Waffen hinzufügen`,
 | 
			
		||||
    inventory_bulkAddFlavourItems: `[UNTRANSLATED] Add Missing Flavour Items`,
 | 
			
		||||
    inventory_bulkAddShipDecorations: `[UNTRANSLATED] Add Missing Ship Decorations`,
 | 
			
		||||
    inventory_bulkAddEvolutionProgress: `Fehlende Incarnon-Entwicklungsfortschritte hinzufügen`,
 | 
			
		||||
    inventory_bulkRankUpSuits: `Alle Warframes auf Max. Rang`,
 | 
			
		||||
    inventory_bulkRankUpWeapons: `Alle Waffen auf Max. Rang`,
 | 
			
		||||
@ -198,8 +208,6 @@ dict = {
 | 
			
		||||
    cheats_dontSubtractVoidTraces: `Void-Spuren nicht verbrauchen`,
 | 
			
		||||
    cheats_dontSubtractConsumables: `Verbrauchsgegenstände (Ausrüstung) nicht verbrauchen`,
 | 
			
		||||
    cheats_unlockAllShipFeatures: `Alle Schiffs-Funktionen freischalten`,
 | 
			
		||||
    cheats_unlockAllShipDecorations: `Alle Schiffsdekorationen freischalten`,
 | 
			
		||||
    cheats_unlockAllFlavourItems: `Alle <abbr title="Animationssets, Glyphen, Farbpaletten usw.">Sammlerstücke</abbr> freischalten`,
 | 
			
		||||
    cheats_unlockAllSkins: `Alle Skins freischalten`,
 | 
			
		||||
    cheats_unlockAllCapturaScenes: `Alle Photora-Szenen freischalten`,
 | 
			
		||||
    cheats_universalPolarityEverywhere: `Universelle Polarität überall`,
 | 
			
		||||
 | 
			
		||||
@ -63,10 +63,16 @@ dict = {
 | 
			
		||||
    code_completed: `Completed`,
 | 
			
		||||
    code_active: `Active`,
 | 
			
		||||
    code_pigment: `Pigment`,
 | 
			
		||||
    code_controller: `Controller cursor`,
 | 
			
		||||
    code_mouseLine: `Line cursor`,
 | 
			
		||||
    code_mouse: `Cursor`,
 | 
			
		||||
    code_itemColorPalette: `|ITEM| Color Palette`,
 | 
			
		||||
    code_mature: `Mature for combat`,
 | 
			
		||||
    code_unmature: `Regress genetic aging`,
 | 
			
		||||
    code_fund: `Fund`,
 | 
			
		||||
    code_funded: `Funded`,
 | 
			
		||||
    code_replays: `Replays`,
 | 
			
		||||
    code_stalker: `Stalker`,
 | 
			
		||||
    code_succChange: `Successfully changed.`,
 | 
			
		||||
    code_requiredInvigorationUpgrade: `You must select both an offensive & defensive upgrade.`,
 | 
			
		||||
    login_description: `Login using your OpenWF account credentials (same as in-game when connecting to this server).`,
 | 
			
		||||
@ -102,12 +108,16 @@ dict = {
 | 
			
		||||
    inventory_kubrowPets: `Beasts`,
 | 
			
		||||
    inventory_evolutionProgress: `Incarnon Evolution Progress`,
 | 
			
		||||
    inventory_Boosters: `Boosters`,
 | 
			
		||||
    inventory_flavourItems: `<abbr title="Animation Sets, Glyphs, Palettes, etc.">Flavour Items</abbr>`,
 | 
			
		||||
    inventory_shipDecorations: `Ship Decorations`,
 | 
			
		||||
    inventory_bulkAddSuits: `Add Missing Warframes`,
 | 
			
		||||
    inventory_bulkAddWeapons: `Add Missing Weapons`,
 | 
			
		||||
    inventory_bulkAddSpaceSuits: `Add Missing Archwings`,
 | 
			
		||||
    inventory_bulkAddSpaceWeapons: `Add Missing Archwing Weapons`,
 | 
			
		||||
    inventory_bulkAddSentinels: `Add Missing Sentinels`,
 | 
			
		||||
    inventory_bulkAddSentinelWeapons: `Add Missing Sentinel Weapons`,
 | 
			
		||||
    inventory_bulkAddFlavourItems: `Add Missing Flavour Items`,
 | 
			
		||||
    inventory_bulkAddShipDecorations: `Add Missing Ship Decorations`,
 | 
			
		||||
    inventory_bulkAddEvolutionProgress: `Add Missing Incarnon Evolution Progress`,
 | 
			
		||||
    inventory_bulkRankUpSuits: `Max Rank All Warframes`,
 | 
			
		||||
    inventory_bulkRankUpWeapons: `Max Rank All Weapons`,
 | 
			
		||||
@ -197,8 +207,6 @@ dict = {
 | 
			
		||||
    cheats_dontSubtractVoidTraces: `Don't Subtract Void Traces`,
 | 
			
		||||
    cheats_dontSubtractConsumables: `Don't Subtract Consumables`,
 | 
			
		||||
    cheats_unlockAllShipFeatures: `Unlock All Ship Features`,
 | 
			
		||||
    cheats_unlockAllShipDecorations: `Unlock All Ship Decorations`,
 | 
			
		||||
    cheats_unlockAllFlavourItems: `Unlock All <abbr title="Animation Sets, Glyphs, Palettes, etc.">Flavor Items</abbr>`,
 | 
			
		||||
    cheats_unlockAllSkins: `Unlock All Skins`,
 | 
			
		||||
    cheats_unlockAllCapturaScenes: `Unlock All Captura Scenes`,
 | 
			
		||||
    cheats_universalPolarityEverywhere: `Universal Polarity Everywhere`,
 | 
			
		||||
 | 
			
		||||
@ -64,10 +64,16 @@ dict = {
 | 
			
		||||
    code_completed: `Completada`,
 | 
			
		||||
    code_active: `Activa`,
 | 
			
		||||
    code_pigment: `Pigmento`,
 | 
			
		||||
    code_controller: `[UNTRANSLATED] Controller cursor`,
 | 
			
		||||
    code_mouseLine: `[UNTRANSLATED] Line cursor`,
 | 
			
		||||
    code_mouse: `[UNTRANSLATED] Cursor`,
 | 
			
		||||
    code_itemColorPalette: `Paleta de colores |ITEM|`,
 | 
			
		||||
    code_mature: `Listo para el combate`,
 | 
			
		||||
    code_unmature: `Regresar el envejecimiento genético`,
 | 
			
		||||
    code_fund: `[UNTRANSLATED] Fund`,
 | 
			
		||||
    code_funded: `[UNTRANSLATED] Funded`,
 | 
			
		||||
    code_replays: `[UNTRANSLATED] Replays`,
 | 
			
		||||
    code_stalker: `Stalker`,
 | 
			
		||||
    code_succChange: `Cambiado correctamente`,
 | 
			
		||||
    code_requiredInvigorationUpgrade: `Debes seleccionar una mejora ofensiva y una defensiva.`,
 | 
			
		||||
    login_description: `Inicia sesión con las credenciales de tu cuenta OpenWF (las mismas que usas en el juego al conectarte a este servidor).`,
 | 
			
		||||
@ -103,12 +109,16 @@ dict = {
 | 
			
		||||
    inventory_kubrowPets: `Bestias`,
 | 
			
		||||
    inventory_evolutionProgress: `Progreso de evolución Incarnon`,
 | 
			
		||||
    inventory_Boosters: `Potenciadores`,
 | 
			
		||||
    inventory_flavourItems: `<abbr title="Conjuntos de animaciones, glifos, paletas, etc.">Ítems estéticos</abbr>`,
 | 
			
		||||
    inventory_shipDecorations: `Decoraciones de nave`,
 | 
			
		||||
    inventory_bulkAddSuits: `Agregar Warframes faltantes`,
 | 
			
		||||
    inventory_bulkAddWeapons: `Agregar armas faltantes`,
 | 
			
		||||
    inventory_bulkAddSpaceSuits: `Agregar Archwings faltantes`,
 | 
			
		||||
    inventory_bulkAddSpaceWeapons: `Agregar armas Archwing faltantes`,
 | 
			
		||||
    inventory_bulkAddSentinels: `Agregar centinelas faltantes`,
 | 
			
		||||
    inventory_bulkAddSentinelWeapons: `Agregar armas de centinela faltantes`,
 | 
			
		||||
    inventory_bulkAddFlavourItems: `[UNTRANSLATED] Add Missing Flavour Items`,
 | 
			
		||||
    inventory_bulkAddShipDecorations: `[UNTRANSLATED] Add Missing Ship Decorations`,
 | 
			
		||||
    inventory_bulkAddEvolutionProgress: `Completar el progreso de evolución Incarnon faltante`,
 | 
			
		||||
    inventory_bulkRankUpSuits: `Maximizar rango de todos los Warframes`,
 | 
			
		||||
    inventory_bulkRankUpWeapons: `Maximizar rango de todas las armas`,
 | 
			
		||||
@ -198,8 +208,6 @@ dict = {
 | 
			
		||||
    cheats_dontSubtractVoidTraces: `No descontar vestigios del Vacío`,
 | 
			
		||||
    cheats_dontSubtractConsumables: `No restar consumibles`,
 | 
			
		||||
    cheats_unlockAllShipFeatures: `Desbloquear todas las funciones de nave`,
 | 
			
		||||
    cheats_unlockAllShipDecorations: `Desbloquear todas las decoraciones de nave`,
 | 
			
		||||
    cheats_unlockAllFlavourItems: `Desbloquear todos los <abbr title="Conjuntos de animaciones, glifos, paletas, etc.">ítems estéticos</abbr>`,
 | 
			
		||||
    cheats_unlockAllSkins: `Desbloquear todas las skins`,
 | 
			
		||||
    cheats_unlockAllCapturaScenes: `Desbloquear todas las escenas de Captura`,
 | 
			
		||||
    cheats_universalPolarityEverywhere: `Polaridad universal en todas partes`,
 | 
			
		||||
 | 
			
		||||
@ -1,7 +1,7 @@
 | 
			
		||||
// French translation by Vitruvio
 | 
			
		||||
dict = {
 | 
			
		||||
    general_inventoryUpdateNote: `Note : Pour voir les changements en jeu, l'inventaire doit être actualisé. Cela se fait en tapant /sync dans le tchat, en visitant un dojo/relais ou en se reconnectant.`,
 | 
			
		||||
    general_inventoryUpdateNoteGameWs: `[UNTRANSLATED] Note: You may need to reopen any menu you are on for changes to be reflected.`,
 | 
			
		||||
    general_inventoryUpdateNoteGameWs: `Note : Rouvrir un menu est nécessaire pour voir les changements.`,
 | 
			
		||||
    general_addButton: `Ajouter`,
 | 
			
		||||
    general_setButton: `Définir`,
 | 
			
		||||
    general_none: `Aucun`,
 | 
			
		||||
@ -11,7 +11,7 @@ dict = {
 | 
			
		||||
    code_loginFail: `Connexion échouée. Vérifiez le mot de passe.`,
 | 
			
		||||
    code_regFail: `Enregistrement impossible. Compte existant?`,
 | 
			
		||||
    code_changeNameConfirm: `Nouveau nom du compte :`,
 | 
			
		||||
    code_changeNameRetry: `[UNTRANSLATED] |NAME| is already taken.`,
 | 
			
		||||
    code_changeNameRetry: `|NAME| est déjà pris.`,
 | 
			
		||||
    code_deleteAccountConfirm: `Supprimer |DISPLAYNAME| (|EMAIL|) ? Cette action est irreversible.`,
 | 
			
		||||
    code_archgun: `Archgun`,
 | 
			
		||||
    code_melee: `Melee`,
 | 
			
		||||
@ -32,8 +32,8 @@ dict = {
 | 
			
		||||
    code_renamePrompt: `Nouveau nom :`,
 | 
			
		||||
    code_remove: `Retirer`,
 | 
			
		||||
    code_addItemsConfirm: `Ajouter |COUNT| items à l'inventaire ?`,
 | 
			
		||||
    code_addTechProjectsConfirm: `[UNTRANSLATED] Are you sure you want to add |COUNT| research to your clan?`,
 | 
			
		||||
    code_addDecoRecipesConfirm: `[UNTRANSLATED] Are you sure you want to add |COUNT| deco recipes to your clan?`,
 | 
			
		||||
    code_addTechProjectsConfirm: `Ajouter |COUNT| recherches au clan ?`,
 | 
			
		||||
    code_addDecoRecipesConfirm: `Ajouter |COUNT| décorations au clan ?`,
 | 
			
		||||
    code_succRankUp: `Montée de niveau effectuée.`,
 | 
			
		||||
    code_noEquipmentToRankUp: `Aucun équipement à monter de niveau.`,
 | 
			
		||||
    code_succAdded: `Ajouté.`,
 | 
			
		||||
@ -64,10 +64,16 @@ dict = {
 | 
			
		||||
    code_completed: `Complétée`,
 | 
			
		||||
    code_active: `Active`,
 | 
			
		||||
    code_pigment: `Pigment`,
 | 
			
		||||
    code_controller: `[UNTRANSLATED] Controller cursor`,
 | 
			
		||||
    code_mouseLine: `[UNTRANSLATED] Line cursor`,
 | 
			
		||||
    code_mouse: `[UNTRANSLATED] Cursor`,
 | 
			
		||||
    code_itemColorPalette: `Palette de couleurs |ITEM|`,
 | 
			
		||||
    code_mature: `Maturer pour le combat`,
 | 
			
		||||
    code_unmature: `Régrésser l'âge génétique`,
 | 
			
		||||
    code_fund: `[UNTRANSLATED] Fund`,
 | 
			
		||||
    code_funded: `[UNTRANSLATED] Funded`,
 | 
			
		||||
    code_fund: `Financer`,
 | 
			
		||||
    code_funded: `Complété`,
 | 
			
		||||
    code_replays: `[UNTRANSLATED] Replays`,
 | 
			
		||||
    code_stalker: `Stalker`,
 | 
			
		||||
    code_succChange: `Changement effectué.`,
 | 
			
		||||
    code_requiredInvigorationUpgrade: `Augmentation offensive et défensive requises.`,
 | 
			
		||||
    login_description: `Connexion avec les informations de connexion OpenWF.`,
 | 
			
		||||
@ -85,8 +91,8 @@ dict = {
 | 
			
		||||
    navbar_cheats: `Cheats`,
 | 
			
		||||
    navbar_import: `Importer`,
 | 
			
		||||
    inventory_addItems: `Ajouter des items`,
 | 
			
		||||
    inventory_addItemByItemType: `[UNTRANSLATED] Raw`,
 | 
			
		||||
    inventory_addItemByItemType_warning: `[UNTRANSLATED] Use this feature at your own risk. It may break your inventory, and you will need to remove items manually if something goes wrong.`,
 | 
			
		||||
    inventory_addItemByItemType: `Brut`,
 | 
			
		||||
    inventory_addItemByItemType_warning: `Cette fonctionnalité comporte des risques. Il faudra rajouter les items manuellement si l'inventaire est compris.`,
 | 
			
		||||
    inventory_suits: `Warframes`,
 | 
			
		||||
    inventory_longGuns: `Armes principales`,
 | 
			
		||||
    inventory_pistols: `Armes secondaires`,
 | 
			
		||||
@ -103,12 +109,16 @@ dict = {
 | 
			
		||||
    inventory_kubrowPets: `Bêtes`,
 | 
			
		||||
    inventory_evolutionProgress: `Progrès de l'évolution Incarnon`,
 | 
			
		||||
    inventory_Boosters: `Boosters`,
 | 
			
		||||
    inventory_flavourItems: `[UNTRANSLATED] <abbr title="Animation Sets, Glyphs, Palettes, etc.">Flavour Items</abbr>`,
 | 
			
		||||
    inventory_shipDecorations: `Décorations du vaisseau`,
 | 
			
		||||
    inventory_bulkAddSuits: `Ajouter les Warframes manquantes`,
 | 
			
		||||
    inventory_bulkAddWeapons: `Ajouter les armes manquantes`,
 | 
			
		||||
    inventory_bulkAddSpaceSuits: `Ajouter les Archwings manquants`,
 | 
			
		||||
    inventory_bulkAddSpaceWeapons: `Ajouter les armes d'Archwing manquantes`,
 | 
			
		||||
    inventory_bulkAddSentinels: `Ajouter les Sentinelles manquantes`,
 | 
			
		||||
    inventory_bulkAddSentinelWeapons: `Ajouter les armes de Sentinelles manquantes`,
 | 
			
		||||
    inventory_bulkAddFlavourItems: `[UNTRANSLATED] Add Missing Flavour Items`,
 | 
			
		||||
    inventory_bulkAddShipDecorations: `[UNTRANSLATED] Add Missing Ship Decorations`,
 | 
			
		||||
    inventory_bulkAddEvolutionProgress: `Ajouter les évolutions Incarnon manquantes`,
 | 
			
		||||
    inventory_bulkRankUpSuits: `Toutes les Warframes au rang max`,
 | 
			
		||||
    inventory_bulkRankUpWeapons: `Toutes les armes au rang max`,
 | 
			
		||||
@ -182,7 +192,7 @@ dict = {
 | 
			
		||||
    cheats_skipTutorial: `Passer le tutoriel`,
 | 
			
		||||
    cheats_skipAllDialogue: `Passer les dialogues`,
 | 
			
		||||
    cheats_unlockAllScans: `Débloquer tous les scans`,
 | 
			
		||||
    cheats_unlockSuccRelog: `[UNTRANSLATED] Success. Please that you'll need to relog for the client to refresh this.`,
 | 
			
		||||
    cheats_unlockSuccRelog: `Succès. Une reconnexion est requise pour appliquer les changements.`,
 | 
			
		||||
    cheats_unlockAllMissions: `Débloquer toutes les missions`,
 | 
			
		||||
    cheats_unlockAllMissions_ok: `Succès. Une actualisation de l'inventaire est nécessaire.`,
 | 
			
		||||
    cheats_infiniteCredits: `Crédits infinis`,
 | 
			
		||||
@ -198,8 +208,6 @@ dict = {
 | 
			
		||||
    cheats_dontSubtractVoidTraces: `Ne pas consommer de Void Traces`,
 | 
			
		||||
    cheats_dontSubtractConsumables: `Ne pas retirer de consommables`,
 | 
			
		||||
    cheats_unlockAllShipFeatures: `Débloquer tous les segments du vaisseau`,
 | 
			
		||||
    cheats_unlockAllShipDecorations: `Débloquer toutes les décorations du vaisseau`,
 | 
			
		||||
    cheats_unlockAllFlavourItems: `Débloquer tous les <abbr title="Animations, Glyphes, Palettes, etc.">Flavor Items</abbr>`,
 | 
			
		||||
    cheats_unlockAllSkins: `Débloquer tous les skins`,
 | 
			
		||||
    cheats_unlockAllCapturaScenes: `Débloquer toutes les scènes captura`,
 | 
			
		||||
    cheats_universalPolarityEverywhere: `Polarités universelles partout`,
 | 
			
		||||
@ -218,7 +226,7 @@ dict = {
 | 
			
		||||
    cheats_baroFullyStocked: `Stock de Baro au max`,
 | 
			
		||||
    cheats_syndicateMissionsRepeatable: `Mission syndicat répétables`,
 | 
			
		||||
    cheats_unlockAllProfitTakerStages: `Débloquer toutes les étapes du Preneur de Profit`,
 | 
			
		||||
    cheats_unlockSuccInventory: `[UNTRANSLATED] Success. Please note that you'll need to resync your inventory, e.g. using the bootstrapper's /sync command, visiting a dojo/relay, or relogging..`,
 | 
			
		||||
    cheats_unlockSuccInventory: `Succès. Une resynchronisation est nécessaire en tapant "/sync" dans le tchat, en visitant un relais ou en se reconnectant.`,
 | 
			
		||||
    cheats_instantFinishRivenChallenge: `Débloquer le challenge Riven instantanément`,
 | 
			
		||||
    cheats_instantResourceExtractorDrones: `Ressources de drones d'extraction instantannées`,
 | 
			
		||||
    cheats_noResourceExtractorDronesDamage: `Aucun dégâts aux drones d'extraction de resources`,
 | 
			
		||||
@ -247,7 +255,7 @@ dict = {
 | 
			
		||||
    cheats_changeSupportedSyndicate: `Allégeance`,
 | 
			
		||||
    cheats_changeButton: `Changer`,
 | 
			
		||||
    cheats_markAllAsRead: `Marquer la boîte de réception comme lue`,
 | 
			
		||||
    cheats_finishInvasionsInOneMission: `[UNTRANSLATED] Finish Invasions in One Mission`,
 | 
			
		||||
    cheats_finishInvasionsInOneMission: `Compléter les invasions en une mission.`,
 | 
			
		||||
 | 
			
		||||
    worldState: `Carte Solaire`,
 | 
			
		||||
    worldState_creditBoost: `Booster de Crédit`,
 | 
			
		||||
@ -257,30 +265,30 @@ dict = {
 | 
			
		||||
    worldState_baroTennoConRelay: `Relais Baro TennoCon`,
 | 
			
		||||
    worldState_starDays: `Jours Stellaires`,
 | 
			
		||||
    worldState_galleonOfGhouls: `Galion des Goules`,
 | 
			
		||||
    worldState_anniversary: `[UNTRANSLATED] Warframe Anniversary`,
 | 
			
		||||
    worldState_useAnniversaryTagForOldGoals: `[UNTRANSLATED] Use <code>Tag</code> from Warframe Anniversary for old Events`,
 | 
			
		||||
    worldState_anniversary: `Anniversaire de Warframe`,
 | 
			
		||||
    worldState_useAnniversaryTagForOldGoals: `Utiliser <code>Tag</code> de l'Anniversaire de Warframe pour les anciens événements`,
 | 
			
		||||
    worldState_ghoulEmergence: `Purge des Goules`,
 | 
			
		||||
    worldState_plagueStar: `Fléau Céleste`,
 | 
			
		||||
    worldState_dogDays: `Bataille d'Eau`,
 | 
			
		||||
    worldState_dogDaysRewards: `[UNTRANSLATED] Dog Days Rewards`,
 | 
			
		||||
    worldState_dogDaysRewards: `Récompenses de la Bataille d'Eau`,
 | 
			
		||||
    worldState_wolfHunt: `Chasse au Loup (2025)`,
 | 
			
		||||
    worldState_orphixVenom: `Venin Orphix`,
 | 
			
		||||
    worldState_longShadow: `La Propagation des Ombres`,
 | 
			
		||||
    worldState_hallowedFlame: `Flamme Hantée`,
 | 
			
		||||
    worldState_hallowedNightmares: `Cauchemars Hantés`,
 | 
			
		||||
    worldState_hallowedNightmaresRewards: `[UNTRANSLATED] Hallowed Nightmares Rewards`,
 | 
			
		||||
    worldState_hallowedNightmaresRewards: `Récompenses Flamme Hantée Cauchemar`,
 | 
			
		||||
    worldState_proxyRebellion: `Rébellion Proxy`,
 | 
			
		||||
    worldState_proxyRebellionRewards: `[UNTRANSLATED] Proxy Rebellion Rewards`,
 | 
			
		||||
    worldState_proxyRebellionRewards: `Récompenses Rébellion Proxy`,
 | 
			
		||||
    worldState_bellyOfTheBeast: `Ventre de la Bête`,
 | 
			
		||||
    worldState_bellyOfTheBeastProgressOverride: `[UNTRANSLATED] Belly of the Beast Progress`,
 | 
			
		||||
    worldState_bellyOfTheBeastProgressOverride: `Progrès du Ventre de la Bête`,
 | 
			
		||||
    worldState_eightClaw: `Huitième Griffe`,
 | 
			
		||||
    worldState_eightClawProgressOverride: `[UNTRANSLATED] Eight Claw Progress`,
 | 
			
		||||
    worldState_eightClawProgressOverride: `Progrès de la Huitième Griffe`,
 | 
			
		||||
    worldState_thermiaFractures: `Crevasses Thermia`,
 | 
			
		||||
    worldState_thermiaFracturesProgressOverride: `[UNTRANSLATED] Thermia Fractures Progress`,
 | 
			
		||||
    worldState_from_year: `[UNTRANSLATED] from |VAL|`,
 | 
			
		||||
    worldState_pre_year: `[UNTRANSLATED] pre |VAL|`,
 | 
			
		||||
    worldState_week: `[UNTRANSLATED] Week |VAL|`,
 | 
			
		||||
    worldState_incompatibleWith: `[UNTRANSLATED] Incompatible with:`,
 | 
			
		||||
    worldState_thermiaFracturesProgressOverride: `Progrès des Fractures Thermia`,
 | 
			
		||||
    worldState_from_year: `de |VAL|`,
 | 
			
		||||
    worldState_pre_year: `pre-|VAL|`,
 | 
			
		||||
    worldState_week: `Semaine |VAL|`,
 | 
			
		||||
    worldState_incompatibleWith: `Incompatible avec :`,
 | 
			
		||||
    enabled: `Activé`,
 | 
			
		||||
    disabled: `Désactivé`,
 | 
			
		||||
    worldState_we1: `Weekend 1`,
 | 
			
		||||
@ -324,8 +332,8 @@ dict = {
 | 
			
		||||
    worldState_varziaFullyStocked: `Stock de Varzia au max`,
 | 
			
		||||
    worldState_varziaOverride: `Rotation de Varzia`,
 | 
			
		||||
 | 
			
		||||
    import_importNote: `[UNTRANSLATED] You can provide a full or partial <code>inventory.php</code> or <code>getShip.php</code> response (client representation) here.`,
 | 
			
		||||
    import_importNote2: `[UNTRANSLATED] All fields that are supported by the importer <b>will be overwritten</b> in your account.`,
 | 
			
		||||
    import_importNote: `Une réponse partielle ou complète de <code>inventory.php</code> ou <code>getShip.php</code> peut être incluse ici.`,
 | 
			
		||||
    import_importNote2: `Tous les champs sont supportés par l'outil d'import <b>serront écrasés</b> sur le compte.`,
 | 
			
		||||
    import_submit: `Soumettre`,
 | 
			
		||||
    import_samples: `Échantillons :`,
 | 
			
		||||
    import_samples_maxFocus: `Toutes les écoles de focus au rang max`,
 | 
			
		||||
@ -394,7 +402,7 @@ dict = {
 | 
			
		||||
 | 
			
		||||
    guildView_cheats: `[UNTRANSLATED] Clan Cheats`,
 | 
			
		||||
    guildView_techProjects: `Recherche`,
 | 
			
		||||
    guildView_vaultDecoRecipes: `[UNTRANSLATED] Dojo Deco Recipes`,
 | 
			
		||||
    guildView_vaultDecoRecipes: `Schémas de décorations de dojo`,
 | 
			
		||||
    guildView_alliance: `Alliance`,
 | 
			
		||||
    guildView_members: `Members`,
 | 
			
		||||
    guildView_pending: `En Attente`,
 | 
			
		||||
@ -414,11 +422,11 @@ dict = {
 | 
			
		||||
    guildView_rank_soldier: `Soldat`,
 | 
			
		||||
    guildView_rank_utility: `Utilitaire`,
 | 
			
		||||
    guildView_rank_warlord: `Seigneur de guerre`,
 | 
			
		||||
    guildView_currency_owned: `[UNTRANSLATED] |COUNT| in Vault.`,
 | 
			
		||||
    guildView_bulkAddTechProjects: `[UNTRANSLATED] Add Missing Research`,
 | 
			
		||||
    guildView_bulkAddVaultDecoRecipes: `[UNTRANSLATED] Add Missing Dojo Deco Recipes`,
 | 
			
		||||
    guildView_bulkFundTechProjects: `[UNTRANSLATED] Fund All Research`,
 | 
			
		||||
    guildView_bulkCompleteTechProjects: `[UNTRANSLATED] Complete All Research`,
 | 
			
		||||
    guildView_currency_owned: `|COUNT| dans le coffre.`,
 | 
			
		||||
    guildView_bulkAddTechProjects: `Ajouter les recherches manquantes`,
 | 
			
		||||
    guildView_bulkAddVaultDecoRecipes: `Ajouter les schémas de décorations de dojo manquantes`,
 | 
			
		||||
    guildView_bulkFundTechProjects: `Financer toutes les recherches`,
 | 
			
		||||
    guildView_bulkCompleteTechProjects: `Compléter toutes les recherches`,
 | 
			
		||||
    guildView_promote: `Promouvoir`,
 | 
			
		||||
    guildView_demote: `Rétrograder`,
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -64,10 +64,16 @@ dict = {
 | 
			
		||||
    code_completed: `Завершено`,
 | 
			
		||||
    code_active: `Активный`,
 | 
			
		||||
    code_pigment: `Пигмент`,
 | 
			
		||||
    code_controller: `Курсор контроллера`,
 | 
			
		||||
    code_mouseLine: `Линейный курсор`,
 | 
			
		||||
    code_mouse: `Курсор`,
 | 
			
		||||
    code_itemColorPalette: `Цветовая палитра: |ITEM|`,
 | 
			
		||||
    code_mature: `Подготовить к сражениям`,
 | 
			
		||||
    code_unmature: `Регрессия генетического старения`,
 | 
			
		||||
    code_fund: `Профинансировать`,
 | 
			
		||||
    code_funded: `Профинансировано`,
 | 
			
		||||
    code_replays: `Повторов`,
 | 
			
		||||
    code_stalker: `Сталкер`,
 | 
			
		||||
    code_succChange: `Успешно изменено.`,
 | 
			
		||||
    code_requiredInvigorationUpgrade: `Вы должны выбрать как атакующее, так и вспомогательное улучшение.`,
 | 
			
		||||
    login_description: `Войдите, используя учетные данные OpenWF (те же, что и в игре при подключении к этому серверу).`,
 | 
			
		||||
@ -103,12 +109,16 @@ dict = {
 | 
			
		||||
    inventory_kubrowPets: `Звери`,
 | 
			
		||||
    inventory_evolutionProgress: `Прогресс эволюции Инкарнонов`,
 | 
			
		||||
    inventory_Boosters: `Бустеры`,
 | 
			
		||||
    inventory_flavourItems: `<abbr title="Наборы анимаций, глифы, палитры и т. д.">Уникальные предметы</abbr>`,
 | 
			
		||||
    inventory_shipDecorations: `Украшения корабля`,
 | 
			
		||||
    inventory_bulkAddSuits: `Добавить отсутствующие Варфреймы`,
 | 
			
		||||
    inventory_bulkAddWeapons: `Добавить отсутствующее оружие`,
 | 
			
		||||
    inventory_bulkAddSpaceSuits: `Добавить отсутствующие Арчвинги`,
 | 
			
		||||
    inventory_bulkAddSpaceWeapons: `Добавить отсутствующее оружие Арчвингов`,
 | 
			
		||||
    inventory_bulkAddSentinels: `Добавить отсутствующих Стражей`,
 | 
			
		||||
    inventory_bulkAddSentinelWeapons: `Добавить отсутствующее оружие Стражей`,
 | 
			
		||||
    inventory_bulkAddFlavourItems: `Добавить отсутствующие Уникальные предметы`,
 | 
			
		||||
    inventory_bulkAddShipDecorations: `Добавить отсутствующие украшения корабля`,
 | 
			
		||||
    inventory_bulkAddEvolutionProgress: `Добавить отсутствующий прогресс эволюции Инкарнонов`,
 | 
			
		||||
    inventory_bulkRankUpSuits: `Макс. ранг всех Варфреймов`,
 | 
			
		||||
    inventory_bulkRankUpWeapons: `Макс. ранг всего оружия`,
 | 
			
		||||
@ -198,8 +208,6 @@ dict = {
 | 
			
		||||
    cheats_dontSubtractVoidTraces: `Не вычитать количество Отголосков Бездны`,
 | 
			
		||||
    cheats_dontSubtractConsumables: `Не вычитать количество расходников`,
 | 
			
		||||
    cheats_unlockAllShipFeatures: `Разблокировать все функции корабля`,
 | 
			
		||||
    cheats_unlockAllShipDecorations: `Разблокировать все украшения корабля`,
 | 
			
		||||
    cheats_unlockAllFlavourItems: `Разблокировать все <abbr title="Наборы анимаций, глифы, палитры и т. д.">уникальные предметы</abbr>`,
 | 
			
		||||
    cheats_unlockAllSkins: `Разблокировать все скины`,
 | 
			
		||||
    cheats_unlockAllCapturaScenes: `Разблокировать все сцены Каптуры`,
 | 
			
		||||
    cheats_universalPolarityEverywhere: `Универсальная полярность везде`,
 | 
			
		||||
 | 
			
		||||
@ -64,10 +64,16 @@ dict = {
 | 
			
		||||
    code_completed: `Завершено`,
 | 
			
		||||
    code_active: `Активний`,
 | 
			
		||||
    code_pigment: `Барвник`,
 | 
			
		||||
    code_controller: `[UNTRANSLATED] Controller cursor`,
 | 
			
		||||
    code_mouseLine: `[UNTRANSLATED] Line cursor`,
 | 
			
		||||
    code_mouse: `[UNTRANSLATED] Cursor`,
 | 
			
		||||
    code_itemColorPalette: `Палітра кольорів «|ITEM|»`,
 | 
			
		||||
    code_mature: `Виростити для бою`,
 | 
			
		||||
    code_unmature: `Обернути старіння`,
 | 
			
		||||
    code_fund: `[UNTRANSLATED] Fund`,
 | 
			
		||||
    code_funded: `[UNTRANSLATED] Funded`,
 | 
			
		||||
    code_replays: `[UNTRANSLATED] Replays`,
 | 
			
		||||
    code_stalker: `Сталкер`,
 | 
			
		||||
    code_succChange: `Успішно змінено.`,
 | 
			
		||||
    code_requiredInvigorationUpgrade: `Ви повинні вибрати як атакуюче, так і допоміжне вдосконалення.`,
 | 
			
		||||
    login_description: `Увійдіть, використовуючи облікові дані OpenWF (ті ж, що й у грі при підключенні до цього серверу).`,
 | 
			
		||||
@ -103,12 +109,16 @@ dict = {
 | 
			
		||||
    inventory_kubrowPets: `Тварини`,
 | 
			
		||||
    inventory_evolutionProgress: `Прогрес еволюції Інкарнонів`,
 | 
			
		||||
    inventory_Boosters: `Посилення`,
 | 
			
		||||
    inventory_flavourItems: `<abbr title="Набори анімацій, гліфи, палітри і т. д.">Унікальні предмети</abbr>`,
 | 
			
		||||
    inventory_shipDecorations: `Прикраси судна`,
 | 
			
		||||
    inventory_bulkAddSuits: `Додати відсутні Ворфрейми`,
 | 
			
		||||
    inventory_bulkAddWeapons: `Додати відсутню зброю`,
 | 
			
		||||
    inventory_bulkAddSpaceSuits: `Додати відсутні Арквінґи`,
 | 
			
		||||
    inventory_bulkAddSpaceWeapons: `Додати відсутню зброю Арквінґів`,
 | 
			
		||||
    inventory_bulkAddSentinels: `Додати відсутніх Вартових`,
 | 
			
		||||
    inventory_bulkAddSentinelWeapons: `Додати відсутню зброю Вартових`,
 | 
			
		||||
    inventory_bulkAddFlavourItems: `[UNTRANSLATED] Add Missing Flavour Items`,
 | 
			
		||||
    inventory_bulkAddShipDecorations: `[UNTRANSLATED] Add Missing Ship Decorations`,
 | 
			
		||||
    inventory_bulkAddEvolutionProgress: `Додати відсутній прогрес еволюції Інкарнонів`,
 | 
			
		||||
    inventory_bulkRankUpSuits: `Макс. рівень всіх Ворфреймів`,
 | 
			
		||||
    inventory_bulkRankUpWeapons: `Макс. рівень всієї зброї`,
 | 
			
		||||
@ -198,8 +208,6 @@ dict = {
 | 
			
		||||
    cheats_dontSubtractVoidTraces: `Не вираховувати кількість Відлуння`,
 | 
			
		||||
    cheats_dontSubtractConsumables: `Не вираховувати кількість витратних матеріалів`,
 | 
			
		||||
    cheats_unlockAllShipFeatures: `Розблокувати всі функції судна`,
 | 
			
		||||
    cheats_unlockAllShipDecorations: `Розблокувати всі прикраси судна`,
 | 
			
		||||
    cheats_unlockAllFlavourItems: `Розблокувати всі <abbr title="Набори анімацій, гліфи, палітри і т. д.">унікальні предмети</abbr>`,
 | 
			
		||||
    cheats_unlockAllSkins: `Розблокувати всі скіни`,
 | 
			
		||||
    cheats_unlockAllCapturaScenes: `Розблокувати всі сцени Світлописця`,
 | 
			
		||||
    cheats_universalPolarityEverywhere: `Будь-яка полярність скрізь`,
 | 
			
		||||
 | 
			
		||||
@ -64,10 +64,16 @@ dict = {
 | 
			
		||||
    code_completed: `已完成`,
 | 
			
		||||
    code_active: `正在执行`,
 | 
			
		||||
    code_pigment: `颜料`,
 | 
			
		||||
    code_controller: `[UNTRANSLATED] Controller cursor`,
 | 
			
		||||
    code_mouseLine: `[UNTRANSLATED] Line cursor`,
 | 
			
		||||
    code_mouse: `[UNTRANSLATED] Cursor`,
 | 
			
		||||
    code_itemColorPalette: `|ITEM| 调色盘`,
 | 
			
		||||
    code_mature: `成长并战备`,
 | 
			
		||||
    code_unmature: `逆转衰老基因`,
 | 
			
		||||
    code_fund: `[UNTRANSLATED] Fund`,
 | 
			
		||||
    code_funded: `[UNTRANSLATED] Funded`,
 | 
			
		||||
    code_replays: `[UNTRANSLATED] Replays`,
 | 
			
		||||
    code_stalker: `追猎者`,
 | 
			
		||||
    code_succChange: `更改成功`,
 | 
			
		||||
    code_requiredInvigorationUpgrade: `您必须同时选择一个进攻型和一个功能型活化属性.`,
 | 
			
		||||
    login_description: `使用您的 OpenWF 账户凭证登录(与游戏内连接本服务器时使用的昵称相同)`,
 | 
			
		||||
@ -103,12 +109,16 @@ dict = {
 | 
			
		||||
    inventory_kubrowPets: `动物同伴`,
 | 
			
		||||
    inventory_evolutionProgress: `灵化之源进度`,
 | 
			
		||||
    inventory_Boosters: `加成器`,
 | 
			
		||||
    inventory_flavourItems: `<abbr title="动作表情、浮印、调色板等">装饰物品</abbr>`,
 | 
			
		||||
    inventory_shipDecorations: `飞船装饰`,
 | 
			
		||||
    inventory_bulkAddSuits: `添加缺失战甲`,
 | 
			
		||||
    inventory_bulkAddWeapons: `添加缺失武器`,
 | 
			
		||||
    inventory_bulkAddSpaceSuits: `添加缺失载具`,
 | 
			
		||||
    inventory_bulkAddSpaceWeapons: `添加缺失载具武器`,
 | 
			
		||||
    inventory_bulkAddSentinels: `添加缺失守护`,
 | 
			
		||||
    inventory_bulkAddSentinelWeapons: `添加缺失守护武器`,
 | 
			
		||||
    inventory_bulkAddFlavourItems: `[UNTRANSLATED] Add Missing Flavour Items`,
 | 
			
		||||
    inventory_bulkAddShipDecorations: `[UNTRANSLATED] Add Missing Ship Decorations`,
 | 
			
		||||
    inventory_bulkAddEvolutionProgress: `添加缺失的灵化之源进度`,
 | 
			
		||||
    inventory_bulkRankUpSuits: `所有战甲升满级`,
 | 
			
		||||
    inventory_bulkRankUpWeapons: `所有武器升满级`,
 | 
			
		||||
@ -198,8 +208,6 @@ dict = {
 | 
			
		||||
    cheats_dontSubtractVoidTraces: `虚空光体无消耗`,
 | 
			
		||||
    cheats_dontSubtractConsumables: `消耗物品使用时无损耗`,
 | 
			
		||||
    cheats_unlockAllShipFeatures: `解锁所有飞船功能`,
 | 
			
		||||
    cheats_unlockAllShipDecorations: `解锁所有飞船装饰`,
 | 
			
		||||
    cheats_unlockAllFlavourItems: `解锁所有<abbr title="动作表情、浮印、调色板等">装饰物品</abbr>`,
 | 
			
		||||
    cheats_unlockAllSkins: `解锁所有外观`,
 | 
			
		||||
    cheats_unlockAllCapturaScenes: `解锁所有Captura场景`,
 | 
			
		||||
    cheats_universalPolarityEverywhere: `全局万用极性`,
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user