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.client.msg.ItemProductionMsg;
|
2024-04-25 08:42:20 -04:00
|
|
|
import engine.objects.City;
|
|
|
|
|
import engine.objects.Item;
|
|
|
|
|
import engine.objects.ItemTemplate;
|
|
|
|
|
import engine.objects.NPC;
|
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-05-12 08:14:06 -04:00
|
|
|
import java.util.concurrent.BlockingQueue;
|
|
|
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
|
|
|
import java.util.concurrent.DelayQueue;
|
|
|
|
|
import java.util.concurrent.ThreadLocalRandom;
|
2024-04-05 20:13:16 -04:00
|
|
|
import java.util.concurrent.atomic.AtomicInteger;
|
|
|
|
|
|
|
|
|
|
public enum ForgeManager implements Runnable {
|
|
|
|
|
|
2024-04-25 08:49:13 -04:00
|
|
|
// MB Dev notes:
|
2024-04-27 07:22:20 -04:00
|
|
|
// Class implements forge rolling mechanics for Magicbane.
|
2024-04-25 08:49:13 -04:00
|
|
|
//
|
2024-04-26 15:56:04 -04:00
|
|
|
// .submit(workOrder) may be called from any thread: (ItemProductionMsgHandler).
|
2024-04-27 08:37:20 -04:00
|
|
|
// Concurrency is managed by same lock used for warehouse (city.cityTransactionLock).
|
2024-04-26 15:56:04 -04:00
|
|
|
// WorkOrders are persisted then reconstituted at bootstrap using table dyn.workorders.
|
2024-04-27 08:37:20 -04:00
|
|
|
// Forge window (ManageNPCMsg) uses item.upgradeDate to serialize completion time.
|
2024-04-25 08:49:13 -04:00
|
|
|
//
|
2024-05-12 07:15:55 -04:00
|
|
|
// Replaces garbage code that looked as if written by a mental patient.
|
2024-04-27 08:37:20 -04:00
|
|
|
//
|
2024-05-12 11:31:25 -04:00
|
|
|
// @TODO Reuse same set of virtual items for the life of a workOrder.
|
2024-04-25 08:49:13 -04:00
|
|
|
|
2024-04-05 20:13:16 -04:00
|
|
|
FORGE_MANAGER;
|
|
|
|
|
|
2024-04-22 16:33:29 -04:00
|
|
|
public static final BlockingQueue<WorkOrder> forge = new DelayQueue<>();
|
2024-04-25 09:22:50 -04:00
|
|
|
public static final AtomicInteger workOrderCounter = new AtomicInteger(0);
|
2024-04-22 07:35:59 -04:00
|
|
|
public static final ConcurrentHashMap<NPC, ConcurrentHashMap.KeySetView<WorkOrder, Boolean>> vendorWorkOrderLookup = new ConcurrentHashMap<>();
|
2024-04-22 00:25:16 -04:00
|
|
|
public static final ConcurrentHashMap<Item, WorkOrder> itemWorkOrderLookup = new ConcurrentHashMap<>();
|
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-25 08:49:13 -04:00
|
|
|
WorkOrder workOrder;
|
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-27 08:47:04 -04:00
|
|
|
// The forge is a blocking priority queue using an epoc sort.
|
|
|
|
|
// workOrders are popped once the workOrder.completionTime has
|
|
|
|
|
// passed.
|
2024-04-25 09:47:58 -04:00
|
|
|
|
2024-04-05 20:13:16 -04:00
|
|
|
try {
|
2024-04-14 12:13:51 -04:00
|
|
|
workOrder = forge.take();
|
|
|
|
|
|
2024-05-11 09:02:54 -04:00
|
|
|
// Early exit for completed workOrders loaded from disk
|
2024-05-12 11:32:00 -04:00
|
|
|
// or vendors who were re-deeded with items still cooking.
|
2024-05-10 09:34:58 -04:00
|
|
|
|
2024-05-11 09:15:11 -04:00
|
|
|
if (workOrder.vendor == null && workOrder.runCompleted.get())
|
2024-05-10 09:34:58 -04:00
|
|
|
continue;
|
|
|
|
|
|
2024-04-26 15:56:04 -04:00
|
|
|
// This workOrder has completed production.
|
|
|
|
|
|
2024-04-24 13:44:22 -04:00
|
|
|
if (workOrder.total_produced >= workOrder.total_to_produce) {
|
2024-04-14 17:02:36 -04:00
|
|
|
|
2024-05-12 07:15:55 -04:00
|
|
|
// CONFIRM_PRODUCE to refresh the interface after we add
|
|
|
|
|
// the Identified flag to the item.
|
2024-04-15 11:07:26 -04:00
|
|
|
|
2024-04-24 13:44:22 -04:00
|
|
|
for (Item workOrderItem : workOrder.cooking) {
|
|
|
|
|
workOrderItem.flags.add(mbEnums.ItemFlags.Identified);
|
|
|
|
|
ItemProductionMsg outMsg = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, workOrderItem, mbEnums.ProductionActionType.CONFIRM_PRODUCE, true);
|
2024-05-12 13:14:42 -04:00
|
|
|
DispatchManager.dispatchMsgToInterestArea(workOrder.vendor, outMsg, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
|
2024-04-24 13:44:22 -04:00
|
|
|
}
|
2024-04-13 08:51:02 -04:00
|
|
|
|
2024-05-12 07:15:55 -04:00
|
|
|
// Set this workOrder to completed and update on disk
|
2024-04-22 15:09:43 -04:00
|
|
|
|
2024-05-11 13:12:10 -04:00
|
|
|
workOrder.runCompleted.set(true);
|
2024-04-27 07:24:57 -04:00
|
|
|
DbManager.WarehouseQueries.WRITE_WORKORDER(workOrder);
|
2024-04-24 13:44:22 -04:00
|
|
|
continue;
|
2024-05-10 09:34:58 -04:00
|
|
|
}
|
2024-04-15 11:30:56 -04:00
|
|
|
|
2024-05-12 07:15:55 -04:00
|
|
|
// PERSIST our current cooking batch of virtual
|
|
|
|
|
// items then add to the vendor inventory
|
2024-04-15 11:30:56 -04:00
|
|
|
|
2024-04-24 13:44:22 -04:00
|
|
|
completeWorkOrderBatch(workOrder);
|
2024-04-15 11:30:56 -04:00
|
|
|
|
2024-05-11 13:12:10 -04:00
|
|
|
// Create new set of in-memory only virtual items
|
2024-04-10 18:01:27 -04:00
|
|
|
|
2024-04-24 13:44:22 -04:00
|
|
|
forgeWorkOrderBatch(workOrder);
|
2024-04-14 16:29:17 -04:00
|
|
|
|
2024-04-24 13:44:22 -04:00
|
|
|
// enQueue this workOrder again; back into the oven
|
|
|
|
|
// until all items for this workOrder are completed.
|
2024-04-14 16:29:17 -04:00
|
|
|
|
2024-04-24 13:44:22 -04:00
|
|
|
forge.add(workOrder);
|
2024-04-05 20:13:16 -04:00
|
|
|
|
2024-04-25 08:51:33 -04:00
|
|
|
// Debugging: Logger.info(workOrder.toString());
|
2024-04-17 11:56:23 -04:00
|
|
|
|
2024-04-24 13:44:22 -04:00
|
|
|
} catch (Exception e) {
|
|
|
|
|
Logger.error(e);
|
|
|
|
|
}
|
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-24 13:47:25 -04:00
|
|
|
// Must have a city to roll anything
|
|
|
|
|
|
|
|
|
|
City city = workOrder.vendor.building.getCity();
|
|
|
|
|
|
|
|
|
|
if (city == null)
|
|
|
|
|
return 58; //58: The formula is beyond the means of this facility
|
|
|
|
|
|
|
|
|
|
// Concurrency is rightly managed by same lock as warehouse
|
|
|
|
|
|
|
|
|
|
city.transactionLock.writeLock().lock();
|
|
|
|
|
|
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-23 15:11:56 -04:00
|
|
|
int validation_result = WorkOrder.validate(workOrder);
|
2024-04-14 12:08:51 -04:00
|
|
|
|
2024-04-25 10:03:37 -04:00
|
|
|
// The return code is used by the caller (ItemProductionMsgHandler)
|
|
|
|
|
// for display of a popup error message to 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-05-11 13:12:10 -04:00
|
|
|
// Configure this production run.
|
2024-04-20 13:21:53 -04:00
|
|
|
|
2024-05-11 13:12:10 -04:00
|
|
|
try {
|
2024-04-25 09:22:50 -04:00
|
|
|
workOrder.workOrderID = workOrderCounter.incrementAndGet();
|
2024-05-12 08:14:06 -04:00
|
|
|
workOrder.rollingDuration = NPCManager.calcRollingDuration(workOrder.vendor, workOrder.templateID);
|
2024-04-21 11:40:27 -04:00
|
|
|
workOrder.completionTime = System.currentTimeMillis() + workOrder.rollingDuration;
|
2024-05-11 09:15:11 -04:00
|
|
|
workOrder.slots_used.set(calcAvailableSlots(workOrder));
|
2024-04-07 22:31:58 -04:00
|
|
|
|
2024-05-11 09:15:11 -04:00
|
|
|
if (workOrder.slots_used.get() == 0)
|
2024-05-09 14:50:47 -04:00
|
|
|
return 58;
|
|
|
|
|
|
2024-04-21 11:40:27 -04:00
|
|
|
workOrder.total_produced = 0;
|
2024-04-23 13:12:49 -04:00
|
|
|
|
2024-05-12 11:34:22 -04:00
|
|
|
// Single item configuration from protocol message (ItemProductionMsg)
|
2024-04-23 13:12:49 -04:00
|
|
|
|
2024-05-11 09:33:54 -04:00
|
|
|
if (workOrder.multiple_slot_request == false && workOrder.total_to_produce == 0)
|
2024-04-23 13:12:49 -04:00
|
|
|
workOrder.total_to_produce = 1;
|
|
|
|
|
|
2024-05-11 13:12:10 -04:00
|
|
|
// Calculate total cost for this production run
|
2024-04-11 13:46:24 -04:00
|
|
|
|
2024-05-11 09:15:11 -04:00
|
|
|
workOrder.total_to_produce *= workOrder.slots_used.get();
|
2024-04-23 13:40:51 -04:00
|
|
|
|
|
|
|
|
workOrder.production_cost = calcProductionCost(workOrder);
|
2024-05-08 15:40:35 -04:00
|
|
|
workOrder.production_cost.forEach((key, value) -> workOrder.production_cost_total.put(key, value * workOrder.total_to_produce));
|
2024-04-10 16:47:02 -04:00
|
|
|
|
2024-05-11 13:12:10 -04:00
|
|
|
// Withdraw gold and resources. Availability has previously been validated.
|
2024-04-19 08:15:17 -04:00
|
|
|
|
2024-04-25 09:22:50 -04:00
|
|
|
if (!WorkOrder.withdrawWorkOrderCost(workOrder))
|
2024-04-21 11:40:27 -04:00
|
|
|
return 58; //58: The formula is beyond the means of this facility
|
2024-04-20 13:01:21 -04:00
|
|
|
|
2024-04-25 08:41:23 -04:00
|
|
|
// Create new batch of virtual items
|
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-27 08:58:28 -04:00
|
|
|
// Assign the new workOrder to the vendor
|
2024-04-20 13:01:21 -04:00
|
|
|
|
2024-04-21 11:39:59 -04:00
|
|
|
vendorWorkOrderLookup.get(workOrder.vendor).add(workOrder);
|
2024-04-27 08:58:28 -04:00
|
|
|
|
|
|
|
|
// Enqueue the new workOrder
|
|
|
|
|
|
2024-04-21 11:40:27 -04:00
|
|
|
forge.add(workOrder);
|
2024-04-20 13:01:21 -04:00
|
|
|
|
2024-04-25 10:04:58 -04:00
|
|
|
// PERSIST workOrder (dyn_workorders)
|
2024-04-22 14:57:00 -04:00
|
|
|
|
2024-04-27 07:24:57 -04:00
|
|
|
DbManager.WarehouseQueries.WRITE_WORKORDER(workOrder);
|
2024-04-22 08:35:58 -04:00
|
|
|
|
2024-04-20 16:52:24 -04:00
|
|
|
} catch (Exception e) {
|
|
|
|
|
Logger.error(e);
|
|
|
|
|
} finally {
|
|
|
|
|
city.transactionLock.writeLock().unlock();
|
|
|
|
|
}
|
2024-04-23 13:15:15 -04:00
|
|
|
Logger.info(workOrder.toString());
|
2024-04-20 13:01:21 -04:00
|
|
|
return validation_result;
|
|
|
|
|
}
|
|
|
|
|
|
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-25 19:57:56 -04:00
|
|
|
// Subtract the slots currently assigned to other workOrders for this vendor
|
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-22 00:19:29 -04:00
|
|
|
availableSlots = availableSlots - npcWorkOrder.cooking.size();
|
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-05-11 09:33:54 -04:00
|
|
|
if (availableSlots > 0 && workOrder.multiple_slot_request == false)
|
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-25 19:57:56 -04:00
|
|
|
// Calculate production cost for a single run of the workOrder
|
2024-04-10 17:22:45 -04:00
|
|
|
|
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);
|
2024-04-23 15:05:03 -04:00
|
|
|
production_cost.putAll(PowersManager._effect_costMaps.get(prefix.getIDString()));
|
2024-04-10 16:47:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (workOrder.suffixToken != 0) {
|
|
|
|
|
EffectsBase suffix = PowersManager.getEffectByToken(workOrder.suffixToken);
|
2024-04-23 15:05:03 -04:00
|
|
|
production_cost.putAll(PowersManager._effect_costMaps.get(suffix.getIDString()));
|
2024-04-10 16:47:02 -04:00
|
|
|
}
|
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-25 08:41:23 -04:00
|
|
|
// Create new virtual 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
|
|
|
|
2024-04-21 08:23:58 -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
|
|
|
|
2024-04-25 08:41:23 -04:00
|
|
|
// item.upgradeDate is serialized (ItemProductionMsg)
|
|
|
|
|
// for vendor forge window completion time.
|
2024-04-16 15:33:39 -04:00
|
|
|
|
2024-04-13 08:51:02 -04:00
|
|
|
forgedItem.setDateToUpgrade(workOrder.completionTime);
|
2024-04-13 08:10:59 -04:00
|
|
|
|
2024-04-25 08:41:23 -04:00
|
|
|
// Assign a prefix and suffix to this item if random rolled
|
2024-04-14 13:51:37 -04:00
|
|
|
|
|
|
|
|
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-25 08:41:23 -04:00
|
|
|
// 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-23 13:20:34 -04:00
|
|
|
else
|
|
|
|
|
forgedItem.flags.add(mbEnums.ItemFlags.Identified);
|
2024-04-13 08:10:59 -04:00
|
|
|
|
2024-04-21 08:23:58 -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-15 12:12:40 -04:00
|
|
|
|
2024-04-19 09:38:01 -04:00
|
|
|
ArrayList<Item> toRemove = new ArrayList<>();
|
|
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
for (Item virutalItem : workOrder.cooking) {
|
2024-04-15 12:12:40 -04:00
|
|
|
|
2024-04-19 10:49:19 -04:00
|
|
|
// Identify completed items
|
|
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
virutalItem.flags.add(mbEnums.ItemFlags.Identified);
|
2024-04-24 13:13:48 -04:00
|
|
|
virutalItem.containerType = mbEnums.ItemContainerType.INVENTORY;
|
2024-04-19 10:49:19 -04:00
|
|
|
|
2024-04-17 13:48:39 -04:00
|
|
|
// Persist item
|
2024-04-17 13:46:10 -04:00
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
Item completedItem = DbManager.ItemQueries.PERSIST(virutalItem);
|
2024-04-16 14:21:29 -04:00
|
|
|
|
2024-04-24 13:13:48 -04:00
|
|
|
// Copy Prefix and Suffix tokens from virtual item.
|
2024-04-20 12:13:33 -04:00
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
completedItem.prefixToken = virutalItem.prefixToken;
|
|
|
|
|
completedItem.suffixToken = virutalItem.suffixToken;
|
2024-04-20 12:28:30 -04:00
|
|
|
|
2024-05-12 07:17:31 -04:00
|
|
|
// Add effects for the tokens. Writes to disk using
|
|
|
|
|
// table dyn_item_enchantment
|
2024-04-24 13:13:48 -04:00
|
|
|
|
2024-04-20 12:37:55 -04:00
|
|
|
ItemManager.applyItemEffects(completedItem);
|
2024-04-20 12:13:33 -04:00
|
|
|
|
2024-04-24 13:13:48 -04:00
|
|
|
// Add to the vendor inventory
|
2024-04-17 11:56:23 -04:00
|
|
|
|
2024-04-17 13:46:10 -04:00
|
|
|
workOrder.vendor.charItemManager.addItemToInventory(completedItem);
|
|
|
|
|
|
|
|
|
|
ItemProductionMsg outMsg1 = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, completedItem, mbEnums.ProductionActionType.DEPOSIT, true);
|
2024-05-12 13:14:42 -04:00
|
|
|
DispatchManager.dispatchMsgToInterestArea(workOrder.vendor, outMsg1, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
|
2024-04-17 13:46:10 -04:00
|
|
|
ItemProductionMsg outMsg2 = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, completedItem, mbEnums.ProductionActionType.CONFIRM_DEPOSIT, true);
|
2024-05-12 13:14:42 -04:00
|
|
|
DispatchManager.dispatchMsgToInterestArea(workOrder.vendor, outMsg2, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
|
2024-04-19 09:30:59 -04:00
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
toRemove.add(virutalItem);
|
2024-04-19 09:38:01 -04:00
|
|
|
}
|
|
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
for (Item virtualItem : toRemove) {
|
2024-04-19 09:38:01 -04:00
|
|
|
|
2024-04-25 08:41:23 -04:00
|
|
|
// Remove virtual items from the forge window
|
2024-04-19 09:30:59 -04:00
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
ItemProductionMsg outMsg = new ItemProductionMsg(workOrder.vendor.building, workOrder.vendor, virtualItem, mbEnums.ProductionActionType.CONFIRM_SETPRICE, true);
|
2024-05-12 13:14:42 -04:00
|
|
|
DispatchManager.dispatchMsgToInterestArea(workOrder.vendor, outMsg, mbEnums.DispatchChannel.SECONDARY, 700, false, false);
|
2024-04-19 09:30:59 -04:00
|
|
|
|
2024-04-25 08:41:23 -04:00
|
|
|
// Remove virtual item from all collections
|
|
|
|
|
|
2024-04-21 08:23:58 -04:00
|
|
|
workOrder.cooking.remove(virtualItem);
|
|
|
|
|
itemWorkOrderLookup.remove(virtualItem);
|
|
|
|
|
DbManager.removeFromCache(virtualItem);
|
2024-04-15 12:12:40 -04:00
|
|
|
}
|
|
|
|
|
}
|
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-25 08:41:23 -04:00
|
|
|
// Completion time for this batch is in the future
|
2024-04-17 12:04:41 -04:00
|
|
|
|
|
|
|
|
workOrder.completionTime = System.currentTimeMillis() + workOrder.rollingDuration;
|
|
|
|
|
|
2024-05-11 09:15:11 -04:00
|
|
|
for (int i = 0; i < workOrder.slots_used.get(); ++i) {
|
2024-04-13 08:19:09 -04:00
|
|
|
|
|
|
|
|
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-05-12 13:14:42 -04:00
|
|
|
DispatchManager.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-22 15:09:43 -04:00
|
|
|
|
2024-04-25 19:57:56 -04:00
|
|
|
// Write updated workOrder to disk
|
2024-04-22 15:09:43 -04:00
|
|
|
|
2024-04-27 07:24:57 -04:00
|
|
|
DbManager.WarehouseQueries.WRITE_WORKORDER(workOrder);
|
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) {
|
|
|
|
|
|
2024-04-25 08:41:23 -04:00
|
|
|
// Random prefix or suffix token based on item.template.modtable
|
|
|
|
|
|
2024-04-14 13:51:37 -04:00
|
|
|
int modifier = 0;
|
|
|
|
|
ModTypeTableEntry modTypeTableEntry = null;
|
2024-04-25 08:49:13 -04:00
|
|
|
ModTableEntry modTableEntry;
|
2024-04-20 11:02:35 -04:00
|
|
|
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-05 20:13:16 -04:00
|
|
|
}
|