forked from OpenWF/SpaceNinjaServer
Compare commits
10 Commits
1f8d437fad
...
2a2c57a932
| Author | SHA1 | Date | |
|---|---|---|---|
| 2a2c57a932 | |||
| 86998b6760 | |||
| be3dd7ab66 | |||
| e6fb675e21 | |||
| fb4c42490e | |||
| 96a15e25df | |||
| ff234c9874 | |||
| 7d3915fe05 | |||
| 4b3e2dfc62 | |||
| 737d013655 |
8
package-lock.json
generated
8
package-lock.json
generated
@ -18,7 +18,7 @@
|
||||
"morgan": "^1.10.0",
|
||||
"ncp": "^2.0.0",
|
||||
"undici": "^7.10.0",
|
||||
"warframe-public-export-plus": "^0.5.90",
|
||||
"warframe-public-export-plus": "^0.5.91",
|
||||
"warframe-riven-info": "^0.1.2",
|
||||
"winston": "^3.17.0",
|
||||
"winston-daily-rotate-file": "^5.0.0",
|
||||
@ -5534,9 +5534,9 @@
|
||||
}
|
||||
},
|
||||
"node_modules/warframe-public-export-plus": {
|
||||
"version": "0.5.90",
|
||||
"resolved": "https://registry.npmjs.org/warframe-public-export-plus/-/warframe-public-export-plus-0.5.90.tgz",
|
||||
"integrity": "sha512-PzYFJ+qOZawPOVx9+hGlOosGdqSJMSRL6c1z3CCXMiY5SEHUs7p9JIMjkqnvg1g7vflVWnyG6E5/2gYkQs0d7w=="
|
||||
"version": "0.5.91",
|
||||
"resolved": "https://registry.npmjs.org/warframe-public-export-plus/-/warframe-public-export-plus-0.5.91.tgz",
|
||||
"integrity": "sha512-xp8rq/dvFC6+urb6vVFRtAmm1v0iE/ZALI3uVGBpblsVB/PWmGxjBBp8l00dCZs67JsqEKcrCcogKwtKTwDc1w=="
|
||||
},
|
||||
"node_modules/warframe-riven-info": {
|
||||
"version": "0.1.2",
|
||||
|
||||
@ -22,7 +22,7 @@
|
||||
"lint:fix": "eslint --fix --ext .ts .",
|
||||
"prettier": "prettier --write .",
|
||||
"update-translations": "cd scripts && node update-translations.cjs",
|
||||
"fix": "npm run update-translations && npm run lint:fix"
|
||||
"fix": "npm run update-translations && npm run prettier"
|
||||
},
|
||||
"license": "GNU",
|
||||
"type": "module",
|
||||
@ -36,7 +36,7 @@
|
||||
"morgan": "^1.10.0",
|
||||
"ncp": "^2.0.0",
|
||||
"undici": "^7.10.0",
|
||||
"warframe-public-export-plus": "^0.5.90",
|
||||
"warframe-public-export-plus": "^0.5.91",
|
||||
"warframe-riven-info": "^0.1.2",
|
||||
"winston": "^3.17.0",
|
||||
"winston-daily-rotate-file": "^5.0.0",
|
||||
|
||||
@ -1,23 +1,91 @@
|
||||
import type { RequestHandler } from "express";
|
||||
import { getAccountIdForRequest } from "../../services/loginService.ts";
|
||||
import { getAccountForRequest } from "../../services/loginService.ts";
|
||||
import { getInventory, addMiscItems, addEquipment, occupySlot } from "../../services/inventoryService.ts";
|
||||
import type { IMiscItem, TFocusPolarity, TEquipmentKey } from "../../types/inventoryTypes/inventoryTypes.ts";
|
||||
import { InventorySlot } from "../../types/inventoryTypes/inventoryTypes.ts";
|
||||
import { logger } from "../../utils/logger.ts";
|
||||
import { ExportFocusUpgrades } from "warframe-public-export-plus";
|
||||
import { Inventory } from "../../models/inventoryModels/inventoryModel.ts";
|
||||
import { version_compare } from "../../helpers/inventoryHelpers.ts";
|
||||
|
||||
export const focusController: RequestHandler = async (req, res) => {
|
||||
const accountId = await getAccountIdForRequest(req);
|
||||
switch (req.query.op) {
|
||||
const account = await getAccountForRequest(req);
|
||||
|
||||
let op = req.query.op as string;
|
||||
const focus2 = account.BuildLabel && version_compare(account.BuildLabel, "2022.04.29.12.53") < 0;
|
||||
if (focus2) {
|
||||
// Focus 2.0
|
||||
switch (req.query.op) {
|
||||
case Focus2Operation.InstallLens:
|
||||
op = "InstallLens";
|
||||
break;
|
||||
case Focus2Operation.UnlockWay:
|
||||
op = "UnlockWay";
|
||||
break;
|
||||
case Focus2Operation.UnlockUpgrade:
|
||||
op = "UnlockUpgrade";
|
||||
break;
|
||||
case Focus2Operation.IncreasePool:
|
||||
op = "IncreasePool";
|
||||
break;
|
||||
case Focus2Operation.LevelUpUpgrade:
|
||||
op = "LevelUpUpgrade";
|
||||
break;
|
||||
case Focus2Operation.ActivateWay:
|
||||
op = "ActivateWay";
|
||||
break;
|
||||
case Focus2Operation.UpdateUpgrade:
|
||||
op = "UpdateUpgrade";
|
||||
break;
|
||||
case Focus2Operation.SentTrainingAmplifier:
|
||||
op = "SentTrainingAmplifier";
|
||||
break;
|
||||
case Focus2Operation.UnbindUpgrade:
|
||||
op = "UnbindUpgrade";
|
||||
break;
|
||||
case Focus2Operation.ConvertShard:
|
||||
op = "ConvertShard";
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
// Focus 3.0
|
||||
switch (req.query.op) {
|
||||
case Focus3Operation.InstallLens:
|
||||
op = "InstallLens";
|
||||
break;
|
||||
case Focus3Operation.UnlockWay:
|
||||
op = "UnlockWay";
|
||||
break;
|
||||
case Focus3Operation.UnlockUpgrade:
|
||||
op = "UnlockUpgrade";
|
||||
break;
|
||||
case Focus3Operation.LevelUpUpgrade:
|
||||
op = "LevelUpUpgrade";
|
||||
break;
|
||||
case Focus3Operation.ActivateWay:
|
||||
op = "ActivateWay";
|
||||
break;
|
||||
case Focus3Operation.SentTrainingAmplifier:
|
||||
op = "SentTrainingAmplifier";
|
||||
break;
|
||||
case Focus3Operation.UnbindUpgrade:
|
||||
op = "UnbindUpgrade";
|
||||
break;
|
||||
case Focus3Operation.ConvertShard:
|
||||
op = "ConvertShard";
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
switch (op) {
|
||||
default:
|
||||
logger.error("Unhandled focus op type: " + String(req.query.op));
|
||||
logger.debug(String(req.body));
|
||||
res.end();
|
||||
break;
|
||||
case FocusOperation.InstallLens: {
|
||||
case "InstallLens": {
|
||||
const request = JSON.parse(String(req.body)) as ILensInstallRequest;
|
||||
const inventory = await getInventory(accountId);
|
||||
const inventory = await getInventory(account._id.toString());
|
||||
const item = inventory[request.Category].id(request.WeaponId);
|
||||
if (item) {
|
||||
item.FocusLens = request.LensType;
|
||||
@ -35,10 +103,10 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
});
|
||||
break;
|
||||
}
|
||||
case FocusOperation.UnlockWay: {
|
||||
case "UnlockWay": {
|
||||
const focusType = (JSON.parse(String(req.body)) as IWayRequest).FocusType;
|
||||
const focusPolarity = focusTypeToPolarity(focusType);
|
||||
const inventory = await getInventory(accountId, "FocusAbility FocusUpgrades FocusXP");
|
||||
const inventory = await getInventory(account._id.toString(), "FocusAbility FocusUpgrades FocusXP");
|
||||
const cost = inventory.FocusAbility ? 50_000 : 0;
|
||||
inventory.FocusAbility ??= focusType;
|
||||
inventory.FocusUpgrades.push({ ItemType: focusType });
|
||||
@ -52,12 +120,29 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
});
|
||||
break;
|
||||
}
|
||||
case FocusOperation.ActivateWay: {
|
||||
case "IncreasePool": {
|
||||
const request = JSON.parse(String(req.body)) as IIncreasePoolRequest;
|
||||
const focusPolarity = focusTypeToPolarity(request.FocusType);
|
||||
const inventory = await getInventory(account._id.toString(), "FocusXP FocusCapacity");
|
||||
let cost = 0;
|
||||
for (let capacity = request.CurrentTotalCapacity; capacity != request.NewTotalCapacity; ++capacity) {
|
||||
cost += increasePoolCost[capacity - 5];
|
||||
}
|
||||
inventory.FocusXP![focusPolarity]! -= cost;
|
||||
inventory.FocusCapacity = request.NewTotalCapacity;
|
||||
await inventory.save();
|
||||
res.json({
|
||||
TotalCapacity: request.NewTotalCapacity,
|
||||
FocusPointCosts: { [focusPolarity]: cost }
|
||||
});
|
||||
break;
|
||||
}
|
||||
case "ActivateWay": {
|
||||
const focusType = (JSON.parse(String(req.body)) as IWayRequest).FocusType;
|
||||
|
||||
await Inventory.updateOne(
|
||||
{
|
||||
accountOwnerId: accountId
|
||||
accountOwnerId: account._id.toString()
|
||||
},
|
||||
{
|
||||
FocusAbility: focusType
|
||||
@ -69,13 +154,20 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
});
|
||||
break;
|
||||
}
|
||||
case FocusOperation.UnlockUpgrade: {
|
||||
case "UnlockUpgrade": {
|
||||
const request = JSON.parse(String(req.body)) as IUnlockUpgradeRequest;
|
||||
const focusPolarity = focusTypeToPolarity(request.FocusTypes[0]);
|
||||
const inventory = await getInventory(accountId);
|
||||
const inventory = await getInventory(account._id.toString());
|
||||
let cost = 0;
|
||||
for (const focusType of request.FocusTypes) {
|
||||
cost += ExportFocusUpgrades[focusType].baseFocusPointCost;
|
||||
if (focusType in ExportFocusUpgrades) {
|
||||
cost += ExportFocusUpgrades[focusType].baseFocusPointCost;
|
||||
} else if (focusType == "/Lotus/Upgrades/Focus/Power/Residual/ChannelEfficiencyFocusUpgrade") {
|
||||
// Zenurik's Inner Might (Focus 2.0)
|
||||
cost += 50_000;
|
||||
} else {
|
||||
logger.warn(`unknown focus upgrade ${focusType}, will unlock it for free`);
|
||||
}
|
||||
inventory.FocusUpgrades.push({ ItemType: focusType, Level: 0 });
|
||||
}
|
||||
inventory.FocusXP![focusPolarity]! -= cost;
|
||||
@ -86,15 +178,20 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
});
|
||||
break;
|
||||
}
|
||||
case FocusOperation.LevelUpUpgrade: {
|
||||
case "LevelUpUpgrade":
|
||||
case "UpdateUpgrade": {
|
||||
const request = JSON.parse(String(req.body)) as ILevelUpUpgradeRequest;
|
||||
const focusPolarity = focusTypeToPolarity(request.FocusInfos[0].ItemType);
|
||||
const inventory = await getInventory(accountId);
|
||||
const inventory = await getInventory(account._id.toString());
|
||||
let cost = 0;
|
||||
for (const focusUpgrade of request.FocusInfos) {
|
||||
cost += focusUpgrade.FocusXpCost;
|
||||
const focusUpgradeDb = inventory.FocusUpgrades.find(entry => entry.ItemType == focusUpgrade.ItemType)!;
|
||||
focusUpgradeDb.Level = focusUpgrade.Level;
|
||||
if (op == "UpdateUpgrade") {
|
||||
focusUpgradeDb.IsActive = focusUpgrade.IsActive;
|
||||
} else {
|
||||
focusUpgradeDb.Level = focusUpgrade.Level;
|
||||
}
|
||||
}
|
||||
inventory.FocusXP![focusPolarity]! -= cost;
|
||||
await inventory.save();
|
||||
@ -104,9 +201,9 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
});
|
||||
break;
|
||||
}
|
||||
case FocusOperation.SentTrainingAmplifier: {
|
||||
case "SentTrainingAmplifier": {
|
||||
const request = JSON.parse(String(req.body)) as ISentTrainingAmplifierRequest;
|
||||
const inventory = await getInventory(accountId);
|
||||
const inventory = await getInventory(account._id.toString());
|
||||
const inventoryChanges = addEquipment(inventory, "OperatorAmps", request.StartingWeaponType, {
|
||||
ModularParts: [
|
||||
"/Lotus/Weapons/Sentients/OperatorAmplifiers/SentTrainingAmplifier/SentAmpTrainingGrip",
|
||||
@ -119,10 +216,10 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
res.json(inventoryChanges.OperatorAmps![0]);
|
||||
break;
|
||||
}
|
||||
case FocusOperation.UnbindUpgrade: {
|
||||
case "UnbindUpgrade": {
|
||||
const request = JSON.parse(String(req.body)) as IUnbindUpgradeRequest;
|
||||
const focusPolarity = focusTypeToPolarity(request.FocusTypes[0]);
|
||||
const inventory = await getInventory(accountId);
|
||||
const inventory = await getInventory(account._id.toString());
|
||||
inventory.FocusXP![focusPolarity]! -= 750_000 * request.FocusTypes.length;
|
||||
addMiscItems(inventory, [
|
||||
{
|
||||
@ -149,7 +246,7 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
});
|
||||
break;
|
||||
}
|
||||
case FocusOperation.ConvertShard: {
|
||||
case "ConvertShard": {
|
||||
const request = JSON.parse(String(req.body)) as IConvertShardRequest;
|
||||
// Tally XP
|
||||
let xp = 0;
|
||||
@ -167,7 +264,7 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
for (const shard of request.Shards) {
|
||||
shard.ItemCount *= -1;
|
||||
}
|
||||
const inventory = await getInventory(accountId);
|
||||
const inventory = await getInventory(account._id.toString());
|
||||
const polarity = request.Polarity;
|
||||
inventory.FocusXP ??= {};
|
||||
inventory.FocusXP[polarity] ??= 0;
|
||||
@ -179,7 +276,8 @@ export const focusController: RequestHandler = async (req, res) => {
|
||||
}
|
||||
};
|
||||
|
||||
enum FocusOperation {
|
||||
// Focus 3.0
|
||||
enum Focus3Operation {
|
||||
InstallLens = "1",
|
||||
UnlockWay = "2",
|
||||
UnlockUpgrade = "3",
|
||||
@ -190,6 +288,20 @@ enum FocusOperation {
|
||||
ConvertShard = "9"
|
||||
}
|
||||
|
||||
// Focus 2.0
|
||||
enum Focus2Operation {
|
||||
InstallLens = "1",
|
||||
UnlockWay = "2",
|
||||
UnlockUpgrade = "3",
|
||||
IncreasePool = "4",
|
||||
LevelUpUpgrade = "5",
|
||||
ActivateWay = "6",
|
||||
UpdateUpgrade = "7", // used to change the IsActive state, same format as ILevelUpUpgradeRequest
|
||||
SentTrainingAmplifier = "9",
|
||||
UnbindUpgrade = "10",
|
||||
ConvertShard = "11"
|
||||
}
|
||||
|
||||
// For UnlockWay & ActivateWay
|
||||
interface IWayRequest {
|
||||
FocusType: string;
|
||||
@ -199,6 +311,13 @@ interface IUnlockUpgradeRequest {
|
||||
FocusTypes: string[];
|
||||
}
|
||||
|
||||
// Focus 2.0
|
||||
interface IIncreasePoolRequest {
|
||||
FocusType: string;
|
||||
CurrentTotalCapacity: number;
|
||||
NewTotalCapacity: number;
|
||||
}
|
||||
|
||||
interface ILevelUpUpgradeRequest {
|
||||
FocusInfos: {
|
||||
ItemType: string;
|
||||
@ -206,6 +325,7 @@ interface ILevelUpUpgradeRequest {
|
||||
IsUniversal: boolean;
|
||||
Level: number;
|
||||
IsActiveAbility: boolean;
|
||||
IsActive?: number; // Focus 2.0
|
||||
}[];
|
||||
}
|
||||
|
||||
@ -240,3 +360,19 @@ const shardValues = {
|
||||
"/Lotus/Types/Gameplay/Eidolon/Resources/SentientShards/SentientShardBrilliantItem": 25_000,
|
||||
"/Lotus/Types/Gameplay/Eidolon/Resources/SentientShards/SentientShardBrilliantTierTwoItem": 40_000
|
||||
};
|
||||
|
||||
// Starting at a capacity of 5 (Source: https://wiki.warframe.com/w/Focus_2.0)
|
||||
const increasePoolCost = [
|
||||
2576, 3099, 3638, 4190, 4755, 5331, 5918, 6514, 7120, 7734, 8357, 8988, 9626, 10271, 10923, 11582, 12247, 12918,
|
||||
13595, 14277, 14965, 15659, 16357, 17061, 17769, 18482, 19200, 19922, 20649, 21380, 22115, 22854, 23597, 24344,
|
||||
25095, 25850, 26609, 27371, 28136, 28905, 29678, 30454, 31233, 32015, 32801, 33590, 34382, 35176, 35974, 36775,
|
||||
37579, 38386, 39195, 40008, 40823, 41641, 42461, 43284, 44110, 44938, 45769, 46603, 47439, 48277, 49118, 49961,
|
||||
50807, 51655, 52505, 53357, 54212, 55069, 55929, 56790, 57654, 58520, 59388, 60258, 61130, 62005, 62881, 63759,
|
||||
64640, 65522, 66407, 67293, 68182, 69072, 69964, 70858, 71754, 72652, 73552, 74453, 75357, 76262, 77169, 78078,
|
||||
78988, 79900, 80814, 81730, 82648, 83567, 84488, 85410, 86334, 87260, 88188, 89117, 90047, 90980, 91914, 92849,
|
||||
93786, 94725, 95665, 96607, 97550, 98495, 99441, 100389, 101338, 102289, 103241, 104195, 105150, 106107, 107065,
|
||||
108024, 108985, 109948, 110911, 111877, 112843, 113811, 114780, 115751, 116723, 117696, 118671, 119647, 120624,
|
||||
121603, 122583, 123564, 124547, 125531, 126516, 127503, 128490, 129479, 130470, 131461, 132454, 133448, 134443,
|
||||
135440, 136438, 137437, 138437, 139438, 140441, 141444, 142449, 143455, 144463, 145471, 146481, 147492, 148503,
|
||||
149517
|
||||
];
|
||||
|
||||
@ -136,7 +136,7 @@ export const manageQuestsController: RequestHandler = async (req, res) => {
|
||||
|
||||
if (currentStage + 1 == questManifest.chainStages?.length) {
|
||||
logger.debug(`Trying to complete last stage with nextStage, calling completeQuest instead`);
|
||||
await completeQuest(inventory, questKey.ItemType);
|
||||
await completeQuest(inventory, questKey.ItemType, true);
|
||||
} else {
|
||||
if (run > 0) {
|
||||
questKey.Progress[currentStage + 1].c = run;
|
||||
|
||||
@ -1,16 +1,9 @@
|
||||
import type { RequestHandler } from "express";
|
||||
import type { ITunables } from "../../types/bootstrapperTypes.ts";
|
||||
|
||||
// This endpoint is specific to the OpenWF Bootstrapper: https://openwf.io/bootstrapper-manual
|
||||
|
||||
interface ITunables {
|
||||
prohibit_skip_mission_start_timer?: boolean;
|
||||
prohibit_fov_override?: boolean;
|
||||
prohibit_freecam?: boolean;
|
||||
prohibit_teleport?: boolean;
|
||||
prohibit_scripts?: boolean;
|
||||
}
|
||||
|
||||
const tunablesController: RequestHandler = (_req, res) => {
|
||||
export const tunablesController: RequestHandler = (_req, res) => {
|
||||
const tunables: ITunables = {};
|
||||
//tunables.prohibit_skip_mission_start_timer = true;
|
||||
//tunables.prohibit_fov_override = true;
|
||||
@ -19,5 +12,3 @@ const tunablesController: RequestHandler = (_req, res) => {
|
||||
//tunables.prohibit_scripts = true;
|
||||
res.json(tunables);
|
||||
};
|
||||
|
||||
export { tunablesController };
|
||||
|
||||
@ -1,5 +1,12 @@
|
||||
import type { RequestHandler } from "express";
|
||||
import { getAccountForRequest } from "../../services/loginService.ts";
|
||||
import { version_compare } from "../../helpers/inventoryHelpers.ts";
|
||||
|
||||
export const getSkuCatalogController: RequestHandler = (_req, res) => {
|
||||
res.sendFile("static/fixed_responses/getSkuCatalog.json", { root: "./" });
|
||||
export const getSkuCatalogController: RequestHandler = async (req, res) => {
|
||||
const account = await getAccountForRequest(req);
|
||||
if (!account.BuildLabel || version_compare(account.BuildLabel, "2025.10.14.16.10") >= 0) {
|
||||
res.sendFile("static/fixed_responses/getSkuCatalogU40.json", { root: "./" });
|
||||
} else {
|
||||
res.sendFile("static/fixed_responses/getSkuCatalog.json", { root: "./" });
|
||||
}
|
||||
};
|
||||
|
||||
@ -146,7 +146,8 @@ const focusUpgradeSchema = new Schema<IFocusUpgrade>(
|
||||
{
|
||||
ItemType: String,
|
||||
Level: Number,
|
||||
IsUniversal: Boolean
|
||||
IsUniversal: Boolean,
|
||||
IsActive: Number
|
||||
},
|
||||
{ _id: false }
|
||||
);
|
||||
@ -1542,6 +1543,8 @@ const inventorySchema = new Schema<IInventoryDatabase, InventoryDocumentProps>(
|
||||
FocusAbility: String,
|
||||
//The treeways of the Focus school.(Active and passive Ability)
|
||||
FocusUpgrades: [focusUpgradeSchema],
|
||||
//Focus 2.0 Pool
|
||||
FocusCapacity: Number,
|
||||
|
||||
//Achievement
|
||||
ChallengeProgress: [challengeProgressSchema],
|
||||
|
||||
@ -81,7 +81,8 @@ export const getRecipe = (uniqueName: string): IRecipe | undefined => {
|
||||
ItemCount: 600
|
||||
}
|
||||
],
|
||||
excludeFromMarket: true
|
||||
excludeFromMarket: true,
|
||||
tradable: false
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
@ -115,7 +115,11 @@ export const addQuestKey = (
|
||||
return inventory.QuestKeys[index - 1].toJSON<IQuestKeyClient>();
|
||||
};
|
||||
|
||||
export const completeQuest = async (inventory: TInventoryDatabaseDocument, questKey: string): Promise<void> => {
|
||||
export const completeQuest = async (
|
||||
inventory: TInventoryDatabaseDocument,
|
||||
questKey: string,
|
||||
sendMessages: boolean = false
|
||||
): Promise<void> => {
|
||||
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
||||
const chainStages = ExportKeys[questKey]?.chainStages;
|
||||
|
||||
@ -161,7 +165,7 @@ export const completeQuest = async (inventory: TInventoryDatabaseDocument, quest
|
||||
const stage = existingQuestKey.Progress[i];
|
||||
if (stage.c <= run) {
|
||||
stage.c = run;
|
||||
await giveKeyChainStageTriggered(inventory, { KeyChain: questKey, ChainStage: i });
|
||||
await giveKeyChainStageTriggered(inventory, { KeyChain: questKey, ChainStage: i }, sendMessages);
|
||||
await giveKeyChainMissionReward(inventory, { KeyChain: questKey, ChainStage: i });
|
||||
}
|
||||
}
|
||||
@ -327,7 +331,8 @@ export const giveKeyChainItem = async (
|
||||
export const giveKeyChainMessage = async (
|
||||
inventory: TInventoryDatabaseDocument,
|
||||
keyChainInfo: IKeyChainRequest,
|
||||
questKey: IQuestKeyDatabase
|
||||
questKey: IQuestKeyDatabase,
|
||||
sendMessage: boolean = true
|
||||
): Promise<void> => {
|
||||
const keyChainMessage = getKeyChainMessage(keyChainInfo);
|
||||
|
||||
@ -336,7 +341,12 @@ export const giveKeyChainMessage = async (
|
||||
keyChainMessage.countedAtt = [];
|
||||
}
|
||||
|
||||
await createMessage(inventory.accountOwnerId, [keyChainMessage]);
|
||||
if (sendMessage) {
|
||||
await createMessage(inventory.accountOwnerId, [keyChainMessage]);
|
||||
} else {
|
||||
if (keyChainMessage.countedAtt?.length) await addItems(inventory, keyChainMessage.countedAtt);
|
||||
if (keyChainMessage.att?.length) await addItems(inventory, keyChainMessage.att);
|
||||
}
|
||||
|
||||
updateQuestStage(inventory, keyChainInfo, { m: true });
|
||||
};
|
||||
@ -384,7 +394,8 @@ export const giveKeyChainMissionReward = async (
|
||||
|
||||
export const giveKeyChainStageTriggered = async (
|
||||
inventory: TInventoryDatabaseDocument,
|
||||
keyChainInfo: IKeyChainRequest
|
||||
keyChainInfo: IKeyChainRequest,
|
||||
sendMessage: boolean = true
|
||||
): Promise<void> => {
|
||||
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
||||
const chainStages = ExportKeys[keyChainInfo.KeyChain]?.chainStages;
|
||||
@ -396,7 +407,7 @@ export const giveKeyChainStageTriggered = async (
|
||||
}
|
||||
|
||||
if (chainStages[keyChainInfo.ChainStage].messageToSendWhenTriggered) {
|
||||
await giveKeyChainMessage(inventory, keyChainInfo, questKey);
|
||||
await giveKeyChainMessage(inventory, keyChainInfo, questKey, sendMessage);
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
@ -35,7 +35,8 @@ import type {
|
||||
IVoidTrader,
|
||||
IVoidTraderOffer,
|
||||
IWorldState,
|
||||
TCircuitGameMode
|
||||
TCircuitGameMode,
|
||||
IFlashSale
|
||||
} from "../types/worldStateTypes.ts";
|
||||
import { toMongoDate, toOid, version_compare } from "../helpers/inventoryHelpers.ts";
|
||||
import { logger } from "../utils/logger.ts";
|
||||
@ -62,7 +63,7 @@ const sortieBosses = [
|
||||
|
||||
type TSortieBoss = (typeof sortieBosses)[number];
|
||||
|
||||
const sortieBossToFaction: Record<TSortieBoss, string> = {
|
||||
const sortieBossToFaction: Record<TSortieBoss, TFaction> = {
|
||||
SORTIE_BOSS_HYENA: "FC_CORPUS",
|
||||
SORTIE_BOSS_KELA: "FC_GRINEER",
|
||||
SORTIE_BOSS_VOR: "FC_GRINEER",
|
||||
@ -375,6 +376,11 @@ export const getSortie = (day: number): ISortie => {
|
||||
const selectedNodes: ISortieMission[] = [];
|
||||
const missionTypes = new Set();
|
||||
|
||||
if (enemyFaction == "FC_INFESTATION") {
|
||||
// MT_RETRIEVAL may not be chosen for infested enemies (https://onlyg.it/OpenWF/SpaceNinjaServer/issues/2907)
|
||||
missionTypes.add("MT_RETRIEVAL");
|
||||
}
|
||||
|
||||
for (let i = 0; i < 3; i++) {
|
||||
const randomIndex = rng.randomInt(0, nodes.length - 1);
|
||||
const node = nodes[randomIndex];
|
||||
@ -1580,21 +1586,16 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
|
||||
}
|
||||
|
||||
if (config.worldState?.qtccAlerts) {
|
||||
const activationTimeStamp = "1759327200000";
|
||||
const expiryTimeStamp = "2000000000000";
|
||||
|
||||
worldState.Alerts.push(
|
||||
{
|
||||
_id: {
|
||||
$oid: "68dc23c42e9d3acfa708ff3b"
|
||||
},
|
||||
Activation: {
|
||||
$date: {
|
||||
$numberLong: "1759327200000"
|
||||
}
|
||||
},
|
||||
Expiry: {
|
||||
$date: {
|
||||
$numberLong: "2000000000000"
|
||||
}
|
||||
},
|
||||
Activation: { $date: { $numberLong: activationTimeStamp } },
|
||||
Expiry: { $date: { $numberLong: expiryTimeStamp } },
|
||||
MissionInfo: {
|
||||
location: "SolNode123",
|
||||
missionType: "MT_SURVIVAL",
|
||||
@ -1618,16 +1619,8 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
|
||||
_id: {
|
||||
$oid: "68dc2466e298b4f04206687a"
|
||||
},
|
||||
Activation: {
|
||||
$date: {
|
||||
$numberLong: "1759327200000"
|
||||
}
|
||||
},
|
||||
Expiry: {
|
||||
$date: {
|
||||
$numberLong: "2000000000000"
|
||||
}
|
||||
},
|
||||
Activation: { $date: { $numberLong: activationTimeStamp } },
|
||||
Expiry: { $date: { $numberLong: expiryTimeStamp } },
|
||||
MissionInfo: {
|
||||
location: "SolNode149",
|
||||
missionType: "MT_DEFENSE",
|
||||
@ -1651,16 +1644,8 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
|
||||
_id: {
|
||||
$oid: "68dc26865e7cb56b820b4252"
|
||||
},
|
||||
Activation: {
|
||||
$date: {
|
||||
$numberLong: "1759327200000"
|
||||
}
|
||||
},
|
||||
Expiry: {
|
||||
$date: {
|
||||
$numberLong: "2000000000000"
|
||||
}
|
||||
},
|
||||
Activation: { $date: { $numberLong: activationTimeStamp } },
|
||||
Expiry: { $date: { $numberLong: expiryTimeStamp } },
|
||||
MissionInfo: {
|
||||
location: "SolNode39",
|
||||
missionType: "MT_EXCAVATE",
|
||||
@ -1681,6 +1666,72 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
|
||||
ForceUnlock: true
|
||||
}
|
||||
);
|
||||
|
||||
const storeItems: (Partial<IFlashSale> & { TypeName: string })[] = [
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImageConquera2021D", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Operator/Tattoos/TattooTennoI", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Operator/Tattoos/TattooTennoH", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Clan/QTCC2024EmblemItem", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Conquera2024Display", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/AvatarImageConqueraGlyphVII", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/AvatarImageConqueraGlyphVI", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Interface/Graphics/CustomUI/ConqueraStyle", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Interface/Graphics/CustomUI/Backgrounds/ConqueraBackground", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Conquera2021Deco", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImageConquera2022A", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImageConquera2021B", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImageConquera2021A", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Scarves/TnCharityRibbonSyandana", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImageConquera2021C", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Venus/Conquera2023CommunityDisplay", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/AvatarImageConqueraGlyphUpdated", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImageConquera", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Sigils/QTCC2023ConqueraSigil", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Sigils/ConqueraSigil", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Effects/Conquera2022Ephemera", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Effects/ConqueraEphemera", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Armor/TnCharityRibbonArmor/ConqueraArmorL", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Armor/TnCharityRibbonArmor/ConqueraArmorA", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Armor/TnCharityRibbonArmor/ConqueraChestRibbon", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Plushies/PlushyProtectorStalker", PremiumOverride: 35 }
|
||||
];
|
||||
|
||||
worldState.FlashSales.push(
|
||||
...storeItems.map(item => ({
|
||||
...{
|
||||
StartDate: { $date: { $numberLong: activationTimeStamp } },
|
||||
EndDate: { $date: { $numberLong: expiryTimeStamp } },
|
||||
ProductExpiryOverride: { $date: { $numberLong: expiryTimeStamp } },
|
||||
ShowInMarket: item.ShowInMarket ?? true,
|
||||
HideFromMarket: item.HideFromMarket ?? false,
|
||||
SupporterPack: item.SupporterPack ?? false,
|
||||
Discount: item.Discount ?? 0,
|
||||
BogoBuy: item.BogoBuy ?? 0,
|
||||
BogoGet: item.BogoGet ?? 0,
|
||||
RegularOverride: item.RegularOverride ?? 0,
|
||||
PremiumOverride: item.PremiumOverride ?? 0
|
||||
},
|
||||
...item
|
||||
}))
|
||||
);
|
||||
|
||||
const seasonalItems = storeItems.map(item => item.TypeName);
|
||||
|
||||
const seasonalCategory = worldState.InGameMarket.LandingPage.Categories.find(
|
||||
c => c.CategoryName == "COMMUNITY"
|
||||
);
|
||||
if (seasonalCategory) {
|
||||
seasonalCategory.Items ??= [];
|
||||
seasonalCategory.Items.push(...seasonalItems);
|
||||
} else {
|
||||
worldState.InGameMarket.LandingPage.Categories.push({
|
||||
CategoryName: "COMMUNITY",
|
||||
Name: "/Lotus/Language/Store/CommunityCategoryTitle",
|
||||
Icon: "community",
|
||||
AddToMenu: true,
|
||||
Items: seasonalItems
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
const isFebruary = date.getUTCMonth() == 1;
|
||||
@ -2048,82 +2099,39 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
|
||||
NightLevel: "/Lotus/Levels/GrineerBeach/GrineerBeachEventNight.level"
|
||||
});
|
||||
|
||||
const baseStoreItem = {
|
||||
ShowInMarket: true,
|
||||
HideFromMarket: false,
|
||||
SupporterPack: false,
|
||||
Discount: 0,
|
||||
BogoBuy: 0,
|
||||
BogoGet: 0,
|
||||
StartDate: { $date: { $numberLong: activationTimeStamp } },
|
||||
EndDate: { $date: { $numberLong: expiryTimeStamp } },
|
||||
ProductExpiryOverride: { $date: { $numberLong: expiryTimeStamp } }
|
||||
};
|
||||
|
||||
const storeItems = [
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/Packages/WaterFightNoggleBundle",
|
||||
PremiumOverride: 240,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFBeastMasterBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFChargerBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFEngineerBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFGruntBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImagePopsicleGrineerPurple",
|
||||
PremiumOverride: 0,
|
||||
RegularOverride: 1
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFHealerBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFHeavyBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFHellionBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFSniperBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFTankBobbleHead",
|
||||
PremiumOverride: 35,
|
||||
RegularOverride: 0
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/SuitCustomizations/ColourPickerRollers",
|
||||
PremiumOverride: 75,
|
||||
RegularOverride: 0
|
||||
}
|
||||
const storeItems: (Partial<IFlashSale> & { TypeName: string })[] = [
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/WaterFightNoggleBundle", PremiumOverride: 240 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFBeastMasterBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFChargerBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFEngineerBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFGruntBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/ImagePopsicleGrineerPurple", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFHealerBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFHeavyBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFHellionBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFSniperBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/Events/WFTankBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/SuitCustomizations/ColourPickerRollers", PremiumOverride: 75 }
|
||||
];
|
||||
|
||||
worldState.FlashSales.push(...storeItems.map(item => ({ ...baseStoreItem, ...item })));
|
||||
worldState.FlashSales.push(
|
||||
...storeItems.map(item => ({
|
||||
...{
|
||||
StartDate: { $date: { $numberLong: activationTimeStamp } },
|
||||
EndDate: { $date: { $numberLong: expiryTimeStamp } },
|
||||
ProductExpiryOverride: { $date: { $numberLong: expiryTimeStamp } },
|
||||
ShowInMarket: item.ShowInMarket ?? true,
|
||||
HideFromMarket: item.HideFromMarket ?? false,
|
||||
SupporterPack: item.SupporterPack ?? false,
|
||||
Discount: item.Discount ?? 0,
|
||||
BogoBuy: item.BogoBuy ?? 0,
|
||||
BogoGet: item.BogoGet ?? 0,
|
||||
RegularOverride: item.RegularOverride ?? 0,
|
||||
PremiumOverride: item.PremiumOverride ?? 0
|
||||
},
|
||||
...item
|
||||
}))
|
||||
);
|
||||
|
||||
const seasonalItems = storeItems.map(item => item.TypeName);
|
||||
|
||||
@ -2762,22 +2770,18 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
|
||||
|
||||
const isOctober = date.getUTCMonth() == 9; // October = month index 9
|
||||
if (config.worldState?.naberusNightsOverride ?? isOctober) {
|
||||
const activationTimeStamp = config.worldState?.naberusNightsOverride
|
||||
? "1727881200000"
|
||||
: Date.UTC(date.getUTCFullYear(), date.getUTCMonth(), 1).toString();
|
||||
|
||||
const expiryTimeStamp = config.worldState?.naberusNightsOverride
|
||||
? "2000000000000"
|
||||
: Date.UTC(date.getUTCFullYear(), date.getUTCMonth() + 1, 1).toString();
|
||||
|
||||
worldState.Goals.push({
|
||||
_id: { $oid: "66fd602de1778d583419e8e7" },
|
||||
Activation: {
|
||||
$date: {
|
||||
$numberLong: config.worldState?.naberusNightsOverride
|
||||
? "1727881200000"
|
||||
: Date.UTC(date.getUTCFullYear(), date.getUTCMonth(), 1).toString()
|
||||
}
|
||||
},
|
||||
Expiry: {
|
||||
$date: {
|
||||
$numberLong: config.worldState?.naberusNightsOverride
|
||||
? "2000000000000"
|
||||
: Date.UTC(date.getUTCFullYear(), date.getUTCMonth() + 1, 1).toString()
|
||||
}
|
||||
},
|
||||
Activation: { $date: { $numberLong: activationTimeStamp } },
|
||||
Expiry: { $date: { $numberLong: expiryTimeStamp } },
|
||||
Count: 0,
|
||||
Goal: 0,
|
||||
Success: 0,
|
||||
@ -2788,6 +2792,181 @@ export const getWorldState = (buildLabel?: string): IWorldState => {
|
||||
Tag: "DeimosHalloween",
|
||||
Node: "DeimosHub"
|
||||
});
|
||||
|
||||
const storeItems: (Partial<IFlashSale> & { TypeName: string })[] = [
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/Halloween2023GlyphBundleA", PremiumOverride: 65 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/Halloween2021GlyphBundle", PremiumOverride: 65 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/Halloween2019GlyphBundleA", PremiumOverride: 65 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/Halloween2019GlyphBundleB", PremiumOverride: 65 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenGlyphBundle", PremiumOverride: 65 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/Halloween2023ArmorBundle", PremiumOverride: 125 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenCrpCircArmorPack", PremiumOverride: 100 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenScarfBundleB", PremiumOverride: 80 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenSkinPack", PremiumOverride: 175 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenShipSkinBundle", PremiumOverride: 80 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenSkinPackC", PremiumOverride: 175 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenSkinPackII", PremiumOverride: 145 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenScarfBundle", PremiumOverride: 130 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/AcolyteNoggleBundle", PremiumOverride: 160 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/AcolyteAreaCasterBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/AcolyteDuellistBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/AcolyteControlBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/AcolyteHeavyBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/AcolyteRogueBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/Items/ShipDecos/AcolyteStrikerBobbleHead", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/SuitCustomizations/ColourPickerHalloweenItemA", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Armor/Halloween2014Wings/Halloween2014ArmArmor", PremiumOverride: 50 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Festivities/PumpkinHead", RegularOverride: 1 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenRegorAxeShield", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019CheshireKavat",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenAkvasto", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenAngstrum", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenBoltor", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019GhostChibiWisp",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenBraton", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/AvatarImageHalloween2016A", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/AvatarImageHalloween2016C", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/AvatarImageHalloween2016B", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenBuzlok", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/AvatarImageHalloween2016D", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019CreepyClem", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenDaikyu", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageHalloween2021Dethcube",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenDragonNikana", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenDualZoren", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019FrankenCorpus",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageHalloween2021Grineer",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenGlaive", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenGalatine", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenGrakata", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenGorgon", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenGlaxion", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenTwinGremlins", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenGrinlok", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Scarves/HalloweenFireFlyScarf", PremiumOverride: 90 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenImperator", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenKronen", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageHalloween2021Lotus",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenJatKittag", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Scarves/HalloweenKyropteraScarf", PremiumOverride: 50 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenKunai", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Liset/LisetSkinHalloween", PremiumOverride: 50 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Liset/LisetInsectSkinHalloween", PremiumOverride: 50 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageChillingGlyphFour",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenMarelok", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenNikana", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenNukor", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageHalloween2021Loid",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenOpticor", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenOrthos", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenParis", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageChillingGlyphTwo",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Armor/CrpCircleArmour/HalloweenCrpCircC", PremiumOverride: 45 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Armor/CrpCircleArmour/HalloweenCrpCircA", PremiumOverride: 50 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Armor/CrpCircleArmour/HalloweenCrpCircL", PremiumOverride: 35 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenScindo", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019GhoulGrave", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageHalloween2021Pumpkin",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSarpa", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageChillingGlyphThree",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSilvaAndAegis", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/AvatarImageChillingGlyphOne",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSoma", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSkana", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019SlimeLoki", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSobek", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSonicor", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSimulor", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenTonkor", PremiumOverride: 20 },
|
||||
{
|
||||
TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019TrickOrBalas",
|
||||
PremiumOverride: 20
|
||||
},
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenSpira", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenStradavar", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenTwinGrakatas", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenArchSword", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenLato", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/AvatarImages/Seasonal/Halloween2019Werefested", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Scarves/HalloweenErosionCape", PremiumOverride: 50 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenVasto", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenDarkSplitSword", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenDarkDagger", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Scarves/HalloweenGrnBannerScarf", PremiumOverride: 75 },
|
||||
{ TypeName: "/Lotus/Upgrades/Skins/Halloween/HalloweenAmprex", PremiumOverride: 20 },
|
||||
{ TypeName: "/Lotus/Types/StoreItems/Packages/HalloweenSkinPackD", PremiumOverride: 180 }
|
||||
];
|
||||
|
||||
worldState.FlashSales.push(
|
||||
...storeItems.map(item => ({
|
||||
...{
|
||||
StartDate: { $date: { $numberLong: activationTimeStamp } },
|
||||
EndDate: { $date: { $numberLong: expiryTimeStamp } },
|
||||
ProductExpiryOverride: { $date: { $numberLong: expiryTimeStamp } },
|
||||
ShowInMarket: item.ShowInMarket ?? true,
|
||||
HideFromMarket: item.HideFromMarket ?? false,
|
||||
SupporterPack: item.SupporterPack ?? false,
|
||||
Discount: item.Discount ?? 0,
|
||||
BogoBuy: item.BogoBuy ?? 0,
|
||||
BogoGet: item.BogoGet ?? 0,
|
||||
RegularOverride: item.RegularOverride ?? 0,
|
||||
PremiumOverride: item.PremiumOverride ?? 0
|
||||
},
|
||||
...item
|
||||
}))
|
||||
);
|
||||
|
||||
const seasonalItems = storeItems.map(item => item.TypeName);
|
||||
const seasonalCategory = worldState.InGameMarket.LandingPage.Categories.find(c => c.CategoryName == "SEASONAL");
|
||||
|
||||
if (seasonalCategory) {
|
||||
seasonalCategory.Items ??= [];
|
||||
seasonalCategory.Items.push(...seasonalItems);
|
||||
} else {
|
||||
worldState.InGameMarket.LandingPage.Categories.push({
|
||||
CategoryName: "SEASONAL",
|
||||
Name: "/Lotus/Language/Store/SeasonalCategoryTitle",
|
||||
Icon: "seasonal",
|
||||
AddToMenu: true,
|
||||
Items: seasonalItems
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
if (config.worldState?.bellyOfTheBeast) {
|
||||
|
||||
@ -8,6 +8,7 @@ import type { IDatabaseAccountJson } from "../types/loginTypes.ts";
|
||||
import type { HydratedDocument } from "mongoose";
|
||||
import { logError, logger } from "../utils/logger.ts";
|
||||
import type { Request } from "express";
|
||||
import type { ITunables } from "../types/bootstrapperTypes.ts";
|
||||
|
||||
let wsServer: WebSocketServer | undefined;
|
||||
let wssServer: WebSocketServer | undefined;
|
||||
@ -89,8 +90,9 @@ interface IWsMsgToClient {
|
||||
logged_out?: boolean;
|
||||
have_game_ws?: boolean;
|
||||
|
||||
// to game
|
||||
// to game/bootstrapper (https://openwf.io/bootstrapper-manual)
|
||||
sync_inventory?: boolean;
|
||||
tunables?: ITunables;
|
||||
}
|
||||
|
||||
const wsOnConnect = (ws: WebSocket, req: http.IncomingMessage): void => {
|
||||
|
||||
9
src/types/bootstrapperTypes.ts
Normal file
9
src/types/bootstrapperTypes.ts
Normal file
@ -0,0 +1,9 @@
|
||||
// This is specific to the OpenWF Bootstrapper: https://openwf.io/bootstrapper-manual
|
||||
export interface ITunables {
|
||||
token?: string;
|
||||
prohibit_skip_mission_start_timer?: boolean;
|
||||
prohibit_fov_override?: boolean;
|
||||
prohibit_freecam?: boolean;
|
||||
prohibit_teleport?: boolean;
|
||||
prohibit_scripts?: boolean;
|
||||
}
|
||||
@ -340,6 +340,7 @@ export interface IInventoryClient extends IDailyAffiliations, InventoryClientEqu
|
||||
EmailItems: ITypeCount[];
|
||||
CompletedSyndicates: string[];
|
||||
FocusXP?: IFocusXP;
|
||||
FocusCapacity?: number;
|
||||
Wishlist: string[];
|
||||
Alignment?: IAlignment;
|
||||
CompletedSorties: string[];
|
||||
@ -638,6 +639,7 @@ export interface IFocusUpgrade {
|
||||
ItemType: string;
|
||||
Level?: number;
|
||||
IsUniversal?: boolean;
|
||||
IsActive?: number; // Focus 2.0
|
||||
}
|
||||
|
||||
export interface IFocusXP {
|
||||
|
||||
409
static/fixed_responses/getSkuCatalogU40.json
Normal file
409
static/fixed_responses/getSkuCatalogU40.json
Normal file
@ -0,0 +1,409 @@
|
||||
{
|
||||
"Skus": [
|
||||
{
|
||||
"productId": 17,
|
||||
"listPrice": {
|
||||
"formatted": "4.99 USD",
|
||||
"raw": "4.99",
|
||||
"value": 4.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "4.99 USD",
|
||||
"raw": "4.99",
|
||||
"value": 4.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 18,
|
||||
"listPrice": {
|
||||
"formatted": "9.99 USD",
|
||||
"raw": "9.99",
|
||||
"value": 9.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "9.99 USD",
|
||||
"raw": "9.99",
|
||||
"value": 9.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 160,
|
||||
"listPrice": {
|
||||
"formatted": "19.99 USD",
|
||||
"raw": "19.99",
|
||||
"value": 19.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "19.99 USD",
|
||||
"raw": "19.99",
|
||||
"value": 19.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 257,
|
||||
"listPrice": {
|
||||
"formatted": "199.99 USD",
|
||||
"raw": "199.99",
|
||||
"value": 199.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "199.99 USD",
|
||||
"raw": "199.99",
|
||||
"value": 199.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 258,
|
||||
"listPrice": {
|
||||
"formatted": "99.99 USD",
|
||||
"raw": "99.99",
|
||||
"value": 99.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "99.99 USD",
|
||||
"raw": "99.99",
|
||||
"value": 99.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 259,
|
||||
"listPrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 785,
|
||||
"listPrice": {
|
||||
"formatted": "19.99 USD",
|
||||
"raw": "19.99",
|
||||
"value": 19.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "19.99 USD",
|
||||
"raw": "19.99",
|
||||
"value": 19.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 786,
|
||||
"listPrice": {
|
||||
"formatted": "39.99 USD",
|
||||
"raw": "39.99",
|
||||
"value": 39.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "39.99 USD",
|
||||
"raw": "39.99",
|
||||
"value": 39.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 787,
|
||||
"listPrice": {
|
||||
"formatted": "79.99 USD",
|
||||
"raw": "79.99",
|
||||
"value": 79.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "79.99 USD",
|
||||
"raw": "79.99",
|
||||
"value": 79.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 979,
|
||||
"listPrice": {
|
||||
"formatted": "4.99 USD",
|
||||
"raw": "4.99",
|
||||
"value": 4.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "4.99 USD",
|
||||
"raw": "4.99",
|
||||
"value": 4.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10037,
|
||||
"listPrice": {
|
||||
"formatted": "54.99 USD",
|
||||
"raw": "54.99",
|
||||
"value": 54.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "54.99 USD",
|
||||
"raw": "54.99",
|
||||
"value": 54.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10054,
|
||||
"listPrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10061,
|
||||
"listPrice": {
|
||||
"formatted": "0.99 USD",
|
||||
"raw": "0.99",
|
||||
"value": 0.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "0.99 USD",
|
||||
"raw": "0.99",
|
||||
"value": 0.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10076,
|
||||
"listPrice": {
|
||||
"formatted": "29.99 USD",
|
||||
"raw": "29.99",
|
||||
"value": 29.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "29.99 USD",
|
||||
"raw": "29.99",
|
||||
"value": 29.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10078,
|
||||
"listPrice": {
|
||||
"formatted": "24.99 USD",
|
||||
"raw": "24.99",
|
||||
"value": 24.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "24.99 USD",
|
||||
"raw": "24.99",
|
||||
"value": 24.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10080,
|
||||
"listPrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10081,
|
||||
"listPrice": {
|
||||
"formatted": "79.99 USD",
|
||||
"raw": "79.99",
|
||||
"value": 79.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "79.99 USD",
|
||||
"raw": "79.99",
|
||||
"value": 79.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10082,
|
||||
"listPrice": {
|
||||
"formatted": "139.99 USD",
|
||||
"raw": "139.99",
|
||||
"value": 139.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "139.99 USD",
|
||||
"raw": "139.99",
|
||||
"value": 139.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10084,
|
||||
"listPrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "49.99 USD",
|
||||
"raw": "49.99",
|
||||
"value": 49.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10085,
|
||||
"listPrice": {
|
||||
"formatted": "30.00 USD",
|
||||
"raw": "30.00",
|
||||
"value": 30
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "30.00 USD",
|
||||
"raw": "30.00",
|
||||
"value": 30
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10086,
|
||||
"listPrice": {
|
||||
"formatted": "60.00 USD",
|
||||
"raw": "60.00",
|
||||
"value": 60
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "60.00 USD",
|
||||
"raw": "60.00",
|
||||
"value": 60
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10087,
|
||||
"listPrice": {
|
||||
"formatted": "90.00 USD",
|
||||
"raw": "90.00",
|
||||
"value": 90
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "90.00 USD",
|
||||
"raw": "90.00",
|
||||
"value": 90
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10088,
|
||||
"listPrice": {
|
||||
"formatted": "90.00 USD",
|
||||
"raw": "90.00",
|
||||
"value": 90
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "90.00 USD",
|
||||
"raw": "90.00",
|
||||
"value": 90
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10089,
|
||||
"listPrice": {
|
||||
"formatted": "40.01 USD",
|
||||
"raw": "40.01",
|
||||
"value": 40.01
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "40.01 USD",
|
||||
"raw": "40.01",
|
||||
"value": 40.01
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10090,
|
||||
"listPrice": {
|
||||
"formatted": "10.01 USD",
|
||||
"raw": "10.01",
|
||||
"value": 10.01
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "10.01 USD",
|
||||
"raw": "10.01",
|
||||
"value": 10.01
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10094,
|
||||
"listPrice": {
|
||||
"formatted": "24.99 USD",
|
||||
"raw": "24.99",
|
||||
"value": 24.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "24.99 USD",
|
||||
"raw": "24.99",
|
||||
"value": 24.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
},
|
||||
{
|
||||
"productId": 10095,
|
||||
"listPrice": {
|
||||
"formatted": "14.99 USD",
|
||||
"raw": "14.99",
|
||||
"value": 14.99
|
||||
},
|
||||
"basePrice": {
|
||||
"formatted": "14.99 USD",
|
||||
"raw": "14.99",
|
||||
"value": 14.99
|
||||
},
|
||||
"currencyCode": "USD",
|
||||
"owned": false
|
||||
}
|
||||
]
|
||||
}
|
||||
@ -832,7 +832,7 @@ function updateInventory() {
|
||||
const td = document.createElement("td");
|
||||
td.classList = "text-end text-nowrap";
|
||||
|
||||
let maxXP = Math.pow(itemMap[item.ItemType].maxLevelCap ?? 30, 2) * 1000;
|
||||
let maxXP = Math.pow(itemMap[item.ItemType]?.maxLevelCap ?? 30, 2) * 1000;
|
||||
if (
|
||||
category != "Suits" &&
|
||||
category != "SpaceSuits" &&
|
||||
@ -859,7 +859,7 @@ function updateInventory() {
|
||||
}
|
||||
}
|
||||
if (
|
||||
itemMap[item.ItemType].maxLevelCap > 30 &&
|
||||
itemMap[item.ItemType]?.maxLevelCap > 30 &&
|
||||
(item.Polarized ?? 0) < (itemMap[item.ItemType].maxLevelCap - 30) / 2
|
||||
) {
|
||||
const a = document.createElement("a");
|
||||
@ -3276,13 +3276,13 @@ function unlockFocusSchool(upgradeType) {
|
||||
return new Promise(resolve => {
|
||||
// Deselect current FocusAbility so we will be able to unlock the way for free
|
||||
$.post({
|
||||
url: "/api/focus.php?" + window.authz + "&op=5",
|
||||
url: "/api/focus.php?" + window.authz + "&op=ActivateWay",
|
||||
contentType: "text/plain",
|
||||
data: JSON.stringify({ FocusType: null })
|
||||
}).done(function () {
|
||||
// Unlock the way now
|
||||
$.post({
|
||||
url: "/api/focus.php?" + window.authz + "&op=2",
|
||||
url: "/api/focus.php?" + window.authz + "&op=UnlockWay",
|
||||
contentType: "text/plain",
|
||||
data: JSON.stringify({
|
||||
FocusType: upgradeType
|
||||
|
||||
@ -1,17 +1,17 @@
|
||||
// German translation by Animan8000
|
||||
dict = {
|
||||
general_inventoryUpdateNote: `Hinweis: Um Änderungen im Spiel zu sehen, musst du dein Inventar neu synchronisieren, z. B. mit dem /sync Befehl des Bootstrappers, durch Besuch eines Dojo/Relais oder durch erneutes Einloggen.`,
|
||||
general_inventoryUpdateNoteGameWs: `[UNTRANSLATED] Note: You may need to reopen any menu you are on for changes to be reflected.`,
|
||||
general_inventoryUpdateNote: `Hinweis: Um Änderungen im Spiel zu sehen, musst du dein Inventar neu synchronisieren, z. B. mit dem /sync Befehl des Bootstrappers im Spielchat, durch Besuch eines Dojo/Relais oder durch erneutes Einloggen.`,
|
||||
general_inventoryUpdateNoteGameWs: `Hinweis: Möglicherweise musst du ein Menü neu öffnen, damit die Änderungen sichtbar werden.`,
|
||||
general_addButton: `Hinzufügen`,
|
||||
general_setButton: `Festlegen`,
|
||||
general_none: `Keines`,
|
||||
general_none: `Nichts`,
|
||||
general_bulkActions: `Massenaktionen`,
|
||||
general_loading: `Lädt...`,
|
||||
|
||||
code_loginFail: `Anmeldung fehlgeschlagen. Bitte überprüfe deine Angaben.`,
|
||||
code_regFail: `Registrierung fehlgeschlagen. Account existiert bereits?`,
|
||||
code_changeNameConfirm: `In welchen Namen möchtest du deinen Account umbenennen?`,
|
||||
code_changeNameRetry: `[UNTRANSLATED] |NAME| is already taken.`,
|
||||
code_changeNameRetry: `|NAME| ist bereits vergeben.`,
|
||||
code_deleteAccountConfirm: `Bist du sicher, dass du deinen Account |DISPLAYNAME| (|EMAIL|) löschen möchtest? Diese Aktion kann nicht rückgängig gemacht werden.`,
|
||||
code_archgun: `Arch-Gewehr`,
|
||||
code_melee: `Nahkampf`,
|
||||
@ -32,8 +32,8 @@ dict = {
|
||||
code_renamePrompt: `Neuen benutzerdefinierten Namen eingeben:`,
|
||||
code_remove: `Entfernen`,
|
||||
code_addItemsConfirm: `Bist du sicher, dass du |COUNT| Gegenstände zu deinem Account hinzufügen möchtest?`,
|
||||
code_addTechProjectsConfirm: `[UNTRANSLATED] Are you sure you want to add |COUNT| research to your clan?`,
|
||||
code_addVaultItemsConfirm: `[UNTRANSLATED] Are you sure you want to add |COUNT| items to your clan vault?`,
|
||||
code_addTechProjectsConfirm: `Bist du sicher, dass du |COUNT| Forschungen zu deinem Clan hinzufügen möchtest?`,
|
||||
code_addVaultItemsConfirm: `Bist du sicher, dass du |COUNT| Gegenstände zu deinem Clan-Tresor hinzufügen möchtest?`,
|
||||
code_succRankUp: `Erfolgreich aufgestiegen.`,
|
||||
code_noEquipmentToRankUp: `Keine Ausstattung zum Rangaufstieg verfügbar.`,
|
||||
code_succAdded: `Erfolgreich hinzugefügt.`,
|
||||
@ -45,7 +45,7 @@ dict = {
|
||||
code_rank: `Rang`,
|
||||
code_rankUp: `Rang erhöhen`,
|
||||
code_rankDown: `Rang verringern`,
|
||||
code_unlockLevelCap: `[UNTRANSLATED] Unlock level cap`,
|
||||
code_unlockLevelCap: `Level-Cap freischalten`,
|
||||
code_count: `Anzahl`,
|
||||
code_focusAllUnlocked: `Alle Fokus-Schulen sind bereits freigeschaltet.`,
|
||||
code_focusUnlocked: `|COUNT| neue Fokus-Schulen freigeschaltet! Ein Inventar-Update wird benötigt, damit die Änderungen im Spiel sichtbar werden.`,
|
||||
@ -65,22 +65,22 @@ 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_controller: `Controller-Cursor`,
|
||||
code_mouseLine: `Linien-Cursor`,
|
||||
code_mouse: `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_fund: `Spenden`,
|
||||
code_funded: `Gespendet`,
|
||||
code_replays: `Wiederholungen`,
|
||||
code_stalker: `Stalker`,
|
||||
code_cutName: `Frisur: |INDEX|`,
|
||||
code_drifterBeardName: `Drifter-Bart: |INDEX|`,
|
||||
code_drifterFaceName: `Drifter-Gesicht: |INDEX|`,
|
||||
code_operatorFaceName: `Operator-Gesicht: |INDEX|`,
|
||||
code_succChange: `Erfolgreich geändert.`,
|
||||
code_requiredInvigorationUpgrade: `[UNTRANSLATED] You must select both an offensive & utility upgrade.`,
|
||||
code_requiredInvigorationUpgrade: `Du musst sowohl ein Offensiv- als auch ein Support-Upgrade auswählen.`,
|
||||
login_description: `Melde dich mit deinem OpenWF-Account an (denselben Angaben wie im Spiel, wenn du dich mit diesem Server verbindest).`,
|
||||
login_emailLabel: `E-Mail-Adresse`,
|
||||
login_passwordLabel: `Passwort`,
|
||||
@ -96,8 +96,8 @@ dict = {
|
||||
navbar_cheats: `Cheats`,
|
||||
navbar_import: `Importieren`,
|
||||
inventory_addItems: `Gegenstände hinzufügen`,
|
||||
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: `Roh`,
|
||||
inventory_addItemByItemType_warning: `Verwende diese Funktion auf eigene Gefahr. Sie kann dein Inventar beschädigen und du musst Gegenstände manuell entfernen, falls etwas schiefgeht.`,
|
||||
inventory_suits: `Warframes`,
|
||||
inventory_longGuns: `Primärwaffen`,
|
||||
inventory_pistols: `Sekundärwaffen`,
|
||||
@ -123,10 +123,10 @@ dict = {
|
||||
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_bulkAddFlavourItems: `Fehlende Sammlerstücke hinzufügen`,
|
||||
inventory_bulkAddShipDecorations: `Fehlende Schiffsdekorationen hinzufügen`,
|
||||
inventory_bulkAddEvolutionProgress: `Fehlende Incarnon-Entwicklungsfortschritte hinzufügen`,
|
||||
inventory_bulkAddWeaponSkins: `[UNTRANSLATED] Add Missing Skins`,
|
||||
inventory_bulkAddWeaponSkins: `Fehlende Skins hinzufügen`,
|
||||
inventory_bulkRankUpSuits: `Alle Warframes auf Max. Rang`,
|
||||
inventory_bulkRankUpWeapons: `Alle Waffen auf Max. Rang`,
|
||||
inventory_bulkRankUpSpaceSuits: `Alle Archwings auf Max. Rang`,
|
||||
@ -179,8 +179,8 @@ dict = {
|
||||
invigorations_utility_EnergyRegen: `+2 Energieregeneration pro Sekunde`,
|
||||
|
||||
detailedView_invigorationOffensiveLabel: `Offensives Upgrade`,
|
||||
detailedView_invigorationUtilityLabel: `[UNTRANSLATED] Utility Upgrade`,
|
||||
detailedView_invigorationExpiryLabel: `[UNTRANSLATED] Invigoration Expiry (optional)`,
|
||||
detailedView_invigorationUtilityLabel: `Support Upgrade`,
|
||||
detailedView_invigorationExpiryLabel: `Kräftigungs-Ablaufdatum (optional)`,
|
||||
|
||||
abilityOverride_label: `Fähigkeitsüberschreibung`,
|
||||
abilityOverride_onSlot: `auf Slot`,
|
||||
@ -199,7 +199,7 @@ dict = {
|
||||
cheats_skipTutorial: `Tutorial überspringen`,
|
||||
cheats_skipAllDialogue: `Alle Dialoge überspringen`,
|
||||
cheats_unlockAllScans: `Alle Scans freischalten`,
|
||||
cheats_unlockSuccRelog: `[UNTRANSLATED] Success. Please note that you'll need to relog for the client to refresh this.`,
|
||||
cheats_unlockSuccRelog: `Erfolgreich. Bitte beachte, dass du dich neu einloggen musst, damit der Client dies aktualisiert.`,
|
||||
cheats_unlockAllMissions: `Alle Missionen freischalten`,
|
||||
cheats_unlockAllMissions_ok: `Erfolgreich. Bitte beachte, dass du ein Dojo/Relais besuchen oder dich neu einloggen musst, damit die Sternenkarte aktualisiert wird.`,
|
||||
cheats_infiniteCredits: `Unendlich Credits`,
|
||||
@ -233,7 +233,7 @@ dict = {
|
||||
cheats_baroFullyStocked: `Baro hat volles Inventar`,
|
||||
cheats_syndicateMissionsRepeatable: `Syndikat-Missionen wiederholbar`,
|
||||
cheats_unlockAllProfitTakerStages: `Alle Profiteintreiber-Phasen freischalten`,
|
||||
cheats_unlockSuccInventory: `[UNTRANSLATED] Success. Please note that you'll need to resync your inventory, e.g. using the bootstrapper's /sync command in game chat, visiting a dojo/relay, or relogging.`,
|
||||
cheats_unlockSuccInventory: `Erfolgreich. Bitte beachte, dass du dein Inventar neu synchronisieren musst, z. B. mit dem /sync Befehl des Bootstrappers im Spielchat, durch Besuch eines Dojo/Relais oder durch erneutes Einloggen.`,
|
||||
cheats_instantFinishRivenChallenge: `Riven-Mod Herausforderung sofort abschließen`,
|
||||
cheats_instantResourceExtractorDrones: `Sofortige Ressourcen-Extraktor-Drohnen`,
|
||||
cheats_noResourceExtractorDronesDamage: `Kein Schaden für Ressourcen-Extraktor-Drohnen`,
|
||||
@ -262,49 +262,49 @@ dict = {
|
||||
cheats_changeSupportedSyndicate: `Unterstütztes Syndikat`,
|
||||
cheats_changeButton: `Ändern`,
|
||||
cheats_markAllAsRead: `Posteingang als gelesen markieren`,
|
||||
cheats_finishInvasionsInOneMission: `[UNTRANSLATED] Finish Invasions in One Mission`,
|
||||
cheats_nemesisHenchmenKillsMultiplierGrineer: `[UNTRANSLATED] Rage Progess Multiplier (Grineer)`,
|
||||
cheats_nemesisHenchmenKillsMultiplierCorpus: `[UNTRANSLATED] Rage Progess Multiplier (Corpus)`,
|
||||
cheats_nemesisAntivirusGainMultiplier: `[UNTRANSLATED] Antivirus Progress Multiplier`,
|
||||
cheats_nemesisHintProgressMultiplierGrineer: `[UNTRANSLATED] Hint Progress Multiplier (Grineer)`,
|
||||
cheats_nemesisHintProgressMultiplierCorpus: `[UNTRANSLATED] Hint Progress Multiplier (Corpus)`,
|
||||
cheats_nemesisExtraWeapon: `[UNTRANSLATED] Extra Nemesis Weapon / Token On Vanquish (0 to disable)`,
|
||||
cheats_finishInvasionsInOneMission: `Invasionen in einer Mission abschließen`,
|
||||
cheats_nemesisHenchmenKillsMultiplierGrineer: `Wut-Fortschrittsmultiplikator (Grineer)`,
|
||||
cheats_nemesisHenchmenKillsMultiplierCorpus: `Wut-Fortschrittsmultiplikator (Corpus)`,
|
||||
cheats_nemesisAntivirusGainMultiplier: `Antivirus-Fortschrittsmultiplikator`,
|
||||
cheats_nemesisHintProgressMultiplierGrineer: `Hinweis-Fortschrittsmultiplikator (Grineer)`,
|
||||
cheats_nemesisHintProgressMultiplierCorpus: `Hinweis-Fortschrittsmultiplikator (Corpus)`,
|
||||
cheats_nemesisExtraWeapon: `Zusätzliche Nemesis-Waffe/-Symbol bei Besiegung (0 zum deaktivieren)`,
|
||||
|
||||
worldState: `Weltstatus`,
|
||||
worldState_creditBoost: `Event Booster: Credit`,
|
||||
worldState_affinityBoost: `Event Booster: Erfahrung`,
|
||||
worldState_resourceBoost: `Event Booster: Ressourcen`,
|
||||
worldState_tennoLiveRelay: `TennoLive Relais`,
|
||||
worldState_baroTennoConRelay: `Baros TennoCon Relais`,
|
||||
worldState_baroTennoConRelay: `Baros TennoCon-Relais`,
|
||||
worldState_starDays: `Sternen-Tage`,
|
||||
worldState_galleonOfGhouls: `Galeone der Ghule`,
|
||||
worldState_anniversary: `[UNTRANSLATED] Warframe Anniversary`,
|
||||
worldState_useAnniversaryTagForOldGoals: `[UNTRANSLATED] Use <code>Tag</code> from Warframe Anniversary for old Events`,
|
||||
worldState_anniversary: `Warframe Jubiläum`,
|
||||
worldState_useAnniversaryTagForOldGoals: `Verwende <code>Tag</code> vom Warframe Jubiläum für alte Events`,
|
||||
worldState_ghoulEmergence: `Ghul Ausrottung`,
|
||||
worldState_plagueStar: `Plagenstern`,
|
||||
worldState_dogDays: `Hitzefrei`,
|
||||
worldState_dogDaysRewards: `[UNTRANSLATED] Dog Days Rewards`,
|
||||
worldState_dogDaysRewards: `Hitzefrei-Belohnungen`,
|
||||
worldState_wolfHunt: `Wolfsjagd`,
|
||||
worldState_voidCorruption: `Void-Korruption (|VAL|)`,
|
||||
worldState_orphixVenom: `Orphix Gift`,
|
||||
worldState_longShadow: `Lange Schatten`,
|
||||
worldState_hallowedFlame: `Geweihte Flamme`,
|
||||
worldState_hallowedNightmares: `Geweihte Albträume`,
|
||||
worldState_hallowedNightmaresRewards: `[UNTRANSLATED] Hallowed Nightmares Rewards`,
|
||||
worldState_naberusNights: `[UNTRANSLATED] Nights of Naberus`,
|
||||
worldState_hallowedNightmaresRewards: `Geweihte Albträume-Belohnungen`,
|
||||
worldState_naberusNights: `Naberus Nacht`,
|
||||
worldState_proxyRebellion: `Proxy-Rebellion`,
|
||||
worldState_proxyRebellionRewards: `[UNTRANSLATED] Proxy Rebellion Rewards`,
|
||||
worldState_proxyRebellionRewards: `Proxy-Rebellion-Belohnungen`,
|
||||
worldState_bellyOfTheBeast: `Das Innere der Bestie`,
|
||||
worldState_bellyOfTheBeastProgressOverride: `[UNTRANSLATED] Belly of the Beast Progress`,
|
||||
worldState_bellyOfTheBeastProgressOverride: `Das Innere der Bestie-Fortschritt`,
|
||||
worldState_eightClaw: `Acht Klauen`,
|
||||
worldState_eightClawProgressOverride: `[UNTRANSLATED] Eight Claw Progress`,
|
||||
worldState_eightClawProgressOverride: `Acht Klauen-Fortschritt`,
|
||||
worldState_thermiaFractures: `Thermische Risse`,
|
||||
worldState_thermiaFracturesProgressOverride: `[UNTRANSLATED] Thermia Fractures Progress`,
|
||||
worldState_qtccAlerts: `[UNTRANSLATED] Quest to Conquer Cancer Alerts`,
|
||||
worldState_from_year: `[UNTRANSLATED] From |VAL|`,
|
||||
worldState_pre_year: `[UNTRANSLATED] Pre-|VAL|`,
|
||||
worldState_week: `[UNTRANSLATED] Week |VAL|`,
|
||||
worldState_incompatibleWith: `[UNTRANSLATED] Incompatible with:`,
|
||||
worldState_thermiaFracturesProgressOverride: `Thermische Risse-Fortschritt`,
|
||||
worldState_qtccAlerts: `Quest zum Kampf gegen Krebs-Alarmierungen`,
|
||||
worldState_from_year: `Von |VAL|`,
|
||||
worldState_pre_year: `Vor |VAL|`,
|
||||
worldState_week: `Woche |VAL|`,
|
||||
worldState_incompatibleWith: `Inkompatibel mit:`,
|
||||
enabled: `Aktiviert`,
|
||||
disabled: `Deaktiviert`,
|
||||
worldState_we1: `Wochenende 1`,
|
||||
@ -348,7 +348,7 @@ dict = {
|
||||
worldState_varziaFullyStocked: `Varzia hat volles Inventar`,
|
||||
worldState_varziaOverride: `Varzia-Angebotsüberschreibung`,
|
||||
|
||||
import_importNote: `[UNTRANSLATED] You can provide a full or partial <code>inventory.php</code> or <code>getShip.php</code> response (client representation) here.`,
|
||||
import_importNote: `Du kannst hier eine vollständige oder teilweise <code>inventory.php</code> oder <code>getShip.php</code> Antwort (Client-Darstellung) einfügen.`,
|
||||
import_importNote2: `Alle Felder, die vom Importer unterstützt werden, <b>werden in deinem Account überschrieben</b>.`,
|
||||
import_submit: `Absenden`,
|
||||
import_samples: `Beispiele:`,
|
||||
@ -416,11 +416,11 @@ dict = {
|
||||
theme_dark: `Dunkles Design`,
|
||||
theme_light: `Helles Design`,
|
||||
|
||||
guildView_cheats: `[UNTRANSLATED] Clan Cheats`,
|
||||
guildView_cheats: `Clan-Cheats`,
|
||||
guildView_techProjects: `Forschung`,
|
||||
guildView_vaultDecoRecipes: `[UNTRANSLATED] Dojo Deco Recipes`,
|
||||
guildView_vaultMiscItems: `[UNTRANSLATED] Resources in Vault`,
|
||||
guildView_vaultShipDecorations: `[UNTRANSLATED] Decorations in Vault`,
|
||||
guildView_vaultDecoRecipes: `Dojo-Deko-Baupläne`,
|
||||
guildView_vaultMiscItems: `Ressourcen im Tresor`,
|
||||
guildView_vaultShipDecorations: `Dekorationen im Tresor`,
|
||||
guildView_alliance: `Allianz`,
|
||||
guildView_members: `Mitglieder`,
|
||||
guildView_pending: `Ausstehend`,
|
||||
@ -440,12 +440,12 @@ dict = {
|
||||
guildView_rank_soldier: `Soldat`,
|
||||
guildView_rank_utility: `Versorger`,
|
||||
guildView_rank_warlord: `Kriegsherr`,
|
||||
guildView_currency_owned: `[UNTRANSLATED] |COUNT| in Vault.`,
|
||||
guildView_bulkAddTechProjects: `[UNTRANSLATED] Add Missing Research`,
|
||||
guildView_bulkAddVaultDecoRecipes: `[UNTRANSLATED] Add Missing Dojo Deco Recipes`,
|
||||
guildView_bulkAddVaultShipDecorations: `[UNTRANSLATED] Add Missing Decorations in Vault`,
|
||||
guildView_bulkFundTechProjects: `[UNTRANSLATED] Fund All Research`,
|
||||
guildView_bulkCompleteTechProjects: `[UNTRANSLATED] Complete All Research`,
|
||||
guildView_currency_owned: `|COUNT| im Tresor.`,
|
||||
guildView_bulkAddTechProjects: `Fehlende Forschungen hinzufügen`,
|
||||
guildView_bulkAddVaultDecoRecipes: `Fehlende Dojo-Deko-Baupläne hinzufügen`,
|
||||
guildView_bulkAddVaultShipDecorations: `Fehlende Dekorationen im Tresor hinzufügen`,
|
||||
guildView_bulkFundTechProjects: `Alle Forschungen spenden`,
|
||||
guildView_bulkCompleteTechProjects: `Alle Forschungen abschließen`,
|
||||
guildView_promote: `Befördern`,
|
||||
guildView_demote: `Degradieren`,
|
||||
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user