/* This file is part of the OdinMS Maple Story Server Copyright (C) 2008 Patrick Huy Matthias Butz Jan Christian Meyer This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation version 3 as published by the Free Software Foundation. You may not use, modify or distribute this program under any other version of the GNU Affero General Public License. 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 Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . */ package scripting.event; import java.io.File; import java.sql.PreparedStatement; import java.sql.SQLException; import tools.Pair; import java.util.ArrayList; import java.util.LinkedList; import java.util.List; import java.util.HashMap; import java.util.Map; import java.util.HashSet; import java.util.Set; import java.util.Iterator; import java.util.Properties; import java.util.concurrent.locks.ReentrantReadWriteLock; import java.util.concurrent.locks.ReentrantReadWriteLock.ReadLock; import java.util.concurrent.locks.ReentrantReadWriteLock.WriteLock; import javax.script.ScriptException; import net.server.world.MapleParty; import net.server.world.MaplePartyCharacter; import provider.MapleDataProviderFactory; import server.MaplePortal; import server.TimerManager; import server.MapleStatEffect; import server.expeditions.MapleExpedition; import server.life.MapleMonster; import server.maps.MapleMap; import server.maps.MapleMapFactory; import tools.DatabaseConnection; import client.MapleCharacter; import client.SkillFactory; import client.Skill; import constants.ItemConstants; import constants.ServerConstants; import java.awt.Point; import java.sql.Connection; import java.util.concurrent.ScheduledFuture; import java.util.logging.Level; import java.util.logging.Logger; import scripting.AbstractPlayerInteraction; import server.MapleItemInformationProvider; import server.life.MapleLifeFactory; import server.life.MapleNPC; import tools.MaplePacketCreator; /** * * @author Matze * @author Ronan */ public class EventInstanceManager { private Map chars = new HashMap<>(); private int leaderId = -1; private List mobs = new LinkedList<>(); private Map killCount = new HashMap<>(); private EventManager em; private MapleMapFactory mapFactory; private String name; private Properties props = new Properties(); private long timeStarted = 0; private long eventTime = 0; private MapleExpedition expedition = null; private List mapIds = new LinkedList(); private List isInstanced = new LinkedList(); private final ReentrantReadWriteLock mutex = new ReentrantReadWriteLock(); private final ReadLock rL = mutex.readLock(); private final WriteLock wL = mutex.writeLock(); private ScheduledFuture event_schedule = null; private boolean disposed = false; private boolean eventCleared = false; // multi-leveled PQ rewards! private Map> collectionSet = new HashMap<>(ServerConstants.MAX_EVENT_LEVELS); private Map> collectionQty = new HashMap<>(ServerConstants.MAX_EVENT_LEVELS); private Map collectionExp = new HashMap<>(ServerConstants.MAX_EVENT_LEVELS); // Exp/Meso rewards by CLEAR on a stage private List onMapClearExp = new ArrayList<>(); private List onMapClearMeso = new ArrayList<>(); // registers player status on an event (null on this Map structure equals to 0) private Map playerGrid = new HashMap<>(); // registers all opened gates on the event. Will help late characters to encounter next stages gates already opened private Map> openedGates = new HashMap<>(); // forces deletion of items not supposed to be held outside of the event, dealt on a player's leaving moment. private Set exclusiveItems = new HashSet<>(); public EventInstanceManager(EventManager em, String name) { this.em = em; this.name = name; mapFactory = new MapleMapFactory(this, MapleDataProviderFactory.getDataProvider(new File(System.getProperty("wzpath") + "/Map.wz")), MapleDataProviderFactory.getDataProvider(new File(System.getProperty("wzpath") + "/String.wz")), (byte) 0, (byte) 1);//Fk this mapFactory.setChannel(em.getChannelServer().getId()); } public void setName(String name) { this.name = name; } public EventManager getEm() { return em; } public int getEventPlayersJobs() { //Bits -> 0: BEGINNER 1: WARRIOR 2: MAGICIAN // 3: BOWMAN 4: THIEF 5: PIRATE int mask = 0; for(MapleCharacter chr: getPlayers()) { mask |= (1 << chr.getJob().getJobNiche()); } return mask; } public void applyEventPlayersItemBuff(int itemId) { List players = getPlayerList(); MapleStatEffect mse = MapleItemInformationProvider.getInstance().getItemEffect(itemId); if(mse != null) { for (MapleCharacter player: players) { mse.applyTo(player); } } } public void applyEventPlayersSkillBuff(int skillId) { applyEventPlayersSkillBuff(skillId, Integer.MAX_VALUE); } public void applyEventPlayersSkillBuff(int skillId, int skillLv) { List players = getPlayerList(); Skill skill = SkillFactory.getSkill(skillId); if(skill != null) { MapleStatEffect mse = skill.getEffect(Math.min(skillLv, skill.getMaxLevel())); if(mse != null) { for (MapleCharacter player: players) { mse.applyTo(player); } } } } public void giveEventPlayersExp(int gain) { giveEventPlayersExp(gain, -1); } public void giveEventPlayersExp(int gain, int mapId) { if(gain == 0) return; List players = getPlayerList(); if(mapId == -1) { for(MapleCharacter mc: players) { mc.gainExp(gain * mc.getExpRate(), true, true); } } else { for(MapleCharacter mc: players) { if(mc.getMapId() == mapId) mc.gainExp(gain * mc.getExpRate(), true, true); } } } public void giveEventPlayersMeso(int gain) { giveEventPlayersMeso(gain, -1); } public void giveEventPlayersMeso(int gain, int mapId) { if(gain == 0) return; List players = getPlayerList(); if(mapId == -1) { for(MapleCharacter mc: players) { mc.gainMeso(gain * mc.getMesoRate()); } } else { for(MapleCharacter mc: players) { if(mc.getMapId() == mapId) mc.gainMeso(gain * mc.getMesoRate()); } } } public void registerPlayer(MapleCharacter chr) { if (chr == null || !chr.isLoggedin()){ return; } try { wL.lock(); try { chars.put(chr.getId(), chr); } finally { wL.unlock(); } chr.setEventInstance(this); em.getIv().invokeFunction("playerEntry", this, chr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public void exitPlayer(MapleCharacter chr) { if (chr == null || !chr.isLoggedin()){ return; } try { unregisterPlayer(chr); em.getIv().invokeFunction("playerExit", this, chr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public void dropMessage(int type, String message) { if(!eventCleared) { for (MapleCharacter chr : getPlayers()) { chr.dropMessage(type, message); } } } public void restartEventTimer(long time) { stopEventTimer(); startEventTimer(time); } public void startEventTimer(long time) { timeStarted = System.currentTimeMillis(); eventTime = time; for(MapleCharacter chr: getPlayers()) { chr.announce(MaplePacketCreator.getClock((int) (time / 1000))); } event_schedule = TimerManager.getInstance().schedule(new Runnable() { public void run() { try { dismissEventTimer(); em.getIv().invokeFunction("scheduledTimeout", EventInstanceManager.this); } catch (ScriptException | NoSuchMethodException ex) { Logger.getLogger(EventManager.class.getName()).log(Level.SEVERE, null, ex); } } }, time); } public void addEventTimer(long time) { if(event_schedule != null) { if(event_schedule.cancel(false)) { long nextTime = getTimeLeft() + time; eventTime += time; event_schedule = TimerManager.getInstance().schedule(new Runnable() { public void run() { try { dismissEventTimer(); em.getIv().invokeFunction("scheduledTimeout", EventInstanceManager.this); } catch (ScriptException | NoSuchMethodException ex) { Logger.getLogger(EventManager.class.getName()).log(Level.SEVERE, null, ex); } } }, nextTime); } } else { startEventTimer(time); } } private void dismissEventTimer() { for(MapleCharacter chr: getPlayers()) { chr.getClient().getSession().write(MaplePacketCreator.removeClock()); } event_schedule = null; eventTime = 0; timeStarted = 0; } public void stopEventTimer() { if(event_schedule != null) { event_schedule.cancel(false); event_schedule = null; } dismissEventTimer(); } public boolean isTimerStarted() { return eventTime > 0 && timeStarted > 0; } public long getTimeLeft() { return eventTime - (System.currentTimeMillis() - timeStarted); } public void registerParty(MapleCharacter chr) { if(chr.isPartyLeader()) { registerParty(chr.getParty(), chr.getMap()); } } public void registerParty(MapleParty party, MapleMap map) { for (MaplePartyCharacter pc : party.getEligibleMembers()) { MapleCharacter c = map.getCharacterById(pc.getId()); registerPlayer(c); } } public void registerExpedition(MapleExpedition exped) { expedition = exped; registerExpeditionTeam(exped, exped.getRecruitingMap().getId()); } private void registerExpeditionTeam(MapleExpedition exped, int recruitMap) { expedition = exped; for(MapleCharacter chr: exped.getMembers()) { if(chr.getMapId() == recruitMap) registerPlayer(chr); } } public void unregisterPlayer(MapleCharacter chr) { try { em.getIv().invokeFunction("playerUnregistered", EventInstanceManager.this, chr); } catch (ScriptException | NoSuchMethodException ex) { Logger.getLogger(EventManager.class.getName()).log(Level.SEVERE, null, ex); } wL.lock(); try { chars.remove(chr.getId()); } finally { wL.unlock(); } gridRemove(chr); dropExclusiveItems(chr); chr.setEventInstance(null); } public int getPlayerCount() { rL.lock(); try { return chars.size(); } finally { rL.unlock(); } } public List getPlayers() { rL.lock(); try { return new ArrayList<>(chars.values()); } finally { rL.unlock(); } } private List getPlayerList() { rL.lock(); try { return new LinkedList<>(chars.values()); } finally { rL.unlock(); } } public void registerMonster(MapleMonster mob) { if (!mob.getStats().isFriendly()) { //We cannot register moon bunny mobs.add(mob); } } public void movePlayer(MapleCharacter chr) { try { em.getIv().invokeFunction("moveMap", this, chr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public void changedMap(MapleCharacter chr, int mapId) { // optional try { em.getIv().invokeFunction("changedMap", this, chr, mapId); } catch (ScriptException | NoSuchMethodException ex) {} } public void afterChangedMap(MapleCharacter chr, int mapId) { // optional try { em.getIv().invokeFunction("afterChangedMap", this, chr, mapId); } catch (ScriptException | NoSuchMethodException ex) {} } public void changedLeader(MapleCharacter ldr) { try { em.getIv().invokeFunction("changedLeader", this, ldr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } leaderId = ldr.getId(); } public void monsterKilled(MapleMonster mob) { mobs.remove(mob); try { em.getIv().invokeFunction("monsterKilled", mob, this); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } if (mobs.isEmpty()) { try { em.getIv().invokeFunction("allMonstersDead", this); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } } public void friendlyKilled(MapleMonster mob) { try { em.getIv().invokeFunction("friendlyKilled", mob, this); } catch (ScriptException | NoSuchMethodException ex) { //optional } } public void playerKilled(MapleCharacter chr) { try { em.getIv().invokeFunction("playerDead", this, chr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public boolean revivePlayer(MapleCharacter chr) { try { Object b = em.getIv().invokeFunction("playerRevive", this, chr); if (b instanceof Boolean) { return (Boolean) b; } } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } return true; } public void playerDisconnected(MapleCharacter chr) { try { em.getIv().invokeFunction("playerDisconnected", this, chr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } /** * * @param chr * @param mob */ public void monsterKilled(MapleCharacter chr, MapleMonster mob) { try { Integer kc = killCount.get(chr); int inc = ((Double) em.getIv().invokeFunction("monsterValue", this, mob.getId())).intValue(); if (kc == null) { kc = inc; } else { kc += inc; } killCount.put(chr, kc); if (expedition != null){ expedition.monsterKilled(chr, mob); } } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public int getKillCount(MapleCharacter chr) { Integer kc = killCount.get(chr); return (kc == null) ? 0 : kc; } public void cancelSchedule() { if(event_schedule != null) { event_schedule.cancel(false); event_schedule = null; } } public void dispose() { try { em.getIv().invokeFunction("dispose", this); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } wL.lock(); try { for(MapleCharacter chr: chars.values()) chr.setEventInstance(null); chars.clear(); mobs.clear(); mapFactory.dispose(); mapFactory = null; } finally { wL.unlock(); } cancelSchedule(); killCount.clear(); disposeExpedition(); if(!eventCleared) em.disposeInstance(name); em = null; } public MapleMapFactory getMapFactory() { return mapFactory; } public void schedule(final String methodName, long delay) { TimerManager.getInstance().schedule(new Runnable() { @Override public void run() { if(em == null) return; try { em.getIv().invokeFunction(methodName, EventInstanceManager.this); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } }, delay); } public String getName() { return name; } public void saveWinner(MapleCharacter chr) { try { Connection con = DatabaseConnection.getConnection(); try (PreparedStatement ps = con.prepareStatement("INSERT INTO eventstats (event, instance, characterid, channel) VALUES (?, ?, ?, ?)")) { ps.setString(1, em.getName()); ps.setString(2, getName()); ps.setInt(3, chr.getId()); ps.setInt(4, chr.getClient().getChannel()); ps.executeUpdate(); } con.close(); } catch (SQLException ex) { ex.printStackTrace(); } } public MapleMap getMapInstance(int mapId) { MapleMap map = mapFactory.getMap(mapId); map.setEventInstance(this); if (!mapFactory.isMapLoaded(mapId)) { if (em.getProperty("shuffleReactors") != null && em.getProperty("shuffleReactors").equals("true")) { map.shuffleReactors(); } } return map; } public void setIntProperty(String key, Integer value) { setProperty(key, value); } public void setProperty(String key, Integer value) { setProperty(key, "" + value); } public void setProperty(String key, String value) { props.setProperty(key, value); } public Object setProperty(String key, String value, boolean prev) { return props.setProperty(key, value); } public String getProperty(String key) { return props.getProperty(key); } public Properties getProperties() { return props; } public int getIntProperty(String key) { return Integer.parseInt(props.getProperty(key)); } public void leftParty(MapleCharacter chr) { try { em.getIv().invokeFunction("leftParty", this, chr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public void disbandParty() { try { em.getIv().invokeFunction("disbandParty", this); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public void clearPQ() { try { em.getIv().invokeFunction("clearPQ", this); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public void removePlayer(MapleCharacter chr) { try { em.getIv().invokeFunction("playerExit", this, chr); } catch (ScriptException | NoSuchMethodException ex) { ex.printStackTrace(); } } public boolean isLeader(MapleCharacter chr) { return (chr.getParty().getLeaderId() == chr.getId()); } public boolean isEventLeader(MapleCharacter chr) { return (chr.getId() == getLeaderId()); } public final MapleMap getInstanceMap(final int mapid) { //gets instance map from the channelserv if (disposed) { return getMapFactory().getMap(mapid); } mapIds.add(mapid); isInstanced.add(false); return getMapFactory().getMap(mapid); } public final boolean disposeIfPlayerBelow(final byte size, final int towarp) { if (disposed) { return true; } if(chars == null) { return false; } MapleMap map = null; if (towarp > 0) { map = this.getMapFactory().getMap(towarp); } List players = getPlayerList(); try { if (players.size() < size) { for (MapleCharacter chr : players) { if (chr == null) { continue; } unregisterPlayer(chr); if (towarp > 0) { chr.changeMap(map, map.getPortal(0)); } } dispose(); return true; } } catch (Exception ex) { ex.printStackTrace(); } return false; } public void spawnNpc(int npcId, Point pos, MapleMap map) { MapleNPC npc = MapleLifeFactory.getNPC(npcId); if (npc != null) { npc.setPosition(pos); npc.setCy(pos.y); npc.setRx0(pos.x + 50); npc.setRx1(pos.x - 50); npc.setFh(map.getFootholds().findBelow(pos).getId()); map.addMapObject(npc); map.broadcastMessage(MaplePacketCreator.spawnNPC(npc)); } } public MapleMonster getMonster(int mid) { return(MapleLifeFactory.getMonster(mid)); } private List convertToIntegerArray(List list) { List intList = new ArrayList<>(); for(Double d: list) intList.add(d.intValue()); return intList; } public void setEventClearStageExp(List gain) { onMapClearExp.clear(); onMapClearExp.addAll(convertToIntegerArray(gain)); } public void setEventClearStageMeso(List gain) { onMapClearMeso.clear(); onMapClearMeso.addAll(convertToIntegerArray(gain)); } public Integer getClearStageExp(int stage) { //stage counts from ONE. if(stage > onMapClearExp.size()) return 0; return onMapClearExp.get(stage - 1); } public Integer getClearStageMeso(int stage) { //stage counts from ONE. if(stage > onMapClearMeso.size()) return 0; return onMapClearMeso.get(stage - 1); } public List getClearStageBonus(int stage) { List list = new ArrayList<>(); list.add(getClearStageExp(stage)); list.add(getClearStageMeso(stage)); return list; } private void dropExclusiveItems(MapleCharacter chr) { AbstractPlayerInteraction api = chr.getClient().getAbstractPlayerInteraction(); for(Integer item: exclusiveItems) { api.removeAll(item); } } public final void setExclusiveItems(List items) { List exclusive = convertToIntegerArray(items); wL.lock(); try { for(Integer item: exclusive) { exclusiveItems.add(item); } } finally { wL.unlock(); } } public final void setEventRewards(List rwds, List qtys, int expGiven) { setEventRewards(1, rwds, qtys, expGiven); } public final void setEventRewards(List rwds, List qtys) { setEventRewards(1, rwds, qtys); } public final void setEventRewards(int eventLevel, List rwds, List qtys) { setEventRewards(eventLevel, rwds, qtys, 0); } public final void setEventRewards(int eventLevel, List rwds, List qtys, int expGiven) { // fixed EXP will be rewarded at the same time the random item is given if(eventLevel <= 0 || eventLevel > ServerConstants.MAX_EVENT_LEVELS) return; eventLevel--; //event level starts from 1 List rewardIds = convertToIntegerArray(rwds); List rewardQtys = convertToIntegerArray(qtys); //rewardsSet and rewardsQty hold temporary values wL.lock(); try { collectionSet.put(eventLevel, rewardIds); collectionQty.put(eventLevel, rewardQtys); collectionExp.put(eventLevel, expGiven); } finally { wL.unlock(); } } private byte getRewardListRequirements(int level) { if(level >= collectionSet.size()) return 0; byte rewardTypes = 0; List list = collectionSet.get(level); for (Integer itemId : list) { rewardTypes |= (1 << ItemConstants.getInventoryType(itemId).getType()); } return rewardTypes; } private boolean hasRewardSlot(MapleCharacter player, int eventLevel) { byte listReq = getRewardListRequirements(eventLevel); //gets all types of items present in the event reward list //iterating over all valid inventory types for(byte type = 1; type <= 5; type++) { if((listReq >> type) % 2 == 1 && !player.hasEmptySlot(type)) return false; } return true; } public final boolean giveEventReward(MapleCharacter player) { return giveEventReward(player, 1); } //gives out EXP & a random item in a similar fashion of when clearing KPQ, LPQ, etc. public final boolean giveEventReward(MapleCharacter player, int eventLevel) { List rewardsSet, rewardsQty; Integer rewardExp; rL.lock(); try { eventLevel--; //event level starts counting from 1 if(eventLevel >= collectionSet.size()) return true; rewardsSet = collectionSet.get(eventLevel); rewardsQty = collectionQty.get(eventLevel); rewardExp = collectionExp.get(eventLevel); } finally { rL.unlock(); } if(rewardExp == null) rewardExp = 0; if(rewardsSet == null || rewardsSet.isEmpty()) { if(rewardExp > 0) player.gainExp(rewardExp); return true; } if(!hasRewardSlot(player, eventLevel)) return false; AbstractPlayerInteraction api = player.getClient().getAbstractPlayerInteraction(); int rnd = (int)Math.floor(Math.random() * rewardsSet.size()); api.gainItem(rewardsSet.get(rnd), rewardsQty.get(rnd).shortValue()); if(rewardExp > 0) player.gainExp(rewardExp); return true; } private void disposeExpedition() { if (expedition != null) { expedition.dispose(eventCleared); em.getChannelServer().getExpeditions().remove(expedition); expedition = null; } } public final void setEventCleared() { eventCleared = true; em.disposeInstance(name); disposeExpedition(); } public final boolean isEventCleared() { return eventCleared; } private boolean isEventTeamLeaderOn() { for(MapleCharacter chr: getPlayers()) { if(chr.getId() == getLeaderId()) return true; } return false; } public final boolean checkEventTeamLacking(boolean testLeader, int minPlayers) { if(eventCleared && getPlayerCount() > 1) return false; if(!eventCleared && testLeader && !isEventTeamLeaderOn()) return true; if(getPlayerCount() < minPlayers) return true; return false; } public final boolean isEventTeamLackingNow(boolean testLeader, int minPlayers, MapleCharacter quitter) { if(eventCleared) { if(getPlayerCount() <= 1) return true; } else { if(testLeader && getLeaderId() == quitter.getId()) return true; if(getPlayerCount() <= minPlayers) return true; } return false; } public final boolean isEventTeamTogether() { rL.lock(); try { if(chars.size() <= 1) return true; Iterator iterator = chars.values().iterator(); MapleCharacter mc = iterator.next(); int mapId = mc.getMapId(); for (; iterator.hasNext();) { mc = iterator.next(); if(mc.getMapId() != mapId) return false; } return true; } finally { rL.unlock(); } } public final void warpEventTeam(int warpFrom, int warpTo) { List players = getPlayerList(); for (MapleCharacter chr : players) { if(chr.getMapId() == warpFrom) chr.changeMap(warpTo); } } public final void warpEventTeam(int warpTo) { List players = getPlayerList(); for (MapleCharacter chr : players) { chr.changeMap(warpTo); } } public final void warpEventTeamToMapSpawnPoint(int warpFrom, int warpTo, int toSp) { List players = getPlayerList(); for (MapleCharacter chr : players) { if(chr.getMapId() == warpFrom) chr.changeMap(warpTo, toSp); } } public final void warpEventTeamToMapSpawnPoint(int warpTo, int toSp) { List players = getPlayerList(); for (MapleCharacter chr : players) { chr.changeMap(warpTo, toSp); } } public final int getLeaderId() { rL.lock(); try { return leaderId; } finally { rL.unlock(); } } public MapleCharacter getLeader() { rL.lock(); try { return chars.get(leaderId); } finally { rL.unlock(); } } public final void setLeader(MapleCharacter chr) { wL.lock(); try { leaderId = chr.getId(); } finally { wL.unlock(); } } public final void showWrongEffect() { showWrongEffect(getLeader().getMapId()); } public final void showWrongEffect(int mapId) { MapleMap map = getMapInstance(mapId); map.broadcastMessage(MaplePacketCreator.showEffect("quest/party/wrong_kor")); map.broadcastMessage(MaplePacketCreator.playSound("Party1/Failed")); } public final void showClearEffect() { showClearEffect(false); } public final void showClearEffect(boolean hasGate) { showClearEffect(hasGate, getLeader().getMapId()); } public final void showClearEffect(int mapId) { showClearEffect(false, mapId); } public final void showClearEffect(boolean hasGate, int mapId) { showClearEffect(hasGate, mapId, "gate", 2); } public final void showClearEffect(int mapId, String mapObj, int newState) { showClearEffect(true, mapId, mapObj, newState); } public final void showClearEffect(boolean hasGate, int mapId, String mapObj, int newState) { MapleMap map = getMapInstance(mapId); map.broadcastMessage(MaplePacketCreator.showEffect("quest/party/clear")); map.broadcastMessage(MaplePacketCreator.playSound("Party1/Clear")); if(hasGate) { map.broadcastMessage(MaplePacketCreator.environmentChange(mapObj, newState)); wL.lock(); try { openedGates.put(map.getId(), new Pair<>(mapObj, newState)); } finally { wL.unlock(); } } } public final void recoverOpenedGate(MapleCharacter chr, int thisMapId) { rL.lock(); try { if(openedGates.containsKey(thisMapId)) { Pair gateData = openedGates.get(thisMapId); chr.announce(MaplePacketCreator.environmentChange(gateData.getLeft(), gateData.getRight())); } } finally { rL.unlock(); } } public final void giveEventPlayersStageReward(int thisStage) { List list = getClearStageBonus(thisStage); // will give bonus exp & mesos to everyone in the event giveEventPlayersExp(list.get(0)); giveEventPlayersMeso(list.get(1)); } public final void linkToNextStage(int thisStage, String eventFamily, int thisMapId) { giveEventPlayersStageReward(thisStage); thisStage--; //stages counts from ONE, scripts from ZERO MapleMap nextStage = getMapInstance(thisMapId); MaplePortal portal = nextStage.getPortal("next00"); if (portal != null) { portal.setScriptName(eventFamily + thisStage); } } public final void linkPortalToScript(int thisStage, String portalName, String scriptName, int thisMapId) { giveEventPlayersStageReward(thisStage); thisStage--; //stages counts from ONE, scripts from ZERO MapleMap nextStage = getMapInstance(thisMapId); MaplePortal portal = nextStage.getPortal(portalName); if (portal != null) { portal.setScriptName(scriptName); } } // registers a player status in an event public final void gridInsert(MapleCharacter chr, int newStatus) { wL.lock(); try { playerGrid.put(chr.getId(), newStatus); } finally { wL.unlock(); } } // unregisters a player status in an event public final void gridRemove(MapleCharacter chr) { wL.lock(); try { playerGrid.remove(chr.getId()); } finally { wL.unlock(); } } // checks a player status public final int gridCheck(MapleCharacter chr) { rL.lock(); try { Integer i = playerGrid.get(chr.getId()); return (i != null) ? i : -1; } finally { rL.unlock(); } } public final void gridClear() { wL.lock(); try { playerGrid.clear(); } finally { wL.unlock(); } } }