package com.planet_ink.coffee_mud.Abilities.Thief; 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.TrackingLibrary; 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 2003-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 Thief_SetAlarm extends ThiefSkill implements Trap { @Override public String ID() { return "Thief_SetAlarm"; } private final static String localizedName = CMLib.lang().L("Set Alarm"); @Override public String name() { return localizedName; } @Override protected int canAffectCode() { return Ability.CAN_EXITS; } @Override protected int canTargetCode() { return Ability.CAN_EXITS; } @Override public int classificationCode() { return Ability.ACODE_THIEF_SKILL | Ability.DOMAIN_TRAPPING; } @Override public int abstractQuality() { return Ability.QUALITY_INDIFFERENT; } private static final String[] triggerStrings = I(new String[] { "SETALARM" }); @Override public String[] triggerStrings() { return triggerStrings; } @Override public int usageType() { return USAGE_MOVEMENT | USAGE_MANA; } protected boolean sprung = false; public Room room1 = null; public Room room2 = null; protected String roomID = null; protected int roomDir = 0; protected int trackLvl= 10; protected int levels = 1000; @Override public boolean isABomb() { return false; } @Override public void activateBomb() { } @Override public boolean sprung() { return sprung; } @Override public boolean disabled() { return false; } @Override public void disable() { unInvoke(); } @Override public void setReset(final int Reset) { } @Override public int getReset() { return 0; } @Override public void resetTrap(final MOB mob) { } @Override public boolean maySetTrap(final MOB mob, final int asLevel) { return false; } @Override public boolean canSetTrapOn(final MOB mob, final Physical P) { return false; } @Override public List<Item> getTrapComponents() { return new Vector<Item>(1); } @Override public boolean canReSetTrap(final MOB mob) { return false; } @Override public String requiresToSet() { return ""; } @Override public Trap setTrap(final MOB mob, final Physical P, final int trapBonus, final int qualifyingClassLevel, final boolean perm) { beneficialAffect(mob, P, qualifyingClassLevel + trapBonus, 0); return (Trap) P.fetchEffect(ID()); } @Override public void spring(final MOB M) { sprung=true; if((!canBeUninvoked()) && (!CMLib.threads().isTicking(this, -1))) { CMLib.threads().startTickDown(this, Tickable.TICKID_SPELL_AFFECT, 1); CMLib.threads().startTickDown(this, Tickable.TICKID_TRAP_RESET, 100); } } public void ensureRooms() { if((room1==null)||(room2==null)) { if((this.roomID!=null) &&(this.roomID.length()>0)) { room1=CMLib.map().getRoom(this.roomID); if(room1 != null) { room2=room1.getRoomInDir(this.roomDir); } } } } public boolean isLocalExempt(final MOB target) { if(target==null) return false; ensureRooms(); final Room R=room1; if((!canBeUninvoked()) &&(!isABomb()) &&(R!=null)) { if((CMLib.law().getLandTitle(R)!=null) &&(CMLib.law().doesHavePriviledgesHere(target,R))) return true; if((target.isMonster()) &&(target.getStartRoom()!=null) &&(target.getStartRoom().getArea()==R.getArea())) return true; } return false; } protected boolean canInvokeTrapOn(final MOB invoker, final MOB target) { if(!isLocalExempt(target)) return true; return false; } @Override public void executeMsg(final Environmental myHost, final CMMsg msg) { if(sprung) { return; } super.executeMsg(myHost,msg); if((msg.amITarget(affected)) &&(msg.targetMinor()==CMMsg.TYP_OPEN)) { if((!msg.amISource(invoker()) &&(canInvokeTrapOn(invoker(),msg.source())) &&(CMLib.dice().rollPercentage()>msg.source().charStats().getSave(CharStats.STAT_SAVE_TRAPS)))) spring(msg.source()); } } @Override public void setMiscText(final String miscText) { super.setMiscText(miscText); sprung=false; if(miscText.length()>0) { levels=CMParms.getParmInt(miscText, "LEVELS", 1000); trackLvl=CMParms.getParmInt(miscText, "TRACKLVL", 10); roomDir=CMParms.getParmInt(miscText, "DIR", 0); roomID=CMParms.getParmStr(miscText, "ROOM", ""); } } @Override public boolean tick(final Tickable ticking, final int tickID) { if(!super.tick(ticking,tickID)) return false; ensureRooms(); if((affected==null) ||(!(affected instanceof Exit)) ||(room1==null) ||(room2==null)) return false; if((tickID == Tickable.TICKID_TRAP_RESET) ||(tickID == Tickable.TICKID_TRAP_DESTRUCTION)) { if(!canBeUninvoked()) { sprung=false; return false; } else unInvoke(); } else if(sprung && (levels>0)) { final MOB realInvoker = invoker(); final MOB invoker=(realInvoker != null) ? realInvoker : CMClass.getFactoryMOB(L("an alarm"), 1, room1); try { final List<Room> rooms=new ArrayList<Room>(); TrackingLibrary.TrackingFlags flags; flags = CMLib.tracking().newFlags() .plus(TrackingLibrary.TrackingFlag.OPENONLY) .plus(TrackingLibrary.TrackingFlag.AREAONLY); CMLib.tracking().getRadiantRooms(room1,rooms,flags,null,trackLvl,null); CMLib.tracking().getRadiantRooms(room2,rooms,flags,null,trackLvl,null); final List<MOB> mobsDone=new ArrayList<MOB>(); room1.showHappens(CMMsg.MSG_NOISE,L("A horrible alarm is going off here.")); room2.showHappens(CMMsg.MSG_NOISE,L("A horrible alarm is going off here.")); final List<MOB> mobsAvailable = new ArrayList<MOB>(rooms.size()); for(int r=0;r<rooms.size();r++) { final Room R=rooms.get(r); if((R!=room1)&&(R!=room2)) { final int dir=CMLib.tracking().radiatesFromDir(R,rooms); if(dir>=0) { for(int i=0;i<R.numInhabitants();i++) { final MOB M=R.fetchInhabitant(i); if((M!=null) &&(M.isMonster()) &&(!M.isInCombat()) &&(CMLib.flags().isMobile(M)) &&(!mobsAvailable.contains(M)) &&(CMLib.flags().canHear(M))) mobsAvailable.add(M); } } } } while((mobsAvailable.size()>0)&&(levels>0)) { final MOB M=mobsAvailable.remove(CMLib.dice().roll(1, mobsAvailable.size(), -1)); final Room R=M.location(); final int dir=CMLib.tracking().radiatesFromDir(R,rooms); if(dir>=0) { if((!M.isInCombat()) &&(!mobsDone.contains(M)) &&(CMLib.flags().canHear(M)) &&(!CMLib.flags().isTracking(M))) { if((R.show(invoker, M, this, CMMsg.MSG_NOISE|CMMsg.MASK_MALICIOUS,L("<T-NAME> hear(s) a loud alarm @x1.",CMLib.directions().getInDirectionName(dir)))) &&(CMLib.dice().rollPercentage()>M.charStats().getSave(CharStats.STAT_SAVE_MIND)) &&(CMLib.dice().rollPercentage()>M.charStats().getSave(CharStats.STAT_SAVE_TRAPS))) { if(CMLib.tracking().autoTrack(M, room1)) { mobsDone.add(M); final Ability A=CMClass.getAbility("WanderHomeLater"); if(A!=null) { final int ticks=trackLvl*2; A.setMiscText("ONCE=true MINTICKS="+ticks+" MAXTICKS="+ticks+" IGNOREPCS=true RESPECTFOLLOW=true"); M.addEffect(A); A.setSavable(false); A.makeLongLasting(); } levels-=M.phyStats().level(); } } Ability A=M.fetchEffect("TemporaryImmunity"); if(A==null) { A=CMClass.getAbility("TemporaryImmunity"); if(A!=null) { A.setSavable(false); A.makeLongLasting(); M.addEffect(A); A.makeLongLasting(); } } if((A!=null)&&(A.text().indexOf(ID())<0)) A.setMiscText("+"+ID()); } } } } finally { if(invoker != realInvoker) invoker.destroy(); // it was a factory mob } } return true; } @Override public boolean invoke(final MOB mob, final List<String> commands, final Physical givenTarget, final boolean auto, final int asLevel) { final String whatToalarm=CMParms.combine(commands,0); Exit alarmThis=null; final int dirCode=CMLib.directions().getGoodDirectionCode(whatToalarm); if(dirCode>=0) alarmThis=mob.location().getExitInDir(dirCode); if((alarmThis==null)||(!alarmThis.hasADoor())) { mob.tell(L("You can't set an alarm that way.")); return false; } if(!super.invoke(mob,commands,givenTarget,auto,asLevel)) return false; final boolean success=proficiencyCheck(mob,0,auto); final CMMsg msg=CMClass.getMsg(mob,alarmThis,this,auto?CMMsg.MSG_OK_ACTION:CMMsg.MSG_THIEF_ACT,CMMsg.MASK_ALWAYS|CMMsg.MSG_THIEF_ACT,CMMsg.MSG_OK_ACTION,(auto?L("@x1 begins to glow!",alarmThis.name()):L("<S-NAME> attempt(s) to lay a trap on @x1.",alarmThis.name()))); if(mob.location().okMessage(mob,msg)) { invoker=mob; mob.location().send(mob,msg); final int levelsLeftToDo = (mob.phyStats().level() * 2) + (2 * super.getXLEVELLevel(mob)); final int trackLevel = 5 + (super.adjustedLevel(mob, asLevel)/10) + super.getXLEVELLevel(mob); room1=mob.location(); room2=mob.location().getRoomInDir(dirCode); if(success) { sprung=false; mob.tell(L("You have set the alarm.")); final Thief_SetAlarm A; if(CMLib.law().doesOwnThisLand(mob,room1)) { A=(Thief_SetAlarm)this.copyOf(); alarmThis.addNonUninvokableEffect(A); CMLib.database().DBUpdateExits(room1); } else A = (Thief_SetAlarm)beneficialAffect(mob,alarmThis,asLevel,trackLevel); if(A!=null) { A.levels = levelsLeftToDo; A.trackLvl= trackLevel; A.setMiscText("LEVELS="+levelsLeftToDo+" TRACKLVL="+trackLevel+" ROOM=\""+CMLib.map().getExtendedRoomID(room1)+"\" DIR="+dirCode); } } else { if(CMLib.dice().rollPercentage()>50) { final Thief_SetAlarm A = (Thief_SetAlarm)beneficialAffect(mob,alarmThis,asLevel,trackLevel); if(A!=null) { mob.location().show(mob,null,CMMsg.MSG_OK_ACTION,L("<S-NAME> trigger(s) the alarm on accident!")); A.levels = levelsLeftToDo; A.trackLvl= trackLevel; A.spring(mob); } } else { mob.tell(L("You fail in your attempt to set an alarm.")); } } } return success; } }