/* * This file is part of the L2J Mobius project. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package com.l2jmobius.gameserver.model.entity; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Calendar; import java.util.Collections; import java.util.Date; import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.concurrent.ConcurrentHashMap; import java.util.logging.Logger; import com.l2jmobius.Config; import com.l2jmobius.commons.database.pool.impl.ConnectionFactory; import com.l2jmobius.gameserver.cache.HtmCache; import com.l2jmobius.gameserver.data.sql.impl.CharNameTable; import com.l2jmobius.gameserver.data.sql.impl.ClanTable; import com.l2jmobius.gameserver.data.xml.impl.ClassListData; import com.l2jmobius.gameserver.data.xml.impl.NpcData; import com.l2jmobius.gameserver.instancemanager.CastleManager; import com.l2jmobius.gameserver.model.L2Clan; import com.l2jmobius.gameserver.model.L2World; import com.l2jmobius.gameserver.model.StatsSet; import com.l2jmobius.gameserver.model.actor.instance.L2PcInstance; import com.l2jmobius.gameserver.model.actor.templates.L2NpcTemplate; import com.l2jmobius.gameserver.model.itemcontainer.Inventory; import com.l2jmobius.gameserver.model.items.instance.L2ItemInstance; import com.l2jmobius.gameserver.model.olympiad.Olympiad; import com.l2jmobius.gameserver.network.SystemMessageId; import com.l2jmobius.gameserver.network.serverpackets.InventoryUpdate; import com.l2jmobius.gameserver.network.serverpackets.NpcHtmlMessage; import com.l2jmobius.gameserver.network.serverpackets.SocialAction; import com.l2jmobius.gameserver.network.serverpackets.SystemMessage; import com.l2jmobius.gameserver.network.serverpackets.UserInfo; import com.l2jmobius.util.StringUtil; /** * Hero entity. * @author godson */ public class Hero { private static final Logger _log = Logger.getLogger(Hero.class.getName()); private static final String GET_HEROES = "SELECT heroes.charId, characters.char_name, heroes.class_id, heroes.count, heroes.played, heroes.claimed FROM heroes, characters WHERE characters.charId = heroes.charId AND heroes.played = 1"; private static final String GET_ALL_HEROES = "SELECT heroes.charId, characters.char_name, heroes.class_id, heroes.count, heroes.played, heroes.claimed FROM heroes, characters WHERE characters.charId = heroes.charId"; private static final String UPDATE_ALL = "UPDATE heroes SET played = 0"; private static final String INSERT_HERO = "INSERT INTO heroes (charId, class_id, count, played, claimed) VALUES (?,?,?,?,?)"; private static final String UPDATE_HERO = "UPDATE heroes SET count = ?, played = ?, claimed = ? WHERE charId = ?"; private static final String GET_CLAN_ALLY = "SELECT characters.clanid AS clanid, coalesce(clan_data.ally_Id, 0) AS allyId FROM characters LEFT JOIN clan_data ON clan_data.clan_id = characters.clanid WHERE characters.charId = ?"; // delete hero items private static final String DELETE_ITEMS = "DELETE FROM items WHERE item_id IN (6842, 6611, 6612, 6613, 6614, 6615, 6616, 6617, 6618, 6619, 6620, 6621, 9388, 9389, 9390) AND owner_id NOT IN (SELECT charId FROM characters WHERE accesslevel > 0)"; private static final Map HEROES = new ConcurrentHashMap<>(); private static final Map COMPLETE_HEROS = new ConcurrentHashMap<>(); private static final Map HERO_COUNTS = new ConcurrentHashMap<>(); private static final Map> HERO_FIGHTS = new ConcurrentHashMap<>(); private static final Map> HERO_DIARY = new ConcurrentHashMap<>(); private static final Map HERO_MESSAGE = new ConcurrentHashMap<>(); public static final String COUNT = "count"; public static final String PLAYED = "played"; public static final String CLAIMED = "claimed"; public static final String CLAN_NAME = "clan_name"; public static final String CLAN_CREST = "clan_crest"; public static final String ALLY_NAME = "ally_name"; public static final String ALLY_CREST = "ally_crest"; public static final int ACTION_RAID_KILLED = 1; public static final int ACTION_HERO_GAINED = 2; public static final int ACTION_CASTLE_TAKEN = 3; protected Hero() { init(); } private void init() { HEROES.clear(); COMPLETE_HEROS.clear(); HERO_COUNTS.clear(); HERO_FIGHTS.clear(); HERO_DIARY.clear(); HERO_MESSAGE.clear(); try (Connection con = ConnectionFactory.getInstance().getConnection(); Statement s1 = con.createStatement(); ResultSet rset = s1.executeQuery(GET_HEROES); PreparedStatement ps = con.prepareStatement(GET_CLAN_ALLY); Statement s2 = con.createStatement(); ResultSet rset2 = s2.executeQuery(GET_ALL_HEROES)) { while (rset.next()) { final StatsSet hero = new StatsSet(); final int charId = rset.getInt(Olympiad.CHAR_ID); hero.set(Olympiad.CHAR_NAME, rset.getString(Olympiad.CHAR_NAME)); hero.set(Olympiad.CLASS_ID, rset.getInt(Olympiad.CLASS_ID)); hero.set(COUNT, rset.getInt(COUNT)); hero.set(PLAYED, rset.getInt(PLAYED)); hero.set(CLAIMED, Boolean.parseBoolean(rset.getString(CLAIMED))); loadFights(charId); loadDiary(charId); loadMessage(charId); processHeros(ps, charId, hero); HEROES.put(charId, hero); } while (rset2.next()) { final StatsSet hero = new StatsSet(); final int charId = rset2.getInt(Olympiad.CHAR_ID); hero.set(Olympiad.CHAR_NAME, rset2.getString(Olympiad.CHAR_NAME)); hero.set(Olympiad.CLASS_ID, rset2.getInt(Olympiad.CLASS_ID)); hero.set(COUNT, rset2.getInt(COUNT)); hero.set(PLAYED, rset2.getInt(PLAYED)); hero.set(CLAIMED, Boolean.parseBoolean(rset2.getString(CLAIMED))); processHeros(ps, charId, hero); COMPLETE_HEROS.put(charId, hero); } } catch (SQLException e) { _log.warning("Hero System: Couldnt load Heroes: " + e.getMessage()); } _log.info("Hero System: Loaded " + HEROES.size() + " Heroes."); _log.info("Hero System: Loaded " + COMPLETE_HEROS.size() + " all time Heroes."); } private void processHeros(PreparedStatement ps, int charId, StatsSet hero) throws SQLException { ps.setInt(1, charId); try (ResultSet rs = ps.executeQuery()) { if (rs.next()) { final int clanId = rs.getInt("clanid"); final int allyId = rs.getInt("allyId"); String clanName = ""; String allyName = ""; int clanCrest = 0; int allyCrest = 0; if (clanId > 0) { clanName = ClanTable.getInstance().getClan(clanId).getName(); clanCrest = ClanTable.getInstance().getClan(clanId).getCrestId(); if (allyId > 0) { allyName = ClanTable.getInstance().getClan(clanId).getAllyName(); allyCrest = ClanTable.getInstance().getClan(clanId).getAllyCrestId(); } } hero.set(CLAN_CREST, clanCrest); hero.set(CLAN_NAME, clanName); hero.set(ALLY_CREST, allyCrest); hero.set(ALLY_NAME, allyName); } ps.clearParameters(); } } private String calcFightTime(long FightTime) { final String format = String.format("%%0%dd", 2); FightTime = FightTime / 1000; final String seconds = String.format(format, FightTime % 60); final String minutes = String.format(format, (FightTime % 3600) / 60); final String time = minutes + ":" + seconds; return time; } /** * Restore hero message from Db. * @param charId */ public void loadMessage(int charId) { try (Connection con = ConnectionFactory.getInstance().getConnection(); PreparedStatement ps = con.prepareStatement("SELECT message FROM heroes WHERE charId=?")) { ps.setInt(1, charId); try (ResultSet rset = ps.executeQuery()) { if (rset.next()) { HERO_MESSAGE.put(charId, rset.getString("message")); } } } catch (SQLException e) { _log.warning("Hero System: Couldnt load Hero Message for CharId: " + charId + ": " + e.getMessage()); } } public void loadDiary(int charId) { final List diary = new ArrayList<>(); int diaryentries = 0; try (Connection con = ConnectionFactory.getInstance().getConnection(); PreparedStatement ps = con.prepareStatement("SELECT * FROM heroes_diary WHERE charId=? ORDER BY time ASC")) { ps.setInt(1, charId); try (ResultSet rset = ps.executeQuery()) { while (rset.next()) { final StatsSet _diaryentry = new StatsSet(); final long time = rset.getLong("time"); final int action = rset.getInt("action"); final int param = rset.getInt("param"); final String date = (new SimpleDateFormat("yyyy-MM-dd HH")).format(new Date(time)); _diaryentry.set("date", date); if (action == ACTION_RAID_KILLED) { final L2NpcTemplate template = NpcData.getInstance().getTemplate(param); if (template != null) { _diaryentry.set("action", template.getName() + " was defeated"); } } else if (action == ACTION_HERO_GAINED) { _diaryentry.set("action", "Gained Hero status"); } else if (action == ACTION_CASTLE_TAKEN) { final Castle castle = CastleManager.getInstance().getCastleById(param); if (castle != null) { _diaryentry.set("action", castle.getName() + " Castle was successfuly taken"); } } diary.add(_diaryentry); diaryentries++; } } HERO_DIARY.put(charId, diary); _log.info("Hero System: Loaded " + diaryentries + " diary entries for Hero: " + CharNameTable.getInstance().getNameById(charId)); } catch (SQLException e) { _log.warning("Hero System: Couldnt load Hero Diary for CharId: " + charId + ": " + e.getMessage()); } } public void loadFights(int charId) { final List fights = new ArrayList<>(); final StatsSet heroCountData = new StatsSet(); final Calendar data = Calendar.getInstance(); data.set(Calendar.DAY_OF_MONTH, 1); data.set(Calendar.HOUR_OF_DAY, 0); data.set(Calendar.MINUTE, 0); data.set(Calendar.MILLISECOND, 0); final long from = data.getTimeInMillis(); int numberoffights = 0; int _victorys = 0; int _losses = 0; int _draws = 0; try (Connection con = ConnectionFactory.getInstance().getConnection(); PreparedStatement ps = con.prepareStatement("SELECT * FROM olympiad_fights WHERE (charOneId=? OR charTwoId=?) AND startvictory"); _victorys++; } else if (winner == 2) { fight.set("result", "loss"); _losses++; } else if (winner == 0) { fight.set("result", "draw"); _draws++; } fights.add(fight); numberoffights++; } } else if (charId == charTwoId) { final String name = CharNameTable.getInstance().getNameById(charOneId); final String cls = ClassListData.getInstance().getClass(charOneClass).getClientCode(); if ((name != null) && (cls != null)) { final StatsSet fight = new StatsSet(); fight.set("oponent", name); fight.set("oponentclass", cls); fight.set("time", calcFightTime(time)); final String date = (new SimpleDateFormat("yyyy-MM-dd HH:mm")).format(new Date(start)); fight.set("start", date); fight.set("classed", classed); if (winner == 1) { fight.set("result", "loss"); _losses++; } else if (winner == 2) { fight.set("result", "victory"); _victorys++; } else if (winner == 0) { fight.set("result", "draw"); _draws++; } fights.add(fight); numberoffights++; } } } } heroCountData.set("victory", _victorys); heroCountData.set("draw", _draws); heroCountData.set("loss", _losses); HERO_COUNTS.put(charId, heroCountData); HERO_FIGHTS.put(charId, fights); _log.info("Hero System: Loaded " + numberoffights + " fights for Hero: " + CharNameTable.getInstance().getNameById(charId)); } catch (SQLException e) { _log.warning("Hero System: Couldnt load Hero fights history for CharId: " + charId + ": " + e); } } public Map getHeroes() { return HEROES; } public int getHeroByClass(int classid) { for (Entry e : HEROES.entrySet()) { if (e.getValue().getInt(Olympiad.CLASS_ID) == classid) { return e.getKey(); } } return 0; } public void resetData() { HERO_DIARY.clear(); HERO_FIGHTS.clear(); HERO_COUNTS.clear(); HERO_MESSAGE.clear(); } public void showHeroDiary(L2PcInstance activeChar, int heroclass, int charid, int page) { final int perpage = 10; final List mainList = HERO_DIARY.get(charid); if (mainList != null) { final NpcHtmlMessage diaryReply = new NpcHtmlMessage(); final String htmContent = HtmCache.getInstance().getHtm(activeChar.getHtmlPrefix(), "html/olympiad/herodiary.htm"); final String heroMessage = HERO_MESSAGE.get(charid); if ((htmContent != null) && (heroMessage != null)) { diaryReply.setHtml(htmContent); diaryReply.replace("%heroname%", CharNameTable.getInstance().getNameById(charid)); diaryReply.replace("%message%", heroMessage); diaryReply.disableValidation(); if (!mainList.isEmpty()) { final List list = new ArrayList<>(mainList); Collections.reverse(list); boolean color = true; final StringBuilder fList = new StringBuilder(500); int counter = 0; int breakat = 0; for (int i = ((page - 1) * perpage); i < list.size(); i++) { breakat = i; final StatsSet diaryEntry = list.get(i); StringUtil.append(fList, ""); if (color) { StringUtil.append(fList, ""); } else { StringUtil.append(fList, "
"); } StringUtil.append(fList, ""); StringUtil.append(fList, ""); StringUtil.append(fList, "
" + diaryEntry.getString("date") + ":xx
" + diaryEntry.getString("action") + "
 
"); StringUtil.append(fList, ""); color = !color; counter++; if (counter >= perpage) { break; } } if (breakat < (list.size() - 1)) { diaryReply.replace("%buttprev%", "