/* * 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.instancemanager; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement; import java.util.List; import java.util.logging.Level; import java.util.logging.Logger; import com.l2jserver.Config; import com.l2jserver.L2DatabaseFactory; import com.l2jserver.gameserver.ItemsAutoDestroy; import com.l2jserver.gameserver.ThreadPoolManager; import com.l2jserver.gameserver.model.L2World; import com.l2jserver.gameserver.model.items.instance.L2ItemInstance; import com.l2jserver.gameserver.model.items.type.L2EtcItemType; import com.l2jserver.util.L2FastList; /** * This class manage all items on ground. * @author Enforcer */ public class ItemsOnGroundManager implements Runnable { private static final Logger _log = Logger.getLogger(ItemsOnGroundManager.class.getName()); protected List _items = new L2FastList<>(true); protected ItemsOnGroundManager() { if (Config.SAVE_DROPPED_ITEM_INTERVAL > 0) { ThreadPoolManager.getInstance().scheduleGeneralAtFixedRate(this, Config.SAVE_DROPPED_ITEM_INTERVAL, Config.SAVE_DROPPED_ITEM_INTERVAL); } load(); } private void load() { // If SaveDroppedItem is false, may want to delete all items previously stored to avoid add old items on reactivate if (!Config.SAVE_DROPPED_ITEM && Config.CLEAR_DROPPED_ITEM_TABLE) { emptyTable(); } if (!Config.SAVE_DROPPED_ITEM) { return; } // if DestroyPlayerDroppedItem was previously false, items currently protected will be added to ItemsAutoDestroy if (Config.DESTROY_DROPPED_PLAYER_ITEM) { String str = null; if (!Config.DESTROY_EQUIPABLE_PLAYER_ITEM) { // Recycle misc. items only str = "UPDATE itemsonground SET drop_time = ? WHERE drop_time = -1 AND equipable = 0"; } else if (Config.DESTROY_EQUIPABLE_PLAYER_ITEM) { // Recycle all items including equip-able str = "UPDATE itemsonground SET drop_time = ? WHERE drop_time = -1"; } try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement ps = con.prepareStatement(str)) { ps.setLong(1, System.currentTimeMillis()); ps.execute(); } catch (Exception e) { _log.log(Level.SEVERE, getClass().getSimpleName() + ": Error while updating table ItemsOnGround " + e.getMessage(), e); } } // Add items to world try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement ps = con.prepareStatement("SELECT object_id,item_id,count,enchant_level,x,y,z,drop_time,equipable FROM itemsonground")) { int count = 0; try (ResultSet rs = ps.executeQuery()) { L2ItemInstance item; while (rs.next()) { item = new L2ItemInstance(rs.getInt(1), rs.getInt(2)); L2World.getInstance().storeObject(item); // this check and.. if (item.isStackable() && (rs.getInt(3) > 1)) { item.setCount(rs.getInt(3)); } // this, are really necessary? if (rs.getInt(4) > 0) { item.setEnchantLevel(rs.getInt(4)); } item.getPosition().setWorldPosition(rs.getInt(5), rs.getInt(6), rs.getInt(7)); item.getPosition().setWorldRegion(L2World.getInstance().getRegion(item.getPosition().getWorldPosition())); item.getPosition().getWorldRegion().addVisibleObject(item); final long dropTime = rs.getLong(8); item.setDropTime(dropTime); item.setProtected(dropTime == -1); item.setIsVisible(true); L2World.getInstance().addVisibleObject(item, item.getPosition().getWorldRegion()); _items.add(item); count++; // add to ItemsAutoDestroy only items not protected if (!Config.LIST_PROTECTED_ITEMS.contains(item.getItemId())) { if (dropTime > -1) { if (((Config.AUTODESTROY_ITEM_AFTER > 0) && (item.getItemType() != L2EtcItemType.HERB)) || ((Config.HERB_AUTO_DESTROY_TIME > 0) && (item.getItemType() == L2EtcItemType.HERB))) { ItemsAutoDestroy.getInstance().addItem(item); } } } } } _log.info(getClass().getSimpleName() + ": Loaded " + count + " items."); } catch (Exception e) { _log.log(Level.SEVERE, getClass().getSimpleName() + ": Error while loading ItemsOnGround " + e.getMessage(), e); } if (Config.EMPTY_DROPPED_ITEM_TABLE_AFTER_LOAD) { emptyTable(); } } public void save(L2ItemInstance item) { if (!Config.SAVE_DROPPED_ITEM) { return; } _items.add(item); } public void removeObject(L2ItemInstance item) { if (Config.SAVE_DROPPED_ITEM) { _items.remove(item); } } public void saveInDb() { run(); } public void cleanUp() { _items.clear(); } public void emptyTable() { try (Connection con = L2DatabaseFactory.getInstance().getConnection(); Statement s = con.createStatement()) { s.executeQuery("DELETE FROM itemsonground"); } catch (Exception e1) { _log.log(Level.SEVERE, getClass().getSimpleName() + ": Error while cleaning table ItemsOnGround " + e1.getMessage(), e1); } } @Override public synchronized void run() { if (!Config.SAVE_DROPPED_ITEM) { return; } emptyTable(); if (_items.isEmpty()) { return; } try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement("INSERT INTO itemsonground(object_id,item_id,count,enchant_level,x,y,z,drop_time,equipable) VALUES(?,?,?,?,?,?,?,?,?)")) { for (L2ItemInstance item : _items) { if (item == null) { continue; } if (CursedWeaponsManager.getInstance().isCursed(item.getItemId())) { continue; // Cursed Items not saved to ground, prevent double save } try { statement.setInt(1, item.getObjectId()); statement.setInt(2, item.getItemId()); statement.setLong(3, item.getCount()); statement.setInt(4, item.getEnchantLevel()); statement.setInt(5, item.getX()); statement.setInt(6, item.getY()); statement.setInt(7, item.getZ()); statement.setLong(8, (item.isProtected() ? -1 : item.getDropTime())); // item is protected or AutoDestroyed statement.setLong(9, (item.isEquipable() ? 1 : 0)); // set equip-able statement.execute(); statement.clearParameters(); } catch (Exception e) { _log.log(Level.SEVERE, getClass().getSimpleName() + ": Error while inserting into table ItemsOnGround: " + e.getMessage(), e); } } } catch (SQLException e) { _log.log(Level.SEVERE, getClass().getSimpleName() + ": SQL error while storing items on ground: " + e.getMessage(), e); } } public static final ItemsOnGroundManager getInstance() { return SingletonHolder._instance; } private static class SingletonHolder { protected static final ItemsOnGroundManager _instance = new ItemsOnGroundManager(); } }