|
|
|
// • ▌ ▄ ·. ▄▄▄· ▄▄ • ▪ ▄▄· ▄▄▄▄· ▄▄▄· ▐▄▄▄ ▄▄▄ .
|
|
|
|
// ·██ ▐███▪▐█ ▀█ ▐█ ▀ ▪██ ▐█ ▌▪▐█ ▀█▪▐█ ▀█ •█▌ ▐█▐▌·
|
|
|
|
// ▐█ ▌▐▌▐█·▄█▀▀█ ▄█ ▀█▄▐█·██ ▄▄▐█▀▀█▄▄█▀▀█ ▐█▐ ▐▌▐▀▀▀
|
|
|
|
// ██ ██▌▐█▌▐█ ▪▐▌▐█▄▪▐█▐█▌▐███▌██▄▪▐█▐█ ▪▐▌██▐ █▌▐█▄▄▌
|
|
|
|
// ▀▀ █▪▀▀▀ ▀ ▀ ·▀▀▀▀ ▀▀▀·▀▀▀ ·▀▀▀▀ ▀ ▀ ▀▀ █▪ ▀▀▀
|
|
|
|
// Magicbane Emulator Project © 2013 - 2022
|
|
|
|
// www.magicbane.com
|
|
|
|
|
|
|
|
|
|
|
|
package engine.objects;
|
|
|
|
|
|
|
|
import engine.Enum;
|
|
|
|
import engine.Enum.ItemType;
|
|
|
|
import engine.Enum.OwnerType;
|
|
|
|
import engine.gameManager.DbManager;
|
|
|
|
import engine.gameManager.PowersManager;
|
|
|
|
import engine.powers.poweractions.AbstractPowerAction;
|
|
|
|
import org.pmw.tinylog.Logger;
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.concurrent.atomic.AtomicInteger;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An immutable, non-persistant implementation of Item
|
|
|
|
*
|
|
|
|
* @author Burfo
|
|
|
|
*/
|
|
|
|
public final class MobLoot extends Item {
|
|
|
|
|
|
|
|
private static final AtomicInteger LastUsedId = new AtomicInteger(0);
|
|
|
|
|
|
|
|
private boolean isDeleted = false;
|
|
|
|
private boolean noSteal;
|
|
|
|
private String prefix = "";
|
|
|
|
private String suffix = "";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new MobLoot.
|
|
|
|
* Do not use this to create Gold.
|
|
|
|
*
|
|
|
|
* @param mob Mob that owns this item
|
|
|
|
* @param ib ItemBase
|
|
|
|
*/
|
|
|
|
public MobLoot(AbstractCharacter mob, ItemBase ib, boolean noSteal) {
|
|
|
|
this(mob, ib, 0, noSteal);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new MobLoot item to hold Gold for the Mob.
|
|
|
|
*
|
|
|
|
* @param mob Mob that owns this item
|
|
|
|
* @param qtyOfGold Quantity of gold
|
|
|
|
*/
|
|
|
|
public MobLoot(AbstractCharacter mob, int qtyOfGold) {
|
|
|
|
this(mob, ItemBase.getGoldItemBase(), qtyOfGold, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new MobLoot.
|
|
|
|
* Primarily used for stackable items that have a quantity.
|
|
|
|
*
|
|
|
|
* @param mob Mob that owns this item
|
|
|
|
* @param ib ItemBase
|
|
|
|
* @param quantity Quantity of the item
|
|
|
|
*/
|
|
|
|
public MobLoot(AbstractCharacter mob, ItemBase ib, int quantity, boolean noSteal) {
|
|
|
|
super(ib, mob.getObjectUUID(),
|
|
|
|
OwnerType.Mob, (byte) 0, (byte) 0, (short) 0,
|
|
|
|
(short) 0, true, false, false, false, true,
|
|
|
|
false, (byte) 0, new ArrayList<>(), generateId());
|
|
|
|
|
|
|
|
if (quantity == 0 && ib.getType() == ItemType.RESOURCE)
|
|
|
|
quantity = 1;
|
|
|
|
|
|
|
|
|
|
|
|
if (quantity > 0)
|
|
|
|
this.setNumOfItems(quantity);
|
|
|
|
|
|
|
|
this.noSteal = noSteal;
|
|
|
|
this.setIsID(this.getItemBase().isAutoID());
|
|
|
|
|
|
|
|
// Class is 'final'; passing 'this' should be okay at the end of the constructor
|
|
|
|
|
|
|
|
DbManager.addToCache(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the MobLoot object from its Id number
|
|
|
|
*
|
|
|
|
* @param id Id Number
|
|
|
|
* @return MobLoot object
|
|
|
|
*/
|
|
|
|
public static MobLoot getFromCache(int id) {
|
|
|
|
return (MobLoot) DbManager.getFromCache(Enum.GameObjectType.MobLoot, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the next available Id number.
|
|
|
|
*
|
|
|
|
* @return Id number
|
|
|
|
*/
|
|
|
|
private static int generateId() {
|
|
|
|
int id = LastUsedId.decrementAndGet();
|
|
|
|
|
|
|
|
//TODO Add a way to reclaim disposed IDs if this becomes a problem
|
|
|
|
if (id == (-10000))
|
|
|
|
Logger.warn("Only 10,000 Id numbers remain useable. Server restart suggested.");
|
|
|
|
else if (id < Integer.MIN_VALUE + 1000)
|
|
|
|
Logger.warn("Only " + (Integer.MIN_VALUE + id)
|
|
|
|
+ " Id numbers remain useable! Server restart suggested.");
|
|
|
|
else if (id == Integer.MIN_VALUE)
|
|
|
|
throw new UnsupportedOperationException("MobLoot has no remaining Id numbers! Restart server immediately!");
|
|
|
|
else if ((id % 10000) == 0)
|
|
|
|
Logger.info(id + " of " + Integer.MIN_VALUE + " Id numbers consumed.");
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts this MotLoot to a persistable Item. Used when a MotLoot is
|
|
|
|
* looted
|
|
|
|
* from a Mob to a Player. Do not call for a Gold item.
|
|
|
|
*
|
|
|
|
* @return An orphaned Item, ready to be moved to the Player's inventory.
|
|
|
|
*/
|
|
|
|
public synchronized Item promoteToItem(PlayerCharacter looter) {
|
|
|
|
|
|
|
|
if (looter == null)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (isDeleted)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (this.getItemBase().getType().equals(ItemType.GOLD))
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
|
|
Item item = (Item) this;
|
|
|
|
|
|
|
|
item.setOwner(looter);
|
|
|
|
//item.setIsID(false);
|
|
|
|
|
|
|
|
item.containerType = Enum.ItemContainerType.INVENTORY;
|
|
|
|
item.setValue(0);
|
|
|
|
item.setName(this.getCustomName());
|
|
|
|
item.setIsID(this.isID());
|
|
|
|
|
|
|
|
if (this.getNumOfItems() > 1)
|
|
|
|
item.setNumOfItems(this.getNumOfItems());
|
|
|
|
|
|
|
|
try {
|
|
|
|
item = DbManager.ItemQueries.ADD_ITEM(item);
|
|
|
|
} catch (Exception e) {
|
|
|
|
Logger.error("e");
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// for (String effectName : this.effectNames)
|
|
|
|
// item.addPermanentEnchantment(effectName, 0);
|
|
|
|
//transfer enchantments to item
|
|
|
|
if (this.prefix.length() != 0)
|
|
|
|
item.addPermanentEnchantment(this.prefix, 0);
|
|
|
|
if (this.suffix.length() != 0)
|
|
|
|
item.addPermanentEnchantment(this.suffix, 0);
|
|
|
|
|
|
|
|
this.junk();
|
|
|
|
if(item.getItemBase().isVorg())
|
|
|
|
Item.BakeVorgStats(item);
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
public synchronized Item promoteToItemForNPC(NPC looter) {
|
|
|
|
|
|
|
|
if (looter == null)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (isDeleted)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (this.getItemBase().getType().equals(ItemType.GOLD))
|
|
|
|
return null;
|
|
|
|
|
|
|
|
Item item = this;
|
|
|
|
item.setOwner(looter);
|
|
|
|
item.containerType = Enum.ItemContainerType.INVENTORY;
|
|
|
|
item.setIsID(true);
|
|
|
|
|
|
|
|
if (this.getNumOfItems() > 1)
|
|
|
|
item.setNumOfItems(this.getNumOfItems());
|
|
|
|
|
|
|
|
try {
|
|
|
|
item = DbManager.ItemQueries.ADD_ITEM(item);
|
|
|
|
} catch (Exception e) {
|
|
|
|
Logger.error(e);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
item.containerType = Enum.ItemContainerType.INVENTORY;
|
|
|
|
|
|
|
|
// for (String effectName : this.effectNames)
|
|
|
|
// item.addPermanentEnchantment(effectName, 0);
|
|
|
|
//transfer enchantments to item
|
|
|
|
try {
|
|
|
|
for (String enchant : this.getEffectNames()) {
|
|
|
|
item.addPermanentEnchantment(enchant, 0);
|
|
|
|
}
|
|
|
|
} catch (Exception e) {
|
|
|
|
Logger.error(e.getMessage());
|
|
|
|
}
|
|
|
|
|
|
|
|
DbManager.NPCQueries.REMOVE_FROM_PRODUCTION_LIST(this.getObjectUUID(), looter.getObjectUUID());
|
|
|
|
looter.removeItemFromForge(this);
|
|
|
|
this.junk();
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
public synchronized void recycle(NPC vendor) {
|
|
|
|
|
|
|
|
//remove from production list for npc in db
|
|
|
|
|
|
|
|
DbManager.NPCQueries.REMOVE_FROM_PRODUCTION_LIST(this.getObjectUUID(), vendor.getObjectUUID());
|
|
|
|
this.removeFromCache();
|
|
|
|
isDeleted = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Junks the item and marks it as deleted
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
protected synchronized void junk() {
|
|
|
|
this.removeFromCache();
|
|
|
|
isDeleted = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determines if this object has been marked as deleted.
|
|
|
|
*
|
|
|
|
* @return True if deleted.
|
|
|
|
*/
|
|
|
|
public boolean isDeleted() {
|
|
|
|
return this.isDeleted;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean noSteal() {
|
|
|
|
return this.noSteal;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void addPermanentEnchantment(String enchantID, int rank, int value, boolean prefix) {
|
|
|
|
AbstractPowerAction apa = PowersManager.getPowerActionByIDString(enchantID);
|
|
|
|
if (apa == null)
|
|
|
|
return;
|
|
|
|
apa.applyEffectForItem(this, rank);
|
|
|
|
|
|
|
|
//limit to 2 effects
|
|
|
|
// if (this.effectNames.size() < 2)
|
|
|
|
// this.effectNames.add(enchantID);
|
|
|
|
if (prefix)
|
|
|
|
this.prefix = enchantID;
|
|
|
|
else
|
|
|
|
this.suffix = enchantID;
|
|
|
|
|
|
|
|
this.getEffectNames().add(enchantID);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* *****
|
|
|
|
* All of the following methods are overridden from
|
|
|
|
* the superclass and intentionally not implemented.
|
|
|
|
* *****
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
public void setOwnerID(int id) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
public synchronized void decrementChargesRemaining() {
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean equipItem(NPC npc, byte slot) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean equipItem(PlayerCharacter pc, byte slot) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean moveItemToBank(NPC npc) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean moveItemToBank(PlayerCharacter pc) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean moveItemToInventory(Corpse corpse) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean moveItemToInventory(NPC npc) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean moveItemToInventory(PlayerCharacter pc) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected boolean moveItemToVault(Account a) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
public void setLastOwner(AbstractWorldObject value) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
public void updateDatabase() {
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not implemented
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
@Deprecated
|
|
|
|
protected void validateItemContainer() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getPrefix() {
|
|
|
|
return prefix;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setPrefix(String prefix) {
|
|
|
|
this.prefix = prefix;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getSuffix() {
|
|
|
|
return suffix;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setSuffix(String suffix) {
|
|
|
|
this.suffix = suffix;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|