/*
* 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 <http://www.gnu.org/licenses/>.
*/
package silentium.gameserver.model;
import java.util.concurrent.Future;
import silentium.commons.utils.Rnd;
import silentium.gameserver.ThreadPoolManager;
import silentium.gameserver.model.actor.instance.L2PcInstance;
import silentium.gameserver.model.actor.instance.L2PenaltyMonsterInstance;
import silentium.gameserver.network.SystemMessageId;
import silentium.gameserver.network.serverpackets.ExFishingHpRegen;
import silentium.gameserver.network.serverpackets.ExFishingStartCombat;
import silentium.gameserver.network.serverpackets.PlaySound;
import silentium.gameserver.network.serverpackets.SystemMessage;
import silentium.gameserver.tables.NpcTable;
import silentium.gameserver.templates.chars.L2NpcTemplate;
public class L2Fishing implements Runnable
{
private L2PcInstance _fisher;
private int _time;
private int _stop = 0;
private int _goodUse = 0;
private int _anim = 0;
private int _mode = 0;
private int _deceptiveMode = 0;
private Future<?> _fishAiTask;
private boolean _thinking;
// Fish datas
private final int _fishId;
private final int _fishMaxHp;
private int _fishCurHp;
private final double _regenHp;
private final boolean _isUpperGrade;
private int _lureType;
@Override
public void run()
{
if (_fisher == null)
return;
if (_fishCurHp >= _fishMaxHp * 2)
{
// The fish got away
_fisher.sendPacket(SystemMessageId.BAIT_STOLEN_BY_FISH);
doDie(false);
}
else if (_time <= 0)
{
// Time is up, so that fish got away
_fisher.sendPacket(SystemMessageId.FISH_SPIT_THE_HOOK);
doDie(false);
}
else
aiTask();
}
// =========================================================
public L2Fishing(L2PcInstance Fisher, L2FishData fish, boolean isNoob, boolean isUpperGrade)
{
_fisher = Fisher;
_fishMaxHp = fish.getHP();
_fishCurHp = _fishMaxHp;
_regenHp = fish.getHpRegen();
_fishId = fish.getId();
_time = fish.getCombatTime() / 1000;
_isUpperGrade = isUpperGrade;
if (isUpperGrade)
{
_deceptiveMode = Rnd.get(100) >= 90 ? 1 : 0;
_lureType = 2;
}
else
{
_deceptiveMode = 0;
_lureType = isNoob ? 0 : 1;
}
_mode = Rnd.get(100) >= 80 ? 1 : 0;
_fisher.broadcastPacket(new ExFishingStartCombat(_fisher, _time, _fishMaxHp, _mode, _lureType, _deceptiveMode));
_fisher.sendPacket(new PlaySound(1, "SF_S_01", 0, 0, 0, 0, 0));
// Succeeded in getting a bite
_fisher.sendPacket(SystemMessageId.GOT_A_BITE);
if (_fishAiTask == null)
_fishAiTask = ThreadPoolManager.getInstance().scheduleEffectAtFixedRate(this, 1000, 1000);
}
public void changeHp(int hp, int pen)
{
_fishCurHp -= hp;
if (_fishCurHp < 0)
_fishCurHp = 0;
_fisher.broadcastPacket(new ExFishingHpRegen(_fisher, _time, _fishCurHp, _mode, _goodUse, _anim, pen, _deceptiveMode));
_anim = 0;
if (_fishCurHp > _fishMaxHp * 2)
{
_fishCurHp = _fishMaxHp * 2;
doDie(false);
return;
}
else if (_fishCurHp == 0)
{
doDie(true);
return;
}
}
public synchronized void doDie(boolean win)
{
if (_fishAiTask != null)
{
_fishAiTask.cancel(false);
_fishAiTask = null;
}
if (_fisher == null)
return;
if (win)
{
int check = Rnd.get(100);
if (check <= 5)
PenaltyMonster();
else
{
_fisher.sendPacket(SystemMessageId.YOU_CAUGHT_SOMETHING);
_fisher.addItem("Fishing", _fishId, 1, null, true);
}
}
_fisher.endFishing(win);
_fisher = null;
}
protected void aiTask()
{
if (_thinking)
return;
_thinking = true;
_time--;
try
{
if (_mode == 1)
{
if (_deceptiveMode == 0)
_fishCurHp += (int) _regenHp;
}
else
{
if (_deceptiveMode == 1)
_fishCurHp += (int) _regenHp;
}
if (_stop == 0)
{
_stop = 1;
int check = Rnd.get(100);
if (check >= 70)
_mode = _mode == 0 ? 1 : 0;
if (_isUpperGrade)
{
check = Rnd.get(100);
if (check >= 90)
_deceptiveMode = _deceptiveMode == 0 ? 1 : 0;
}
}
else
_stop--;
}
finally
{
_thinking = false;
if (_anim != 0)
_fisher.broadcastPacket(new ExFishingHpRegen(_fisher, _time, _fishCurHp, _mode, 0, _anim, 0, _deceptiveMode));
else
_fisher.sendPacket(new ExFishingHpRegen(_fisher, _time, _fishCurHp, _mode, 0, _anim, 0, _deceptiveMode));
}
}
public void useRealing(int dmg, int pen)
{
_anim = 2;
if (Rnd.get(100) > 90)
{
_fisher.sendPacket(SystemMessageId.FISH_RESISTED_ATTEMPT_TO_BRING_IT_IN);
_goodUse = 0;
changeHp(0, pen);
return;
}
if (_fisher == null)
return;
if (_mode == 1)
{
if (_deceptiveMode == 0)
{
// Reeling is successful, Damage: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.REELING_SUCCESFUL_S1_DAMAGE).addNumber(dmg));
if (pen == 50)
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.REELING_SUCCESSFUL_PENALTY_S1).addNumber(pen));
_goodUse = 1;
changeHp(dmg, pen);
}
else
{
// Reeling failed, Damage: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.FISH_RESISTED_REELING_S1_HP_REGAINED).addNumber(dmg));
_goodUse = 2;
changeHp(-dmg, pen);
}
}
else
{
if (_deceptiveMode == 0)
{
// Reeling failed, Damage: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.FISH_RESISTED_REELING_S1_HP_REGAINED).addNumber(dmg));
_goodUse = 2;
changeHp(-dmg, pen);
}
else
{
// Reeling is successful, Damage: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.REELING_SUCCESFUL_S1_DAMAGE).addNumber(dmg));
if (pen == 50)
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.REELING_SUCCESSFUL_PENALTY_S1).addNumber(pen));
_goodUse = 1;
changeHp(dmg, pen);
}
}
}
public void usePomping(int dmg, int pen)
{
_anim = 1;
if (Rnd.get(100) > 90)
{
_fisher.sendPacket(SystemMessageId.FISH_RESISTED_ATTEMPT_TO_BRING_IT_IN);
_goodUse = 0;
changeHp(0, pen);
return;
}
if (_fisher == null)
return;
if (_mode == 0)
{
if (_deceptiveMode == 0)
{
// Pumping is successful. Damage: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.PUMPING_SUCCESFUL_S1_DAMAGE).addNumber(dmg));
if (pen == 50)
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.PUMPING_SUCCESSFUL_PENALTY_S1).addNumber(pen));
_goodUse = 1;
changeHp(dmg, pen);
}
else
{
// Pumping failed, Regained: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.FISH_RESISTED_PUMPING_S1_HP_REGAINED).addNumber(dmg));
_goodUse = 2;
changeHp(-dmg, pen);
}
}
else
{
if (_deceptiveMode == 0)
{
// Pumping failed, Regained: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.FISH_RESISTED_PUMPING_S1_HP_REGAINED).addNumber(dmg));
_goodUse = 2;
changeHp(-dmg, pen);
}
else
{
// Pumping is successful. Damage: $s1
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.PUMPING_SUCCESFUL_S1_DAMAGE).addNumber(dmg));
if (pen == 50)
_fisher.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.PUMPING_SUCCESSFUL_PENALTY_S1).addNumber(pen));
_goodUse = 1;
changeHp(dmg, pen);
}
}
}
private void PenaltyMonster()
{
int lvl = (int) Math.round(_fisher.getLevel() * 0.1);
int npcid;
_fisher.sendPacket(SystemMessageId.YOU_CAUGHT_SOMETHING_SMELLY_THROW_IT_BACK);
switch (lvl)
{
case 0:
case 1:
npcid = 18319;
break;
case 2:
npcid = 18320;
break;
case 3:
npcid = 18321;
break;
case 4:
npcid = 18322;
break;
case 5:
npcid = 18323;
break;
case 6:
npcid = 18324;
break;
case 7:
npcid = 18325;
break;
case 8:
npcid = 18326;
break;
default:
npcid = 18319;
break;
}
L2NpcTemplate temp;
temp = NpcTable.getInstance().getTemplate(npcid);
if (temp != null)
{
try
{
L2Spawn spawn = new L2Spawn(temp);
spawn.setLocx(_fisher.getX());
spawn.setLocy(_fisher.getY());
spawn.setLocz(_fisher.getZ() + 20);
spawn.setHeading(_fisher.getHeading());
spawn.stopRespawn();
((L2PenaltyMonsterInstance) spawn.doSpawn().scheduleDespawn(3 * 60 * 1000)).setPlayerToKill(_fisher);
}
catch (Exception e)
{
// Nothing
}
}
}
}