Factions3/src/com/massivecraft/factions/entity/UPlayer.java

739 lines
19 KiB
Java
Raw Normal View History

2013-04-22 09:37:53 +02:00
package com.massivecraft.factions.entity;
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.entity.Player;
2011-07-18 22:06:02 +02:00
2013-04-22 09:37:53 +02:00
import com.massivecraft.factions.EconomyParticipator;
import com.massivecraft.factions.FFlag;
import com.massivecraft.factions.Factions;
import com.massivecraft.factions.Lang;
import com.massivecraft.factions.Rel;
import com.massivecraft.factions.RelationParticipator;
import com.massivecraft.factions.event.FactionsEventChunkChange;
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.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;
import com.massivecraft.mcore.util.MUtil;
import com.massivecraft.mcore.util.SenderUtil;
2013-04-25 12:04:01 +02:00
import com.massivecraft.mcore.util.Txt;
2011-02-06 13:36:11 +01:00
public class UPlayer extends SenderEntity<UPlayer> implements EconomyParticipator
{
// -------------------------------------------- //
2013-04-12 08:56:26 +02:00
// META
// -------------------------------------------- //
public static UPlayer get(Object oid)
{
return UPlayerColls.get().get2(oid);
}
// -------------------------------------------- //
// OVERRIDE: ENTITY
// -------------------------------------------- //
2013-04-12 08:56:26 +02:00
@Override
public UPlayer load(UPlayer 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);
2013-04-23 12:14:36 +02:00
this.setPower(that.power);
return this;
2013-04-12 08:56:26 +02:00
}
@Override
public boolean isDefault()
{
2013-04-23 12:14:36 +02:00
if (this.hasFaction()) return false;
// Role means nothing without a faction.
// Title means nothing without a faction.
2013-04-23 14:00:18 +02:00
if (this.getPowerRounded() != (int) Math.round(UConf.get(this).defaultPlayerPower)) return false;
2013-04-17 15:30:21 +02:00
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.
2013-04-23 12:14:36 +02:00
// Each player belong to a faction.
2013-04-24 14:31:10 +02:00
// Null means default for the universe.
private String factionId = null;
2013-04-17 15:30:21 +02:00
// What role does the player have in the faction?
2013-04-24 14:31:10 +02:00
// Null means default for the universe.
2013-04-17 15:30:21 +02:00
private Rel role = null;
// What title does the player have in the faction?
2013-04-23 12:14:36 +02:00
// The title is just for fun. It's not connected to any game mechanic.
2013-04-17 15:30:21 +02:00
// 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.
2013-04-23 12:14:36 +02:00
// If the title contains raw markup, such as "<white>" instead of "§f" it will not be parsed and "<white>" will be displayed.
2013-04-24 14:31:10 +02:00
//
// Null means the player has no title.
2013-04-17 15:30:21 +02:00
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-23 12:14:36 +02:00
// Each player has an individual power level.
// The power level for online players is occasionally updated by a recurring task and the power should stay the same for offline players.
// For that reason the value is to be considered correct when you pick it. Do not call the power update method.
2013-04-24 14:31:10 +02:00
// Null means default for the universe.
private Double power = null;
2013-04-17 15:30:21 +02:00
2013-04-24 15:14:15 +02:00
// The id for the faction this uplayer is currently autoclaiming for.
// NOTE: This field will not be saved to the database ever.
// Null means the player isn't auto claiming.
private transient Faction autoClaimFaction = null;
public Faction getAutoClaimFaction() { return this.autoClaimFaction; }
public void setAutoClaimFaction(Faction autoClaimFaction) { this.autoClaimFaction = autoClaimFaction; }
2013-04-17 15:30:21 +02:00
// -------------------------------------------- //
2013-04-24 15:14:15 +02:00
// FIELDS: MULTIVERSE PROXY
2013-04-17 15:30:21 +02:00
// -------------------------------------------- //
2013-04-24 15:14:15 +02:00
public boolean isMapAutoUpdating() { return MPlayer.get(this).isMapAutoUpdating(); }
public void setMapAutoUpdating(boolean mapAutoUpdating) { MPlayer.get(this).setMapAutoUpdating(mapAutoUpdating); }
2013-04-17 15:30:21 +02:00
2013-04-24 15:14:15 +02:00
public boolean isUsingAdminMode() { return MPlayer.get(this).isUsingAdminMode(); }
public void setUsingAdminMode(boolean usingAdminMode) { MPlayer.get(this).setUsingAdminMode(usingAdminMode); }
2013-04-17 15:30:21 +02:00
// -------------------------------------------- //
2013-04-23 12:14:36 +02:00
// CORE UTILITIES
2013-04-17 15:30:21 +02:00
// -------------------------------------------- //
public void resetFactionData()
2013-04-17 15:30:21 +02:00
{
// The default neutral faction
this.setFactionId(null);
this.setRole(null);
this.setTitle(null);
2013-04-24 15:14:15 +02:00
this.setAutoClaimFaction(null);
2013-04-17 15:30:21 +02:00
}
2013-04-23 12:14:36 +02:00
/*
public boolean isPresent(boolean requireFetchable)
{
if (!this.isOnline()) return false;
if (requireFetchable)
{
}
else
{
}
PS ps = Mixin.getSenderPs(this.getId());
if (ps == null) return false;
String psUniverse = Factions.get().getMultiverse().getUniverseForWorldName(ps.getWorld());
if (!psUniverse.equals(this.getUniverse())) return false;
if (!requireFetchable) return true;
Player player = this.getPlayer();
if (player == null) return false;
if (player.isDead()) return false;
return true;
}
*/
2013-04-17 15:30:21 +02:00
// -------------------------------------------- //
// FIELD: factionId
// -------------------------------------------- //
2013-04-23 14:00:18 +02:00
public String getDefaultFactionId()
{
return UConf.get(this).defaultPlayerFactionId;
}
2013-04-17 15:30:21 +02:00
// This method never returns null
public String getFactionId()
2011-10-08 23:22:02 +02:00
{
2013-04-23 14:00:18 +02:00
if (this.factionId == null) return this.getDefaultFactionId();
return this.factionId;
}
2013-04-17 15:30:21 +02:00
// This method never returns null
public Faction getFaction()
{
Faction ret = FactionColls.get().get(this).get(this.getFactionId());
2013-04-23 14:00:18 +02:00
if (ret == null) ret = FactionColls.get().get(this).get(UConf.get(this).defaultPlayerFactionId);
return ret;
}
public boolean hasFaction()
{
return !this.getFactionId().equals(UConf.get(this).factionIdNone);
}
// 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)
{
// Clean input
String target = factionId;
// Detect Nochange
if (MUtil.equals(this.factionId, target)) return;
// Get the raw old value
String oldFactionId = this.factionId;
// Apply
this.factionId = target;
// Must be attached and initialized
if (!this.attached()) return;
2013-04-22 15:05:00 +02:00
if (!Factions.get().isDatabaseInitialized()) return;
if (oldFactionId == null) oldFactionId = this.getDefaultFactionId();
// Update index
Faction oldFaction = FactionColls.get().get(this).get(oldFactionId);
2013-04-25 12:04:01 +02:00
Faction faction = this.getFaction();
2013-04-25 12:04:01 +02:00
if (oldFaction != null) oldFaction.uplayers.remove(this);
if (faction != null) faction.uplayers.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-23 14:00:18 +02:00
public Rel getDefaultRole()
{
return UConf.get(this).defaultPlayerRole;
}
2013-04-17 15:30:21 +02:00
public Rel getRole()
{
2013-04-23 14:00:18 +02:00
if (this.role == null) return this.getDefaultRole();
2013-04-17 15:30:21 +02:00
return this.role;
}
2011-10-08 23:22:02 +02:00
2013-04-17 15:30:21 +02:00
public void setRole(Rel role)
{
// Clean input
Rel target = role;
// Detect Nochange
if (MUtil.equals(this.role, target)) return;
// Apply
this.role = target;
// Mark as changed
2013-04-17 15:30:21 +02:00
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)
{
// Clean input
String target = title;
if (target != null)
2013-04-17 15:30:21 +02:00
{
target = target.trim();
if (target.length() == 0)
2013-04-17 15:30:21 +02:00
{
target = null;
2013-04-17 15:30:21 +02:00
}
}
2013-04-25 16:54:55 +02:00
// NOTE: That we parse the title here is considered part of the 1.8 --> 2.0 migration.
// This should be removed once the migration phase is considered to be over.
if (target != null)
{
target = Txt.parse(target);
}
// Detect Nochange
if (MUtil.equals(this.title, target)) return;
// Apply
this.title = target;
// Mark as changed
2013-04-17 15:30:21 +02:00
this.changed();
}
2011-10-12 17:25:01 +02:00
// -------------------------------------------- //
// FIELD: powerBoost
// -------------------------------------------- //
public double getPowerBoost()
{
Double ret = this.powerBoost;
if (ret == null) ret = 0D;
return ret;
}
public void setPowerBoost(Double powerBoost)
{
// Clean input
Double target = powerBoost;
if (target == null || target == 0) target = null;
// Detect Nochange
if (MUtil.equals(this.powerBoost, target)) return;
// Apply
this.powerBoost = target;
// Mark as changed
this.changed();
}
public boolean hasPowerBoost()
{
return this.getPowerBoost() != 0D;
}
2013-04-18 10:35:40 +02:00
// -------------------------------------------- //
// FIELD: power
2011-03-22 17:20:21 +01:00
// -------------------------------------------- //
// MIXIN: RAW
public double getPowerMaxUniversal()
{
return Factions.get().getPowerMixin().getMaxUniversal(this);
}
public double getPowerMax()
{
return Factions.get().getPowerMixin().getMax(this);
}
public double getPowerMin()
{
return Factions.get().getPowerMixin().getMin(this);
}
public double getPowerPerHour()
{
return Factions.get().getPowerMixin().getPerHour(this);
}
public double getPowerPerDeath()
{
return Factions.get().getPowerMixin().getPerDeath(this);
}
// MIXIN: FINER
public double getLimitedPower(double power)
{
power = Math.max(power, this.getPowerMin());
power = Math.min(power, this.getPowerMax());
return power;
}
public int getPowerMaxRounded()
{
return (int) Math.round(this.getPowerMax());
}
public int getPowerMinRounded()
{
return (int) Math.round(this.getPowerMin());
}
public int getPowerMaxUniversalRounded()
{
return (int) Math.round(this.getPowerMaxUniversal());
}
2013-04-18 10:35:40 +02:00
// RAW
2013-04-23 14:00:18 +02:00
public double getDefaultPower()
{
return UConf.get(this).defaultPlayerPower;
}
2013-04-18 10:35:40 +02:00
public double getPower()
{
Double ret = this.power;
2013-04-23 14:00:18 +02:00
if (ret == null) ret = this.getDefaultPower();
ret = this.getLimitedPower(ret);
return ret;
2013-04-18 10:35:40 +02:00
}
2013-04-23 12:14:36 +02:00
public void setPower(Double power)
2013-04-18 10:35:40 +02:00
{
// Clean input
Double target = power;
// Detect Nochange
if (MUtil.equals(this.power, target)) return;
// Apply
this.power = target;
// Mark as changed
2013-04-23 12:14:36 +02:00
this.changed();
2013-04-18 10:35:40 +02:00
}
// FINER
public int getPowerRounded()
{
return (int) Math.round(this.getPower());
}
// -------------------------------------------- //
// TITLE, NAME, FACTION NAME AND CHAT
// -------------------------------------------- //
public String getName()
{
return this.getFixedId();
}
public String getFactionName()
{
2013-04-23 14:00:18 +02:00
Faction faction = this.getFaction();
if (faction.isNone()) return "";
return faction.getName();
}
// Base concatenations:
2013-04-25 16:54:55 +02:00
public String getNameAndSomething(String color, String something)
{
2013-04-25 16:54:55 +02:00
String ret = "";
ret += color;
ret += this.getRole().getPrefix();
2013-04-25 16:02:37 +02:00
if (something != null && something.length() > 0)
2013-04-17 15:30:21 +02:00
{
2013-04-25 16:54:55 +02:00
ret += something;
ret += " ";
ret += color;
}
ret += this.getName();
return ret;
}
2013-04-25 16:54:55 +02:00
public String getNameAndFactionName()
{
return this.getNameAndSomething("", this.getFactionName());
}
public String getNameAndTitle(String color)
{
2013-04-17 15:30:21 +02:00
if (this.hasTitle())
{
2013-04-25 16:54:55 +02:00
return this.getNameAndSomething(color, this.getTitle());
2013-04-17 15:30:21 +02:00
}
else
{
2013-04-25 16:54:55 +02:00
return this.getNameAndSomething(color, null);
2013-04-17 15:30:21 +02:00
}
}
// Colored concatenations:
// These are used in information messages
public String getNameAndTitle(Faction faction)
{
2013-04-25 16:54:55 +02:00
return this.getNameAndTitle(this.getColorTo(faction).toString());
}
public String getNameAndTitle(UPlayer uplayer)
{
2013-04-25 16:54:55 +02:00
return this.getNameAndTitle(this.getColorTo(uplayer).toString());
}
// -------------------------------------------- //
// 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);
}
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 BoardColls.get().getFactionAt(Mixin.getSenderPs(this.getId())) == this.getFaction();
2011-02-06 13:36:11 +01:00
}
public boolean isInEnemyTerritory()
{
// TODO: Use Mixin to get this PS instead
return BoardColls.get().getFactionAt(Mixin.getSenderPs(this.getId())).getRelationTo(this) == Rel.ENEMY;
2011-02-06 13:36:11 +01:00
}
// -------------------------------------------- //
// ACTIONS
// -------------------------------------------- //
2011-03-22 20:36:33 +01:00
public void leave()
{
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
if (myFaction.getUPlayers().size() > 1)
{
if (!permanent && this.getRole() == Rel.LEADER)
{
msg("<b>You must give the leader role to someone else first.");
return;
}
if (!UConf.get(myFaction).canLeaveWithNegativePower && this.getPower() < 0)
{
msg("<b>You cannot leave until your power is positive.");
return;
}
}
2013-04-19 14:08:45 +02:00
// Event
2013-04-25 16:02:37 +02:00
FactionsEventMembershipChange membershipChangeEvent = new FactionsEventMembershipChange(this.getSender(), this, myFaction, MembershipChangeReason.LEAVE);
2013-04-19 14:08:45 +02:00
membershipChangeEvent.run();
if (membershipChangeEvent.isCancelled()) return;
if (myFaction.isNormal())
{
for (UPlayer uplayer : myFaction.getUPlayersWhereOnline(true))
{
uplayer.msg("%s<i> left %s<i>.", this.describeTo(uplayer, true), myFaction.describeTo(uplayer));
}
if (MConf.get().logFactionLeave)
{
Factions.get().log(this.getName()+" left the faction: "+myFaction.getName());
}
}
this.resetFactionData();
if (myFaction.isNormal() && !permanent && myFaction.getUPlayers().isEmpty())
{
2011-03-22 20:36:33 +01:00
// Remove this faction
for (UPlayer uplayer : UPlayerColls.get().get(this).getAllOnline())
{
uplayer.msg("<i>%s<i> was disbanded.", myFaction.describeTo(uplayer, true));
2011-03-22 20:36:33 +01:00
}
2011-10-10 01:21:05 +02:00
myFaction.detach();
if (MConf.get().logFactionDisband)
{
Factions.get().log("The faction "+myFaction.getName()+" ("+myFaction.getId()+") was disbanded due to the last player ("+this.getName()+") leaving.");
}
2011-03-22 20:36:33 +01:00
}
}
public boolean tryClaim(Faction newFaction, PS ps, boolean verbooseChange, boolean verbooseSame)
{
PS chunk = ps.getChunk(true);
Faction oldFaction = BoardColls.get().getFactionAt(chunk);
UConf uconf = UConf.get(newFaction);
MConf mconf = MConf.get();
2011-10-22 16:00:24 +02:00
// Validate
if (newFaction == oldFaction)
{
msg("%s<i> already owns this land.", newFaction.describeTo(this, true));
2011-10-22 16:00:24 +02:00
return true;
}
if (!this.isUsingAdminMode() && newFaction.isNormal())
{
int ownedLand = newFaction.getLandCount();
if (mconf.worldsNoClaiming.contains(ps.getWorld()))
{
msg("<b>Sorry, this world has land claiming disabled.");
return false;
}
if (newFaction.getUPlayers().size() < uconf.claimsRequireMinFactionMembers)
{
msg("Factions must have at least <h>%s<b> members to claim land.", uconf.claimsRequireMinFactionMembers);
return false;
}
if (uconf.claimedLandsMax != 0 && ownedLand >= uconf.claimedLandsMax && ! newFaction.getFlag(FFlag.INFPOWER))
{
msg("<b>Limit reached. You can't claim more land.");
return false;
}
if (ownedLand >= newFaction.getPowerRounded())
{
msg("<b>You can't claim more land. You need more power.");
return false;
}
2013-04-25 12:04:01 +02:00
if (oldFaction.isNormal())
{
2013-04-25 12:04:01 +02:00
if (!uconf.claimingFromOthersAllowed)
{
2013-04-25 12:04:01 +02:00
msg("<b>You may not claim land from others.");
return false;
}
2013-04-25 12:04:01 +02:00
if (oldFaction.getRelationTo(newFaction).isAtLeast(Rel.TRUCE))
{
2013-04-25 12:04:01 +02:00
msg("<b>You can't claim this land due to your relation with the current owner.");
return false;
}
if
(
uconf.claimsMustBeConnected
&& newFaction.getLandCountInWorld(ps.getWorld()) > 0
&& !BoardColls.get().isConnectedPs(ps, newFaction)
&& (!uconf.claimsCanBeUnconnectedIfOwnedByOtherFaction || !oldFaction.isNormal())
)
{
if (uconf.claimsCanBeUnconnectedIfOwnedByOtherFaction)
{
msg("<b>You can only claim additional land which is connected to your first claim or controlled by another faction!");
}
else
{
msg("<b>You can only claim additional land which is connected to your first claim!");
}
return false;
}
if (!oldFaction.hasLandInflation())
{
msg("%s<i> owns this land and is strong enough to keep it.", oldFaction.getName(this));
return false;
}
if ( ! BoardColls.get().isBorderPs(ps))
{
msg("<b>You must start claiming land at the border of the territory.");
return false;
}
}
}
2011-10-22 16:00:24 +02:00
// Event
FactionsEventChunkChange event = new FactionsEventChunkChange(sender, chunk, newFaction);
event.run();
if (event.isCancelled()) return false;
// Apply
BoardColls.get().setFactionAt(chunk, newFaction);
// Inform
Set<UPlayer> informees = new HashSet<UPlayer>();
informees.add(this);
if (newFaction.isNormal())
{
informees.addAll(newFaction.getUPlayers());
}
if (oldFaction.isNormal())
{
informees.addAll(oldFaction.getUPlayers());
}
if (MConf.get().logLandClaims)
{
informees.add(UPlayer.get(SenderUtil.getConsole()));
}
for (UPlayer informee : informees)
{
2013-04-25 12:04:01 +02:00
String chunkString = Txt.parse("<h>%s <h>%d %d", Mixin.getWorldDisplayName(chunk.getWorld()), chunk.getChunkX(), chunk.getChunkZ());
informee.msg("<h>%s<i> did %s %s <i>for <h>%s<i> from <h>%s<i>.", this.describeTo(informee, true), event.getType().toString().toLowerCase(), chunkString, newFaction.describeTo(informee), oldFaction.describeTo(informee));
}
return true;
}
2011-03-19 13:00:03 +01:00
}