2013-04-11 08:49:14 +02:00
|
|
|
package com.massivecraft.factions;
|
|
|
|
|
2013-04-11 10:11:14 +02:00
|
|
|
import java.lang.reflect.Type;
|
2013-04-11 08:49:14 +02:00
|
|
|
import java.util.ArrayList;
|
2013-04-11 09:38:08 +02:00
|
|
|
import java.util.Collections;
|
2013-04-11 08:49:14 +02:00
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.Map.Entry;
|
|
|
|
import java.util.concurrent.ConcurrentSkipListMap;
|
|
|
|
|
|
|
|
import org.bukkit.ChatColor;
|
|
|
|
|
|
|
|
import com.massivecraft.factions.iface.RelationParticipator;
|
|
|
|
import com.massivecraft.factions.integration.LWCFeatures;
|
|
|
|
import com.massivecraft.factions.util.AsciiCompass;
|
|
|
|
import com.massivecraft.mcore.ps.PS;
|
|
|
|
import com.massivecraft.mcore.store.Entity;
|
|
|
|
import com.massivecraft.mcore.util.Txt;
|
2013-04-11 10:11:14 +02:00
|
|
|
import com.massivecraft.mcore.xlib.gson.reflect.TypeToken;
|
2013-04-11 08:49:14 +02:00
|
|
|
|
2013-04-12 15:10:11 +02:00
|
|
|
public class Board extends Entity<Board> implements BoardInterface
|
2013-04-11 08:49:14 +02:00
|
|
|
{
|
2013-04-11 10:11:14 +02:00
|
|
|
public static final transient Type MAP_TYPE = new TypeToken<Map<PS, TerritoryAccess>>(){}.getType();
|
|
|
|
|
2013-04-11 08:49:14 +02:00
|
|
|
// -------------------------------------------- //
|
|
|
|
// META
|
|
|
|
// -------------------------------------------- //
|
|
|
|
|
|
|
|
public static Board get(Object oid)
|
|
|
|
{
|
|
|
|
return BoardColl.get().get(oid);
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------- //
|
2013-04-11 09:25:26 +02:00
|
|
|
// OVERRIDE: ENTITY
|
2013-04-11 08:49:14 +02:00
|
|
|
// -------------------------------------------- //
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Board load(Board that)
|
|
|
|
{
|
|
|
|
this.map = that.map;
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean isDefault()
|
|
|
|
{
|
|
|
|
if (this.map == null) return true;
|
|
|
|
if (this.map.isEmpty()) return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------- //
|
|
|
|
// FIELDS
|
|
|
|
// -------------------------------------------- //
|
|
|
|
|
2013-04-11 09:38:08 +02:00
|
|
|
// TODO: Make TerritoryAccess immutable.
|
|
|
|
|
|
|
|
private ConcurrentSkipListMap<PS, TerritoryAccess> map;
|
|
|
|
public Map<PS, TerritoryAccess> getMap() { return Collections.unmodifiableMap(this.map); }
|
|
|
|
|
|
|
|
// -------------------------------------------- //
|
|
|
|
// CONSTRUCT
|
|
|
|
// -------------------------------------------- //
|
|
|
|
|
|
|
|
public Board()
|
|
|
|
{
|
|
|
|
this.map = new ConcurrentSkipListMap<PS, TerritoryAccess>();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Board(Map<PS, TerritoryAccess> map)
|
|
|
|
{
|
|
|
|
this.map = new ConcurrentSkipListMap<PS, TerritoryAccess>(map);
|
|
|
|
}
|
2013-04-11 08:49:14 +02:00
|
|
|
|
|
|
|
// -------------------------------------------- //
|
2013-04-11 09:25:26 +02:00
|
|
|
// OVERRIDE: BOARD
|
2013-04-11 08:49:14 +02:00
|
|
|
// -------------------------------------------- //
|
|
|
|
|
2013-04-11 09:25:26 +02:00
|
|
|
// GET
|
|
|
|
|
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public TerritoryAccess getTerritoryAccessAt(PS ps)
|
|
|
|
{
|
2013-04-11 09:25:26 +02:00
|
|
|
if (ps == null) return null;
|
2013-04-11 08:49:14 +02:00
|
|
|
ps = ps.getChunkCoords(true);
|
|
|
|
TerritoryAccess ret = this.map.get(ps);
|
2013-04-12 09:51:14 +02:00
|
|
|
if (ret == null) ret = new TerritoryAccess(Const.FACTIONID_NONE);
|
2013-04-11 08:49:14 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public Faction getFactionAt(PS ps)
|
|
|
|
{
|
2013-04-11 09:25:26 +02:00
|
|
|
if (ps == null) return null;
|
2013-04-11 08:49:14 +02:00
|
|
|
return this.getTerritoryAccessAt(ps).getHostFaction();
|
|
|
|
}
|
|
|
|
|
|
|
|
// SET
|
|
|
|
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public void setTerritoryAccessAt(PS ps, TerritoryAccess territoryAccess)
|
|
|
|
{
|
|
|
|
ps = ps.getChunkCoords(true);
|
|
|
|
|
2013-04-12 09:51:14 +02:00
|
|
|
if (territoryAccess == null || (territoryAccess.getHostFactionId().equals(Const.FACTIONID_NONE) && territoryAccess.isDefault()))
|
2013-04-11 08:49:14 +02:00
|
|
|
{
|
|
|
|
// TODO: Listen to an event instead!
|
2013-04-18 17:47:42 +02:00
|
|
|
// NOTE: And this is probably the place where the event should be triggered!
|
2013-04-11 08:49:14 +02:00
|
|
|
if (ConfServer.onUnclaimResetLwcLocks && LWCFeatures.getEnabled())
|
|
|
|
{
|
|
|
|
LWCFeatures.clearAllChests(ps);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.map.remove(ps);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.map.put(ps, territoryAccess);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.changed();
|
|
|
|
}
|
|
|
|
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public void setFactionAt(PS ps, Faction faction)
|
|
|
|
{
|
|
|
|
TerritoryAccess territoryAccess = null;
|
|
|
|
if (faction != null)
|
|
|
|
{
|
|
|
|
territoryAccess = new TerritoryAccess(faction.getId());
|
|
|
|
}
|
|
|
|
this.setTerritoryAccessAt(ps, territoryAccess);
|
|
|
|
}
|
|
|
|
|
|
|
|
// REMOVE
|
|
|
|
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public void removeAt(PS ps)
|
|
|
|
{
|
|
|
|
this.setTerritoryAccessAt(ps, null);
|
|
|
|
}
|
|
|
|
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
|
|
|
public void removeAll(Faction faction)
|
2013-04-11 08:49:14 +02:00
|
|
|
{
|
2013-04-11 09:25:26 +02:00
|
|
|
String factionId = faction.getId();
|
|
|
|
|
2013-04-11 08:49:14 +02:00
|
|
|
for (Entry<PS, TerritoryAccess> entry : this.map.entrySet())
|
|
|
|
{
|
|
|
|
TerritoryAccess territoryAccess = entry.getValue();
|
|
|
|
if ( ! territoryAccess.getHostFactionId().equals(factionId)) continue;
|
|
|
|
|
|
|
|
PS ps = entry.getKey();
|
|
|
|
this.removeAt(ps);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Removes orphaned foreign keys
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public void clean()
|
|
|
|
{
|
|
|
|
for (Entry<PS, TerritoryAccess> entry : this.map.entrySet())
|
|
|
|
{
|
|
|
|
TerritoryAccess territoryAccess = entry.getValue();
|
2013-04-12 09:47:43 +02:00
|
|
|
|
|
|
|
if (FactionColl.get().containsId(territoryAccess.getHostFactionId())) continue;
|
2013-04-11 08:49:14 +02:00
|
|
|
|
|
|
|
PS ps = entry.getKey();
|
|
|
|
this.removeAt(ps);
|
|
|
|
|
|
|
|
Factions.get().log("Board cleaner removed "+territoryAccess.getHostFactionId()+" from "+entry.getKey());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// COUNT
|
2013-04-11 09:25:26 +02:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public int getCount(Faction faction)
|
|
|
|
{
|
|
|
|
return this.getCount(faction.getId());
|
|
|
|
}
|
2013-04-11 08:49:14 +02:00
|
|
|
|
|
|
|
public int getCount(String factionId)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
for (TerritoryAccess territoryAccess : this.map.values())
|
|
|
|
{
|
|
|
|
if(territoryAccess.getHostFactionId().equals(factionId)) ret += 1;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// NEARBY DETECTION
|
2013-04-11 09:25:26 +02:00
|
|
|
|
2013-04-11 08:49:14 +02:00
|
|
|
// Is this coord NOT completely surrounded by coords claimed by the same faction?
|
|
|
|
// Simpler: Is there any nearby coord with a faction other than the faction here?
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public boolean isBorderPs(PS ps)
|
|
|
|
{
|
|
|
|
PS nearby = null;
|
|
|
|
Faction faction = this.getFactionAt(ps);
|
|
|
|
|
|
|
|
nearby = ps.withChunkX(ps.getChunkX() +1);
|
|
|
|
if (faction != this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
nearby = ps.withChunkX(ps.getChunkX() -1);
|
|
|
|
if (faction != this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
nearby = ps.withChunkZ(ps.getChunkZ() +1);
|
|
|
|
if (faction != this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
nearby = ps.withChunkZ(ps.getChunkZ() -1);
|
|
|
|
if (faction != this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Is this coord connected to any coord claimed by the specified faction?
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public boolean isConnectedPs(PS ps, Faction faction)
|
|
|
|
{
|
|
|
|
PS nearby = null;
|
|
|
|
|
|
|
|
nearby = ps.withChunkX(ps.getChunkX() +1);
|
|
|
|
if (faction == this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
nearby = ps.withChunkX(ps.getChunkX() -1);
|
|
|
|
if (faction == this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
nearby = ps.withChunkZ(ps.getChunkZ() +1);
|
|
|
|
if (faction == this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
nearby = ps.withChunkZ(ps.getChunkZ() -1);
|
|
|
|
if (faction == this.getFactionAt(nearby)) return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// MAP GENERATION
|
|
|
|
|
2013-04-11 09:25:26 +02:00
|
|
|
@Override
|
2013-04-11 08:49:14 +02:00
|
|
|
public ArrayList<String> getMap(RelationParticipator observer, PS centerPs, double inDegrees)
|
|
|
|
{
|
|
|
|
centerPs = centerPs.getChunkCoords(true);
|
|
|
|
|
|
|
|
ArrayList<String> ret = new ArrayList<String>();
|
|
|
|
Faction centerFaction = this.getFactionAt(centerPs);
|
|
|
|
|
|
|
|
ret.add(Txt.titleize("("+centerPs.getChunkX() + "," + centerPs.getChunkZ()+") "+centerFaction.getTag(observer)));
|
|
|
|
|
|
|
|
int halfWidth = Const.MAP_WIDTH / 2;
|
|
|
|
int halfHeight = Const.MAP_HEIGHT / 2;
|
|
|
|
|
|
|
|
PS topLeftPs = centerPs.plusChunkCoords(-halfWidth, -halfHeight);
|
|
|
|
|
|
|
|
int width = halfWidth * 2 + 1;
|
|
|
|
int height = halfHeight * 2 + 1;
|
|
|
|
|
|
|
|
//Make room for the list of tags
|
|
|
|
height--;
|
|
|
|
|
|
|
|
Map<Faction, Character> fList = new HashMap<Faction, Character>();
|
|
|
|
int chrIdx = 0;
|
|
|
|
|
|
|
|
// For each row
|
|
|
|
for (int dz = 0; dz < height; dz++)
|
|
|
|
{
|
|
|
|
// Draw and add that row
|
|
|
|
String row = "";
|
|
|
|
for (int dx = 0; dx < width; dx++)
|
|
|
|
{
|
|
|
|
if(dx == halfWidth && dz == halfHeight)
|
|
|
|
{
|
|
|
|
row += ChatColor.AQUA+"+";
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
PS herePs = topLeftPs.plusChunkCoords(dx, dz);
|
|
|
|
Faction hereFaction = this.getFactionAt(herePs);
|
|
|
|
if (hereFaction.isNone())
|
|
|
|
{
|
|
|
|
row += ChatColor.GRAY+"-";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!fList.containsKey(hereFaction))
|
|
|
|
fList.put(hereFaction, Const.MAP_KEY_CHARS[chrIdx++]);
|
|
|
|
char fchar = fList.get(hereFaction);
|
|
|
|
row += hereFaction.getColorTo(observer) + "" + fchar;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret.add(row);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the compass
|
|
|
|
ArrayList<String> asciiCompass = AsciiCompass.getAsciiCompass(inDegrees, ChatColor.RED, Txt.parse("<a>"));
|
|
|
|
|
|
|
|
// Add the compass
|
|
|
|
ret.set(1, asciiCompass.get(0)+ret.get(1).substring(3*3));
|
|
|
|
ret.set(2, asciiCompass.get(1)+ret.get(2).substring(3*3));
|
|
|
|
ret.set(3, asciiCompass.get(2)+ret.get(3).substring(3*3));
|
|
|
|
|
|
|
|
String fRow = "";
|
|
|
|
for (Faction keyfaction : fList.keySet())
|
|
|
|
{
|
|
|
|
fRow += ""+keyfaction.getColorTo(observer) + fList.get(keyfaction) + ": " + keyfaction.getTag() + " ";
|
|
|
|
}
|
|
|
|
fRow = fRow.trim();
|
|
|
|
ret.add(fRow);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|