/* * 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 net.sf.l2j.gameserver; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.util.List; import java.util.Map; import java.util.concurrent.ScheduledFuture; import java.util.logging.Logger; import javolution.util.FastList; import javolution.util.FastMap; import net.sf.l2j.Config; import net.sf.l2j.L2DatabaseFactory; import net.sf.l2j.gameserver.ai.CtrlIntention; import net.sf.l2j.gameserver.datatables.ClanTable; import net.sf.l2j.gameserver.datatables.MapRegionTable; import net.sf.l2j.gameserver.datatables.NpcTable; import net.sf.l2j.gameserver.datatables.SpawnTable; import net.sf.l2j.gameserver.model.L2CharPosition; import net.sf.l2j.gameserver.model.L2Clan; import net.sf.l2j.gameserver.model.L2ItemInstance; import net.sf.l2j.gameserver.model.L2Party; import net.sf.l2j.gameserver.model.L2Spawn; import net.sf.l2j.gameserver.model.L2World; import net.sf.l2j.gameserver.model.SpawnListener; import net.sf.l2j.gameserver.model.actor.instance.L2FestivalMonsterInstance; import net.sf.l2j.gameserver.model.actor.instance.L2NpcInstance; import net.sf.l2j.gameserver.model.actor.instance.L2PcInstance; import net.sf.l2j.gameserver.model.base.Experience; import net.sf.l2j.gameserver.network.SystemMessageId; import net.sf.l2j.gameserver.serverpackets.CreatureSay; import net.sf.l2j.gameserver.serverpackets.MagicSkillUse; import net.sf.l2j.gameserver.serverpackets.PledgeShowInfoUpdate; import net.sf.l2j.gameserver.serverpackets.SystemMessage; import net.sf.l2j.gameserver.templates.L2NpcTemplate; import net.sf.l2j.gameserver.templates.StatsSet; import net.sf.l2j.gameserver.util.Util; import net.sf.l2j.util.Rnd; /** * Seven Signs Festival of Darkness Engine * * TODO: * - Archer mobs should target healer characters over other party members. * - ADDED 29 Sep: Players that leave a party during the Seven Signs Festival will now take damage and cannot be healed. * * @author Tempy */ public class SevenSignsFestival implements SpawnListener { protected static final Logger _log = Logger.getLogger(SevenSignsFestival.class.getName()); private static SevenSignsFestival _instance; private static final String GET_CLAN_NAME = "SELECT clan_name FROM clan_data WHERE clan_id = (SELECT clanid FROM characters WHERE char_name = ?)"; /** * These length settings are important! :) * All times are relative to the ELAPSED time (in ms) since a festival begins. * * Festival manager start is the time after the server starts to begin the first * festival cycle. * * The cycle length should ideally be at least 2x longer than the festival length. * This allows ample time for players to sign-up to participate in the festival. * * The intermission is the time between the festival participants being moved * to the "arenas" and the spawning of the first set of mobs. * * The monster swarm time is the time before the monsters swarm to the center of the arena, * after they are spawned. * * The chest spawn time is for when the bonus festival chests spawn, usually * towards the end of the festival. */ public static final long FESTIVAL_SIGNUP_TIME = Config.ALT_FESTIVAL_CYCLE_LENGTH - Config.ALT_FESTIVAL_LENGTH - 60000; // Key Constants \\ private static final int FESTIVAL_MAX_OFFSET_X = 230; private static final int FESTIVAL_MAX_OFFSET_Y = 230; private static final int FESTIVAL_DEFAULT_RESPAWN = 60; // Specify in seconds! public static final int FESTIVAL_COUNT = 5; public static final int FESTIVAL_LEVEL_MAX_31 = 0; public static final int FESTIVAL_LEVEL_MAX_42 = 1; public static final int FESTIVAL_LEVEL_MAX_53 = 2; public static final int FESTIVAL_LEVEL_MAX_64 = 3; public static final int FESTIVAL_LEVEL_MAX_NONE = 4; public static final int[] FESTIVAL_LEVEL_SCORES = {60, 70, 100, 120, 150}; // 500 maximum possible score public static final int FESTIVAL_OFFERING_ID = 5901; public static final int FESTIVAL_OFFERING_VALUE = 5; //////////////////////// \\\\\\\\\\\\\\\\\\\\\\\\\\ /* * The following contains all the necessary spawn data for: * - Player Start Locations * - Witches * - Monsters * - Chests * * All data is given by: X, Y, Z (coords), Heading, NPC ID (if necessary) * This may be moved externally in time, but the data should not change. */ public static final int[][] FESTIVAL_DAWN_PLAYER_SPAWNS = { {-79187, 113186, -4895, 0}, // 31 and below {-75918, 110137, -4895, 0}, // 42 and below {-73835, 111969, -4895, 0}, // 53 and below {-76170, 113804, -4895, 0}, // 64 and below {-78927, 109528, -4895, 0} // No level limit }; public static final int[][] FESTIVAL_DUSK_PLAYER_SPAWNS = { {-77200, 88966, -5151, 0}, // 31 and below {-76941, 85307, -5151, 0}, // 42 and below {-74855, 87135, -5151, 0}, // 53 and below {-80208, 88222, -5151, 0}, // 64 and below {-79954, 84697, -5151, 0} // No level limit }; protected static final int[][] FESTIVAL_DAWN_WITCH_SPAWNS = { {-79183, 113052, -4891, 0, 31132}, // 31 and below {-75916, 110270, -4891, 0, 31133}, // 42 and below {-73979, 111970, -4891, 0, 31134}, // 53 and below {-76174, 113663, -4891, 0, 31135}, // 64 and below {-78930, 109664, -4891, 0, 31136} // No level limit }; protected static final int[][] FESTIVAL_DUSK_WITCH_SPAWNS = { {-77199, 88830, -5147, 0, 31142}, // 31 and below {-76942, 85438, -5147, 0, 31143}, // 42 and below {-74990, 87135, -5147, 0, 31144}, // 53 and below {-80207, 88222, -5147, 0, 31145}, // 64 and below {-79952, 84833, -5147, 0, 31146} // No level limit }; protected static final int[][][] FESTIVAL_DAWN_PRIMARY_SPAWNS = { { /* Level 31 and Below - Offering of the Branded */ {-78537, 113839, -4895, -1, 18009}, {-78466, 113852, -4895, -1, 18010}, {-78509, 113899, -4895, -1, 18010}, {-78481, 112557, -4895, -1, 18009}, {-78559, 112504, -4895, -1, 18010}, {-78489, 112494, -4895, -1, 18010}, {-79803, 112543, -4895, -1, 18012}, {-79854, 112492, -4895, -1, 18013}, {-79886, 112557, -4895, -1, 18014}, {-79821, 113811, -4895, -1, 18015}, {-79857, 113896, -4895, -1, 18017}, {-79878, 113816, -4895, -1, 18018}, // Archers and Marksmen \\ {-79190, 113660, -4895, -1, 18011}, {-78710, 113188, -4895, -1, 18011}, {-79190, 112730, -4895, -1, 18016}, {-79656, 113188, -4895, -1, 18016} }, { /* Level 42 and Below - Apostate Offering */ {-76558, 110784, -4895, -1, 18019}, {-76607, 110815, -4895, -1, 18020}, // South West {-76559, 110820, -4895, -1, 18020}, {-75277, 110792, -4895, -1, 18019}, {-75225, 110801, -4895, -1, 18020}, // South East {-75262, 110832, -4895, -1, 18020}, {-75249, 109441, -4895, -1, 18022}, {-75278, 109495, -4895, -1, 18023}, // North East {-75223, 109489, -4895, -1, 18024}, {-76556, 109490, -4895, -1, 18025}, {-76607, 109469, -4895, -1, 18027}, // North West {-76561, 109450, -4895, -1, 18028}, // Archers and Marksmen \\ {-76399, 110144, -4895, -1, 18021}, {-75912, 110606, -4895, -1, 18021}, {-75444, 110144, -4895, -1, 18026}, {-75930, 109665, -4895, -1, 18026} }, { /* Level 53 and Below - Witch's Offering */ {-73184, 111319, -4895, -1, 18029}, {-73135, 111294, -4895, -1, 18030}, // South West {-73185, 111281, -4895, -1, 18030}, {-74477, 111321, -4895, -1, 18029}, {-74523, 111293, -4895, -1, 18030}, // South East {-74481, 111280, -4895, -1, 18030}, {-74489, 112604, -4895, -1, 18032}, {-74491, 112660, -4895, -1, 18033}, // North East {-74527, 112629, -4895, -1, 18034}, {-73197, 112621, -4895, -1, 18035}, {-73142, 112631, -4895, -1, 18037}, // North West {-73182, 112656, -4895, -1, 18038}, // Archers and Marksmen \\ {-73834, 112430, -4895, -1, 18031}, {-74299, 111959, -4895, -1, 18031}, {-73841, 111491, -4895, -1, 18036}, {-73363, 111959, -4895, -1, 18036} }, { /* Level 64 and Below - Dark Omen Offering */ {-75543, 114461, -4895, -1, 18039}, {-75514, 114493, -4895, -1, 18040}, // South West {-75488, 114456, -4895, -1, 18040}, {-75521, 113158, -4895, -1, 18039}, {-75504, 113110, -4895, -1, 18040}, // South East {-75489, 113142, -4895, -1, 18040}, {-76809, 113143, -4895, -1, 18042}, {-76860, 113138, -4895, -1, 18043}, // North East {-76831, 113112, -4895, -1, 18044}, {-76831, 114441, -4895, -1, 18045}, {-76840, 114490, -4895, -1, 18047}, // North West {-76864, 114455, -4895, -1, 18048}, // Archers and Marksmen \\ {-75703, 113797, -4895, -1, 18041}, {-76180, 114263, -4895, -1, 18041}, {-76639, 113797, -4895, -1, 18046}, {-76180, 113337, -4895, -1, 18046} }, { /* No Level Limit - Offering of Forbidden Path */ {-79576, 108881, -4895, -1, 18049}, {-79592, 108835, -4895, -1, 18050}, // South West {-79614, 108871, -4895, -1, 18050}, {-79586, 110171, -4895, -1, 18049}, {-79589, 110216, -4895, -1, 18050}, // South East {-79620, 110177, -4895, -1, 18050}, {-78825, 110182, -4895, -1, 18052}, {-78238, 110182, -4895, -1, 18053}, // North East {-78266, 110218, -4895, -1, 18054}, {-78275, 108883, -4895, -1, 18055}, {-78267, 108839, -4895, -1, 18057}, // North West {-78241, 108871, -4895, -1, 18058}, // Archers and Marksmen \\ {-79394, 109538, -4895, -1, 18051}, {-78929, 109992, -4895, -1, 18051}, {-78454, 109538, -4895, -1, 18056}, {-78929, 109053, -4895, -1, 18056} } }; protected static final int[][][] FESTIVAL_DUSK_PRIMARY_SPAWNS = { { /* Level 31 and Below - Offering of the Branded */ {-76542, 89653, -5151, -1, 18009}, {-76509, 89637, -5151, -1, 18010}, {-76548, 89614, -5151, -1, 18010}, {-76539, 88326, -5151, -1, 18009}, {-76512, 88289, -5151, -1, 18010}, {-76546, 88287, -5151, -1, 18010}, {-77879, 88308, -5151, -1, 18012}, {-77886, 88310, -5151, -1, 18013}, {-77879, 88278, -5151, -1, 18014}, {-77857, 89605, -5151, -1, 18015}, {-77858, 89658, -5151, -1, 18017}, {-77891, 89633, -5151, -1, 18018}, // Archers and Marksmen \\ {-76728, 88962, -5151, -1, 18011}, {-77194, 88494, -5151, -1, 18011}, {-77660, 88896, -5151, -1, 18016}, {-77195, 89438, -5151, -1, 18016} }, { /* Level 42 and Below - Apostate's Offering */ {-77585, 84650, -5151, -1, 18019}, {-77628, 84643, -5151, -1, 18020}, {-77607, 84613, -5151, -1, 18020}, {-76603, 85946, -5151, -1, 18019}, {-77606, 85994, -5151, -1, 18020}, {-77638, 85959, -5151, -1, 18020}, {-76301, 85960, -5151, -1, 18022}, {-76257, 85972, -5151, -1, 18023}, {-76286, 85992, -5151, -1, 18024}, {-76281, 84667, -5151, -1, 18025}, {-76291, 84611, -5151, -1, 18027}, {-76257, 84616, -5151, -1, 18028}, // Archers and Marksmen \\ {-77419, 85307, -5151, -1, 18021}, {-76952, 85768, -5151, -1, 18021}, {-76477, 85312, -5151, -1, 18026}, {-76942, 84832, -5151, -1, 18026} }, { /* Level 53 and Below - Witch's Offering */ {-74211, 86494, -5151, -1, 18029}, {-74200, 86449, -5151, -1, 18030}, {-74167, 86464, -5151, -1, 18030}, {-75495, 86482, -5151, -1, 18029}, {-75540, 86473, -5151, -1, 18030}, {-75509, 86445, -5151, -1, 18030}, {-75509, 87775, -5151, -1, 18032}, {-75518, 87826, -5151, -1, 18033}, {-75542, 87780, -5151, -1, 18034}, {-74214, 87789, -5151, -1, 18035}, {-74169, 87801, -5151, -1, 18037}, {-74198, 87827, -5151, -1, 18038}, // Archers and Marksmen \\ {-75324, 87135, -5151, -1, 18031}, {-74852, 87606, -5151, -1, 18031}, {-74388, 87146, -5151, -1, 18036}, {-74856, 86663, -5151, -1, 18036} }, { /* Level 64 and Below - Dark Omen Offering */ {-79560, 89007, -5151, -1, 18039}, {-79521, 89016, -5151, -1, 18040}, {-79544, 89047, -5151, -1, 18040}, {-79552, 87717, -5151, -1, 18039}, {-79552, 87673, -5151, -1, 18040}, {-79510, 87702, -5151, -1, 18040}, {-80866, 87719, -5151, -1, 18042}, {-80897, 87689, -5151, -1, 18043}, {-80850, 87685, -5151, -1, 18044}, {-80848, 89013, -5151, -1, 18045}, {-80887, 89051, -5151, -1, 18047}, {-80891, 89004, -5151, -1, 18048}, // Archers and Marksmen \\ {-80205, 87895, -5151, -1, 18041}, {-80674, 88350, -5151, -1, 18041}, {-80209, 88833, -5151, -1, 18046}, {-79743, 88364, -5151, -1, 18046} }, { /* No Level Limit - Offering of Forbidden Path */ {-80624, 84060, -5151, -1, 18049}, {-80621, 84007, -5151, -1, 18050}, {-80590, 84039, -5151, -1, 18050}, {-80605, 85349, -5151, -1, 18049}, {-80639, 85363, -5151, -1, 18050}, {-80611, 85385, -5151, -1, 18050}, {-79311, 85353, -5151, -1, 18052}, {-79277, 85384, -5151, -1, 18053}, {-79273, 85539, -5151, -1, 18054}, {-79297, 84054, -5151, -1, 18055}, {-79285, 84006, -5151, -1, 18057}, {-79260, 84040, -5151, -1, 18058}, // Archers and Marksmen \\ {-79945, 85171, -5151, -1, 18051}, {-79489, 84707, -5151, -1, 18051}, {-79952, 84222, -5151, -1, 18056}, {-80423, 84703, -5151, -1, 18056} } }; protected static final int[][][] FESTIVAL_DAWN_SECONDARY_SPAWNS = { { /* 31 and Below */ {-78757, 112834, -4895, -1, 18016}, {-78581, 112834, -4895, -1, 18016}, {-78822, 112526, -4895, -1, 18011}, {-78822, 113702, -4895, -1, 18011}, {-78822, 113874, -4895, -1, 18011}, {-79524, 113546, -4895, -1, 18011}, {-79693, 113546, -4895, -1, 18011}, {-79858, 113546, -4895, -1, 18011}, {-79545, 112757, -4895, -1, 18016}, {-79545, 112586, -4895, -1, 18016}, }, { /* 42 and Below */ {-75565, 110580, -4895, -1, 18026}, {-75565, 110740, -4895, -1, 18026}, {-75577, 109776, -4895, -1, 18021}, {-75413, 109776, -4895, -1, 18021}, {-75237, 109776, -4895, -1, 18021}, {-76274, 109468, -4895, -1, 18021}, {-76274, 109635, -4895, -1, 18021}, {-76274, 109795, -4895, -1, 18021}, {-76351, 110500, -4895, -1, 18056}, {-76528, 110500, -4895, -1, 18056}, }, { /* 53 and Below */ {-74191, 111527, -4895, -1, 18036}, {-74191, 111362, -4895, -1, 18036}, {-73495, 111611, -4895, -1, 18031}, {-73327, 111611, -4895, -1, 18031}, {-73154, 111611, -4895, -1, 18031}, {-73473, 112301, -4895, -1, 18031}, {-73473, 112475, -4895, -1, 18031}, {-73473, 112649, -4895, -1, 18031}, {-74270, 112326, -4895, -1, 18036}, {-74443, 112326, -4895, -1, 18036}, }, { /* 64 and Below */ {-75738, 113439, -4895, -1, 18046}, {-75571, 113439, -4895, -1, 18046}, {-75824, 114141, -4895, -1, 18041}, {-75824, 114309, -4895, -1, 18041}, {-75824, 114477, -4895, -1, 18041}, {-76513, 114158, -4895, -1, 18041}, {-76683, 114158, -4895, -1, 18041}, {-76857, 114158, -4895, -1, 18041}, {-76535, 113357, -4895, -1, 18056}, {-76535, 113190, -4895, -1, 18056}, }, { /* No Level Limit */ {-79350, 109894, -4895, -1, 18056}, {-79534, 109894, -4895, -1, 18056}, {-79285, 109187, -4895, -1, 18051}, {-79285, 109019, -4895, -1, 18051}, {-79285, 108860, -4895, -1, 18051}, {-78587, 109172, -4895, -1, 18051}, {-78415, 109172, -4895, -1, 18051}, {-78249, 109172, -4895, -1, 18051}, {-78575, 109961, -4895, -1, 18056}, {-78575, 110130, -4895, -1, 18056}, } }; protected static final int[][][] FESTIVAL_DUSK_SECONDARY_SPAWNS = { { /* 31 and Below */ {-76844, 89304, -5151, -1, 18011}, {-76844, 89479, -5151, -1, 18011}, {-76844, 89649, -5151, -1, 18011}, {-77544, 89326, -5151, -1, 18011}, {-77716, 89326, -5151, -1, 18011}, {-77881, 89326, -5151, -1, 18011}, {-77561, 88530, -5151, -1, 18016}, {-77561, 88364, -5151, -1, 18016}, {-76762, 88615, -5151, -1, 18016}, {-76594, 88615, -5151, -1, 18016}, }, { /* 42 and Below */ {-77307, 84969, -5151, -1, 18021}, {-77307, 84795, -5151, -1, 18021}, {-77307, 84623, -5151, -1, 18021}, {-76614, 84944, -5151, -1, 18021}, {-76433, 84944, -5151, -1, 18021}, {-7626-1, 84944, -5151, -1, 18021}, {-76594, 85745, -5151, -1, 18026}, {-76594, 85910, -5151, -1, 18026}, {-77384, 85660, -5151, -1, 18026}, {-77555, 85660, -5151, -1, 18026}, }, { /* 53 and Below */ {-74517, 86782, -5151, -1, 18031}, {-74344, 86782, -5151, -1, 18031}, {-74185, 86782, -5151, -1, 18031}, {-74496, 87464, -5151, -1, 18031}, {-74496, 87636, -5151, -1, 18031}, {-74496, 87815, -5151, -1, 18031}, {-75298, 87497, -5151, -1, 18036}, {-75460, 87497, -5151, -1, 18036}, {-75219, 86712, -5151, -1, 18036}, {-75219, 86531, -5151, -1, 18036}, }, { /* 64 and Below */ {-79851, 88703, -5151, -1, 18041}, {-79851, 88868, -5151, -1, 18041}, {-79851, 89040, -5151, -1, 18041}, {-80548, 88722, -5151, -1, 18041}, {-80711, 88722, -5151, -1, 18041}, {-80883, 88722, -5151, -1, 18041}, {-80565, 87916, -5151, -1, 18046}, {-80565, 87752, -5151, -1, 18046}, {-79779, 87996, -5151, -1, 18046}, {-79613, 87996, -5151, -1, 18046}, }, { /* No Level Limit */ {-79271, 84330, -5151, -1, 18051}, {-79448, 84330, -5151, -1, 18051}, {-79601, 84330, -5151, -1, 18051}, {-80311, 84367, -5151, -1, 18051}, {-80311, 84196, -5151, -1, 18051}, {-80311, 84015, -5151, -1, 18051}, {-80556, 85049, -5151, -1, 18056}, {-80384, 85049, -5151, -1, 18056}, {-79598, 85127, -5151, -1, 18056}, {-79598, 85303, -5151, -1, 18056}, } }; protected static final int[][][] FESTIVAL_DAWN_CHEST_SPAWNS = { { /* Level 31 and Below */ {-78999, 112957, -4927, -1, 18109}, {-79153, 112873, -4927, -1, 18109}, {-79256, 112873, -4927, -1, 18109}, {-79368, 112957, -4927, -1, 18109}, {-79481, 113124, -4927, -1, 18109}, {-79481, 113275, -4927, -1, 18109}, {-79364, 113398, -4927, -1, 18109}, {-79213, 113500, -4927, -1, 18109}, {-79099, 113500, -4927, -1, 18109}, {-78960, 113398, -4927, -1, 18109}, {-78882, 113235, -4927, -1, 18109}, {-78882, 113099, -4927, -1, 18109}, }, { /* Level 42 and Below */ {-76119, 110383, -4927, -1, 18110}, {-75980, 110442, -4927, -1, 18110}, {-75848, 110442, -4927, -1, 18110}, {-75720, 110383, -4927, -1, 18110}, {-75625, 110195, -4927, -1, 18110}, {-75625, 110063, -4927, -1, 18110}, {-75722, 109908, -4927, -1, 18110}, {-75863, 109832, -4927, -1, 18110}, {-75989, 109832, -4927, -1, 18110}, {-76130, 109908, -4927, -1, 18110}, {-76230, 110079, -4927, -1, 18110}, {-76230, 110215, -4927, -1, 18110}, }, { /* Level 53 and Below */ {-74055, 111781, -4927, -1, 18111}, {-74144, 111938, -4927, -1, 18111}, {-74144, 112075, -4927, -1, 18111}, {-74055, 112173, -4927, -1, 18111}, {-73885, 112289, -4927, -1, 18111}, {-73756, 112289, -4927, -1, 18111}, {-73574, 112141, -4927, -1, 18111}, {-73511, 112040, -4927, -1, 18111}, {-73511, 111912, -4927, -1, 18111}, {-73574, 111772, -4927, -1, 18111}, {-73767, 111669, -4927, -1, 18111}, {-73899, 111669, -4927, -1, 18111}, }, { /* Level 64 and Below */ {-76008, 113566, -4927, -1, 18112}, {-76159, 113485, -4927, -1, 18112}, {-76267, 113485, -4927, -1, 18112}, {-76386, 113566, -4927, -1, 18112}, {-76482, 113748, -4927, -1, 18112}, {-76482, 113885, -4927, -1, 18112}, {-76371, 114029, -4927, -1, 18112}, {-76220, 114118, -4927, -1, 18112}, {-76092, 114118, -4927, -1, 18112}, {-75975, 114029, -4927, -1, 18112}, {-75861, 11385-1, -4927, -1, 18112}, {-75861, 113713, -4927, -1, 18112}, }, { /* No Level Limit */ {-79100, 109782, -4927, -1, 18113}, {-78962, 109853, -4927, -1, 18113}, {-78851, 109853, -4927, -1, 18113}, {-78721, 109782, -4927, -1, 18113}, {-78615, 109596, -4927, -1, 18113}, {-78615, 109453, -4927, -1, 18113}, {-78746, 109300, -4927, -1, 18113}, {-78881, 109203, -4927, -1, 18113}, {-79027, 109203, -4927, -1, 18113}, {-79159, 109300, -4927, -1, 18113}, {-79240, 109480, -4927, -1, 18113}, {-79240, 109615, -4927, -1, 18113}, } }; protected static final int[][][] FESTIVAL_DUSK_CHEST_SPAWNS = { { /* Level 31 and Below */ {-77016, 88726, -5183, -1, 18114}, {-77136, 88646, -5183, -1, 18114}, {-77247, 88646, -5183, -1, 18114}, {-77380, 88726, -5183, -1, 18114}, {-77512, 88883, -5183, -1, 18114}, {-77512, 89053, -5183, -1, 18114}, {-77378, 89287, -5183, -1, 18114}, {-77254, 89238, -5183, -1, 18114}, {-77095, 89238, -5183, -1, 18114}, {-76996, 89287, -5183, -1, 18114}, {-76901, 89025, -5183, -1, 18114}, {-76901, 88891, -5183, -1, 18114}, }, { /* Level 42 and Below */ {-77128, 85553, -5183, -1, 18115}, {-77036, 85594, -5183, -1, 18115}, {-76919, 85594, -5183, -1, 18115}, {-76755, 85553, -5183, -1, 18115}, {-76635, 85392, -5183, -1, 18115}, {-76635, 85216, -5183, -1, 18115}, {-76761, 85025, -5183, -1, 18115}, {-76908, 85004, -5183, -1, 18115}, {-77041, 85004, -5183, -1, 18115}, {-77138, 85025, -5183, -1, 18115}, {-77268, 85219, -5183, -1, 18115}, {-77268, 85410, -5183, -1, 18115}, }, { /* Level 53 and Below */ {-75150, 87303, -5183, -1, 18116}, {-75150, 87175, -5183, -1, 18116}, {-75150, 87175, -5183, -1, 18116}, {-75150, 87303, -5183, -1, 18116}, {-74943, 87433, -5183, -1, 18116}, {-74767, 87433, -5183, -1, 18116}, {-74556, 87306, -5183, -1, 18116}, {-74556, 87184, -5183, -1, 18116}, {-74556, 87184, -5183, -1, 18116}, {-74556, 87306, -5183, -1, 18116}, {-74757, 86830, -5183, -1, 18116}, {-74927, 86830, -5183, -1, 18116}, }, { /* Level 64 and Below */ {-80010, 88128, -5183, -1, 18117}, {-80113, 88066, -5183, -1, 18117}, {-80220, 88066, -5183, -1, 18117}, {-80359, 88128, -5183, -1, 18117}, {-80467, 88267, -5183, -1, 18117}, {-80467, 88436, -5183, -1, 18117}, {-80381, 88639, -5183, -1, 18117}, {-80278, 88577, -5183, -1, 18117}, {-80142, 88577, -5183, -1, 18117}, {-80028, 88639, -5183, -1, 18117}, {-79915, 88466, -5183, -1, 18117}, {-79915, 88322, -5183, -1, 18117}, }, { /* No Level Limit */ {-80153, 84947, -5183, -1, 18118}, {-80003, 84962, -5183, -1, 18118}, {-79848, 84962, -5183, -1, 18118}, {-79742, 84947, -5183, -1, 18118}, {-79668, 84772, -5183, -1, 18118}, {-79668, 84619, -5183, -1, 18118}, {-79772, 84471, -5183, -1, 18118}, {-79888, 84414, -5183, -1, 18118}, {-80023, 84414, -5183, -1, 18118}, {-80166, 84471, -5183, -1, 18118}, {-80253, 84600, -5183, -1, 18118}, {-80253, 84780, -5183, -1, 18118}, } }; //////////////////////// \\\\\\\\\\\\\\\\\\\\\\\\\\ protected FestivalManager _managerInstance; protected ScheduledFuture _managerScheduledTask; protected int _signsCycle = SevenSigns.getInstance().getCurrentCycle(); protected int _festivalCycle; protected long _nextFestivalCycleStart; protected long _nextFestivalStart; protected boolean _festivalInitialized; protected boolean _festivalInProgress; protected List _accumulatedBonuses; // The total bonus available (in Ancient Adena) boolean _noPartyRegister; private L2NpcInstance _dawnChatGuide; private L2NpcInstance _duskChatGuide; protected Map> _dawnFestivalParticipants; protected Map> _duskFestivalParticipants; protected Map> _dawnPreviousParticipants; protected Map> _duskPreviousParticipants; private Map _dawnFestivalScores; private Map _duskFestivalScores; /** * _festivalData is essentially an instance of the seven_signs_festival table and * should be treated as such. * * Data is initially accessed by the related Seven Signs cycle, with _signsCycle representing * data for the current round of Festivals. * * The actual table data is stored as a series of StatsSet constructs. These are accessed by * the use of an offset based on the number of festivals, thus: * * offset = FESTIVAL_COUNT + festivalId * (Data for Dawn is always accessed by offset > FESTIVAL_COUNT) */ private Map> _festivalData; public SevenSignsFestival() { _accumulatedBonuses = new FastList(); _dawnFestivalParticipants = new FastMap>(); _dawnPreviousParticipants = new FastMap>(); _dawnFestivalScores = new FastMap(); _duskFestivalParticipants = new FastMap>(); _duskPreviousParticipants = new FastMap>(); _duskFestivalScores = new FastMap(); _festivalData = new FastMap>(); restoreFestivalData(); if (SevenSigns.getInstance().isSealValidationPeriod()) { _log.info("SevenSignsFestival: Initialization bypassed due to Seal Validation in effect."); return; } L2Spawn.addSpawnListener(this); startFestivalManager(); } public static SevenSignsFestival getInstance() { if (_instance == null) _instance = new SevenSignsFestival(); return _instance; } /** * Returns the associated name (level range) to a given festival ID. * * @param int festivalID * @return String festivalName */ public static final String getFestivalName(int festivalID) { String festivalName; switch (festivalID) { case FESTIVAL_LEVEL_MAX_31: festivalName = "Level 31 or lower"; break; case FESTIVAL_LEVEL_MAX_42: festivalName = "Level 42 or lower"; break; case FESTIVAL_LEVEL_MAX_53: festivalName = "Level 53 or lower"; break; case FESTIVAL_LEVEL_MAX_64: festivalName = "Level 64 or lower"; break; default: festivalName = "No Level Limit"; break; } return festivalName; } /** * Returns the maximum allowed player level for the given festival type. * * @param festivalId * @return int maxLevel */ public static final int getMaxLevelForFestival(int festivalId) { int maxLevel = (Experience.MAX_LEVEL - 1); switch (festivalId) { case SevenSignsFestival.FESTIVAL_LEVEL_MAX_31: maxLevel = 31; break; case SevenSignsFestival.FESTIVAL_LEVEL_MAX_42: maxLevel = 42; break; case SevenSignsFestival.FESTIVAL_LEVEL_MAX_53: maxLevel = 53; break; case SevenSignsFestival.FESTIVAL_LEVEL_MAX_64: maxLevel = 64; break; } return maxLevel; } /** * Returns true if the monster ID given is of an archer/marksman type. * * @param npcId * @return boolean isArcher */ protected static final boolean isFestivalArcher(int npcId) { if (npcId < 18009 || npcId > 18108) return false; int identifier = npcId%10; return (identifier == 4 || identifier == 9); } /** * Returns true if the monster ID given is a festival chest. * * @param npcId * @return boolean isChest */ protected static final boolean isFestivalChest(int npcId) { return (npcId < 18109 || npcId > 18118); } /** * Primarily used to terminate the Festival Manager, when the Seven Signs period changes. * * @return ScheduledFuture festManagerScheduler */ protected final ScheduledFuture getFestivalManagerSchedule() { if (_managerScheduledTask == null) startFestivalManager(); return _managerScheduledTask; } /** * Used to start the Festival Manager, if the current period is not Seal Validation. */ protected void startFestivalManager() { // Start the Festival Manager for the first time after the server has started // at the specified time, then invoke it automatically after every cycle. FestivalManager fm = new FestivalManager(); setNextFestivalStart(Config.ALT_FESTIVAL_MANAGER_START + FESTIVAL_SIGNUP_TIME); _managerScheduledTask = ThreadPoolManager.getInstance().scheduleGeneralAtFixedRate(fm, Config.ALT_FESTIVAL_MANAGER_START, Config.ALT_FESTIVAL_CYCLE_LENGTH); _log.info("SevenSignsFestival: The first Festival of Darkness cycle begins in " + (Config.ALT_FESTIVAL_MANAGER_START / 60000) + " minute(s)."); } /** * Restores saved festival data, basic settings from the properties file * and past high score data from the database. * * @throws Exception */ protected void restoreFestivalData() { Connection con = null; PreparedStatement statement = null; ResultSet rset = null; if (Config.DEBUG) _log.info("SevenSignsFestival: Restoring festival data. Current SS Cycle: " + _signsCycle); try { con = L2DatabaseFactory.getInstance().getConnection(); statement = con.prepareStatement("SELECT festivalId, cabal, cycle, date, score, members " + "FROM seven_signs_festival"); rset = statement.executeQuery(); while (rset.next()) { int festivalCycle = rset.getInt("cycle"); int festivalId = rset.getInt("festivalId"); String cabal = rset.getString("cabal"); StatsSet festivalDat = new StatsSet(); festivalDat.set("festivalId", festivalId); festivalDat.set("cabal", cabal); festivalDat.set("cycle", festivalCycle); festivalDat.set("date", rset.getString("date")); festivalDat.set("score", rset.getInt("score")); festivalDat.set("members", rset.getString("members")); if (Config.DEBUG) _log.info("SevenSignsFestival: Loaded data from DB for (Cycle = " + festivalCycle + ", Oracle = " + cabal + ", Festival = "+ getFestivalName(festivalId)); if (cabal.equals("dawn")) festivalId += FESTIVAL_COUNT; Map tempData = _festivalData.get(festivalCycle); if (tempData == null) tempData = new FastMap(); tempData.put(festivalId, festivalDat); _festivalData.put(festivalCycle, tempData); } rset.close(); statement.close(); String query = "SELECT festival_cycle, "; for (int i = 0; i < FESTIVAL_COUNT-1; i++) query += "accumulated_bonus" + String.valueOf(i) + ", "; query += "accumulated_bonus" + String.valueOf(FESTIVAL_COUNT -1) + " "; query += "FROM seven_signs_status WHERE id=0"; statement = con.prepareStatement(query); rset = statement.executeQuery(); while (rset.next()) { _festivalCycle = rset.getInt("festival_cycle"); for(int i = 0; i < FESTIVAL_COUNT; i++) _accumulatedBonuses.add(i, rset.getInt("accumulated_bonus" + String.valueOf(i))); } rset.close(); statement.close(); con.close(); if (Config.DEBUG) _log.info("SevenSignsFestival: Loaded data from database."); } catch (SQLException e) { _log.severe("SevenSignsFestival: Failed to load configuration: " + e); } finally { try { rset.close(); statement.close(); con.close(); } catch (SQLException e) {} } } /** * Stores current festival data, basic settings to the properties file * and past high score data to the database. * * If updateSettings = true, then all Seven Signs data is updated in the database. * * @param updateSettings * @throws Exception */ public void saveFestivalData(boolean updateSettings) { Connection con = null; PreparedStatement statement = null; if (Config.DEBUG) _log.info("SevenSignsFestival: Saving festival data to disk."); try { con = L2DatabaseFactory.getInstance().getConnection(); for (Map currCycleData : _festivalData.values()) { for (StatsSet festivalDat : currCycleData.values()) { int festivalCycle = festivalDat.getInteger("cycle"); int festivalId = festivalDat.getInteger("festivalId"); String cabal = festivalDat.getString("cabal"); // Try to update an existing record. statement = con.prepareStatement( "UPDATE seven_signs_festival SET date=?, score=?, members=? WHERE cycle=? AND cabal=? AND festivalId=?"); statement.setLong(1, Long.valueOf(festivalDat.getString("date"))); statement.setInt(2, festivalDat.getInteger("score")); statement.setString(3, festivalDat.getString("members")); statement.setInt(4, festivalCycle); statement.setString(5, cabal); statement.setInt(6, festivalId); // If there was no record to update, assume it doesn't exist and add a new one, // otherwise continue with the next record to store. if (statement.executeUpdate() > 0) { if (Config.DEBUG) _log.info("SevenSignsFestival: Updated data in DB (Cycle = " + festivalCycle + ", Cabal = " + cabal + ", FestID = " + festivalId + ")"); statement.close(); continue; } statement.close(); statement = con.prepareStatement( "INSERT INTO seven_signs_festival (festivalId, cabal, cycle, date, score, members) VALUES (?,?,?,?,?,?)"); statement.setInt(1, festivalId); statement.setString(2, cabal); statement.setInt(3, festivalCycle); statement.setLong(4, Long.valueOf(festivalDat.getString("date"))); statement.setInt(5, festivalDat.getInteger("score")); statement.setString(6, festivalDat.getString("members")); statement.execute(); statement.close(); if (Config.DEBUG) _log.info("SevenSignsFestival: Inserted data in DB (Cycle = " + festivalCycle + ", Cabal = " + cabal + ", FestID = " + festivalId + ")"); } } con.close(); // Updates Seven Signs DB data also, so call only if really necessary. if (updateSettings) SevenSigns.getInstance().saveSevenSignsData(null, true); } catch (SQLException e) { _log.severe("SevenSignsFestival: Failed to save configuration: " + e); } finally { try { statement.close(); con.close(); } catch (Exception e) {} } } /** * If a clan member is a member of the highest-ranked party in the Festival of Darkness, 100 points are added per member */ protected void rewardHighestRanked() { String[] partyMembers; StatsSet overallData = getOverallHighestScoreData(FESTIVAL_LEVEL_MAX_31); if (overallData != null) { partyMembers = overallData.getString("members").split(","); for (String partyMemberName : partyMembers) addReputationPointsForPartyMemberClan(partyMemberName); } overallData = getOverallHighestScoreData(FESTIVAL_LEVEL_MAX_42); if (overallData != null) { partyMembers = overallData.getString("members").split(","); for (String partyMemberName : partyMembers) addReputationPointsForPartyMemberClan(partyMemberName); } overallData = getOverallHighestScoreData(FESTIVAL_LEVEL_MAX_53); if (overallData != null) { partyMembers = overallData.getString("members").split(","); for (String partyMemberName : partyMembers) addReputationPointsForPartyMemberClan(partyMemberName); } overallData = getOverallHighestScoreData(FESTIVAL_LEVEL_MAX_64); if (overallData != null) { partyMembers = overallData.getString("members").split(","); for (String partyMemberName : partyMembers) addReputationPointsForPartyMemberClan(partyMemberName); } overallData = getOverallHighestScoreData(FESTIVAL_LEVEL_MAX_NONE); if (overallData != null) { partyMembers = overallData.getString("members").split(","); for (String partyMemberName : partyMembers) addReputationPointsForPartyMemberClan(partyMemberName); } } private void addReputationPointsForPartyMemberClan(String partyMemberName) { L2PcInstance player = L2World.getInstance().getPlayer(partyMemberName); if (player != null) { if (player.getClan() != null) { player.getClan().setReputationScore(player.getClan().getReputationScore()+100, true); player.getClan().broadcastToOnlineMembers(new PledgeShowInfoUpdate(player.getClan())); SystemMessage sm = new SystemMessage(SystemMessageId.CLAN_MEMBER_S1_WAS_IN_HIGHEST_RANKED_PARTY_IN_FESTIVAL_OF_DARKNESS_AND_GAINED_S2_REPUTATION); sm.addString(partyMemberName); sm.addNumber(100); player.getClan().broadcastToOnlineMembers(sm); } } else { java.sql.Connection con = null; try { con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(GET_CLAN_NAME); statement.setString(1, partyMemberName); ResultSet rset = statement.executeQuery(); if (rset.next()) { String clanName = rset.getString("clan_name"); if (clanName != null) { L2Clan clan = ClanTable.getInstance().getClanByName(clanName); if (clan != null) { clan.setReputationScore(clan.getReputationScore()+100, true); clan.broadcastToOnlineMembers(new PledgeShowInfoUpdate(clan)); SystemMessage sm = new SystemMessage(SystemMessageId.CLAN_MEMBER_S1_WAS_IN_HIGHEST_RANKED_PARTY_IN_FESTIVAL_OF_DARKNESS_AND_GAINED_S2_REPUTATION); sm.addString(partyMemberName); sm.addNumber(100); clan.broadcastToOnlineMembers(sm); } } } rset.close(); statement.close(); } catch (Exception e) { _log.warning("could not get clan name of " + partyMemberName + ": "+e); } finally { try { con.close(); } catch (Exception e) {} } } } /** * Used to reset all festival data at the beginning of a new quest event period. */ protected void resetFestivalData(boolean updateSettings) { _festivalCycle = 0; _signsCycle = SevenSigns.getInstance().getCurrentCycle(); // Set all accumulated bonuses back to 0. for (int i = 0; i < FESTIVAL_COUNT; i++) _accumulatedBonuses.set(i, 0); _dawnFestivalParticipants.clear(); _dawnPreviousParticipants.clear(); _dawnFestivalScores.clear(); _duskFestivalParticipants.clear(); _duskPreviousParticipants.clear(); _duskFestivalScores.clear(); // Set up a new data set for the current cycle of festivals Map newData = new FastMap(); for (int i = 0; i < FESTIVAL_COUNT * 2; i++) { int festivalId = i; if (i >= FESTIVAL_COUNT) festivalId -= FESTIVAL_COUNT; // Create a new StatsSet with "default" data for Dusk StatsSet tempStats = new StatsSet(); tempStats.set("festivalId", festivalId); tempStats.set("cycle", _signsCycle); tempStats.set("date", "0"); tempStats.set("score", 0); tempStats.set("members", ""); if (i >= FESTIVAL_COUNT) tempStats.set("cabal", SevenSigns.getCabalShortName(SevenSigns.CABAL_DAWN)); else tempStats.set("cabal", SevenSigns.getCabalShortName(SevenSigns.CABAL_DUSK)); newData.put(i, tempStats); } // Add the newly created cycle data to the existing festival data, and // subsequently save it to the database. _festivalData.put(_signsCycle, newData); saveFestivalData(updateSettings); // Remove any unused blood offerings from online players. for (L2PcInstance onlinePlayer : L2World.getInstance().getAllPlayers()) { try { L2ItemInstance bloodOfferings = onlinePlayer.getInventory().getItemByItemId(FESTIVAL_OFFERING_ID); if (bloodOfferings != null) onlinePlayer.destroyItem("SevenSigns", bloodOfferings, null, false); } catch (NullPointerException e) {} } _log.info("SevenSignsFestival: Reinitialized engine for next competition period."); } public final int getCurrentFestivalCycle() { return _festivalCycle; } public final boolean isFestivalInitialized() { return _festivalInitialized; } public final boolean isFestivalInProgress() { return _festivalInProgress; } public void setNextCycleStart() { _nextFestivalCycleStart = System.currentTimeMillis() + Config.ALT_FESTIVAL_CYCLE_LENGTH; } public void setNextFestivalStart(long milliFromNow) { _nextFestivalStart = System.currentTimeMillis() + milliFromNow; } public final int getMinsToNextCycle() { if (SevenSigns.getInstance().isSealValidationPeriod()) return -1; return Math.round((_nextFestivalCycleStart - System.currentTimeMillis()) / 60000); } public final int getMinsToNextFestival() { if (SevenSigns.getInstance().isSealValidationPeriod()) return -1; return Math.round((_nextFestivalStart - System.currentTimeMillis()) / 60000) + 1; } public final String getTimeToNextFestivalStr() { if (SevenSigns.getInstance().isSealValidationPeriod()) return "This is the Seal Validation period. Festivals will resume next week."; return "The next festival will begin in " + getMinsToNextFestival() + " minute(s)."; } /** * Returns the current festival ID and oracle ID that the specified player is in, * but will return the default of {-1, -1} if the player is not found as a participant. * * @param player * @return int[] playerFestivalInfo */ public final int[] getFestivalForPlayer(L2PcInstance player) { int[] playerFestivalInfo = {-1, -1}; int festivalId = 0; while (festivalId < FESTIVAL_COUNT) { List participants = _dawnFestivalParticipants.get(festivalId); // If there are no participants in this festival, move on to the next. if (participants != null && participants.contains(player)) { playerFestivalInfo[0] = SevenSigns.CABAL_DAWN; playerFestivalInfo[1] = festivalId; return playerFestivalInfo; } festivalId++; participants = _duskFestivalParticipants.get(festivalId); if (participants != null && participants.contains(player)) { playerFestivalInfo[0] = SevenSigns.CABAL_DUSK; playerFestivalInfo[1] = festivalId; return playerFestivalInfo; } festivalId++; } // Return default data if the player is not found as a participant. return playerFestivalInfo; } public final boolean isParticipant(L2PcInstance player) { if (SevenSigns.getInstance().isSealValidationPeriod()) return false; if (_managerInstance == null) return false; for (List participants : _dawnFestivalParticipants.values()) if (participants.contains(player)) return true; for (List participants : _duskFestivalParticipants.values()) if (participants.contains(player)) return true; return false; } public final List getParticipants(int oracle, int festivalId) { if (oracle == SevenSigns.CABAL_DAWN) return _dawnFestivalParticipants.get(festivalId); return _duskFestivalParticipants.get(festivalId); } public final List getPreviousParticipants(int oracle, int festivalId) { if (oracle == SevenSigns.CABAL_DAWN) return _dawnPreviousParticipants.get(festivalId); return _duskPreviousParticipants.get(festivalId); } public void setParticipants(int oracle, int festivalId, L2Party festivalParty) { List participants = new FastList(); if (festivalParty != null) { participants = festivalParty.getPartyMembers(); if (Config.DEBUG) _log.info("SevenSignsFestival: " + festivalParty.getPartyMembers().toString() + " have signed up to the " + SevenSigns.getCabalShortName(oracle) + " " + getFestivalName(festivalId) + " festival."); } if (oracle == SevenSigns.CABAL_DAWN) _dawnFestivalParticipants.put(festivalId, participants); else _duskFestivalParticipants.put(festivalId, participants); } public void updateParticipants(L2PcInstance player, L2Party festivalParty) { if (!isParticipant(player)) return; final int[] playerFestInfo = getFestivalForPlayer(player); final int oracle = playerFestInfo[0]; final int festivalId = playerFestInfo[1]; if (festivalId > -1) { if (_festivalInitialized) { L2DarknessFestival festivalInst = _managerInstance.getFestivalInstance(oracle, festivalId); if (festivalParty == null) for (L2PcInstance partyMember : getParticipants(oracle, festivalId)) festivalInst.relocatePlayer(partyMember, true); else festivalInst.relocatePlayer(player, true); } setParticipants(oracle, festivalId, festivalParty); // Check on disconect if min player in party if (festivalParty.getMemberCount() < Config.ALT_FESTIVAL_MIN_PLAYER) { updateParticipants(player, festivalParty); festivalParty.removePartyMember(player); } } } public final int getFinalScore(int oracle, int festivalId) { if (oracle == SevenSigns.CABAL_DAWN) return _dawnFestivalScores.get(festivalId); return _duskFestivalScores.get(festivalId); } public final int getHighestScore(int oracle, int festivalId) { return getHighestScoreData(oracle, festivalId).getInteger("score"); } /** * Returns a stats set containing the highest score this cycle for the * the specified cabal and associated festival ID. * * @param oracle * @param festivalId * @return StatsSet festivalDat */ public final StatsSet getHighestScoreData(int oracle, int festivalId) { int offsetId = festivalId; if (oracle == SevenSigns.CABAL_DAWN) offsetId += 5; // Attempt to retrieve existing score data (if found), otherwise create a // new blank data set and display a console warning. StatsSet currData = null; try { currData = _festivalData.get(_signsCycle).get(offsetId); } catch (Exception e) { currData = new StatsSet(); currData.set("score", 0); currData.set("members", ""); if (Config.DEBUG) _log.info("SevenSignsFestival: Data missing for " + SevenSigns.getCabalName(oracle) + ", FestivalID = " + festivalId + " (Current Cycle " + _signsCycle + ")"); } return currData; } /** * Returns a stats set containing the highest ever recorded * score data for the specified festival. * * @param festivalId * @return StatsSet result */ public final StatsSet getOverallHighestScoreData(int festivalId) { StatsSet result = null; int highestScore = 0; for (Map currCycleData : _festivalData.values()) { for (StatsSet currFestData : currCycleData.values()) { int currFestID = currFestData.getInteger("festivalId"); int festivalScore = currFestData.getInteger("score"); if (currFestID != festivalId) continue; if (festivalScore > highestScore) { highestScore = festivalScore; result = currFestData; } } } return result; } /** * Set the final score details for the last participants of the specified festival data. * Returns true if the score is higher than that previously recorded this cycle. * * @param player * @param oracle * @param festivalId * @param offeringScore * @return boolean isHighestScore */ public boolean setFinalScore(L2PcInstance player, int oracle, int festivalId, int offeringScore) { List partyMembers; int currDawnHighScore = getHighestScore(SevenSigns.CABAL_DAWN, festivalId); int currDuskHighScore = getHighestScore(SevenSigns.CABAL_DUSK, festivalId); int thisCabalHighScore = 0; int otherCabalHighScore = 0; if (oracle == SevenSigns.CABAL_DAWN) { thisCabalHighScore = currDawnHighScore; otherCabalHighScore = currDuskHighScore; _dawnFestivalScores.put(festivalId, offeringScore); } else { thisCabalHighScore = currDuskHighScore; otherCabalHighScore = currDawnHighScore; _duskFestivalScores.put(festivalId, offeringScore); } StatsSet currFestData = getHighestScoreData(oracle, festivalId); // Check if this is the highest score for this level range so far for the player's cabal. if (offeringScore > thisCabalHighScore) { // If the current score is greater than that for the other cabal, // then they already have the points from this festival. if (thisCabalHighScore < otherCabalHighScore) return false; partyMembers = new FastList(); List prevParticipants = getPreviousParticipants(oracle, festivalId); // Record a string list of the party members involved. for (L2PcInstance partyMember : prevParticipants) { try { partyMembers.add(partyMember.getName()); } catch (NullPointerException e) {} } // Update the highest scores and party list. currFestData.set("date", String.valueOf(System.currentTimeMillis())); currFestData.set("score", offeringScore); currFestData.set("members", Util.implodeString(partyMembers, ",")); if (Config.DEBUG) _log.info("SevenSignsFestival: " + player.getName() + "'s party has the highest score (" + offeringScore + ") so far for " + SevenSigns.getCabalName(oracle) + " in " + getFestivalName(festivalId)); // Only add the score to the cabal's overall if it's higher than the other cabal's score. if (offeringScore > otherCabalHighScore) { int contribPoints = FESTIVAL_LEVEL_SCORES[festivalId]; // Give this cabal the festival points, while deducting them from the other. SevenSigns.getInstance().addFestivalScore(oracle, contribPoints); if (Config.DEBUG) _log.info("SevenSignsFestival: This is the highest score overall so far for the " + getFestivalName(festivalId) + " festival!"); } saveFestivalData(true); return true; } return false; } public final int getAccumulatedBonus(int festivalId) { return _accumulatedBonuses.get(festivalId); } public final int getTotalAccumulatedBonus() { int totalAccumBonus = 0; for (int accumBonus : _accumulatedBonuses) totalAccumBonus += accumBonus; return totalAccumBonus; } public void addAccumulatedBonus(int festivalId, int stoneType, int stoneAmount) { int eachStoneBonus = 0; switch (stoneType) { case SevenSigns.SEAL_STONE_BLUE_ID: eachStoneBonus = SevenSigns.SEAL_STONE_BLUE_VALUE; break; case SevenSigns.SEAL_STONE_GREEN_ID: eachStoneBonus = SevenSigns.SEAL_STONE_GREEN_VALUE; break; case SevenSigns.SEAL_STONE_RED_ID: eachStoneBonus = SevenSigns.SEAL_STONE_RED_VALUE; break; } int newTotalBonus = _accumulatedBonuses.get(festivalId) + (stoneAmount * eachStoneBonus); _accumulatedBonuses.set(festivalId, newTotalBonus); } /** * Calculate and return the proportion of the accumulated bonus for the festival * where the player was in the winning party, if the winning party's cabal won the event. * * The accumulated bonus is then updated, with the player's share deducted. * * @param player * @return playerBonus (the share of the bonus for the party) */ public final int distribAccumulatedBonus(L2PcInstance player) { int playerBonus = 0; String playerName = player.getName(); int playerCabal = SevenSigns.getInstance().getPlayerCabal(player); if (playerCabal != SevenSigns.getInstance().getCabalHighestScore()) return 0; if (_festivalData.get(_signsCycle) != null) for (StatsSet festivalData : _festivalData.get(_signsCycle).values()) { if (festivalData.getString("members").indexOf(playerName) > -1) { int festivalId = festivalData.getInteger("festivalId"); int numPartyMembers = festivalData.getString("members").split(",").length; int totalAccumBonus = _accumulatedBonuses.get(festivalId); playerBonus = totalAccumBonus / numPartyMembers; _accumulatedBonuses.set(festivalId, totalAccumBonus - playerBonus); break; } } return playerBonus; } /** * Used to send a "shout" message to all players currently present in an Oracle. * Primarily used for Festival Guide and Witch related speech. * * @param senderName * @param message */ public void sendMessageToAll(String senderName, String message) { if (_dawnChatGuide == null || _duskChatGuide == null) return; CreatureSay cs = new CreatureSay(_dawnChatGuide.getObjectId(), 1, senderName, message); _dawnChatGuide.broadcastPacket(cs); cs = new CreatureSay(_duskChatGuide.getObjectId(), 1, senderName, message); _duskChatGuide.broadcastPacket(cs); } /** * Basically a wrapper-call to signal to increase the challenge of the specified festival. * * @param oracle * @param festivalId * @return boolean isChalIncreased */ public final boolean increaseChallenge(int oracle, int festivalId) { L2DarknessFestival festivalInst = _managerInstance.getFestivalInstance(oracle, festivalId); return festivalInst.increaseChallenge(); } /** * Used with the SpawnListener, to update the required "chat guide" instances, * for use with announcements in the oracles. * * @param npc */ public void npcSpawned(L2NpcInstance npc) { if (npc == null) return; int npcId = npc.getNpcId(); // If the spawned NPC ID matches the ones we need, assign their instances. if (npcId == 31127) { if (Config.DEBUG) _log.config("SevenSignsFestival: Instance found for NPC ID 31127 (" + npc.getObjectId() + ")."); _dawnChatGuide = npc; } if (npcId == 31137) { if (Config.DEBUG) _log.config("SevenSignsFestival: Instance found for NPC ID 31137 (" + npc.getObjectId() + ")."); _duskChatGuide = npc; } } /** * The FestivalManager class is the main runner of all the festivals. * It is used for easier integration and management of all running festivals. * * @author Tempy */ private class FestivalManager implements Runnable { protected Map _festivalInstances; public FestivalManager() { _festivalInstances = new FastMap(); _managerInstance = this; // Increment the cycle counter. _festivalCycle++; // Set the next start timers. setNextCycleStart(); setNextFestivalStart(Config.ALT_FESTIVAL_CYCLE_LENGTH - FESTIVAL_SIGNUP_TIME); } public synchronized void run() { // The manager shouldn't be running if Seal Validation is in effect. if (SevenSigns.getInstance().isSealValidationPeriod()) return; // If the next period is due to start before the end of this // festival cycle, then don't run it. if (SevenSigns.getInstance().getMilliToPeriodChange() < Config.ALT_FESTIVAL_CYCLE_LENGTH) return; if (Config.DEBUG) _log.info("SevenSignsFestival: Festival manager initialized. Those wishing to participate have " + getMinsToNextFestival() + " minute(s) to sign up."); sendMessageToAll("Festival Guide", "The main event will start in " + getMinsToNextFestival() + " minutes. Please register now."); // Stand by until the allowed signup period has elapsed. try { wait(FESTIVAL_SIGNUP_TIME); } catch (InterruptedException e) { } // Clear past participants, they can no longer register their score if not done so already. _dawnPreviousParticipants.clear(); _duskPreviousParticipants.clear(); // Get rid of random monsters that avoided deletion after last festival for (L2DarknessFestival festivalInst : _festivalInstances.values()) festivalInst.unspawnMobs(); // Start only if participants signed up _noPartyRegister = true; while (_noPartyRegister) { if ((_duskFestivalParticipants.isEmpty() && _dawnFestivalParticipants.isEmpty())) { try { setNextCycleStart(); setNextFestivalStart(Config.ALT_FESTIVAL_CYCLE_LENGTH - FESTIVAL_SIGNUP_TIME); wait(Config.ALT_FESTIVAL_CYCLE_LENGTH - FESTIVAL_SIGNUP_TIME); for (L2DarknessFestival festivalInst : _festivalInstances.values()) { if (!festivalInst._npcInsts.isEmpty()) { festivalInst.unspawnMobs(); } } } catch (InterruptedException e) { } } else { _noPartyRegister = false; } } /* INITIATION */ // Set the festival timer to 0, as it is just beginning. long elapsedTime = 0; // Create the instances for the festivals in both Oracles, // but only if they have participants signed up for them. for (int i = 0; i < FESTIVAL_COUNT; i++) { if (_duskFestivalParticipants.get(i) != null) _festivalInstances.put(10 + i, new L2DarknessFestival(SevenSigns.CABAL_DUSK, i)); if (_dawnFestivalParticipants.get(i) != null) _festivalInstances.put(20 + i, new L2DarknessFestival(SevenSigns.CABAL_DAWN, i)); } // Prevent future signups while festival is in progress. _festivalInitialized = true; setNextFestivalStart(Config.ALT_FESTIVAL_CYCLE_LENGTH); sendMessageToAll("Festival Guide", "The main event is now starting."); if (Config.DEBUG) _log.info("SevenSignsFestival: The current set of festivals will begin in " + (Config.ALT_FESTIVAL_FIRST_SPAWN / 60000) + " minute(s)."); // Stand by for a short length of time before starting the festival. try { wait(Config.ALT_FESTIVAL_FIRST_SPAWN); } catch (InterruptedException e) { } elapsedTime = Config.ALT_FESTIVAL_FIRST_SPAWN; // Participants can now opt to increase the challenge, if desired. _festivalInProgress = true; /* PROPOGATION */ // Sequentially set all festivals to begin, spawn the Festival Witch and notify participants. for (L2DarknessFestival festivalInst : _festivalInstances.values()) { festivalInst.festivalStart(); festivalInst.sendMessageToParticipants("The festival is about to begin!"); } if (Config.DEBUG) _log.info("SevenSignsFestival: Each of the festivals will end in " + (Config.ALT_FESTIVAL_LENGTH / 60000) + " minutes. New participants can signup then."); // After a short time period, move all idle spawns to the center of the arena. try { wait(Config.ALT_FESTIVAL_FIRST_SWARM - Config.ALT_FESTIVAL_FIRST_SPAWN); } catch (InterruptedException e) { } elapsedTime += Config.ALT_FESTIVAL_FIRST_SWARM - Config.ALT_FESTIVAL_FIRST_SPAWN; for (L2DarknessFestival festivalInst : _festivalInstances.values()) festivalInst.moveMonstersToCenter(); // Stand by until the time comes for the second spawn. try { wait(Config.ALT_FESTIVAL_SECOND_SPAWN - Config.ALT_FESTIVAL_FIRST_SWARM); } catch (InterruptedException e) { } // Spawn an extra set of monsters (archers) on the free platforms with // a faster respawn when killed. for (L2DarknessFestival festivalInst : _festivalInstances.values()) { festivalInst.spawnFestivalMonsters(FESTIVAL_DEFAULT_RESPAWN / 2, 2); festivalInst.sendMessageToParticipants("The festival will end in " + ((Config.ALT_FESTIVAL_LENGTH - Config.ALT_FESTIVAL_SECOND_SPAWN) / 60000) + " minute(s)."); } elapsedTime += Config.ALT_FESTIVAL_SECOND_SPAWN - Config.ALT_FESTIVAL_FIRST_SWARM; // After another short time period, again move all idle spawns to the center of the arena. try { wait(Config.ALT_FESTIVAL_SECOND_SWARM - Config.ALT_FESTIVAL_SECOND_SPAWN); } catch (InterruptedException e) { } for (L2DarknessFestival festivalInst : _festivalInstances.values()) festivalInst.moveMonstersToCenter(); elapsedTime += Config.ALT_FESTIVAL_SECOND_SWARM - Config.ALT_FESTIVAL_SECOND_SPAWN; // Stand by until the time comes for the chests to be spawned. try { wait(Config.ALT_FESTIVAL_CHEST_SPAWN - Config.ALT_FESTIVAL_SECOND_SWARM); } catch (InterruptedException e) { } // Spawn the festival chests, which enable the team to gain greater rewards // for each chest they kill. for (L2DarknessFestival festivalInst : _festivalInstances.values()) { festivalInst.spawnFestivalMonsters(FESTIVAL_DEFAULT_RESPAWN, 3); festivalInst.sendMessageToParticipants("The chests have spawned! Be quick, the festival will end soon."); } elapsedTime += Config.ALT_FESTIVAL_CHEST_SPAWN - Config.ALT_FESTIVAL_SECOND_SWARM; // Stand by and wait until it's time to end the festival. try { wait(Config.ALT_FESTIVAL_LENGTH - elapsedTime); } catch (InterruptedException e) { } // Participants can no longer opt to increase the challenge, as the festival will soon close. _festivalInProgress = false; /* TERMINATION */ // Sequentially begin the ending sequence for all running festivals. for (L2DarknessFestival festivalInst : _festivalInstances.values()) festivalInst.festivalEnd(); // Clear the participants list for the next round of signups. _dawnFestivalParticipants.clear(); _duskFestivalParticipants.clear(); // Allow signups for the next festival cycle. _festivalInitialized = false; sendMessageToAll("Festival Witch", "That will do! I'll move you to the outside soon."); if (Config.DEBUG) _log.info("SevenSignsFestival: The next set of festivals begin in " + getMinsToNextFestival() + " minute(s)."); } /** * Returns the running instance of a festival for the given Oracle and festivalID. *
* A null value is returned if there are no participants in that festival. * * @param oracle * @param festivalId * @return L2DarknessFestival festivalInst */ public final L2DarknessFestival getFestivalInstance(int oracle, int festivalId) { if (!isFestivalInitialized()) return null; /* Compute the offset if a Dusk instance is required. * * ID: 0 1 2 3 4 * Dusk 1: 10 11 12 13 14 * Dawn 2: 20 21 22 23 24 */ festivalId += (oracle == SevenSigns.CABAL_DUSK) ? 10 : 20; return _festivalInstances.get(festivalId); } /** * Returns the number of currently running festivals WITH participants. * * @return int Count */ public final int getInstanceCount() { return _festivalInstances.size(); } } /** * Each running festival is represented by an L2DarknessFestival class. * It contains all the spawn information and data for the running festival. * * All festivals are managed by the FestivalManager class, which must be initialized first. * * @author Tempy */ private class L2DarknessFestival { protected final int _cabal; protected final int _levelRange; protected boolean _challengeIncreased; private FestivalSpawn _startLocation; private FestivalSpawn _witchSpawn; private L2NpcInstance _witchInst; List _npcInsts; private List _participants; private Map _originalLocations; protected L2DarknessFestival(int cabal, int levelRange) { _cabal = cabal; _levelRange = levelRange; _originalLocations = new FastMap(); _npcInsts = new FastList(); if (cabal == SevenSigns.CABAL_DAWN) { _participants = _dawnFestivalParticipants.get(levelRange); _witchSpawn = new FestivalSpawn(FESTIVAL_DAWN_WITCH_SPAWNS[levelRange]); _startLocation = new FestivalSpawn(FESTIVAL_DAWN_PLAYER_SPAWNS[levelRange]); } else { _participants = _duskFestivalParticipants.get(levelRange); _witchSpawn = new FestivalSpawn(FESTIVAL_DUSK_WITCH_SPAWNS[levelRange]); _startLocation = new FestivalSpawn(FESTIVAL_DUSK_PLAYER_SPAWNS[levelRange]); } // FOR TESTING! if (_participants == null) _participants = new FastList(); festivalInit(); } protected void festivalInit() { boolean isPositive; if (Config.DEBUG) _log.info("SevenSignsFestival: Initializing festival for " + SevenSigns.getCabalShortName(_cabal) + " (" + getFestivalName(_levelRange) + ")"); // Teleport all players to arena and notify them. if (_participants.size() > 0) { try { for (L2PcInstance participant : _participants) { _originalLocations.put(participant, new FestivalSpawn(participant.getX(), participant.getY(), participant.getZ(), participant.getHeading())); // Randomize the spawn point around the specific centerpoint for each player. int x = _startLocation._x; int y = _startLocation._y; isPositive = (Rnd.nextInt(2) == 1); if (isPositive) { x += Rnd.nextInt(FESTIVAL_MAX_OFFSET_X); y += Rnd.nextInt(FESTIVAL_MAX_OFFSET_Y); } else { x -= Rnd.nextInt(FESTIVAL_MAX_OFFSET_X); y -= Rnd.nextInt(FESTIVAL_MAX_OFFSET_Y); } participant.getAI().setIntention(CtrlIntention.AI_INTENTION_IDLE); participant.teleToLocation(x, y, _startLocation._z, true); // Remove all buffs from all participants on entry. Works like the skill Cancel. participant.stopAllEffects(); // Remove any stray blood offerings in inventory L2ItemInstance bloodOfferings = participant.getInventory().getItemByItemId(FESTIVAL_OFFERING_ID); if (bloodOfferings != null) participant.destroyItem("SevenSigns", bloodOfferings, null, true); } } catch (NullPointerException e) { // deleteMe handling should teleport party out in case of disconnect } } L2NpcTemplate witchTemplate = NpcTable.getInstance().getTemplate(_witchSpawn._npcId); // Spawn the festival witch for this arena try { L2Spawn npcSpawn = new L2Spawn(witchTemplate); npcSpawn.setLocx(_witchSpawn._x); npcSpawn.setLocy(_witchSpawn._y); npcSpawn.setLocz(_witchSpawn._z); npcSpawn.setHeading(_witchSpawn._heading); npcSpawn.setAmount(1); npcSpawn.setRespawnDelay(1); // Needed as doSpawn() is required to be called also for the NpcInstance it returns. npcSpawn.startRespawn(); SpawnTable.getInstance().addNewSpawn(npcSpawn, false); _witchInst = npcSpawn.doSpawn(); if (Config.DEBUG) _log.fine("SevenSignsFestival: Spawned the Festival Witch " + npcSpawn.getNpcid() + " at " + _witchSpawn._x + " " + _witchSpawn._y + " " + _witchSpawn._z); } catch (Exception e) { _log.warning("SevenSignsFestival: Error while spawning Festival Witch ID " + _witchSpawn._npcId + ": " + e); } // Make it appear as though the Witch has apparated there. MagicSkillUse msu = new MagicSkillUse(_witchInst, _witchInst, 2003, 1, 1, 0); _witchInst.broadcastPacket(msu); // And another one...:D msu = new MagicSkillUse(_witchInst, _witchInst, 2133, 1, 1, 0); _witchInst.broadcastPacket(msu); // Send a message to all participants from the witch. sendMessageToParticipants("The festival will begin in 2 minutes."); } protected void festivalStart() { if (Config.DEBUG) _log.info("SevenSignsFestival: Starting festival for " + SevenSigns.getCabalShortName(_cabal) + " (" + getFestivalName(_levelRange) + ")"); spawnFestivalMonsters(FESTIVAL_DEFAULT_RESPAWN, 0); } protected void moveMonstersToCenter() { boolean isPositive; if (Config.DEBUG) _log.info("SevenSignsFestival: Moving spawns to arena center for festival " + SevenSigns.getCabalShortName(_cabal) + " (" + getFestivalName(_levelRange) + ")"); for (L2FestivalMonsterInstance festivalMob : _npcInsts) { if (festivalMob.isDead()) continue; // Only move monsters that are idle or doing their usual functions. CtrlIntention currIntention = festivalMob.getAI().getIntention(); if (currIntention != CtrlIntention.AI_INTENTION_IDLE && currIntention != CtrlIntention.AI_INTENTION_ACTIVE) continue; int x = _startLocation._x; int y = _startLocation._y; /* * Random X and Y coords around the player start location, up to half of the * maximum allowed offset are generated to prevent the mobs from all moving * to the exact same place. */ isPositive = (Rnd.nextInt(2) == 1); if (isPositive) { x += Rnd.nextInt(FESTIVAL_MAX_OFFSET_X); y += Rnd.nextInt(FESTIVAL_MAX_OFFSET_Y); } else { x -= Rnd.nextInt(FESTIVAL_MAX_OFFSET_X); y -= Rnd.nextInt(FESTIVAL_MAX_OFFSET_Y); } L2CharPosition moveTo = new L2CharPosition(x, y, _startLocation._z, Rnd.nextInt(65536)); festivalMob.setRunning(); festivalMob.getAI().setIntention(CtrlIntention.AI_INTENTION_MOVE_TO, moveTo); } } public void setSpawnRate(int respawnDelay) { if (Config.DEBUG) _log.info("SevenSignsFestival: Modifying spawn rate of festival mobs to " + respawnDelay + " ms for festival " + SevenSigns.getCabalShortName(_cabal) + " (" + getFestivalName(_levelRange) + ")"); for (L2FestivalMonsterInstance monsterInst : _npcInsts) monsterInst.getSpawn().setRespawnDelay(respawnDelay); } /** * Used to spawn monsters unique to the festival. *
* Valid SpawnTypes:
* 0 - All Primary Monsters (starting monsters) *
* 1 - Same as 0, but without archers/marksmen. (used for challenge increase) *
* 2 - Secondary Monsters (archers) *
* 3 - Festival Chests * * @param respawnDelay * @param spawnType */ protected void spawnFestivalMonsters(int respawnDelay, int spawnType) { int[][] _npcSpawns = null; switch (spawnType) { case 0: case 1: _npcSpawns = (_cabal == SevenSigns.CABAL_DAWN) ? FESTIVAL_DAWN_PRIMARY_SPAWNS[_levelRange] : FESTIVAL_DUSK_PRIMARY_SPAWNS[_levelRange]; break; case 2: _npcSpawns = (_cabal == SevenSigns.CABAL_DAWN) ? FESTIVAL_DAWN_SECONDARY_SPAWNS[_levelRange] : FESTIVAL_DUSK_SECONDARY_SPAWNS[_levelRange]; break; case 3: _npcSpawns = (_cabal == SevenSigns.CABAL_DAWN) ? FESTIVAL_DAWN_CHEST_SPAWNS[_levelRange] : FESTIVAL_DUSK_CHEST_SPAWNS[_levelRange]; break; } for (int i = 0; i < _npcSpawns.length; i++) { FestivalSpawn currSpawn = new FestivalSpawn(_npcSpawns[i]); // Only spawn archers/marksmen if specified to do so. if (spawnType == 1 && isFestivalArcher(currSpawn._npcId)) continue; L2NpcTemplate npcTemplate = NpcTable.getInstance().getTemplate(currSpawn._npcId); try { L2Spawn npcSpawn = new L2Spawn(npcTemplate); npcSpawn.setLocx(currSpawn._x); npcSpawn.setLocy(currSpawn._y); npcSpawn.setLocz(currSpawn._z); npcSpawn.setHeading(Rnd.nextInt(65536)); npcSpawn.setAmount(1); npcSpawn.setRespawnDelay(respawnDelay); // Needed as doSpawn() is required to be called also for the NpcInstance it returns. npcSpawn.startRespawn(); SpawnTable.getInstance().addNewSpawn(npcSpawn, false); L2FestivalMonsterInstance festivalMob = (L2FestivalMonsterInstance)npcSpawn.doSpawn(); // Set the offering bonus to 2x or 5x the amount per kill, // if this spawn is part of an increased challenge or is a festival chest. if (spawnType == 1) festivalMob.setOfferingBonus(2); else if (spawnType == 3) festivalMob.setOfferingBonus(5); _npcInsts.add(festivalMob); if (Config.DEBUG) _log.fine("SevenSignsFestival: Spawned NPC ID " + currSpawn._npcId + " at " + currSpawn._x + " " + currSpawn._y + " " + currSpawn._z); } catch (Exception e) { _log.warning("SevenSignsFestival: Error while spawning NPC ID " + currSpawn._npcId + ": " + e); } } } protected boolean increaseChallenge() { if (_challengeIncreased) return false; // Set this flag to true to make sure that this can only be done once. _challengeIncreased = true; if (Config.DEBUG) _log.info("SevenSignsFestival: " + _participants.get(0).getName() + "'s team have opted to increase the festival challenge!"); // Spawn more festival monsters, but this time with a twist. spawnFestivalMonsters(FESTIVAL_DEFAULT_RESPAWN, 1); return true; } public void sendMessageToParticipants(String message) { if (_participants.size() > 0) { CreatureSay cs = new CreatureSay(_witchInst.getObjectId(), 0, "Festival Witch", message); for (L2PcInstance participant : _participants) { try { participant.sendPacket(cs); } catch (NullPointerException e) { } } } } protected void festivalEnd() { if (Config.DEBUG) _log.info("SevenSignsFestival: Ending festival for " + SevenSigns.getCabalShortName(_cabal) + " (" + getFestivalName(_levelRange) + ")"); if (_participants.size() > 0) { for (L2PcInstance participant : _participants) { try { relocatePlayer(participant, false); participant.sendMessage("The festival has ended. Your party leader must now register your score before the next festival takes place."); } catch (NullPointerException e) { } } if (_cabal == SevenSigns.CABAL_DAWN) _dawnPreviousParticipants.put(_levelRange, _participants); else _duskPreviousParticipants.put(_levelRange, _participants); } _participants = null; unspawnMobs(); } protected void unspawnMobs() { // Delete all the NPCs in the current festival arena. if (_witchInst != null) { _witchInst.getSpawn().stopRespawn(); _witchInst.deleteMe(); } if (_npcInsts != null) for (L2FestivalMonsterInstance monsterInst : _npcInsts) if (monsterInst != null) { monsterInst.getSpawn().stopRespawn(); monsterInst.deleteMe(); } } public void relocatePlayer(L2PcInstance participant, boolean isRemoving) { try { FestivalSpawn origPosition = _originalLocations.get(participant); if (isRemoving) _originalLocations.remove(participant); participant.getAI().setIntention(CtrlIntention.AI_INTENTION_IDLE); participant.teleToLocation(origPosition._x, origPosition._y, origPosition._z, true); participant.sendMessage("You have been removed from the festival arena."); } catch (Exception e) { // If an exception occurs, just move the player to the nearest town. try { participant.teleToLocation(MapRegionTable.TeleportWhereType.Town); participant.sendMessage("You have been removed from the festival arena."); } catch (NullPointerException e2) {} } } } private class FestivalSpawn { protected final int _x; protected final int _y; protected final int _z; protected final int _heading; protected final int _npcId; protected FestivalSpawn(int x, int y, int z, int heading) { _x = x; _y = y; _z = z; // Generate a random heading if no positive one given. _heading = (heading < 0) ? Rnd.nextInt(65536) : heading; _npcId = -1; } protected FestivalSpawn(int[] spawnData) { _x = spawnData[0]; _y = spawnData[1]; _z = spawnData[2]; _heading = (spawnData[3] < 0) ? Rnd.nextInt(65536) : spawnData[3]; if (spawnData.length > 4) _npcId = spawnData[4]; else _npcId = -1; } } }