/* * 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 static com.l2jserver.gameserver.model.itemcontainer.PcInventory.MAX_ADENA; import java.util.List; 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.serverpackets.ActionFailed; import com.l2jserver.gameserver.network.serverpackets.ExBuySellListPacket; import com.l2jserver.gameserver.network.serverpackets.StatusUpdate; import com.l2jserver.gameserver.util.Util; /** * RequestSellItem client packet class. */ public final class RequestSellItem extends L2GameClientPacket { private static final String _C__1E_REQUESTSELLITEM = "[C] 1E RequestSellItem"; //private static Logger _log = Logger.getLogger(RequestSellItem.class.getName()); private static final int BATCH_LENGTH = 16; // length of the one item private int _listId; private Item[] _items = null; /** * packet type id 0x1e * * sample * * 1e * 00 00 00 00 // list id * 02 00 00 00 // number of items * * 71 72 00 10 // object id * ea 05 00 00 // item id * 01 00 00 00 // item count * * 76 4b 00 10 // object id * 2e 0a 00 00 // item id * 01 00 00 00 // item count * * format: cdd (ddd) * @param decrypt */ @Override protected void readImpl() { _listId = readD(); int count = readD(); if (count <= 0 || count > Config.MAX_ITEM_IN_PACKET || count * BATCH_LENGTH != _buf.remaining()) { return; } _items = new Item[count]; for (int i = 0; i < count; i++) { int objectId = readD(); int itemId = readD(); long cnt = readQ(); if (objectId < 1 || itemId < 1 || cnt < 1) { _items = null; return; } _items[i] = new Item(objectId, itemId, cnt); } } @Override protected void runImpl() { this.processSell(); } protected void processSell() { L2PcInstance player = getClient().getActiveChar(); if (player == null) return; if (!getClient().getFloodProtectors().getTransaction().tryPerformAction("buy")) { player.sendMessage("You buying too fast."); return; } if (_items == null) { sendPacket(ActionFailed.STATIC_PACKET); return; } // Alt game - Karma punishment if (!Config.ALT_GAME_KARMA_PLAYER_CAN_SHOP && player.getKarma() > 0) { 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) // Target not a merchant || target.getInstanceId() != player.getInstanceId() || !player.isInsideRadius(target, INTERACTION_DISTANCE, true, false))) // Distance is too far { sendPacket(ActionFailed.STATIC_PACKET); return; } L2Character merchant = null; double taxRate = 0; if (target instanceof L2MerchantInstance || target instanceof L2MerchantSummonInstance) merchant = (L2Character) target; else if (!player.isGM()) { sendPacket(ActionFailed.STATIC_PACKET); return; } L2TradeList list = null; 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 totalPrice = 0; // Proceed the sell for (Item i : _items) { L2ItemInstance item = player.checkItemManipulation(i.getObjectId(), i.getCount(), "sell"); if (item == null || (!item.isSellable())) continue; long price = item.getReferencePrice() / 2; totalPrice += price * i.getCount(); if ((MAX_ADENA / i.getCount()) < price || totalPrice > MAX_ADENA) { Util.handleIllegalPlayerAction(player, "Warning!! Character " + player.getName() + " of account " + player.getAccountName() + " tried to purchase over " + MAX_ADENA + " adena worth of goods.", Config.DEFAULT_PUNISH); return; } if (Config.ALLOW_REFUND) item = player.getInventory().transferItem("Sell", i.getObjectId(), i.getCount(), player.getRefund(), player, merchant); else item = player.getInventory().destroyItem("Sell", i.getObjectId(), i.getCount(), player, merchant); } player.addAdena("Sell", totalPrice, merchant, false); // Update current load as well StatusUpdate su = new StatusUpdate(player); su.addAttribute(StatusUpdate.CUR_LOAD, player.getCurrentLoad()); player.sendPacket(su); player.sendPacket(new ExBuySellListPacket(player, list, taxRate, true)); } private class Item { private final int _objectId; private final int _itemId; private final long _count; public Item(int objId, int id, long num) { _objectId = objId; _itemId = id; _count = num; } public int getObjectId() { return _objectId; } @SuppressWarnings("unused") public int getItemId() { return _itemId; } public long getCount() { return _count; } } /* (non-Javadoc) * @see com.l2jserver.gameserver.clientpackets.ClientBasePacket#getType() */ @Override public String getType() { return _C__1E_REQUESTSELLITEM; } }