/
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.Libraries;
import com.planet_ink.coffee_mud.core.interfaces.*;
import com.planet_ink.coffee_mud.core.*;
import com.planet_ink.coffee_mud.Libraries.interfaces.*;
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.io.PrintWriter;
import java.util.*;


/* 
   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 CharCreation extends StdLibrary implements CharCreationLibrary
{
    public String ID(){return "CharCreation";}
    public Hashtable pendingLogins=new Hashtable();
    private DVector autoTitles=null;

    public void reRollStats(MOB mob, CharStats C)
    {
        // from Ashera
        int basemax = CMProps.getIntVar(CMProps.SYSTEMI_BASEMAXSTAT);
        int basemin = 3;

        int points = CMProps.getIntVar(CMProps.SYSTEMI_MAXSTAT);
        // Make sure there are enough points
        if (points < ((basemin + 1) * CharStats.NUM_BASE_STATS))
            points = (basemin + 1) * CharStats.NUM_BASE_STATS;
        
        // Make sure there aren't too many points
        if (points > (basemax - 1) * CharStats.NUM_BASE_STATS) 
                points = (basemax - 1) * CharStats.NUM_BASE_STATS;
       
        int[] stats=new int[CharStats.NUM_BASE_STATS];
        for(int i=0;i<stats.length;i++)
            stats[i]=basemin;
       
        // Subtract stat minimums from point total to get distributable points
        int pointsLeft = points - (basemin * CharStats.NUM_BASE_STATS);

        while (pointsLeft > 0)
        {
            int whichStat = CMLib.dice().roll(1,CharStats.NUM_BASE_STATS,-1);
            if(stats[whichStat]<basemax)
            {
                stats[whichStat]++;
                --pointsLeft;
            }
        }

        for(int i=0;i<CharStats.NUM_BASE_STATS;i++)
            C.setStat(i,stats[i]);
    }

    public boolean classOkForMe(MOB mob, CharClass thisClass, int theme)
    {
        if((CMProps.isTheme(thisClass.availabilityCode()))
           &&(CMath.bset(thisClass.availabilityCode(),theme))
           &&(!CMath.bset(thisClass.availabilityCode(),Area.THEME_SKILLONLYMASK))
           &&((CMProps.getVar(CMProps.SYSTEM_MULTICLASS).startsWith("NO"))
              ||(CMProps.getVar(CMProps.SYSTEM_MULTICLASS).startsWith("MULTI"))
              ||(thisClass.baseClass().equals(thisClass.ID())
              ||(thisClass.ID().equals("Apprentice"))))
           &&thisClass.qualifiesForThisClass(mob,true))
            return true;
        return false;
    }

    public Vector classQualifies(MOB mob, int theme)
    {
        Vector them=new Vector();
        HashSet doneClasses=new HashSet();
        for(Enumeration c=CMClass.charClasses();c.hasMoreElements();)
        {
            CharClass C=(CharClass)c.nextElement();
            if(doneClasses.contains(C.ID())) continue;
            C=CMClass.getCharClass(C.ID());
            doneClasses.add(C.ID());
            if(classOkForMe(mob,C,theme))
                them.addElement(C);
        }
        return them;
    }

    public boolean isOkName(String login)
    {
        if(login.length()>20) return false;
        if(login.length()<3) return false;

        if(login.trim().indexOf(" ")>=0) return false;

        login=login.toUpperCase().trim();
        Vector V=CMParms.parse(login);
        for(int v=V.size()-1;v>=0;v--)
        {
            String str=(String)V.elementAt(v);
            if((" THE A AN ").indexOf(" "+str+" ")>=0)
                V.removeElementAt(v);
        }
        for(int v=0;v<V.size();v++)
        {
            String str=(String)V.elementAt(v);
            if((" YOU SHIT FUCK CUNT ALL FAGGOT ASSHOLE ARSEHOLE PUSSY COCK SLUT BITCH DAMN CRAP GOD JESUS CHRIST NOBODY SOMEBODY MESSIAH ADMIN SYSOP ").indexOf(" "+str+" ")>=0)
                return false;
        }
        Vector V2=CMParms.parseCommas(CMProps.getVar(CMProps.SYSTEM_BADNAMES),true);
        for(int v2=0;v2<V2.size();v2++)
        {
            String str2=(String)V2.elementAt(v2);
            if(str2.length()>0)
            for(int v=0;v<V.size();v++)
            {
                String str=(String)V.elementAt(v);
                if((str.length()>0)
                &&(str.equalsIgnoreCase(str2)))
                    return false;
            }
        }

        for(int c=0;c<login.length();c++)
        {
            char C=Character.toUpperCase(login.charAt(c));
            if(("ABCDEFGHIJKLMNOPQRSTUVWXYZ ").indexOf(C)<0)
                return false;
        }
        for(Enumeration d=CMLib.map().deities();d.hasMoreElements();)
        {
            MOB D=(MOB)d.nextElement();
            if((CMLib.english().containsString(D.ID(),login))
            ||(CMLib.english().containsString(D.Name(),login)))
                return false;
        }
        for(Enumeration m=CMClass.mobTypes();m.hasMoreElements();)
        {
            MOB M=(MOB)m.nextElement();
            if((CMLib.english().containsString(M.Name(),login))
            ||(CMLib.english().containsString(M.name(),login)))
                return false;
        }

        for(Enumeration e=CMLib.clans().clans();e.hasMoreElements();)
        {
            Clan C=(Clan)e.nextElement();
            if((CMLib.english().containsString(C.clanID(),login))
            ||(CMLib.english().containsString(C.name(),login)))
                return false;
        }

        for(Enumeration e=CMLib.map().players();e.hasMoreElements();)
        {
            MOB tm=(MOB)e.nextElement();
            if((CMLib.english().containsString(tm.ID(),login))
            ||(CMLib.english().containsString(tm.Name(),login)))
                return false;

        }
        for(int c=0;c<login.length();c++)
        {
            char C=Character.toUpperCase(login.charAt(c));
            if(("ABCDEFGHIJKLMNOPQRSTUVWXYZ ").indexOf(C)<0)
                return false;
        }
        return !CMSecurity.isBanned(login);
    }

    public void reloadTerminal(MOB mob)
    {
        if(mob==null) return;
        
        Session S=mob.session();
        if(S==null) return;
        
        S.initTelnetMode(mob.getBitmap());
        if((CMath.bset(mob.getBitmap(),MOB.ATT_MXP))
        &&(!CMSecurity.isDisabled("MXP")))
        {
            if(S.clientTelnetMode(Session.TELNET_MXP))
            {
                StringBuffer mxpText=Resources.getFileResource("text/mxp.txt",true);
                if(mxpText!=null)
                    S.rawPrintln("\033[6z"+mxpText.toString()+"\n\r");
            }
            else
                mob.tell("MXP codes have been disabled for this session.");
        }
        else
        if(S.clientTelnetMode(Session.TELNET_MXP))
        {
            S.changeTelnetMode(Session.TELNET_MXP,false);
            S.setClientTelnetMode(Session.TELNET_MXP,false);
        }
        
        if((CMath.bset(mob.getBitmap(),MOB.ATT_SOUND))
        &&(!CMSecurity.isDisabled("MSP")))
        {
            if(!S.clientTelnetMode(Session.TELNET_MSP))
                mob.tell("MSP sounds have been disabled for this session.");
        }
        else
        if(S.clientTelnetMode(Session.TELNET_MSP))
        {
            S.changeTelnetMode(Session.TELNET_MSP,false);
            S.setClientTelnetMode(Session.TELNET_MSP,false);
        }
    }
    
    public void showTheNews(MOB mob)
    {
        reloadTerminal(mob);
        Command C=CMClass.getCommand("PollCmd");
        try{ C.execute(mob,null);}catch(Exception e){}
        
        if((mob.session()==null)
        ||(mob.isMonster())
        ||(CMath.bset(mob.getBitmap(),MOB.ATT_DAILYMESSAGE)))
            return;

        C=CMClass.getCommand("MOTD");
        try{ C.execute(mob,CMParms.parse("MOTD NEW PAUSE"));}catch(Exception e){}
    }

    public boolean checkExpiration(MOB mob)
    {
        if(!CMProps.getBoolVar(CMProps.SYSTEMB_ACCOUNTEXPIRATION)) return true;
        MOB newMob=CMLib.map().getLoadPlayer(mob.Name());
        if(CMSecurity.isASysOp(newMob)) return true;
        if((newMob.playerStats()!=null)
        &&(newMob.playerStats().getAccountExpiration()<=System.currentTimeMillis()))
        {
            mob.tell("\n\r"+CMProps.getVar(CMProps.SYSTEM_EXPCONTACTLINE)+"\n\r\n\r");
            mob.session().setKillFlag(true);
            if(pendingLogins.containsKey(mob.Name().toUpperCase()))
               pendingLogins.remove(mob.Name().toUpperCase());
            return false;
        }
        return true;
    }

    public boolean createCharacter(MOB mob, String login, Session session)
        throws java.io.IOException
    {
        if(session.confirm("\n\r'"+CMStrings.capitalizeAndLower(login)+"' does not exist.\n\rIs this a new character you would like to create (y/N)?","N"))
        {
            login=CMStrings.capitalizeAndLower(login.trim());
            session.println(null,null,null,"\n\r\n\r"+new CMFile(Resources.buildResourcePath("text")+"newchar.txt",null,true).text().toString());
            
            boolean emailPassword=((CMProps.getVar(CMProps.SYSTEM_EMAILREQ).toUpperCase().startsWith("PASS"))&&(CMProps.getVar(CMProps.SYSTEM_MAILBOX).length()>0));

            String password="";
            if(!emailPassword)
            while(password.length()==0)
            {
                password=session.prompt("\n\rEnter a password: ","");
                if(password.length()==0)
                	session.println("\n\rYou must enter a password to continue.");
            }

            mob.setName(login);
            mob.setPlayerStats((PlayerStats)CMClass.getCommon("DefaultPlayerStats"));
            mob.playerStats().setPassword(password);

            boolean emailReq=(!CMProps.getVar(CMProps.SYSTEM_EMAILREQ).toUpperCase().startsWith("OPTION"));
            while(true)
            {
                String newEmail=session.prompt("\n\rEnter your e-mail address:");
                String confirmEmail=newEmail;
                if(emailPassword) session.println("This email address will be used to send you a password.");
                if(emailReq) confirmEmail=session.prompt("Confirm that '"+newEmail+"' is correct by re-entering.\n\rRe-enter:");
                if(((newEmail.length()>6)&&(newEmail.indexOf("@")>0)&&((newEmail.equalsIgnoreCase(confirmEmail))))
                   ||(!emailReq))
                {
                    mob.playerStats().setEmail(newEmail);
                    break;
                }
                session.println("\n\rThat email address combination was invalid.\n\r");
            }
            if((mob.playerStats().getEmail()!=null)&&CMSecurity.isBanned(mob.playerStats().getEmail()))
            {
                session.println("\n\rYou are unwelcome.  No one likes you here. Go away.\n\r\n\r");
                if(mob.session()==session)
                	session.setKillFlag(true);
                if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                   pendingLogins.remove(mob.Name().toUpperCase());
                return false;
            }
            
            Log.sysOut("FrontDoor","Creating user: "+mob.Name());

            mob.setBitmap(MOB.ATT_AUTOEXITS|MOB.ATT_AUTOWEATHER);
            if(session.confirm("\n\rDo you want ANSI colors (Y/n)?","Y"))
                mob.setBitmap(CMath.setb(mob.getBitmap(),MOB.ATT_ANSI));
            else
            {
                mob.setBitmap(CMath.unsetb(mob.getBitmap(),MOB.ATT_ANSI));
                session.setServerTelnetMode(Session.TELNET_ANSI,false);
                session.setClientTelnetMode(Session.TELNET_ANSI,false);
            }
            if((session.clientTelnetMode(Session.TELNET_MSP))
            &&(!CMSecurity.isDisabled("MSP")))
                mob.setBitmap(mob.getBitmap()|MOB.ATT_SOUND);
            if((session.clientTelnetMode(Session.TELNET_MXP))
            &&(!CMSecurity.isDisabled("MXP")))
                mob.setBitmap(mob.getBitmap()|MOB.ATT_MXP);

            int themeCode=CMProps.getIntVar(CMProps.SYSTEMI_MUDTHEME);
            int theme=Area.THEME_FANTASY;
            switch(themeCode)
            {
                case Area.THEME_FANTASY:
                case Area.THEME_HEROIC:
                case Area.THEME_TECHNOLOGY:
                    theme=themeCode;
                    break;
                default:
                    theme=-1;
                    String choices="";
                    String selections="";
                    if(CMath.bset(themeCode,Area.THEME_FANTASY)){ choices+="F"; selections+="/F";}
                    if(CMath.bset(themeCode,Area.THEME_HEROIC)){ choices+="H"; selections+="/H";}
                    if(CMath.bset(themeCode,Area.THEME_TECHNOLOGY)){ choices+="T"; selections+="/T";}
                    if(choices.length()==0)
                    {
                        choices="F";
                        selections="/F";
                    }
                    while((theme<0)&&(!session.killFlag()))
                    {
                    	session.println(null,null,null,new CMFile(Resources.buildResourcePath("text")+"themes.txt",null,true).text().toString());
                    	session.print("\n\r^!Please select from the following:^N "+selections.substring(1)+"\n\r");
                        String themeStr=session.choose(": ",choices,"");
                        if(themeStr.toUpperCase().startsWith("F"))
                            theme=Area.THEME_FANTASY;
                        if(themeStr.toUpperCase().startsWith("H"))
                            theme=Area.THEME_HEROIC;
                        if(themeStr.toUpperCase().startsWith("T"))
                            theme=Area.THEME_TECHNOLOGY;
                    }
                    break;
            }
            if(!CMSecurity.isDisabled("RACES"))
            	session.println(null,null,null,new CMFile(Resources.buildResourcePath("text")+"races.txt",null,true).text().toString());

            StringBuffer listOfRaces=new StringBuffer("[");
            boolean tmpFirst = true;
            HashSet doneRaces=new HashSet();
            for(Enumeration r=CMClass.races();r.hasMoreElements();)
            {
                Race R=(Race)r.nextElement();
                if(doneRaces.contains(R.ID())) continue;
                R=CMClass.getRace(R.ID());
                doneRaces.add(R.ID());
                if((CMProps.isTheme(R.availabilityCode()))
                &&(!CMath.bset(R.availabilityCode(),Area.THEME_SKILLONLYMASK))
                &&(CMath.bset(R.availabilityCode(),theme)))
                {
                    if (!tmpFirst)
                        listOfRaces.append(", ");
                    else
                        tmpFirst = false;
                    listOfRaces.append("^H"+R.name()+"^N");
                }
            }
            listOfRaces.append("]");
            Race newRace=null;
            if(CMSecurity.isDisabled("RACES"))
            {
                newRace=CMClass.getRace("PlayerRace");
                if(newRace==null)
                    newRace=CMClass.getRace("StdRace");
            }
            while(newRace==null)
            {
            	session.print("\n\r^!Please choose from the following races (?):^N\n\r");
            	session.print(listOfRaces.toString());
                String raceStr=session.prompt("\n\r: ","");
                if(raceStr.trim().equalsIgnoreCase("?"))
                	session.println(null,null,null,"\n\r"+new CMFile(Resources.buildResourcePath("text")+"races.txt",null,true).text().toString());
                else
                {
                    newRace=CMClass.getRace(raceStr);
                    if((newRace!=null)&&((!CMProps.isTheme(newRace.availabilityCode()))
                                            ||(!CMath.bset(newRace.availabilityCode(),theme))
                                            ||(CMath.bset(newRace.availabilityCode(),Area.THEME_SKILLONLYMASK))))
                        newRace=null;
                    if(newRace==null)
                        for(Enumeration r=CMClass.races();r.hasMoreElements();)
                        {
                            Race R=(Race)r.nextElement();
                            if((R.name().equalsIgnoreCase(raceStr))
                            &&(CMProps.isTheme(R.availabilityCode()))
                            &&(CMath.bset(R.availabilityCode(),theme))
                            &&(!CMath.bset(R.availabilityCode(),Area.THEME_SKILLONLYMASK)))
                            {
                                newRace=R;
                                break;
                            }
                        }
                    if(newRace==null)
                        for(Enumeration r=CMClass.races();r.hasMoreElements();)
                        {
                            Race R=(Race)r.nextElement();
                            if((R.name().toUpperCase().startsWith(raceStr.toUpperCase()))
                            &&(CMProps.isTheme(R.availabilityCode()))
                            &&(CMath.bset(R.availabilityCode(),theme))
                            &&(!CMath.bset(R.availabilityCode(),Area.THEME_SKILLONLYMASK)))
                            {
                                newRace=R;
                                break;
                            }
                        }
                    if(newRace!=null)
                    {
                        StringBuffer str=CMLib.help().getHelpText(newRace.ID().toUpperCase(),mob,false);
                        if(str!=null) session.println("\n\r^N"+str.toString()+"\n\r");
                        if(!session.confirm("^!Is ^H"+newRace.name()+"^N^! correct (Y/n)?^N","Y"))
                            newRace=null;
                    }
                }
            }
            mob.baseCharStats().setMyRace(newRace);

            mob.baseState().setHitPoints(CMProps.getIntVar(CMProps.SYSTEMI_STARTHP));
            mob.baseState().setMovement(CMProps.getIntVar(CMProps.SYSTEMI_STARTMOVE));
            mob.baseState().setMana(CMProps.getIntVar(CMProps.SYSTEMI_STARTMANA));

            String Gender="";
            while(Gender.length()==0)
                Gender=session.choose("\n\r^!What is your gender (M/F)?^N","MF","");

            mob.baseCharStats().setStat(CharStats.STAT_GENDER,Gender.toUpperCase().charAt(0));
            mob.baseCharStats().getMyRace().startRacing(mob,false);

            if((CMProps.getBoolVar(CMProps.SYSTEMB_ACCOUNTEXPIRATION))&&(mob.playerStats()!=null))
                mob.playerStats().setAccountExpiration(System.currentTimeMillis()+(1000*60*60*24*CMProps.getIntVar(CMProps.SYSTEMI_TRIALDAYS)));

            session.println(null,null,null,"\n\r\n\r"+new CMFile(Resources.buildResourcePath("text")+"stats.txt",null,true).text().toString());

            boolean mayCont=true;
            StringBuffer listOfClasses=new StringBuffer("??? no classes ???");
            while(mayCont)
            {
                reRollStats(mob,mob.baseCharStats());
                mob.recoverCharStats();
                Vector V=classQualifies(mob,theme);
                if(V.size()>0)
                {
                    StringBuffer classes=new StringBuffer("");
                    listOfClasses = new StringBuffer("");
                    for(int v=0;v<V.size();v++)
                        if(v==V.size()-1)
                        {
                            if (v != 0)
                            {
                                classes.append("^?and ^?");
                                listOfClasses.append("^?or ^?");
                            }
                            classes.append(((CharClass)V.elementAt(v)).name());
                            listOfClasses.append(((CharClass)V.elementAt(v)).name());
                        }
                        else
                        {
                            classes.append(((CharClass)V.elementAt(v)).name()+"^?, ^?");
                            listOfClasses.append(((CharClass)V.elementAt(v)).name()+"^?, ^?");
                        }

                    int max=CMProps.getIntVar(CMProps.SYSTEMI_BASEMAXSTAT);
                    StringBuffer statstr=new StringBuffer("Your current stats are: \n\r");
                    CharStats CT=mob.charStats();
                    statstr.append(CMStrings.padRight("Strength",15)+": "+CMStrings.padRight(Integer.toString(CT.getStat(CharStats.STAT_STRENGTH)),2)+"/"+(max+CT.getStat(CharStats.STAT_MAX_STRENGTH_ADJ))+"\n\r");
                    statstr.append(CMStrings.padRight("Intelligence",15)+": "+CMStrings.padRight(Integer.toString(CT.getStat(CharStats.STAT_INTELLIGENCE)),2)+"/"+(max+CT.getStat(CharStats.STAT_MAX_INTELLIGENCE_ADJ))+"\n\r");
                    statstr.append(CMStrings.padRight("Dexterity",15)+": "+CMStrings.padRight(Integer.toString(CT.getStat(CharStats.STAT_DEXTERITY)),2)+"/"+(max+CT.getStat(CharStats.STAT_MAX_DEXTERITY_ADJ))+"\n\r");
                    statstr.append(CMStrings.padRight("Wisdom",15)+": "+CMStrings.padRight(Integer.toString(CT.getStat(CharStats.STAT_WISDOM)),2)+"/"+(max+CT.getStat(CharStats.STAT_MAX_WISDOM_ADJ))+"\n\r");
                    statstr.append(CMStrings.padRight("Constitution",15)+": "+CMStrings.padRight(Integer.toString(CT.getStat(CharStats.STAT_CONSTITUTION)),2)+"/"+(max+CT.getStat(CharStats.STAT_MAX_CONSTITUTION_ADJ))+"\n\r");
                    statstr.append(CMStrings.padRight("Charisma",15)+": "+CMStrings.padRight(Integer.toString(CT.getStat(CharStats.STAT_CHARISMA)),2)+"/"+(max+CT.getStat(CharStats.STAT_MAX_CHARISMA_ADJ))+"\n\r");
                    statstr.append(CMStrings.padRight("TOTAL POINTS",15)+": "+CMProps.getIntVar(CMProps.SYSTEMI_MAXSTAT)+"/"+(CMProps.getIntVar(CMProps.SYSTEMI_BASEMAXSTAT)*6));
                    session.println(statstr.toString());
                    if(!CMSecurity.isDisabled("CLASSES")
                    &&!mob.baseCharStats().getMyRace().classless())
                    	session.println("\n\rThis would qualify you for ^H"+classes.toString()+"^N.");

                    if(!session.confirm("^!Would you like to re-roll (y/N)?^N","N"))
                        mayCont=false;
                }
            }
            if(!CMSecurity.isDisabled("CLASSES")
            &&!mob.baseCharStats().getMyRace().classless())
                session.println(null,null,null,new CMFile(Resources.buildResourcePath("text")+"classes.txt",null,true).text().toString());

            CharClass newClass=null;
            Vector qualClasses=classQualifies(mob,theme);
            if(CMSecurity.isDisabled("CLASSES")||mob.baseCharStats().getMyRace().classless())
            {
                if(CMSecurity.isDisabled("CLASSES"))
                    newClass=CMClass.getCharClass("PlayerClass");
                if((newClass==null)&&(qualClasses.size()>0))
                    newClass=(CharClass)qualClasses.elementAt(CMLib.dice().roll(1,qualClasses.size(),-1));
                if(newClass==null)
                    newClass=CMClass.getCharClass("PlayerClass");
                if(newClass==null)
                    newClass=CMClass.getCharClass("StdCharClass");
            }
            else
            if(qualClasses.size()==0)
            {
                newClass=CMClass.getCharClass("Apprentice");
                if(newClass==null) newClass=CMClass.getCharClass("StdCharClass");
            }
            else
            if(qualClasses.size()==1)
                newClass=(CharClass)qualClasses.firstElement();
            else
            while(newClass==null)
            {
                session.print("\n\r^!Please choose from the following Classes:\n\r");
                session.print("^H[" + listOfClasses.toString() + "]^N");
                String ClassStr=session.prompt("\n\r: ","");
                if(ClassStr.trim().equalsIgnoreCase("?"))
                    session.println(null,null,null,"\n\r"+new CMFile(Resources.buildResourcePath("text")+"classes.txt",null,true).text().toString());
                else
                {
                    newClass=CMClass.findCharClass(ClassStr);
                    if(newClass==null)
                    for(Enumeration c=qualClasses.elements();c.hasMoreElements();)
                    {
                        CharClass C=(CharClass)c.nextElement();
                        if(C.name().equalsIgnoreCase(ClassStr))
                        {
                            newClass=C;
                            break;
                        }
                    }
                    if(newClass==null)
                    for(Enumeration c=qualClasses.elements();c.hasMoreElements();)
                    {
                        CharClass C=(CharClass)c.nextElement();
                        if(C.name().toUpperCase().startsWith(ClassStr.toUpperCase()))
                        {
                            newClass=C;
                            break;
                        }
                    }
                    if((newClass!=null)&&(classOkForMe(mob,newClass,theme)))
                    {
                        StringBuffer str=CMLib.help().getHelpText(newClass.ID().toUpperCase(),mob,false);
                        if(str!=null) session.println("\n\r^N"+str.toString()+"\n\r");
                        if(!session.confirm("^NIs ^H"+newClass.name()+"^N correct (Y/n)?","Y"))
                            newClass=null;
                    }
                    else
                        newClass=null;
                }
            }
            mob.baseEnvStats().setLevel(1);
            mob.baseCharStats().setCurrentClass(newClass);
            mob.baseCharStats().setClassLevel(newClass,1);
            mob.baseEnvStats().setSensesMask(0);


            Item r=CMClass.getItem("Ration");
            Item w=CMClass.getItem("Waterskin");
            Item t=CMClass.getItem("Torch");
            mob.addInventory(r);
            mob.addInventory(w);
            mob.addInventory(t);
            mob.setWimpHitPoint(5);

            CMLib.utensils().outfit(mob,mob.baseCharStats().getMyRace().outfit(mob));

            Ability A=CMClass.getAbility("Allergies");
            if(A!=null) A.invoke(mob,mob,true,0);

            mob.recoverCharStats();
            mob.recoverEnvStats();
            mob.recoverMaxState();
            mob.resetToMaxState();

            Faction F=null;
            Vector mine=null;
            int defaultValue=0;
            for(Enumeration e=CMLib.factions().factionSet().elements();e.hasMoreElements();)
            {
                F=(Faction)e.nextElement();
                mine=F.findChoices(mob);
                defaultValue=F.findAutoDefault(mob);
                if(defaultValue!=Integer.MAX_VALUE)
                    mob.addFaction(F.factionID(),defaultValue);
                if(mine.size()==1)
                    mob.addFaction(F.factionID(),((Integer)mine.firstElement()).intValue());
                else
                if(mine.size()>1)
                {
                    if((F.choiceIntro()!=null)&&(F.choiceIntro().length()>0))
                        session.println(null,null,null,"\n\r\n\r"+new CMFile(Resources.makeFileResourceName(F.choiceIntro()),null,true).text().toString());
                    StringBuffer menu=new StringBuffer("Select one: ");
                    Vector namedChoices=new Vector();
                    for(int m=0;m<mine.size();m++)
                    {
                        Faction.FactionRange FR=CMLib.factions().getRange(F.factionID(),((Integer)mine.elementAt(m)).intValue());
                        if(FR!=null)
                        {
                            namedChoices.addElement(FR.name().toUpperCase());
                            menu.append(FR.name()+", ");
                        }
                        else
                            namedChoices.addElement(""+((Integer)mine.elementAt(m)).intValue());
                    }
                    if(mine.size()==namedChoices.size())
                    {
                        String alignment="";
                        while((!namedChoices.contains(alignment))
                        &&(!session.killFlag()))
                        {
                            alignment=session.prompt(menu.toString().substring(0,menu.length()-2)+".\n\r: ","").toUpperCase();
                            if(!namedChoices.contains(alignment))
                                for(int i=0;i<namedChoices.size();i++)
                                    if(((String)namedChoices.elementAt(i)).startsWith(alignment.toUpperCase()))
                                    { alignment=(String)namedChoices.elementAt(i); break;}
                            if(!namedChoices.contains(alignment))
                                for(int i=0;i<namedChoices.size();i++)
                                    if(((String)namedChoices.elementAt(i)).indexOf(alignment.toUpperCase())>=0)
                                    { alignment=(String)namedChoices.elementAt(i); break;}
                        }
                        if(!session.killFlag())
                        {
                            int valueIndex=namedChoices.indexOf(alignment);
                            if(valueIndex>=0)
                                mob.addFaction(F.factionID(),((Integer)mine.elementAt(valueIndex)).intValue());
                        }
                    }
                }
            }
            mob.baseCharStats().getCurrentClass().startCharacter(mob,false,false);
            CMLib.utensils().outfit(mob,mob.baseCharStats().getCurrentClass().outfit(mob));
            mob.setStartRoom(CMLib.map().getDefaultStartRoom(mob));
            mob.baseCharStats().setStat(CharStats.STAT_AGE,mob.playerStats().initializeBirthday(0,mob.baseCharStats().getMyRace()));
            session.println(null,null,null,"\n\r\n\r"+new CMFile(Resources.buildResourcePath("text")+"newchardone.txt",null,true).text().toString());
            session.prompt("");
            boolean logoff=false;
            if(emailPassword)
            {
                password="";
                for(int i=0;i<6;i++)
                    password+=(char)('a'+CMLib.dice().roll(1,26,-1));
                mob.playerStats().setPassword(password);
                CMLib.database().DBUpdatePassword(mob);
                CMLib.database().DBWriteJournal(CMProps.getVar(CMProps.SYSTEM_MAILBOX),
                          mob.Name(),
                          mob.Name(),
                          "Password for "+mob.Name(),
                          "Your password for "+mob.Name()+" is: "+mob.playerStats().password()+"\n\rYou can login by pointing your mud client at "+CMProps.getVar(CMProps.SYSTEM_MUDDOMAIN)+" port(s):"+CMProps.getVar(CMProps.SYSTEM_MUDPORTS)+".\n\rYou may use the PASSWORD command to change it once you are online.",-1);
                session.println("Your account has been created.  You will receive an email with your password shortly.");
                try{Thread.sleep(2000);}catch(Exception e){}
                if(mob.session()==session)
	                session.setKillFlag(true);
            }
            else
            {
            	if(mob.session()==session)
	                reloadTerminal(mob);
                mob.bringToLife(mob.getStartRoom(),true);
                mob.location().showOthers(mob,mob.location(),CMMsg.MASK_ALWAYS|CMMsg.MSG_ENTER,"<S-NAME> appears!");
            }
            mob.playerStats().leveledDateTime(0);
            CMLib.database().DBCreateCharacter(mob);
            CMLib.map().addPlayer(mob);

            if((session==null)||(mob.playerStats()==null)) return false;
            mob.playerStats().setLastIP(session.getAddress());
            Log.sysOut("FrontDoor","Created user: "+mob.Name());
            CMProps.addNewUserByIP(session.getAddress());
            for(int s=0;s<CMLib.sessions().size();s++)
            {
                Session S=CMLib.sessions().elementAt(s);
                if((S!=null)
                &&(S.mob()!=null)
                &&((!CMLib.flags().isCloaked(mob))||(CMSecurity.isASysOp(S.mob())))
                &&(CMath.bset(S.mob().getBitmap(),MOB.ATT_AUTONOTIFY))
                &&(S.mob().playerStats()!=null)
                &&((S.mob().playerStats().getFriends().contains(mob.Name())||S.mob().playerStats().getFriends().contains("All"))))
                    S.mob().tell("^X"+mob.Name()+" has just been created.^.^?");
            }
            if((CMProps.getVar(CMProps.SYSTEM_PKILL).startsWith("ALWAYS"))
            &&(!CMath.bset(mob.getBitmap(),MOB.ATT_PLAYERKILL)))
                mob.setBitmap(mob.getBitmap()|MOB.ATT_PLAYERKILL);
            if((CMProps.getVar(CMProps.SYSTEM_PKILL).startsWith("NEVER"))
            &&(CMath.bset(mob.getBitmap(),MOB.ATT_PLAYERKILL)))
                mob.setBitmap(mob.getBitmap()-MOB.ATT_PLAYERKILL);
            CMLib.database().DBUpdatePlayer(mob);
            Vector channels=CMLib.channels().getFlaggedChannelNames("NEWPLAYERS");
            for(int i=0;i<channels.size();i++)
                CMLib.commands().postChannel((String)channels.elementAt(i),mob.getClanID(),mob.Name()+" has just been created.",true);
            CMLib.coffeeTables().bump(mob,CoffeeTableRow.STAT_LOGINS);
            CMLib.coffeeTables().bump(mob,CoffeeTableRow.STAT_NEWPLAYERS);
            if(pendingLogins.containsKey(mob.Name().toUpperCase()))
               pendingLogins.remove(mob.Name().toUpperCase());
            return !logoff;
        }
        if(pendingLogins.containsKey(mob.Name().toUpperCase()))
            pendingLogins.remove(mob.Name().toUpperCase());
         return false;
    }
    
    private boolean loginsDisabled(MOB mob)
    {
        if((CMSecurity.isDisabled("LOGINS"))&&(!CMSecurity.isASysOp(mob)))
        {
			StringBuffer rejectText=Resources.getFileResource("text/nologins.txt",true);
			if((rejectText!=null)&&(rejectText.length()>0))
				mob.session().println(rejectText.toString());
			try{Thread.sleep(1000);}catch(Exception e){}
            mob.session().setKillFlag(true);
            if(pendingLogins.containsKey(mob.Name().toUpperCase()))
               pendingLogins.remove(mob.Name().toUpperCase());
            return true;
        }
        return false;
    }
    
    
    // 0=no login, 1=normal login, 2=swap
    public int login(MOB mob, int attempt)
        throws java.io.IOException
    {
        if(mob==null) return 0;
        if(mob.session()==null) return 0;

        boolean wizi=false;

        String login=mob.session().prompt("name: ");
        if(login==null) return 0;
        login=login.trim();
        if(login.length()==0) return 0;
        if(login.endsWith(" !"))
        {
            login=login.substring(0,login.length()-2);
            login=login.trim();
            wizi=true;
        }
        
        boolean found=CMLib.database().DBUserSearch(mob,login);
        if(found)
        {
            mob.session().print("password: ");
            String password=mob.session().blockingIn();
            PlayerStats pstats=mob.playerStats();

            if((pstats!=null)
            &&(pstats.password().equalsIgnoreCase(password))
            &&(mob.Name().trim().length()>0))
            {
                if(CMSecurity.isBanned(mob.Name()))
                {
                    mob.tell("\n\rYou are unwelcome.  No one likes you here. Go away.\n\r\n\r");
                    mob.session().setKillFlag(true);
                    if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                       pendingLogins.remove(mob.Name().toUpperCase());
                    return 0;
                }
                if(((pstats.getEmail()==null)||(pstats.getEmail().length()==0))
                   &&(!CMProps.getVar(CMProps.SYSTEM_EMAILREQ).toUpperCase().startsWith("OPTION")))
                {
                    Command C=CMClass.getCommand("Email");
                    if(C!=null)
                    {
                        if(!C.execute(mob,null))
                            return 0;
                    }
                    CMLib.database().DBUpdateEmail(mob);
                }
                if((pstats.getEmail()!=null)&&CMSecurity.isBanned(pstats.getEmail()))
                {
                    mob.tell("\n\rYou are unwelcome.  No one likes you here. Go away.\n\r\n\r");
                    mob.session().setKillFlag(true);
                    if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                       pendingLogins.remove(mob.Name().toUpperCase());
                    return 0;
                }
                if(!checkExpiration(mob)) return 0;
                Long L=(Long)pendingLogins.get(mob.Name().toUpperCase());
                if((L!=null)&&((System.currentTimeMillis()-L.longValue())<(10*60*1000)))
                {
                    mob.session().println("A previous login is still pending.  Please be patient.");
                    return 0;
                }
                if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                   pendingLogins.remove(mob.Name().toUpperCase());
                pendingLogins.put(mob.Name().toUpperCase(),new Long(System.currentTimeMillis()));

                for(int s=0;s<CMLib.sessions().size();s++)
                {
                    Session thisSession=CMLib.sessions().elementAt(s);
                    if((thisSession.mob()!=null)&&(thisSession!=mob.session()))
                    {
                        if((thisSession.mob().Name().equals(mob.Name())))
                        {
                            Room oldRoom=thisSession.mob().location();
                            if(oldRoom!=null)
                            while(oldRoom.isInhabitant(thisSession.mob()))
                                oldRoom.delInhabitant(thisSession.mob());
                            mob.session().setMob(thisSession.mob());
                            thisSession.mob().setSession(mob.session());
                            thisSession.setMob(null);
                            thisSession.setKillFlag(true);
                            Log.sysOut("FrontDoor","Session swap for "+mob.session().mob().Name()+".");
                            reloadTerminal(mob.session().mob());
                            mob.session().mob().bringToLife(oldRoom,false);
                            if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                               pendingLogins.remove(mob.Name().toUpperCase());
                            return 2;
                        }
                    }
                }

                // count number of multiplays
                int numAtAddress=0;
                try{
                for(int s=0;s<CMLib.sessions().size();s++)
                {
                    if((CMLib.sessions().elementAt(s)!=mob.session())
                    &&(mob.session().getAddress().equalsIgnoreCase((CMLib.sessions().elementAt(s).getAddress()))))
                        numAtAddress++;
                }
                }catch(Exception e){}
                
                if((CMProps.getIntVar(CMProps.SYSTEMI_MAXCONNSPERIP)>0)
                &&(numAtAddress>=CMProps.getIntVar(CMProps.SYSTEMI_MAXCONNSPERIP))
                &&(!CMSecurity.isDisabled("MAXCONNSPERIP")))
                {
                    mob.session().println("The maximum player limit has already been reached for your IP address.");
                    if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                        pendingLogins.remove(mob.Name().toUpperCase());
                    return 0;
                }
                
                MOB oldMOB=mob;
                if(CMLib.map().getPlayer(oldMOB.Name())!=null)
                {
                    oldMOB.session().setMob(CMLib.map().getPlayer(oldMOB.Name()));
                    mob=oldMOB.session().mob();
                    mob.setSession(oldMOB.session());
                    if(mob!=oldMOB)
                        oldMOB.setSession(null);
                    if(loginsDisabled(mob)) 
                    	return 0;
                    if(wizi)
                    {
                        Command C=CMClass.getCommand("WizInv");
                        if((C!=null)&&(C.securityCheck(mob)||C.securityCheck(mob)))
                            C.execute(mob,CMParms.makeVector("WIZINV"));
                    }
                    showTheNews(mob);
                    mob.bringToLife(mob.location(),false);
                    CMLib.coffeeTables().bump(mob,CoffeeTableRow.STAT_LOGINS);
                    mob.location().showOthers(mob,mob.location(),CMMsg.MASK_ALWAYS|CMMsg.MSG_ENTER,"<S-NAME> appears!");
                    for(int f=0;f<mob.numFollowers();f++)
                    {
                        MOB follower=mob.fetchFollower(f);
                        Room R=follower.location();
                        if((follower!=null)
                        &&(follower.isMonster())
                        &&(!follower.isPossessing())
                        &&((R==null)||(!R.isInhabitant(follower))))
                        {
                            if(R==null) R=mob.location();
                            follower.setLocation(R);
                            follower.setFollowing(mob); // before for bestow names sake
                            follower.bringToLife(R,false);
                            follower.setFollowing(mob);
                            R.showOthers(follower,R,CMMsg.MASK_ALWAYS|CMMsg.MSG_ENTER,"<S-NAME> appears!");
                        }
                    }
                }
                else
                {
                    CMLib.database().DBReadPlayer(mob);
                    if(loginsDisabled(mob)) 
                    	return 0;
                    if(wizi)
                    {
                        Command C=CMClass.getCommand("WizInv");
                        if((C!=null)&&(C.securityCheck(mob)||C.securityCheck(mob)))
                            C.execute(mob,CMParms.makeVector("WIZINV"));
                    }
                    showTheNews(mob);
                    mob.bringToLife(mob.location(),true);
                    CMLib.coffeeTables().bump(mob,CoffeeTableRow.STAT_LOGINS);
                    mob.location().showOthers(mob,mob.location(),CMMsg.MASK_ALWAYS|CMMsg.MSG_ENTER,"<S-NAME> appears!");
                    CMLib.database().DBReadFollowers(mob,true);
                }
                if((mob.session()!=null)&&(mob.playerStats()!=null))
                    mob.playerStats().setLastIP(mob.session().getAddress());
                for(int s=0;s<CMLib.sessions().size();s++)
                {
                    Session S=CMLib.sessions().elementAt(s);
                    if((S!=null)
                    &&(S.mob()!=null)
                    &&(S.mob()!=mob)
                    &&((!CMLib.flags().isCloaked(mob))||(CMSecurity.isASysOp(S.mob())))
                    &&(CMath.bset(S.mob().getBitmap(),MOB.ATT_AUTONOTIFY))
                    &&(S.mob().playerStats()!=null)
                    &&((S.mob().playerStats().getFriends().contains(mob.Name())||S.mob().playerStats().getFriends().contains("All"))))
                        S.mob().tell("^X"+mob.Name()+" has logged on.^.^?");
                }
                if((CMProps.getVar(CMProps.SYSTEM_PKILL).startsWith("ALWAYS"))
                &&(!CMath.bset(mob.getBitmap(),MOB.ATT_PLAYERKILL)))
                    mob.setBitmap(mob.getBitmap()|MOB.ATT_PLAYERKILL);
                if((CMProps.getVar(CMProps.SYSTEM_PKILL).startsWith("NEVER"))
                &&(CMath.bset(mob.getBitmap(),MOB.ATT_PLAYERKILL)))
                    mob.setBitmap(mob.getBitmap()-MOB.ATT_PLAYERKILL);
                Vector channels=CMLib.channels().getFlaggedChannelNames("LOGINS");
                if(!CMLib.flags().isCloaked(mob))
                for(int i=0;i<channels.size();i++)
                    CMLib.commands().postChannel((String)channels.elementAt(i),mob.getClanID(),mob.Name()+" has logged on.",true);
                if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                   pendingLogins.remove(mob.Name().toUpperCase());
            }
            else
            {
                String name=mob.Name();
                Log.sysOut("FrontDoor","Failed login: "+mob.Name());
                mob.setName("");
                mob.setPlayerStats(null);
                mob.session().println("\n\rInvalid password.\n\r");
                if(pendingLogins.containsKey(mob.Name().toUpperCase()))
                   pendingLogins.remove(mob.Name().toUpperCase());
                if((!mob.session().killFlag())
                &&(pstats!=null)
                &&(pstats.getEmail().length()>0)
                &&(pstats.getEmail().indexOf("@")>0)
                &&(attempt>2)
                &&(CMProps.getVar(CMProps.SYSTEM_MUDDOMAIN).length()>0))
                {
                    if(mob.session().confirm("Would you like you have your password e-mailed to you (y/N)? ","N"))
                    {
                        if(CMLib.smtp().emailIfPossible(CMProps.getVar(CMProps.SYSTEM_SMTPSERVERNAME),
                                                   "passwords@"+CMProps.getVar(CMProps.SYSTEM_MUDDOMAIN).toLowerCase(),
                                                   "noreply@"+CMProps.getVar(CMProps.SYSTEM_MUDDOMAIN).toLowerCase(),
                                                   pstats.getEmail(),
                                                   "Password for "+name,
                                                   "Your password for "+name+" at "+CMProps.getVar(CMProps.SYSTEM_MUDDOMAIN)+" is: '"+pstats.password()+"'."))
                            mob.session().println("Email sent.\n\r");
                        else
                            mob.session().println("Error sending email.\n\r");
                        mob.session().setKillFlag(true);
                    }
                }
                return 0;
            }
        }
        else
        {
            if(CMSecurity.isDisabled("NEWPLAYERS"))
            {
                mob.session().print("\n\r'"+CMStrings.capitalizeAndLower(login)+"' is not recognized.\n\r");
                mob.setName("");
                mob.setPlayerStats(null);
            }
            else
            if(!isOkName(login))
            {
                mob.session().println("\n\rThat name is unrecognized.\n\rThat name is also not available for new players.\n\r  Choose another name (no spaces allowed)!\n\r");
                mob.setName("");
                mob.setPlayerStats(null);
            }
            else
            if((CMProps.getIntVar(CMProps.SYSTEMI_MUDTHEME)==0)
            ||(CMSecurity.isDisabled("LOGINS")))
            {
                mob.session().print("\n\r'"+CMStrings.capitalizeAndLower(login)+"' does not exist.\n\rThis server is not accepting new accounts.\n\r\n\r");
                mob.setName("");
                mob.setPlayerStats(null);
            }
            else
            if((CMProps.getIntVar(CMProps.SYSTEMI_MAXNEWPERIP)>0)
            &&(CMProps.getCountNewUserByIP(mob.session().getAddress())>=CMProps.getIntVar(CMProps.SYSTEMI_MAXNEWPERIP))
            &&(!CMSecurity.isDisabled("MAXCONNSPERIP")))
            {
                mob.session().println("\n\rThat name is unrecognized.\n\rAlso, the maximum daily new player limit has already been reached for your location.");
                mob.setName("");
                mob.setPlayerStats(null);
            }
            else
                return createCharacter(mob,login,mob.session())?1:0;
            if(pendingLogins.containsKey(mob.Name().toUpperCase()))
               pendingLogins.remove(mob.Name().toUpperCase());
            return 0;
        }
        if((mob!=null)&&(mob.session()!=null))
            mob.session().println("\n\r");
        if(pendingLogins.containsKey(mob.Name().toUpperCase()))
           pendingLogins.remove(mob.Name().toUpperCase());
        return 1;
    }
    
    public String evaluateAutoTitle(String row, boolean addIfPossible)
    {
        if(row.trim().startsWith("#")||row.trim().startsWith(";")||(row.trim().length()==0)) return null;
        int x=row.indexOf("=");
        while((x>=1)&&(row.charAt(x-1)=='\\')) x=row.indexOf("=",x+1);
        if(x<0) return "Error: Invalid line! Not comment, whitespace, and does not contain an = sign!"; 
        String title=row.substring(0,x).trim();
        String mask=row.substring(x+1).trim();
        
        if(title.length()==0)return "Error: Blank title: "+title+"="+mask+"!";
        if(mask.length()==0)return "Error: Blank mask: "+title+"="+mask+"!";
        if(autoTitles.contains(title)) return "Error: Duplicate title: "+title+"="+mask+"!";
        autoTitles.addElement(title,mask,CMLib.masking().maskCompile(mask));
        return null;
    }
    public boolean isExistingAutoTitle(String title)
    {
        if(autoTitles==null) reloadAutoTitles();
        for(int v=0;v<autoTitles.size();v++)
            if(((String)autoTitles.elementAt(v,1)).toUpperCase().equalsIgnoreCase(title.trim()))
                return true;
        return false;
    }
    
    public Enumeration autoTitles()
    {
        if(autoTitles==null) reloadAutoTitles();
        return autoTitles.getDimensionVector(1).elements();
    }
    
    public String getAutoTitleMask(String title)
    {
        if(autoTitles==null) reloadAutoTitles();
        int x=autoTitles.indexOf(title);
        if(x<0) return "";
        return (String)autoTitles.elementAt(x,2);
    }
    
    
    public boolean evaluateAutoTitles(MOB mob)
    {
        if(mob==null) return false;
        PlayerStats P=mob.playerStats();
        if(P==null) return false;
        if(autoTitles==null) reloadAutoTitles();
        String title=null;
        Vector mask=null;
        int pdex=0;
        Vector PT=P.getTitles();
        boolean somethingDone=false;
        for(int t=0;t<autoTitles.size();t++)
        {
            mask=(Vector)autoTitles.elementAt(t,3);
            title=(String)autoTitles.elementAt(t,1);
            pdex=PT.contains(title)?t:-1;
            if(CMLib.masking().maskCheck(mask,mob,true))
            {
                if(pdex<0)
                {
                    if(PT.size()>0)
                        PT.insertElementAt(title,0);
                    else
                        PT.addElement(title);
                    somethingDone=true;
                }
            }
            else
            if(pdex>=0){ somethingDone=true; PT.removeElementAt(pdex);}
        }
        return somethingDone;
    }
    public void reloadAutoTitles()
    {
        autoTitles=new DVector(3);
        Vector V=Resources.getFileLineVector(Resources.getFileResource("titles.txt",true));
        String WKID=null;
        for(int v=0;v<V.size();v++)
        {
            String row=(String)V.elementAt(v);
            WKID=evaluateAutoTitle(row,true);
            if(WKID==null) continue;
            if(WKID.startsWith("Error: "))
                Log.errOut("CharCreation",WKID);
        }
        for(Enumeration e=CMLib.map().players();e.hasMoreElements();)
        {
            MOB M=(MOB)e.nextElement();
            if(M.playerStats()!=null)
            {
                if((evaluateAutoTitles(M))&&(!CMLib.flags().isInTheGame(M,true)))
                    CMLib.database().DBUpdatePlayerStatsOnly(M);
            }
        }
    }
}