2
0

DoorTable.java 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  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 com.l2jserver.gameserver.datatables;
  16. import gnu.trove.map.hash.TIntObjectHashMap;
  17. import java.io.BufferedReader;
  18. import java.io.File;
  19. import java.io.FileNotFoundException;
  20. import java.io.FileReader;
  21. import java.io.IOException;
  22. import java.io.LineNumberReader;
  23. import java.util.ArrayList;
  24. import java.util.StringTokenizer;
  25. import java.util.logging.Level;
  26. import java.util.logging.Logger;
  27. import com.l2jserver.Config;
  28. import com.l2jserver.gameserver.idfactory.IdFactory;
  29. import com.l2jserver.gameserver.instancemanager.ClanHallManager;
  30. import com.l2jserver.gameserver.instancemanager.InstanceManager;
  31. import com.l2jserver.gameserver.instancemanager.MapRegionManager;
  32. import com.l2jserver.gameserver.model.StatsSet;
  33. import com.l2jserver.gameserver.model.actor.instance.L2DoorInstance;
  34. import com.l2jserver.gameserver.model.actor.templates.L2CharTemplate;
  35. import com.l2jserver.gameserver.model.entity.ClanHall;
  36. import com.l2jserver.gameserver.model.entity.clanhall.SiegableHall;
  37. import com.l2jserver.gameserver.pathfinding.AbstractNodeLoc;
  38. public class DoorTable
  39. {
  40. private static final Logger _log = Logger.getLogger(DoorTable.class.getName());
  41. private final TIntObjectHashMap<L2DoorInstance> _staticItems;
  42. private final TIntObjectHashMap<ArrayList<L2DoorInstance>> _regions;
  43. protected DoorTable()
  44. {
  45. _staticItems = new TIntObjectHashMap<L2DoorInstance>();
  46. _regions = new TIntObjectHashMap<ArrayList<L2DoorInstance>>();
  47. parseData();
  48. onStart();
  49. }
  50. public void reloadAll()
  51. {
  52. respawn();
  53. }
  54. public void respawn()
  55. {
  56. _staticItems.clear();
  57. _regions.clear();
  58. parseData();
  59. }
  60. public void parseData()
  61. {
  62. final File doorData = new File(Config.DATAPACK_ROOT, "data/door.csv");
  63. try (
  64. FileReader fr = new FileReader(doorData);
  65. BufferedReader br = new BufferedReader(fr);
  66. LineNumberReader lnr = new LineNumberReader(br))
  67. {
  68. String line = null;
  69. _log.info("Searching clan halls doors:");
  70. while ((line = lnr.readLine()) != null)
  71. {
  72. if ((line.trim().length() == 0) || line.startsWith("#"))
  73. {
  74. continue;
  75. }
  76. L2DoorInstance door = parseList(line, false);
  77. putDoor(door);
  78. door.spawnMe(door.getX(), door.getY(), door.getZ());
  79. }
  80. _log.info("DoorTable: Loaded " + _staticItems.size() + " Door Templates for " + _regions.size() + " regions.");
  81. }
  82. catch (FileNotFoundException e)
  83. {
  84. _log.warning("door.csv is missing in data folder");
  85. }
  86. catch (IOException e)
  87. {
  88. _log.log(Level.WARNING, "Error while creating door table " + e.getMessage(), e);
  89. }
  90. }
  91. /**
  92. * Parses door list.
  93. * @param line string
  94. * @param commanderDoor whether the door is commander door (fortress)
  95. * @return created door instance
  96. */
  97. public static L2DoorInstance parseList(final String line, final boolean commanderDoor)
  98. {
  99. StringTokenizer st = new StringTokenizer(line, ";");
  100. L2DoorInstance door = null;
  101. try
  102. {
  103. String name = st.nextToken();
  104. int id = Integer.parseInt(st.nextToken());
  105. int x = Integer.parseInt(st.nextToken());
  106. int y = Integer.parseInt(st.nextToken());
  107. int z = Integer.parseInt(st.nextToken());
  108. int rangeXMin = Integer.parseInt(st.nextToken());
  109. int rangeYMin = Integer.parseInt(st.nextToken());
  110. int rangeZMin = Integer.parseInt(st.nextToken());
  111. int rangeXMax = Integer.parseInt(st.nextToken());
  112. int rangeYMax = Integer.parseInt(st.nextToken());
  113. int rangeZMax = Integer.parseInt(st.nextToken());
  114. int hp = Integer.parseInt(st.nextToken());
  115. int pdef = Integer.parseInt(st.nextToken());
  116. int mdef = Integer.parseInt(st.nextToken());
  117. int emitter = Integer.parseInt(st.nextToken());
  118. boolean unlockable = false;
  119. if (st.hasMoreTokens())
  120. {
  121. unlockable = Boolean.parseBoolean(st.nextToken());
  122. }
  123. boolean startOpen = false;
  124. if (st.hasMoreTokens())
  125. {
  126. startOpen = Boolean.parseBoolean(st.nextToken());
  127. }
  128. boolean targetable = true;
  129. if (st.hasMoreTokens())
  130. {
  131. targetable = Boolean.parseBoolean(st.nextToken());
  132. }
  133. int hallId = 0;
  134. if (st.hasMoreTokens())
  135. {
  136. hallId = Integer.parseInt(st.nextToken());
  137. }
  138. if (rangeXMin > rangeXMax)
  139. {
  140. _log.severe("Error in door data, XMin > XMax, ID:" + id);
  141. }
  142. if (rangeYMin > rangeYMax)
  143. {
  144. _log.severe("Error in door data, YMin > YMax, ID:" + id);
  145. }
  146. if (rangeZMin > rangeZMax)
  147. {
  148. _log.severe("Error in door data, ZMin > ZMax, ID:" + id);
  149. }
  150. int collisionRadius; // (max) radius for movement checks
  151. if ((rangeXMax - rangeXMin) > (rangeYMax - rangeYMin))
  152. {
  153. collisionRadius = rangeYMax - rangeYMin;
  154. }
  155. else
  156. {
  157. collisionRadius = rangeXMax - rangeXMin;
  158. }
  159. StatsSet npcDat = new StatsSet();
  160. npcDat.set("npcId", id);
  161. npcDat.set("level", 0);
  162. npcDat.set("jClass", "door");
  163. npcDat.set("baseSTR", 0);
  164. npcDat.set("baseCON", 0);
  165. npcDat.set("baseDEX", 0);
  166. npcDat.set("baseINT", 0);
  167. npcDat.set("baseWIT", 0);
  168. npcDat.set("baseMEN", 0);
  169. npcDat.set("baseShldDef", 0);
  170. npcDat.set("baseShldRate", 0);
  171. npcDat.set("baseAccCombat", 38);
  172. npcDat.set("baseEvasRate", 38);
  173. npcDat.set("baseCritRate", 38);
  174. // npcDat.set("name", "");
  175. npcDat.set("collision_radius", collisionRadius);
  176. npcDat.set("collision_height", rangeZMax - rangeZMin);
  177. npcDat.set("sex", "male");
  178. npcDat.set("type", "");
  179. npcDat.set("baseAtkRange", 0);
  180. npcDat.set("baseMpMax", 0);
  181. npcDat.set("baseCpMax", 0);
  182. npcDat.set("rewardExp", 0);
  183. npcDat.set("rewardSp", 0);
  184. npcDat.set("basePAtk", 0);
  185. npcDat.set("baseMAtk", 0);
  186. npcDat.set("basePAtkSpd", 0);
  187. npcDat.set("aggroRange", 0);
  188. npcDat.set("baseMAtkSpd", 0);
  189. npcDat.set("rhand", 0);
  190. npcDat.set("lhand", 0);
  191. npcDat.set("armor", 0);
  192. npcDat.set("baseWalkSpd", 0);
  193. npcDat.set("baseRunSpd", 0);
  194. npcDat.set("name", name);
  195. npcDat.set("baseHpMax", hp);
  196. npcDat.set("baseHpReg", 3.e-3f);
  197. npcDat.set("baseMpReg", 3.e-3f);
  198. npcDat.set("basePDef", pdef);
  199. npcDat.set("baseMDef", mdef);
  200. L2CharTemplate template = new L2CharTemplate(npcDat);
  201. door = new L2DoorInstance(IdFactory.getInstance().getNextId(), template, id, name, unlockable);
  202. door.setRange(rangeXMin, rangeYMin, rangeZMin, rangeXMax, rangeYMax, rangeZMax);
  203. door.setCurrentHpMp(door.getMaxHp(), door.getMaxMp());
  204. door.setXYZInvisible(x, y, z);
  205. door.setMapRegion(MapRegionManager.getInstance().getMapRegionLocId(x, y));
  206. door.setEmitter(emitter);
  207. door.setTargetable(targetable);
  208. if (hallId > 0)
  209. {
  210. ClanHall hall = ClanHallManager.getAllClanHalls().get(hallId);
  211. if (hall != null)
  212. {
  213. door.setClanHall(hall);
  214. hall.getDoors().add(door);
  215. if (hall.isSiegableHall())
  216. {
  217. ((SiegableHall) hall).getDoorDefault().add(line);
  218. }
  219. }
  220. }
  221. if (commanderDoor)
  222. {
  223. door.setIsCommanderDoor(startOpen);
  224. }
  225. else
  226. {
  227. door.setOpen(startOpen);
  228. }
  229. }
  230. catch (Exception e)
  231. {
  232. _log.log(Level.SEVERE, "Error in door data at line: " + line, e);
  233. }
  234. return door;
  235. }
  236. public L2DoorInstance getDoor(Integer id)
  237. {
  238. return _staticItems.get(id);
  239. }
  240. public void putDoor(L2DoorInstance door)
  241. {
  242. _staticItems.put(door.getDoorId(), door);
  243. if (_regions.contains(door.getMapRegion()))
  244. {
  245. _regions.get(door.getMapRegion()).add(door);
  246. }
  247. else
  248. {
  249. final ArrayList<L2DoorInstance> region = new ArrayList<L2DoorInstance>();
  250. region.add(door);
  251. _regions.put(door.getMapRegion(), region);
  252. }
  253. }
  254. public L2DoorInstance[] getDoors()
  255. {
  256. return _staticItems.values(new L2DoorInstance[0]);
  257. }
  258. /**
  259. * Performs a check and sets up a scheduled task for those doors that require auto opening/closing.
  260. */
  261. public void checkAutoOpen()
  262. {
  263. for (L2DoorInstance doorInst : getDoors())
  264. {
  265. // Tower of Insolence (open_time 120 seconds)
  266. // TODO: (close_time 120 seconds & random_time 120 seconds)
  267. if (doorInst.getDoorName().startsWith("toi_"))
  268. {
  269. doorInst.setAutoActionDelay(120000);
  270. }
  271. // Devil Isle (open_time 120 seconds)
  272. // TODO: (close_time 120 seconds & random_time 30 seconds)
  273. else if (doorInst.getDoorName().startsWith("di_"))
  274. {
  275. doorInst.setAutoActionDelay(120000);
  276. }
  277. // Garden of Eva (open_time 300 seconds)
  278. // TODO: (close_time 20 seconds & random_time 120 seconds)
  279. else if (doorInst.getDoorName().startsWith("goe_"))
  280. {
  281. doorInst.setAutoActionDelay(300000);
  282. }
  283. // Kratei's Cube (open_time 20 seconds)
  284. // TODO: (close_time 15 seconds & random_time 10 seconds)
  285. else if (doorInst.getDoorName().startsWith("kc20_"))
  286. {
  287. doorInst.setAutoActionDelay(20000);
  288. }
  289. // Kratei's Cube (open_time 30 seconds)
  290. // TODO: (close_time 15 seconds & random_time 10 seconds)
  291. else if (doorInst.getDoorName().startsWith("kc30_"))
  292. {
  293. doorInst.setAutoActionDelay(30000);
  294. }
  295. // Kratei's Cube (open_time 25 seconds)
  296. // TODO: (close_time 15 seconds & random_time 10 seconds)
  297. else if (doorInst.getDoorName().startsWith("kc25_"))
  298. {
  299. doorInst.setAutoActionDelay(25000);
  300. }
  301. // Kratei's Cube (open_time 15 seconds)
  302. // TODO: (close_time 15 seconds & random_time 10 seconds)
  303. else if (doorInst.getDoorName().startsWith("kc15_"))
  304. {
  305. doorInst.setAutoActionDelay(15000);
  306. }
  307. // Kratei's Cube (open_time 10 seconds)
  308. // TODO: (close_time 15 seconds & random_time 5 seconds)
  309. else if (doorInst.getDoorName().startsWith("kc10_"))
  310. {
  311. doorInst.setAutoActionDelay(10000);
  312. }
  313. // Kratei's Cube (open_time 10 seconds)
  314. // TODO: (close_time 15 seconds & random_time 10 seconds)
  315. else if (doorInst.getDoorName().startsWith("kc10b_"))
  316. {
  317. doorInst.setAutoActionDelay(10000);
  318. }
  319. // Kratei's Cube (open_time 14 seconds)
  320. // TODO: (close_time 15 seconds & random_time 10 seconds)
  321. else if (doorInst.getDoorName().startsWith("kc14_"))
  322. {
  323. doorInst.setAutoActionDelay(14000);
  324. }
  325. // Kratei's Cube (open_time 23 seconds)
  326. // TODO: (close_time 15 seconds & random_time 10 seconds)
  327. else if (doorInst.getDoorName().startsWith("kc23_"))
  328. {
  329. doorInst.setAutoActionDelay(23000);
  330. }
  331. // Kratei's Cube (open_time 18 seconds)
  332. // TODO: (close_time 15 seconds & random_time 10 seconds)
  333. else if (doorInst.getDoorName().startsWith("kc18_"))
  334. {
  335. doorInst.setAutoActionDelay(18000);
  336. }
  337. // Kratei's Cube (open_time 26 seconds)
  338. // TODO: (close_time 15 seconds & random_time 10 seconds)
  339. else if (doorInst.getDoorName().startsWith("kc26_"))
  340. {
  341. doorInst.setAutoActionDelay(26000);
  342. }
  343. }
  344. }
  345. public boolean checkIfDoorsBetween(AbstractNodeLoc start, AbstractNodeLoc end, int instanceId)
  346. {
  347. return checkIfDoorsBetween(start.getX(), start.getY(), start.getZ(), end.getX(), end.getY(), end.getZ(), instanceId);
  348. }
  349. public boolean checkIfDoorsBetween(int x, int y, int z, int tx, int ty, int tz, int instanceId)
  350. {
  351. ArrayList<L2DoorInstance> allDoors;
  352. if ((instanceId > 0) && (InstanceManager.getInstance().getInstance(instanceId) != null))
  353. {
  354. allDoors = InstanceManager.getInstance().getInstance(instanceId).getDoors();
  355. }
  356. else
  357. {
  358. allDoors = _regions.get(MapRegionManager.getInstance().getMapRegionLocId(x, y));
  359. }
  360. if (allDoors == null)
  361. {
  362. return false;
  363. }
  364. for (L2DoorInstance doorInst : allDoors)
  365. {
  366. if (doorInst.getXMax() == 0)
  367. {
  368. continue;
  369. }
  370. // line segment goes through box
  371. // first basic checks to stop most calculations short
  372. // phase 1, x
  373. if (((x <= doorInst.getXMax()) && (tx >= doorInst.getXMin())) || ((tx <= doorInst.getXMax()) && (x >= doorInst.getXMin())))
  374. {
  375. // phase 2, y
  376. if (((y <= doorInst.getYMax()) && (ty >= doorInst.getYMin())) || ((ty <= doorInst.getYMax()) && (y >= doorInst.getYMin())))
  377. {
  378. // phase 3, basically only z remains but now we calculate it with another formula (by rage)
  379. // in some cases the direct line check (only) in the beginning isn't sufficient,
  380. // when char z changes a lot along the path
  381. if ((doorInst.getCurrentHp() > 0) && !doorInst.getOpen())
  382. {
  383. int px1 = doorInst.getXMin();
  384. int py1 = doorInst.getYMin();
  385. int pz1 = doorInst.getZMin();
  386. int px2 = doorInst.getXMax();
  387. int py2 = doorInst.getYMax();
  388. int pz2 = doorInst.getZMax();
  389. int l = tx - x;
  390. int m = ty - y;
  391. int n = tz - z;
  392. int dk;
  393. if ((dk = ((doorInst.getA() * l) + (doorInst.getB() * m) + (doorInst.getC() * n))) == 0)
  394. {
  395. continue; // Parallel
  396. }
  397. float p = (float) ((doorInst.getA() * x) + (doorInst.getB() * y) + (doorInst.getC() * z) + doorInst.getD()) / (float) dk;
  398. int fx = (int) (x - (l * p));
  399. int fy = (int) (y - (m * p));
  400. int fz = (int) (z - (n * p));
  401. if (((Math.min(x, tx) <= fx) && (fx <= Math.max(x, tx))) && ((Math.min(y, ty) <= fy) && (fy <= Math.max(y, ty))) && ((Math.min(z, tz) <= fz) && (fz <= Math.max(z, tz))))
  402. {
  403. if ((((fx >= px1) && (fx <= px2)) || ((fx >= px2) && (fx <= px1))) && (((fy >= py1) && (fy <= py2)) || ((fy >= py2) && (fy <= py1))) && (((fz >= pz1) && (fz <= pz2)) || ((fz >= pz2) && (fz <= pz1))))
  404. {
  405. return true; // Door between
  406. }
  407. }
  408. }
  409. }
  410. }
  411. }
  412. return false;
  413. }
  414. private void onStart()
  415. {
  416. try
  417. {
  418. checkAutoOpen();
  419. }
  420. catch (NullPointerException e)
  421. {
  422. _log.log(Level.WARNING, "There are errors in your Door.csv file. Update door.csv", e);
  423. }
  424. }
  425. public static DoorTable getInstance()
  426. {
  427. return SingletonHolder._instance;
  428. }
  429. private static class SingletonHolder
  430. {
  431. protected static final DoorTable _instance = new DoorTable();
  432. }
  433. }