939 lines
25 KiB
Java
939 lines
25 KiB
Java
/*
|
|
* Copyright (C) 2004-2015 L2J Server
|
|
*
|
|
* This file is part of L2J Server.
|
|
*
|
|
* L2J Server 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.
|
|
*
|
|
* L2J Server 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
package com.l2jserver.gameserver.model.entity;
|
|
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.time.DayOfWeek;
|
|
import java.util.ArrayList;
|
|
import java.util.Collection;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Objects;
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
import java.util.concurrent.CopyOnWriteArrayList;
|
|
import java.util.concurrent.ScheduledFuture;
|
|
import java.util.logging.Level;
|
|
import java.util.logging.Logger;
|
|
|
|
import javax.xml.parsers.DocumentBuilderFactory;
|
|
|
|
import org.w3c.dom.Document;
|
|
import org.w3c.dom.NamedNodeMap;
|
|
import org.w3c.dom.Node;
|
|
|
|
import com.l2jserver.Config;
|
|
import com.l2jserver.gameserver.ThreadPoolManager;
|
|
import com.l2jserver.gameserver.data.xml.impl.DoorData;
|
|
import com.l2jserver.gameserver.data.xml.impl.NpcData;
|
|
import com.l2jserver.gameserver.enums.ChatType;
|
|
import com.l2jserver.gameserver.enums.InstanceReenterType;
|
|
import com.l2jserver.gameserver.enums.InstanceRemoveBuffType;
|
|
import com.l2jserver.gameserver.instancemanager.InstanceManager;
|
|
import com.l2jserver.gameserver.model.L2Spawn;
|
|
import com.l2jserver.gameserver.model.L2World;
|
|
import com.l2jserver.gameserver.model.L2WorldRegion;
|
|
import com.l2jserver.gameserver.model.Location;
|
|
import com.l2jserver.gameserver.model.StatsSet;
|
|
import com.l2jserver.gameserver.model.TeleportWhereType;
|
|
import com.l2jserver.gameserver.model.actor.L2Character;
|
|
import com.l2jserver.gameserver.model.actor.L2Npc;
|
|
import com.l2jserver.gameserver.model.actor.instance.L2DoorInstance;
|
|
import com.l2jserver.gameserver.model.actor.instance.L2PcInstance;
|
|
import com.l2jserver.gameserver.model.actor.templates.L2DoorTemplate;
|
|
import com.l2jserver.gameserver.model.actor.templates.L2NpcTemplate;
|
|
import com.l2jserver.gameserver.model.holders.InstanceReenterTimeHolder;
|
|
import com.l2jserver.gameserver.model.instancezone.InstanceWorld;
|
|
import com.l2jserver.gameserver.network.SystemMessageId;
|
|
import com.l2jserver.gameserver.network.serverpackets.CreatureSay;
|
|
import com.l2jserver.gameserver.network.serverpackets.SystemMessage;
|
|
import com.l2jserver.gameserver.util.Broadcast;
|
|
|
|
/**
|
|
* Main class for game instances.
|
|
* @author evill33t, GodKratos
|
|
*/
|
|
public final class Instance
|
|
{
|
|
private static final Logger _log = Logger.getLogger(Instance.class.getName());
|
|
|
|
private final int _id;
|
|
private String _name;
|
|
private int _ejectTime = Config.EJECT_DEAD_PLAYER_TIME;
|
|
/** Allow random walk for NPCs, global parameter. */
|
|
private boolean _allowRandomWalk = true;
|
|
private final List<Integer> _players = new CopyOnWriteArrayList<>();
|
|
private final List<L2Npc> _npcs = new CopyOnWriteArrayList<>();
|
|
private final Map<Integer, L2DoorInstance> _doors = new ConcurrentHashMap<>();
|
|
private final Map<String, List<L2Spawn>> _manualSpawn = new HashMap<>();
|
|
private Location _spawnLoc = null;
|
|
private boolean _allowSummon = true;
|
|
private long _emptyDestroyTime = -1;
|
|
private long _lastLeft = -1;
|
|
private long _instanceStartTime = -1;
|
|
private long _instanceEndTime = -1;
|
|
private boolean _isPvPInstance = false;
|
|
private boolean _showTimer = false;
|
|
private boolean _isTimerIncrease = true;
|
|
private String _timerText = "";
|
|
// Instance reset data
|
|
private InstanceReenterType _type = InstanceReenterType.NONE;
|
|
private final List<InstanceReenterTimeHolder> _resetData = new ArrayList<>();
|
|
// Instance remove buffs data
|
|
private InstanceRemoveBuffType _removeBuffType = InstanceRemoveBuffType.NONE;
|
|
private final List<Integer> _exceptionList = new ArrayList<>();
|
|
|
|
protected ScheduledFuture<?> _checkTimeUpTask = null;
|
|
protected final Map<Integer, ScheduledFuture<?>> _ejectDeadTasks = new HashMap<>();
|
|
|
|
public Instance(int id)
|
|
{
|
|
_id = id;
|
|
_instanceStartTime = System.currentTimeMillis();
|
|
}
|
|
|
|
public Instance(int id, String name)
|
|
{
|
|
_id = id;
|
|
_name = name;
|
|
_instanceStartTime = System.currentTimeMillis();
|
|
}
|
|
|
|
/**
|
|
* @return the ID of this instance.
|
|
*/
|
|
public int getId()
|
|
{
|
|
return _id;
|
|
}
|
|
|
|
/**
|
|
* @return the name of this instance
|
|
*/
|
|
public String getName()
|
|
{
|
|
return _name;
|
|
}
|
|
|
|
public void setName(String name)
|
|
{
|
|
_name = name;
|
|
}
|
|
|
|
/**
|
|
* @return the eject time
|
|
*/
|
|
public int getEjectTime()
|
|
{
|
|
return _ejectTime;
|
|
}
|
|
|
|
/**
|
|
* @param ejectTime the player eject time upon death
|
|
*/
|
|
public void setEjectTime(int ejectTime)
|
|
{
|
|
_ejectTime = ejectTime;
|
|
}
|
|
|
|
/**
|
|
* @return whether summon friend type skills are allowed for this instance
|
|
*/
|
|
public boolean isSummonAllowed()
|
|
{
|
|
return _allowSummon;
|
|
}
|
|
|
|
/**
|
|
* Sets the status for the instance for summon friend type skills
|
|
* @param b
|
|
*/
|
|
public void setAllowSummon(boolean b)
|
|
{
|
|
_allowSummon = b;
|
|
}
|
|
|
|
/**
|
|
* Returns true if entire instance is PvP zone
|
|
* @return
|
|
*/
|
|
public boolean isPvPInstance()
|
|
{
|
|
return _isPvPInstance;
|
|
}
|
|
|
|
/**
|
|
* Sets PvP zone status of the instance
|
|
* @param b
|
|
*/
|
|
public void setPvPInstance(boolean b)
|
|
{
|
|
_isPvPInstance = b;
|
|
}
|
|
|
|
/**
|
|
* Set the instance duration task
|
|
* @param duration in milliseconds
|
|
*/
|
|
public void setDuration(int duration)
|
|
{
|
|
if (_checkTimeUpTask != null)
|
|
{
|
|
_checkTimeUpTask.cancel(true);
|
|
}
|
|
|
|
_checkTimeUpTask = ThreadPoolManager.getInstance().scheduleGeneral(new CheckTimeUp(duration), 500);
|
|
_instanceEndTime = System.currentTimeMillis() + duration + 500;
|
|
}
|
|
|
|
/**
|
|
* Set time before empty instance will be removed
|
|
* @param time in milliseconds
|
|
*/
|
|
public void setEmptyDestroyTime(long time)
|
|
{
|
|
_emptyDestroyTime = time;
|
|
}
|
|
|
|
/**
|
|
* Checks if the player exists within this instance
|
|
* @param objectId
|
|
* @return true if player exists in instance
|
|
*/
|
|
public boolean containsPlayer(int objectId)
|
|
{
|
|
return _players.contains(objectId);
|
|
}
|
|
|
|
/**
|
|
* Adds the specified player to the instance
|
|
* @param objectId Players object ID
|
|
*/
|
|
public void addPlayer(int objectId)
|
|
{
|
|
_players.add(objectId);
|
|
}
|
|
|
|
/**
|
|
* Removes the specified player from the instance list.
|
|
* @param objectId the player's object Id
|
|
*/
|
|
public void removePlayer(Integer objectId)
|
|
{
|
|
_players.remove(objectId);
|
|
if (_players.isEmpty() && (_emptyDestroyTime >= 0))
|
|
{
|
|
_lastLeft = System.currentTimeMillis();
|
|
setDuration((int) (_instanceEndTime - System.currentTimeMillis() - 500));
|
|
}
|
|
}
|
|
|
|
public void addNpc(L2Npc npc)
|
|
{
|
|
_npcs.add(npc);
|
|
}
|
|
|
|
public void removeNpc(L2Npc npc)
|
|
{
|
|
if (npc.getSpawn() != null)
|
|
{
|
|
npc.getSpawn().stopRespawn();
|
|
}
|
|
_npcs.remove(npc);
|
|
}
|
|
|
|
/**
|
|
* Adds a door into the instance
|
|
* @param doorId - from doors.xml
|
|
* @param set - StatsSet for initializing door
|
|
*/
|
|
public void addDoor(int doorId, StatsSet set)
|
|
{
|
|
if (_doors.containsKey(doorId))
|
|
{
|
|
_log.warning("Door ID " + doorId + " already exists in instance " + getId());
|
|
return;
|
|
}
|
|
|
|
final L2DoorInstance newdoor = new L2DoorInstance(new L2DoorTemplate(set));
|
|
newdoor.setInstanceId(getId());
|
|
newdoor.setCurrentHp(newdoor.getMaxHp());
|
|
newdoor.spawnMe(newdoor.getTemplate().getX(), newdoor.getTemplate().getY(), newdoor.getTemplate().getZ());
|
|
_doors.put(doorId, newdoor);
|
|
}
|
|
|
|
public List<Integer> getPlayers()
|
|
{
|
|
return _players;
|
|
}
|
|
|
|
public List<L2Npc> getNpcs()
|
|
{
|
|
return _npcs;
|
|
}
|
|
|
|
public Collection<L2DoorInstance> getDoors()
|
|
{
|
|
return _doors.values();
|
|
}
|
|
|
|
public L2DoorInstance getDoor(int id)
|
|
{
|
|
return _doors.get(id);
|
|
}
|
|
|
|
public long getInstanceEndTime()
|
|
{
|
|
return _instanceEndTime;
|
|
}
|
|
|
|
public long getInstanceStartTime()
|
|
{
|
|
return _instanceStartTime;
|
|
}
|
|
|
|
public boolean isShowTimer()
|
|
{
|
|
return _showTimer;
|
|
}
|
|
|
|
public boolean isTimerIncrease()
|
|
{
|
|
return _isTimerIncrease;
|
|
}
|
|
|
|
public String getTimerText()
|
|
{
|
|
return _timerText;
|
|
}
|
|
|
|
/**
|
|
* @return the spawn location for this instance to be used when leaving the instance
|
|
*/
|
|
public Location getSpawnLoc()
|
|
{
|
|
return _spawnLoc;
|
|
}
|
|
|
|
/**
|
|
* Sets the spawn location for this instance to be used when leaving the instance
|
|
* @param loc
|
|
*/
|
|
public void setSpawnLoc(Location loc)
|
|
{
|
|
_spawnLoc = loc;
|
|
}
|
|
|
|
public void removePlayers()
|
|
{
|
|
for (Integer objectId : _players)
|
|
{
|
|
final L2PcInstance player = L2World.getInstance().getPlayer(objectId);
|
|
if ((player != null) && (player.getInstanceId() == getId()))
|
|
{
|
|
player.setInstanceId(0);
|
|
if (getSpawnLoc() != null)
|
|
{
|
|
player.teleToLocation(getSpawnLoc(), true);
|
|
}
|
|
else
|
|
{
|
|
player.teleToLocation(TeleportWhereType.TOWN);
|
|
}
|
|
}
|
|
}
|
|
_players.clear();
|
|
}
|
|
|
|
public void removeNpcs()
|
|
{
|
|
for (L2Npc mob : _npcs)
|
|
{
|
|
if (mob != null)
|
|
{
|
|
if (mob.getSpawn() != null)
|
|
{
|
|
mob.getSpawn().stopRespawn();
|
|
}
|
|
mob.deleteMe();
|
|
}
|
|
}
|
|
_npcs.clear();
|
|
_manualSpawn.clear();
|
|
}
|
|
|
|
public void removeSpawnedNpcs()
|
|
{
|
|
_npcs.stream().filter(Objects::nonNull).forEach(L2Npc::deleteMe);
|
|
_npcs.clear();
|
|
}
|
|
|
|
public void removeDoors()
|
|
{
|
|
for (L2DoorInstance door : _doors.values())
|
|
{
|
|
if (door != null)
|
|
{
|
|
L2WorldRegion region = door.getWorldRegion();
|
|
door.decayMe();
|
|
|
|
if (region != null)
|
|
{
|
|
region.removeVisibleObject(door);
|
|
}
|
|
|
|
door.getKnownList().removeAllKnownObjects();
|
|
L2World.getInstance().removeObject(door);
|
|
}
|
|
}
|
|
_doors.clear();
|
|
}
|
|
|
|
/**
|
|
* Spawns group of instance NPC's
|
|
* @param groupName - name of group from XML definition to spawn
|
|
* @return list of spawned NPC's
|
|
*/
|
|
public List<L2Npc> spawnGroup(String groupName)
|
|
{
|
|
List<L2Npc> ret = null;
|
|
if (_manualSpawn.containsKey(groupName))
|
|
{
|
|
final List<L2Spawn> manualSpawn = _manualSpawn.get(groupName);
|
|
ret = new ArrayList<>(manualSpawn.size());
|
|
|
|
for (L2Spawn spawnDat : manualSpawn)
|
|
{
|
|
ret.add(spawnDat.doSpawn());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_log.warning(getName() + " instance: cannot spawn NPC's, wrong group name: " + groupName);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
public void loadInstanceTemplate(String filename)
|
|
{
|
|
Document doc = null;
|
|
File xml = new File(Config.DATAPACK_ROOT, "data/instances/" + filename);
|
|
|
|
try
|
|
{
|
|
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
|
|
factory.setValidating(false);
|
|
factory.setIgnoringComments(true);
|
|
doc = factory.newDocumentBuilder().parse(xml);
|
|
|
|
for (Node n = doc.getFirstChild(); n != null; n = n.getNextSibling())
|
|
{
|
|
if ("instance".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
parseInstance(n);
|
|
}
|
|
}
|
|
}
|
|
catch (IOException e)
|
|
{
|
|
_log.log(Level.WARNING, "Instance: can not find " + xml.getAbsolutePath() + " ! " + e.getMessage(), e);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
_log.log(Level.WARNING, "Instance: error while loading " + xml.getAbsolutePath() + " ! " + e.getMessage(), e);
|
|
}
|
|
}
|
|
|
|
private void parseInstance(Node n) throws Exception
|
|
{
|
|
L2Spawn spawnDat;
|
|
L2NpcTemplate npcTemplate;
|
|
_name = n.getAttributes().getNamedItem("name").getNodeValue();
|
|
Node a = n.getAttributes().getNamedItem("ejectTime");
|
|
if (a != null)
|
|
{
|
|
_ejectTime = 1000 * Integer.parseInt(a.getNodeValue());
|
|
}
|
|
a = n.getAttributes().getNamedItem("allowRandomWalk");
|
|
if (a != null)
|
|
{
|
|
_allowRandomWalk = Boolean.parseBoolean(a.getNodeValue());
|
|
}
|
|
Node first = n.getFirstChild();
|
|
for (n = first; n != null; n = n.getNextSibling())
|
|
{
|
|
if ("activityTime".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("val");
|
|
if (a != null)
|
|
{
|
|
_checkTimeUpTask = ThreadPoolManager.getInstance().scheduleGeneral(new CheckTimeUp(Integer.parseInt(a.getNodeValue()) * 60000), 15000);
|
|
_instanceEndTime = System.currentTimeMillis() + (Long.parseLong(a.getNodeValue()) * 60000) + 15000;
|
|
}
|
|
}
|
|
// @formatter:off
|
|
/*
|
|
else if ("timeDelay".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("val");
|
|
if (a != null)
|
|
{
|
|
instance.setTimeDelay(Integer.parseInt(a.getNodeValue()));
|
|
}
|
|
}
|
|
*/
|
|
// @formatter:on
|
|
else if ("allowSummon".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("val");
|
|
if (a != null)
|
|
{
|
|
setAllowSummon(Boolean.parseBoolean(a.getNodeValue()));
|
|
}
|
|
}
|
|
else if ("emptyDestroyTime".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("val");
|
|
if (a != null)
|
|
{
|
|
_emptyDestroyTime = Long.parseLong(a.getNodeValue()) * 1000;
|
|
}
|
|
}
|
|
else if ("showTimer".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("val");
|
|
if (a != null)
|
|
{
|
|
_showTimer = Boolean.parseBoolean(a.getNodeValue());
|
|
}
|
|
a = n.getAttributes().getNamedItem("increase");
|
|
if (a != null)
|
|
{
|
|
_isTimerIncrease = Boolean.parseBoolean(a.getNodeValue());
|
|
}
|
|
a = n.getAttributes().getNamedItem("text");
|
|
if (a != null)
|
|
{
|
|
_timerText = a.getNodeValue();
|
|
}
|
|
}
|
|
else if ("PvPInstance".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("val");
|
|
if (a != null)
|
|
{
|
|
setPvPInstance(Boolean.parseBoolean(a.getNodeValue()));
|
|
}
|
|
}
|
|
else if ("doorlist".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
for (Node d = n.getFirstChild(); d != null; d = d.getNextSibling())
|
|
{
|
|
int doorId = 0;
|
|
if ("door".equalsIgnoreCase(d.getNodeName()))
|
|
{
|
|
doorId = Integer.parseInt(d.getAttributes().getNamedItem("doorId").getNodeValue());
|
|
StatsSet set = new StatsSet();
|
|
set.add(DoorData.getInstance().getDoorTemplate(doorId));
|
|
for (Node bean = d.getFirstChild(); bean != null; bean = bean.getNextSibling())
|
|
{
|
|
if ("set".equalsIgnoreCase(bean.getNodeName()))
|
|
{
|
|
NamedNodeMap attrs = bean.getAttributes();
|
|
String setname = attrs.getNamedItem("name").getNodeValue();
|
|
String value = attrs.getNamedItem("val").getNodeValue();
|
|
set.set(setname, value);
|
|
}
|
|
}
|
|
addDoor(doorId, set);
|
|
}
|
|
}
|
|
}
|
|
else if ("spawnlist".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
for (Node group = n.getFirstChild(); group != null; group = group.getNextSibling())
|
|
{
|
|
if ("group".equalsIgnoreCase(group.getNodeName()))
|
|
{
|
|
String spawnGroup = group.getAttributes().getNamedItem("name").getNodeValue();
|
|
List<L2Spawn> manualSpawn = new ArrayList<>();
|
|
for (Node d = group.getFirstChild(); d != null; d = d.getNextSibling())
|
|
{
|
|
int npcId = 0, x = 0, y = 0, z = 0, heading = 0, respawn = 0, respawnRandom = 0;
|
|
Boolean allowRandomWalk = null;
|
|
if ("spawn".equalsIgnoreCase(d.getNodeName()))
|
|
{
|
|
|
|
npcId = Integer.parseInt(d.getAttributes().getNamedItem("npcId").getNodeValue());
|
|
x = Integer.parseInt(d.getAttributes().getNamedItem("x").getNodeValue());
|
|
y = Integer.parseInt(d.getAttributes().getNamedItem("y").getNodeValue());
|
|
z = Integer.parseInt(d.getAttributes().getNamedItem("z").getNodeValue());
|
|
heading = Integer.parseInt(d.getAttributes().getNamedItem("heading").getNodeValue());
|
|
respawn = Integer.parseInt(d.getAttributes().getNamedItem("respawn").getNodeValue());
|
|
if (d.getAttributes().getNamedItem("respawnRandom") != null)
|
|
{
|
|
respawnRandom = Integer.parseInt(d.getAttributes().getNamedItem("respawnRandom").getNodeValue());
|
|
}
|
|
if (d.getAttributes().getNamedItem("allowRandomWalk") != null)
|
|
{
|
|
allowRandomWalk = Boolean.valueOf(d.getAttributes().getNamedItem("allowRandomWalk").getNodeValue());
|
|
}
|
|
npcTemplate = NpcData.getInstance().getTemplate(npcId);
|
|
if (npcTemplate != null)
|
|
{
|
|
spawnDat = new L2Spawn(npcTemplate);
|
|
spawnDat.setX(x);
|
|
spawnDat.setY(y);
|
|
spawnDat.setZ(z);
|
|
spawnDat.setAmount(1);
|
|
spawnDat.setHeading(heading);
|
|
spawnDat.setRespawnDelay(respawn, respawnRandom);
|
|
if (respawn == 0)
|
|
{
|
|
spawnDat.stopRespawn();
|
|
}
|
|
else
|
|
{
|
|
spawnDat.startRespawn();
|
|
}
|
|
spawnDat.setInstanceId(getId());
|
|
if (allowRandomWalk == null)
|
|
{
|
|
spawnDat.setIsNoRndWalk(!_allowRandomWalk);
|
|
}
|
|
else
|
|
{
|
|
spawnDat.setIsNoRndWalk(!allowRandomWalk);
|
|
}
|
|
if (spawnGroup.equals("general"))
|
|
{
|
|
spawnDat.doSpawn();
|
|
}
|
|
else
|
|
{
|
|
manualSpawn.add(spawnDat);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_log.warning("Instance: Data missing in NPC table for ID: " + npcId + " in Instance " + getId());
|
|
}
|
|
}
|
|
}
|
|
if (!manualSpawn.isEmpty())
|
|
{
|
|
_manualSpawn.put(spawnGroup, manualSpawn);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ("spawnpoint".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
try
|
|
{
|
|
int x = Integer.parseInt(n.getAttributes().getNamedItem("spawnX").getNodeValue());
|
|
int y = Integer.parseInt(n.getAttributes().getNamedItem("spawnY").getNodeValue());
|
|
int z = Integer.parseInt(n.getAttributes().getNamedItem("spawnZ").getNodeValue());
|
|
_spawnLoc = new Location(x, y, z);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
_log.log(Level.WARNING, "Error parsing instance xml: " + e.getMessage(), e);
|
|
_spawnLoc = null;
|
|
}
|
|
}
|
|
else if ("reenter".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("additionStyle");
|
|
if (a != null)
|
|
{
|
|
_type = InstanceReenterType.valueOf(a.getNodeValue());
|
|
}
|
|
|
|
for (Node d = n.getFirstChild(); d != null; d = d.getNextSibling())
|
|
{
|
|
long time = -1;
|
|
DayOfWeek day = null;
|
|
int hour = -1;
|
|
int minute = -1;
|
|
|
|
if ("reset".equalsIgnoreCase(d.getNodeName()))
|
|
{
|
|
a = d.getAttributes().getNamedItem("time");
|
|
if (a != null)
|
|
{
|
|
time = Long.parseLong(a.getNodeValue());
|
|
|
|
if (time > 0)
|
|
{
|
|
_resetData.add(new InstanceReenterTimeHolder(time));
|
|
break;
|
|
}
|
|
}
|
|
else if (time == -1)
|
|
{
|
|
a = d.getAttributes().getNamedItem("day");
|
|
if (a != null)
|
|
{
|
|
day = DayOfWeek.valueOf(a.getNodeValue().toUpperCase());
|
|
}
|
|
|
|
a = d.getAttributes().getNamedItem("hour");
|
|
if (a != null)
|
|
{
|
|
hour = Integer.parseInt(a.getNodeValue());
|
|
}
|
|
|
|
a = d.getAttributes().getNamedItem("minute");
|
|
if (a != null)
|
|
{
|
|
minute = Integer.parseInt(a.getNodeValue());
|
|
}
|
|
_resetData.add(new InstanceReenterTimeHolder(day, hour, minute));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ("removeBuffs".equalsIgnoreCase(n.getNodeName()))
|
|
{
|
|
a = n.getAttributes().getNamedItem("type");
|
|
if (a != null)
|
|
{
|
|
_removeBuffType = InstanceRemoveBuffType.valueOf(a.getNodeValue().toUpperCase());
|
|
}
|
|
|
|
for (Node d = n.getFirstChild(); d != null; d = d.getNextSibling())
|
|
{
|
|
if ("skill".equalsIgnoreCase(d.getNodeName()))
|
|
{
|
|
a = d.getAttributes().getNamedItem("id");
|
|
if (a != null)
|
|
{
|
|
_exceptionList.add(Integer.parseInt(a.getNodeValue()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
protected void doCheckTimeUp(int remaining)
|
|
{
|
|
CreatureSay cs = null;
|
|
int timeLeft;
|
|
int interval;
|
|
|
|
if (_players.isEmpty() && (_emptyDestroyTime == 0))
|
|
{
|
|
remaining = 0;
|
|
interval = 500;
|
|
}
|
|
else if (_players.isEmpty() && (_emptyDestroyTime > 0))
|
|
{
|
|
|
|
Long emptyTimeLeft = (_lastLeft + _emptyDestroyTime) - System.currentTimeMillis();
|
|
if (emptyTimeLeft <= 0)
|
|
{
|
|
interval = 0;
|
|
remaining = 0;
|
|
}
|
|
else if ((remaining > 300000) && (emptyTimeLeft > 300000))
|
|
{
|
|
interval = 300000;
|
|
remaining = remaining - 300000;
|
|
}
|
|
else if ((remaining > 60000) && (emptyTimeLeft > 60000))
|
|
{
|
|
interval = 60000;
|
|
remaining = remaining - 60000;
|
|
}
|
|
else if ((remaining > 30000) && (emptyTimeLeft > 30000))
|
|
{
|
|
interval = 30000;
|
|
remaining = remaining - 30000;
|
|
}
|
|
else
|
|
{
|
|
interval = 10000;
|
|
remaining = remaining - 10000;
|
|
}
|
|
}
|
|
else if (remaining > 300000)
|
|
{
|
|
timeLeft = remaining / 60000;
|
|
interval = 300000;
|
|
SystemMessage sm = SystemMessage.getSystemMessage(SystemMessageId.THIS_DUNGEON_WILL_EXPIRE_IN_S1_MINUTE_S_YOU_WILL_BE_FORCED_OUT_OF_THE_DUNGEON_WHEN_THE_TIME_EXPIRES);
|
|
sm.addString(Integer.toString(timeLeft));
|
|
Broadcast.toPlayersInInstance(sm, getId());
|
|
remaining = remaining - 300000;
|
|
}
|
|
else if (remaining > 60000)
|
|
{
|
|
timeLeft = remaining / 60000;
|
|
interval = 60000;
|
|
SystemMessage sm = SystemMessage.getSystemMessage(SystemMessageId.THIS_DUNGEON_WILL_EXPIRE_IN_S1_MINUTE_S_YOU_WILL_BE_FORCED_OUT_OF_THE_DUNGEON_WHEN_THE_TIME_EXPIRES);
|
|
sm.addString(Integer.toString(timeLeft));
|
|
Broadcast.toPlayersInInstance(sm, getId());
|
|
remaining = remaining - 60000;
|
|
}
|
|
else if (remaining > 30000)
|
|
{
|
|
timeLeft = remaining / 1000;
|
|
interval = 30000;
|
|
cs = new CreatureSay(0, ChatType.ALLIANCE, "Notice", timeLeft + " seconds left.");
|
|
remaining = remaining - 30000;
|
|
}
|
|
else
|
|
{
|
|
timeLeft = remaining / 1000;
|
|
interval = 10000;
|
|
cs = new CreatureSay(0, ChatType.ALLIANCE, "Notice", timeLeft + " seconds left.");
|
|
remaining = remaining - 10000;
|
|
}
|
|
if (cs != null)
|
|
{
|
|
for (Integer objectId : _players)
|
|
{
|
|
final L2PcInstance player = L2World.getInstance().getPlayer(objectId);
|
|
if ((player != null) && (player.getInstanceId() == getId()))
|
|
{
|
|
player.sendPacket(cs);
|
|
}
|
|
}
|
|
}
|
|
cancelTimer();
|
|
if (remaining >= 10000)
|
|
{
|
|
_checkTimeUpTask = ThreadPoolManager.getInstance().scheduleGeneral(new CheckTimeUp(remaining), interval);
|
|
}
|
|
else
|
|
{
|
|
_checkTimeUpTask = ThreadPoolManager.getInstance().scheduleGeneral(new TimeUp(), interval);
|
|
}
|
|
}
|
|
|
|
public void cancelTimer()
|
|
{
|
|
if (_checkTimeUpTask != null)
|
|
{
|
|
_checkTimeUpTask.cancel(true);
|
|
}
|
|
}
|
|
|
|
public void cancelEjectDeadPlayer(L2PcInstance player)
|
|
{
|
|
final ScheduledFuture<?> task = _ejectDeadTasks.remove(player.getObjectId());
|
|
if (task != null)
|
|
{
|
|
task.cancel(true);
|
|
}
|
|
}
|
|
|
|
public void addEjectDeadTask(L2PcInstance player)
|
|
{
|
|
if ((player != null))
|
|
{
|
|
_ejectDeadTasks.put(player.getObjectId(), ThreadPoolManager.getInstance().scheduleGeneral(() ->
|
|
{
|
|
if (player.isDead() && (player.getInstanceId() == getId()))
|
|
{
|
|
player.setInstanceId(0);
|
|
if (getSpawnLoc() != null)
|
|
{
|
|
player.teleToLocation(getSpawnLoc(), true);
|
|
}
|
|
else
|
|
{
|
|
player.teleToLocation(TeleportWhereType.TOWN);
|
|
}
|
|
}
|
|
}, _ejectTime));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param killer the character that killed the {@code victim}
|
|
* @param victim the character that was killed by the {@code killer}
|
|
*/
|
|
public final void notifyDeath(L2Character killer, L2Character victim)
|
|
{
|
|
final InstanceWorld instance = InstanceManager.getInstance().getPlayerWorld(victim.getActingPlayer());
|
|
if (instance != null)
|
|
{
|
|
instance.onDeath(killer, victim);
|
|
}
|
|
}
|
|
|
|
public class CheckTimeUp implements Runnable
|
|
{
|
|
private final int _remaining;
|
|
|
|
public CheckTimeUp(int remaining)
|
|
{
|
|
_remaining = remaining;
|
|
}
|
|
|
|
@Override
|
|
public void run()
|
|
{
|
|
doCheckTimeUp(_remaining);
|
|
}
|
|
}
|
|
|
|
public class TimeUp implements Runnable
|
|
{
|
|
@Override
|
|
public void run()
|
|
{
|
|
InstanceManager.getInstance().destroyInstance(getId());
|
|
}
|
|
}
|
|
|
|
public InstanceReenterType getReenterType()
|
|
{
|
|
return _type;
|
|
}
|
|
|
|
public void setReenterType(InstanceReenterType type)
|
|
{
|
|
_type = type;
|
|
}
|
|
|
|
public List<InstanceReenterTimeHolder> getReenterData()
|
|
{
|
|
return _resetData;
|
|
}
|
|
|
|
public boolean isRemoveBuffEnabled()
|
|
{
|
|
return getRemoveBuffType() != InstanceRemoveBuffType.NONE;
|
|
}
|
|
|
|
public InstanceRemoveBuffType getRemoveBuffType()
|
|
{
|
|
return _removeBuffType;
|
|
}
|
|
|
|
public List<Integer> getBuffExceptionList()
|
|
{
|
|
return _exceptionList;
|
|
}
|
|
} |