/* * This program is free software: you can redistribute it and/or modify it under * the terms of the GNU General Public License as published by the Free Software * Foundation, either version 3 of the License, or (at your option) any later * version. * * 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 General Public License for more * details. * * You should have received a copy of the GNU General Public License along with * this program. If not, see . */ package com.l2jserver.gameserver.network.clientpackets; import static com.l2jserver.gameserver.model.actor.L2Npc.INTERACTION_DISTANCE; import java.util.List; import java.util.logging.Logger; import com.l2jserver.Config; import com.l2jserver.gameserver.TradeController; import com.l2jserver.gameserver.model.L2ItemInstance; import com.l2jserver.gameserver.model.L2Object; import com.l2jserver.gameserver.model.L2TradeList; import com.l2jserver.gameserver.model.actor.L2Character; import com.l2jserver.gameserver.model.actor.instance.L2MerchantInstance; import com.l2jserver.gameserver.model.actor.instance.L2MerchantSummonInstance; import com.l2jserver.gameserver.model.actor.instance.L2PcInstance; import com.l2jserver.gameserver.network.SystemMessageId; import com.l2jserver.gameserver.network.serverpackets.ActionFailed; import com.l2jserver.gameserver.network.serverpackets.ExBuySellListPacket; import com.l2jserver.gameserver.network.serverpackets.StatusUpdate; import com.l2jserver.gameserver.network.serverpackets.SystemMessage; import com.l2jserver.gameserver.templates.item.L2Item; import com.l2jserver.gameserver.util.Util; /** * RequestRefundItem client packet class. */ public final class RequestRefundItem extends L2GameClientPacket { private static final String _C__D0_75_REQUESTREFUNDITEM = "[C] D0:75 RequestRefundItem"; private static final Logger _log = Logger.getLogger(RequestRefundItem.class.getName()); private static final int BATCH_LENGTH = 4; // length of the one item private int _listId; private int[] _items = null; @Override protected void readImpl() { _listId = readD(); final int count = readD(); if (count <= 0 || count > Config.MAX_ITEM_IN_PACKET || count * BATCH_LENGTH != _buf.remaining()) return; _items = new int[count]; for (int i = 0; i < count; i++) _items[i] = readD(); } @Override protected void runImpl() { final L2PcInstance player = getClient().getActiveChar(); if (player == null) return; if (!getClient().getFloodProtectors().getTransaction().tryPerformAction("refund")) { player.sendMessage("You using refund too fast."); return; } if (_items == null) { sendPacket(ActionFailed.STATIC_PACKET); return; } if (!player.hasRefund()) { sendPacket(ActionFailed.STATIC_PACKET); return; } L2Object target = player.getTarget(); if (!player.isGM() && (target == null // No target (ie GM Shop) || !(target instanceof L2MerchantInstance || target instanceof L2MerchantSummonInstance) || player.getInstanceId() != target.getInstanceId() || !player.isInsideRadius(target, INTERACTION_DISTANCE, true, false))) // Distance is too far { sendPacket(ActionFailed.STATIC_PACKET); return; } L2Character merchant = null; if (target instanceof L2MerchantInstance || target instanceof L2MerchantSummonInstance) merchant = (L2Character) target; else if (!player.isGM()) { sendPacket(ActionFailed.STATIC_PACKET); return; } L2TradeList list = null; double taxRate = 0; if (merchant != null) { List lists; if (merchant instanceof L2MerchantInstance) { lists = TradeController.getInstance().getBuyListByNpcId(((L2MerchantInstance) merchant).getNpcId()); taxRate = ((L2MerchantInstance) merchant).getMpc().getTotalTaxRate(); } else { lists = TradeController.getInstance().getBuyListByNpcId(((L2MerchantSummonInstance) merchant).getNpcId()); taxRate = 50; } if (!player.isGM()) { if (lists == null) { Util.handleIllegalPlayerAction(player, "Warning!! Character " + player.getName() + " of account " + player.getAccountName() + " sent a false BuyList list_id " + _listId, Config.DEFAULT_PUNISH); return; } for (L2TradeList tradeList : lists) { if (tradeList.getListId() == _listId) list = tradeList; } } else list = TradeController.getInstance().getBuyList(_listId); } else list = TradeController.getInstance().getBuyList(_listId); if (list == null) { Util.handleIllegalPlayerAction(player, "Warning!! Character " + player.getName() + " of account " + player.getAccountName() + " sent a false BuyList list_id " + _listId, Config.DEFAULT_PUNISH); return; } long weight = 0; long adena = 0; long slots = 0; L2ItemInstance[] refund = player.getRefund().getItems(); int[] objectIds = new int[_items.length]; for (int i = 0; i < _items.length; i++) { int idx = _items[i]; if (idx < 0 || idx >= refund.length) { Util.handleIllegalPlayerAction(player, "Warning!! Character " + player.getName() + " of account " + player.getAccountName() + " sent invalid refund index", Config.DEFAULT_PUNISH); return; } // check for duplicates - indexes for (int j = i + 1; j < _items.length; j++) if (idx == _items[j]) { Util.handleIllegalPlayerAction(player, "Warning!! Character " + player.getName() + " of account " + player.getAccountName() + " sent duplicate refund index", Config.DEFAULT_PUNISH); return; } final L2ItemInstance item = refund[idx]; final L2Item template = item.getItem(); objectIds[i] = item.getObjectId(); // second check for duplicates - object ids for (int j = 0; j < i; j++) if (objectIds[i] == objectIds[j]) { Util.handleIllegalPlayerAction(player, "Warning!! Character " + player.getName() + " of account " + player.getAccountName() + " has duplicate items in refund list", Config.DEFAULT_PUNISH); return; } long count = item.getCount(); weight += count * template.getWeight(); adena += count * template.getReferencePrice() / 2; if (!template.isStackable()) slots += count; else if (player.getInventory().getItemByItemId(template.getItemId()) == null) slots++; } if (weight > Integer.MAX_VALUE || weight < 0 || !player.getInventory().validateWeight((int) weight)) { sendPacket(new SystemMessage(SystemMessageId.WEIGHT_LIMIT_EXCEEDED)); sendPacket(ActionFailed.STATIC_PACKET); return; } if (slots > Integer.MAX_VALUE || slots < 0 || !player.getInventory().validateCapacity((int) slots)) { sendPacket(new SystemMessage(SystemMessageId.SLOTS_FULL)); sendPacket(ActionFailed.STATIC_PACKET); return; } if ((adena < 0) || !player.reduceAdena("Refund", adena, player.getLastFolkNPC(), false)) { sendPacket(new SystemMessage(SystemMessageId.YOU_NOT_ENOUGH_ADENA)); sendPacket(ActionFailed.STATIC_PACKET); return; } for (int i = 0; i < _items.length; i++) { L2ItemInstance item = player.getRefund().transferItem("Refund", objectIds[i], Long.MAX_VALUE, player.getInventory(), player, player.getLastFolkNPC()); if (item == null) { _log.warning("Error refunding object for char " + player.getName() + " (newitem == null)"); continue; } } // Update current load status on player StatusUpdate su = new StatusUpdate(player.getObjectId()); su.addAttribute(StatusUpdate.CUR_LOAD, player.getCurrentLoad()); player.sendPacket(su); player.sendPacket(new ExBuySellListPacket(player, list, taxRate, true)); } /* (non-Javadoc) * @see com.l2jserver.gameserver.clientpackets.ClientBasePacket#getType() */ @Override public String getType() { return _C__D0_75_REQUESTREFUNDITEM; } }