Factions/src/com/massivecraft/factions/FPlayer.java

883 lines
24 KiB
Java
Raw Normal View History

2011-07-18 22:06:02 +02:00
package com.massivecraft.factions;
2011-02-06 13:36:11 +01:00
2011-10-22 16:00:24 +02:00
import java.util.HashSet;
import java.util.Set;
2011-02-06 13:36:11 +01:00
import org.bukkit.ChatColor;
import org.bukkit.Location;
2011-02-06 13:36:11 +01:00
import org.bukkit.entity.Player;
2011-07-18 22:06:02 +02:00
import com.massivecraft.factions.event.FactionsEventLandClaim;
2013-04-19 14:08:45 +02:00
import com.massivecraft.factions.event.FactionsEventMembershipChange;
import com.massivecraft.factions.event.FactionsEventMembershipChange.MembershipChangeReason;
2011-10-12 17:25:01 +02:00
import com.massivecraft.factions.iface.EconomyParticipator;
import com.massivecraft.factions.iface.RelationParticipator;
import com.massivecraft.factions.integration.Econ;
import com.massivecraft.factions.integration.LWCFeatures;
import com.massivecraft.factions.integration.SpoutFeatures;
import com.massivecraft.factions.integration.Worldguard;
2011-10-12 17:25:01 +02:00
import com.massivecraft.factions.util.RelationUtil;
2013-04-18 10:35:40 +02:00
import com.massivecraft.mcore.mixin.Mixin;
import com.massivecraft.mcore.ps.PS;
2013-04-12 08:56:26 +02:00
import com.massivecraft.mcore.store.SenderEntity;
2013-04-18 10:35:40 +02:00
import com.massivecraft.mcore.util.TimeUnit;
2013-04-10 10:32:04 +02:00
import com.massivecraft.mcore.util.Txt;
2011-02-06 13:36:11 +01:00
2013-04-12 08:56:26 +02:00
public class FPlayer extends SenderEntity<FPlayer> implements EconomyParticipator
{
// -------------------------------------------- //
2013-04-12 08:56:26 +02:00
// META
// -------------------------------------------- //
2013-04-12 08:56:26 +02:00
public static FPlayer get(Object oid)
{
2013-04-12 08:56:26 +02:00
return FPlayerColl.get().get(oid);
}
// -------------------------------------------- //
// OVERRIDE: ENTITY
// -------------------------------------------- //
2013-04-12 08:56:26 +02:00
@Override
public FPlayer load(FPlayer that)
{
this.setFactionId(that.factionId);
2013-04-17 15:30:21 +02:00
this.setRole(that.role);
this.setTitle(that.title);
this.setPowerBoost(that.powerBoost);
this.power = that.power;
this.lastPowerUpdateTime = that.lastPowerUpdateTime;
return this;
2013-04-12 08:56:26 +02:00
}
@Override
public boolean isDefault()
{
if (this.hasFaction()) return false;
2013-04-17 15:30:21 +02:00
// Note: we do not check role or title here since they mean nothing without a faction.
// TODO: This line looks obnoxious, investigate it.
2013-04-18 10:35:40 +02:00
if (this.getPowerRounded() != this.getPowerMaxRounded() && this.getPowerRounded() != (int) Math.round(ConfServer.powerStarting)) return false;
2013-04-12 08:56:26 +02:00
2013-04-17 15:30:21 +02:00
if (this.hasPowerBoost()) return false;
2013-04-12 08:56:26 +02:00
return true;
}
// -------------------------------------------- //
2013-04-17 15:30:21 +02:00
// FIELDS: RAW
// -------------------------------------------- //
2013-04-17 15:30:21 +02:00
// In this section of the source code we place the field declarations only.
// Each field has it's own section further down since just the getter and setter logic takes up quite some place.
2013-04-17 15:30:21 +02:00
// This is a foreign key.
// A players always belongs to a faction.
// If null the player belongs to the no-faction faction called Wilderness.
private String factionId = null;
2013-04-17 15:30:21 +02:00
// What role does the player have in the faction?
// The default value here is MEMBER since that one would be one of the most common ones and our goal is to save database space.
// A note to self is that we can not change it from member to anything else just because we feel like it, that would corrupt database content.
private Rel role = null;
// What title does the player have in the faction?
// The title is just for fun. It's completely meaningless.
// The default case is no title since it's what you start with and also the most common case.
// The player title is similar to the faction description.
//
// Question: Can the title contain chat colors?
// Answer: Yes but in such case the policy is that they already must be parsed using Txt.parse.
// If they contain markup it should not be parsed in case we coded the system correctly.
private String title = null;
// Player usually do not have a powerboost. It defaults to 0.
// The powerBoost is a custom increase/decrease to default and maximum power.
// Note that player powerBoost and faction powerBoost are very similar.
private Double powerBoost = null;
2013-04-18 10:35:40 +02:00
// This field contains the last calculated value of the players power.
// The power calculation is lazy which means that the power is calculated first when you try to view the value.
2013-04-17 15:30:21 +02:00
private double power;
2013-04-18 10:35:40 +02:00
// This is the timestamp for the last calculation of the power.
// The value is used for the lazy calculation described above.
2013-04-17 15:30:21 +02:00
private long lastPowerUpdateTime;
// -------------------------------------------- //
// FIELDS: RAW TRANSIENT
// -------------------------------------------- //
// Where did this player stand the last time we checked?
private transient PS currentChunk = null;
public PS getCurrentChunk() { return this.currentChunk; }
public void setCurrentChunk(PS currentChunk) { this.currentChunk = currentChunk.getChunk(true); }
// FIELD: mapAutoUpdating
private transient boolean mapAutoUpdating = false;
public void setMapAutoUpdating(boolean mapAutoUpdating) { this.mapAutoUpdating = mapAutoUpdating; }
public boolean isMapAutoUpdating() { return mapAutoUpdating; }
// FIELD: autoClaimEnabled
private transient Faction autoClaimFor = null;
public Faction getAutoClaimFor() { return autoClaimFor; }
public void setAutoClaimFor(Faction faction) { this.autoClaimFor = faction; }
private transient boolean usingAdminMode = false;
public boolean isUsingAdminMode() { return this.usingAdminMode; }
public void setUsingAdminMode(boolean val) { this.usingAdminMode = val; }
// FIELD: loginPvpDisabled
//private transient boolean loginPvpDisabled;
2013-04-17 15:30:21 +02:00
// FIELD: account
public String getAccountId() { return this.getId(); }
// -------------------------------------------- //
// CONSTRUCT
// -------------------------------------------- //
// GSON need this noarg constructor.
public FPlayer()
{
this.resetFactionData(false);
2013-04-18 10:35:40 +02:00
this.power = ConfServer.powerStarting;
2013-04-17 15:30:21 +02:00
this.lastPowerUpdateTime = System.currentTimeMillis();
if ( ! ConfServer.newPlayerStartingFactionID.equals(Const.FACTIONID_NONE) && FactionColl.get().containsId(ConfServer.newPlayerStartingFactionID))
{
this.factionId = ConfServer.newPlayerStartingFactionID;
}
}
public final void resetFactionData(boolean doSpoutUpdate)
{
// TODO: Should we not rather use ConfServer.newPlayerStartingFactionID here?
// The default neutral faction
this.setFactionId(null);
this.setRole(null);
this.setTitle(null);
this.autoClaimFor = null;
if (doSpoutUpdate)
{
SpoutFeatures.updateTitle(this, null);
SpoutFeatures.updateTitle(null, this);
SpoutFeatures.updateCape(this.getPlayer(), null);
}
}
public void resetFactionData()
{
this.resetFactionData(true);
}
// -------------------------------------------- //
// FIELD: factionId
// -------------------------------------------- //
// This method never returns null
public String getFactionId()
2011-10-08 23:22:02 +02:00
{
if (this.factionId == null) return Const.FACTIONID_NONE;
return this.factionId;
}
2013-04-17 15:30:21 +02:00
// This method never returns null
public Faction getFaction()
{
Faction ret = FactionColl.get().get(this.getFactionId());
if (ret == null) ret = FactionColl.get().get(Const.FACTIONID_NONE);
return ret;
}
public boolean hasFaction()
{
return !this.getFactionId().equals(Const.FACTIONID_NONE);
}
// This setter is so long because it search for default/null case and takes care of updating the faction member index
public void setFactionId(String factionId)
{
// Avoid null input
if (factionId == null) factionId = Const.FACTIONID_NONE;
// Get the old value
String oldFactionId = this.getFactionId();
// Ignore nochange
if (factionId.equals(oldFactionId)) return;
// Apply change
if (factionId.equals(Const.FACTIONID_NONE))
{
this.factionId = null;
}
else
{
this.factionId = factionId;
}
// Next we must be attached and inited
if (!this.attached()) return;
if (!this.getColl().inited()) return;
if (!FactionColl.get().inited()) return;
// Spout Derp
SpoutFeatures.updateTitle(this, null);
SpoutFeatures.updateTitle(null, this);
// Update index
Faction oldFaction = FactionColl.get().get(oldFactionId);
Faction faction = FactionColl.get().get(factionId);
oldFaction.fplayers.remove(this);
faction.fplayers.add(this);
// Mark as changed
this.changed();
}
public void setFaction(Faction faction)
{
this.setFactionId(faction.getId());
2011-10-08 23:22:02 +02:00
}
2013-04-17 15:30:21 +02:00
// -------------------------------------------- //
2011-10-08 23:22:02 +02:00
// FIELD: role
2013-04-17 15:30:21 +02:00
// -------------------------------------------- //
2011-10-08 23:22:02 +02:00
2013-04-17 15:30:21 +02:00
public Rel getRole()
{
if (this.role == null) return Rel.MEMBER;
return this.role;
}
2011-10-08 23:22:02 +02:00
2013-04-17 15:30:21 +02:00
public void setRole(Rel role)
{
if (role == null || role == Rel.MEMBER)
{
this.role = null;
}
else
{
this.role = role;
}
SpoutFeatures.updateTitle(this, null);
this.changed();
}
2011-10-08 23:22:02 +02:00
2013-04-12 08:11:11 +02:00
// -------------------------------------------- //
2013-04-17 15:30:21 +02:00
// FIELD: title
2013-04-12 08:11:11 +02:00
// -------------------------------------------- //
2013-04-17 15:30:21 +02:00
public boolean hasTitle()
{
return this.title != null;
}
2011-10-09 18:35:39 +02:00
2013-04-17 15:30:21 +02:00
public String getTitle()
{
if (this.hasTitle()) return this.title;
return Lang.PLAYER_NOTITLE;
}
2011-10-08 23:22:02 +02:00
2013-04-17 15:30:21 +02:00
public void setTitle(String title)
{
if (title != null)
{
title = title.trim();
if (title.length() == 0)
{
title = null;
}
}
this.title = title;
this.changed();
}
2011-10-12 17:25:01 +02:00
2011-03-22 17:20:21 +01:00
// -------------------------------------------- //
2013-04-17 15:30:21 +02:00
// FIELD: powerBoost
2011-03-22 17:20:21 +01:00
// -------------------------------------------- //
2013-04-17 15:30:21 +02:00
public double getPowerBoost()
{
2013-04-17 15:30:21 +02:00
Double ret = this.powerBoost;
if (ret == null) ret = 0D;
return ret;
2011-03-19 13:00:03 +01:00
}
2013-04-17 15:30:21 +02:00
public void setPowerBoost(Double powerBoost)
{
2013-04-17 15:30:21 +02:00
if (powerBoost == null || powerBoost == 0)
2011-10-10 01:21:05 +02:00
{
2013-04-17 15:30:21 +02:00
powerBoost = null;
2011-10-10 01:21:05 +02:00
}
2013-04-17 15:30:21 +02:00
this.powerBoost = powerBoost;
this.changed();
2011-10-10 01:21:05 +02:00
}
2013-04-17 15:30:21 +02:00
public boolean hasPowerBoost()
2011-10-10 01:21:05 +02:00
{
2013-04-17 15:30:21 +02:00
return this.getPowerBoost() != 0D;
2011-03-18 17:33:23 +01:00
}
2011-03-22 17:20:21 +01:00
// -------------------------------------------- //
2013-04-18 10:35:40 +02:00
// FIELD: lastPowerUpdateTime
// -------------------------------------------- //
// RAW
public long getLastPowerUpdateTime()
{
return this.lastPowerUpdateTime;
}
public void setLastPowerUpdateTime(long lastPowerUpdateTime)
{
this.lastPowerUpdateTime = lastPowerUpdateTime;
this.changed();
}
// -------------------------------------------- //
// FIELD: power
2011-03-22 17:20:21 +01:00
// -------------------------------------------- //
2013-04-18 10:35:40 +02:00
// RAW
public double getPower()
{
this.recalculatePower();
2013-04-18 10:35:40 +02:00
return this.power;
}
public void setPower(double power)
{
this.setPower(power, System.currentTimeMillis());
}
public void setPower(double power, long now)
{
power = Math.min(power, this.getPowerMax());
power = Math.max(power, this.getPowerMin());
// Nochange
if (this.power == power) return;
this.power = power;
this.setLastPowerUpdateTime(now);
this.changed();
}
public double getPowerMax()
{
return ConfServer.powerMax + this.getPowerBoost();
2013-04-18 10:35:40 +02:00
}
public double getPowerMin()
{
return ConfServer.powerMin + this.getPowerBoost();
2013-04-18 10:35:40 +02:00
}
public void recalculatePower()
2013-04-18 10:35:40 +02:00
{
this.recalculatePower(this.isOnline());
2013-04-18 10:35:40 +02:00
}
private static final transient long POWER_RECALCULATION_MINIMUM_WAIT_MILLIS = 10 * TimeUnit.MILLIS_PER_SECOND;
public void recalculatePower(boolean online)
2013-04-18 10:35:40 +02:00
{
// Is the player really on this server?
// We use the sender ps mixin to fetch the current player location.
// If the PS is null it's OK. We assume the player is here if we do not know.
PS ps = Mixin.getSenderPs(this.getId());
if (ps != null && !ps.isWorldLoadedOnThisServer()) return;
// Get the now
long now = System.currentTimeMillis();
// We will only update if a certain amount of time has passed.
if (this.getLastPowerUpdateTime() + POWER_RECALCULATION_MINIMUM_WAIT_MILLIS >= now) return;
// Calculate millis passed
long millisPassed = now - this.getLastPowerUpdateTime();
// Note that we updated
this.setLastPowerUpdateTime(now);
// We consider dead players to be offline.
if (online)
{
Player thisPlayer = this.getPlayer();
if (thisPlayer != null && thisPlayer.isDead())
{
online = false;
}
}
// Depending on online state pick the config values
double powerPerHour = online ? ConfServer.powerPerHourOnline : ConfServer.powerPerHourOffline;
double powerLimitGain = online ? ConfServer.powerLimitGainOnline : ConfServer.powerLimitGainOffline;
double powerLimitLoss = online ? ConfServer.powerLimitLossOnline : ConfServer.powerLimitLossOffline;
// Apply the negative divisor thingy
if (ConfServer.scaleNegativePower && this.power < 0)
{
powerPerHour += (Math.sqrt(Math.abs(this.power)) * Math.abs(this.power)) / ConfServer.scaleNegativeDivisor;
}
// Calculate delta and target
double powerDelta = powerPerHour * millisPassed / TimeUnit.MILLIS_PER_HOUR;
double powerTarget = this.power + powerDelta;
// Check Gain and Loss limits
if (powerDelta >= 0)
{
// Gain
if (powerTarget > powerLimitGain)
{
if (this.power > powerLimitGain)
{
// Did already cross --> Just freeze
powerTarget = this.power;
}
else
{
// Crossing right now --> Snap to limit
powerTarget = powerLimitGain;
}
}
}
else
{
// Loss
if (powerTarget < powerLimitLoss)
{
if (this.power < powerLimitLoss)
{
// Did already cross --> Just freeze
powerTarget = this.power;
}
else
{
// Crossing right now --> Snap to limit
powerTarget = powerLimitLoss;
}
}
}
this.setPower(powerTarget, now);
}
// FINER
public int getPowerRounded()
{
return (int) Math.round(this.getPower());
}
public int getPowerMaxRounded()
{
return (int) Math.round(this.getPowerMax());
}
public int getPowerMinRounded()
{
2013-04-18 10:35:40 +02:00
return (int) Math.round(this.getPowerMin());
2011-03-22 20:36:33 +01:00
}
2013-04-18 10:35:40 +02:00
// -------------------------------------------- //
// TITLE, NAME, FACTION TAG AND CHAT
// -------------------------------------------- //
public String getName()
{
return this.getFixedId();
}
public String getTag()
{
if ( ! this.hasFaction())
{
return "";
}
return this.getFaction().getTag();
}
// Base concatenations:
public String getNameAndSomething(String something)
{
String ret = this.role.getPrefix();
2013-04-17 15:30:21 +02:00
if (something.length() > 0)
{
ret += something+" ";
}
ret += this.getName();
return ret;
}
public String getNameAndTitle()
{
2013-04-17 15:30:21 +02:00
if (this.hasTitle())
{
return this.getNameAndSomething(this.getTitle());
}
else
{
return this.getName();
}
}
public String getNameAndTag()
{
return this.getNameAndSomething(this.getTag());
}
// Colored concatenations:
// These are used in information messages
public String getNameAndTitle(Faction faction)
{
return this.getColorTo(faction)+this.getNameAndTitle();
}
public String getNameAndTitle(FPlayer fplayer)
{
return this.getColorTo(fplayer)+this.getNameAndTitle();
}
// Chat Tag:
// These are injected into the format of global chat messages.
public String getChatTag()
{
2011-03-19 13:00:03 +01:00
if ( ! this.hasFaction()) {
return "";
}
2013-04-09 13:15:25 +02:00
return String.format(ConfServer.chatTagFormat, this.role.getPrefix()+this.getTag());
}
// Colored Chat Tag
public String getChatTag(Faction faction)
{
2011-03-19 13:00:03 +01:00
if ( ! this.hasFaction()) {
return "";
}
2011-10-12 17:25:01 +02:00
return this.getRelationTo(faction).getColor()+getChatTag();
}
public String getChatTag(FPlayer fplayer)
{
if ( ! this.hasFaction())
{
return "";
}
return this.getColorTo(fplayer)+getChatTag();
}
// -------------------------------------------- //
// RELATION AND RELATION COLORS
// -------------------------------------------- //
2011-10-12 17:25:01 +02:00
@Override
2011-10-24 11:07:06 +02:00
public String describeTo(RelationParticipator observer, boolean ucfirst)
{
2011-10-24 11:07:06 +02:00
return RelationUtil.describeThatToMe(this, observer, ucfirst);
}
2011-10-12 17:25:01 +02:00
@Override
2011-10-24 11:07:06 +02:00
public String describeTo(RelationParticipator observer)
{
2011-10-24 11:07:06 +02:00
return RelationUtil.describeThatToMe(this, observer);
}
2011-10-12 17:25:01 +02:00
@Override
2011-10-24 11:07:06 +02:00
public Rel getRelationTo(RelationParticipator observer)
{
2011-10-24 11:07:06 +02:00
return RelationUtil.getRelationOfThatToMe(this, observer);
}
2011-10-12 17:25:01 +02:00
@Override
2011-10-24 11:07:06 +02:00
public Rel getRelationTo(RelationParticipator observer, boolean ignorePeaceful)
{
2011-10-24 11:07:06 +02:00
return RelationUtil.getRelationOfThatToMe(this, observer, ignorePeaceful);
}
public Rel getRelationToLocation()
{
// TODO: Use some built in system to get sender
return BoardColl.get().getFactionAt(PS.valueOf(this.getPlayer())).getRelationTo(this);
}
2011-10-12 17:25:01 +02:00
@Override
2011-10-24 11:07:06 +02:00
public ChatColor getColorTo(RelationParticipator observer)
2011-10-12 17:25:01 +02:00
{
2011-10-24 11:07:06 +02:00
return RelationUtil.getColorOfThatToMe(this, observer);
2011-10-12 17:25:01 +02:00
}
// -------------------------------------------- //
// HEALTH
// -------------------------------------------- //
public void heal(int amnt)
{
2011-02-06 13:36:11 +01:00
Player player = this.getPlayer();
if (player == null)
{
2011-02-06 13:36:11 +01:00
return;
}
player.setHealth(player.getHealth() + amnt);
}
// -------------------------------------------- //
// TERRITORY
// -------------------------------------------- //
public boolean isInOwnTerritory()
{
// TODO: Use Mixin to get this PS instead
return BoardColl.get().getFactionAt(PS.valueOf(this.getPlayer())) == this.getFaction();
2011-02-06 13:36:11 +01:00
}
public boolean isInEnemyTerritory()
{
// TODO: Use Mixin to get this PS instead
return BoardColl.get().getFactionAt(PS.valueOf(this.getPlayer())).getRelationTo(this) == Rel.ENEMY;
2011-02-06 13:36:11 +01:00
}
public void sendFactionHereMessage()
{
if (SpoutFeatures.updateTerritoryDisplay(this))
{
return;
}
Faction factionHere = BoardColl.get().getFactionAt(this.getCurrentChunk());
2013-04-10 10:32:04 +02:00
String msg = Txt.parse("<i>")+" ~ "+factionHere.getTag(this);
if (factionHere.hasDescription())
{
msg += " - "+factionHere.getDescription();
}
2011-02-06 13:36:11 +01:00
this.sendMessage(msg);
}
// -------------------------------------------- //
// ACTIONS
// -------------------------------------------- //
2011-03-22 20:36:33 +01:00
public void leave(boolean makePay)
{
2011-03-22 20:36:33 +01:00
Faction myFaction = this.getFaction();
2013-04-19 14:08:45 +02:00
boolean permanent = myFaction.getFlag(FFlag.PERMANENT);
2011-03-22 20:36:33 +01:00
2013-04-19 14:08:45 +02:00
if (!permanent && this.getRole() == Rel.LEADER && myFaction.getFPlayers().size() > 1)
{
2011-10-10 13:40:24 +02:00
msg("<b>You must give the admin role to someone else first.");
2011-03-22 20:36:33 +01:00
return;
}
2013-04-09 13:15:25 +02:00
if (!ConfServer.canLeaveWithNegativePower && this.getPower() < 0)
{
2011-10-10 13:40:24 +02:00
msg("<b>You cannot leave until your power is positive.");
return;
}
2013-04-19 14:08:45 +02:00
// Event
FactionsEventMembershipChange membershipChangeEvent = new FactionsEventMembershipChange(sender, this, myFaction, MembershipChangeReason.LEAVE);
membershipChangeEvent.run();
if (membershipChangeEvent.isCancelled()) return;
// Am I the last one in the faction?
if (myFaction.getFPlayers().size() == 1)
{
// Transfer all money
2013-04-19 09:50:33 +02:00
if (Econ.isEnabled())
Econ.transferMoney(this, myFaction, this, Econ.getBalance(myFaction.getAccountId()));
}
if (myFaction.isNormal())
{
for (FPlayer fplayer : myFaction.getFPlayersWhereOnline(true))
{
fplayer.msg("%s<i> left %s<i>.", this.describeTo(fplayer, true), myFaction.describeTo(fplayer));
}
2013-04-09 13:15:25 +02:00
if (ConfServer.logFactionLeave)
Factions.get().log(this.getName()+" left the faction: "+myFaction.getTag());
}
this.resetFactionData();
2013-04-19 14:08:45 +02:00
if (myFaction.isNormal() && !permanent && myFaction.getFPlayers().isEmpty())
{
2011-03-22 20:36:33 +01:00
// Remove this faction
2013-04-12 08:56:26 +02:00
for (FPlayer fplayer : FPlayerColl.get().getAllOnline())
{
fplayer.msg("<i>%s<i> was disbanded.", myFaction.describeTo(fplayer, true));
2011-03-22 20:36:33 +01:00
}
2011-10-10 01:21:05 +02:00
myFaction.detach();
2013-04-09 13:15:25 +02:00
if (ConfServer.logFactionDisband)
Factions.get().log("The faction "+myFaction.getTag()+" ("+myFaction.getId()+") was disbanded due to the last player ("+this.getName()+") leaving.");
2011-03-22 20:36:33 +01:00
}
}
2011-10-22 16:00:24 +02:00
public boolean canClaimForFactionAtLocation(Faction forFaction, Location location, boolean notifyFailure)
{
2011-10-22 16:00:24 +02:00
String error = null;
PS ps = PS.valueOf(location);
Faction myFaction = this.getFaction();
Faction currentFaction = BoardColl.get().getFactionAt(ps);
int ownedLand = forFaction.getLandCount();
2011-10-22 16:00:24 +02:00
2013-04-09 13:15:25 +02:00
if (ConfServer.worldGuardChecking && Worldguard.checkForRegionsInChunk(location))
{
// Checks for WorldGuard regions in the chunk attempting to be claimed
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>This land is protected");
}
else if (ConfServer.worldsNoClaiming.contains(ps.getWorld()))
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>Sorry, this world has land claiming disabled.");
}
2013-04-16 11:27:03 +02:00
else if (this.isUsingAdminMode())
{
2011-10-22 16:00:24 +02:00
return true;
}
2011-10-22 16:00:24 +02:00
else if (forFaction == currentFaction)
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("%s<i> already own this land.", forFaction.describeTo(this, true));
}
else if ( ! FPerm.TERRITORY.has(this, forFaction, true))
{
return false;
}
2013-04-09 13:15:25 +02:00
else if (forFaction.getFPlayers().size() < ConfServer.claimsRequireMinFactionMembers)
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("Factions must have at least <h>%s<b> members to claim land.", ConfServer.claimsRequireMinFactionMembers);
}
2011-10-22 16:00:24 +02:00
else if (ownedLand >= forFaction.getPowerRounded())
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>You can't claim more land! You need more power!");
2011-10-22 16:00:24 +02:00
}
2013-04-09 13:15:25 +02:00
else if (ConfServer.claimedLandsMax != 0 && ownedLand >= ConfServer.claimedLandsMax && ! forFaction.getFlag(FFlag.INFPOWER))
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>Limit reached. You can't claim more land!");
}
2013-04-09 13:15:25 +02:00
else if ( ! ConfServer.claimingFromOthersAllowed && currentFaction.isNormal())
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>You may not claim land from others.");
}
2011-10-25 22:18:54 +02:00
else if (currentFaction.getRelationTo(forFaction).isAtLeast(Rel.TRUCE) && ! currentFaction.isNone())
2011-10-22 16:00:24 +02:00
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>You can't claim this land due to your relation with the current owner.");
2011-10-22 16:00:24 +02:00
}
else if
(
2013-04-09 13:15:25 +02:00
ConfServer.claimsMustBeConnected
2013-04-16 11:27:03 +02:00
&& ! this.isUsingAdminMode()
&& myFaction.getLandCountInWorld(ps.getWorld()) > 0
&& !BoardColl.get().isConnectedPs(ps, myFaction)
2013-04-09 13:15:25 +02:00
&& (!ConfServer.claimsCanBeUnconnectedIfOwnedByOtherFaction || !currentFaction.isNormal())
)
{
2013-04-09 13:15:25 +02:00
if (ConfServer.claimsCanBeUnconnectedIfOwnedByOtherFaction)
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>You can only claim additional land which is connected to your first claim or controlled by another faction!");
else
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>You can only claim additional land which is connected to your first claim!");
}
2011-10-22 16:00:24 +02:00
else if (currentFaction.isNormal())
{
2011-10-23 22:08:57 +02:00
if ( ! currentFaction.hasLandInflation())
{
// TODO more messages WARN current faction most importantly
2013-04-10 10:32:04 +02:00
error = Txt.parse("%s<i> owns this land and is strong enough to keep it.", currentFaction.getTag(this));
}
else if ( ! BoardColl.get().isBorderPs(ps))
{
2013-04-10 10:32:04 +02:00
error = Txt.parse("<b>You must start claiming land at the border of the territory.");
}
}
2011-10-22 16:00:24 +02:00
if (notifyFailure && error != null)
{
msg(error);
}
return error == null;
}
public boolean attemptClaim(Faction forFaction, Location location, boolean notifyFailure)
{
// notifyFailure is false if called by auto-claim; no need to notify on every failure for it
// return value is false on failure, true on success
PS flocation = PS.valueOf(location).getChunk(true);
Faction currentFaction = BoardColl.get().getFactionAt(flocation);
2011-10-22 16:00:24 +02:00
int ownedLand = forFaction.getLandCount();
2011-10-22 16:00:24 +02:00
if ( ! this.canClaimForFactionAtLocation(forFaction, location, notifyFailure)) return false;
2011-10-23 22:08:57 +02:00
// TODO: Add flag no costs??
// if economy is enabled and they're not on the bypass list, make sure they can pay
2013-04-19 09:50:33 +02:00
boolean mustPay = Econ.isEnabled() && ! this.isUsingAdminMode();
double cost = 0.0;
EconomyParticipator payee = null;
if (mustPay)
{
cost = Econ.calculateClaimCost(ownedLand, currentFaction.isNormal());
if (ConfServer.econClaimUnconnectedFee != 0.0 && forFaction.getLandCountInWorld(flocation.getWorld()) > 0 && !BoardColl.get().isConnectedPs(flocation, forFaction))
2013-04-09 13:15:25 +02:00
cost += ConfServer.econClaimUnconnectedFee;
2013-04-09 13:15:25 +02:00
if(ConfServer.bankEnabled && ConfServer.bankFactionPaysLandCosts && this.hasFaction())
payee = this.getFaction();
else
payee = this;
if ( ! Econ.hasAtLeast(payee, cost, "to claim this land")) return false;
}
// Event
FactionsEventLandClaim event = new FactionsEventLandClaim(sender, forFaction, flocation);
event.run();
if (event.isCancelled()) return false;
// then make 'em pay (if applicable)
// TODO: The economy integration should cancel the event above!
2013-04-19 09:50:33 +02:00
if (mustPay && ! Econ.modifyMoney(payee, -cost, "claim this land")) return false;
// TODO: The LWC integration should listen to Monitor for the claim event.
2013-04-09 13:15:25 +02:00
if (LWCFeatures.getEnabled() && forFaction.isNormal() && ConfServer.onCaptureResetLwcLocks)
{
2013-04-18 21:03:38 +02:00
LWCFeatures.clearOtherProtections(flocation, this.getFaction());
}
// announce success
2011-10-22 16:00:24 +02:00
Set<FPlayer> informTheseFPlayers = new HashSet<FPlayer>();
informTheseFPlayers.add(this);
informTheseFPlayers.addAll(forFaction.getFPlayersWhereOnline(true));
for (FPlayer fp : informTheseFPlayers)
{
2011-10-22 16:00:24 +02:00
fp.msg("<h>%s<i> claimed land for <h>%s<i> from <h>%s<i>.", this.describeTo(fp, true), forFaction.describeTo(fp), currentFaction.describeTo(fp));
}
2011-10-22 16:00:24 +02:00
BoardColl.get().setFactionAt(flocation, forFaction);
SpoutFeatures.updateTerritoryDisplayLoc(flocation);
2013-04-09 13:15:25 +02:00
if (ConfServer.logLandClaims)
Factions.get().log(this.getName()+" claimed land at ("+flocation.getChunkX()+","+flocation.getChunkZ()+") for the faction: "+forFaction.getTag());
return true;
}
2011-03-19 13:00:03 +01:00
}