package com.planet_ink.coffee_mud.Behaviors; import com.planet_ink.coffee_mud.core.interfaces.*; import com.planet_ink.coffee_mud.core.*; import com.planet_ink.coffee_mud.core.collections.*; import com.planet_ink.coffee_mud.Abilities.interfaces.*; import com.planet_ink.coffee_mud.Areas.interfaces.*; import com.planet_ink.coffee_mud.Behaviors.interfaces.*; import com.planet_ink.coffee_mud.CharClasses.interfaces.*; import com.planet_ink.coffee_mud.Commands.interfaces.*; import com.planet_ink.coffee_mud.Common.interfaces.*; import com.planet_ink.coffee_mud.Exits.interfaces.*; import com.planet_ink.coffee_mud.Items.interfaces.*; import com.planet_ink.coffee_mud.Libraries.interfaces.*; import com.planet_ink.coffee_mud.Locales.interfaces.*; import com.planet_ink.coffee_mud.MOBS.interfaces.*; import com.planet_ink.coffee_mud.Races.interfaces.*; import java.util.*; /* Copyright 2004-2019 Bo Zimmerman Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ public class TaxCollector extends StdBehavior { @Override public String ID() { return "TaxCollector"; } @Override protected int canImproveCode() { return Behavior.CAN_MOBS; } protected DVector demanded = null; protected DVector paid = null; protected long waitTime = 1000 * 60 * 2; protected long graceTime = 1000 * 60 * 60; protected int lastMonthChecked = -1; protected List<LandTitle> taxableProperties = new Vector<LandTitle>(); protected Set<String> peopleWhoOwe = new HashSet<String>(); protected String treasuryRoomID = null; protected Container treasuryContainer = null; public final static int OWE_TOTAL = 0; public final static int OWE_CITIZENTAX = 1; public final static int OWE_BACKTAXES = 2; public final static int OWE_FINES = 3; @Override public String accountForYourself() { return "tax collecting"; } @Override public void setParms(final String newParms) { super.setParms(newParms); demanded=null; paid=null; waitTime=CMParms.getParmInt(newParms,"WAIT",1000*60*2); graceTime=CMParms.getParmInt(newParms,"GRACE",1000*60*60); } protected boolean belongsToAnOweingClan(final MOB M) { for(final String clanID : peopleWhoOwe) { if(M.getClanRole(clanID)!=null) return true; } return false; } public double[] totalMoneyOwed(final MOB collector,final MOB M) { final double[] owed=new double[4]; if((peopleWhoOwe.contains(M.Name())) ||(belongsToAnOweingClan(M))) { for(int t=0;t<taxableProperties.size();t++) { final LandTitle T=taxableProperties.get(t); if((T.getOwnerName().equals(M.Name()) ||(M.getClanRole(T.getOwnerName())!=null)) &&(T.backTaxes()>0)) owed[OWE_BACKTAXES]+=T.backTaxes(); } } final LegalBehavior B=CMLib.law().getLegalBehavior(M.location()); if(B!=null) { final Area A2=CMLib.law().getLegalObject(M.location()); if(A2!=null) owed[OWE_FINES]=B.finesOwed(M); if((A2!=null) &&(!B.isAnyOfficer(A2,M)) &&(!B.isJudge(A2,M))) { final Law theLaw=B.legalInfo(A2); if(theLaw!=null) { final double cittax=CMath.s_double((String)theLaw.taxLaws().get("CITTAX")); if(cittax>0.0) owed[OWE_CITIZENTAX]=CMath.mul(CMLib.beanCounter().getTotalAbsoluteShopKeepersValue(M,collector),CMath.div(cittax,100.0)); } } } else owed[OWE_CITIZENTAX]=CMath.div(CMLib.beanCounter().getTotalAbsoluteShopKeepersValue(M,collector),10.0); owed[OWE_TOTAL]=owed[OWE_CITIZENTAX]+owed[OWE_BACKTAXES]+owed[OWE_FINES]; owed[OWE_TOTAL]=Math.round(owed[OWE_TOTAL]); return owed; } @Override public void executeMsg(final Environmental host, final CMMsg msg) { super.executeMsg(host,msg); if((host!=null)&&(host instanceof MOB)) { final MOB mob=(MOB)host; if(msg.amITarget(mob) &&(msg.targetMinor()==CMMsg.TYP_GIVE) &&(msg.tool() instanceof Coins)) { double paidAmount=((Coins)msg.tool()).getTotalValue(); if(paidAmount<=0.0) return; final double[] owed=totalMoneyOwed(mob,msg.source()); if(treasuryRoomID!=null) { final Room treasuryR=CMLib.map().getRoom(treasuryRoomID); if(treasuryR!=null) { final Coins COIN=CMLib.beanCounter().makeBestCurrency(CMLib.beanCounter().getCurrency(mob),paidAmount,treasuryR,treasuryContainer); if(COIN!=null) COIN.putCoinsBack(); } } if((demanded!=null)&&(demanded.contains(msg.source()))) { final int demanDex=demanded.indexOf(msg.source()); if(demanDex>=0) { paidAmount-=owed[OWE_CITIZENTAX]; demanded.removeElementAt(demanDex); } } if(paid.contains(msg.source())) paid.removeElement(msg.source()); paid.addElement(msg.source(),Long.valueOf(System.currentTimeMillis())); if(owed[OWE_FINES]>0) { final LegalBehavior B=CMLib.law().getLegalBehavior(msg.source().location()); final Area A2=CMLib.law().getLegalObject(msg.source().location()); if((B!=null)&&(A2!=null)) { if(paidAmount>=owed[OWE_FINES]) { paidAmount-=owed[OWE_FINES]; B.modifyAssessedFines(0.0,msg.source()); } else { owed[OWE_FINES]-=paidAmount; paidAmount=0; B.modifyAssessedFines(owed[OWE_FINES],msg.source()); } } } int numProperties = 0; int numBackTaxesUnpaid = 0; boolean paidBackTaxes = false; for(int i=0;i<taxableProperties.size();i++) { final LandTitle T=taxableProperties.get(i); if(T.getOwnerName().equals(msg.source().Name()) ||(msg.source().getClanRole(T.getOwnerName())!=null)) { numProperties++; if(T.backTaxes()>0) { numBackTaxesUnpaid++; if(paidAmount>=0) { if(paidAmount>=T.backTaxes()) { paidAmount-=T.backTaxes(); T.setBackTaxes(0); T.updateTitle(); numBackTaxesUnpaid--; paidBackTaxes=true; } else { paidAmount=0; T.setBackTaxes(T.backTaxes()-(int)Math.round(paidAmount)); T.updateTitle(); break; } } } } } if((paidBackTaxes) &&(numBackTaxesUnpaid==0) &&(mob.location()!=null)) { final LegalBehavior B=CMLib.law().getLegalBehavior(mob.location().getArea()); if((B!=null)&&(!msg.source().isMonster())) { final Area A2=CMLib.law().getLegalObject(mob.location().getArea()); B.aquit(A2,msg.source(),new String[]{"TAXEVASION"}); } } if((paidAmount>0) &&(numProperties>0)) { for(int i=0;i<taxableProperties.size();i++) { final LandTitle T=taxableProperties.get(i); if(((T.getOwnerName().equals(msg.source().Name()))) &&(paidAmount>0)) { T.setBackTaxes(T.backTaxes()-(int)Math.round(CMath.div(paidAmount,numProperties))); T.updateTitle(); paidAmount-=CMath.div(paidAmount,numProperties); } } } msg.addTrailerMsg(CMClass.getMsg(mob,msg.source(),null,CMMsg.MSG_SPEAK,L("<S-NAME> says 'Very good. Your taxes are paid in full.' to <T-NAMESELF>."))); } } } @Override public boolean okMessage(final Environmental host, final CMMsg msg) { if((host==null)||(!(host instanceof MOB))) return super.okMessage(host,msg); final MOB mob=(MOB)host; if(msg.amITarget(mob) &&(msg.targetMinor()==CMMsg.TYP_GIVE) &&(msg.tool() instanceof Coins)) { final String currency=CMLib.beanCounter().getCurrency(mob); final double[] owe=totalMoneyOwed(mob,msg.source()); final double coins=((Coins)msg.tool()).getTotalValue(); if((paid!=null)&&(paid.contains(msg.source()))) owe[OWE_TOTAL]-=owe[OWE_CITIZENTAX]; final String owed=CMLib.beanCounter().nameCurrencyShort(currency,owe[OWE_TOTAL]); if((!((Coins)msg.tool()).getCurrency().equals(CMLib.beanCounter().getCurrency(mob)))) { msg.source().tell(L("@x1 refuses your money.",mob.name(msg.source()))); CMLib.commands().postSay(mob,msg.source(),L("I don't accept that kind of currency."),false,false); return false; } if(coins<owe[OWE_TOTAL]) { msg.source().tell(L("@x1 refuses your money.",mob.name(msg.source()))); CMLib.commands().postSay(mob,msg.source(),L("That's not enough. You owe @x1. Try again.",owed),false,false); return false; } } return true; } @Override public boolean tick(final Tickable ticking, final int tickID) { super.tick(ticking,tickID); if((tickID!=Tickable.TICKID_MOB)||(!(ticking instanceof MOB))) return true; if(!CMProps.getBoolVar(CMProps.Bool.MUDSTARTED)) return true; final MOB mob=(MOB)ticking; if(demanded==null) demanded=new DVector(2); if(paid==null) paid=new DVector(2); for(int i=paid.size()-1;i>=0;i--) { final Long L=(Long)paid.elementAt(i,2); if((System.currentTimeMillis()-L.longValue())>graceTime) paid.removeElementAt(i); } final Room R=mob.location(); if((R!=null) &&(lastMonthChecked!=R.getArea().getTimeObj().getMonth())) { lastMonthChecked=R.getArea().getTimeObj().getMonth(); final Law theLaw=CMLib.law().getTheLaw(R,mob); if(theLaw!=null) { final Area A2=CMLib.law().getLegalObject(R); final String taxs=(String)theLaw.taxLaws().get("PROPERTYTAX"); LandTitle T=null; peopleWhoOwe.clear(); if((taxs!=null) &&(taxs.length()>0) &&(CMath.s_double(taxs)>0)) { taxableProperties=CMLib.law().getAllUniqueLandTitles(A2.getMetroMap(),"*",false); for(int v=0;v<taxableProperties.size();v++) { T=taxableProperties.get(v); if((!peopleWhoOwe.contains(T.getOwnerName())) &&(T.backTaxes()>0)) peopleWhoOwe.add(T.getOwnerName()); } } else taxableProperties.clear(); final Law.TreasurySet treas=theLaw.getTreasuryNSafe(A2); treasuryRoomID=CMLib.map().getExtendedRoomID(treas.room); treasuryContainer=treas.container; } } if((R!=null) &&(!mob.isInCombat()) &&(CMLib.flags().isAliveAwakeMobileUnbound(mob,true)) &&(R.numInhabitants()>1)) { final MOB M=R.fetchRandomInhabitant(); if((M!=null) &&(M!=mob) &&(!CMLib.flags().isAnimalIntelligence(M)) &&(!CMSecurity.isAllowed(M,R,CMSecurity.SecFlag.ORDER)) &&(!CMSecurity.isAllowed(M,R,CMSecurity.SecFlag.CMDROOMS)) &&(CMLib.clans().findCommonRivalrousClans(mob, M).size()==0) &&(CMLib.flags().canBeSeenBy(M,mob))) { final int demandDex=demanded.indexOf(M); if((demandDex>=0) &&((System.currentTimeMillis()-((Long)demanded.elementAt(demandDex,2)).longValue())>waitTime)) { final LegalBehavior B=CMLib.law().getLegalBehavior(R.getArea()); if(M.isMonster() &&(M.getStartRoom()!=null) &&(CMLib.law().getLandTitle(M.getStartRoom())==null)) demanded.removeElementAt(demandDex); else if(B!=null) { B.accuse(CMLib.law().getLegalObject(R),M,mob,new String[]{"TAXEVASION"}); CMLib.commands().postSay(mob,M,L("Can't pay huh? Well, you'll be hearing from the law -- THAT's for sure!"),false,false); } else { CMLib.commands().postSay(mob,M,L("You know what they say about death and taxes, so if you won't pay ... DIE!!!!"),false,false); CMLib.combat().postAttack(mob,M,mob.fetchWieldedItem()); demanded.removeElementAt(demandDex); } } if((!paid.contains(M))&&(demandDex<0)) { final double[] owe=totalMoneyOwed(mob,M); final StringBuffer say=new StringBuffer(""); final String currency=CMLib.beanCounter().getCurrency(mob); final double denomination=CMLib.beanCounter().getLowestDenomination(currency); if(owe[OWE_CITIZENTAX]>1.0) { final String moneyStr = CMLib.beanCounter().getDenominationName(currency,denomination,Math.round(CMath.div(owe[OWE_CITIZENTAX],denomination))); say.append(L("You owe @x1 in local taxes. ",moneyStr)); } if(owe[OWE_BACKTAXES]>1.0) { final String moneyStr = CMLib.beanCounter().getDenominationName(currency,denomination,Math.round(CMath.div(owe[OWE_BACKTAXES],denomination))); say.append(L("You owe @x1 in back property taxes",moneyStr)); } if(owe[OWE_FINES]>1.0) { final String moneyStr = CMLib.beanCounter().getDenominationName(currency,denomination,Math.round(CMath.div(owe[OWE_FINES],denomination))); say.append(L("You owe @x1 in fines",moneyStr)); } if(say.length()>0) { CMLib.commands().postSay(mob,M,L("@x1. You must pay me immediately or face the consequences.",say.toString()),false,false); demanded.addElement(M,Long.valueOf(System.currentTimeMillis())); if(M.isMonster()) { final List<String> V=new Vector<String>(); V.add("GIVE"); V.add(""+Math.round(owe[OWE_TOTAL])); V.add(mob.name()); M.doCommand(V,MUDCmdProcessor.METAFLAG_FORCED); } } } } final Item I=R.getRandomItem(); if((I!=null)&&(I instanceof Coins)) CMLib.commands().postGet(mob,I.container(),I,false); } return true; } }