1366 lines
40 KiB
Java
1366 lines
40 KiB
Java
package com.massivecraft.factions.entity;
|
|
|
|
|
|
import com.massivecraft.factions.Factions;
|
|
import com.massivecraft.factions.FactionsIndex;
|
|
import com.massivecraft.factions.FactionsParticipator;
|
|
import com.massivecraft.factions.Rel;
|
|
import com.massivecraft.factions.RelationParticipator;
|
|
import com.massivecraft.factions.entity.MPerm.MPermable;
|
|
import com.massivecraft.factions.predicate.PredicateCommandSenderFaction;
|
|
import com.massivecraft.factions.predicate.PredicateMPlayerRank;
|
|
import com.massivecraft.factions.util.MiscUtil;
|
|
import com.massivecraft.factions.util.RelationUtil;
|
|
import com.massivecraft.massivecore.Couple;
|
|
import com.massivecraft.massivecore.Identified;
|
|
import com.massivecraft.massivecore.collections.MassiveList;
|
|
import com.massivecraft.massivecore.collections.MassiveMap;
|
|
import com.massivecraft.massivecore.collections.MassiveMapDef;
|
|
import com.massivecraft.massivecore.collections.MassiveSet;
|
|
import com.massivecraft.massivecore.mixin.MixinMessage;
|
|
import com.massivecraft.massivecore.money.Money;
|
|
import com.massivecraft.massivecore.predicate.PredicateAnd;
|
|
import com.massivecraft.massivecore.predicate.PredicateVisibleTo;
|
|
import com.massivecraft.massivecore.ps.PS;
|
|
import com.massivecraft.massivecore.store.Entity;
|
|
import com.massivecraft.massivecore.store.EntityInternalMap;
|
|
import com.massivecraft.massivecore.store.SenderColl;
|
|
import com.massivecraft.massivecore.util.IdUtil;
|
|
import com.massivecraft.massivecore.util.MUtil;
|
|
import com.massivecraft.massivecore.util.Txt;
|
|
import org.bukkit.Bukkit;
|
|
import org.bukkit.ChatColor;
|
|
import org.bukkit.command.CommandSender;
|
|
import org.bukkit.entity.Player;
|
|
|
|
import java.util.Arrays;
|
|
import java.util.Collection;
|
|
import java.util.Collections;
|
|
import java.util.Comparator;
|
|
import java.util.Iterator;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Map.Entry;
|
|
import java.util.Optional;
|
|
import java.util.Set;
|
|
import java.util.stream.Collectors;
|
|
|
|
public class Faction extends Entity<Faction> implements FactionsParticipator, MPerm.MPermable {
|
|
// -------------------------------------------- //
|
|
// CONSTANTS
|
|
// -------------------------------------------- //
|
|
|
|
public static final String NODESCRIPTION = Txt.parse("<em><silver>no description set");
|
|
public static final String NOMOTD = Txt.parse("<em><silver>no message of the day set");
|
|
|
|
// -------------------------------------------- //
|
|
// META
|
|
// -------------------------------------------- //
|
|
|
|
public static Faction get(Object oid) {
|
|
return FactionColl.get().get(oid);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// OVERRIDE: ENTITY
|
|
// -------------------------------------------- //
|
|
|
|
@Override
|
|
public Faction load(Faction that) {
|
|
this.setName(that.name);
|
|
this.setDescription(that.description);
|
|
this.setMotd(that.motd);
|
|
this.setCreatedAtMillis(that.createdAtMillis);
|
|
this.warps.load(that.warps);
|
|
this.setPowerBoost(that.powerBoost);
|
|
this.money = that.money;
|
|
this.invitations.load(that.invitations);
|
|
this.ranks.load(that.ranks);
|
|
this.votes.load(that.votes);
|
|
this.setRelationWishes(that.relationWishes);
|
|
this.setFlagIds(that.flags);
|
|
this.perms = that.perms;
|
|
this.tax = that.tax;
|
|
|
|
return this;
|
|
}
|
|
|
|
@Override
|
|
public void preDetach(String id) {
|
|
if (!this.isLive()) {
|
|
return;
|
|
}
|
|
|
|
// NOTE: Existence check is required for compatibility with some plugins.
|
|
// If they have money ...
|
|
if (Money.exists(this)) {
|
|
// ... remove it.
|
|
Money.set(this, null, 0, "Factions");
|
|
}
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// VERSION
|
|
// -------------------------------------------- //
|
|
|
|
public int version = 4;
|
|
|
|
// -------------------------------------------- //
|
|
// FIELDS: RAW
|
|
// -------------------------------------------- //
|
|
// In this section of the source code we place the field declarations only.
|
|
// Each field has its own section further down since just the getter and setter logic takes up quite some place.
|
|
|
|
// The actual faction id looks something like "54947df8-0e9e-4471-a2f9-9af509fb5889" and that is not too easy to
|
|
// remember for humans.
|
|
// Thus, we make use of a name. Since the id is used in all foreign key situations changing the name is fine.
|
|
// Null should never happen. The name must not be null.
|
|
private String name = null;
|
|
|
|
// Factions can optionally set a description for themselves.
|
|
// This description can for example be seen in territorial alerts.
|
|
// Null means the faction has no description.
|
|
private String description = null;
|
|
|
|
// Factions can optionally set a message of the day.
|
|
// This message will be shown when logging on to the server.
|
|
// Null means the faction has no motd
|
|
private String motd = null;
|
|
|
|
// We store the creation date for the faction.
|
|
// It can be displayed on info pages etc.
|
|
private long createdAtMillis = System.currentTimeMillis();
|
|
|
|
// Factions can set a few significant locations (warps)
|
|
private EntityInternalMap<Warp> warps = new EntityInternalMap<>(this, Warp.class);
|
|
|
|
// Factions usually do not have a powerboost. It defaults to 0.
|
|
// The powerBoost is a custom increase/decrease to default and maximum power.
|
|
// Null means the faction has powerBoost (0).
|
|
private Double powerBoost = null;
|
|
|
|
// The money a Faction has
|
|
// null means 0.0
|
|
private Double money = null;
|
|
|
|
// This is the ids of the invited players.
|
|
// They are actually "senderIds" since you can invite "@console" to your faction.
|
|
private EntityInternalMap<Invitation> invitations = new EntityInternalMap<>(this, Invitation.class);
|
|
|
|
// This is where all the ranks are, they are faction specific
|
|
private EntityInternalMap<Rank> ranks = this.createRankMap();
|
|
|
|
// This is the votes currently open in the faction
|
|
private EntityInternalMap<Vote> votes = new EntityInternalMap<>(this, Vote.class);
|
|
|
|
// The keys in this map are factionIds.
|
|
// Null means no special relation whishes.
|
|
private MassiveMapDef<String, Rel> relationWishes = new MassiveMapDef<>();
|
|
|
|
// The flag overrides are modifications to the default values.
|
|
// Null means default.
|
|
private MassiveMapDef<String, Boolean> flags = new MassiveMapDef<>();
|
|
|
|
private Map<String, Set<String>> perms = this.createNewPermMap();
|
|
|
|
// What is the base tax on members of the faction?
|
|
// Specific taxes on ranks or players.
|
|
public static String IDENTIFIER_TAX_BASE = "base";
|
|
|
|
private Map<String, Double> tax = new MassiveMap<>();
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: id
|
|
// -------------------------------------------- //
|
|
|
|
// FINER
|
|
|
|
public boolean isNone() {
|
|
return this.getId().equals(Factions.ID_NONE);
|
|
}
|
|
|
|
public boolean isNormal() {
|
|
return !this.isNone();
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: name
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
@Override
|
|
public String getName() {
|
|
return this.name;
|
|
}
|
|
|
|
public void setName(String name) {
|
|
// Clean input
|
|
String target = name;
|
|
|
|
// Detect Nochange
|
|
if (MUtil.equals(this.name, target)) {
|
|
return;
|
|
}
|
|
|
|
// Apply
|
|
this.name = target;
|
|
|
|
// Mark as changed
|
|
this.changed();
|
|
}
|
|
|
|
// FINER
|
|
|
|
public String getComparisonName() {
|
|
return MiscUtil.getComparisonString(this.getName());
|
|
}
|
|
|
|
public String getName(String prefix) {
|
|
return prefix + this.getName();
|
|
}
|
|
|
|
public String getName(RelationParticipator observer) {
|
|
if (observer == null) {
|
|
return getName();
|
|
}
|
|
return this.getName(this.getColorTo(observer).toString());
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: description
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
public boolean hasDescription() {
|
|
return this.description != null;
|
|
}
|
|
|
|
public String getDescription() {
|
|
return this.description;
|
|
}
|
|
|
|
public void setDescription(String description) {
|
|
// Clean input
|
|
String target = clean(description);
|
|
|
|
// Detect Nochange
|
|
if (MUtil.equals(this.description, target)) {
|
|
return;
|
|
}
|
|
|
|
// Apply
|
|
this.description = target;
|
|
|
|
// Mark as changed
|
|
this.changed();
|
|
}
|
|
|
|
// FINER
|
|
|
|
public String getDescriptionDesc() {
|
|
String motd = this.getDescription();
|
|
if (motd == null) {
|
|
motd = NODESCRIPTION;
|
|
}
|
|
return motd;
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: motd
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
public boolean hasMotd() {
|
|
return this.motd != null;
|
|
}
|
|
|
|
public String getMotd() {
|
|
return this.motd;
|
|
}
|
|
|
|
public void setMotd(String motd) {
|
|
// Clean input
|
|
String target = clean(motd);
|
|
|
|
// Detect Nochange
|
|
if (MUtil.equals(this.motd, target)) {
|
|
return;
|
|
}
|
|
|
|
// Apply
|
|
this.motd = target;
|
|
|
|
// Mark as changed
|
|
this.changed();
|
|
}
|
|
|
|
// FINER
|
|
|
|
public String getMotdDesc() {
|
|
return getMotdDesc(this.getMotd());
|
|
}
|
|
|
|
private static String getMotdDesc(String motd) {
|
|
if (motd == null) {
|
|
motd = NOMOTD;
|
|
}
|
|
return motd;
|
|
}
|
|
|
|
public List<Object> getMotdMessages() {
|
|
// Create
|
|
List<Object> ret = new MassiveList<>();
|
|
|
|
// Fill
|
|
Object title = this.getName() + " - Message of the Day";
|
|
title = Txt.titleize(title);
|
|
ret.add(title);
|
|
|
|
String motd = this.getMotdDesc();
|
|
List<String> motds = Arrays.asList(motd.split("\\\\n"));
|
|
motds = motds.stream().map(s -> Txt.parse("<i>") + s).collect(Collectors.toList());
|
|
ret.addAll(motds);
|
|
|
|
ret.add("");
|
|
|
|
// Return
|
|
return ret;
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: createdAtMillis
|
|
// -------------------------------------------- //
|
|
|
|
public long getCreatedAtMillis() {
|
|
return this.createdAtMillis;
|
|
}
|
|
|
|
public void setCreatedAtMillis(long createdAtMillis) {
|
|
// Detect Nochange
|
|
if (MUtil.equals(this.createdAtMillis, createdAtMillis)) {
|
|
return;
|
|
}
|
|
|
|
// Apply
|
|
this.createdAtMillis = createdAtMillis;
|
|
|
|
// Mark as changed
|
|
this.changed();
|
|
}
|
|
|
|
public long getAge() {
|
|
return this.getAge(System.currentTimeMillis());
|
|
}
|
|
|
|
public long getAge(long now) {
|
|
return now - this.getCreatedAtMillis();
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: warp
|
|
// -------------------------------------------- //
|
|
|
|
public EntityInternalMap<Warp> getWarps() {
|
|
return this.warps;
|
|
}
|
|
|
|
public Warp getWarp(Object oid) {
|
|
if (oid == null) {
|
|
throw new NullPointerException("oid");
|
|
}
|
|
Warp warp = this.getWarps().get(oid);
|
|
if (warp == null) {
|
|
return null;
|
|
}
|
|
|
|
if (!warp.verifyIsValid()) {
|
|
return null;
|
|
}
|
|
return warp;
|
|
}
|
|
|
|
public PS getWarpPS(Object oid) {
|
|
if (oid == null) {
|
|
throw new NullPointerException("oid");
|
|
}
|
|
Warp warp = this.getWarp(oid);
|
|
if (warp == null) {
|
|
return null;
|
|
}
|
|
return warp.getLocation();
|
|
}
|
|
|
|
public String addWarp(Warp warp) {
|
|
return this.getWarps().attach(warp);
|
|
}
|
|
|
|
public Warp removeWarp(Warp warp) {
|
|
return this.getWarps().detachEntity(warp);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: powerBoost
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
@Override
|
|
public double getPowerBoost() {
|
|
Double ret = this.powerBoost;
|
|
if (ret == null) {
|
|
ret = 0D;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
@Override
|
|
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();
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: money
|
|
// -------------------------------------------- //
|
|
|
|
public double getMoney() {
|
|
if (!MConf.get().econEnabled) {
|
|
throw new UnsupportedOperationException("econ not enabled");
|
|
}
|
|
if (!MConf.get().bankEnabled) {
|
|
throw new UnsupportedOperationException("bank not enabled");
|
|
}
|
|
if (!MConf.get().useNewMoneySystem) {
|
|
throw new UnsupportedOperationException("this server does not use the new econ system");
|
|
}
|
|
|
|
return this.convertGet(this.money, 0D);
|
|
}
|
|
|
|
public void setMoney(Double money) {
|
|
if (!MConf.get().econEnabled) {
|
|
throw new UnsupportedOperationException("econ not enabled");
|
|
}
|
|
if (!MConf.get().bankEnabled) {
|
|
throw new UnsupportedOperationException("bank not enabled");
|
|
}
|
|
if (!MConf.get().useNewMoneySystem) {
|
|
throw new UnsupportedOperationException("this server does not use the new econ system");
|
|
}
|
|
|
|
this.money = this.convertSet(money, this.money, 0D);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: invitations
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
public EntityInternalMap<Invitation> getInvitations() {
|
|
return this.invitations;
|
|
}
|
|
|
|
// FINER
|
|
|
|
public boolean isInvited(String playerId) {
|
|
return this.getInvitations().containsKey(playerId);
|
|
}
|
|
|
|
public boolean isInvited(MPlayer mplayer) {
|
|
return this.isInvited(mplayer.getId());
|
|
}
|
|
|
|
public boolean uninvite(String playerId) {
|
|
return this.getInvitations().detachId(playerId) != null;
|
|
}
|
|
|
|
public boolean uninvite(MPlayer mplayer) {
|
|
return uninvite(mplayer.getId());
|
|
}
|
|
|
|
public void invite(String playerId, Invitation invitation) {
|
|
uninvite(playerId);
|
|
this.invitations.attach(invitation, playerId);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: ranks
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
public EntityInternalMap<Rank> getRanks() {
|
|
return this.ranks;
|
|
}
|
|
|
|
// FINER
|
|
|
|
public boolean hasRank(Rank rank) {
|
|
return this.getRanks().containsKey(rank.getId());
|
|
}
|
|
|
|
public Rank getRank(String rankId) {
|
|
if (rankId == null) {
|
|
throw new NullPointerException("rankId");
|
|
}
|
|
return this.getRanks().getFixed(rankId);
|
|
}
|
|
|
|
private EntityInternalMap<Rank> createRankMap() {
|
|
EntityInternalMap<Rank> ret = new EntityInternalMap<>(this, Rank.class);
|
|
|
|
MConf.get().defaultRanks.stream()
|
|
.map(Rank::copy)
|
|
.forEach(ret::attach);
|
|
|
|
return ret;
|
|
}
|
|
|
|
public Rank getLeaderRank() {
|
|
Rank ret = null;
|
|
for (Rank rank : this.getRanks().getAll()) {
|
|
if (ret != null && ret.isMoreThan(rank)) {
|
|
continue;
|
|
}
|
|
|
|
ret = rank;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
public Rank getLowestRank() {
|
|
Rank ret = null;
|
|
for (Rank rank : this.getRanks().getAll()) {
|
|
if (ret != null && ret.isLessThan(rank)) {
|
|
continue;
|
|
}
|
|
|
|
ret = rank;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: votes
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
public EntityInternalMap<Vote> getVotes() {
|
|
return this.votes;
|
|
}
|
|
|
|
public void addVote(Vote vote) {
|
|
if (vote == null) {
|
|
throw new NullPointerException("vote");
|
|
}
|
|
this.getVotes().attach(vote);
|
|
}
|
|
|
|
public Optional<Vote> getVoteByName(String name) {
|
|
if (name == null) {
|
|
throw new NullPointerException("name");
|
|
}
|
|
return this.getVotes().getAll().stream().filter(vote -> vote.getName().equalsIgnoreCase(name)).findFirst();
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: relationWish
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
public Map<String, Rel> getRelationWishes() {
|
|
return this.relationWishes;
|
|
}
|
|
|
|
public void setRelationWishes(Map<String, Rel> relationWishes) {
|
|
// Clean input
|
|
MassiveMapDef<String, Rel> target = new MassiveMapDef<>(relationWishes);
|
|
|
|
// Detect Nochange
|
|
if (MUtil.equals(this.relationWishes, target)) {
|
|
return;
|
|
}
|
|
|
|
// Apply
|
|
this.relationWishes = target;
|
|
|
|
// Mark as changed
|
|
this.changed();
|
|
}
|
|
|
|
// FINER
|
|
|
|
public Rel getRelationWish(String factionId) {
|
|
Rel ret = this.getRelationWishes().get(factionId);
|
|
if (ret == null) {
|
|
ret = Rel.NEUTRAL;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
public Rel getRelationWish(Faction faction) {
|
|
return this.getRelationWish(faction.getId());
|
|
}
|
|
|
|
public void setRelationWish(String factionId, Rel rel) {
|
|
Map<String, Rel> relationWishes = this.getRelationWishes();
|
|
if (rel == null || rel == Rel.NEUTRAL) {
|
|
relationWishes.remove(factionId);
|
|
} else {
|
|
relationWishes.put(factionId, rel);
|
|
}
|
|
this.setRelationWishes(relationWishes);
|
|
}
|
|
|
|
public void setRelationWish(Faction faction, Rel rel) {
|
|
this.setRelationWish(faction.getId(), rel);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: flagOverrides
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
|
|
public Map<MFlag, Boolean> getFlags() {
|
|
// We start with default values ...
|
|
Map<MFlag, Boolean> ret = new MassiveMap<>();
|
|
for (MFlag mflag : MFlag.getAll()) {
|
|
ret.put(mflag, mflag.isStandard());
|
|
}
|
|
|
|
// ... and if anything is explicitly set we use that info ...
|
|
Iterator<Map.Entry<String, Boolean>> iter = this.flags.entrySet().iterator();
|
|
while (iter.hasNext()) {
|
|
// ... for each entry ...
|
|
Map.Entry<String, Boolean> entry = iter.next();
|
|
|
|
// ... extract id and remove null values ...
|
|
String id = entry.getKey();
|
|
if (id == null) {
|
|
iter.remove();
|
|
this.changed();
|
|
continue;
|
|
}
|
|
|
|
// ... resolve object and skip unknowns ...
|
|
MFlag mflag = MFlag.get(id);
|
|
if (mflag == null) {
|
|
continue;
|
|
}
|
|
|
|
ret.put(mflag, entry.getValue());
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
public void setFlags(Map<MFlag, Boolean> flags) {
|
|
Map<String, Boolean> flagIds = new MassiveMap<>();
|
|
for (Map.Entry<MFlag, Boolean> entry : flags.entrySet()) {
|
|
flagIds.put(entry.getKey().getId(), entry.getValue());
|
|
}
|
|
setFlagIds(flagIds);
|
|
}
|
|
|
|
public void setFlagIds(Map<String, Boolean> flagIds) {
|
|
// Clean input
|
|
MassiveMapDef<String, Boolean> target = new MassiveMapDef<>();
|
|
for (Map.Entry<String, Boolean> entry : flagIds.entrySet()) {
|
|
String key = entry.getKey();
|
|
if (key == null) {
|
|
continue;
|
|
}
|
|
key = key.toLowerCase(); // Lowercased Keys Version 2.6.0 --> 2.7.0
|
|
|
|
Boolean value = entry.getValue();
|
|
if (value == null) {
|
|
continue;
|
|
}
|
|
|
|
target.put(key, value);
|
|
}
|
|
|
|
// Detect Nochange
|
|
if (MUtil.equals(this.flags, target)) {
|
|
return;
|
|
}
|
|
|
|
// Apply
|
|
this.flags = new MassiveMapDef<>(target);
|
|
|
|
// Mark as changed
|
|
this.changed();
|
|
}
|
|
|
|
// FINER
|
|
|
|
public boolean getFlag(String flagId) {
|
|
if (flagId == null) {
|
|
throw new NullPointerException("flagId");
|
|
}
|
|
|
|
Boolean ret = this.flags.get(flagId);
|
|
if (ret != null) {
|
|
return ret;
|
|
}
|
|
|
|
MFlag flag = MFlag.get(flagId);
|
|
if (flag == null) {
|
|
throw new NullPointerException("flag");
|
|
}
|
|
|
|
return flag.isStandard();
|
|
}
|
|
|
|
public boolean getFlag(MFlag flag) {
|
|
if (flag == null) {
|
|
throw new NullPointerException("flag");
|
|
}
|
|
|
|
String flagId = flag.getId();
|
|
if (flagId == null) {
|
|
throw new NullPointerException("flagId");
|
|
}
|
|
|
|
Boolean ret = this.flags.get(flagId);
|
|
if (ret != null) {
|
|
return ret;
|
|
}
|
|
|
|
return flag.isStandard();
|
|
}
|
|
|
|
public Boolean setFlag(String flagId, boolean value) {
|
|
if (flagId == null) {
|
|
throw new NullPointerException("flagId");
|
|
}
|
|
|
|
Boolean ret = this.flags.put(flagId, value);
|
|
if (ret == null || ret != value) {
|
|
this.changed();
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
public Boolean setFlag(MFlag flag, boolean value) {
|
|
if (flag == null) {
|
|
throw new NullPointerException("flag");
|
|
}
|
|
|
|
String flagId = flag.getId();
|
|
if (flagId == null) {
|
|
throw new NullPointerException("flagId");
|
|
}
|
|
|
|
Boolean ret = this.flags.put(flagId, value);
|
|
if (ret == null || ret != value) {
|
|
this.changed();
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: perms
|
|
// -------------------------------------------- //
|
|
|
|
public Map<String, Set<String>> getPerms() {
|
|
return this.perms;
|
|
}
|
|
|
|
public Map<String, Set<String>> createNewPermMap() {
|
|
Map<String, Set<String>> ret = new MassiveMap<>();
|
|
for (MPerm mperm : MPerm.getAll()) {
|
|
Set<String> granteds = mperm2granteds(mperm);
|
|
|
|
ret.put(mperm.getId(), granteds);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
private Set<String> mperm2granteds(MPerm mperm) {
|
|
String permId = mperm.getId();
|
|
Set<String> value = new MassiveSet<>(MConf.get().perm2default.get(permId));
|
|
Set<String> additions = new MassiveSet<>();
|
|
outer:
|
|
for (Iterator<String> it = value.iterator(); it.hasNext(); ) {
|
|
String granted = it.next();
|
|
for (Rank rank : this.getRanks().getAll()) {
|
|
if (granted.equalsIgnoreCase(rank.getName())) {
|
|
it.remove();
|
|
additions.add(rank.getId());
|
|
continue outer;
|
|
}
|
|
}
|
|
}
|
|
value.addAll(additions);
|
|
return value;
|
|
}
|
|
|
|
// IS PERMITTED
|
|
|
|
public boolean isPlayerPermitted(MPlayer mplayer, String permId) {
|
|
if (isPermitted(mplayer.getId(), permId)) {
|
|
return true;
|
|
}
|
|
if (isPermitted(mplayer.getFaction().getId(), permId)) {
|
|
return true;
|
|
}
|
|
if (isPermitted(mplayer.getRank().getId(), permId)) {
|
|
return true;
|
|
}
|
|
return isPermitted(RelationUtil.getRelationOfThatToMe(mplayer, this).toString(), permId);
|
|
}
|
|
|
|
public boolean isPlayerPermitted(MPlayer mplayer, MPerm mperm) {
|
|
return isPlayerPermitted(mplayer, mperm.getId());
|
|
}
|
|
|
|
public boolean isFactionPermitted(Faction faction, String permId) {
|
|
if (isPermitted(faction.getId(), permId)) {
|
|
return true;
|
|
}
|
|
return isPermitted(RelationUtil.getRelationOfThatToMe(faction, this).toString(), permId);
|
|
}
|
|
|
|
public boolean isFactionPermitted(Faction faction, MPerm mperm) {
|
|
return isFactionPermitted(faction, mperm.getId());
|
|
}
|
|
|
|
public Set<String> getPermitted(String permId) {
|
|
if (permId == null) {
|
|
throw new NullPointerException("permId");
|
|
}
|
|
Set<String> permables = this.perms.get(permId);
|
|
|
|
if (permables == null) {
|
|
// No perms were found, but likely this is just a new MPerm.
|
|
// So if this does not exist in the database, throw an error.
|
|
if (!doesPermExist(permId)) {
|
|
throw new NullPointerException(permId + " caused null");
|
|
}
|
|
|
|
permables = new MassiveSet<>();
|
|
this.perms.put(permId, permables);
|
|
}
|
|
|
|
return permables;
|
|
}
|
|
|
|
public Set<String> getPermitted(MPerm mperm) {
|
|
return getPermitted(mperm.getId());
|
|
}
|
|
|
|
public Set<MPermable> getPermittedPermables(String permId) {
|
|
return MPerm.idsToMPermables(getPermitted(permId));
|
|
}
|
|
|
|
public Set<MPermable> getPermittedPermables(MPerm mperm) {
|
|
return getPermittedPermables(mperm.getId());
|
|
}
|
|
|
|
public boolean isPermitted(String permableId, String permId) {
|
|
if (permableId == null) {
|
|
throw new NullPointerException("permableId");
|
|
}
|
|
if (permId == null) {
|
|
throw new NullPointerException("permId");
|
|
}
|
|
|
|
// TODO: Isn't this section redundant and just a copy of that from getPermitted?
|
|
Set<String> permables = this.perms.get(permId);
|
|
if (permables == null) {
|
|
// No perms were found, but likely this is just a new MPerm.
|
|
// So if this does not exist in the database, throw an error.
|
|
if (!doesPermExist(permId)) {
|
|
throw new NullPointerException(permId + " caused null");
|
|
}
|
|
|
|
// Otherwise handle it
|
|
return false;
|
|
}
|
|
|
|
return getPermitted(permId).contains(permableId);
|
|
}
|
|
|
|
// SET PERMITTED
|
|
|
|
public boolean setPermitted(MPerm.MPermable mpermable, String permId, boolean permitted) {
|
|
boolean changed = false;
|
|
|
|
Set<String> perms = this.perms.get(permId);
|
|
if (perms == null) {
|
|
// No perms were found, but likely this is just a new MPerm.
|
|
// So if this does not exist in the database, throw an error.
|
|
if (!doesPermExist(permId)) {
|
|
throw new NullPointerException(permId + " caused null");
|
|
}
|
|
|
|
// Otherwise handle it
|
|
Set<String> permables = new MassiveSet<>();
|
|
this.perms.put(permId, permables);
|
|
changed = true;
|
|
}
|
|
|
|
if (permitted) {
|
|
changed = this.getPerms().get(permId).add(mpermable.getId()) | changed;
|
|
} else {
|
|
changed = this.getPerms().get(permId).remove(mpermable.getId()) | changed;
|
|
}
|
|
if (changed) {
|
|
this.changed();
|
|
}
|
|
return changed;
|
|
}
|
|
|
|
public boolean setPermitted(MPerm.MPermable mpermable, MPerm mperm, boolean permitted) {
|
|
return setPermitted(mpermable, mperm.getId(), permitted);
|
|
}
|
|
|
|
public void setPermittedRelations(String permId, Collection<MPerm.MPermable> permables) {
|
|
Set<String> ids = permables.stream().map(MPerm.MPermable::getId).collect(Collectors.toSet());
|
|
this.getPerms().put(permId, ids);
|
|
}
|
|
|
|
public void setPermittedRelations(MPerm perm, Collection<MPerm.MPermable> permables) {
|
|
setPermittedRelations(perm.getId(), permables);
|
|
}
|
|
|
|
|
|
private boolean doesPermExist(String permId) {
|
|
return MPermColl.get().getFixed(permId) != null;
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FIELD: tax
|
|
// -------------------------------------------- //
|
|
|
|
// RAW
|
|
public Map<String, Double> getTax() {
|
|
return this.tax;
|
|
}
|
|
|
|
// FINER GET
|
|
public Double getTaxFor(String id) {
|
|
if (id == null) {
|
|
throw new NullPointerException("id");
|
|
}
|
|
return this.tax.get(id);
|
|
}
|
|
|
|
public Double getTaxFor(Identified identified) {
|
|
if (identified == null) {
|
|
throw new NullPointerException("identified");
|
|
}
|
|
return this.getTaxFor(identified.getId());
|
|
}
|
|
|
|
public double getTaxForPlayer(MPlayer mplayer) {
|
|
return getTaxAndReasonForPlayer(mplayer).map(Entry::getValue).orElse(0D);
|
|
}
|
|
|
|
public Optional<Entry<String, Double>> getTaxAndReasonForPlayer(MPlayer mplayer) {
|
|
if (mplayer == null) {
|
|
throw new NullPointerException("mplayer");
|
|
}
|
|
|
|
if (mplayer.getFaction() != this) {
|
|
throw new IllegalArgumentException("Player " + mplayer.getId() + " not in " + this.getId());
|
|
}
|
|
|
|
Double ret = null;
|
|
|
|
ret = this.getTaxFor(mplayer);
|
|
if (ret != null) {
|
|
return Optional.of(new Couple<>(mplayer.getId(), ret));
|
|
}
|
|
|
|
ret = this.getTaxFor(mplayer.getRank());
|
|
if (ret != null) {
|
|
return Optional.of(new Couple<>(mplayer.getRank().getId(), ret));
|
|
}
|
|
|
|
ret = this.getTaxFor(IDENTIFIER_TAX_BASE);
|
|
if (ret != null) {
|
|
return Optional.of(new Couple<>(IDENTIFIER_TAX_BASE, ret));
|
|
}
|
|
|
|
return Optional.empty();
|
|
}
|
|
|
|
// FINER SET
|
|
public Double setTaxFor(String id, Double value) {
|
|
this.changed();
|
|
return this.tax.put(id, value);
|
|
}
|
|
|
|
public Double setTaxFor(Identified identified, Double value) {
|
|
return this.setTaxFor(identified.getId(), value);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// OVERRIDE: RelationParticipator
|
|
// -------------------------------------------- //
|
|
|
|
@Override
|
|
public String describeTo(RelationParticipator observer, boolean ucfirst) {
|
|
return RelationUtil.describeThatToMe(this, observer, ucfirst);
|
|
}
|
|
|
|
@Override
|
|
public String describeTo(RelationParticipator observer) {
|
|
return RelationUtil.describeThatToMe(this, observer);
|
|
}
|
|
|
|
@Override
|
|
public Rel getRelationTo(RelationParticipator observer) {
|
|
return RelationUtil.getRelationOfThatToMe(this, observer);
|
|
}
|
|
|
|
@Override
|
|
public Rel getRelationTo(RelationParticipator observer, boolean ignorePeaceful) {
|
|
return RelationUtil.getRelationOfThatToMe(this, observer, ignorePeaceful);
|
|
}
|
|
|
|
@Override
|
|
public ChatColor getColorTo(RelationParticipator observer) {
|
|
return RelationUtil.getColorOfThatToMe(this, observer);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// OVERRIDE: permable
|
|
// -------------------------------------------- //
|
|
|
|
@Override
|
|
public String getDisplayName(Object senderObject) {
|
|
MPlayer mplayer = MPlayer.get(senderObject);
|
|
if (mplayer == null) {
|
|
return this.getName();
|
|
}
|
|
|
|
return this.describeTo(mplayer);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// POWER
|
|
// -------------------------------------------- //
|
|
// TODO: Implement a has enough feature.
|
|
|
|
public double getPower() {
|
|
if (this.getFlag(MFlag.getFlagInfpower())) {
|
|
return 999999;
|
|
}
|
|
|
|
double ret = 0;
|
|
for (MPlayer mplayer : this.getMPlayers()) {
|
|
ret += mplayer.getPower();
|
|
}
|
|
|
|
ret = this.limitWithPowerMax(ret);
|
|
ret += this.getPowerBoost();
|
|
|
|
return ret;
|
|
}
|
|
|
|
public double getPowerMax() {
|
|
if (this.getFlag(MFlag.getFlagInfpower())) {
|
|
return 999999;
|
|
}
|
|
|
|
double ret = 0;
|
|
for (MPlayer mplayer : this.getMPlayers()) {
|
|
ret += mplayer.getPowerMax();
|
|
}
|
|
|
|
ret = this.limitWithPowerMax(ret);
|
|
ret += this.getPowerBoost();
|
|
|
|
return ret;
|
|
}
|
|
|
|
private double limitWithPowerMax(double power) {
|
|
// NOTE: 0.0 powerMax means there is no max power
|
|
double powerMax = MConf.get().factionPowerMax;
|
|
|
|
return powerMax <= 0 || power < powerMax ? power : powerMax;
|
|
}
|
|
|
|
public int getPowerRounded() {
|
|
return (int) Math.round(this.getPower());
|
|
}
|
|
|
|
public int getPowerMaxRounded() {
|
|
return (int) Math.round(this.getPowerMax());
|
|
}
|
|
|
|
public int getLandCount() {
|
|
return BoardColl.get().getCount(this);
|
|
}
|
|
|
|
public int getLandCountInWorld(String worldName) {
|
|
return Board.get(worldName).getCount(this);
|
|
}
|
|
|
|
public boolean hasLandInflation() {
|
|
return this.getLandCount() > this.getPowerRounded();
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// WORLDS
|
|
// -------------------------------------------- //
|
|
|
|
public Set<String> getClaimedWorlds() {
|
|
return BoardColl.get().getClaimedWorlds(this);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FOREIGN KEY: MPLAYER
|
|
// -------------------------------------------- //
|
|
|
|
public List<MPlayer> getMPlayers() {
|
|
return new MassiveList<>(FactionsIndex.get().getMPlayers(this));
|
|
}
|
|
|
|
public List<MPlayer> getMPlayers(java.util.function.Predicate<? super MPlayer> where, Comparator<? super MPlayer> orderby, Integer limit, Integer offset) {
|
|
return MUtil.transform(this.getMPlayers(), where, orderby, limit, offset);
|
|
}
|
|
|
|
public List<MPlayer> getMPlayersWhere(java.util.function.Predicate<? super MPlayer> predicate) {
|
|
return this.getMPlayers(predicate, null, null, null);
|
|
}
|
|
|
|
public List<MPlayer> getMPlayersWhereOnline(boolean online) {
|
|
return this.getMPlayersWhere(online ? SenderColl.PREDICATE_ONLINE : SenderColl.PREDICATE_OFFLINE);
|
|
}
|
|
|
|
public List<MPlayer> getMPlayersWhereOnlineTo(Object senderObject) {
|
|
return this.getMPlayersWhere(PredicateAnd.get(SenderColl.PREDICATE_ONLINE, PredicateVisibleTo.get(senderObject)));
|
|
}
|
|
|
|
public List<MPlayer> getMPlayersWhereRank(Rank rank) {
|
|
return this.getMPlayersWhere(PredicateMPlayerRank.get(rank));
|
|
}
|
|
|
|
public MPlayer getLeader() {
|
|
List<MPlayer> ret = this.getMPlayersWhereRank(this.getLeaderRank());
|
|
if (ret.size() == 0) {
|
|
return null;
|
|
}
|
|
return ret.get(0);
|
|
}
|
|
|
|
public Set<String> getMPlayerIds() {
|
|
return this.getMPlayers().stream().map(MPlayer::getId).collect(Collectors.toSet());
|
|
}
|
|
|
|
public List<CommandSender> getOnlineCommandSenders() {
|
|
// Create Ret
|
|
List<CommandSender> ret = new MassiveList<>();
|
|
|
|
// Fill Ret
|
|
for (CommandSender sender : IdUtil.getLocalSenders()) {
|
|
if (MUtil.isntSender(sender)) {
|
|
continue;
|
|
}
|
|
|
|
MPlayer mplayer = MPlayer.get(sender);
|
|
if (mplayer.getFaction() != this) {
|
|
continue;
|
|
}
|
|
|
|
ret.add(sender);
|
|
}
|
|
|
|
// Return Ret
|
|
return ret;
|
|
}
|
|
|
|
public List<Player> getOnlinePlayers() {
|
|
// Create Ret
|
|
List<Player> ret = new MassiveList<>();
|
|
|
|
// Fill Ret
|
|
for (Player player : Bukkit.getOnlinePlayers()) {
|
|
if (MUtil.isntPlayer(player)) {
|
|
continue;
|
|
}
|
|
|
|
MPlayer mplayer = MPlayer.get(player);
|
|
if (mplayer.getFaction() != this) {
|
|
continue;
|
|
}
|
|
|
|
ret.add(player);
|
|
}
|
|
|
|
// Return Ret
|
|
return ret;
|
|
}
|
|
|
|
// used when current leader is about to be removed from the faction; promotes new leader, or disbands faction if no other members left
|
|
public void promoteNewLeader() {
|
|
if (!this.isNormal()) {
|
|
return;
|
|
}
|
|
if (this.getFlag(MFlag.getFlagPermanent()) && MConf.get().permanentFactionsDisableLeaderPromotion) {
|
|
return;
|
|
}
|
|
|
|
MPlayer oldLeader = this.getLeader();
|
|
Rank leaderRank = oldLeader.getRank();
|
|
|
|
List<MPlayer> replacements = Collections.emptyList();
|
|
for (Rank rank = leaderRank; rank != null; rank = rank.getRankBelow()) {
|
|
//Skip first
|
|
if (rank == leaderRank) {
|
|
continue;
|
|
}
|
|
|
|
replacements = this.getMPlayersWhereRank(rank);
|
|
if (!replacements.isEmpty()) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if we found a replacement
|
|
if (replacements == null || replacements.isEmpty()) {
|
|
// faction leader is the only member; one-man faction
|
|
if (this.getFlag(MFlag.getFlagPermanent())) {
|
|
if (oldLeader != null) {
|
|
oldLeader.setRank(this.getLeaderRank().getRankBelow());
|
|
}
|
|
return;
|
|
}
|
|
|
|
// no members left and faction isn't permanent, so disband it
|
|
if (MConf.get().logFactionDisband) {
|
|
Factions.get().log("The faction " + this.getName() + " (" + this.getId() + ") has been disbanded since it has no members left.");
|
|
}
|
|
|
|
for (MPlayer mplayer : MPlayerColl.get().getAllOnline()) {
|
|
mplayer.msg("<i>The faction %s<i> was disbanded.", this.getName(mplayer));
|
|
}
|
|
|
|
this.detach();
|
|
} else {
|
|
// promote new faction leader
|
|
if (oldLeader != null) {
|
|
oldLeader.setRank(this.getLeaderRank().getRankBelow());
|
|
}
|
|
|
|
replacements.get(0).setRank(this.getLeaderRank());
|
|
this.msg("<i>Faction leader <h>%s<i> has been removed. %s<i> has been promoted as the new faction leader.", oldLeader == null ? "" : oldLeader.getName(), replacements.get(0).getName());
|
|
Factions.get().log("Faction " + this.getName() + " (" + this.getId() + ") leader was removed. Replacement leader: " + replacements.get(0).getName());
|
|
}
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// FACTION ONLINE STATE
|
|
// -------------------------------------------- //
|
|
|
|
public boolean isAllMPlayersOffline() {
|
|
return this.getMPlayersWhereOnline(true).size() == 0;
|
|
}
|
|
|
|
public boolean isAnyMPlayersOnline() {
|
|
return !this.isAllMPlayersOffline();
|
|
}
|
|
|
|
public boolean isFactionConsideredOffline() {
|
|
return this.isAllMPlayersOffline();
|
|
}
|
|
|
|
public boolean isFactionConsideredOnline() {
|
|
return !this.isFactionConsideredOffline();
|
|
}
|
|
|
|
public boolean isExplosionsAllowed() {
|
|
boolean explosions = this.getFlag(MFlag.getFlagExplosions());
|
|
boolean offlineexplosions = this.getFlag(MFlag.getFlagOfflineexplosions());
|
|
|
|
if (explosions && offlineexplosions) {
|
|
return true;
|
|
}
|
|
if (!explosions && !offlineexplosions) {
|
|
return false;
|
|
}
|
|
|
|
boolean online = this.isFactionConsideredOnline();
|
|
|
|
return (online && explosions) || (!online && offlineexplosions);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// MESSAGES
|
|
// -------------------------------------------- //
|
|
// These methods are simply proxied in from the Mixin.
|
|
|
|
// CONVENIENCE SEND MESSAGE
|
|
|
|
public boolean sendMessage(Object message) {
|
|
return MixinMessage.get().messagePredicate(new PredicateCommandSenderFaction(this), message);
|
|
}
|
|
|
|
public boolean sendMessage(Object... messages) {
|
|
return MixinMessage.get().messagePredicate(new PredicateCommandSenderFaction(this), messages);
|
|
}
|
|
|
|
public boolean sendMessage(Collection<Object> messages) {
|
|
return MixinMessage.get().messagePredicate(new PredicateCommandSenderFaction(this), messages);
|
|
}
|
|
|
|
// CONVENIENCE MSG
|
|
|
|
public boolean msg(String msg) {
|
|
return MixinMessage.get().msgPredicate(new PredicateCommandSenderFaction(this), msg);
|
|
}
|
|
|
|
public boolean msg(String msg, Object... args) {
|
|
return MixinMessage.get().msgPredicate(new PredicateCommandSenderFaction(this), msg, args);
|
|
}
|
|
|
|
public boolean msg(Collection<String> msgs) {
|
|
return MixinMessage.get().msgPredicate(new PredicateCommandSenderFaction(this), msgs);
|
|
}
|
|
|
|
// -------------------------------------------- //
|
|
// UTIL
|
|
// -------------------------------------------- //
|
|
|
|
// FIXME this probably needs to be moved elsewhere
|
|
public static String clean(String message) {
|
|
String target = message;
|
|
if (target == null) {
|
|
return null;
|
|
}
|
|
|
|
target = target.trim();
|
|
if (target.isEmpty()) {
|
|
target = null;
|
|
}
|
|
|
|
return target;
|
|
}
|
|
|
|
}
|