SevenSigns.java 49 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416
  1. /*
  2. * This program is free software: you can redistribute it and/or modify it under
  3. * the terms of the GNU General Public License as published by the Free Software
  4. * Foundation, either version 3 of the License, or (at your option) any later
  5. * version.
  6. *
  7. * This program is distributed in the hope that it will be useful, but WITHOUT
  8. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
  9. * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
  10. * details.
  11. *
  12. * You should have received a copy of the GNU General Public License along with
  13. * this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. package net.sf.l2j.gameserver;
  16. import java.sql.Connection;
  17. import java.sql.PreparedStatement;
  18. import java.sql.ResultSet;
  19. import java.sql.SQLException;
  20. import java.util.Calendar;
  21. import java.util.Map;
  22. import java.util.logging.Logger;
  23. import javolution.util.FastMap;
  24. import net.sf.l2j.Config;
  25. import net.sf.l2j.L2DatabaseFactory;
  26. import net.sf.l2j.gameserver.datatables.MapRegionTable;
  27. import net.sf.l2j.gameserver.instancemanager.CastleManager;
  28. import net.sf.l2j.gameserver.model.AutoChatHandler;
  29. import net.sf.l2j.gameserver.model.AutoSpawnHandler;
  30. import net.sf.l2j.gameserver.model.L2World;
  31. import net.sf.l2j.gameserver.model.AutoSpawnHandler.AutoSpawnInstance;
  32. import net.sf.l2j.gameserver.model.actor.instance.L2PcInstance;
  33. import net.sf.l2j.gameserver.network.SystemMessageId;
  34. import net.sf.l2j.gameserver.serverpackets.SSQInfo;
  35. import net.sf.l2j.gameserver.serverpackets.SystemMessage;
  36. import net.sf.l2j.gameserver.templates.StatsSet;
  37. /**
  38. * Seven Signs Engine
  39. *
  40. * TODO:
  41. * - Implementation of the Seal of Strife for sieges.
  42. *
  43. * @author Tempy
  44. */
  45. public class SevenSigns
  46. {
  47. protected static final Logger _log = Logger.getLogger(SevenSigns.class.getName());
  48. private static SevenSigns _instance;
  49. // Basic Seven Signs Constants \\
  50. public static final String SEVEN_SIGNS_DATA_FILE = "config/signs.properties";
  51. public static final String SEVEN_SIGNS_HTML_PATH = "data/html/seven_signs/";
  52. public static final int CABAL_NULL = 0;
  53. public static final int CABAL_DUSK = 1;
  54. public static final int CABAL_DAWN = 2;
  55. public static final int SEAL_NULL = 0;
  56. public static final int SEAL_AVARICE = 1;
  57. public static final int SEAL_GNOSIS = 2;
  58. public static final int SEAL_STRIFE = 3;
  59. public static final int PERIOD_COMP_RECRUITING = 0;
  60. public static final int PERIOD_COMPETITION = 1;
  61. public static final int PERIOD_COMP_RESULTS = 2;
  62. public static final int PERIOD_SEAL_VALIDATION = 3;
  63. public static final int PERIOD_START_HOUR = 18;
  64. public static final int PERIOD_START_MINS = 00;
  65. public static final int PERIOD_START_DAY = Calendar.MONDAY;
  66. // The quest event and seal validation periods last for approximately one week
  67. // with a 15 minutes "interval" period sandwiched between them.
  68. public static final int PERIOD_MINOR_LENGTH = 900000;
  69. public static final int PERIOD_MAJOR_LENGTH = 604800000 - PERIOD_MINOR_LENGTH;
  70. public static final int ANCIENT_ADENA_ID = 5575;
  71. public static final int RECORD_SEVEN_SIGNS_ID = 5707;
  72. public static final int CERTIFICATE_OF_APPROVAL_ID = 6388;
  73. public static final int RECORD_SEVEN_SIGNS_COST = 500;
  74. public static final int ADENA_JOIN_DAWN_COST = 50000;
  75. // NPC Related Constants \\
  76. public static final int ORATOR_NPC_ID = 31094;
  77. public static final int PREACHER_NPC_ID = 31093;
  78. public static final int MAMMON_MERCHANT_ID = 31113;
  79. public static final int MAMMON_BLACKSMITH_ID = 31126;
  80. public static final int MAMMON_MARKETEER_ID = 31092;
  81. public static final int SPIRIT_IN_ID = 31111;
  82. public static final int SPIRIT_OUT_ID = 31112;
  83. public static final int LILITH_NPC_ID = 25283;
  84. public static final int ANAKIM_NPC_ID = 25286;
  85. public static final int CREST_OF_DAWN_ID = 31170;
  86. public static final int CREST_OF_DUSK_ID = 31171;
  87. // Seal Stone Related Constants \\
  88. public static final int SEAL_STONE_BLUE_ID = 6360;
  89. public static final int SEAL_STONE_GREEN_ID = 6361;
  90. public static final int SEAL_STONE_RED_ID = 6362;
  91. public static final int SEAL_STONE_BLUE_VALUE = 3;
  92. public static final int SEAL_STONE_GREEN_VALUE = 5;
  93. public static final int SEAL_STONE_RED_VALUE = 10;
  94. public static final int BLUE_CONTRIB_POINTS = 3;
  95. public static final int GREEN_CONTRIB_POINTS = 5;
  96. public static final int RED_CONTRIB_POINTS = 10;
  97. private final Calendar _calendar = Calendar.getInstance();
  98. protected int _activePeriod;
  99. protected int _currentCycle;
  100. protected double _dawnStoneScore;
  101. protected double _duskStoneScore;
  102. protected int _dawnFestivalScore;
  103. protected int _duskFestivalScore;
  104. protected int _compWinner;
  105. protected int _previousWinner;
  106. private Map<Integer, StatsSet> _signsPlayerData;
  107. private Map<Integer, Integer> _signsSealOwners;
  108. private Map<Integer, Integer> _signsDuskSealTotals;
  109. private Map<Integer, Integer> _signsDawnSealTotals;
  110. private static AutoSpawnInstance _merchantSpawn;
  111. private static AutoSpawnInstance _blacksmithSpawn;
  112. private static AutoSpawnInstance _spiritInSpawn;
  113. private static AutoSpawnInstance _spiritOutSpawn;
  114. private static AutoSpawnInstance _lilithSpawn;
  115. private static AutoSpawnInstance _anakimSpawn;
  116. private static AutoSpawnInstance _crestofdawnspawn;
  117. private static AutoSpawnInstance _crestofduskspawn;
  118. private static Map<Integer, AutoSpawnInstance> _oratorSpawns;
  119. private static Map<Integer, AutoSpawnInstance> _preacherSpawns;
  120. private static Map<Integer, AutoSpawnInstance> _marketeerSpawns;
  121. public SevenSigns()
  122. {
  123. _signsPlayerData = new FastMap<Integer, StatsSet>();
  124. _signsSealOwners = new FastMap<Integer, Integer>();
  125. _signsDuskSealTotals = new FastMap<Integer, Integer>();
  126. _signsDawnSealTotals = new FastMap<Integer, Integer>();
  127. try
  128. {
  129. restoreSevenSignsData();
  130. }
  131. catch (Exception e) {
  132. _log.severe("SevenSigns: Failed to load configuration: " + e);
  133. }
  134. _log.info("SevenSigns: Currently in the " + getCurrentPeriodName() + " period!");
  135. initializeSeals();
  136. if (isSealValidationPeriod())
  137. if (getCabalHighestScore() == CABAL_NULL)
  138. _log.info("SevenSigns: The competition ended with a tie last week.");
  139. else
  140. _log.info("SevenSigns: The " + getCabalName(getCabalHighestScore()) + " were victorious last week.");
  141. else
  142. if (getCabalHighestScore() == CABAL_NULL)
  143. _log.info("SevenSigns: The competition, if the current trend continues, will end in a tie this week.");
  144. else
  145. _log.info("SevenSigns: The " + getCabalName(getCabalHighestScore()) + " are in the lead this week.");
  146. synchronized (this)
  147. {
  148. setCalendarForNextPeriodChange();
  149. long milliToChange = getMilliToPeriodChange();
  150. // Schedule a time for the next period change.
  151. SevenSignsPeriodChange sspc = new SevenSignsPeriodChange();
  152. ThreadPoolManager.getInstance().scheduleGeneral(sspc, milliToChange);
  153. // Thanks to http://rainbow.arch.scriptmania.com/scripts/timezone_countdown.html for help with this.
  154. double numSecs = (milliToChange / 1000) % 60;
  155. double countDown = ((milliToChange / 1000) - numSecs) / 60;
  156. int numMins = (int)Math.floor(countDown % 60);
  157. countDown = (countDown - numMins) / 60;
  158. int numHours = (int)Math.floor(countDown % 24);
  159. int numDays = (int)Math.floor((countDown - numHours) / 24);
  160. _log.info("SevenSigns: Next period begins in " + numDays + " days, " + numHours + " hours and " + numMins + " mins.");
  161. }
  162. }
  163. /**
  164. * Registers all random spawns and auto-chats for Seven Signs NPCs,
  165. * along with spawns for the Preachers of Doom and Orators of Revelations
  166. * at the beginning of the Seal Validation period.
  167. *
  168. */
  169. public void spawnSevenSignsNPC()
  170. {
  171. _merchantSpawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(MAMMON_MERCHANT_ID, false);
  172. _blacksmithSpawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(MAMMON_BLACKSMITH_ID, false);
  173. _marketeerSpawns = AutoSpawnHandler.getInstance().getAutoSpawnInstances(MAMMON_MARKETEER_ID);
  174. _spiritInSpawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(SPIRIT_IN_ID, false);
  175. _spiritOutSpawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(SPIRIT_OUT_ID, false);
  176. _lilithSpawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(LILITH_NPC_ID, false);
  177. _anakimSpawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(ANAKIM_NPC_ID, false);
  178. _crestofdawnspawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(CREST_OF_DAWN_ID, false);
  179. _crestofduskspawn = AutoSpawnHandler.getInstance().getAutoSpawnInstance(CREST_OF_DUSK_ID, false);
  180. _oratorSpawns = AutoSpawnHandler.getInstance().getAutoSpawnInstances(ORATOR_NPC_ID);
  181. _preacherSpawns = AutoSpawnHandler.getInstance().getAutoSpawnInstances(PREACHER_NPC_ID);
  182. if (isSealValidationPeriod() || isCompResultsPeriod())
  183. {
  184. for (AutoSpawnInstance spawnInst: _marketeerSpawns.values())
  185. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, true);
  186. if (getSealOwner(SEAL_GNOSIS) == getCabalHighestScore() && getSealOwner(SEAL_GNOSIS) != CABAL_NULL)
  187. {
  188. if (!Config.ANNOUNCE_MAMMON_SPAWN)
  189. _blacksmithSpawn.setBroadcast(false);
  190. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_blacksmithSpawn.getObjectId(), true).isSpawnActive())
  191. AutoSpawnHandler.getInstance().setSpawnActive(_blacksmithSpawn, true);
  192. for (AutoSpawnInstance spawnInst: _oratorSpawns.values())
  193. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(spawnInst.getObjectId(), true).isSpawnActive())
  194. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, true);
  195. for (AutoSpawnInstance spawnInst: _preacherSpawns.values())
  196. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(spawnInst.getObjectId(), true).isSpawnActive())
  197. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, true);
  198. if (!AutoChatHandler.getInstance().getAutoChatInstance(PREACHER_NPC_ID, false).isActive() && !AutoChatHandler.getInstance().getAutoChatInstance(ORATOR_NPC_ID, false).isActive())
  199. AutoChatHandler.getInstance().setAutoChatActive(true);
  200. }
  201. else
  202. {
  203. AutoSpawnHandler.getInstance().setSpawnActive(_blacksmithSpawn, false);
  204. for (AutoSpawnInstance spawnInst: _oratorSpawns.values())
  205. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, false);
  206. for (AutoSpawnInstance spawnInst: _preacherSpawns.values())
  207. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, false);
  208. AutoChatHandler.getInstance().setAutoChatActive(false);
  209. }
  210. if (getSealOwner(SEAL_AVARICE) == getCabalHighestScore() && getSealOwner(SEAL_AVARICE) != CABAL_NULL)
  211. {
  212. if (!Config.ANNOUNCE_MAMMON_SPAWN)
  213. _merchantSpawn.setBroadcast(false);
  214. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_merchantSpawn.getObjectId(), true).isSpawnActive())
  215. AutoSpawnHandler.getInstance().setSpawnActive(_merchantSpawn, true);
  216. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_spiritInSpawn.getObjectId(), true).isSpawnActive())
  217. AutoSpawnHandler.getInstance().setSpawnActive(_spiritInSpawn, true);
  218. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_spiritOutSpawn.getObjectId(), true).isSpawnActive())
  219. AutoSpawnHandler.getInstance().setSpawnActive(_spiritOutSpawn, true);
  220. switch (getCabalHighestScore())
  221. {
  222. case CABAL_DAWN:
  223. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_lilithSpawn.getObjectId(), true).isSpawnActive())
  224. AutoSpawnHandler.getInstance().setSpawnActive(_lilithSpawn, true);
  225. AutoSpawnHandler.getInstance().setSpawnActive(_anakimSpawn, false);
  226. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_crestofdawnspawn.getObjectId(), true).isSpawnActive())
  227. AutoSpawnHandler.getInstance().setSpawnActive(_crestofdawnspawn, true);
  228. AutoSpawnHandler.getInstance().setSpawnActive(_crestofduskspawn, false);
  229. break;
  230. case CABAL_DUSK:
  231. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_anakimSpawn.getObjectId(), true).isSpawnActive())
  232. AutoSpawnHandler.getInstance().setSpawnActive(_anakimSpawn, true);
  233. AutoSpawnHandler.getInstance().setSpawnActive(_lilithSpawn, false);
  234. if (!AutoSpawnHandler.getInstance().getAutoSpawnInstance(_crestofduskspawn.getObjectId(), true).isSpawnActive())
  235. AutoSpawnHandler.getInstance().setSpawnActive(_crestofduskspawn, true);
  236. AutoSpawnHandler.getInstance().setSpawnActive(_crestofdawnspawn, false);
  237. break;
  238. }
  239. }
  240. else
  241. {
  242. AutoSpawnHandler.getInstance().setSpawnActive(_merchantSpawn, false);
  243. AutoSpawnHandler.getInstance().setSpawnActive(_lilithSpawn, false);
  244. AutoSpawnHandler.getInstance().setSpawnActive(_anakimSpawn, false);
  245. AutoSpawnHandler.getInstance().setSpawnActive(_crestofdawnspawn, false);
  246. AutoSpawnHandler.getInstance().setSpawnActive(_crestofduskspawn, false);
  247. AutoSpawnHandler.getInstance().setSpawnActive(_spiritInSpawn, false);
  248. AutoSpawnHandler.getInstance().setSpawnActive(_spiritOutSpawn, false);
  249. }
  250. }
  251. else
  252. {
  253. AutoSpawnHandler.getInstance().setSpawnActive(_merchantSpawn, false);
  254. AutoSpawnHandler.getInstance().setSpawnActive(_blacksmithSpawn, false);
  255. AutoSpawnHandler.getInstance().setSpawnActive(_lilithSpawn, false);
  256. AutoSpawnHandler.getInstance().setSpawnActive(_anakimSpawn, false);
  257. AutoSpawnHandler.getInstance().setSpawnActive(_crestofdawnspawn, false);
  258. AutoSpawnHandler.getInstance().setSpawnActive(_crestofduskspawn, false);
  259. AutoSpawnHandler.getInstance().setSpawnActive(_spiritInSpawn, false);
  260. AutoSpawnHandler.getInstance().setSpawnActive(_spiritOutSpawn, false);
  261. for (AutoSpawnInstance spawnInst: _oratorSpawns.values())
  262. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, false);
  263. for (AutoSpawnInstance spawnInst: _preacherSpawns.values())
  264. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, false);
  265. for (AutoSpawnInstance spawnInst: _marketeerSpawns.values())
  266. AutoSpawnHandler.getInstance().setSpawnActive(spawnInst, false);
  267. AutoChatHandler.getInstance().setAutoChatActive(false);
  268. }
  269. }
  270. public static SevenSigns getInstance()
  271. {
  272. if (_instance == null)
  273. _instance = new SevenSigns();
  274. return _instance;
  275. }
  276. public static int calcContributionScore(int blueCount, int greenCount, int redCount)
  277. {
  278. int contrib = blueCount * BLUE_CONTRIB_POINTS;
  279. contrib += greenCount * GREEN_CONTRIB_POINTS;
  280. contrib += redCount * RED_CONTRIB_POINTS;
  281. return contrib;
  282. }
  283. public static int calcAncientAdenaReward(int blueCount, int greenCount, int redCount)
  284. {
  285. int reward = blueCount * SEAL_STONE_BLUE_VALUE;
  286. reward += greenCount * SEAL_STONE_GREEN_VALUE;
  287. reward += redCount * SEAL_STONE_RED_VALUE;
  288. return reward;
  289. }
  290. public static final String getCabalShortName(int cabal)
  291. {
  292. switch (cabal)
  293. {
  294. case CABAL_DAWN:
  295. return "dawn";
  296. case CABAL_DUSK:
  297. return "dusk";
  298. }
  299. return "No Cabal";
  300. }
  301. public static final String getCabalName(int cabal)
  302. {
  303. switch (cabal)
  304. {
  305. case CABAL_DAWN:
  306. return "Lords of Dawn";
  307. case CABAL_DUSK:
  308. return "Revolutionaries of Dusk";
  309. }
  310. return "No Cabal";
  311. }
  312. public static final String getSealName(int seal, boolean shortName)
  313. {
  314. String sealName = (!shortName) ? "Seal of " : "";
  315. switch (seal)
  316. {
  317. case SEAL_AVARICE:
  318. sealName += "Avarice";
  319. break;
  320. case SEAL_GNOSIS:
  321. sealName += "Gnosis";
  322. break;
  323. case SEAL_STRIFE:
  324. sealName += "Strife";
  325. break;
  326. }
  327. return sealName;
  328. }
  329. public final int getCurrentCycle()
  330. {
  331. return _currentCycle;
  332. }
  333. public final int getCurrentPeriod()
  334. {
  335. return _activePeriod;
  336. }
  337. private final int getDaysToPeriodChange()
  338. {
  339. int numDays = _calendar.get(Calendar.DAY_OF_WEEK) - PERIOD_START_DAY;
  340. if (numDays < 0)
  341. return 0 - numDays;
  342. return 7 - numDays;
  343. }
  344. public final long getMilliToPeriodChange()
  345. {
  346. long currTimeMillis = System.currentTimeMillis();
  347. long changeTimeMillis = _calendar.getTimeInMillis();
  348. return (changeTimeMillis - currTimeMillis);
  349. }
  350. protected void setCalendarForNextPeriodChange()
  351. {
  352. // Calculate the number of days until the next period
  353. // A period starts at 18:00 pm (local time), like on official servers.
  354. switch (getCurrentPeriod())
  355. {
  356. case PERIOD_SEAL_VALIDATION:
  357. case PERIOD_COMPETITION:
  358. int daysToChange = getDaysToPeriodChange();
  359. if (daysToChange == 7)
  360. if (_calendar.get(Calendar.HOUR_OF_DAY) < PERIOD_START_HOUR)
  361. daysToChange = 0;
  362. else if (_calendar.get(Calendar.HOUR_OF_DAY) == PERIOD_START_HOUR && _calendar.get(Calendar.MINUTE) < PERIOD_START_MINS)
  363. daysToChange = 0;
  364. // Otherwise...
  365. if (daysToChange > 0)
  366. _calendar.add(Calendar.DATE, daysToChange);
  367. _calendar.set(Calendar.HOUR_OF_DAY, PERIOD_START_HOUR);
  368. _calendar.set(Calendar.MINUTE, PERIOD_START_MINS);
  369. break;
  370. case PERIOD_COMP_RECRUITING:
  371. case PERIOD_COMP_RESULTS:
  372. _calendar.add(Calendar.MILLISECOND, PERIOD_MINOR_LENGTH);
  373. break;
  374. }
  375. }
  376. public final String getCurrentPeriodName()
  377. {
  378. String periodName = null;
  379. switch (_activePeriod)
  380. {
  381. case PERIOD_COMP_RECRUITING:
  382. periodName = "Quest Event Initialization";
  383. break;
  384. case PERIOD_COMPETITION:
  385. periodName = "Competition (Quest Event)";
  386. break;
  387. case PERIOD_COMP_RESULTS:
  388. periodName = "Quest Event Results";
  389. break;
  390. case PERIOD_SEAL_VALIDATION:
  391. periodName = "Seal Validation";
  392. break;
  393. }
  394. return periodName;
  395. }
  396. public final boolean isSealValidationPeriod()
  397. {
  398. return (_activePeriod == PERIOD_SEAL_VALIDATION);
  399. }
  400. public final boolean isCompResultsPeriod()
  401. {
  402. return (_activePeriod == PERIOD_COMP_RESULTS);
  403. }
  404. public final int getCurrentScore(int cabal)
  405. {
  406. double totalStoneScore = _dawnStoneScore + _duskStoneScore;
  407. switch (cabal)
  408. {
  409. case CABAL_NULL:
  410. return 0;
  411. case CABAL_DAWN:
  412. return Math.round((float)(_dawnStoneScore / ((float)totalStoneScore == 0 ? 1 : totalStoneScore)) * 500) + _dawnFestivalScore;
  413. case CABAL_DUSK:
  414. return Math.round((float)(_duskStoneScore / ((float)totalStoneScore == 0 ? 1 : totalStoneScore)) * 500) + _duskFestivalScore;
  415. }
  416. return 0;
  417. }
  418. public final double getCurrentStoneScore(int cabal)
  419. {
  420. switch (cabal)
  421. {
  422. case CABAL_NULL:
  423. return 0;
  424. case CABAL_DAWN:
  425. return _dawnStoneScore;
  426. case CABAL_DUSK:
  427. return _duskStoneScore;
  428. }
  429. return 0;
  430. }
  431. public final int getCurrentFestivalScore(int cabal)
  432. {
  433. switch (cabal)
  434. {
  435. case CABAL_NULL:
  436. return 0;
  437. case CABAL_DAWN:
  438. return _dawnFestivalScore;
  439. case CABAL_DUSK:
  440. return _duskFestivalScore;
  441. }
  442. return 0;
  443. }
  444. public final int getCabalHighestScore()
  445. {
  446. if (getCurrentScore(CABAL_DUSK) == getCurrentScore(CABAL_DAWN))
  447. return CABAL_NULL;
  448. else if (getCurrentScore(CABAL_DUSK) > getCurrentScore(CABAL_DAWN))
  449. return CABAL_DUSK;
  450. else
  451. return CABAL_DAWN;
  452. }
  453. public final int getSealOwner(int seal)
  454. {
  455. return _signsSealOwners.get(seal);
  456. }
  457. public final int getSealProportion(int seal, int cabal)
  458. {
  459. if (cabal == CABAL_NULL)
  460. return 0;
  461. else if (cabal == CABAL_DUSK)
  462. return _signsDuskSealTotals.get(seal);
  463. else
  464. return _signsDawnSealTotals.get(seal);
  465. }
  466. public final int getTotalMembers(int cabal)
  467. {
  468. int cabalMembers = 0;
  469. String cabalName = getCabalShortName(cabal);
  470. for (StatsSet sevenDat : _signsPlayerData.values())
  471. if (sevenDat.getString("cabal").equals(cabalName))
  472. cabalMembers++;
  473. return cabalMembers;
  474. }
  475. public final StatsSet getPlayerData(L2PcInstance player)
  476. {
  477. if (!hasRegisteredBefore(player))
  478. return null;
  479. return _signsPlayerData.get(player.getObjectId());
  480. }
  481. public int getPlayerStoneContrib(L2PcInstance player)
  482. {
  483. if (!hasRegisteredBefore(player))
  484. return 0;
  485. int stoneCount = 0;
  486. StatsSet currPlayer = getPlayerData(player);
  487. stoneCount += currPlayer.getInteger("red_stones");
  488. stoneCount += currPlayer.getInteger("green_stones");
  489. stoneCount += currPlayer.getInteger("blue_stones");
  490. return stoneCount;
  491. }
  492. public int getPlayerContribScore(L2PcInstance player)
  493. {
  494. if (!hasRegisteredBefore(player))
  495. return 0;
  496. StatsSet currPlayer = getPlayerData(player);
  497. return currPlayer.getInteger("contribution_score");
  498. }
  499. public int getPlayerAdenaCollect(L2PcInstance player)
  500. {
  501. if (!hasRegisteredBefore(player))
  502. return 0;
  503. return _signsPlayerData.get(player.getObjectId()).getInteger("ancient_adena_amount");
  504. }
  505. public int getPlayerSeal(L2PcInstance player)
  506. {
  507. if (!hasRegisteredBefore(player))
  508. return SEAL_NULL;
  509. return getPlayerData(player).getInteger("seal");
  510. }
  511. public int getPlayerCabal(L2PcInstance player)
  512. {
  513. if (!hasRegisteredBefore(player))
  514. return CABAL_NULL;
  515. String playerCabal = getPlayerData(player).getString("cabal");
  516. if (playerCabal.equalsIgnoreCase("dawn"))
  517. return CABAL_DAWN;
  518. else if (playerCabal.equalsIgnoreCase("dusk"))
  519. return CABAL_DUSK;
  520. else
  521. return CABAL_NULL;
  522. }
  523. /**
  524. * Restores all Seven Signs data and settings, usually called at server startup.
  525. *
  526. * @throws Exception
  527. */
  528. protected void restoreSevenSignsData()
  529. {
  530. Connection con = null;
  531. PreparedStatement statement = null;
  532. ResultSet rset = null;
  533. try
  534. {
  535. con = L2DatabaseFactory.getInstance().getConnection();
  536. statement = con.prepareStatement("SELECT char_obj_id, cabal, seal, red_stones, green_stones, blue_stones, " +
  537. "ancient_adena_amount, contribution_score FROM seven_signs");
  538. rset = statement.executeQuery();
  539. while (rset.next())
  540. {
  541. int charObjId = rset.getInt("char_obj_id");
  542. StatsSet sevenDat = new StatsSet();
  543. sevenDat.set("char_obj_id", charObjId);
  544. sevenDat.set("cabal", rset.getString("cabal"));
  545. sevenDat.set("seal", rset.getInt("seal"));
  546. sevenDat.set("red_stones", rset.getInt("red_stones"));
  547. sevenDat.set("green_stones", rset.getInt("green_stones"));
  548. sevenDat.set("blue_stones", rset.getInt("blue_stones"));
  549. sevenDat.set("ancient_adena_amount", rset.getDouble("ancient_adena_amount"));
  550. sevenDat.set("contribution_score", rset.getDouble("contribution_score"));
  551. if (Config.DEBUG)
  552. _log.info("SevenSigns: Loaded data from DB for char ID " + charObjId + " (" + sevenDat.getString("cabal") + ")");
  553. _signsPlayerData.put(charObjId, sevenDat);
  554. }
  555. rset.close();
  556. statement.close();
  557. statement = con.prepareStatement("SELECT * FROM seven_signs_status WHERE id=0");
  558. rset = statement.executeQuery();
  559. while (rset.next())
  560. {
  561. _currentCycle = rset.getInt("current_cycle");
  562. _activePeriod = rset.getInt("active_period");
  563. _previousWinner = rset.getInt("previous_winner");
  564. _dawnStoneScore = rset.getDouble("dawn_stone_score");
  565. _dawnFestivalScore = rset.getInt("dawn_festival_score");
  566. _duskStoneScore = rset.getDouble("dusk_stone_score");
  567. _duskFestivalScore = rset.getInt("dusk_festival_score");
  568. _signsSealOwners.put(SEAL_AVARICE, rset.getInt("avarice_owner"));
  569. _signsSealOwners.put(SEAL_GNOSIS, rset.getInt("gnosis_owner"));
  570. _signsSealOwners.put(SEAL_STRIFE, rset.getInt("strife_owner"));
  571. _signsDawnSealTotals.put(SEAL_AVARICE, rset.getInt("avarice_dawn_score"));
  572. _signsDawnSealTotals.put(SEAL_GNOSIS, rset.getInt("gnosis_dawn_score"));
  573. _signsDawnSealTotals.put(SEAL_STRIFE, rset.getInt("strife_dawn_score"));
  574. _signsDuskSealTotals.put(SEAL_AVARICE, rset.getInt("avarice_dusk_score"));
  575. _signsDuskSealTotals.put(SEAL_GNOSIS, rset.getInt("gnosis_dusk_score"));
  576. _signsDuskSealTotals.put(SEAL_STRIFE, rset.getInt("strife_dusk_score"));
  577. }
  578. rset.close();
  579. statement.close();
  580. statement = con.prepareStatement("UPDATE seven_signs_status SET date=? WHERE id=0");
  581. statement.setInt(1, Calendar.getInstance().get(Calendar.DAY_OF_WEEK));
  582. statement.execute();
  583. statement.close();
  584. con.close();
  585. }
  586. catch (SQLException e)
  587. {
  588. _log.severe("SevenSigns: Unable to load Seven Signs data from database: " + e);
  589. }
  590. finally
  591. {
  592. try
  593. {
  594. rset.close();
  595. statement.close();
  596. con.close();
  597. }
  598. catch (Exception e) {}
  599. }
  600. // Festival data is loaded now after the Seven Signs engine data.
  601. }
  602. /**
  603. * Saves all Seven Signs data, both to the database and properties file (if updateSettings = True).
  604. * Often called to preserve data integrity and synchronization with DB, in case of errors.
  605. * <BR>
  606. * If player != null, just that player's data is updated in the database, otherwise all player's data is
  607. * sequentially updated.
  608. *
  609. * @param player
  610. * @param updateSettings
  611. * @throws Exception
  612. */
  613. public void saveSevenSignsData(L2PcInstance player, boolean updateSettings)
  614. {
  615. Connection con = null;
  616. PreparedStatement statement = null;
  617. if (Config.DEBUG)
  618. _log.info("SevenSigns: Saving data to disk.");
  619. try
  620. {
  621. con = L2DatabaseFactory.getInstance().getConnection();
  622. for (StatsSet sevenDat : _signsPlayerData.values())
  623. {
  624. if (player != null)
  625. if (sevenDat.getInteger("char_obj_id") != player.getObjectId())
  626. continue;
  627. statement = con.prepareStatement(
  628. "UPDATE seven_signs SET cabal=?, seal=?, red_stones=?, " +
  629. "green_stones=?, blue_stones=?, " +
  630. "ancient_adena_amount=?, contribution_score=? " +
  631. "WHERE char_obj_id=?");
  632. statement.setString(1, sevenDat.getString("cabal"));
  633. statement.setInt(2, sevenDat.getInteger("seal"));
  634. statement.setInt(3, sevenDat.getInteger("red_stones"));
  635. statement.setInt(4, sevenDat.getInteger("green_stones"));
  636. statement.setInt(5, sevenDat.getInteger("blue_stones"));
  637. statement.setDouble(6, sevenDat.getDouble("ancient_adena_amount"));
  638. statement.setDouble(7, sevenDat.getDouble("contribution_score"));
  639. statement.setInt(8, sevenDat.getInteger("char_obj_id"));
  640. statement.execute();
  641. statement.close();
  642. if (Config.DEBUG)
  643. _log.info("SevenSigns: Updated data in database for char ID " + sevenDat.getInteger("char_obj_id") + " (" + sevenDat.getString("cabal") + ")");
  644. }
  645. if (updateSettings)
  646. {
  647. String sqlQuery = "UPDATE seven_signs_status SET current_cycle=?, active_period=?, previous_winner=?, " +
  648. "dawn_stone_score=?, dawn_festival_score=?, dusk_stone_score=?, dusk_festival_score=?, " +
  649. "avarice_owner=?, gnosis_owner=?, strife_owner=?, avarice_dawn_score=?, gnosis_dawn_score=?, " +
  650. "strife_dawn_score=?, avarice_dusk_score=?, gnosis_dusk_score=?, strife_dusk_score=?, " +
  651. "festival_cycle=?, ";
  652. for (int i = 0; i < (SevenSignsFestival.FESTIVAL_COUNT); i++)
  653. sqlQuery += "accumulated_bonus" + String.valueOf(i) + "=?, ";
  654. sqlQuery += "date=? WHERE id=0";
  655. statement = con.prepareStatement(sqlQuery);
  656. statement.setInt(1, _currentCycle);
  657. statement.setInt(2, _activePeriod);
  658. statement.setInt(3, _previousWinner);
  659. statement.setDouble(4, _dawnStoneScore);
  660. statement.setInt(5, _dawnFestivalScore);
  661. statement.setDouble(6, _duskStoneScore);
  662. statement.setInt(7, _duskFestivalScore);
  663. statement.setInt(8, _signsSealOwners.get(SEAL_AVARICE));
  664. statement.setInt(9, _signsSealOwners.get(SEAL_GNOSIS));
  665. statement.setInt(10, _signsSealOwners.get(SEAL_STRIFE));
  666. statement.setInt(11, _signsDawnSealTotals.get(SEAL_AVARICE));
  667. statement.setInt(12, _signsDawnSealTotals.get(SEAL_GNOSIS));
  668. statement.setInt(13, _signsDawnSealTotals.get(SEAL_STRIFE));
  669. statement.setInt(14, _signsDuskSealTotals.get(SEAL_AVARICE));
  670. statement.setInt(15, _signsDuskSealTotals.get(SEAL_GNOSIS));
  671. statement.setInt(16, _signsDuskSealTotals.get(SEAL_STRIFE));
  672. statement.setInt(17, SevenSignsFestival.getInstance().getCurrentFestivalCycle());
  673. for (int i = 0; i < SevenSignsFestival.FESTIVAL_COUNT; i++)
  674. statement.setInt(18 + i, SevenSignsFestival.getInstance().getAccumulatedBonus(i));
  675. statement.setInt(18 + SevenSignsFestival.FESTIVAL_COUNT, Calendar.getInstance().get(Calendar.DAY_OF_WEEK));
  676. statement.execute();
  677. statement.close();
  678. con.close();
  679. if (Config.DEBUG)
  680. _log.info("SevenSigns: Updated data in database.");
  681. }
  682. }
  683. catch (SQLException e)
  684. {
  685. _log.severe("SevenSigns: Unable to save data to database: " + e);
  686. }
  687. finally
  688. {
  689. try {
  690. statement.close();
  691. con.close();
  692. }
  693. catch (Exception e) {}
  694. }
  695. }
  696. /**
  697. * Used to reset the cabal details of all players, and update the database.<BR>
  698. * Primarily used when beginning a new cycle, and should otherwise never be called.
  699. */
  700. protected void resetPlayerData()
  701. {
  702. if (Config.DEBUG)
  703. _log.info("SevenSigns: Resetting player data for new event period.");
  704. // Reset each player's contribution data as well as seal and cabal.
  705. for (StatsSet sevenDat : _signsPlayerData.values())
  706. {
  707. int charObjId = sevenDat.getInteger("char_obj_id");
  708. // Reset the player's cabal and seal information
  709. sevenDat.set("cabal", "");
  710. sevenDat.set("seal", SEAL_NULL);
  711. sevenDat.set("contribution_score", 0);
  712. _signsPlayerData.put(charObjId, sevenDat);
  713. }
  714. }
  715. /**
  716. * Tests whether the specified player has joined a cabal in the past.
  717. *
  718. * @param player
  719. * @return boolean hasRegistered
  720. */
  721. private boolean hasRegisteredBefore(L2PcInstance player)
  722. {
  723. return _signsPlayerData.containsKey(player.getObjectId());
  724. }
  725. /**
  726. * Used to specify cabal-related details for the specified player. This method
  727. * checks to see if the player has registered before and will update the database
  728. * if necessary.
  729. * <BR>
  730. * Returns the cabal ID the player has joined.
  731. *
  732. * @param player
  733. * @param chosenCabal
  734. * @param chosenSeal
  735. * @return int cabal
  736. */
  737. public int setPlayerInfo(L2PcInstance player, int chosenCabal, int chosenSeal)
  738. {
  739. int charObjId = player.getObjectId();
  740. Connection con = null;
  741. PreparedStatement statement = null;
  742. StatsSet currPlayerData = getPlayerData(player);
  743. if (currPlayerData != null)
  744. {
  745. // If the seal validation period has passed,
  746. // cabal information was removed and so "re-register" player
  747. currPlayerData.set("cabal", getCabalShortName(chosenCabal));
  748. currPlayerData.set("seal", chosenSeal);
  749. _signsPlayerData.put(charObjId, currPlayerData);
  750. }
  751. else
  752. {
  753. currPlayerData = new StatsSet();
  754. currPlayerData.set("char_obj_id", charObjId);
  755. currPlayerData.set("cabal", getCabalShortName(chosenCabal));
  756. currPlayerData.set("seal", chosenSeal);
  757. currPlayerData.set("red_stones", 0);
  758. currPlayerData.set("green_stones", 0);
  759. currPlayerData.set("blue_stones", 0);
  760. currPlayerData.set("ancient_adena_amount", 0);
  761. currPlayerData.set("contribution_score", 0);
  762. _signsPlayerData.put(charObjId, currPlayerData);
  763. // Update data in database, as we have a new player signing up.
  764. try
  765. {
  766. con = L2DatabaseFactory.getInstance().getConnection();
  767. statement = con.prepareStatement(
  768. "INSERT INTO seven_signs (char_obj_id, cabal, seal) VALUES (?,?,?)");
  769. statement.setInt(1, charObjId);
  770. statement.setString(2, getCabalShortName(chosenCabal));
  771. statement.setInt(3, chosenSeal);
  772. statement.execute();
  773. statement.close();
  774. con.close();
  775. if (Config.DEBUG)
  776. _log.info("SevenSigns: Inserted data in DB for char ID " + currPlayerData.getInteger("char_obj_id") + " (" + currPlayerData.getString("cabal") + ")");
  777. }
  778. catch (SQLException e)
  779. {
  780. _log.severe("SevenSigns: Failed to save data: " + e);
  781. }
  782. finally
  783. {
  784. try
  785. {
  786. statement.close();
  787. con.close();
  788. }
  789. catch (Exception e) {}
  790. }
  791. }
  792. // Increasing Seal total score for the player chosen Seal.
  793. if (currPlayerData.getString("cabal") == "dawn")
  794. _signsDawnSealTotals.put(chosenSeal, _signsDawnSealTotals.get(chosenSeal) + 1);
  795. else
  796. _signsDuskSealTotals.put(chosenSeal, _signsDuskSealTotals.get(chosenSeal) + 1);
  797. saveSevenSignsData(player, true);
  798. if (Config.DEBUG)
  799. _log.info("SevenSigns: " + player.getName() + " has joined the " + getCabalName(chosenCabal) + " for the " + getSealName(chosenSeal, false) + "!");
  800. return chosenCabal;
  801. }
  802. /**
  803. * Returns the amount of ancient adena the specified player can claim, if any.<BR>
  804. * If removeReward = True, all the ancient adena owed to them is removed, then
  805. * DB is updated.
  806. *
  807. * @param player
  808. * @param removeReward
  809. * @return int rewardAmount
  810. */
  811. public int getAncientAdenaReward(L2PcInstance player, boolean removeReward)
  812. {
  813. StatsSet currPlayer = getPlayerData(player);
  814. int rewardAmount = currPlayer.getInteger("ancient_adena_amount");
  815. currPlayer.set("red_stones", 0);
  816. currPlayer.set("green_stones", 0);
  817. currPlayer.set("blue_stones", 0);
  818. currPlayer.set("ancient_adena_amount", 0);
  819. if (removeReward)
  820. {
  821. _signsPlayerData.put(player.getObjectId(), currPlayer);
  822. saveSevenSignsData(player, true);
  823. }
  824. return rewardAmount;
  825. }
  826. /**
  827. * Used to add the specified player's seal stone contribution points
  828. * to the current total for their cabal. Returns the point score the
  829. * contribution was worth.
  830. *
  831. * Each stone count <B>must be</B> broken down and specified by the stone's color.
  832. *
  833. * @param player
  834. * @param blueCount
  835. * @param greenCount
  836. * @param redCount
  837. * @return int contribScore
  838. */
  839. public int addPlayerStoneContrib(L2PcInstance player, int blueCount, int greenCount, int redCount)
  840. {
  841. StatsSet currPlayer = getPlayerData(player);
  842. int contribScore = calcContributionScore(blueCount, greenCount, redCount);
  843. int totalAncientAdena = currPlayer.getInteger("ancient_adena_amount") + calcAncientAdenaReward(blueCount, greenCount, redCount);
  844. int totalContribScore = currPlayer.getInteger("contribution_score") + contribScore;
  845. if (totalContribScore > Config.ALT_MAXIMUM_PLAYER_CONTRIB)
  846. return -1;
  847. currPlayer.set("red_stones", currPlayer.getInteger("red_stones") + redCount);
  848. currPlayer.set("green_stones", currPlayer.getInteger("green_stones") + greenCount);
  849. currPlayer.set("blue_stones", currPlayer.getInteger("blue_stones") + blueCount);
  850. currPlayer.set("ancient_adena_amount", totalAncientAdena);
  851. currPlayer.set("contribution_score", totalContribScore);
  852. _signsPlayerData.put(player.getObjectId(), currPlayer);
  853. switch (getPlayerCabal(player))
  854. {
  855. case CABAL_DAWN:
  856. _dawnStoneScore += contribScore;
  857. break;
  858. case CABAL_DUSK:
  859. _duskStoneScore += contribScore;
  860. break;
  861. }
  862. saveSevenSignsData(player, true);
  863. if (Config.DEBUG)
  864. _log.info("SevenSigns: " + player.getName() + " contributed " + contribScore + " seal stone points to their cabal.");
  865. return contribScore;
  866. }
  867. /**
  868. * Adds the specified number of festival points to the specified cabal.
  869. * Remember, the same number of points are <B>deducted from the rival cabal</B>
  870. * to maintain proportionality.
  871. *
  872. * @param cabal
  873. * @param amount
  874. */
  875. public void addFestivalScore(int cabal, int amount)
  876. {
  877. if (cabal == CABAL_DUSK) {
  878. _duskFestivalScore += amount;
  879. // To prevent negative scores!
  880. if (_dawnFestivalScore >= amount)
  881. _dawnFestivalScore -= amount;
  882. }
  883. else {
  884. _dawnFestivalScore += amount;
  885. if (_duskFestivalScore >= amount)
  886. _duskFestivalScore -= amount;
  887. }
  888. }
  889. /**
  890. * Send info on the current Seven Signs period to the specified player.
  891. *
  892. * @param player
  893. */
  894. public void sendCurrentPeriodMsg(L2PcInstance player)
  895. {
  896. SystemMessage sm = null;
  897. switch (getCurrentPeriod())
  898. {
  899. case PERIOD_COMP_RECRUITING:
  900. sm = new SystemMessage(SystemMessageId.PREPARATIONS_PERIOD_BEGUN);
  901. break;
  902. case PERIOD_COMPETITION:
  903. sm = new SystemMessage(SystemMessageId.COMPETITION_PERIOD_BEGUN);
  904. break;
  905. case PERIOD_COMP_RESULTS:
  906. sm = new SystemMessage(SystemMessageId.RESULTS_PERIOD_BEGUN);
  907. break;
  908. case PERIOD_SEAL_VALIDATION:
  909. sm = new SystemMessage(SystemMessageId.VALIDATION_PERIOD_BEGUN);
  910. break;
  911. }
  912. player.sendPacket(sm);
  913. }
  914. /**
  915. * Sends the built-in system message specified by sysMsgId to all online players.
  916. *
  917. * @param sysMsgId
  918. */
  919. public void sendMessageToAll(SystemMessageId sysMsgId)
  920. {
  921. SystemMessage sm = new SystemMessage(sysMsgId);
  922. for (L2PcInstance player : L2World.getInstance().getAllPlayers())
  923. player.sendPacket(sm);
  924. }
  925. /**
  926. * Used to initialize the seals for each cabal. (Used at startup or at beginning of a new cycle).
  927. * This method should be called after <B>resetSeals()</B> and <B>calcNewSealOwners()</B> on a new cycle.
  928. */
  929. protected void initializeSeals()
  930. {
  931. for (Integer currSeal : _signsSealOwners.keySet())
  932. {
  933. int sealOwner = _signsSealOwners.get(currSeal);
  934. if (sealOwner != CABAL_NULL)
  935. if (isSealValidationPeriod())
  936. _log.info("SevenSigns: The " + getCabalName(sealOwner) + " have won the " + getSealName(currSeal, false) + ".");
  937. else
  938. _log.info("SevenSigns: The " + getSealName(currSeal, false) + " is currently owned by " + getCabalName(sealOwner) + ".");
  939. else
  940. _log.info("SevenSigns: The " + getSealName(currSeal, false) + " remains unclaimed.");
  941. }
  942. }
  943. /**
  944. * Only really used at the beginning of a new cycle, this method resets all seal-related data.
  945. */
  946. protected void resetSeals()
  947. {
  948. _signsDawnSealTotals.put(SEAL_AVARICE, 0);
  949. _signsDawnSealTotals.put(SEAL_GNOSIS, 0);
  950. _signsDawnSealTotals.put(SEAL_STRIFE, 0);
  951. _signsDuskSealTotals.put(SEAL_AVARICE, 0);
  952. _signsDuskSealTotals.put(SEAL_GNOSIS, 0);
  953. _signsDuskSealTotals.put(SEAL_STRIFE, 0);
  954. }
  955. /**
  956. * Calculates the ownership of the three Seals of the Seven Signs,
  957. * based on various criterion.
  958. * <BR><BR>
  959. * Should only ever called at the beginning of a new cycle.
  960. */
  961. protected void calcNewSealOwners()
  962. {
  963. if (Config.DEBUG)
  964. {
  965. _log.info("SevenSigns: (Avarice) Dawn = " + _signsDawnSealTotals.get(SEAL_AVARICE) + ", Dusk = " + _signsDuskSealTotals.get(SEAL_AVARICE));
  966. _log.info("SevenSigns: (Gnosis) Dawn = " + _signsDawnSealTotals.get(SEAL_GNOSIS) + ", Dusk = " + _signsDuskSealTotals.get(SEAL_GNOSIS));
  967. _log.info("SevenSigns: (Strife) Dawn = " + _signsDawnSealTotals.get(SEAL_STRIFE) + ", Dusk = " + _signsDuskSealTotals.get(SEAL_STRIFE));
  968. }
  969. for (Integer currSeal : _signsDawnSealTotals.keySet())
  970. {
  971. int prevSealOwner = _signsSealOwners.get(currSeal);
  972. int newSealOwner = CABAL_NULL;
  973. int dawnProportion = getSealProportion(currSeal, CABAL_DAWN);
  974. int totalDawnMembers = getTotalMembers(CABAL_DAWN) == 0 ? 1 : getTotalMembers(CABAL_DAWN);
  975. int dawnPercent = Math.round(((float)dawnProportion / (float)totalDawnMembers) * 100);
  976. int duskProportion = getSealProportion(currSeal, CABAL_DUSK);
  977. int totalDuskMembers = getTotalMembers(CABAL_DUSK) == 0 ? 1 : getTotalMembers(CABAL_DUSK);
  978. int duskPercent = Math.round(((float)duskProportion / (float)totalDuskMembers) * 100);
  979. /*
  980. * - If a Seal was already closed or owned by the opponent and the new winner wants
  981. * to assume ownership of the Seal, 35% or more of the members of the Cabal must
  982. * have chosen the Seal. If they chose less than 35%, they cannot own the Seal.
  983. *
  984. * - If the Seal was owned by the winner in the previous Seven Signs, they can retain
  985. * that seal if 10% or more members have chosen it. If they want to possess a new Seal,
  986. * at least 35% of the members of the Cabal must have chosen the new Seal.
  987. */
  988. switch (prevSealOwner)
  989. {
  990. case CABAL_NULL:
  991. switch (getCabalHighestScore())
  992. {
  993. case CABAL_NULL:
  994. newSealOwner = CABAL_NULL;
  995. break;
  996. case CABAL_DAWN:
  997. if (dawnPercent >= 35)
  998. newSealOwner = CABAL_DAWN;
  999. else
  1000. newSealOwner = CABAL_NULL;
  1001. break;
  1002. case CABAL_DUSK:
  1003. if (duskPercent >= 35)
  1004. newSealOwner = CABAL_DUSK;
  1005. else
  1006. newSealOwner = CABAL_NULL;
  1007. break;
  1008. }
  1009. break;
  1010. case CABAL_DAWN:
  1011. switch (getCabalHighestScore())
  1012. {
  1013. case CABAL_NULL:
  1014. if (dawnPercent >= 10)
  1015. newSealOwner = CABAL_DAWN;
  1016. else
  1017. newSealOwner = CABAL_NULL;
  1018. break;
  1019. case CABAL_DAWN:
  1020. if (dawnPercent >= 10)
  1021. newSealOwner = CABAL_DAWN;
  1022. else
  1023. newSealOwner = CABAL_NULL;
  1024. break;
  1025. case CABAL_DUSK:
  1026. if (duskPercent >= 35)
  1027. newSealOwner = CABAL_DUSK;
  1028. else if (dawnPercent >= 10)
  1029. newSealOwner = CABAL_DAWN;
  1030. else
  1031. newSealOwner = CABAL_NULL;
  1032. break;
  1033. }
  1034. break;
  1035. case CABAL_DUSK:
  1036. switch (getCabalHighestScore())
  1037. {
  1038. case CABAL_NULL:
  1039. if (duskPercent >= 10)
  1040. newSealOwner = CABAL_DUSK;
  1041. else
  1042. newSealOwner = CABAL_NULL;
  1043. break;
  1044. case CABAL_DAWN:
  1045. if (dawnPercent >= 35)
  1046. newSealOwner = CABAL_DAWN;
  1047. else if (duskPercent >= 10)
  1048. newSealOwner = CABAL_DUSK;
  1049. else
  1050. newSealOwner = CABAL_NULL;
  1051. break;
  1052. case CABAL_DUSK:
  1053. if (duskPercent >= 10)
  1054. newSealOwner = CABAL_DUSK;
  1055. else
  1056. newSealOwner = CABAL_NULL;
  1057. break;
  1058. }
  1059. break;
  1060. }
  1061. _signsSealOwners.put(currSeal, newSealOwner);
  1062. // Alert all online players to new seal status.
  1063. switch (currSeal)
  1064. {
  1065. case SEAL_AVARICE:
  1066. if (newSealOwner == CABAL_DAWN)
  1067. sendMessageToAll(SystemMessageId.DAWN_OBTAINED_AVARICE);
  1068. else if (newSealOwner == CABAL_DUSK)
  1069. sendMessageToAll(SystemMessageId.DUSK_OBTAINED_AVARICE);
  1070. break;
  1071. case SEAL_GNOSIS:
  1072. if (newSealOwner == CABAL_DAWN)
  1073. sendMessageToAll(SystemMessageId.DAWN_OBTAINED_GNOSIS);
  1074. else if (newSealOwner == CABAL_DUSK)
  1075. sendMessageToAll(SystemMessageId.DUSK_OBTAINED_GNOSIS);
  1076. break;
  1077. case SEAL_STRIFE:
  1078. if (newSealOwner == CABAL_DAWN)
  1079. sendMessageToAll(SystemMessageId.DAWN_OBTAINED_STRIFE);
  1080. else if (newSealOwner == CABAL_DUSK)
  1081. sendMessageToAll(SystemMessageId.DUSK_OBTAINED_STRIFE);
  1082. CastleManager.getInstance().validateTaxes(newSealOwner);
  1083. break;
  1084. }
  1085. }
  1086. }
  1087. /**
  1088. * This method is called to remove all players from catacombs and
  1089. * necropolises, who belong to the losing cabal.
  1090. * <BR><BR>
  1091. * Should only ever called at the beginning of Seal Validation.
  1092. */
  1093. protected void teleLosingCabalFromDungeons(String compWinner)
  1094. {
  1095. for (L2PcInstance onlinePlayer : L2World.getInstance().getAllPlayers())
  1096. {
  1097. StatsSet currPlayer = getPlayerData(onlinePlayer);
  1098. if (isSealValidationPeriod() || isCompResultsPeriod())
  1099. {
  1100. if (!onlinePlayer.isGM() && onlinePlayer.isIn7sDungeon() && !currPlayer.getString("cabal").equals(compWinner))
  1101. {
  1102. onlinePlayer.teleToLocation(MapRegionTable.TeleportWhereType.Town);
  1103. onlinePlayer.setIsIn7sDungeon(false);
  1104. onlinePlayer.sendMessage("You have been teleported to the nearest town due to the beginning of the Seal Validation period.");
  1105. }
  1106. }
  1107. else
  1108. {
  1109. if (!onlinePlayer.isGM() && onlinePlayer.isIn7sDungeon() && !currPlayer.getString("cabal").equals(""))
  1110. {
  1111. onlinePlayer.teleToLocation(MapRegionTable.TeleportWhereType.Town);
  1112. onlinePlayer.setIsIn7sDungeon(false);
  1113. onlinePlayer.sendMessage("You have been teleported to the nearest town because you have not signed for any cabal.");
  1114. }
  1115. }
  1116. }
  1117. }
  1118. /**
  1119. * The primary controller of period change of the Seven Signs system.
  1120. * This runs all related tasks depending on the period that is about to begin.
  1121. *
  1122. * @author Tempy
  1123. */
  1124. protected class SevenSignsPeriodChange implements Runnable
  1125. {
  1126. public void run()
  1127. {
  1128. /*
  1129. * Remember the period check here refers to the period just ENDED!
  1130. */
  1131. final int periodEnded = getCurrentPeriod();
  1132. _activePeriod++;
  1133. switch (periodEnded)
  1134. {
  1135. case PERIOD_COMP_RECRUITING: // Initialization
  1136. // Start the Festival of Darkness cycle.
  1137. SevenSignsFestival.getInstance().startFestivalManager();
  1138. // Send message that Competition has begun.
  1139. sendMessageToAll(SystemMessageId.QUEST_EVENT_PERIOD_BEGUN);
  1140. break;
  1141. case PERIOD_COMPETITION: // Results Calculation
  1142. // Send message that Competition has ended.
  1143. sendMessageToAll(SystemMessageId.QUEST_EVENT_PERIOD_ENDED);
  1144. int compWinner = getCabalHighestScore();
  1145. // Schedule a stop of the festival engine.
  1146. SevenSignsFestival.getInstance().getFestivalManagerSchedule().cancel(false);
  1147. calcNewSealOwners();
  1148. switch (compWinner)
  1149. {
  1150. case CABAL_DAWN:
  1151. sendMessageToAll(SystemMessageId.DAWN_WON);
  1152. break;
  1153. case CABAL_DUSK:
  1154. sendMessageToAll(SystemMessageId.DUSK_WON);
  1155. break;
  1156. }
  1157. _previousWinner = compWinner;
  1158. break;
  1159. case PERIOD_COMP_RESULTS: // Seal Validation
  1160. // Perform initial Seal Validation set up.
  1161. initializeSeals();
  1162. // Send message that Seal Validation has begun.
  1163. sendMessageToAll(SystemMessageId.SEAL_VALIDATION_PERIOD_BEGUN);
  1164. _log.info("SevenSigns: The " + getCabalName(_previousWinner) + " have won the competition with " + getCurrentScore(_previousWinner) + " points!");
  1165. break;
  1166. case PERIOD_SEAL_VALIDATION: // Reset for New Cycle
  1167. SevenSignsFestival.getInstance().rewardHighestRanked(); // reward highest ranking members from cycle
  1168. // Ensure a cycle restart when this period ends.
  1169. _activePeriod = PERIOD_COMP_RECRUITING;
  1170. // Send message that Seal Validation has ended.
  1171. sendMessageToAll(SystemMessageId.SEAL_VALIDATION_PERIOD_ENDED);
  1172. // Reset all data
  1173. resetPlayerData();
  1174. resetSeals();
  1175. // Reset all Festival-related data and remove any unused blood offerings.
  1176. // NOTE: A full update of Festival data in the database is also performed.
  1177. SevenSignsFestival.getInstance().resetFestivalData(false);
  1178. _dawnStoneScore = 0;
  1179. _duskStoneScore = 0;
  1180. _dawnFestivalScore = 0;
  1181. _duskFestivalScore = 0;
  1182. _currentCycle++;
  1183. break;
  1184. }
  1185. // Make sure all Seven Signs data is saved for future use.
  1186. saveSevenSignsData(null, true);
  1187. teleLosingCabalFromDungeons(getCabalShortName(getCabalHighestScore()));
  1188. SSQInfo ss = new SSQInfo();
  1189. for (L2PcInstance player : L2World.getInstance().getAllPlayers())
  1190. player.sendPacket(ss);
  1191. spawnSevenSignsNPC();
  1192. _log.info("SevenSigns: The " + getCurrentPeriodName() + " period has begun!");
  1193. setCalendarForNextPeriodChange();
  1194. SevenSignsPeriodChange sspc = new SevenSignsPeriodChange();
  1195. ThreadPoolManager.getInstance().scheduleGeneral(sspc, getMilliToPeriodChange());
  1196. }
  1197. }
  1198. }