Files
prestonbane/src/engine/gameManager/ForgeManager.java
T

393 lines
15 KiB
Java
Raw Normal View History

2024-04-05 20:13:16 -04:00
// • ▌ ▄ ·. ▄▄▄· ▄▄ • ▪ ▄▄· ▄▄▄▄· ▄▄▄· ▐▄▄▄ ▄▄▄ .
// ·██ ▐███▪▐█ ▀█ ▐█ ▀ ▪██ ▐█ ▌▪▐█ ▀█▪▐█ ▀█ •█▌ ▐█▐▌·
// ▐█ ▌▐▌▐█·▄█▀▀█ ▄█ ▀█▄▐█·██ ▄▄▐█▀▀█▄▄█▀▀█ ▐█▐ ▐▌▐▀▀▀
// ██ ██▌▐█▌▐█ ▪▐▌▐█▄▪▐█▐█▌▐███▌██▄▪▐█▐█ ▪▐▌██▐ █▌▐█▄▄▌
// ▀▀ █▪▀▀▀ ▀ ▀ ·▀▀▀▀ ▀▀▀·▀▀▀ ·▀▀▀▀ ▀ ▀ ▀▀ █▪ ▀▀▀
// Magicbane Emulator Project © 2013 - 2022
// www.magicbane.com
package engine.gameManager;
2024-04-20 09:34:49 -04:00
import engine.loot.ModTableEntry;
2024-04-14 13:51:37 -04:00
import engine.loot.ModTypeTableEntry;
2024-04-05 20:13:16 -04:00
import engine.loot.WorkOrder;
2024-04-10 16:06:09 -04:00
import engine.mbEnums;
2024-04-14 15:38:24 -04:00
import engine.net.DispatchMessage;
import engine.net.client.msg.ItemProductionMsg;
import engine.objects.*;
2024-04-10 16:47:02 -04:00
import engine.powers.EffectsBase;
2024-04-07 22:28:07 -04:00
import org.pmw.tinylog.Logger;
2024-04-05 20:13:16 -04:00
2024-04-14 14:48:59 -04:00
import java.util.ArrayList;
2024-04-08 04:17:31 -04:00
import java.util.HashMap;
2024-04-05 20:13:16 -04:00
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.DelayQueue;
2024-04-14 13:51:37 -04:00
import java.util.concurrent.ThreadLocalRandom;
2024-04-13 10:06:44 -04:00
import java.util.concurrent.TimeUnit;
2024-04-05 20:13:16 -04:00
import java.util.concurrent.atomic.AtomicInteger;
public enum ForgeManager implements Runnable {
FORGE_MANAGER;
2024-04-13 11:28:48 -04:00
private static final BlockingQueue<WorkOrder> forge = new DelayQueue<>();
2024-04-06 07:26:17 -04:00
public static final AtomicInteger wordOrderCounter = new AtomicInteger(0);
2024-04-14 14:48:59 -04:00
public static final HashMap<NPC, ArrayList<WorkOrder>> vendorWorkOrderLookup = new HashMap<>();
2024-04-15 14:45:34 -04:00
public static final HashMap<Item, WorkOrder> itemWorkOrderLookup = new HashMap<>();
2024-04-05 20:13:16 -04:00
@Override
2024-04-14 14:48:59 -04:00
2024-04-05 20:13:16 -04:00
public void run() {
2024-04-14 15:02:52 -04:00
WorkOrder workOrder = null;
2024-04-05 20:13:16 -04:00
2024-04-14 15:02:52 -04:00
while (true) {
2024-04-14 12:13:51 -04:00
2024-04-05 20:13:16 -04:00
try {
2024-04-14 12:13:51 -04:00
workOrder = forge.take();
2024-04-14 12:14:17 -04:00
} catch (Exception e) {
2024-04-14 12:13:51 -04:00
Logger.error(e);
}
2024-04-21 14:10:35 -04:00
// Do not process null workOrders
if (workOrder == null)
continue;
2024-04-15 11:01:53 -04:00
if (workOrder.total_produced >= workOrder.total_to_produce) {
2024-04-14 17:02:36 -04:00
2024-04-15 11:30:56 -04:00
// Complete this workOrder.
2024-04-15 11:07:26 -04:00
2024-04-16 15:17:15 -04:00
for (Item workOrderItem : workOrder.cooking) {
2024-04-19 10:49:19 -04:00
workOrderItem.flags.add(mbEnums.ItemFlags.Identified);
2024-04-16 15:17:15 -04:00
ItemProductionMsg outMsg = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, workOrderItem, mbEnums.ProductionActionType.CONFIRM_PRODUCE, true);
DispatchMessage.dispatchMsgToInterestArea(workOrder.vendor, outMsg, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
2024-04-16 15:40:10 -04:00
}
2024-04-13 08:51:02 -04:00
2024-04-14 14:11:22 -04:00
workOrder.runCompleted = true;
2024-04-15 11:30:56 -04:00
}
2024-04-16 15:40:10 -04:00
if (workOrder.runCompleted)
2024-04-14 14:11:22 -04:00
continue;
2024-04-15 11:30:56 -04:00
2024-04-17 11:56:23 -04:00
// Move current cooking batch to vendor inventory
2024-04-15 11:30:56 -04:00
2024-04-17 11:56:23 -04:00
completeWorkOrderBatch(workOrder);
2024-04-10 18:01:27 -04:00
2024-04-16 15:34:39 -04:00
// Create new set of in-memory only virtual items
2024-04-14 16:29:17 -04:00
2024-04-20 12:37:55 -04:00
forgeWorkOrderBatch(workOrder);
2024-04-14 16:29:17 -04:00
2024-04-14 14:11:22 -04:00
// enQueue this workOrder again; back into the oven
// until all items for this workOrder are completed.
2024-04-05 20:13:16 -04:00
2024-04-14 14:11:22 -04:00
forge.add(workOrder);
2024-04-17 11:56:23 -04:00
// Debugging
2024-04-14 14:11:22 -04:00
Logger.info(workOrder.toString());
2024-04-05 20:13:16 -04:00
}
2024-04-07 22:15:06 -04:00
}
2024-04-05 20:13:16 -04:00
2024-04-07 22:53:41 -04:00
public static void start() {
2024-04-08 16:26:29 -04:00
Thread forgeManager;
forgeManager = new Thread(FORGE_MANAGER);
forgeManager.setName("Forge Manager");
forgeManager.start();
2024-04-07 22:53:41 -04:00
}
2024-04-14 12:08:51 -04:00
public static int submit(WorkOrder workOrder) {
2024-04-20 13:26:44 -04:00
// Make sure vendor can roll the formulae, warehouse can
// afford this wordOrder and other related checks.
2024-04-20 13:21:53 -04:00
2024-04-20 12:33:55 -04:00
int validation_result = ItemManager.validate(workOrder);
2024-04-14 12:08:51 -04:00
2024-04-20 13:21:53 -04:00
// The return code is used by the submitter as a
2024-04-20 13:26:44 -04:00
// popup error message for the player.
2024-04-20 13:21:53 -04:00
2024-04-14 12:08:51 -04:00
if (validation_result != 0)
return validation_result;
2024-04-08 17:42:47 -04:00
2024-04-21 14:08:27 -04:00
// Concurrency is managed by same lock as warehouse
City city = workOrder.vendor.building.getCity();
2024-04-21 14:08:27 -04:00
if (city == null)
return 58; //58: The formula is beyond the means of this facility
city.transactionLock.writeLock().lock();
try {
2024-04-21 11:40:27 -04:00
// Configure this production run.
2024-04-20 13:21:53 -04:00
2024-04-21 11:40:27 -04:00
workOrder.workOrderID = wordOrderCounter.incrementAndGet();
workOrder.rollingDuration = ForgeManager.calcRollingDuration(workOrder);
workOrder.completionTime = System.currentTimeMillis() + workOrder.rollingDuration;
workOrder.slots_used = calcAvailableSlots(workOrder);
2024-04-07 22:31:58 -04:00
2024-04-21 11:40:27 -04:00
workOrder.total_produced = 0;
workOrder.total_to_produce *= workOrder.slots_used;
workOrder.production_cost = calcProductionCost(workOrder);
2024-04-10 16:47:02 -04:00
2024-04-21 11:40:27 -04:00
// Set total cost for this production run
2024-04-11 13:46:24 -04:00
2024-04-21 11:40:27 -04:00
workOrder.production_cost_total.putAll(workOrder.production_cost);
workOrder.production_cost_total.forEach((key, value) -> workOrder.production_cost_total.compute(key, (k, v) -> v * workOrder.total_to_produce));
2024-04-10 16:47:02 -04:00
2024-04-21 11:40:27 -04:00
// Deduct gold cost from building
2024-04-19 08:15:17 -04:00
2024-04-21 11:40:27 -04:00
if (!debitWorkOrderCost(workOrder))
return 58; //58: The formula is beyond the means of this facility
2024-04-20 13:01:21 -04:00
2024-04-21 11:40:27 -04:00
// Create in-memory items and add to collections
2024-04-20 13:01:21 -04:00
2024-04-21 11:40:27 -04:00
forgeWorkOrderBatch(workOrder);
2024-04-20 13:01:21 -04:00
2024-04-21 11:40:27 -04:00
// Submit workOrder for next completion cycle
2024-04-20 13:01:21 -04:00
vendorWorkOrderLookup.get(workOrder.vendor).add(workOrder);
2024-04-21 11:40:27 -04:00
forge.add(workOrder);
2024-04-20 13:01:21 -04:00
} catch (Exception e) {
Logger.error(e);
} finally {
city.transactionLock.writeLock().unlock();
}
2024-04-20 13:01:21 -04:00
return validation_result;
}
2024-04-07 23:21:35 -04:00
public static long calcRollingDuration(WorkOrder workOrder) {
2024-04-12 16:43:22 -04:00
float rollingDuration;
2024-04-12 16:52:43 -04:00
rollingDuration = workOrder.vendor.getBuilding().getRank() * -5L + 40;
2024-04-13 10:06:44 -04:00
rollingDuration = TimeUnit.MINUTES.toMillis((long) rollingDuration);
2024-04-12 16:43:22 -04:00
rollingDuration *= Float.parseFloat(ConfigManager.MB_PRODUCTION_RATE.getValue());
2024-04-10 17:22:45 -04:00
ItemTemplate template = ItemTemplate.templates.get(workOrder.templateID);
2024-04-07 23:21:35 -04:00
// Bane circles
2024-04-10 17:22:45 -04:00
if (template.item_bane_rank > 0)
2024-04-12 16:43:22 -04:00
rollingDuration = (long) template.item_bane_rank * 60 * 60 * 3 * 1000 * Float.parseFloat(ConfigManager.MB_PRODUCTION_RATE.getValue());
2024-04-07 23:21:35 -04:00
2024-04-12 16:43:22 -04:00
return (long) rollingDuration;
2024-04-07 23:21:35 -04:00
}
2024-04-08 13:06:22 -04:00
2024-04-08 17:40:38 -04:00
public static int calcAvailableSlots(WorkOrder workOrder) {
2024-04-08 13:06:22 -04:00
2024-04-10 17:22:45 -04:00
// Slots available in a forge are based on the npc rank
2024-04-08 13:06:22 -04:00
int availableSlots = workOrder.vendor.getRank();
2024-04-20 13:26:44 -04:00
// Subtract slots currently being used by npc workOrders
2024-04-10 17:22:45 -04:00
2024-04-14 14:48:59 -04:00
for (WorkOrder npcWorkOrder : ForgeManager.vendorWorkOrderLookup.get(workOrder.vendor))
2024-04-13 07:54:26 -04:00
availableSlots = availableSlots - npcWorkOrder.slots_used;
2024-04-08 13:06:22 -04:00
2024-04-20 13:26:44 -04:00
// Single item rolls are always a single slot
2024-04-10 17:22:45 -04:00
2024-04-16 15:40:10 -04:00
if (availableSlots > 0 && !workOrder.multiple_slot_request)
2024-04-10 17:22:45 -04:00
availableSlots = 1;
2024-04-08 13:06:22 -04:00
return availableSlots;
}
2024-04-10 16:06:09 -04:00
public static HashMap<mbEnums.ResourceType, Integer> calcProductionCost(WorkOrder workOrder) {
2024-04-10 16:47:02 -04:00
2024-04-10 17:22:45 -04:00
// Calculate the production cost for a single run of this workOrder
2024-04-10 16:06:09 -04:00
HashMap<mbEnums.ResourceType, Integer> production_cost = new HashMap<>();
2024-04-10 16:47:02 -04:00
ItemTemplate template = ItemTemplate.templates.get(workOrder.templateID);
// Add gold and resource costs from template
production_cost.put(mbEnums.ResourceType.GOLD, template.item_value);
production_cost.putAll(template.item_resource_cost);
// Calculate cost of prefix and suffix
if (workOrder.prefixToken != 0) {
EffectsBase prefix = PowersManager.getEffectByToken(workOrder.prefixToken);
EffectsBase prefixValue = PowersManager.getEffectByIDString(prefix.getIDString() + 'A');
production_cost.putAll(prefixValue.getResourcesForEffect());
}
if (workOrder.suffixToken != 0) {
EffectsBase suffix = PowersManager.getEffectByToken(workOrder.suffixToken);
EffectsBase suffixValue = PowersManager.getEffectByIDString(suffix.getIDString() + 'A');
production_cost.putAll(suffixValue.getResourcesForEffect());
}
2024-04-10 16:06:09 -04:00
return production_cost;
}
2024-04-12 17:29:52 -04:00
2024-04-13 08:10:59 -04:00
public static Item forgeItem(WorkOrder workOrder) {
2024-04-16 15:33:39 -04:00
// Create new item from specified template
2024-04-14 13:51:37 -04:00
2024-04-16 15:33:39 -04:00
ItemTemplate template = ItemTemplate.templates.get(workOrder.templateID);
2024-04-13 08:10:59 -04:00
Item forgedItem = new Item(workOrder.templateID);
2024-04-16 15:33:39 -04:00
// forgedItem gets a negative id; a virtual item which is not persisted
2024-04-16 15:33:39 -04:00
2024-04-19 10:32:36 -04:00
forgedItem.objectUUID = ItemManager.lastNegativeID.getAndDecrement();
2024-04-13 08:10:59 -04:00
forgedItem.containerType = mbEnums.ItemContainerType.FORGE;
forgedItem.ownerID = workOrder.vendor.getObjectUUID();
2024-04-16 15:33:39 -04:00
// The UpgradeDate for the item is serialized for the
// vendor forge window
2024-04-13 08:51:02 -04:00
forgedItem.setDateToUpgrade(workOrder.completionTime);
2024-04-13 08:10:59 -04:00
2024-04-14 13:51:37 -04:00
// Give prefix and suffix to this item if random rolled
if (workOrder.prefixToken == 0)
forgedItem.prefixToken = calcRandomMod(workOrder.vendor, mbEnums.ItemModType.PREFIX, template.modTable);
2024-04-14 15:19:55 -04:00
else
forgedItem.prefixToken = workOrder.prefixToken;
2024-04-14 13:51:37 -04:00
if (workOrder.suffixToken == 0)
forgedItem.suffixToken = calcRandomMod(workOrder.vendor, mbEnums.ItemModType.SUFFIX, template.modTable);
2024-04-14 15:19:55 -04:00
else
forgedItem.suffixToken = workOrder.suffixToken;
2024-04-14 13:51:37 -04:00
2024-04-13 10:01:35 -04:00
// Forged random rolled items are unidentified until completed
2024-04-13 08:10:59 -04:00
2024-04-13 10:01:35 -04:00
if (workOrder.prefixToken == 0 && workOrder.suffixToken == 0)
forgedItem.flags.remove(mbEnums.ItemFlags.Identified);
2024-04-13 08:10:59 -04:00
// Add virtual item to in-memory caches
workOrder.cooking.add(forgedItem);
DbManager.addToCache(forgedItem);
itemWorkOrderLookup.put(forgedItem, workOrder);
2024-04-13 08:10:59 -04:00
return forgedItem;
}
2024-04-13 08:19:09 -04:00
2024-04-17 11:56:23 -04:00
public static void completeWorkOrderBatch(WorkOrder workOrder) {
2024-04-19 09:38:01 -04:00
ArrayList<Item> toRemove = new ArrayList<>();
for (Item virutalItem : workOrder.cooking) {
2024-04-19 10:49:19 -04:00
// Identify completed items
virutalItem.flags.add(mbEnums.ItemFlags.Identified);
2024-04-19 10:49:19 -04:00
2024-04-17 13:48:39 -04:00
// Persist item
Item completedItem = DbManager.ItemQueries.PERSIST(virutalItem);
2024-04-16 14:21:29 -04:00
2024-04-20 12:13:33 -04:00
// Apply Item effects for Prefix and Suffix tokens
completedItem.prefixToken = virutalItem.prefixToken;
completedItem.suffixToken = virutalItem.suffixToken;
2024-04-20 12:28:30 -04:00
2024-04-20 12:37:55 -04:00
ItemManager.applyItemEffects(completedItem);
2024-04-20 12:13:33 -04:00
2024-04-17 11:56:23 -04:00
// add to the vendor inventory
workOrder.vendor.charItemManager.addItemToInventory(completedItem);
ItemProductionMsg outMsg1 = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, completedItem, mbEnums.ProductionActionType.DEPOSIT, true);
DispatchMessage.dispatchMsgToInterestArea(workOrder.vendor, outMsg1, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
ItemProductionMsg outMsg2 = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, completedItem, mbEnums.ProductionActionType.CONFIRM_DEPOSIT, true);
DispatchMessage.dispatchMsgToInterestArea(workOrder.vendor, outMsg2, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
2024-04-19 09:30:59 -04:00
toRemove.add(virutalItem);
2024-04-19 09:38:01 -04:00
}
// Remove the negativeID virtual item from all collections
for (Item virtualItem : toRemove) {
2024-04-19 09:38:01 -04:00
2024-04-19 09:30:59 -04:00
// Remove the virtual items from the forge window
ItemProductionMsg outMsg = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, virtualItem, mbEnums.ProductionActionType.CONFIRM_SETPRICE, true);
2024-04-19 09:30:59 -04:00
DispatchMessage.dispatchMsgToInterestArea(workOrder.vendor, outMsg, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
workOrder.cooking.remove(virtualItem);
itemWorkOrderLookup.remove(virtualItem);
DbManager.removeFromCache(virtualItem);
}
}
2024-04-16 15:40:10 -04:00
2024-04-20 12:37:55 -04:00
public static void forgeWorkOrderBatch(WorkOrder workOrder) {
2024-04-13 08:19:09 -04:00
2024-04-17 12:04:41 -04:00
// New completion time for this batch
workOrder.completionTime = System.currentTimeMillis() + workOrder.rollingDuration;
2024-04-13 08:19:09 -04:00
for (int i = 0; i < workOrder.slots_used; ++i) {
Item forged_item = forgeItem(workOrder);
2024-04-14 15:00:25 -04:00
2024-04-14 15:38:24 -04:00
// Update NPC window
2024-04-14 17:30:35 -04:00
ItemProductionMsg outMsg = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, forged_item, mbEnums.ProductionActionType.CONFIRM_PRODUCE, true);
2024-04-14 15:38:24 -04:00
DispatchMessage.dispatchMsgToInterestArea(workOrder.vendor, outMsg, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
2024-04-14 14:07:06 -04:00
workOrder.total_produced = workOrder.total_produced + 1;
2024-04-13 08:19:09 -04:00
}
}
2024-04-14 13:13:28 -04:00
2024-04-14 13:51:37 -04:00
public static int calcRandomMod(NPC vendor, mbEnums.ItemModType itemModType, int modTable) {
int modifier = 0;
ModTypeTableEntry modTypeTableEntry = null;
2024-04-20 11:02:35 -04:00
ModTableEntry modTableEntry = null;
int rollForModifier;
2024-04-14 13:51:37 -04:00
switch (itemModType) {
case PREFIX:
2024-04-20 11:13:31 -04:00
int randomPrefix = vendor.getModTypeTable().get(vendor.getItemModTable().indexOf(modTable));
modTypeTableEntry = ModTypeTableEntry.rollTable(randomPrefix, ThreadLocalRandom.current().nextInt(1, 100 + 1));
2024-04-14 13:51:37 -04:00
break;
case SUFFIX:
2024-04-20 11:13:31 -04:00
int randomSuffix = vendor.getModSuffixTable().get(vendor.getItemModTable().indexOf(modTable));
modTypeTableEntry = ModTypeTableEntry.rollTable(randomSuffix, ThreadLocalRandom.current().nextInt(1, 100 + 1));
2024-04-14 13:51:37 -04:00
break;
}
if (modTypeTableEntry == null)
return 0;
2024-04-14 13:13:28 -04:00
2024-04-20 11:02:35 -04:00
rollForModifier = ThreadLocalRandom.current().nextInt(1, 100 + 1);
2024-04-14 13:13:28 -04:00
2024-04-20 11:02:35 -04:00
if (rollForModifier < 80) {
2024-04-20 11:05:30 -04:00
int randomModifier = LootManager.TableRoll(vendor.getLevel(), false);
modTableEntry = ModTableEntry.rollTable(modTypeTableEntry.modTableID, randomModifier);
2024-04-20 11:53:53 -04:00
EffectsBase effectsBase = PowersManager.getEffectByIDString(modTableEntry.action);
modifier = effectsBase.getToken();
2024-04-20 09:59:04 -04:00
}
2024-04-14 13:13:28 -04:00
2024-04-14 13:51:37 -04:00
return modifier;
2024-04-14 13:13:28 -04:00
}
2024-04-19 08:59:10 -04:00
2024-04-20 13:21:53 -04:00
public static boolean debitWorkOrderCost(WorkOrder workOrder) {
int strongbox = workOrder.vendor.building.getStrongboxValue();
int overflow = strongbox - workOrder.production_cost_total.get(mbEnums.ResourceType.GOLD);
if (overflow > 1 || workOrder.production_cost_total.size() > 1) {
// Deduct total cost from warehouse
if (workOrder.vendor.building.getCity() == null)
return false;
Warehouse warehouse = workOrder.vendor.building.getCity().warehouse;
if (warehouse == null)
return false;
workOrder.production_cost_total.forEach((key, value) -> warehouse.resources.compute(key, (k, v) -> warehouse.resources.get(key) - value));
2024-04-20 15:54:31 -04:00
DbManager.WarehouseQueries.UPDATE_WAREHOUSE(warehouse);
2024-04-20 13:21:53 -04:00
}
return true;
}
2024-04-05 20:13:16 -04:00
}