/*
* 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 gnu.trove.TIntIntHashMap;
import java.util.ArrayList;
import java.util.List;
import javolution.util.FastList;
import javolution.util.FastMap;
import com.l2jserver.Config;
import com.l2jserver.gameserver.ThreadPoolManager;
import com.l2jserver.gameserver.model.actor.L2Character;
import com.l2jserver.gameserver.model.actor.instance.L2PcInstance;
import com.l2jserver.gameserver.model.entity.BlockCheckerEngine;
import com.l2jserver.gameserver.model.entity.TvTEvent;
import com.l2jserver.gameserver.model.itemcontainer.PcInventory;
import com.l2jserver.gameserver.model.olympiad.OlympiadManager;
import com.l2jserver.gameserver.network.SystemMessageId;
import com.l2jserver.gameserver.network.serverpackets.ExCubeGameAddPlayer;
import com.l2jserver.gameserver.network.serverpackets.ExCubeGameChangeTeam;
import com.l2jserver.gameserver.network.serverpackets.ExCubeGameRemovePlayer;
import com.l2jserver.gameserver.network.serverpackets.L2GameServerPacket;
import com.l2jserver.gameserver.network.serverpackets.SystemMessage;
/**
* @author BiggBoss
*/
public final class HandysBlockCheckerManager
{
/*
* This class manage the player add/remove, team change and
* event arena status, as the clearance of the participants
* list or liberate the arena
*/
// All the participants and their team classifed by arena
private static ArenaParticipantsHolder[] _arenaPlayers;
// Arena votes to start the game
private static TIntIntHashMap _arenaVotes = new TIntIntHashMap();
// Arena Status, True = is being used, otherwise, False
private static FastMap _arenaStatus;
// Registration request penalty (10 seconds)
private static FastList _registrationPenalty = new FastList();
/**
* Return the number of event-start votes for the spcified
* arena id
* @param arenaId
* @return int (number of votes)
*/
public synchronized int getArenaVotes(int arenaId)
{
return _arenaVotes.get(arenaId);
}
/**
* Add a new vote to start the event for the specified
* arena id
* @param arena
*/
public synchronized void increaseArenaVotes(int arena)
{
int newVotes = _arenaVotes.get(arena) + 1;
ArenaParticipantsHolder holder = _arenaPlayers[arena];
if(newVotes > holder.getAllPlayers().size() / 2 && !holder.getEvent().isStarted())
{
clearArenaVotes(arena);
if(holder.getBlueTeamSize() == 0 || holder.getRedTeamSize() == 0)
return;
if(Config.HBCE_FAIR_PLAY)
holder.checkAndShuffle();
ThreadPoolManager.getInstance().executeTask(holder.getEvent().new StartEvent());
}
else
{
_arenaVotes.put(arena, newVotes);
}
}
/**
* Will clear the votes queue (of event start) for the
* specified arena id
* @param arena
*/
public synchronized void clearArenaVotes(int arena)
{
_arenaVotes.put(arena, 0);
}
private HandysBlockCheckerManager()
{
// Initialize arena status
if(_arenaStatus == null)
{
_arenaStatus = new FastMap();
_arenaStatus.put(0, false);
_arenaStatus.put(1, false);
_arenaStatus.put(2, false);
_arenaStatus.put(3, false);
}
}
/**
* Returns the players holder
* @param arena
* @return ArenaParticipantsHolder
*/
public ArenaParticipantsHolder getHolder(int arena)
{
return _arenaPlayers[arena];
}
/**
* Initializes the participants holder
*/
public void startUpParticipantsQueue()
{
_arenaPlayers = new ArenaParticipantsHolder[4];
for(int i = 0; i < 4; ++i)
{
_arenaPlayers[i] = new ArenaParticipantsHolder(i);
}
}
/**
* Add the player to the specified arena (throught the specified
* arena manager) and send the needed server -> client packets
* @param player
* @param arenaId
* @return
*/
public boolean addPlayerToArena(L2PcInstance player, int arenaId)
{
ArenaParticipantsHolder holder = _arenaPlayers[arenaId];
synchronized(holder)
{
boolean isRed;
for(int i = 0; i < 4; i++)
{
if(_arenaPlayers[i].getAllPlayers().contains(player))
{
SystemMessage msg = SystemMessage.getSystemMessage(SystemMessageId.C1_IS_ALREADY_REGISTERED_ON_THE_MATCH_WAITING_LIST);
msg.addCharName(player);
player.sendPacket(msg);
return false;
}
}
if(player.isCursedWeaponEquipped())
{
player.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.CANNOT_REGISTER_PROCESSING_CURSED_WEAPON));
return false;
}
if(TvTEvent.isPlayerParticipant(player.getObjectId()) || player.isInOlympiadMode())
{
player.sendMessage("Couldnt register you due other event participation");
return false;
}
if(OlympiadManager.getInstance().isRegistered(player))
{
OlympiadManager.getInstance().unRegisterNoble(player);
player.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.COLISEUM_OLYMPIAD_KRATEIS_APPLICANTS_CANNOT_PARTICIPATE));
}
/*
if(UnderGroundColiseum.getInstance().isRegisteredPlayer(player))
{
UngerGroundColiseum.getInstance().removeParticipant(player);
player.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.COLISEUM_OLYMPIAD_KRATEIS_APPLICANTS_CANNOT_PARTICIPATE));
}
if(KrateiCubeManager.getInstance().isRegisteredPlayer(player))
{
KrateiCubeManager.getInstance().removeParticipant(player);
player.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.COLISEUM_OLYMPIAD_KRATEIS_APPLICANTS_CANNOT_PARTICIPATE));
}
*/
if(_registrationPenalty.contains(player.getObjectId()))
{
player.sendPacket(SystemMessage.getSystemMessage(SystemMessageId.CANNOT_REQUEST_REGISTRATION_10_SECS_AFTER));
return false;
}
if(holder.getBlueTeamSize() < holder.getRedTeamSize())
{
holder.addPlayer(player, 1);
isRed = false;
}
else
{
holder.addPlayer(player, 0);
isRed = true;
}
holder.broadCastPacketToTeam(new ExCubeGameAddPlayer(player, isRed));
return true;
}
}
/**
* Will remove the specified player from the specified
* team and arena and will send the needed packet to all
* his team mates / enemy team mates
* @param player
* @param arenaId
* @param team
*/
public void removePlayer(L2PcInstance player, int arenaId, int team)
{
ArenaParticipantsHolder holder = _arenaPlayers[arenaId];
synchronized(holder)
{
boolean isRed = team == 0 ? true : false;
holder.removePlayer(player, team);
holder.broadCastPacketToTeam(new ExCubeGameRemovePlayer(player, isRed));
// End event if theres an empty team
int teamSize = isRed? holder.getRedTeamSize() : holder.getBlueTeamSize();
if(teamSize == 0)
holder.getEvent().endEventAbnormally();
Integer objId = player.getObjectId();
if(!_registrationPenalty.contains(objId))
_registrationPenalty.add(objId);
schedulePenaltyRemoval(objId);
}
}
/**
* Will change the player from one team to other (if possible)
* and will send the needed packets
* @param player
* @param arena
* @param team
*/
public void changePlayerToTeam(L2PcInstance player, int arena, int team)
{
ArenaParticipantsHolder holder = _arenaPlayers[arena];
synchronized(holder)
{
boolean isFromRed = holder._redPlayers.contains(player);
if(isFromRed && holder.getBlueTeamSize() == 6)
{
player.sendMessage("The team is full");
return;
}
else if(!isFromRed && holder.getRedTeamSize() == 6)
{
player.sendMessage("The team is full");
return;
}
int futureTeam = isFromRed? 1 : 0;
holder.addPlayer(player, futureTeam);
if(isFromRed)
holder.removePlayer(player, 0);
else
holder.removePlayer(player, 1);
holder.broadCastPacketToTeam(new ExCubeGameChangeTeam(player, isFromRed));
}
}
/**
* Will erase all participants from the specified holder
* @param arenaId
*/
public synchronized void clearPaticipantQueueByArenaId(int arenaId)
{
_arenaPlayers[arenaId].clearPlayers();
}
/**
* Returns true if arena is holding an event at this momment
* @param arenaId
* @return boolean
*/
public boolean arenaIsBeingUsed(int arenaId)
{
if(arenaId < 0 || arenaId > 3)
return false;
return _arenaStatus.get(arenaId);
}
/**
* Set the specified arena as being used
* @param arenaId
*/
public void setArenaBeingUsed(int arenaId)
{
_arenaStatus.put(arenaId, true);
}
/**
* Set as free the specified arena for future
* events
* @param arenaId
*/
public void setArenaFree(int arenaId)
{
_arenaStatus.put(arenaId, false);
}
/**
* Called when played logs out while participating
* in Block Checker Event
* @param player
*/
public void onDisconnect(L2PcInstance player)
{
int arena = player.getBlockCheckerArena();
int team = getHolder(arena).getPlayerTeam(player);
HandysBlockCheckerManager.getInstance().removePlayer(player, arena, team);
if (player.getTeam() > 0 )
{
player.stopAllEffects();
// Remove team aura
player.setTeam(0);
// Remove the event items
PcInventory inv = player.getInventory();
if(inv.getItemByItemId(13787) != null)
{
long count = inv.getInventoryItemCount(13787, 0);
inv.destroyItemByItemId("Handys Block Checker", 13787, count, player, player);
}
if(inv.getItemByItemId(13788) != null)
{
long count = inv.getInventoryItemCount(13788, 0);
inv.destroyItemByItemId("Handys Block Checker", 13788, count, player, player);
}
player.setInsideZone(L2Character.ZONE_PVP, false);
// Teleport Back
player.teleToLocation(-57478, -60367, -2370);
}
}
public static HandysBlockCheckerManager getInstance()
{
return SingletonHolder._instance;
}
private static class SingletonHolder
{
private static HandysBlockCheckerManager _instance = new HandysBlockCheckerManager();
}
public class ArenaParticipantsHolder
{
int _arena;
List _redPlayers;
List _bluePlayers;
BlockCheckerEngine _engine;
public ArenaParticipantsHolder(int arena)
{
_arena = arena;
_redPlayers = new ArrayList(6);
_bluePlayers = new ArrayList(6);
_engine = new BlockCheckerEngine(this, _arena);
}
public List getRedPlayers()
{
return _redPlayers;
}
public List getBluePlayers()
{
return _bluePlayers;
}
public ArrayList getAllPlayers()
{
ArrayList all = new ArrayList(12);
all.addAll(_redPlayers);
all.addAll(_bluePlayers);
return all;
}
public void addPlayer(L2PcInstance player, int team)
{
if(team == 0)
_redPlayers.add(player);
else
_bluePlayers.add(player);
}
public void removePlayer(L2PcInstance player, int team)
{
if(team == 0)
_redPlayers.remove(player);
else
_bluePlayers.remove(player);
}
public int getPlayerTeam(L2PcInstance player)
{
if(_redPlayers.contains(player))
return 0;
else if(_bluePlayers.contains(player))
return 1;
else
return -1;
}
public int getRedTeamSize()
{
return _redPlayers.size();
}
public int getBlueTeamSize()
{
return _bluePlayers.size();
}
public void broadCastPacketToTeam(L2GameServerPacket packet)
{
for(L2PcInstance p : _redPlayers)
p.sendPacket(packet);
for(L2PcInstance p : _bluePlayers)
p.sendPacket(packet);
}
public void clearPlayers()
{
_redPlayers.clear();
_bluePlayers.clear();
}
public BlockCheckerEngine getEvent()
{
return _engine;
}
public void updateEvent()
{
_engine.updatePlayersOnStart(this);
}
private void checkAndShuffle()
{
int redSize = _redPlayers.size();
int blueSize = _bluePlayers.size();
if(redSize > blueSize + 1)
{
broadCastPacketToTeam(SystemMessage.getSystemMessage(SystemMessageId.TEAM_ADJUSTED_BECAUSE_WRONG_POPULATION_RATIO));
int needed = redSize - (blueSize + 1);
for(int i = 0; i < needed+1; i++)
{
L2PcInstance plr = _redPlayers.get(i);
if(plr == null) continue;
changePlayerToTeam(plr, this._arena, 1);
}
}
else if(blueSize > redSize + 1)
{
broadCastPacketToTeam(SystemMessage.getSystemMessage(SystemMessageId.TEAM_ADJUSTED_BECAUSE_WRONG_POPULATION_RATIO));
int needed = blueSize - (redSize + 1);
for(int i = 0; i < needed+1; i++)
{
L2PcInstance plr = _bluePlayers.get(i);
if(plr == null) continue;
changePlayerToTeam(plr, this._arena, 0);
}
}
}
}
private void schedulePenaltyRemoval(int objId)
{
ThreadPoolManager.getInstance().scheduleGeneral(new PenaltyRemove(objId), 10000);
}
private class PenaltyRemove implements Runnable
{
Integer objectId;
public PenaltyRemove(Integer id)
{
objectId = id;
}
@Override
public void run()
{
try
{
_registrationPenalty.remove(objectId);
}
catch(Exception e)
{
e.printStackTrace();
}
}
}
}