/* This file is part of the HeavenMS MapleStory Server Copyleft (L) 2016 - 2019 RonanLana This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation version 3 as published by the Free Software Foundation. You may not use, modify or distribute this program under any other version of the GNU Affero General Public License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . */ package client.processor.action; import client.MapleCharacter; import client.MapleClient; import client.inventory.Equip; import client.inventory.Item; import client.inventory.MapleInventoryType; import client.inventory.manipulator.MapleInventoryManipulator; import config.YamlConfig; import constants.game.GameConstants; import constants.inventory.ItemConstants; import net.packet.InPacket; import server.MakerItemFactory; import server.MakerItemFactory.MakerItemCreateEntry; import server.MapleItemInformationProvider; import tools.FilePrinter; import tools.PacketCreator; import tools.Pair; import java.util.LinkedHashMap; import java.util.LinkedList; import java.util.List; import java.util.Map; /** * * @author Ronan */ public class MakerProcessor { private static MapleItemInformationProvider ii = MapleItemInformationProvider.getInstance(); public static void makerAction(InPacket p, MapleClient c) { if (c.tryacquireClient()) { try { int type = p.readInt(); int toCreate = p.readInt(); int toDisassemble = -1, pos = -1; boolean makerSucceeded = true; MakerItemCreateEntry recipe; Map reagentids = new LinkedHashMap<>(); int stimulantid = -1; if(type == 3) { // building monster crystal int fromLeftover = toCreate; toCreate = ii.getMakerCrystalFromLeftover(toCreate); if(toCreate == -1) { c.sendPacket(PacketCreator.serverNotice(1, ii.getName(fromLeftover) + " is unavailable for Monster Crystal conversion.")); c.sendPacket(PacketCreator.makerEnableActions()); return; } recipe = MakerItemFactory.generateLeftoverCrystalEntry(fromLeftover, toCreate); } else if(type == 4) { // disassembling p.readInt(); // 1... probably inventory type pos = p.readInt(); Item it = c.getPlayer().getInventory(MapleInventoryType.EQUIP).getItem((short) pos); if(it != null && it.getItemId() == toCreate) { toDisassemble = toCreate; Pair>> pair = generateDisassemblyInfo(toDisassemble); if(pair != null) { recipe = MakerItemFactory.generateDisassemblyCrystalEntry(toDisassemble, pair.getLeft(), pair.getRight()); } else { c.sendPacket(PacketCreator.serverNotice(1, ii.getName(toCreate) + " is unavailable for Monster Crystal disassembly.")); c.sendPacket(PacketCreator.makerEnableActions()); return; } } else { c.sendPacket(PacketCreator.serverNotice(1, "An unknown error occurred when trying to apply that item for disassembly.")); c.sendPacket(PacketCreator.makerEnableActions()); return; } } else { if(ItemConstants.isEquipment(toCreate)) { // only equips uses stimulant and reagents if(p.readByte() != 0) { // stimulant stimulantid = ii.getMakerStimulant(toCreate); if(!c.getAbstractPlayerInteraction().haveItem(stimulantid)) { stimulantid = -1; } } int reagents = Math.min(p.readInt(), getMakerReagentSlots(toCreate)); for(int i = 0; i < reagents; i++) { // crystals int reagentid = p.readInt(); if(ItemConstants.isMakerReagent(reagentid)) { Short rs = reagentids.get(reagentid); if(rs == null) { reagentids.put(reagentid, (short) 1); } else { reagentids.put(reagentid, (short) (rs + 1)); } } } List> toUpdate = new LinkedList<>(); for(Map.Entry r : reagentids.entrySet()) { int qty = c.getAbstractPlayerInteraction().getItemQuantity(r.getKey()); if(qty < r.getValue()) { toUpdate.add(new Pair<>(r.getKey(), (short) qty)); } } // remove those not present on player inventory if(!toUpdate.isEmpty()) { for(Pair rp : toUpdate) { if(rp.getRight() > 0) { reagentids.put(rp.getLeft(), rp.getRight()); } else { reagentids.remove(rp.getLeft()); } } } if(!reagentids.isEmpty()) { if(!removeOddMakerReagents(toCreate, reagentids)) { c.sendPacket(PacketCreator.serverNotice(1, "You can only use WATK and MATK Strengthening Gems on weapon items.")); c.sendPacket(PacketCreator.makerEnableActions()); return; } } } recipe = MakerItemFactory.getItemCreateEntry(toCreate, stimulantid, reagentids); } short createStatus = getCreateStatus(c, recipe); switch(createStatus) { case -1:// non-available for Maker itemid has been tried to forge FilePrinter.printError(FilePrinter.EXPLOITS, "Player " + c.getPlayer().getName() + " tried to craft itemid " + toCreate + " using the Maker skill."); c.sendPacket(PacketCreator.serverNotice(1, "The requested item could not be crafted on this operation.")); c.sendPacket(PacketCreator.makerEnableActions()); break; case 1: // no items c.sendPacket(PacketCreator.serverNotice(1, "You don't have all required items in your inventory to make " + ii.getName(toCreate) + ".")); c.sendPacket(PacketCreator.makerEnableActions()); break; case 2: // no meso c.sendPacket(PacketCreator.serverNotice(1, "You don't have enough mesos (" + GameConstants.numberWithCommas(recipe.getCost()) + ") to complete this operation.")); c.sendPacket(PacketCreator.makerEnableActions()); break; case 3: // no req level c.sendPacket(PacketCreator.serverNotice(1, "You don't have enough level to complete this operation.")); c.sendPacket(PacketCreator.makerEnableActions()); break; case 4: // no req skill level c.sendPacket(PacketCreator.serverNotice(1, "You don't have enough Maker level to complete this operation.")); c.sendPacket(PacketCreator.makerEnableActions()); break; case 5: // inventory full c.sendPacket(PacketCreator.serverNotice(1, "Your inventory is full.")); c.sendPacket(PacketCreator.makerEnableActions()); break; default: if(toDisassemble != -1) { MapleInventoryManipulator.removeFromSlot(c, MapleInventoryType.EQUIP, (short) pos, (short) 1, false); } else { for (Pair pair : recipe.getReqItems()) { c.getAbstractPlayerInteraction().gainItem(pair.getLeft(), (short) -pair.getRight(), false); } } int cost = recipe.getCost(); if(stimulantid == -1 && reagentids.isEmpty()) { if(cost > 0) c.getPlayer().gainMeso(-cost, false); for (Pair pair : recipe.getGainItems()) { c.getPlayer().setCS(true); c.getAbstractPlayerInteraction().gainItem(pair.getLeft(), pair.getRight().shortValue(), false); c.getPlayer().setCS(false); } } else { toCreate = recipe.getGainItems().get(0).getLeft(); if(stimulantid != -1) c.getAbstractPlayerInteraction().gainItem(stimulantid, (short) -1, false); if(!reagentids.isEmpty()) { for(Map.Entry r : reagentids.entrySet()) { c.getAbstractPlayerInteraction().gainItem(r.getKey(), (short) (-1 * r.getValue()), false); } } if(cost > 0) c.getPlayer().gainMeso(-cost, false); makerSucceeded = addBoostedMakerItem(c, toCreate, stimulantid, reagentids); } // thanks inhyuk for noticing missing MAKER_RESULT packets if (type == 3) { c.sendPacket(PacketCreator.makerResultCrystal(recipe.getGainItems().get(0).getLeft(), recipe.getReqItems().get(0).getLeft())); } else if (type == 4) { c.sendPacket(PacketCreator.makerResultDesynth(recipe.getReqItems().get(0).getLeft(), recipe.getCost(), recipe.getGainItems())); } else { c.sendPacket(PacketCreator.makerResult(makerSucceeded, recipe.getGainItems().get(0).getLeft(), recipe.getGainItems().get(0).getRight(), recipe.getCost(), recipe.getReqItems(), stimulantid, new LinkedList<>(reagentids.keySet()))); } c.sendPacket(PacketCreator.showMakerEffect(makerSucceeded)); c.getPlayer().getMap().broadcastMessage(c.getPlayer(), PacketCreator.showForeignMakerEffect(c.getPlayer().getId(), makerSucceeded), false); if(toCreate == 4260003 && type == 3 && c.getPlayer().getQuestStatus(6033) == 1) { c.getAbstractPlayerInteraction().setQuestProgress(6033, 1); } } } finally { c.releaseClient(); } } } // checks and prevents hackers from PE'ing Maker operations with invalid operations private static boolean removeOddMakerReagents(int toCreate, Map reagentids) { Map reagentType = new LinkedHashMap<>(); List toRemove = new LinkedList<>(); boolean isWeapon = ItemConstants.isWeapon(toCreate) || YamlConfig.config.server.USE_MAKER_PERMISSIVE_ATKUP; // thanks Vcoc for finding a case where a weapon wouldn't be counted as such due to a bounding on isWeapon for(Map.Entry r : reagentids.entrySet()) { int curRid = r.getKey(); int type = r.getKey() / 100; if(type < 42502 && !isWeapon) { // only weapons should gain w.att/m.att from these. return false; //toRemove.add(curRid); } else { Integer tableRid = reagentType.get(type); if(tableRid != null) { if(tableRid < curRid) { toRemove.add(tableRid); reagentType.put(type, curRid); } else { toRemove.add(curRid); } } else { reagentType.put(type, curRid); } } } // removing less effective gems of repeated type for(Integer i : toRemove) { reagentids.remove(i); } // the Maker skill will use only one of each gem for(Integer i : reagentids.keySet()) { reagentids.put(i, (short) 1); } return true; } private static int getMakerReagentSlots(int itemId) { try { int eqpLevel = ii.getEquipLevelReq(itemId); if(eqpLevel < 78) { return 1; } else if(eqpLevel >= 78 && eqpLevel < 108) { return 2; } else { return 3; } } catch(NullPointerException npe) { return 0; } } private static Pair>> generateDisassemblyInfo(int itemId) { int recvFee = ii.getMakerDisassembledFee(itemId); if(recvFee > -1) { List> gains = ii.getMakerDisassembledItems(itemId); if(!gains.isEmpty()) { return new Pair<>(recvFee, gains); } } return null; } public static int getMakerSkillLevel(MapleCharacter chr) { return chr.getSkillLevel((chr.getJob().getId() / 1000) * 10000000 + 1007); } private static short getCreateStatus(MapleClient c, MakerItemCreateEntry recipe) { if(recipe.isInvalid()) { return -1; } if(!hasItems(c, recipe)) { return 1; } if(c.getPlayer().getMeso() < recipe.getCost()) { return 2; } if(c.getPlayer().getLevel() < recipe.getReqLevel()) { return 3; } if(getMakerSkillLevel(c.getPlayer()) < recipe.getReqSkillLevel()) { return 4; } List addItemids = new LinkedList<>(); List addQuantity = new LinkedList<>(); List rmvItemids = new LinkedList<>(); List rmvQuantity = new LinkedList<>(); for (Pair p : recipe.getReqItems()) { rmvItemids.add(p.getLeft()); rmvQuantity.add(p.getRight()); } for (Pair p : recipe.getGainItems()) { addItemids.add(p.getLeft()); addQuantity.add(p.getRight()); } if (!c.getAbstractPlayerInteraction().canHoldAllAfterRemoving(addItemids, addQuantity, rmvItemids, rmvQuantity)) { return 5; } return 0; } private static boolean hasItems(MapleClient c, MakerItemCreateEntry recipe) { for (Pair p : recipe.getReqItems()) { int itemId = p.getLeft(); if (c.getPlayer().getInventory(ItemConstants.getInventoryType(itemId)).countById(itemId) < p.getRight()) { return false; } } return true; } private static boolean addBoostedMakerItem(MapleClient c, int itemid, int stimulantid, Map reagentids) { if(stimulantid != -1 && !ii.rollSuccessChance(90.0)) { return false; } Item item = ii.getEquipById(itemid); if(item == null) return false; Equip eqp = (Equip)item; if(ItemConstants.isAccessory(item.getItemId()) && eqp.getUpgradeSlots() <= 0) eqp.setUpgradeSlots(3); if(YamlConfig.config.server.USE_ENHANCED_CRAFTING == true) { if(!(c.getPlayer().isGM() && YamlConfig.config.server.USE_PERFECT_GM_SCROLL)) { eqp.setUpgradeSlots((byte)(eqp.getUpgradeSlots() + 1)); } item = MapleItemInformationProvider.getInstance().scrollEquipWithId(eqp, 2049100, true, 2049100, c.getPlayer().isGM()); } if(!reagentids.isEmpty()) { Map stats = new LinkedHashMap<>(); List randOption = new LinkedList<>(); List randStat = new LinkedList<>(); for(Map.Entry r : reagentids.entrySet()) { Pair reagentBuff = ii.getMakerReagentStatUpgrade(r.getKey()); if(reagentBuff != null) { String s = reagentBuff.getLeft(); if(s.substring(0, 4).contains("rand")) { if(s.substring(4).equals("Stat")) { randStat.add((short) (reagentBuff.getRight() * r.getValue())); } else { randOption.add((short) (reagentBuff.getRight() * r.getValue())); } } else { String stat = s.substring(3); if(!stat.equals("ReqLevel")) { // improve req level... really? switch (stat) { case "MaxHP": stat = "MHP"; break; case "MaxMP": stat = "MMP"; break; } Integer d = stats.get(stat); if(d == null) { stats.put(stat, reagentBuff.getRight() * r.getValue()); } else { stats.put(stat, d + (reagentBuff.getRight() * r.getValue())); } } } } } ii.improveEquipStats(eqp, stats); for(Short sh : randStat) { ii.scrollOptionEquipWithChaos(eqp, sh, false); } for(Short sh : randOption) { ii.scrollOptionEquipWithChaos(eqp, sh, true); } } if(stimulantid != -1) { eqp = ii.randomizeUpgradeStats(eqp); } MapleInventoryManipulator.addFromDrop(c, item, false, -1); return true; } }