/
com/planet_ink/coffee_mud/Abilities/
com/planet_ink/coffee_mud/Abilities/Common/
com/planet_ink/coffee_mud/Abilities/Diseases/
com/planet_ink/coffee_mud/Abilities/Druid/
com/planet_ink/coffee_mud/Abilities/Fighter/
com/planet_ink/coffee_mud/Abilities/Prayers/
com/planet_ink/coffee_mud/Abilities/Properties/
com/planet_ink/coffee_mud/Abilities/Skills/
com/planet_ink/coffee_mud/Abilities/Songs/
com/planet_ink/coffee_mud/Abilities/Spells/
com/planet_ink/coffee_mud/Abilities/Thief/
com/planet_ink/coffee_mud/Abilities/Traps/
com/planet_ink/coffee_mud/Areas/interfaces/
com/planet_ink/coffee_mud/Behaviors/
com/planet_ink/coffee_mud/CharClasses/interfaces/
com/planet_ink/coffee_mud/Commands/
com/planet_ink/coffee_mud/Commands/interfaces/
com/planet_ink/coffee_mud/Exits/interfaces/
com/planet_ink/coffee_mud/Items/Armor/
com/planet_ink/coffee_mud/Items/Basic/
com/planet_ink/coffee_mud/Items/MiscMagic/
com/planet_ink/coffee_mud/Items/Software/
com/planet_ink/coffee_mud/Items/Weapons/
com/planet_ink/coffee_mud/Libraries/interfaces/
com/planet_ink/coffee_mud/Locales/
com/planet_ink/coffee_mud/Locales/interfaces/
com/planet_ink/coffee_mud/MOBS/
com/planet_ink/coffee_mud/MOBS/interfaces/
com/planet_ink/coffee_mud/Races/
com/planet_ink/coffee_mud/Races/interfaces/
com/planet_ink/coffee_mud/WebMacros/
com/planet_ink/coffee_mud/WebMacros/interfaces/
com/planet_ink/coffee_mud/application/
com/planet_ink/coffee_mud/core/smtp/
com/planet_ink/siplet/applet/
lib/
resources/examples/
resources/fakedb/
resources/quests/delivery/
resources/quests/diseased/
resources/quests/drowning/
resources/quests/gobwar/
resources/quests/holidays/
resources/quests/robbed/
resources/quests/smurfocide/
resources/quests/stolen/
resources/quests/templates/
resources/quests/treasurehunt/
resources/quests/vengeance/
web/
web/admin.templates/
web/admin/images/
web/pub.templates/
web/pub/images/mxp/
web/pub/sounds/
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.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.Locales.interfaces.*;
import com.planet_ink.coffee_mud.MOBS.interfaces.*;
import com.planet_ink.coffee_mud.Races.interfaces.*;



import java.util.*;
import java.io.IOException;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;

/*
   Copyright 2000-2006 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 Arrest extends StdBehavior implements LegalBehavior
{
	public String ID(){return "Arrest";}
	public long flags(){return Behavior.FLAG_LEGALBEHAVIOR;}
	protected int canImproveCode(){return Behavior.CAN_AREAS;}

	protected boolean loadAttempt=false;

	protected String getLawParms(){ return getParms();}
    protected Hashtable finesAssessed=new Hashtable();
    public boolean isFullyControlledByClan(){return true;}

    public boolean frame(Area myArea, MOB accused, MOB framed)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        LegalWarrant W=null;
        if(laws!=null)
            for(int i=0;(W=laws.getWarrant(accused,i))!=null;i++)
                if(W.criminal()==accused)
                {
                    W.setCriminal(framed);
                    return true;
                }
        return false;
    }
    public boolean arrest(Area myArea, MOB officer, MOB accused)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        LegalWarrant W=(laws!=null)?laws.getWarrant(accused,0):null;
        if((W!=null)&&((W.arrestingOfficer()==null)||(W.arrestingOfficer().location()!=accused.location())))
        {
            W.setArrestingOfficer(myArea,officer);
            CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),"You are under arrest "+restOfCharges(laws,W.criminal())+"! Sit down on the ground immediately!",false,false);
            W.setState(Law.STATE_ARRESTING);
            return true;
        }
        return false;
    }
    public Vector warrantInfo(Area myArea)
    {
        Vector V=new Vector();
        if(!theLawIsEnabled()) return V;
        Law laws=getLaws(myArea,false);
        boolean debugging=CMSecurity.isDebugging("ARREST");
        if(laws!=null)
            for(int i=0;i<laws.warrants().size();i++)
            {
                LegalWarrant W=(LegalWarrant)laws.warrants().elementAt(i);
                if(isStillACrime(W,debugging))
                {
                    Vector V2=new Vector();
                    V2.addElement(W.criminal().name());
                    if(W.victim()==null) V2.addElement("");
                    else V2.addElement(W.victim().name());
                    if(W.witness()==null) V2.addElement("");
                    else V2.addElement(W.witness().name());
                    V2.addElement(fixCharge(W));
                    V.addElement(V2);
                }
            }
        return V;
    }
    
    public int revoltChance(){return 0;}
    public Law legalInfo(Area myArea)
    {
        if(!theLawIsEnabled()) return null;
        return getLaws(myArea,false);
    }
    public boolean isElligibleOfficer(Area myArea, MOB mob)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if((mob.isMonster())
        &&(mob.location()!=null)
        &&(laws!=null)
        &&(isElligibleOfficer(laws,mob,mob.location().getArea())))
            return true;
        return false;
    }
    public boolean hasWarrant(Area myArea, MOB accused)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        return (laws!=null)?((laws.getWarrant(accused,0))!=null):false;
    }
    public boolean isAnyOfficer(Area myArea, MOB mob)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if((mob.isMonster())
        &&(mob.location()!=null)
        &&(laws!=null)
        &&(isAnyKindOfOfficer(laws,mob)))
            return true;
        return false;
    }
    public boolean isJudge(Area myArea, MOB mob)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if((mob.isMonster())
        &&(mob.location()!=null)
        &&(laws!=null)
        &&(isTheJudge(laws,mob)))
            return true;
        return false;
    }
    public void modifyAssessedFines(double d, MOB mob)
    {
        Double D=(Double)finesAssessed.get(mob);
        if(D!=null) finesAssessed.remove(mob);
        if(d>0) finesAssessed.put(mob,new Double(d));
    }
        
    public double finesOwed(MOB mob)
    {
        if(!theLawIsEnabled()) return 0.0;
        Double D=(Double)finesAssessed.get(mob);
        if(D!=null) return D.doubleValue();
        return 0.0;
    }
    
    public boolean updateLaw(Area myArea)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if(laws!=null)
        {
            laws.resetLaw();
            if(getLawParms().equalsIgnoreCase("custom")
            &&(myArea!=null))
            {
                CMLib.database().DBReCreateData(myArea.Name(),"ARREST",myArea.Name()+"/ARREST",laws.rawLawString());
                return true;
            }
        }
        return false;
    }
    public String rulingClan()
    {
        return "";
    }
    public String conquestInfo(Area myArea)
    {
        return "";
    }
    public int controlPoints()
    {
        return 0;
    }
	public void setControlPoints(String clanID, int newControlPoints){}
	
	public Vector getWarrantsOf(Area myArea, String name)
    {
        Vector V=new Vector();
        if(!theLawIsEnabled()) return V;
        Law laws=getLaws(myArea,false);
        boolean debugging=CMSecurity.isDebugging("ARREST");
        for(int i=0;i<laws.warrants().size();i++)
        {
            LegalWarrant W=(LegalWarrant)laws.warrants().elementAt(i);
            if((isStillACrime(W,debugging))
            &&((name==null)||(CMLib.english().containsString(W.criminal().name(),name))))
            {
                W.setLastOffense(System.currentTimeMillis()+EXPIRATION_MILLIS+10);
                V.addElement(W);
            }
        }
        return V;
    }
    public Vector getWarrantsOf(Area myArea, MOB accused)
    {
        Vector V=new Vector();
        if(!theLawIsEnabled()) return V;
        Law laws=getLaws(myArea,false);
        boolean debugging=CMSecurity.isDebugging("ARREST");
        for(int i=0;i<laws.warrants().size();i++)
        {
            LegalWarrant W=(LegalWarrant)laws.warrants().elementAt(i);
            if((isStillACrime(W,debugging))&&((accused==null)||(W.criminal()==accused)))
                V.addElement(W);
        }
        return V;
    }
    
    public boolean addWarrant(Law laws, LegalWarrant W)
    {
        if(!theLawIsEnabled()) return false;
        if((laws!=null)&&(!laws.warrants().contains(W)))
        {
            laws.warrants().addElement(W);
        	if(W.criminal()!=null)
        	{
	            Vector channels=CMLib.channels().getFlaggedChannelNames("WARRANTS");
	            for(int i=0;i<channels.size();i++)
	                CMLib.commands().postChannel((String)channels.elementAt(i),"",W.criminal().name()+" has been accused of "+fixCharge(W)+".",true);
        	}
            return true;
        }
        return false;
    }
    public boolean addWarrant(Area myArea, LegalWarrant W)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        return addWarrant(laws,W);
    }
    public boolean addWarrant(Area myArea, MOB accused, MOB victim, String crimeLocs, String crimeFlags, String crime, String sentence, String warnMsg)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if(laws!=null)
            return fillOutWarrant(accused,laws,myArea,victim,crimeLocs,crimeFlags,crime,sentence,warnMsg);
        return false;
    }
    public boolean deleteWarrant(Area myArea, LegalWarrant W)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if((laws!=null)&&(laws.warrants().contains(W)))
        {
            laws.warrants().removeElement(W);
            return true;
        }
        return false;
    }
    public boolean aquit(Area myArea, MOB accused, Vector lawStrings)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if(laws!=null)
        {
            boolean debugging=CMSecurity.isDebugging("ARREST");
            String[] info=null;
            for(int v=0;v<lawStrings.size();v++)
            {
                String brokenLaw=(String)lawStrings.elementAt(v);
                if((laws.basicCrimes().containsKey(brokenLaw))&&(laws.basicCrimes().get(brokenLaw) instanceof String[]))
                {   info=(String[])laws.basicCrimes().get(brokenLaw);   break; }
                else
                if((laws.taxLaws().containsKey(brokenLaw))&&(laws.taxLaws().get(brokenLaw) instanceof String[]))
                {   info=(String[])laws.taxLaws().get(brokenLaw);   break; }
                else
                if((laws.abilityCrimes().containsKey(brokenLaw))&&(laws.abilityCrimes().get(brokenLaw) instanceof String[]))
                {   info=(String[])laws.abilityCrimes().get(brokenLaw);   break; }
            }
            if(info==null) return false;
            for(int i=0;i<laws.warrants().size();i++)
            {
                LegalWarrant W=(LegalWarrant)laws.warrants().elementAt(i);
                if((isStillACrime(W,debugging))
                &&(W.criminal()==accused)
                &&(W.crime().equalsIgnoreCase(info[Law.BIT_CRIMENAME])))
                {
                    laws.warrants().removeElement(W);
                    return true;
                }
            }
        }
        return false;
    }
    public boolean isJailRoom(Area myArea, Vector jails)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if(laws!=null)
        {
            Vector rooms=getRooms(myArea,laws.jailRooms());
            boolean answer=false;
            for(int i=0;i<jails.size();i++)
                answer=answer||rooms.contains(jails.elementAt(i));
            return answer;
        }
        return false;
    }
    public boolean accuse(Area myArea, MOB accused, MOB victim, Vector lawStrings)
    {
        if(!theLawIsEnabled()) return false;
        Law laws=getLaws(myArea,false);
        if(laws!=null)
        {
            String[] info=null;
            for(int v=0;v<lawStrings.size();v++)
            {
                String brokenLaw=(String)lawStrings.elementAt(v);
                if((laws.basicCrimes().containsKey(brokenLaw))&&(laws.basicCrimes().get(brokenLaw) instanceof String[]))
                    info=(String[])laws.basicCrimes().get(brokenLaw);
                else
                if((laws.taxLaws().containsKey(brokenLaw))&&(laws.taxLaws().get(brokenLaw) instanceof String[]))
                    info=(String[])laws.taxLaws().get(brokenLaw);
                else
                if((laws.abilityCrimes().containsKey(brokenLaw))&&(laws.abilityCrimes().get(brokenLaw) instanceof String[]))
                    info=(String[])laws.abilityCrimes().get(brokenLaw);
                if(info!=null)
                {
                    if((info[Law.BIT_CRIMENAME]!=null)
                    &&(info[Law.BIT_CRIMENAME].length()>0))
                        break;
                    info=null;
                }
            }
            if(info==null) return false;
            fillOutWarrant(accused,
                            laws,
                            myArea,
                            (victim==accused)?null:victim,
                            info[Law.BIT_CRIMELOCS],
                            info[Law.BIT_CRIMEFLAGS],
                            info[Law.BIT_CRIMENAME],
                            info[Law.BIT_SENTENCE],
                            info[Law.BIT_WARNMSG]);
            return true;
        }
        return false;
    }

	public void setParms(String newParms)
	{
		super.setParms(newParms);
		loadAttempt=false;
	}

	protected boolean defaultModifiableNames(){return true;}

	protected Law getLaws(Environmental what, boolean cleanOnly)
	{
		String lawName=getLawParms();

		boolean modifiableLaw=false;
		boolean modifiableNames=defaultModifiableNames();

		Law laws=null;
		if((lawName.equalsIgnoreCase("custom"))&&(what!=null))
		{
			modifiableLaw=true;
			laws=(Law)Resources.getResource("LEGAL-"+what.Name());
		}
		else
		{
			if(lawName.length()==0)
				lawName="laws.ini";
			laws=(Law)Resources.getResource("LEGAL-"+lawName);
			modifiableNames=false;
		}
		if((laws==null)&&(cleanOnly)) return null;

		if(laws==null)
		{
			Properties lawprops=new Properties();
			try
			{
				if((lawName.equalsIgnoreCase("custom"))&&(what!=null))
				{
					Vector data=CMLib.database().DBReadData(what.Name(),"ARREST",what.Name()+"/ARREST");
					if((data!=null)&&(data.size()>0))
					{
						data=(Vector)data.firstElement();
						if((data!=null)&&(data.size()>0))
						{
							String s=CMStrings.replaceAll((String)data.elementAt(3),"~","\n");
							s=CMStrings.replaceAll(s,"`","'");
							lawprops.load(new ByteArrayInputStream(CMStrings.strToBytes(s)));
						}
						else
						{
							String s=Law.defaultLaw;
							lawprops.load(new ByteArrayInputStream(CMStrings.strToBytes(s)));
							s=CMStrings.replaceAll(s,"\n","~");
							s=CMStrings.replaceAll(s,"'","`");
							CMLib.database().DBCreateData(what.Name(),"ARREST",what.Name()+"/ARREST",s);
						}
					}
					else
					{
						String s=Law.defaultLaw;
						lawprops.load(new ByteArrayInputStream(CMStrings.strToBytes(s)));
						s=CMStrings.replaceAll(s,"\n","~");
						s=CMStrings.replaceAll(s,"\r","~");
						s=CMStrings.replaceAll(s,"'","`");
						CMLib.database().DBCreateData(what.Name(),"ARREST",what.Name()+"/ARREST",s);
					}
				}
				if(lawprops.isEmpty())
                    lawprops.load(new ByteArrayInputStream(new CMFile(Resources.makeFileResourceName(lawName),null,false).raw()));
			}
			catch(IOException e)
			{
				if(!loadAttempt)
				{
					Log.errOut("Arrest","Unable to load: "+lawName+", legal system inoperable.");
					loadAttempt=true;
				}
				return (Law)CMClass.getCommon("DefaultLawSet");
			}
			loadAttempt=true;
            laws=(Law)CMClass.getCommon("DefaultLawSet");
			laws.initialize(this,lawprops,modifiableNames,modifiableLaw);
			if(lawName.equalsIgnoreCase("custom")&&(what!=null))
				Resources.submitResource("LEGAL-"+what.name(),laws);
			else
				Resources.submitResource("LEGAL-"+lawName,laws);
		}
		return laws;
	}

	public void unCuff(MOB mob)
	{
		Ability A=mob.fetchEffect("Skill_HandCuff");
		if(A!=null) A.unInvoke();
	}


	public void dismissOfficer(MOB officer)
	{
		if(officer==null) return;
		if((officer.getStartRoom()!=null)
		&&(officer.location()!=null)
		&&(officer.getStartRoom()==officer.location()))
			return;
		if(officer.isMonster())
			CMLib.tracking().wanderAway(officer,true,true);
	}

	public MOB getAWitnessHere(Room R, MOB accused)
	{
		if(R!=null)
		for(int i=0;i<R.numInhabitants();i++)
		{
			MOB M=R.fetchInhabitant(i);
			if(M.isMonster()
			&&(M!=accused)
			&&(M.charStats().getStat(CharStats.STAT_INTELLIGENCE)>3)
			&&(CMLib.dice().rollPercentage()<=(CMLib.flags().isEvil(accused)?25:(CMLib.flags().isGood(accused)?95:50))))
				return M;
		}
		return null;
	}

	public MOB getWitness(Area A, MOB accused)
	{
		Room R=accused.location();

		if((A!=null)&&(!A.inMyMetroArea(R.getArea())))
			return null;
		MOB M=getAWitnessHere(R,accused);
		if(M!=null) return M;

		if(R!=null)
		for(int i=0;i<Directions.NUM_DIRECTIONS;i++)
		{
			Room R2=R.getRoomInDir(i);
			M=getAWitnessHere(R2,accused);
			if(M!=null) return M;
		}
		return null;
	}

	public boolean isAnyKindOfOfficer(Law laws, MOB M)
	{
		if((M.isMonster())
		&&(M.location()!=null)
		&&(CMLib.flags().isMobile(M)))
		{
			if((laws.officerNames().size()<=0)
			||(((String)laws.officerNames().firstElement()).equals("@")))
			   return false;
			for(int i=0;i<laws.officerNames().size();i++)
				if((CMLib.english().containsString(M.displayText(),(String)laws.officerNames().elementAt(i))
				||(CMLib.english().containsString(M.Name(),(String)laws.officerNames().elementAt(i)))))
					return true;
		}
		return false;
	}

	public boolean isElligibleOfficer(Law laws, MOB M, Area myArea)
	{
		if((M!=null)&&(M.isMonster())&&(M.location()!=null))
		{
			if((myArea!=null)&&(!myArea.inMyMetroArea(M.location().getArea()))) return false;

			if(isAnyKindOfOfficer(laws,M)
			&&(!isBusyWithJustice(laws,M))
			&&(CMLib.flags().aliveAwakeMobile(M,true))
			&&(!M.isInCombat()))
				return true;
		}
		return false;
	}

	public MOB getElligibleOfficerHere(Law laws,
									   Area myArea,
									   Room R,
									   MOB criminal,
									   MOB victim)
	{
		if(R==null) return null;
		for(int i=0;i<R.numInhabitants();i++)
		{
			MOB M=R.fetchInhabitant(i);
			if((M!=null)
			&&(M!=criminal)
			&&(M.location()!=null)
			&&(myArea.inMyMetroArea(M.location().getArea()))
			&&((victim==null)||(M!=victim))
			&&(isElligibleOfficer(laws,M,myArea))
			&&(CMLib.flags().canBeSeenBy(criminal,M)))
				return M;
		}
		return null;
	}

	public MOB getAnyElligibleOfficer(Law laws,
									  Area myArea,
									  MOB criminal,
									  MOB victim)
	{
		Room R=criminal.location();
		if(R==null) return null;
		if((myArea!=null)&&(!myArea.inMyMetroArea(R.getArea()))) return null;
		MOB M=getElligibleOfficerHere(laws,myArea,R,criminal,victim);
		if(M==null)
			for(Enumeration e=myArea.getMetroMap();e.hasMoreElements();)
			{
				Room R2=(Room)e.nextElement();
				M=getElligibleOfficerHere(laws,myArea,R2,criminal,victim);
				if(M!=null) break;
			}
		return M;
	}

	public MOB getElligibleOfficer(Law laws,
								   Area myArea,
								   MOB criminal,
								   MOB victim)
	{
		Room R=criminal.location();
		if(R==null) return null;
		if((myArea!=null)&&(!myArea.inMyMetroArea(R.getArea()))) return null;
		MOB M=getElligibleOfficerHere(laws,myArea,R,criminal,victim);
		if(M!=null) return M;
		for(int d=0;d<Directions.NUM_DIRECTIONS;d++)
		{
			Room R2=R.getRoomInDir(d);
			if(R2!=null)
			{
				M=getElligibleOfficerHere(laws,myArea,R2,criminal,victim);
				if(M!=null)
				{
					int direction=Directions.getOpDirectionCode(d);
					CMLib.tracking().move(M,direction,false,false);
					if(M.location()==R) return M;
				}
			}
		}
		return null;
	}

	public boolean canFocusOn(MOB officer, MOB criminal)
	{
		CMMsg msg=CMClass.getMsg(officer,criminal,CMMsg.MSG_LOOK,"<S-NAME> look(s) closely at <T-NAME>.");
		if((officer!=null)&&(officer.location()!=null)&&(criminal.location()==officer.location()))
		{
			if(!officer.location().okMessage(officer,msg))
				return false;
			if(msg.sourceMessage().indexOf("<T-NAME>")<0)
				return false;
			if((criminal.name().toUpperCase().equals(criminal.Name().toUpperCase()))
		        ||(criminal.name().toUpperCase().startsWith(criminal.Name().toUpperCase()+" "))
		        ||(criminal.name().toUpperCase().endsWith(" "+criminal.Name().toUpperCase())))
			    return true;
		}
		return true;
	}

	public boolean isStillACrime(LegalWarrant W, boolean debugging)
	{
		// will witness talk, or victim press charges?
		HashSet H=W.criminal().getGroupMembers(new HashSet());
		if((W.witness()!=null)&&W.witness().amDead()) 
	    {
		    if(debugging) Log.debugOut("ARREST", "Witness is DEAD!");
		    return false;
	    }
		if(W.arrestingOfficer()!=null)
		{
			if(W.witness()==W.arrestingOfficer())
				return true;
			if((W.victim()!=null)&&(W.victim()==W.arrestingOfficer()))
				return true;
		}

		if((W.witness()!=null)&&H.contains(W.witness())) 
	    {
		    if(debugging) Log.debugOut("ARREST", "Witness is a friend of the accused!");
		    return false;
	    }
		if((W.victim()!=null)&&(H.contains(W.victim()))) 
	    {
		    if(debugging) Log.debugOut("ARREST", "Victim is a friend of the accused!");
		    return false;
	    }
		// crimes expire after three real days
		if((W.lastOffense()>0)&&((System.currentTimeMillis()-W.lastOffense())>EXPIRATION_MILLIS))
	    {
		    if(debugging) Log.debugOut("ARREST","Crime has expired: "+W.lastOffense());
		    return false;
	    }
		return true;
	}

    public Vector getRelevantWarrants(Vector warrants, LegalWarrant W, MOB criminal)
    {
        Vector V=new Vector();
        if(W!=null) V.addElement(W);
        for(int w2=0;w2<warrants.size();w2++)
        {
            LegalWarrant W2=(LegalWarrant)warrants.elementAt(w2);
            if((W2.criminal()==criminal)
            &&(W2!=W)
            &&((W==null)
                ||(W2.crime()==null)
                ||(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE))
                ||(W2.crime().equalsIgnoreCase(W.crime()))))
                V.addElement(W2);
        }
        return V;
    }
    
    public double getFine(Law laws, LegalWarrant W, MOB criminal)
    {
        String s=null;
        if(CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE))
        {
            s=W.getActionParm(Law.ACTIONMASK_FINE);
            if((s==null)||(s.length()==0)||(!CMath.isNumber(s))) return 0;
            return CMath.s_double(s);
        }
        double fine=0.0;
        Vector V=getRelevantWarrants(laws.warrants(),W,criminal);
        for(int w2=0;w2<V.size();w2++)
        {
            LegalWarrant W2=(LegalWarrant)V.elementAt(w2);
            if(!CMath.bset(W2.actionCode(),Law.ACTIONMASK_SEPARATE))
            {
                s=W.getActionParm(Law.ACTIONMASK_FINE);
                if((s!=null)&&(s.length()>0)&&(CMath.isNumber(s)))
                    fine+=CMath.s_double(s);
            }
        }
        return fine;
    }
    protected String getDetainParm(Law laws, LegalWarrant W, MOB criminal)
    {
        String s=null;
        if(CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE))
        {
            s=W.getActionParm(Law.ACTIONMASK_DETAIN);
            if((s==null)||(s.length()==0)) return ""; 
        }
        s=W.getActionParm(Law.ACTIONMASK_DETAIN);
        if((s==null)||(s.length()==0))
        {
            Vector V=getRelevantWarrants(laws.warrants(),W,criminal);
            for(int w2=0;w2<V.size();w2++)
            {
                LegalWarrant W2=(LegalWarrant)V.elementAt(w2);
                if(!CMath.bset(W2.actionCode(),Law.ACTIONMASK_SEPARATE))
                {
                    s=W.getActionParm(Law.ACTIONMASK_DETAIN);
                    if((s!=null)&&(s.length()>0))
                        break;
                }
            }
        }
        if(s!=null) 
        {
            return s;
        }
        return "";
    }
    protected String getDetainRoom(Law laws, LegalWarrant W, MOB criminal)
    {
        String s=getDetainParm(laws,W,criminal);
        if((s==null)||(s.length()==0)) return "";
        int x=s.indexOf(",");
        if((x<0)||(!CMath.isInteger(s.substring(x+1)))) return s;
        return s.substring(0,x);
    }
    protected int getDetainTime(Law laws, LegalWarrant W, MOB criminal)
    {
        String s=getDetainParm(laws,W,criminal);
        if((s==null)||(s.length()==0)) return -1;
        int x=s.indexOf(",");
        if((x<0)||(!CMath.isInteger(s.substring(x+1)))) 
            return laws.jailTimes()[0].intValue();
        return CMath.s_int(s.substring(x+1));
    }
	public int highestCrimeAction(Law laws, LegalWarrant W, MOB criminal)
	{
		int highest=0;
        if(CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE))
            return W.actionCode();
        Vector V=getRelevantWarrants(laws.warrants(),W,criminal);
		for(int w2=0;w2<V.size();w2++)
		{
			LegalWarrant W2=(LegalWarrant)V.elementAt(w2);
            if(!CMath.bset(W2.actionCode(),Law.ACTIONMASK_SEPARATE))
            {
    			if(((W2.actionCode()&Law.ACTION_MASK)+W2.offenses())>(highest&Law.ACTION_MASK))
    				highest=(W2.actionCode()&Law.ACTION_MASK)+((W2.offenses()<4)?W2.offenses():3);
            }
		}
        for(int w2=0;w2<V.size();w2++)
        {
            LegalWarrant W2=(LegalWarrant)V.elementAt(w2);
            if((!CMath.bset(W2.actionCode(),Law.ACTIONMASK_SEPARATE))
            &&(highest<((W2.actionCode()&Law.ACTION_MASK)+4)))
                highest++;
        }
		if(highest>Law.ACTION_HIGHEST) highest=Law.ACTION_HIGHEST;
		int adjusted=highest;
		if((CMLib.flags().isGood(criminal))&&(adjusted>0))
			adjusted--;
		return adjusted;
	}

	public boolean isBusyWithJustice(Law laws, MOB M)
	{
		for(int w=0;w<laws.warrants().size();w++)
		{
			LegalWarrant W=(LegalWarrant)laws.warrants().elementAt(w);
			if(W.arrestingOfficer()!=null)
			{
				if(W.criminal()==M) return true;
				else
				if(W.arrestingOfficer()==M) return true;
			}
		}
		return false;
	}

	public String fixCharge(LegalWarrant W)
	{
		if(W==null) return "";
		String charge=W.crime();
		if(W.victim()==null) return charge;
		if(charge.indexOf("<T-NAME>")<0) return charge;
		return charge.replaceFirst("<T-NAME>",W.victim().name());
	}

	public String restOfCharges(Law laws, MOB mob)
	{
		StringBuffer msg=new StringBuffer("");
		for(int w=0;(laws.getWarrant(mob,w)!=null);w++)
		{
			LegalWarrant W=laws.getWarrant(mob,w);
			if(W!=null)
			{
				if(w==0)
					msg.append("for "+fixCharge(W));
				else
				if(laws.getWarrant(mob,w+1)==null)
					msg.append(", and for "+fixCharge(W));
				else
					msg.append(", for "+fixCharge(W));
			}
		}
		return msg.toString();
	}

	public void makePeace(Room R)
	{
		if(R==null) return;
		for(int i=0;i<R.numInhabitants();i++)
		{
			MOB inhab=R.fetchInhabitant(i);
			if((inhab!=null)&&(inhab.isInCombat()))
				inhab.makePeace();
		}
	}

	public boolean isTheJudge(Law laws, MOB M)
	{
		if(((M.isMonster()||M.soulMate()!=null))
		&&(!CMLib.flags().isMobile(M))
		&&(M.location()!=null))
		{
			if((laws.judgeNames().size()<=0)||(((String)laws.judgeNames().firstElement()).equals("@")))
				return false;
			for(int i=0;i<laws.judgeNames().size();i++)
			{
				if((CMLib.english().containsString(M.displayText(),(String)laws.judgeNames().elementAt(i)))
				||(CMLib.english().containsString(M.Name(),(String)laws.judgeNames().elementAt(i))))
					return true;
			}
		}
		return false;
	}

	public MOB getTheJudgeHere(Law laws, Room R)
	{
		for(int i=0;i<R.numInhabitants();i++)
		{
			MOB M=R.fetchInhabitant(i);
			if(isTheJudge(laws,M))
				return M;
		}
		return null;
	}

	public Room findTheJudge(Law laws, Area myArea)
	{
		for(Enumeration r=myArea.getMetroMap();r.hasMoreElements();)
		{
			Room R=(Room)r.nextElement();
			for(int i=0;i<R.numInhabitants();i++)
			{
				MOB M=R.fetchInhabitant(i);
				if(isTheJudge(laws,M))
					return R;
			}
		}
		return null;
	}

	public boolean trackTheJudge(MOB officer, Area myArea, Law laws)
	{
		CMLib.tracking().stopTracking(officer);
		Ability A=CMClass.getAbility("Skill_Track");
		if(A!=null)
		{
			Room R=findTheJudge(laws,myArea);
			if(R!=null)
			{
				A.invoke(officer,CMParms.parse("\""+CMLib.map().getExtendedRoomID(R)+"\""),R,true,0);
				return true;
			}
		}
		return false;
	}

	public Room getReleaseRoom(Law laws, Area myArea, MOB criminal, LegalWarrant W)
	{
		Room room=null;
		if((criminal.isMonster())&&(criminal.getStartRoom()!=null))
			room=criminal.getStartRoom();
		else
        {
    		if((laws.releaseRooms().size()==0)||(((String)laws.releaseRooms().firstElement()).equals("@")))
    			return (Room)myArea.getMetroMap().nextElement();
            if(criminal.location()!=null)
    			room=getRoom(criminal.location().getArea(),laws.releaseRooms());
			if(room==null) room=getRoom(myArea,laws.releaseRooms());
			if(room==null) room=findTheJudge(laws,myArea);
			if(room==null) room=(Room)myArea.getMetroMap().nextElement();
        }
		return room;
	}


	public boolean isTroubleMaker(MOB M)
	{
		if(M==null) return false;
		for(int b=0;b<M.numBehaviors();b++)
		{
			Behavior B=M.fetchBehavior(b);
			if((B!=null)&&(CMath.bset(B.flags(),Behavior.FLAG_TROUBLEMAKING)))
				return true;
		}
		return false;
	}

	public Vector getRooms(Area A, Vector V)
	{
	    Vector finalV=new Vector();
		Room jail=null;
		if(V.size()==0) return finalV;
		for(int v=0;v<V.size();v++)
		{
			String which=(String)V.elementAt(v);
			jail=getRoom(A,which);
			if((jail!=null)
			&&(!finalV.contains(jail)))
			    finalV.addElement(jail);
		}
		return finalV;
	}
    public Room getRoom(Area A, String which)
    {
        Room jail=null;
        jail=CMLib.map().getRoom(which);
        if(jail==null)
        for(Enumeration r=A.getMetroMap();r.hasMoreElements();)
        {
            Room R=(Room)r.nextElement();
            if(CMLib.english().containsString(R.displayText(),which))
            { jail=R; break; }
        }
        if(jail==null)
        for(Enumeration r=A.getMetroMap();r.hasMoreElements();)
        {
            Room R=(Room)r.nextElement();
            if(CMLib.english().containsString(R.description(),which))
            { jail=R; break; }
        }
        return jail;
    }
	public Room getRoom(Area A, Vector V)
	{
		if(V.size()==0) return null;
		String which=(String)V.elementAt(CMLib.dice().roll(1,V.size(),-1));
        return getRoom(A,which);
	}

	public void fileAllWarrants(Law laws, LegalWarrant W1, MOB mob)
	{
		LegalWarrant W=null;
		Vector V=new Vector();
        if((W1!=null)&&(CMath.bset(W1.actionCode(),Law.ACTIONMASK_SEPARATE)))
        {
            for(int i=0;(W=laws.getWarrant(mob,i))!=null;i++)
                if((W.criminal()==mob)&&(W1.crime().equalsIgnoreCase(W.crime())))
                    V.addElement(W);
        }
        else
		for(int i=0;(W=laws.getWarrant(mob,i))!=null;i++)
			if(W.criminal()==mob)
				V.addElement(W);
		for(int v=0;v<V.size();v++)
		{
			W=(LegalWarrant)V.elementAt(v);
			laws.warrants().removeElement(W);
			if(W.crime()!=null)
			{
				boolean found=false;
				for(int w=0;w<laws.oldWarrants().size();w++)
				{
					LegalWarrant oW=(LegalWarrant)laws.oldWarrants().elementAt(w);
					if((oW.criminal()==mob)
					&&(oW.crime()!=null)
					&&(oW.crime().equals(W.crime())))
						found=true;
				}
				if(!found)
				{
					W.setOffenses(W.offenses()+1);
					laws.oldWarrants().addElement(W);
				}
			}
		}
	}

	public Room findTheJail(MOB mob, Area myArea, Law laws)
	{
		Room jail=null;
		if((laws.jailRooms().size()==0)||(((String)laws.jailRooms().firstElement()).equals("@")))
			return null;
		jail=getRoom(mob.location().getArea(),laws.jailRooms());
		if(jail==null) jail=getRoom(myArea,laws.jailRooms());
		return jail;
	}

    public Room findTheDetentionCenter(MOB mob, Area myArea, Law laws, LegalWarrant W)
    {
        String detentionCenter=getDetainRoom(laws,W,W.criminal());
        if(detentionCenter.length()==0) return null;
        Room detainer=getRoom(mob.location().getArea(),detentionCenter);
        if(detainer==null) detainer=getRoom(myArea,detentionCenter);
        return detainer;
    }
    
	public boolean judgeMe(Law laws, MOB judge, MOB officer, MOB criminal, LegalWarrant W, Area A, boolean debugging)
	{
        Vector relevantCrimes=getRelevantWarrants(laws.warrants(),W,criminal);
        if(CMath.bset(W.actionCode(),Law.ACTIONMASK_SKIPTRIAL))
            judge=officer;
        if(debugging)Log.debugOut("Arrest",criminal.Name()+" judged for "+W.crime()+" has base action "+W.actionCode()+", and final judgement "+highestCrimeAction(laws,W,W.criminal()));
        boolean totallyDone=false;
		switch(highestCrimeAction(laws,W,W.criminal())&Law.ACTION_MASK)
		{
		case Law.ACTION_WARN:
			{
			if((judge==null)&&(officer!=null)) judge=officer;
			StringBuffer str=new StringBuffer("");
			str.append(criminal.name()+", you are in trouble for "+restOfCharges(laws,criminal)+".  ");
			for(int w2=0;w2<relevantCrimes.size();w2++)
			{
				LegalWarrant W2=(LegalWarrant)relevantCrimes.elementAt(w2);
				if(W2.criminal()==criminal)
				{
					if(W2.witness()!=null)
						str.append("The charge of "+fixCharge(W2)+" was witnessed by "+W2.witness().name()+".  ");
					if((W2.warnMsg()!=null)&&(W2.warnMsg().length()>0))
						str.append(W2.warnMsg()+"  ");
					if((W2.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
						str.append(laws.getMessage(Law.MSG_PREVOFF)+"  ");
				}
			}
			if((laws.getMessage(Law.MSG_WARNING).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_DETAIN)))
				str.append(laws.getMessage(Law.MSG_WARNING)+"  ");
			CMLib.commands().postSay(judge,criminal,str.toString(),false,false);
			}
			totallyDone=true;
            break;
		case Law.ACTION_THREATEN:
			{
			if((judge==null)&&(officer!=null)) judge=officer;
			StringBuffer str=new StringBuffer("");
			str.append(criminal.name()+", you are in trouble for "+restOfCharges(laws,criminal)+".  ");
			for(int w2=0;w2<relevantCrimes.size();w2++)
			{
				LegalWarrant W2=(LegalWarrant)relevantCrimes.elementAt(w2);
				if(W2.criminal()==criminal)
				{
					if(W2.witness()!=null)
						str.append("The charge of "+fixCharge(W2)+" was witnessed by "+W2.witness().name()+".  ");
					if((W2.warnMsg()!=null)&&(W2.warnMsg().length()>0))
						str.append(W2.warnMsg()+"  ");
					if((W2.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
						str.append(laws.getMessage(Law.MSG_PREVOFF)+"  ");
				}
			}
			if((laws.getMessage(Law.MSG_THREAT).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_DETAIN)))
				str.append(laws.getMessage(Law.MSG_THREAT)+"  ");
			CMLib.commands().postSay(judge,criminal,str.toString(),false,false);
			}
            totallyDone=true;
            break;
		case Law.ACTION_PAROLE1:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.paroleMessages(0).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.paroleMessages(0),false,false);
				W.setJailTime(laws.paroleTimes(0));
				W.setState(Law.STATE_PAROLING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_PAROLE2:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.paroleMessages(1).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.paroleMessages(1),false,false);
				W.setJailTime(laws.paroleTimes(1));
				W.setState(Law.STATE_PAROLING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_PAROLE3:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.paroleMessages(2).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.paroleMessages(2),false,false);
				W.setJailTime(laws.paroleTimes(2));
				W.setState(Law.STATE_PAROLING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_PAROLE4:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.paroleMessages(3).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.paroleMessages(3),false,false);
				W.setJailTime(laws.paroleTimes(3));
				W.setState(Law.STATE_PAROLING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_JAIL1:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.jailMessages(0).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.jailMessages(0),false,false);
				W.setJailTime(laws.jailTimes(0));
				W.setState(Law.STATE_JAILING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_JAIL2:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.jailMessages(1).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.jailMessages(1),false,false);
				W.setJailTime(laws.jailTimes(1));
				W.setState(Law.STATE_JAILING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_JAIL3:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.jailMessages(2).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.jailMessages(2),false,false);
				W.setJailTime(laws.jailTimes(2));
				W.setState(Law.STATE_JAILING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_JAIL4:
			if(judge!=null)
			{
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.jailMessages(3).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.jailMessages(3),false,false);
				W.setJailTime(laws.jailTimes(3));
				W.setState(Law.STATE_JAILING);
			}
            totallyDone=false;
            break;
		case Law.ACTION_EXECUTE:
			if(judge!=null)
			{
				criminal.setFollowing(null);
				if((W.offenses()>0)&&(laws.getMessage(Law.MSG_PREVOFF).length()>0)&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
					CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PREVOFF),false,false);
				if(laws.getMessage(Law.MSG_EXECUTE).length()>0)
					CMLib.commands().postSay(judge,criminal,laws.getMessage(Law.MSG_EXECUTE),false,false);
				W.setState(Law.STATE_EXECUTING);
			}
            totallyDone=false;
            break;
		}
        if((totallyDone)&&(CMath.bset(W.actionCode(),Law.ACTIONMASK_FINE)))
        {
            double fines=getFine(laws,W,criminal);
            if((judge==null)&&(officer!=null)) judge=officer;
            if((fines>0.0)&&(judge!=null))
            {
                CMLib.commands().postSay(judge,criminal,"You are hereby fined "+CMLib.beanCounter().nameCurrencyShort(judge,fines)+", payable to the local tax assessor.",false,false);
                Double D=(Double)finesAssessed.get(criminal);
                if(D==null)
                    D=new Double(0.0);
                else
                    finesAssessed.remove(criminal);
                finesAssessed.put(criminal,new Double(D.doubleValue()+fines));
            }
        }
        if((totallyDone)&&(CMath.bset(W.actionCode(),Law.ACTIONMASK_DETAIN)))
        { 
            W.setState(Law.STATE_DETAINING);
            if(officer!=null)W.setArrestingOfficer(A,officer);
            if(debugging)Log.debugOut("Arrest","Putting the above crime into a detain state, officer="+(W.arrestingOfficer()!=null)+".");
            return false;
        }
		return totallyDone;
	}

	public boolean fillOutWarrant(MOB mob,
								  Law laws,
								  Area myArea,
								  Environmental target,
								  String crimeLocs,
								  String crimeFlags,
								  String crime,
								  String sentence,
								  String warnMsg)
	{
		if(mob.amDead())
		{
			if(CMSecurity.isDebugging("ARREST")) Log.debugOut("ARREST",mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+" * IS DEAD!");
		    return false;
		}
		if(mob.location()==null)
		{
			if(CMSecurity.isDebugging("ARREST")) Log.debugOut("ARREST",mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Accused is not here.");
		    return false;
		}
		if((myArea!=null)&&(!myArea.inMyMetroArea(mob.location().getArea())))
		{
			if(CMSecurity.isDebugging("ARREST")) Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Accused is not in the area.");
		    return false;
		}

		if(isAnyKindOfOfficer(laws,mob)
		||(isTheJudge(laws,mob))
		||CMSecurity.isAllowed(mob,mob.location(),"ABOVELAW"))
		{
			if(CMSecurity.isDebugging("ARREST")) Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Accused is an officer ("+isAnyKindOfOfficer(laws,mob)+"), judge ("+isTheJudge(laws,mob)+"), or above the law ("+CMSecurity.isAllowed(mob,mob.location(),"ABOVELAW")+").");
		    return false;
		}

		// is there a witness
		MOB witness=getWitness(myArea,mob);
		boolean requiresWitness=true;

		// is there a victim (if necessary)
		MOB victim=null;
		if((target!=null)&&(target instanceof MOB))
			victim=(MOB)target;
		if(mob==victim) 
		{
			if(CMSecurity.isDebugging("ARREST")) Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Accused and victim are the same.");
		    return false;
		}

		// any special circumstances?
		if(crimeFlags.trim().length()>0)
		{
			Vector V=CMParms.parse(crimeFlags.toUpperCase());
			for(int v=0;v<V.size();v++)
			{
				String str=(String)V.elementAt(v);
				if(str.endsWith("WITNESS")&&(str.length()<9))
				{
					if(str.startsWith("!"))
						requiresWitness=false;
					else
					if((witness!=null)&&(witness.location()!=mob.location()))
					{
						if(CMSecurity.isDebugging("ARREST")) 
						    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Witness required, but not present.");
					   return false;
					}
				}
				else
				if(str.endsWith("COMBAT")&&(str.length()<8))
				{
					if(mob.isInCombat())
					{
						if(str.startsWith("!")) 
						{
							if(CMSecurity.isDebugging("ARREST")) 
							    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* In combat, but shouldn't be!");
						    return false;
						}
					}
					else
						if(!str.startsWith("!"))
						{
							if(CMSecurity.isDebugging("ARREST")) 
							    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Not in combat, but should be!");
						    return false;
						}

				}
				else
				if(str.endsWith("RECENTLY")&&(str.length()<10))
				{
					LegalWarrant W=laws.getOldWarrant(mob,crime,false);
					long thisTime=System.currentTimeMillis();
					if((W!=null)&&((thisTime-W.lastOffense())<600000))
					{
						if(str.startsWith("!"))
						{
							if(CMSecurity.isDebugging("ARREST")) 
							    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Not recently, but is!");
						    return false;
						}
					}
					else
						if(!str.startsWith("!"))
						{
							if(CMSecurity.isDebugging("ARREST")) 
							    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Recently required, but it isn't!");
						    return false;
						}
				}
			}
		}
		if((requiresWitness)&&(witness==null))
		{
			if(CMSecurity.isDebugging("ARREST")) 
			    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Witness required, and none present: "+(witness==null?null:witness.Name())+".");
		    return false;
		}

		// is the location significant to this crime?
		if(crimeLocs.trim().length()>0)
		{
			boolean aCrime=false;
			Vector V=CMParms.parse(crimeLocs);
			String display=mob.location().displayText().toUpperCase().trim();
			for(int v=0;v<V.size();v++)
			{
				String str=((String)V.elementAt(v)).toUpperCase();
				if(str.endsWith("INDOORS")&&(str.length()<9))
				{
					if((mob.location().domainType()&Room.INDOORS)>0)
					{
						if(str.startsWith("!"))
						{
							if(CMSecurity.isDebugging("ARREST")) 
							    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Shouldn't be indoors, but is!");
						    return false;
						}
					}
					else
						if(!str.startsWith("!"))
						{
							if(CMSecurity.isDebugging("ARREST")) 
							    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Should be indoors, but isn't!");
						    return false;
						}
					aCrime=true;
				}
				else
				if(str.endsWith("HOME")&&(str.length()<6))
				{
					if(CMLib.law().doesHavePriviledgesHere(mob,mob.location()))
						if(str.startsWith("!")) 
						{
							if(CMSecurity.isDebugging("ARREST")) 
							    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Should not be home, but is!");
						    return false;
						}
					if(!str.startsWith("!"))
					{
						if(CMSecurity.isDebugging("ARREST")) 
						    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Should be home, but is not!");
					    return false;
					}
					aCrime=true;
				}
				else
				if(str.startsWith("!")&&(CMLib.english().containsString(display,str.substring(1))))
				{
					if(CMSecurity.isDebugging("ARREST")) 
					    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Should not be at '"+str.substring(1)+"', but is!");
				    return false;
				}
				else
				if(CMLib.english().containsString(display,str))
				{ aCrime=true; break;}
			}
			if(!aCrime)
			{
				if(CMSecurity.isDebugging("ARREST")) 
				    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Crime flag failure!");
			    return false;
			}
		}

		// is the victim a protected race?
		if((victim!=null)&&(!(victim instanceof Deity)))
		{
			if(!CMLib.masking().maskCheck(laws.getMessage(Law.MSG_PROTECTEDMASK),victim,false))
			{
				if(CMSecurity.isDebugging("ARREST")) 
				    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Victim is not a protected race!");
				return false;
			}
		}

		// does a warrant already exist?
		LegalWarrant W=null;
		for(int i=0;(W=laws.getWarrant(mob,i))!=null;i++)
		{
			if((W.criminal()==mob)
			&&(W.victim()==victim)
			&&(W.crime().equals(crime)))
			{
				if(CMSecurity.isDebugging("ARREST")) 
				    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Warrant already exists.");
				return false;
			}
		}
		if(W==null) W=laws.getOldWarrant(mob,crime,true);
		if(W==null) W=(LegalWarrant)CMClass.getCommon("DefaultArrestWarrant");

		// fill out the warrant!
		W.setCriminal(mob);
		W.setVictim(victim);
		W.setCrime(crime);
		W.setState(Law.STATE_SEEKING);
		W.setWitness(witness);
		W.setLastOffense(System.currentTimeMillis());
		W.setWarnMsg(warnMsg);
		sentence=sentence.trim();
        Vector sentences=CMParms.parse(sentence);
        W.setActionCode(0);
        for(int v=0;v<sentences.size();v++)
        {
            String s=(String)sentences.elementAt(v);
            int x=s.indexOf("=");
            String parm=null;
            if(x>0)
            {
                parm=s.substring(x+1);
                s=s.substring(0,x+1);
            }
            boolean actionCodeSet=false;
            for(int i=0;i<Law.ACTION_DESCS.length;i++)
                if(s.equalsIgnoreCase(Law.ACTION_DESCS[i]))
                { 
                    actionCodeSet=true; 
                    W.setActionCode(W.actionCode()|i);
                    if(parm!=null)
                        W.addActionParm(i,parm);
                }
            if(!actionCodeSet)
                for(int i=0;i<Law.ACTIONMASK_DESCS.length;i++)
                    if(s.equalsIgnoreCase(Law.ACTIONMASK_DESCS[i]))
                    { 
                        actionCodeSet=true; 
                        W.setActionCode(W.actionCode()|Law.ACTIONMASK_CODES[i]);
                        if(parm!=null)
                            W.addActionParm(Law.ACTIONMASK_CODES[i],parm);
                    }
            if(!actionCodeSet)
            {
                Log.errOut("Arrest","Unknown sentence: "+s+" for crime "+crime);
                return false;
            }
        }

		if((W.victim()!=null)&&(isTroubleMaker(W.victim()))&&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE)))
			W.setActionCode(W.actionCode()/2);

		if((isStillACrime(W,CMSecurity.isDebugging("ARREST")))
		&&((W.witness()==null)||CMLib.flags().canBeSeenBy(W.criminal(),W.witness())))
		{
			if(CMSecurity.isDebugging("ARREST")) 
			    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Warrant filled out.");
			addWarrant(laws,W);
		}
		else
			if(CMSecurity.isDebugging("ARREST")) 
			    Log.debugOut("ARREST", mob.name()+", data: "+crimeLocs+"->"+crimeFlags+"->"+crime+"->"+sentence+"* Warrant fails the is a crime check.");
		return true;
	}

	protected boolean isAnUltimateAuthorityHere(MOB M, Law laws)
	{
		if(CMSecurity.isAllowed(M,M.location(),"ABOVELAW")||(isTheJudge(laws,M)))
			return true;
		return false;
	}

	protected boolean theLawIsEnabled()
	{
		return ((CMProps.getBoolVar(CMProps.SYSTEMB_MUDSTARTED))
				&&(!CMSecurity.isDisabled("ARREST")));
	}

    public void testEntryLaw(Law laws, Area myArea, MOB testMOB, Room R)
    {
        if((laws.basicCrimes().containsKey("NUDITY"))
        &&(!testMOB.isMonster())
        &&(testMOB.fetchFirstWornItem(Item.WORN_LEGS)==null)
        &&(testMOB.getWearPositions(Item.WORN_LEGS)>0)
        &&(testMOB.fetchFirstWornItem(Item.WORN_WAIST)==null)
        &&(testMOB.getWearPositions(Item.WORN_WAIST)>0)
        &&(testMOB.fetchFirstWornItem(Item.WORN_ABOUT_BODY)==null)
        &&(testMOB.getWearPositions(Item.WORN_ABOUT_BODY)>0))
        {
            String info[]=(String[])laws.basicCrimes().get("NUDITY");
            fillOutWarrant(testMOB,
                            laws,
                           myArea,
                           null,
                           info[Law.BIT_CRIMELOCS],
                           info[Law.BIT_CRIMEFLAGS],
                           info[Law.BIT_CRIMENAME],
                           info[Law.BIT_SENTENCE],
                           info[Law.BIT_WARNMSG]);
        }
        

        Item w=null;
        if((laws.basicCrimes().containsKey("ARMED"))
        &&((!testMOB.isMonster())||(!myArea.inMyMetroArea(CMLib.map().getStartArea(testMOB))))
        &&((w=testMOB.fetchWieldedItem())!=null)
        &&(w instanceof Weapon)
        &&(((Weapon)w).weaponClassification()!=Weapon.CLASS_NATURAL)
        &&(((Weapon)w).weaponClassification()!=Weapon.CLASS_HAMMER)
        &&(((Weapon)w).weaponClassification()!=Weapon.CLASS_STAFF)
        &&(CMLib.flags().isSeen(w))
        &&(!CMLib.flags().isHidden(w))
        &&(!CMLib.flags().isInvisible(w)))
        {
            String info[]=(String[])laws.basicCrimes().get("ARMED");
            if((testMOB.session()==null)
            ||(((System.currentTimeMillis()-testMOB.session().getLastNPCFight())>30000)
                &&((System.currentTimeMillis()-testMOB.session().getLastPKFight())>30000)))
            {
                fillOutWarrant(testMOB,
                               laws,
                               myArea,
                               null,
                               info[Law.BIT_CRIMELOCS],
                               info[Law.BIT_CRIMEFLAGS],
                               info[Law.BIT_CRIMENAME],
                               info[Law.BIT_SENTENCE],
                               info[Law.BIT_WARNMSG]);
            }
        }
        
        if((laws.basicCrimes().containsKey("TRESPASSING"))
        &&((CMLib.masking().maskCheck(laws.getMessage(Law.MSG_TRESPASSERMASK),testMOB,false))
            ||(testMOB.isMonster()
                &&(testMOB.getStartRoom()!=null)
                &&(testMOB.getStartRoom().getArea()!=R.getArea())
                &&(CMLib.flags().isPossiblyAggressive(testMOB))
                &&((testMOB.amFollowing()==null)
                        ||((!testMOB.amFollowing().isMonster())&&(testMOB.amFollowing().location()==testMOB.location())))
                &&(!CMLib.masking().maskCheck(laws.getMessage(Law.MSG_PROTECTEDMASK),testMOB,false)))))
        {
            String[] info=(String[])laws.basicCrimes().get("TRESPASSING");
            fillOutWarrant(testMOB,
                            laws,
                           myArea,
                           null,
                           info[Law.BIT_CRIMELOCS],
                           info[Law.BIT_CRIMEFLAGS],
                           info[Law.BIT_CRIMENAME],
                           info[Law.BIT_SENTENCE],
                           info[Law.BIT_WARNMSG]);
        }
    }
    
    
	public void executeMsg(Environmental affecting, CMMsg msg)
	{
		super.executeMsg(affecting, msg);
		if(!(affecting instanceof Area)) return;
		if(!theLawIsEnabled()) return;

		Area myArea=(Area)affecting;
		Law laws=getLaws(affecting,false);
		if(!laws.lawIsActivated()) return;
		if(msg.source()==null) return;

		// the archons pardon
		if((msg.sourceMinor()==CMMsg.TYP_SPEAK)
		&&(msg.sourceMessage()!=null)
		&&(isAnUltimateAuthorityHere(msg.source(),laws)))
		{
			int x=msg.sourceMessage().toUpperCase().indexOf("I HEREBY PARDON ");
			if(x>0)
			{
				int y=msg.sourceMessage().lastIndexOf("'");
				if(y<x)	y=msg.sourceMessage().lastIndexOf("`");
				String name=null;
				if(y>x)
					name=msg.sourceMessage().substring(x+16,y).trim();
				else
					name=msg.sourceMessage().substring(x+16).trim();
				Vector warrs=(Vector)laws.warrants().clone();
				if(name.length()>0)
				for(int i=warrs.size()-1;i>=0;i--)
				{
					LegalWarrant W=(LegalWarrant)warrs.elementAt(i);
					if((W.criminal()!=null)&&(CMLib.english().containsString(W.criminal().Name(),name)))
					{
						Ability A=W.criminal().fetchEffect("Prisoner");
						if(A!=null) A.unInvoke();
						if(W.jail()!=W.criminal().location())
						{
							if(W.arrestingOfficer()!=null)
								dismissOfficer(W.arrestingOfficer());
							laws.warrants().removeElement(W);
						}
						else
						{
							W.setCrime("pardoned");
							W.setOffenses(0);
						}
					}
				}
			}
		}

		if((msg.sourceMinor()==CMMsg.TYP_DEATH)
		&&(msg.tool()!=null)
		&&(msg.tool() instanceof MOB)
		&&(laws.basicCrimes().containsKey("MURDER")))
		{
			MOB criminal=(MOB)msg.tool();
			for(int i=laws.warrants().size()-1;i>=0;i--)
			{
				LegalWarrant W=(LegalWarrant)laws.warrants().elementAt(i);
				if((W.victim()!=null)
				&&(W.criminal()!=null)
				&&(W.victim()==msg.source())
                &&(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE))
				&&(W.criminal()==criminal))
					laws.warrants().removeElement(W);
			}
			String[] bits=(String[])laws.basicCrimes().get("MURDER");
			fillOutWarrant(criminal,
						   laws,
						   myArea,
						   msg.source(),
						   bits[Law.BIT_CRIMELOCS],
						   bits[Law.BIT_CRIMEFLAGS],
						   bits[Law.BIT_CRIMENAME],
						   bits[Law.BIT_SENTENCE],
						   bits[Law.BIT_WARNMSG]);
			return;
		}

		if(isAnyKindOfOfficer(laws,msg.source())||(isTheJudge(laws,msg.source())))
        {
            if((msg.sourceMinor()==CMMsg.TYP_ENTER)
            &&(msg.target() instanceof Room))
            {
                Room R=(Room)msg.target();
                MOB M=null;
                for(int m=0;m<R.numInhabitants();m++)
                {
                    M=R.fetchInhabitant(m);
                    if((M!=null)
                    &&(M!=msg.source())
                    &&(!isAnyKindOfOfficer(laws,M))
                    &&(!isTheJudge(laws,M)))
                        testEntryLaw(laws,myArea,M,R);
                }
            }
			return;
        }

        if((msg.source().isMonster())&&(!laws.arrestMobs()))
            return;

		if(!CMLib.flags().aliveAwakeMobile(msg.source(),true))
			return;

        Room R=msg.source().location();
		if(R==null) return;

		if((msg.tool()!=null)
		&&(msg.tool() instanceof Ability)
		&&(msg.othersMessage()!=null)
		&&((laws.abilityCrimes().containsKey(msg.tool().ID().toUpperCase()))
                ||(laws.abilityCrimes().containsKey(CMLib.flags().getAbilityType((Ability)msg.tool())))
                ||(laws.abilityCrimes().containsKey(CMLib.flags().getAbilityDomain((Ability)msg.tool())))))
		{
			String[] info=(String[])laws.abilityCrimes().get(msg.tool().ID().toUpperCase());
            if(info==null) info=(String[])laws.abilityCrimes().get(CMLib.flags().getAbilityType((Ability)msg.tool()));
            if(info==null) info=(String[])laws.abilityCrimes().get(CMLib.flags().getAbilityDomain((Ability)msg.tool()));
			fillOutWarrant(msg.source(),
						    laws,
							myArea,
							msg.target(),
							info[Law.BIT_CRIMELOCS],
							info[Law.BIT_CRIMEFLAGS],
							info[Law.BIT_CRIMENAME],
							info[Law.BIT_SENTENCE],
							info[Law.BIT_WARNMSG]);
		}

		for(int a=0;a<msg.source().numAllEffects();a++)
		{
			Ability A=msg.source().fetchEffect(a);
			if((A!=null)
			&&(!A.isAutoInvoked())
			&&((A.canBeUninvoked()||(!msg.source().isMonster())))
			&&((laws.abilityCrimes().containsKey("$"+A.ID().toUpperCase()))
                ||(laws.abilityCrimes().containsKey("$"+CMLib.flags().getAbilityType(A)))
                ||(laws.abilityCrimes().containsKey("$"+CMLib.flags().getAbilityDomain(A)))))
			{
				String[] info=(String[])laws.abilityCrimes().get("$"+A.ID().toUpperCase());
                if(info==null) info=(String[])laws.abilityCrimes().get("$"+CMLib.flags().getAbilityType(A));
                if(info==null) info=(String[])laws.abilityCrimes().get("$"+CMLib.flags().getAbilityDomain(A));
				fillOutWarrant(msg.source(),
								laws,
								myArea,
								null,
								info[Law.BIT_CRIMELOCS],
								info[Law.BIT_CRIMEFLAGS],
								info[Law.BIT_CRIMENAME],
								info[Law.BIT_SENTENCE],
								info[Law.BIT_WARNMSG]);
			}
		}

		if((CMath.bset(msg.targetCode(),CMMsg.MASK_MALICIOUS))
		&&(msg.target()!=null)
		&&(!CMath.bset(msg.sourceCode(),CMMsg.MASK_ALWAYS))
		&&((msg.tool()==null)||(msg.source().isMine(msg.tool())))
		&&(msg.target()!=msg.source())
		&&(!msg.target().name().equals(msg.source().name()))
		&&(msg.target() instanceof MOB))
		{
			if(isTheJudge(laws,(MOB)msg.target()))
			{
				if(!msg.source().isMonster())
				for(int i=0;i<R.numInhabitants();i++)
				{
					MOB M=R.fetchInhabitant(i);
					if((M!=null)
					&&(M!=msg.target())
					&&(M!=msg.source())
					&&(M.getVictim()!=msg.source())
					&&(isAnyKindOfOfficer(laws,M)))
					{
						if(msg.source().amFollowing()==M)
							msg.source().setFollowing(null);
						CMLib.commands().postSay(M,null,"Ack! Treason! Die!",false,false);
						M.setVictim(msg.source());
					}
				}
			}
			else
			{
				boolean justResisting=false;
				boolean turnAbout=false;
				boolean targetIsOfficer=isAnyKindOfOfficer(laws,(MOB)msg.target());
				String[] assaultInfo=(String[])laws.basicCrimes().get("ASSAULT");
				String[] murderInfo=(String[])laws.basicCrimes().get("MURDER");
				if((assaultInfo!=null)&&(murderInfo!=null))
				for(int i=laws.warrants().size()-1;i>=0;i--)
				{
					LegalWarrant W=(LegalWarrant)laws.warrants().elementAt(i);
					if(targetIsOfficer
					&&(W.criminal()==msg.source())
					&&(W.arrestingOfficer()!=null)
					&&(W.criminal().location()!=null)
					&&(W.criminal().location().isInhabitant(W.arrestingOfficer())))
						justResisting=true;
					else
					if((!targetIsOfficer)
					&&(W.victim()==msg.source())
					&&(W.criminal()==msg.target())
					&&(W.crime().equals(assaultInfo[Law.BIT_CRIMENAME])||W.crime().equals(murderInfo[Law.BIT_CRIMENAME]))
					&&(isStillACrime(W,false)))
						turnAbout=true;
					else
					if((!targetIsOfficer)
					&&(W.victim()==msg.target())
					&&(W.criminal()==msg.source())
					&&(W.crime().equals(murderInfo[Law.BIT_CRIMENAME]))
					&&(isStillACrime(W,false)))
						turnAbout=true;
				}
				if(justResisting)
				{
					if(laws.basicCrimes().containsKey("RESISTINGARREST"))
					{
						String[] info=(String[])laws.basicCrimes().get("RESISTINGARREST");
						fillOutWarrant(msg.source(),
										laws,
										myArea,
										null,
										info[Law.BIT_CRIMELOCS],
										info[Law.BIT_CRIMEFLAGS],
										info[Law.BIT_CRIMENAME],
										info[Law.BIT_SENTENCE],
										info[Law.BIT_WARNMSG]);
					}
				}
				else
				if((!turnAbout)
				&&(assaultInfo!=null)
				&&((msg.source().isMonster())||(!isTroubleMaker((MOB)msg.target()))))
					fillOutWarrant(msg.source(),
									laws,
									myArea,
									msg.target(),
									assaultInfo[Law.BIT_CRIMELOCS],
									assaultInfo[Law.BIT_CRIMEFLAGS],
									assaultInfo[Law.BIT_CRIMENAME],
									assaultInfo[Law.BIT_SENTENCE],
									assaultInfo[Law.BIT_WARNMSG]);
			}
		}

		if((msg.othersCode()!=CMMsg.NO_EFFECT)
		   &&(msg.othersMessage()!=null))
		{
		    if((msg.targetMinor()==CMMsg.TYP_GET)
		    &&(msg.target() instanceof Item)
		    &&(laws.bannedSubstances().size()>0))
		    {
		        String rsc=RawMaterial.RESOURCE_DESCS[((Item)msg.target()).material()&RawMaterial.RESOURCE_MASK].toUpperCase();
				for(int i=0;i<laws.bannedSubstances().size();i++)
				{
					Vector V=(Vector)laws.bannedSubstances().elementAt(i);
					for(int v=0;v<V.size();v++)
					{
						if((CMLib.english().containsString(msg.target().name(),(String)V.elementAt(v)))
						||rsc.equalsIgnoreCase((String)V.elementAt(v)))
						{
							String[] info=(String[])laws.bannedBits().elementAt(i);
							fillOutWarrant(msg.source(),
											laws,
											myArea,
											msg.target(),
											info[Law.BIT_CRIMELOCS],
											info[Law.BIT_CRIMEFLAGS],
											info[Law.BIT_CRIMENAME],
											info[Law.BIT_SENTENCE],
											info[Law.BIT_WARNMSG]);
						}
					}
				}

		    }
			if(msg.sourceMinor()==CMMsg.TYP_ENTER)
                testEntryLaw(laws,myArea,msg.source(),R);
            
			for(int i=0;i<laws.otherCrimes().size();i++)
			{
				Vector V=(Vector)laws.otherCrimes().elementAt(i);
				for(int v=0;v<V.size();v++)
				{
					if(CMLib.english().containsString(msg.othersMessage(),(String)V.elementAt(v)))
					{
						String[] info=(String[])laws.otherBits().elementAt(i);
						fillOutWarrant(msg.source(),
										laws,
										myArea,
										msg.target(),
										info[Law.BIT_CRIMELOCS],
										info[Law.BIT_CRIMEFLAGS],
										info[Law.BIT_CRIMENAME],
										info[Law.BIT_SENTENCE],
										info[Law.BIT_WARNMSG]);
					}
				}
			}
		}
	}


	public void haveMobReactToLaw(MOB mob, MOB officer)
	{
		if((mob.isMonster())&&(!CMLib.flags().isSitting(mob))&&(mob.amFollowing()==null)&&(!mob.isInCombat()))
		{
			boolean good=CMLib.flags().isGood(mob);
			boolean evil=CMLib.flags().isEvil(mob);
			boolean neutral=(!good)&&(!evil);
			if(evil
			||(neutral&&(CMLib.dice().rollPercentage()>50))
			||(CMLib.flags().flaggedBehaviors(mob,Behavior.FLAG_POTENTIALLYAGGRESSIVE).size()>0))
			{
				if(mob.envStats().level()>(officer.envStats().level()/2))
					mob.setVictim(officer);
				else
				if(!CMLib.flags().isAnimalIntelligence(mob))
					mob.enqueCommand(CMParms.parse("FLEE"),1);
			}
			else
			if((good||neutral)
			&&(!CMLib.flags().isAnimalIntelligence(mob)))
			{
				mob.makePeace();
				mob.doCommand(CMParms.parse("SIT"));
			}
			else
			if((CMLib.flags().isAnimalIntelligence(mob))&&(CMLib.dice().rollPercentage()>50))
			{
				mob.makePeace();
				mob.doCommand(CMParms.parse("SIT"));
			}
		}
	}

	public boolean tick(Tickable ticking, int tickID)
	{
		super.tick(ticking,tickID);

		if(!(ticking instanceof Area)) return true;
		if(tickID!=Tickable.TICKID_AREA) return true;
		Area myArea=(Area)ticking;

		if(!theLawIsEnabled())return true;

		Law laws=getLaws(myArea,false);
		if(!laws.lawIsActivated())
		{
			laws.warrants().clear();
			laws.oldWarrants().clear();
			return true;
		}
		boolean debugging=CMSecurity.isDebugging("ARREST");

		laws.propertyTaxTick(myArea,debugging);

		HashSet handled=new HashSet();
		Vector warrs=(Vector)laws.warrants().clone();
		for(int w=0;w<warrs.size();w++)
		{
			LegalWarrant W=(LegalWarrant)warrs.elementAt(w);
			if((W.criminal()==null)||(W.criminal().location()==null))
            {
                if(debugging) Log.debugOut("Arrest","Tick: "+W.crime()+": Criminal or Location is null. Skipping.");
				continue;
            }

			if(!isStillACrime(W,debugging))
			{
				unCuff(W.criminal());
				if(W.arrestingOfficer()!=null)
					dismissOfficer(W.arrestingOfficer());
				W.setArrestingOfficer(myArea,null);
				W.setOffenses(W.offenses()+1);
				laws.oldWarrants().addElement(W);
				laws.warrants().removeElement(W);
                if(debugging) Log.debugOut("Arrest","Tick: "+W.crime()+": No longer a crime.");
				continue;
			}

            
			if(!CMath.bset(W.actionCode(),Law.ACTIONMASK_SEPARATE))
            {
                if(handled.contains(W.criminal().Name()))
    				continue;
                handled.add(W.criminal().Name());
            }
            else
            {
                if(handled.contains(W.criminal().Name()+"/"+W.crime()))
                    continue;
                handled.add(W.criminal().Name()+"/"+W.crime());
            }
            if(debugging) Log.debugOut("Arrest","Tick: Handling "+W.crime()+" for "+W.criminal().Name()+": State "+W.state());
            
    		processWarrant(myArea, laws, W, debugging);
		}
		return true;
	}

    protected void fileArrestResister(Law laws, Area myArea, LegalWarrant W)
    {
        if((W.criminal()!=null)
        &&(W.arrestingOfficer()!=null)
        &&(!W.arrestingOfficer().amDead())
        &&(!W.crime().equalsIgnoreCase("pardoned"))
        &&(!CMLib.flags().isInTheGame(W.criminal(),true))
        &&(isStillACrime(W,false)))
        {
            if(laws.basicCrimes().containsKey("RESISTINGARREST"))
            {
                String[] info=(String[])laws.basicCrimes().get("RESISTINGARREST");
                fillOutWarrant(W.criminal(),
                                laws,
                                myArea,
                                null,
                                info[Law.BIT_CRIMELOCS],
                                info[Law.BIT_CRIMEFLAGS],
                                info[Law.BIT_CRIMENAME],
                                info[Law.BIT_SENTENCE],
                                info[Law.BIT_WARNMSG]);
            }
        }
    }
    
	protected void processWarrant(Area myArea, Law laws, LegalWarrant W, boolean debugging)
	{
		switch(W.state())
		{
		case Law.STATE_SEEKING:
			{
				MOB officer=W.arrestingOfficer();
				if((officer==null)||(!W.criminal().location().isInhabitant(officer)))
				   officer=null;
				if(officer==null)
					officer=getElligibleOfficer(laws,myArea,W.criminal(),W.victim());
				W.setTravelAttemptTime(0);
				if((officer!=null)
				&&(W.criminal().location()!=null)
				&&(W.criminal().location().isInhabitant(officer))
				&&(!W.criminal().amDead())
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(CMLib.flags().canBeSeenBy(W.criminal(),officer))
				&&(canFocusOn(officer,W.criminal())))
				{
					if(CMSecurity.isAllowed(W.criminal(),W.criminal().location(),"ABOVELAW"))
					{
						CMLib.commands().postSay(officer,W.criminal(),"Damn, I can't arrest you.",false,false);
						if(CMSecurity.isAllowedEverywhere(W.criminal(),"ABOVELAW"))
						{
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());
							W.setArrestingOfficer(myArea,null);
						}
					}
					else
					if(W.crime().equalsIgnoreCase("pardoned"))
					{
						fileAllWarrants(laws,W,W.criminal());
						unCuff(W.criminal());
						W.setArrestingOfficer(myArea,null);
					}
					else
					if(judgeMe(laws,null,officer,W.criminal(),W,myArea,debugging))
					{
						fileAllWarrants(laws,W,W.criminal());
						unCuff(W.criminal());
						dismissOfficer(officer);
						W.setArrestingOfficer(myArea,null);
					}
					else
                    if(W.state()!=Law.STATE_DETAINING)
					{
						if(!CMLib.flags().isAnimalIntelligence(W.criminal()))
						{
							W.setArrestingOfficer(myArea,officer);
	        	            LegalWarrant copKillerW=laws.getCopkiller(myArea,this,W.criminal());
	        	            if(copKillerW!=null)
	        	            {
								CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),laws.getMessage(Law.MSG_COPKILLER),false,false);
	    	            		W.setState(Law.STATE_SUBDUEING);
	    	            		W.arrestingOfficer().setVictim(W.criminal());
	    	            		processWarrant(myArea, laws, W, debugging);
	    	            		return;
	        	            }
                            LegalWarrant lawResistW=laws.getLawresister(myArea,this,W.criminal());
                            if(lawResistW!=null)
                            {
                                CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),laws.getMessage(Law.MSG_RESISTFIGHT),false,false);
                                W.setState(Law.STATE_SUBDUEING);
                                W.arrestingOfficer().setVictim(W.criminal());
                                processWarrant(myArea, laws, W, debugging);
                                return;
                            }
                            
							CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),"You are under arrest "+restOfCharges(laws,W.criminal())+"! Sit down on the ground immediately!",false,false);
							W.setState(Law.STATE_ARRESTING);
						}
						else
						{
							W.setArrestingOfficer(myArea,officer);
							CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),"You are headed to the pound for "+restOfCharges(laws,W.criminal())+"!",false,false);
							W.setState(Law.STATE_ARRESTING);
						}
					}
				}
				else
				if(W.crime().equalsIgnoreCase("pardoned"))
				{
					fileAllWarrants(laws,W,W.criminal());
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
				}
			}
			break;
		case Law.STATE_ARRESTING:
			{
				MOB officer=W.arrestingOfficer();
				W.setTravelAttemptTime(0);
				if((officer!=null)
				&&(W.criminal().location()!=null)
				&&(W.criminal().location().isInhabitant(officer))
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(!W.criminal().amDead())
				&&(CMLib.flags().aliveAwakeMobile(officer,true))
				&&(CMLib.flags().canBeSeenBy(W.criminal(),officer)))
				{
					if(officer.isInCombat())
					{
						if(officer.getVictim()==W.criminal())
						{
							CMLib.commands().postSay(officer,W.criminal(),laws.getMessage(Law.MSG_RESISTFIGHT),false,false);
							W.setState(Law.STATE_SUBDUEING);
						}
						else
						{
							W.setArrestingOfficer(myArea,null);
							W.setState(Law.STATE_SEEKING);
						}
					}
					else
					if(W.crime().equalsIgnoreCase("pardoned"))
					{
						fileAllWarrants(laws,W,W.criminal());
						unCuff(W.criminal());
						W.setArrestingOfficer(myArea,null);
					}
					else
					{
						haveMobReactToLaw(W.criminal(),officer);
						W.setState(Law.STATE_SUBDUEING);
						if(CMLib.flags().isSitting(W.criminal())||CMLib.flags().isSleeping(W.criminal()))
						{
							if(!CMLib.flags().isAnimalIntelligence(W.criminal()))
								CMLib.commands().postSay(officer,W.criminal(),laws.getMessage(Law.MSG_NORESIST),false,false);
						}
						else
							CMLib.commands().postSay(officer,W.criminal(),laws.getMessage(Law.MSG_RESISTWARN),false,false);
						if(W.criminal().isMonster())
							haveMobReactToLaw(W.criminal(),officer);
					}
				}
				else
				{
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
				}
			}
			break;
		case Law.STATE_SUBDUEING:
			{
				MOB officer=W.arrestingOfficer();
				if((officer!=null)
				&&(W.criminal().location()!=null)
				&&(W.criminal().location().isInhabitant(officer))
				&&(!W.criminal().amDead())
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(CMLib.flags().aliveAwakeMobile(officer,true))
				&&(CMLib.flags().canBeSeenBy(W.criminal(),officer)))
				{
					W.setTravelAttemptTime(0);
					haveMobReactToLaw(W.criminal(),officer);
					if(W.crime().equalsIgnoreCase("pardoned"))
					{
						fileAllWarrants(laws,W,W.criminal());
						unCuff(W.criminal());
						W.setArrestingOfficer(myArea,null);
					}
					else
					if(CMLib.flags().isStanding(W.criminal()))
					{
						if(!W.arrestingOfficer().isInCombat())
							CMLib.commands().postSay(officer,W.criminal(),laws.getMessage(Law.MSG_RESIST),false,false);

						Ability A=CMClass.getAbility("Skill_ArrestingSap");
						if(A!=null){
							int curPoints=(int)Math.round(CMath.div(W.criminal().curState().getHitPoints(),W.criminal().maxState().getHitPoints())*100.0);
							A.setProficiency(100);
							A.setAbilityCode(10);
							if(!A.invoke(officer,W.criminal(),(curPoints<=25),0))
							{
								A=CMClass.getAbility("Skill_Trip");
								if(A!=null)
								{
									A.setProficiency(100);
									A.setAbilityCode(30);
									if(!A.invoke(officer,W.criminal(),(curPoints<=50),0))
										CMLib.combat().postAttack(officer,W.criminal(),officer.fetchWieldedItem());
								}
							}
						}
					}
					Ability cuff=W.criminal().fetchEffect("Skill_HandCuff");
					if((CMLib.flags().isSitting(W.criminal())||(cuff!=null)||(CMLib.flags().isSleeping(W.criminal())))
					&&(!W.criminal().amDead())
					&&(CMLib.flags().isInTheGame(W.criminal(),true)))
					{
						makePeace(officer.location());
						// cuff him!
						if(CMLib.flags().isAnimalIntelligence(W.criminal()))
							W.setState(Law.STATE_JAILING);
						else
							W.setState(Law.STATE_MOVING);
						if(cuff!=null){ cuff.unInvoke(); W.criminal().delEffect(cuff);}
						Ability A=CMClass.getAbility("Skill_HandCuff");
						if(A!=null)	A.invoke(officer,W.criminal(),true,0);
						W.criminal().makePeace();
						makePeace(officer.location());
						A=W.criminal().fetchEffect("Skill_ArrestingSap");
						if(A!=null)A.unInvoke();
						A=W.criminal().fetchEffect("Fighter_Whomp");
						if(A!=null)A.unInvoke();
						A=W.criminal().fetchEffect("Skill_Trip");
						if(A!=null)A.unInvoke();
						makePeace(officer.location());
						CMLib.commands().postStand(W.criminal(),true);
						W.setTravelAttemptTime(System.currentTimeMillis());
						if(trackTheJudge(officer,myArea,laws))
							makePeace(officer.location());
						else
						{
							makePeace(officer.location());
							CMLib.commands().postSay(officer,W.criminal(),"Since there is no judge, you may go.",false,false);
							W.setTravelAttemptTime(0);
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());
							if(W.arrestingOfficer()!=null)
								dismissOfficer(W.arrestingOfficer());
						}
					}
					else
					{
						if(officer!=null)
                        {
							CMLib.commands().postSay(officer,null,"Hmph.",false,false);
                            fileArrestResister(laws,myArea,W);
                        }
						W.setTravelAttemptTime(0);
						unCuff(W.criminal());
						W.setArrestingOfficer(myArea,null);
						W.setState(Law.STATE_SEEKING);
					}
				}
				else
				{
					if(officer!=null)
                    {
						CMLib.commands().postSay(officer,null,"Darn.",false,false);
                        fileArrestResister(laws,myArea,W);
                    }
					W.setTravelAttemptTime(0);
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
				}
			}
			break;
		case Law.STATE_MOVING:
			{
				MOB officer=W.arrestingOfficer();

				if((officer!=null)
				&&(W.criminal().location().isInhabitant(officer))
				&&(!W.criminal().amDead())
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(!W.crime().equalsIgnoreCase("pardoned"))
				&&((W.travelAttemptTime()==0)||((System.currentTimeMillis()-W.travelAttemptTime())<(5*60*1000)))
				&&(CMLib.flags().aliveAwakeMobile(officer,true)))
				{
					if(W.criminal().curState().getMovement()<50)
						W.criminal().curState().setMovement(50);
					if(officer.curState().getMovement()<50)
						officer.curState().setMovement(50);
					makePeace(officer.location());
					if(officer.isMonster()) CMLib.commands().postLook(officer,true);
					if(getTheJudgeHere(laws,officer.location())!=null)
						W.setState(Law.STATE_REPORTING);
					else
					if(CMLib.flags().flaggedAffects(officer,Ability.FLAG_TRACKING).size()==0)
					{
						if(!trackTheJudge(officer,myArea,laws))
						{
							CMLib.commands().postSay(officer,null,"Now where was that court?.",false,false);
							W.setTravelAttemptTime(0);
							unCuff(W.criminal());
							W.setArrestingOfficer(myArea,null);
							W.setState(Law.STATE_SEEKING);
						}
					}
					else
					if((CMLib.dice().rollPercentage()>75)&&(laws.chitChat().size()>0))
						CMLib.commands().postSay(officer,W.criminal(),(String)laws.chitChat().elementAt(CMLib.dice().roll(1,laws.chitChat().size(),-1)),false,false);
				}
				else
				{
					if(officer!=null)
                    {
						CMLib.commands().postSay(officer,null,"Drat! Lost another one!",false,false);
                        fileArrestResister(laws,myArea,W);
                    }
					W.setTravelAttemptTime(0);
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
				}
			}
			break;
		case Law.STATE_REPORTING:
			{
				MOB officer=W.arrestingOfficer();
				if((officer!=null)
				&&(W.criminal().location().isInhabitant(officer))
				&&(!W.criminal().amDead())
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(!W.crime().equalsIgnoreCase("pardoned"))
				&&(CMLib.flags().aliveAwakeMobile(officer,true)))
				{
					MOB judge=getTheJudgeHere(laws,officer.location());
					if(judge==null)
					{
						W.setState(Law.STATE_MOVING);
						if(!trackTheJudge(officer,myArea,laws))
						{
							CMLib.commands().postSay(officer,null,"Where was that darn court!",false,false);
							W.setTravelAttemptTime(0);
							unCuff(W.criminal());
							W.setArrestingOfficer(myArea,null);
							W.setState(Law.STATE_SEEKING);
						}

					}
					else
					if(CMLib.flags().aliveAwakeMobile(judge,true))
					{
						W.setTravelAttemptTime(0);
						String sirmaam="Sir";
						if(Character.toString((char)judge.charStats().getStat(CharStats.STAT_GENDER)).equalsIgnoreCase("F"))
							sirmaam="Ma'am";
						CMLib.commands().postSay(officer,judge,sirmaam+", "+W.criminal().name()+" has been arrested "+restOfCharges(laws,W.criminal())+".",false,false);
                        Vector warrants=getRelevantWarrants(laws.warrants(),W,W.criminal());
						for(int w2=0;w2<warrants.size();w2++)
						{
							LegalWarrant W2=(LegalWarrant)warrants.elementAt(w2);
							if(W2.witness()!=null)
								CMLib.commands().postSay(officer,judge,"The charge of "+fixCharge(W2)+" was witnessed by "+W2.witness().name()+".",false,false);
						}
						W.setState(Law.STATE_WAITING);
						if((highestCrimeAction(laws,W,W.criminal())==Law.ACTION_EXECUTE)
                        &&(judge.location()!=null))
						{
							Vector channels=CMLib.channels().getFlaggedChannelNames("EXECUTIONS");
                            for(int i=0;i<channels.size();i++)
								CMLib.commands().postChannel(judge,(String)channels.elementAt(i),W.criminal().Name()+" is being executed at "+judge.location().displayText()+" for "+W.criminal().charStats().hisher()+" crimes.",true);
						}
					}
					else
					{
						CMLib.commands().postSay(officer,W.criminal(),"I guess court is not in session today.",false,false);
						W.setTravelAttemptTime(0);
						unCuff(W.criminal());
						W.setArrestingOfficer(myArea,null);
						W.setState(Law.STATE_SEEKING);
					}
				}
				else
				{
					if(officer!=null)
                    {
						CMLib.commands().postSay(officer,null,"Wha? Where'd he go?",false,false);
                        fileArrestResister(laws,myArea,W);
                    }
					W.setTravelAttemptTime(0);
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
				}
			}
			break;
		case Law.STATE_WAITING:
			{
				MOB officer=W.arrestingOfficer();
				if((officer!=null)
				&&(!W.criminal().amDead())
				&&(W.criminal().location().isInhabitant(officer))
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(!W.crime().equalsIgnoreCase("pardoned"))
				&&(CMLib.flags().aliveAwakeMobile(officer,true)))
				{
					MOB judge=getTheJudgeHere(laws,officer.location());
					if(judge==null)
					{
						W.setState(Law.STATE_MOVING);
						if(!trackTheJudge(officer,myArea,laws))
						{
							CMLib.commands().postSay(officer,null,"Where was that darn court?!",false,false);
							W.setTravelAttemptTime(0);
							unCuff(W.criminal());
							W.setArrestingOfficer(myArea,null);
							W.setState(Law.STATE_SEEKING);
						}
					}
					else
					if(CMLib.flags().aliveAwakeMobile(judge,true))
					{
						if(judgeMe(laws,judge,officer,W.criminal(),W,myArea,debugging))
						{
							W.setTravelAttemptTime(0);
							unCuff(W.criminal());
							dismissOfficer(officer);
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());
							W.setArrestingOfficer(myArea,null);
						}
						// else, still stuff to do
					}
					else
					{
						CMLib.commands().postSay(officer,W.criminal(),"Court is not in session today.",false,false);
						W.setTravelAttemptTime(0);
						unCuff(W.criminal());
						W.setArrestingOfficer(myArea,null);
						W.setState(Law.STATE_SEEKING);
					}
				}
				else
				{
					if(officer!=null)
                    {
						CMLib.commands().postSay(officer,null,"Wha? Huh?",false,false);
                        fileArrestResister(laws,myArea,W);
                    }
					W.setTravelAttemptTime(0);
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
				}
			}
			break;
		case Law.STATE_PAROLING:
			{
				W.setTravelAttemptTime(0);
				MOB officer=W.arrestingOfficer();
				if((officer!=null)
				&&(!W.criminal().amDead())
				&&(W.criminal().location().isInhabitant(officer))
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(CMLib.flags().aliveAwakeMobile(officer,true))
				&&(!W.crime().equalsIgnoreCase("pardoned"))
				&&(CMLib.flags().canBeSeenBy(W.criminal(),officer)))
				{
					MOB judge=getTheJudgeHere(laws,officer.location());
					fileAllWarrants(laws,W,W.criminal());
					unCuff(W.criminal());
					if((judge!=null)
					&&(CMLib.flags().aliveAwakeMobile(judge,true)))
					{
						judge.location().show(judge,W.criminal(),CMMsg.MSG_OK_VISUAL,"<S-NAME> put(s) <T-NAME> on parole!");
						Ability A=CMClass.getAbility("Prisoner");
						A.startTickDown(judge,W.criminal(),W.jailTime());
						W.criminal().recoverEnvStats();
						W.criminal().recoverCharStats();
						CMLib.commands().postSay(judge,W.criminal(),laws.getMessage(Law.MSG_PAROLEDISMISS),false,false);
						dismissOfficer(officer);
						W.setArrestingOfficer(myArea,null);
						W.criminal().tell("\n\r\n\r");
						if(W.criminal().isMonster())
							CMLib.tracking().wanderAway(W.criminal(),true,true);
					}
					else
					{
						if(officer!=null)
							CMLib.commands().postSay(officer,null,"No court today.",false,false);
						unCuff(W.criminal());
						if(W.arrestingOfficer()!=null)
							dismissOfficer(W.arrestingOfficer());
						W.setArrestingOfficer(myArea,null);
						W.setState(Law.STATE_SEEKING);
					}
				}
				else
				{
					if(officer!=null)
						CMLib.commands().postSay(officer,null,"That was wierd.",false,false);
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
				}
			}
			break;
		case Law.STATE_JAILING:
			{
				MOB officer=W.arrestingOfficer();
				if((officer!=null)
				&&(!W.criminal().amDead())
				&&(W.criminal().location().isInhabitant(officer))
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(CMLib.flags().aliveAwakeMobile(officer,true))
				&&(!W.crime().equalsIgnoreCase("pardoned"))
				&&(CMLib.flags().canBeSeenBy(W.criminal(),officer)))
				{
					Room jail=findTheJail(W.criminal(),myArea,laws);
					if(jail!=null)
					{

						Ability A=W.criminal().fetchEffect("Prisoner");
						if(A!=null){ A.unInvoke(); W.criminal().delEffect(A);}

						makePeace(officer.location());
						W.setJail(jail);
						// cuff him!
						W.setState(Law.STATE_MOVING2);
						A=CMClass.getAbility("Skill_HandCuff");
						if((A!=null)&&(!CMLib.flags().isBoundOrHeld(W.criminal())))
							A.invoke(officer,W.criminal(),true,0);
						W.criminal().makePeace();
						makePeace(officer.location());
						CMLib.commands().postStand(W.criminal(),true);
						CMLib.tracking().stopTracking(officer);
						A=CMClass.getAbility("Skill_Track");
						if(A!=null)
						{
							W.setTravelAttemptTime(System.currentTimeMillis());
							A.setAbilityCode(1);
							A.invoke(officer,CMParms.parse(CMLib.map().getExtendedRoomID(jail)),jail,true,0);
						}
						if(officer.fetchEffect("Skill_Track")==null)
						{
							W.setTravelAttemptTime(0);
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());
							CMLib.commands().postSay(officer,W.criminal(),"I can't find the jail, you are free to go.",false,false);
							dismissOfficer(officer);
							W.setArrestingOfficer(myArea,null);
						}
						makePeace(officer.location());
					}
					else
					{
						W.setTravelAttemptTime(0);
						fileAllWarrants(laws,W,W.criminal());
						unCuff(W.criminal());
						CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),"But since there IS no jail, I will let you go.",false,false);
						dismissOfficer(officer);
						W.setArrestingOfficer(myArea,null);
					}
				}
				else
				{
					if(officer!=null)
                    {
						CMLib.commands().postSay(officer,null,"Crazy.",false,false);
                        fileArrestResister(laws,myArea,W);
                    }
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
					W.setTravelAttemptTime(0);
				}
			}
			break;
            case Law.STATE_DETAINING:
            {
                MOB officer=W.arrestingOfficer();
                if((officer!=null)
                &&(!W.criminal().amDead())
                &&(W.criminal().location().isInhabitant(officer))
                &&(CMLib.flags().isInTheGame(W.criminal(),true))
                &&(CMLib.flags().aliveAwakeMobile(officer,true))
                &&(!W.crime().equalsIgnoreCase("pardoned"))
                &&(CMLib.flags().canBeSeenBy(W.criminal(),officer)))
                {
                    Room jail=findTheDetentionCenter(W.criminal(),myArea,laws,W);
                    int time=getDetainTime(laws,W,W.criminal());
                    if((jail!=null)&&(time>=0))
                    {
                        Ability A=W.criminal().fetchEffect("Prisoner");
                        if(A!=null){ A.unInvoke(); W.criminal().delEffect(A);}

                        makePeace(officer.location());
                        W.setJail(jail);
                        W.setJailTime(time);
                        // cuff him!
                        W.setState(Law.STATE_MOVING3);
                        A=CMClass.getAbility("Skill_HandCuff");
                        W.criminal().baseEnvStats().setDisposition(W.criminal().baseEnvStats().disposition()|EnvStats.IS_SITTING);
                        W.criminal().envStats().setDisposition(W.criminal().envStats().disposition()|EnvStats.IS_SITTING);
                        if((A!=null)&&(!CMLib.flags().isBoundOrHeld(W.criminal())))
                            A.invoke(officer,W.criminal(),true,0);
                        W.criminal().makePeace();
                        makePeace(officer.location());
                        CMLib.commands().postStand(W.criminal(),true);
                        CMLib.tracking().stopTracking(officer);
                        A=CMClass.getAbility("Skill_Track");
                        if(A!=null)
                        {
                            W.setTravelAttemptTime(System.currentTimeMillis());
                            A.setAbilityCode(1);
                            A.invoke(officer,CMParms.parse(CMLib.map().getExtendedRoomID(jail)),jail,true,0);
                        }
                        if(officer.fetchEffect("Skill_Track")==null)
                        {
                            W.setTravelAttemptTime(0);
                            fileAllWarrants(laws,W,W.criminal());
                            unCuff(W.criminal());
                            CMLib.commands().postSay(officer,W.criminal(),"I can't find the detention center, you are free to go.",false,false);
                            dismissOfficer(officer);
                            W.setArrestingOfficer(myArea,null);
                        }
                        makePeace(officer.location());
                    }
                    else
                    {
                        W.setTravelAttemptTime(0);
                        fileAllWarrants(laws,W,W.criminal());
                        unCuff(W.criminal());
                        CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),"But since there IS no detention center, I will let you go.",false,false);
                        dismissOfficer(officer);
                        W.setArrestingOfficer(myArea,null);
                    }
                }
                else
                {
                    if(officer!=null)
                    {
                        CMLib.commands().postSay(officer,null,"Sad.",false,false);
                        fileArrestResister(laws,myArea,W);
                        dismissOfficer(officer);
                    }
                    W.setTravelAttemptTime(0);
                    fileAllWarrants(laws,W,W.criminal());
                    unCuff(W.criminal());
                    W.setArrestingOfficer(myArea,null);
                    W.setState(Law.STATE_SEEKING);
                }
            }
            break;
		case Law.STATE_EXECUTING:
			{
				MOB officer=W.arrestingOfficer();
				if((officer!=null)
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&(!W.criminal().amDead())
				&&(W.criminal().location().isInhabitant(officer))
				&&(CMLib.flags().aliveAwakeMobile(officer,true))
				&&(!W.crime().equalsIgnoreCase("pardoned"))
				&&(CMLib.flags().canBeSeenBy(W.criminal(),officer))
				&&(canFocusOn(officer,W.criminal())))
				{
					MOB judge=getTheJudgeHere(laws,officer.location());
					if((judge!=null)
					&&(CMLib.flags().aliveAwakeMobile(judge,true))
					&&(judge.location()==W.criminal().location()))
					{
						dismissOfficer(officer);
						Ability A=CMClass.getAbility("Prisoner");
						A.startTickDown(judge,W.criminal(),100);
					    A=judge.fetchAbility("Fighter_Behead");
					    if(A==null)A=judge.fetchAbility("Prayer_Stoning");
					    boolean served=false;
					    if(A!=null)
					    {
					        A.setProficiency(100);
					        served=A.invoke(judge,W.criminal(),false,0);
					    }
						fileAllWarrants(laws,W,W.criminal());
						W.criminal().recoverEnvStats();
						W.criminal().recoverCharStats();
						if(!served)
							CMLib.combat().postAttack(judge,W.criminal(),judge.fetchWieldedItem());
						W.setArrestingOfficer(myArea,null);
						W.setTravelAttemptTime(0);
						unCuff(W.criminal());
					}
					else
					{
						if(officer!=null)
							CMLib.commands().postSay(officer,null,"Looks like court is not in session.",false,false);
						W.setTravelAttemptTime(0);
						unCuff(W.criminal());
						if(W.arrestingOfficer()!=null)
							dismissOfficer(W.arrestingOfficer());
						W.setArrestingOfficer(myArea,null);
						W.setState(Law.STATE_SEEKING);
					}
				}
				else
				{
					if(officer!=null)
                    {
						CMLib.commands().postSay(officer,null,"Didn't see that coming.",false,false);
                        fileArrestResister(laws,myArea,W);
                    }
					W.setTravelAttemptTime(0);
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
				}
			}
			break;
		case Law.STATE_MOVING2:
			{
				MOB officer=W.arrestingOfficer();
				if((officer!=null)
				&&(!W.criminal().amDead())
				&&(W.criminal().location().isInhabitant(officer))
				&&(CMLib.flags().isInTheGame(W.criminal(),true))
				&&((W.travelAttemptTime()==0)||((System.currentTimeMillis()-W.travelAttemptTime())<(5*60*1000)))
				&&(CMLib.flags().aliveAwakeMobile(officer,true))
				&&(W.jail()!=null))
				{
					if(W.criminal().curState().getMovement()<50)
						W.criminal().curState().setMovement(50);
					if(officer.curState().getMovement()<50)
						officer.curState().setMovement(50);
					makePeace(officer.location());
					if(officer.isMonster()) CMLib.commands().postLook(officer,true);
					if(W.jail()==W.criminal().location())
					{
						unCuff(W.criminal());
						Ability A=CMClass.getAbility("Prisoner");
						if(A!=null)A.startTickDown(officer,W.criminal(),W.jailTime());
						W.criminal().recoverEnvStats();
						W.criminal().recoverCharStats();
						dismissOfficer(officer);
						if(W.criminal().fetchEffect("Prisoner")==null)
						{
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());
						}
						else
							W.setState(Law.STATE_RELEASE);
					}
					else
					if(CMLib.flags().flaggedAffects(officer,Ability.FLAG_TRACKING).size()==0)
					{
						Ability A=CMClass.getAbility("Skill_Track");
						if(A!=null)
						{
							CMLib.tracking().stopTracking(officer);
							A.setAbilityCode(1); // tells track to cache the path
							A.invoke(officer,CMParms.parse(CMLib.map().getExtendedRoomID(W.jail())),W.jail(),true,0);
						}
						if(officer.fetchEffect("Skill_Track")==null)
						{
							W.setTravelAttemptTime(0);
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());
							CMLib.commands().postSay(officer,W.criminal(),"I lost the jail, so you are free to go.",false,false);
							dismissOfficer(officer);
							W.setArrestingOfficer(myArea,null);
						}
					}
					else
					if((CMLib.dice().rollPercentage()>75)&&(laws.chitChat2().size()>0))
						CMLib.commands().postSay(officer,W.criminal(),(String)laws.chitChat2().elementAt(CMLib.dice().roll(1,laws.chitChat2().size(),-1)),false,false);
				}
				else
				{
                    fileArrestResister(laws,myArea,W);
					unCuff(W.criminal());
					W.setArrestingOfficer(myArea,null);
					W.setState(Law.STATE_SEEKING);
					W.setTravelAttemptTime(0);
				}
			}
			break;
            case Law.STATE_MOVING3:
            {
                MOB officer=W.arrestingOfficer();
                if((officer!=null)
                &&(!W.criminal().amDead())
                &&(W.criminal().location().isInhabitant(officer))
                &&(CMLib.flags().isInTheGame(W.criminal(),true))
                &&((W.travelAttemptTime()==0)||((System.currentTimeMillis()-W.travelAttemptTime())<(5*60*1000)))
                &&(CMLib.flags().aliveAwakeMobile(officer,true))
                &&(W.jail()!=null))
                {
                    if(W.criminal().curState().getMovement()<50)
                        W.criminal().curState().setMovement(50);
                    if(officer.curState().getMovement()<50)
                        officer.curState().setMovement(50);
                    makePeace(officer.location());
                    if(officer.isMonster()) CMLib.commands().postLook(officer,true);
                    if(W.jail()==W.criminal().location())
                    {
                        unCuff(W.criminal());
                        Ability A=CMClass.getAbility("Prisoner");
                        if(A!=null)A.startTickDown(officer,W.criminal(),W.jailTime());
                        W.criminal().recoverEnvStats();
                        W.criminal().recoverCharStats();
                        dismissOfficer(officer);
                        if(W.criminal().fetchEffect("Prisoner")==null)
                        {
                            fileAllWarrants(laws,W,W.criminal());
                            unCuff(W.criminal());
                        }
                        else
                            W.setState(Law.STATE_RELEASE);
                    }
                    else
                    if(CMLib.flags().flaggedAffects(officer,Ability.FLAG_TRACKING).size()==0)
                    {
                        Ability A=CMClass.getAbility("Skill_Track");
                        if(A!=null)
                        {
                            CMLib.tracking().stopTracking(officer);
                            A.setAbilityCode(1); // tells track to cache the path
                            A.invoke(officer,CMParms.parse(CMLib.map().getExtendedRoomID(W.jail())),W.jail(),true,0);
                        }
                        if(officer.fetchEffect("Skill_Track")==null)
                        {
                            W.setTravelAttemptTime(0);
                            fileAllWarrants(laws,W,W.criminal());
                            unCuff(W.criminal());
                            CMLib.commands().postSay(officer,W.criminal(),"I lost the detention center, so you are free to go.",false,false);
                            dismissOfficer(officer);
                            W.setArrestingOfficer(myArea,null);
                        }
                    }
                    else
                    if((CMLib.dice().rollPercentage()>75)&&(laws.chitChat3().size()>0))
                        CMLib.commands().postSay(officer,W.criminal(),(String)laws.chitChat3().elementAt(CMLib.dice().roll(1,laws.chitChat3().size(),-1)),false,false);
                }
                else
                {
                    fileArrestResister(laws,myArea,W);
                    unCuff(W.criminal());
                    W.setArrestingOfficer(myArea,null);
                    W.setState(Law.STATE_SEEKING);
                    W.setTravelAttemptTime(0);
                    fileAllWarrants(laws,W,W.criminal());
                }
            }
            break;
		case Law.STATE_RELEASE:
			{
				if(((W.criminal().fetchEffect("Prisoner")==null)||(W.crime().equalsIgnoreCase("pardoned")))
				&&(W.jail()!=null))
				{
					Ability P=W.criminal().fetchEffect("Prisoner");
					if(P!=null) P.unInvoke();
                    if(CMath.bset(highestCrimeAction(laws,W,W.criminal()),Law.ACTIONMASK_NORELEASE))
                    {
                        W.setTravelAttemptTime(0);
                        fileAllWarrants(laws,W,W.criminal());
                        unCuff(W.criminal());
                        W.setArrestingOfficer(myArea,null);
                    }
                    else
					if(W.criminal().location()==W.jail())
					{
						MOB officer=W.arrestingOfficer();
						if((officer==null)
						||(!CMLib.flags().aliveAwakeMobile(officer,true))
						||(W.criminal().amDead())
						||(!CMLib.flags().isInTheGame(W.criminal(),true))
						||(!W.criminal().location().isInhabitant(officer)))
						{
							W.setArrestingOfficer(myArea,getAnyElligibleOfficer(laws,W.jail().getArea(),W.criminal(),W.victim()));
							if(W.arrestingOfficer()==null) W.setArrestingOfficer(myArea,getAnyElligibleOfficer(laws,myArea,W.criminal(),W.victim()));
							if(W.arrestingOfficer()==null) break;
							officer=W.arrestingOfficer();
							W.jail().bringMobHere(officer,false);
							if(!canFocusOn(officer,W.criminal()))
							{
								W.jail().show(officer,W.criminal(),CMMsg.MSG_QUIETMOVEMENT,"<S-NAME> arrive(s) to release <T-NAME>, but can't find <T-HIM-HER>.");
								dismissOfficer(officer);
								W.setArrestingOfficer(myArea,null);
							}
							else
								W.jail().show(officer,W.criminal(),CMMsg.MSG_QUIETMOVEMENT,"<S-NAME> arrive(s) to release <T-NAME>.");
							Ability A=CMClass.getAbility("Skill_HandCuff");
							if((A!=null)&&(!CMLib.flags().isBoundOrHeld(W.criminal())))
								A.invoke(officer,W.criminal(),true,0);
						}
						W.setReleaseRoom(getReleaseRoom(laws,myArea,W.criminal(),W));
						W.criminal().makePeace();
						makePeace(officer.location());
						CMLib.tracking().stopTracking(officer);
						Ability A=CMClass.getAbility("Skill_Track");
						if(A!=null)
						{
							W.setTravelAttemptTime(System.currentTimeMillis());
							A.invoke(officer,CMParms.parse(CMLib.map().getExtendedRoomID(W.releaseRoom())),W.releaseRoom(),true,0);
						}
						if(officer.fetchEffect("Skill_Track")==null)
						{
							W.setTravelAttemptTime(0);
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());
							CMLib.commands().postSay(officer,W.criminal(),"Well, you can always recall.",false,false);
							dismissOfficer(officer);
							W.setArrestingOfficer(myArea,null);
						}
					}
					else
					if(W.releaseRoom()!=null)
					{
						MOB officer=W.arrestingOfficer();
						if(W.criminal().location()==W.releaseRoom())
						{
							fileAllWarrants(laws,W,W.criminal());
							unCuff(W.criminal());

							if(officer!=null)
							{
								if((CMLib.flags().aliveAwakeMobile(officer,true))
								&&(W.criminal().location().isInhabitant(officer)))
									CMLib.commands().postSay(officer,null,laws.getMessage(Law.MSG_LAWFREE),false,false);
								dismissOfficer(officer);
							}
							W.setTravelAttemptTime(0);
						}
						else
						{
							if((officer!=null)
							&&(CMLib.flags().aliveAwakeMobile(officer,true))
							&&(W.criminal().location().isInhabitant(officer))
							&&((W.travelAttemptTime()==0)||((System.currentTimeMillis()-W.travelAttemptTime())<(5*60*1000))))
							{
								if(officer.isMonster()) CMLib.commands().postLook(officer,true);
								if(W.criminal().curState().getMovement()<20)
									W.criminal().curState().setMovement(20);
								if(officer.curState().getMovement()<20)
									officer.curState().setMovement(20);
								if(W.arrestingOfficer().fetchEffect("Skill_Track")==null)
								{
									CMLib.tracking().stopTracking(officer);
									Ability A=CMClass.getAbility("Skill_Track");
									if(A!=null)	A.invoke(officer,CMParms.parse(CMLib.map().getExtendedRoomID(W.releaseRoom())),W.releaseRoom(),true,0);
									if(W.arrestingOfficer().fetchEffect("Skill_Track")==null)
									{
										W.setTravelAttemptTime(0);
										fileAllWarrants(laws,W,W.criminal());
										unCuff(W.criminal());
										CMLib.commands().postSay(W.arrestingOfficer(),W.criminal(),"Don't worry, you can always recall.",false,false);
										dismissOfficer(W.arrestingOfficer());
										W.setArrestingOfficer(myArea,null);
									}
								}
							}
							else
							{
								if(officer!=null)
									CMLib.commands().postSay(officer,null,"There's always recall.",false,false);
								W.setTravelAttemptTime(0);
								fileAllWarrants(laws,W,W.criminal());
								unCuff(W.criminal());
								if(officer!=null)
									dismissOfficer(officer);
							}
						}
					}
					else
					{
						if(W.arrestingOfficer()!=null)
							CMLib.commands().postSay(W.arrestingOfficer(),null,"Well, he can always recall.",false,false);
						W.setTravelAttemptTime(0);
						fileAllWarrants(laws,W,W.criminal());
						unCuff(W.criminal());
						if(W.arrestingOfficer()!=null)
							dismissOfficer(W.arrestingOfficer());
					}
				}
			}
			break;
		}
	}
}