package com.planet_ink.coffee_mud.Abilities.Spells;
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.Libraries.interfaces.TrackingLibrary.TrackingFlag;
import com.planet_ink.coffee_mud.Libraries.interfaces.TrackingLibrary.TrackingFlags;
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 2016-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 Spell_Lighthouse extends Spell
{
@Override
public String ID()
{
return "Spell_Lighthouse";
}
private final static String localizedName = CMLib.lang().L("Lighthouse");
@Override
public String name()
{
return localizedName;
}
private final static String localizedStaticDisplay = CMLib.lang().L("(Lighthouse)");
@Override
public String displayText()
{
return localizedStaticDisplay;
}
@Override
public int abstractQuality()
{
return Ability.QUALITY_OK_SELF;
}
@Override
protected int canAffectCode()
{
return CAN_MOBS;
}
@Override
public int classificationCode()
{
return Ability.ACODE_SPELL | Ability.DOMAIN_EVOCATION;
}
@Override
public int castingQuality(final MOB mob, final Physical target)
{
if(mob!=null)
{
if(!CMLib.flags().canBeSeenBy(mob.location(), mob))
return super.castingQuality(mob, target,Ability.QUALITY_BENEFICIAL_SELF);
}
return super.castingQuality(mob,target);
}
@Override
public void affectPhyStats(final Physical affected, final PhyStats affectableStats)
{
if(!(affected instanceof Room))
affectableStats.setDisposition(affectableStats.disposition()|PhyStats.IS_LIGHTSOURCE);
if(CMLib.flags().isInDark(affected))
affectableStats.setDisposition(affectableStats.disposition()-PhyStats.IS_DARK);
}
protected Set<Room> roomSet=new HashSet<Room>();
protected volatile Room lastRoom=null;
protected volatile Room lastShipRoom=null;
protected volatile int lastDir = -1;
public void addToRoom(final Room R, final Item fromShip, final int fromDir)
{
if(R==null)
return;
final Ability A=CMClass.getAbility("Spell_Light");
if((A!=null)&&(R.fetchEffect(ID())==null))
{
final boolean isInDark=CMLib.flags().isInDark(R);
A.setInvoker(invoker());
A.setSavable(false);
A.setExpirationDate(System.currentTimeMillis()+8000);
R.addEffect(A);
R.recoverPhyStats();
R.recoverRoomStats();
if(isInDark && (!CMLib.flags().isInDark(R)))
{
String dirName;
if(fromShip != null)
{
dirName=fromShip.Name();
R.showHappens(CMMsg.MSG_OK_VISUAL,L("A bright light shines in from @x1.",dirName));
}
else
if(fromDir >= 0)
{
if(R.getArea() instanceof BoardableShip)
dirName=CMLib.directions().getFromShipDirectionName(fromDir);
else
dirName=CMLib.directions().getFromCompassDirectionName(fromDir);
R.showHappens(CMMsg.MSG_OK_VISUAL,L("A bright light shines in from @x1.",dirName));
}
else
{
R.showHappens(CMMsg.MSG_OK_VISUAL,L("A bright light shines in."));
}
}
}
}
public void delFromRoom(final Room R, final Item fromShip, final int fromDir)
{
if(R==null)
return;
try
{
final Ability A=R.fetchEffect("Spell_Light");
if((A!=null)&&(!A.isSavable())&&(A.invoker()==invoker()))
{
final boolean isInDark=CMLib.flags().isInDark(R);
R.delEffect(A);
R.recoverPhyStats();
final boolean darkNow=CMLib.flags().isInDark(R);
R.addEffect(A);
R.recoverPhyStats();
if(!isInDark && darkNow)
{
String dirName;
if(fromShip != null)
{
dirName=fromShip.Name();
R.showHappens(CMMsg.MSG_OK_VISUAL,L("A bright light from @x1 disappears.",dirName));
}
else
if(fromDir >= 0)
{
if(R.getArea() instanceof BoardableShip)
dirName=CMLib.directions().getFromShipDirectionName(fromDir);
else
dirName=CMLib.directions().getFromCompassDirectionName(fromDir);
R.showHappens(CMMsg.MSG_OK_VISUAL,L("A bright light from @x1 disappears.",dirName));
}
else
{
R.showHappens(CMMsg.MSG_OK_VISUAL,L("A bright light disappears."));
}
R.delEffect(A);
A.destroy();
R.recoverPhyStats();
R.recoverRoomStats();
}
}
}
catch(final Exception e)
{
Log.errOut(e);
}
}
@Override
public void unInvoke()
{
// undo the affects of this spell
final boolean canBeUninvoked=canBeUninvoked();
final MOB invoker=invoker();
final Room room=CMLib.map().roomLocation(affected);
if(canBeUninvoked&&(room!=null)&&(affected instanceof MOB))
room.show((MOB)affected,null,CMMsg.MSG_OK_VISUAL,L("The lighthouse above <S-NAME> dims."));
super.unInvoke();
this.invoker=invoker;
if(canBeUninvoked&&(room!=null))
{
final List<Room> rooms=new LinkedList<Room>();
synchronized(roomSet)
{
rooms.addAll(roomSet);
roomSet.clear();
}
for(final Room R : rooms)
this.delFromRoom(R, null, -1);
}
if(canBeUninvoked&&(room!=null))
room.recoverRoomStats();
}
protected int nextDir(final Room R)
{
if(lastDir < 0)
lastDir=Directions.NORTH;
int dir=lastDir;
if(dir >= Directions.CODES().length)
dir=0;
else
dir++;
while(((R.getRoomInDir(dir)==null)
||(R.getExitInDir(dir)==null)
||(!R.getExitInDir(dir).isOpen()))
&&(dir != lastDir))
{
if(dir >= Directions.CODES().length)
dir=0;
else
dir++;
}
return dir;
}
@Override
public boolean tick(final Tickable ticking, final int tickID)
{
if(!super.tick(ticking, tickID))
return false;
final Room currentRoom = CMLib.map().roomLocation(affected);
if((currentRoom == null)||(!currentRoom.isHere(affected)))
{
unInvoke();
return false;
}
if(((currentRoom.domainType()&Room.INDOORS)==0)
&&(currentRoom.getArea() instanceof BoardableShip))
{
final Item shipItem = ((BoardableShip)currentRoom.getArea()).getShipItem();
final Room shipRoom = CMLib.map().roomLocation(shipItem);
if((shipRoom == null)||(!shipRoom.isHere(shipItem)))
{
unInvoke();
return false;
}
lastRoom=currentRoom;
final Set<Room> newRooms = new HashSet<Room>();
final TrackingFlags flags=CMLib.tracking().newFlags()
.plus(TrackingFlag.AREAONLY)
.plus(TrackingFlag.OPENONLY)
.plus(TrackingFlag.OUTDOORONLY);
newRooms.add(currentRoom);
final int range=10 + super.getXLEVELLevel(invoker())+(2*super.getXMAXRANGELevel(invoker()));
newRooms.addAll(CMLib.tracking().getRadiantRooms(currentRoom, flags, range));
final int prevDir=Directions.getOpDirectionCode(this.lastDir);
this.lastDir = nextDir(shipRoom);
newRooms.add(shipRoom);
if((this.lastDir>=0)
&&(shipRoom.getRoomInDir(this.lastDir)!=null)
&&(shipRoom.getExitInDir(this.lastDir)!=null)
&&(shipRoom.getExitInDir(this.lastDir).isOpen()))
newRooms.add(shipRoom.getRoomInDir(this.lastDir));
List<Room> oldRooms;
synchronized(roomSet)
{
oldRooms=new LinkedList<Room>(roomSet);
}
for(final Room R : oldRooms)
{
if(!newRooms.contains(R))
this.delFromRoom(R, shipItem, prevDir);
}
for(final Room R : newRooms)
{
if(!oldRooms.contains(R))
this.addToRoom(R, shipItem, Directions.getOpDirectionCode(this.lastDir));
}
synchronized(roomSet)
{
synchronized(roomSet)
{
this.roomSet.clear();
this.roomSet.addAll(newRooms);
}
}
}
else
{
lastRoom=currentRoom;
final List<Room> newRooms = new LinkedList<Room>();
newRooms.add(currentRoom);
final int prevDir=Directions.getOpDirectionCode(this.lastDir);
this.lastDir = nextDir(currentRoom);
newRooms.add(currentRoom);
if((this.lastDir>=0)
&&(currentRoom.getRoomInDir(this.lastDir)!=null)
&&(currentRoom.getExitInDir(this.lastDir)!=null)
&&(currentRoom.getExitInDir(this.lastDir).isOpen()))
newRooms.add(currentRoom.getRoomInDir(this.lastDir));
List<Room> oldRooms;
synchronized(roomSet)
{
oldRooms=new LinkedList<Room>(roomSet);
}
for(final Room R : oldRooms)
{
if(!newRooms.contains(R))
this.delFromRoom(R, null, prevDir);
}
for(final Room R : newRooms)
{
if(!oldRooms.contains(R))
this.addToRoom(R, null, Directions.getOpDirectionCode(this.lastDir));
}
synchronized(roomSet)
{
synchronized(roomSet)
{
this.roomSet.clear();
this.roomSet.addAll(newRooms);
}
}
}
return true;
}
@Override
public boolean invoke(final MOB mob, final List<String> commands, final Physical givenTarget, final boolean auto, final int asLevel)
{
MOB target=mob;
if((auto)&&(givenTarget!=null)&&(givenTarget instanceof MOB))
target=(MOB)givenTarget;
if(target.fetchEffect(this.ID())!=null)
{
mob.tell(target,null,null,L("<S-NAME> already <S-HAS-HAVE> a lighthouse."));
return false;
}
if(!super.invoke(mob,commands,givenTarget,auto,asLevel))
return false;
final boolean success=proficiencyCheck(mob,0,auto);
final Room room=mob.location();
if((success)&&(room!=null))
{
final CMMsg msg=CMClass.getMsg(mob,target,this,verbalCastCode(mob,target,auto),auto?L("^S<S-NAME> attain(s) a pulsing lighthouse above <S-HIS-HER> head!"):L("^S<S-NAME> invoke(s) a lighthouse above <S-HIS-HER> head!^?"));
if(room.okMessage(mob,msg))
{
room.send(mob,msg);
beneficialAffect(mob,target,asLevel,0);
room.recoverRoomStats();
}
}
else
beneficialWordsFizzle(mob,mob.location(),L("<S-NAME> attempt(s) to invoke a lighthouse, but fail(s)."));
return success;
}
}