/*
* 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.templates.item;
import java.util.ArrayList;
import java.util.List;
import java.util.logging.Logger;
import com.l2jserver.Config;
import com.l2jserver.gameserver.model.Elementals;
import com.l2jserver.gameserver.model.L2Effect;
import com.l2jserver.gameserver.model.L2ItemInstance;
import com.l2jserver.gameserver.model.L2Object;
import com.l2jserver.gameserver.model.L2Skill;
import com.l2jserver.gameserver.model.actor.L2Character;
import com.l2jserver.gameserver.model.actor.instance.L2SummonInstance;
import com.l2jserver.gameserver.network.SystemMessageId;
import com.l2jserver.gameserver.network.serverpackets.SystemMessage;
import com.l2jserver.gameserver.skills.Env;
import com.l2jserver.gameserver.skills.conditions.Condition;
import com.l2jserver.gameserver.skills.funcs.Func;
import com.l2jserver.gameserver.skills.funcs.FuncTemplate;
import com.l2jserver.gameserver.templates.StatsSet;
import com.l2jserver.gameserver.templates.effects.EffectTemplate;
import javolution.util.FastList;
/**
* This class contains all informations concerning the item (weapon, armor, etc).
* Mother class of :
*
L2Armor
* L2EtcItem
* L2Weapon
* @version $Revision: 1.7.2.2.2.5 $ $Date: 2005/04/06 18:25:18 $
*/
public abstract class L2Item
{
public static final int TYPE1_WEAPON_RING_EARRING_NECKLACE = 0;
public static final int TYPE1_SHIELD_ARMOR = 1;
public static final int TYPE1_ITEM_QUESTITEM_ADENA = 4;
public static final int TYPE2_WEAPON = 0;
public static final int TYPE2_SHIELD_ARMOR = 1;
public static final int TYPE2_ACCESSORY = 2;
public static final int TYPE2_QUEST = 3;
public static final int TYPE2_MONEY = 4;
public static final int TYPE2_OTHER = 5;
public static final int TYPE2_PET_WOLF = 6;
public static final int TYPE2_PET_HATCHLING = 7;
public static final int TYPE2_PET_STRIDER = 8;
public static final int TYPE2_PET_BABY = 9;
public static final int TYPE2_PET_EVOLVEDWOLF = 10;
public static final int SLOT_NONE = 0x0000;
public static final int SLOT_UNDERWEAR = 0x0001;
public static final int SLOT_R_EAR = 0x0002;
public static final int SLOT_L_EAR = 0x0004;
public static final int SLOT_LR_EAR = 0x00006;
public static final int SLOT_NECK = 0x0008;
public static final int SLOT_R_FINGER = 0x0010;
public static final int SLOT_L_FINGER = 0x0020;
public static final int SLOT_LR_FINGER = 0x0030;
public static final int SLOT_HEAD = 0x0040;
public static final int SLOT_R_HAND = 0x0080;
public static final int SLOT_L_HAND = 0x0100;
public static final int SLOT_GLOVES = 0x0200;
public static final int SLOT_CHEST = 0x0400;
public static final int SLOT_LEGS = 0x0800;
public static final int SLOT_FEET = 0x1000;
public static final int SLOT_BACK = 0x2000;
public static final int SLOT_LR_HAND = 0x4000;
public static final int SLOT_FULL_ARMOR = 0x8000;
public static final int SLOT_HAIR = 0x010000;
public static final int SLOT_ALLDRESS = 0x020000;
public static final int SLOT_HAIR2 = 0x040000;
public static final int SLOT_HAIRALL = 0x080000;
public static final int SLOT_R_BRACELET = 0x100000;
public static final int SLOT_L_BRACELET = 0x200000;
public static final int SLOT_DECO = 0x400000;
public static final int SLOT_BELT = 0x10000000;
public static final int SLOT_WOLF = -100;
public static final int SLOT_HATCHLING = -101;
public static final int SLOT_STRIDER = -102;
public static final int SLOT_BABYPET = -103;
public static final int SLOT_GREATWOLF = -104;
public static final int SLOT_MULTI_ALLWEAPON = SLOT_LR_HAND | SLOT_R_HAND;
public static final int MATERIAL_STEEL = 0x00; // ??
public static final int MATERIAL_FINE_STEEL = 0x01; // ??
public static final int MATERIAL_BLOOD_STEEL = 0x02; // ??
public static final int MATERIAL_BRONZE = 0x03; // ??
public static final int MATERIAL_SILVER = 0x04; // ??
public static final int MATERIAL_GOLD = 0x05; // ??
public static final int MATERIAL_MITHRIL = 0x06; // ??
public static final int MATERIAL_ORIHARUKON = 0x07; // ??
public static final int MATERIAL_PAPER = 0x08; // ??
public static final int MATERIAL_WOOD = 0x09; // ??
public static final int MATERIAL_CLOTH = 0x0a; // ??
public static final int MATERIAL_LEATHER = 0x0b; // ??
public static final int MATERIAL_BONE = 0x0c; // ??
public static final int MATERIAL_HORN = 0x0d; // ??
public static final int MATERIAL_DAMASCUS = 0x0e; // ??
public static final int MATERIAL_ADAMANTAITE = 0x0f; // ??
public static final int MATERIAL_CHRYSOLITE = 0x10; // ??
public static final int MATERIAL_CRYSTAL = 0x11; // ??
public static final int MATERIAL_LIQUID = 0x12; // ??
public static final int MATERIAL_SCALE_OF_DRAGON = 0x13; // ??
public static final int MATERIAL_DYESTUFF = 0x14; // ??
public static final int MATERIAL_COBWEB = 0x15; // ??
public static final int MATERIAL_SEED = 0x15; // ??
public static final int CRYSTAL_NONE = 0x00; // ??
public static final int CRYSTAL_D = 0x01; // ??
public static final int CRYSTAL_C = 0x02; // ??
public static final int CRYSTAL_B = 0x03; // ??
public static final int CRYSTAL_A = 0x04; // ??
public static final int CRYSTAL_S = 0x05; // ??
public static final int CRYSTAL_S80 = 0x06; // ??
public static final int CRYSTAL_S84 = 0x07; // ??
private static final int[] crystalItemId =
{
0, 1458, 1459, 1460, 1461, 1462, 1462, 1462
};
private static final int[] crystalEnchantBonusArmor =
{
0, 11, 6, 11, 19, 25, 25, 25
};
private static final int[] crystalEnchantBonusWeapon =
{
0, 90, 45, 67, 144, 250, 250, 250
};
private final int _itemId;
private final String _name;
private final int _type1; // needed for item list (inventory)
private final int _type2; // different lists for armor, weapon, etc
private final int _weight;
private final boolean _crystallizable;
private final boolean _stackable;
private final int _materialType;
private final int _crystalType; // default to none-grade
private final int _duration;
private final int _time;
private final int _bodyPart;
private final int _referencePrice;
private final int _crystalCount;
private final boolean _sellable;
private final boolean _dropable;
private final boolean _destroyable;
private final boolean _tradeable;
private final boolean _depositable;
private final boolean _common;
private final boolean _heroItem;
private final boolean _pvpItem;
@SuppressWarnings({ "rawtypes" })
protected final Enum _type;
protected Elementals _elementals = null;
protected FuncTemplate[] _funcTemplates;
protected EffectTemplate[] _effectTemplates;
protected L2Skill[] _skills;
protected List _preConditions = new FastList();
protected static final Func[] _emptyFunctionSet = new Func[0];
protected static final L2Effect[] _emptyEffectSet = new L2Effect[0];
protected static final Logger _log = Logger.getLogger(L2Item.class.getName());
/**
* Constructor of the L2Item that fill class variables.
* Variables filled :
* type
* _itemId
* _name
* _type1 & _type2
* _weight
* _crystallizable
* _stackable
* _materialType & _crystalType & _crystlaCount
* _duration
* _bodypart
* _referencePrice
* _sellable
* @param type : Enum designating the type of the item
* @param set : StatsSet corresponding to a set of couples (key,value) for description of the item
*/
protected L2Item(Enum> type, StatsSet set)
{
_type = type;
_itemId = set.getInteger("item_id");
_name = set.getString("name");
_type1 = set.getInteger("type1"); // needed for item list (inventory)
_type2 = set.getInteger("type2"); // different lists for armor, weapon, etc
_weight = set.getInteger("weight");
_crystallizable = set.getBool("crystallizable");
_stackable = set.getBool("stackable", false);
_materialType = set.getInteger("material");
_crystalType = set.getInteger("crystal_type", CRYSTAL_NONE); // default to none-grade
_duration = set.getInteger("duration");
_time = set.getInteger("time");
_bodyPart = set.getInteger("bodypart");
_referencePrice = set.getInteger("price");
_crystalCount = set.getInteger("crystal_count", 0);
_sellable = set.getBool("sellable", true);
_dropable = set.getBool("dropable", true);
_destroyable = set.getBool("destroyable", true);
_tradeable = set.getBool("tradeable", true);
_depositable = set.getBool("depositable", true);
_common = (_itemId >= 12006 && _itemId <= 12361) || (_itemId >= 11605 && _itemId <= 12308);
_heroItem = (_itemId >= 6611 && _itemId <= 6621) || (_itemId >= 9388 && _itemId <= 9390) || _itemId == 6842;
_pvpItem = (_itemId >= 10667 && _itemId <= 10792) || (_itemId >= 10793 && _itemId <= 10835) || (_itemId >= 12852 && _itemId <= 12977) || (_itemId >= 14363 && _itemId <= 14519) || (_itemId >= 14520 && _itemId <= 14525) || _itemId == 14528 || _itemId == 14529 || _itemId == 14558;
}
/**
* Returns the itemType.
* @return Enum
*/
@SuppressWarnings({ "rawtypes" })
public Enum getItemType()
{
return _type;
}
/**
* Returns the duration of the item
* @return int
*/
public final int getDuration()
{
return _duration;
}
/**
* Returns the time of the item
* @return int
*/
public final int getTime()
{
return _time;
}
/**
* Returns the ID of the iden
* @return int
*/
public final int getItemId()
{
return _itemId;
}
public abstract int getItemMask();
/**
* Return the type of material of the item
* @return int
*/
public final int getMaterialType()
{
return _materialType;
}
/**
* Returns the type 2 of the item
* @return int
*/
public final int getType2()
{
return _type2;
}
/**
* Returns the weight of the item
* @return int
*/
public final int getWeight()
{
return _weight;
}
/**
* Returns if the item is crystallizable
* @return boolean
*/
public final boolean isCrystallizable()
{
return _crystallizable;
}
/**
* Return the type of crystal if item is crystallizable
* @return int
*/
public final int getCrystalType()
{
return _crystalType;
}
/**
* Return the type of crystal if item is crystallizable
* @return int
*/
public final int getCrystalItemId()
{
return crystalItemId[_crystalType];
}
/**
* Returns the grade of the item.
* Concept :
* In fact, this fucntion returns the type of crystal of the item.
* @return int
*/
public final int getItemGrade()
{
return getCrystalType();
}
/**
* Returns the grade of the item.
* For grades S80 and S84 return S
* @return int
*/
public final int getItemGradeSPlus()
{
switch (getItemGrade())
{
case CRYSTAL_S80:
case CRYSTAL_S84:
return CRYSTAL_S;
default:
return getItemGrade();
}
}
/**
* Returns the quantity of crystals for crystallization
* @return int
*/
public final int getCrystalCount()
{
return _crystalCount;
}
/**
* Returns the quantity of crystals for crystallization on specific enchant level
* @return int
*/
public final int getCrystalCount(int enchantLevel)
{
if (enchantLevel > 3)
switch (_type2)
{
case TYPE2_SHIELD_ARMOR:
case TYPE2_ACCESSORY:
return _crystalCount + crystalEnchantBonusArmor[getCrystalType()] * (3 * enchantLevel - 6);
case TYPE2_WEAPON:
return _crystalCount + crystalEnchantBonusWeapon[getCrystalType()] * (2 * enchantLevel - 3);
default:
return _crystalCount;
}
else if (enchantLevel > 0)
switch (_type2)
{
case TYPE2_SHIELD_ARMOR:
case TYPE2_ACCESSORY:
return _crystalCount + crystalEnchantBonusArmor[getCrystalType()] * enchantLevel;
case TYPE2_WEAPON:
return _crystalCount + crystalEnchantBonusWeapon[getCrystalType()] * enchantLevel;
default:
return _crystalCount;
}
else
return _crystalCount;
}
/**
* Returns the name of the item
* @return String
*/
public final String getName()
{
return _name;
}
/**
* Returns the base elemental of the item
* @return Elementals
*/
public final Elementals getElementals()
{
return _elementals;
}
/**
* Sets the base elemental of the item
*/
public final void setElementals(Elementals element)
{
if (_elementals != null)
{
_log.warning("Item " + getName() + "(" + getItemId() + ") has more than one element definition!");
return;
}
_elementals = element;
}
/**
* Return the part of the body used with the item.
* @return int
*/
public final int getBodyPart()
{
return _bodyPart;
}
/**
* Returns the type 1 of the item
* @return int
*/
public final int getType1()
{
return _type1;
}
/**
* Returns if the item is stackable
* @return boolean
*/
public final boolean isStackable()
{
return _stackable;
}
/**
* Returns if the item is consumable
* @return boolean
*/
public boolean isConsumable()
{
return false;
}
public boolean isEquipable()
{
return this.getBodyPart() != 0 && !(this.getItemType() instanceof L2EtcItemType);
}
/**
* Returns the price of reference of the item
* @return int
*/
public final int getReferencePrice()
{
return (isConsumable() ? (int)(_referencePrice * Config.RATE_CONSUMABLE_COST) : _referencePrice);
}
/**
* Returns if the item can be sold
* @return boolean
*/
public final boolean isSellable()
{
return _sellable;
}
/**
* Returns if the item can dropped
* @return boolean
*/
public final boolean isDropable()
{
return _dropable;
}
/**
* Returns if the item can destroy
* @return boolean
*/
public final boolean isDestroyable()
{
return _destroyable;
}
/**
* Returns if the item can add to trade
* @return boolean
*/
public final boolean isTradeable()
{
return _tradeable;
}
/**
* Returns if the item can be put into warehouse
* @return boolean
*/
public final boolean isDepositable()
{
return _depositable;
}
/**
* Returns if item is common
* @return boolean
*/
public final boolean isCommon()
{
return _common;
}
/**
* Returns if item is hero-only
* @return
*/
public final boolean isHeroItem()
{
return _heroItem;
}
/**
* Returns if item is pvp
* @return
*/
public final boolean isPvpItem()
{
return _pvpItem;
}
/**
* Returns if item is for hatchling
* @return boolean
*/
public boolean isForHatchling()
{
return (_type2 == TYPE2_PET_HATCHLING);
}
/**
* Returns if item is for strider
* @return boolean
*/
public boolean isForStrider()
{
return (_type2 == TYPE2_PET_STRIDER);
}
/**
* Returns if item is for wolf
* @return boolean
*/
public boolean isForWolf()
{
return (_type2 == TYPE2_PET_WOLF);
}
/**
* Returns if item is for Great wolf
* @return boolean
*/
public boolean isForEvolvedWolf()
{
return (_type2 == TYPE2_PET_EVOLVEDWOLF);
}
/**
* Returns if item is for wolf
* @return boolean
*/
public boolean isForBabyPet()
{
return (_type2 == TYPE2_PET_BABY);
}
/**
* Returns array of Func objects containing the list of functions used by the item
* @param instance : L2ItemInstance pointing out the item
* @param player : L2Character pointing out the player
* @return Func[] : array of functions
*/
public Func[] getStatFuncs(L2ItemInstance instance, L2Character player)
{
if (_funcTemplates == null || _funcTemplates.length == 0)
return _emptyFunctionSet;
ArrayList funcs = new ArrayList(_funcTemplates.length);
Env env = new Env();
env.player = player;
env.target = player;
env.item = instance;
Func f;
for (FuncTemplate t : _funcTemplates)
{
f = t.getFunc(env, this); // skill is owner
if (f != null)
funcs.add(f);
}
if (funcs.isEmpty())
return _emptyFunctionSet;
return funcs.toArray(new Func[funcs.size()]);
}
/**
* Returns the effects associated with the item.
* @param instance : L2ItemInstance pointing out the item
* @param player : L2Character pointing out the player
* @return L2Effect[] : array of effects generated by the item
*/
public L2Effect[] getEffects(L2ItemInstance instance, L2Character player)
{
if (_effectTemplates == null || _effectTemplates.length == 0)
return _emptyEffectSet;
ArrayList effects = new ArrayList(_effectTemplates.length);
Env env = new Env();
env.player = player;
env.target = player;
env.item = instance;
L2Effect e;
for (EffectTemplate et : _effectTemplates)
{
e = et.getEffect(env);
if (e != null)
{
e.scheduleEffect();
effects.add(e);
}
}
if (effects.isEmpty())
return _emptyEffectSet;
return effects.toArray(new L2Effect[effects.size()]);
}
/**
* Returns effects of skills associated with the item.
* @param caster : L2Character pointing out the caster
* @param target : L2Character pointing out the target
* @return L2Effect[] : array of effects generated by the skill
public L2Effect[] getSkillEffects(L2Character caster, L2Character target)
{
if (_skills == null)
return _emptyEffectSet;
List effects = new FastList();
for (L2Skill skill : _skills)
{
if (!skill.checkCondition(caster, target, true))
continue; // Skill condition not met
if (target.getFirstEffect(skill.getId()) != null)
target.removeEffect(target.getFirstEffect(skill.getId()));
for (L2Effect e : skill.getEffects(caster, target))
effects.add(e);
}
if (effects.isEmpty())
return _emptyEffectSet;
return effects.toArray(new L2Effect[effects.size()]);
}
*/
/**
* Add the FuncTemplate f to the list of functions used with the item
* @param f : FuncTemplate to add
*/
public void attach(FuncTemplate f)
{
switch(f.stat)
{
case FIRE_RES:
case FIRE_POWER:
setElementals(new Elementals(Elementals.FIRE, (int) f.lambda.calc(null)));
break;
case WATER_RES:
case WATER_POWER:
setElementals(new Elementals(Elementals.WATER, (int) f.lambda.calc(null)));
break;
case WIND_RES:
case WIND_POWER:
setElementals(new Elementals(Elementals.WIND, (int) f.lambda.calc(null)));
break;
case EARTH_RES:
case EARTH_POWER:
setElementals(new Elementals(Elementals.EARTH, (int) f.lambda.calc(null)));
break;
case HOLY_RES:
case HOLY_POWER:
setElementals(new Elementals(Elementals.HOLY, (int) f.lambda.calc(null)));
break;
case DARK_RES:
case DARK_POWER:
setElementals(new Elementals(Elementals.DARK, (int) f.lambda.calc(null)));
break;
}
// If _functTemplates is empty, create it and add the FuncTemplate f in it
if (_funcTemplates == null)
{
_funcTemplates = new FuncTemplate[]
{
f
};
}
else
{
int len = _funcTemplates.length;
FuncTemplate[] tmp = new FuncTemplate[len + 1];
// Definition : arraycopy(array source, begins copy at this position of source, array destination, begins copy at this position in dest,
// number of components to be copied)
System.arraycopy(_funcTemplates, 0, tmp, 0, len);
tmp[len] = f;
_funcTemplates = tmp;
}
}
/**
* Add the EffectTemplate effect to the list of effects generated by the item
* @param effect : EffectTemplate
*/
public void attach(EffectTemplate effect)
{
if (_effectTemplates == null)
{
_effectTemplates = new EffectTemplate[]
{
effect
};
}
else
{
int len = _effectTemplates.length;
EffectTemplate[] tmp = new EffectTemplate[len + 1];
// Definition : arraycopy(array source, begins copy at this position of source, array destination, begins copy at this position in dest,
// number of components to be copied)
System.arraycopy(_effectTemplates, 0, tmp, 0, len);
tmp[len] = effect;
_effectTemplates = tmp;
}
}
/**
* Add the L2Skill skill to the list of skills generated by the item
* @param skill : L2Skill
*/
public void attach(L2Skill skill)
{
if (_skills == null)
{
_skills = new L2Skill[]
{
skill
};
}
else
{
int len = _skills.length;
L2Skill[] tmp = new L2Skill[len + 1];
// Definition : arraycopy(array source, begins copy at this position of source, array destination, begins copy at this position in dest,
// number of components to be copied)
System.arraycopy(_skills, 0, tmp, 0, len);
tmp[len] = skill;
_skills = tmp;
}
}
public final void attach(Condition c)
{
if (!_preConditions.contains(c))
_preConditions.add(c);
}
public final L2Skill[] getItemSkills()
{
return _skills;
}
public boolean checkCondition(L2Character activeChar, L2Object target, boolean sendMessage)
{
if (activeChar.isGM() && !Config.GM_ITEM_RESTRICTION)
return true;
Env env = new Env();
env.player = activeChar;
if (target instanceof L2Character) // TODO: object or char?
env.target = (L2Character)target;
for (Condition preCondition : _preConditions)
{
if (preCondition == null)
return true;
if (!preCondition.test(env))
{
if (activeChar instanceof L2SummonInstance)
{
((L2SummonInstance)activeChar).getOwner().sendPacket(new SystemMessage(SystemMessageId.PET_CANNOT_USE_ITEM));
return false;
}
if (sendMessage)
{
String msg = preCondition.getMessage();
int msgId = preCondition.getMessageId();
if (msg != null)
{
activeChar.sendMessage(msg);
}
else if (msgId !=0)
{
SystemMessage sm = new SystemMessage(msgId);
if (preCondition.isAddName())
sm.addItemName(_itemId);
activeChar.sendPacket(sm);
}
}
return false;
}
}
return true;
}
/**
* Returns the name of the item
* @return String
*/
@Override
public String toString()
{
return _name+"("+_itemId+")";
}
}