package com.planet_ink.coffee_mud.Abilities.Common;
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 2002-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 Hunting extends GatheringSkill
{
@Override
public String ID()
{
return "Hunting";
}
public Hunting()
{
super();
displayText=L("You are hunting...");
verb=L("hunting");
}
private final static String localizedName = CMLib.lang().L("Hunting");
@Override
public String name()
{
return localizedName;
}
private static final String[] triggerStrings =I(new String[] {"HUNT","HUNTING"});
@Override
public String[] triggerStrings()
{
return triggerStrings;
}
@Override
public int classificationCode()
{
return Ability.ACODE_COMMON_SKILL|Ability.DOMAIN_GATHERINGSKILL;
}
// common recipe definition indexes
protected static final int RCP_RESOURCE= 0;
protected static final int RCP_DOMAIN = 1;
protected static final int RCP_FREQ = 2;
protected static final int RCP_MOB = 3;
protected MOB found = null;
protected String foundShortName = "";
public Room nearByRoom()
{
final List<Integer> possibilities=new ArrayList<Integer>(Directions.NUM_DIRECTIONS());
for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--)
{
if(d!=Directions.UP)
{
final Room room=activityRoom.getRoomInDir(d);
final Exit exit=activityRoom.getExitInDir(d);
if((room!=null)&&(exit!=null)&&(exit.isOpen()))
possibilities.add(Integer.valueOf(d));
}
}
if(possibilities.size()>0)
{
final int dir=possibilities.get(CMLib.dice().roll(1,possibilities.size(),-1)).intValue();
return activityRoom.getRoomInDir(dir);
}
return null;
}
public void moveFound()
{
if(found.location()==null)
return;
final List<Integer> possibilities=new ArrayList<Integer>();
for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--)
{
final Room room=found.location().getRoomInDir(d);
if((room == null)
||(CMLib.flags().isAiryRoom(room)&&(!CMLib.flags().isFlying(found))))
{
final Exit exit=found.location().getExitInDir(d);
if((exit!=null)
&&(exit.isOpen())
&&(CMLib.flags().canBreatheHere(found, room)))
possibilities.add(Integer.valueOf(d));
}
}
if(possibilities.size()>0)
{
final int dir=possibilities.get(CMLib.dice().roll(1,possibilities.size(),-1)).intValue();
CMLib.tracking().walk(found,dir,true,false);
}
}
@Override
public boolean tick(final Tickable ticking, final int tickID)
{
if((affected!=null)&&(affected instanceof MOB)&&(tickID==Tickable.TICKID_MOB))
{
final MOB mob=(MOB)affected;
activityRoom=mob.location();
if((found!=null)&&(found.amDead()))
{
found.setLocation(null);
found.destroy();
unInvoke();
}
else
if((found!=null)
&&(found.location()!=null)
&&(CMLib.flags().isAliveAwakeMobile(found,true))
&&(!found.isInCombat()))
{
if(found.location()==mob.location())
{
if((mob.isMonster())
&&(CMLib.flags().isAliveAwakeMobile(mob,true))
&&(CMLib.flags().canBeSeenBy(found,mob))
&&(!mob.isInCombat()))
CMLib.combat().postAttack(mob,found,mob.fetchWieldedItem());
else
moveFound();
}
}
if(tickUp==0)
{
if(found!=null)
{
super.adjustYieldBasedOnRoomSpam(1, mob.location());
if(CMLib.flags().isWateryRoom(mob.location()))
commonTell(mob,L("You have found some @x1 signs!",foundShortName));
else
commonTell(mob,L("You have found some @x1 tracks!",foundShortName));
commonTell(mob,L("You need to find the @x1 nearby before the trail goes cold!",foundShortName));
displayText=L("You are hunting for @x1",found.name());
verb=L("hunting for @x1",found.name());
found.basePhyStats().setLevel(mob.basePhyStats().level());
found.recoverPhyStats();
final Ability A=CMClass.getAbility("Prop_ModExperience");
A.setMiscText("=20%");
found.addNonUninvokableEffect(A);
found.bringToLife(nearByRoom(),true);
CMLib.beanCounter().clearZeroMoney(found,null);
found.setMoneyVariation(0);
}
else
{
final StringBuffer str=new StringBuffer(L("You can't seem to find any game around here.\n\r"));
final int d=lookingForMat(RawMaterial.MATERIAL_FLESH,mob.location());
if(d<0)
str.append(L("You might try elsewhere."));
else
str.append(L("You might try @x1.",CMLib.directions().getInDirectionName(d)));
commonTell(mob,str.toString());
unInvoke();
}
}
else
if(mob.isMonster()
&&(CMLib.dice().rollPercentage()>50)
&&(CMLib.flags().isMobile(mob))
&&(CMLib.flags().isAliveAwakeMobile(mob,true))
&&(CMLib.flags().canSenseEnteringLeaving(found,mob)))
{
for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--)
{
final Room R=mob.location().getRoomInDir(d);
if((R!=null)&&(R==found.location()))
{
CMLib.tracking().walk(mob,d,false,false);
break;
}
}
}
}
return super.tick(ticking,tickID);
}
@Override
public void unInvoke()
{
if(canBeUninvoked())
{
if(affected instanceof MOB)
{
final MOB mob=(MOB)affected;
if((found!=null)&&(!found.amDead())&&(found.location()!=null)&&(!found.isInCombat()))
{
if(found.location()==mob.location())
moveFound();
found.location().delInhabitant(found);
found.setLocation(null);
found.destroy();
mob.location().show(mob,null,getActivityMessageType(),L("<S-NAME> <S-HAS-HAVE> lost the trail."));
}
}
}
super.unInvoke();
}
@Override
public boolean invoke(final MOB mob, final List<String> commands, final Physical givenTarget, final boolean auto, final int asLevel)
{
if(super.checkStop(mob, commands))
return true;
verb=L("hunting");
found=null;
activityRoom=null;
final Room R=mob.location();
if(R==null)
return false;
if(!super.invoke(mob,commands,givenTarget,auto,asLevel))
return false;
int resourceType=R.myResource();
if((resourceType&RawMaterial.RESOURCE_MASK)>=RawMaterial.CODES.NAMES().length)
{
resourceType=0;
Log.errOut("Hunting","Room "+CMLib.map().getExtendedRoomID(R)+" had resource code "+R.myResource());
}
final String resourceName = RawMaterial.CODES.NAME(resourceType).toUpperCase();
final int domainType = R.domainType()&(~Room.INDOORS);
final String domainName;
if((R.domainType()&Room.INDOORS)==0)
domainName = (domainType < Room.DOMAIN_OUTDOOR_DESCS.length) ? Room.DOMAIN_OUTDOOR_DESCS[domainType].toUpperCase() : "";
else
domainName = (domainType < Room.DOMAIN_INDOORS_DESCS.length) ? Room.DOMAIN_INDOORS_DESCS[domainType].toUpperCase() : "";
if((proficiencyCheck(mob,0,auto))
&&(super.checkIfAnyYield(mob.location()))
&&(nearByRoom()!=null))
{
final List<List<String>> recipes = loadRecipes("hunting.txt");
if((recipes != null)
&&(recipes.size()>0))
{
int totalWeights = 0;
final List<List<String>> subset=new ArrayList<List<String>>();
for(final List<String> subl : recipes)
{
if(subl.size()<4)
continue;
if(subl.get(RCP_RESOURCE).toUpperCase().equals(resourceName)
&&((subl.get(RCP_DOMAIN).equals("*"))
||(subl.get(RCP_DOMAIN).equals(domainName))
||(subl.get(RCP_DOMAIN).equals(R.ID()))))
{
subset.add(subl);
totalWeights += CMath.s_int(subl.get(RCP_FREQ));
}
}
List<String> winl = null;
if(totalWeights > 0)
{
final int winner=CMLib.dice().roll(1, totalWeights, -1);
int current = 0;
for(final List<String> subl : subset)
{
current += CMath.s_int(subl.get(RCP_FREQ));
if(winner < current)
{
winl = subl;
break;
}
}
if(winl == null)
winl=subset.get(subset.size()-1);
}
if(winl != null)
{
final String mobID = winl.get(RCP_MOB);
MOB stdM = CMClass.getMOB(mobID);
if(stdM == null)
stdM = CMLib.catalog().getCatalogMob(mobID);
if(stdM == null)
{
commonTell(mob,L("There are no signs of life here."));
return false;
}
MOB genM=stdM;
if(!genM.isGeneric())
{
if(genM.baseCharStats().getMyRace().useRideClass())
genM=CMClass.getMOB("GenRideable");
else
genM=CMClass.getMOB("GenMob");
genM.setBaseCharStats((CharStats)stdM.baseCharStats().copyOf());
genM.setBasePhyStats((PhyStats)stdM.basePhyStats().copyOf());
genM.setBaseState((CharState)stdM.baseState().copyOf());
for(final GenericBuilder.GenMOBCode stat : GenericBuilder.GenMOBCode.values())
{
if(stat != GenericBuilder.GenMOBCode.ABILITY) // because this screws up gen hit points
genM.setStat(stat.name(), CMLib.coffeeMaker().getGenMobStat(stdM,stat.name()));
}
genM.basePhyStats().setRejuv(0);
genM.recoverCharStats();
genM.recoverPhyStats();
genM.recoverMaxState();
genM.resetToMaxState();
}
final int moblevel = Math.round(CMath.sqrt(mob.phyStats().level())); // pity on artisans
final int addlevel = (moblevel < CMProps.getIntVar(CMProps.Int.EXPRATE)) ? 0 : (moblevel - CMProps.getIntVar(CMProps.Int.EXPRATE));
genM.basePhyStats().setLevel(genM.basePhyStats().level() + addlevel);
genM.basePhyStats().setRejuv(0);
genM.recoverCharStats();
genM.recoverPhyStats();
genM.recoverMaxState();
genM.resetToMaxState();
CMLib.leveler().fillOutMOB(genM,genM.basePhyStats().level());
genM.basePhyStats().setRejuv(0);
genM.recoverPhyStats();
found=genM;
foundShortName=found.name();
int x=0;
if((x=foundShortName.lastIndexOf(' '))>=0)
foundShortName=foundShortName.substring(x).trim().toLowerCase();
found.setLocation(null);
if(found instanceof Rideable)
{
if((CMLib.flags().canBreatheThis(found, RawMaterial.RESOURCE_FRESHWATER)
|| CMLib.flags().canBreatheThis(found, RawMaterial.RESOURCE_SALTWATER))
&&(!CMLib.flags().canBreatheThis(found, RawMaterial.RESOURCE_AIR)))
((Rideable)found).setRideBasis(Rideable.RIDEABLE_WATER);
else
((Rideable)found).setRideBasis(Rideable.RIDEABLE_LAND);
}
}
}
}
final int duration=10+mob.phyStats().level()+(super.getXTIMELevel(mob)*2);
final CMMsg msg=CMClass.getMsg(mob,found,this,getActivityMessageType(),L("<S-NAME> start(s) hunting."));
if(R.okMessage(mob,msg))
{
R.send(mob,msg);
found=(MOB)msg.target();
beneficialAffect(mob,mob,asLevel,duration);
}
return true;
}
}