var digits = "0123456789";
var lowercaseLetters = "abcdefghijklmnopqrstuvwxyz"
var uppercaseLetters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
//var whitespace = " \t\n\r";
var whitespace = " ";
var decimalPointDelimiter = "."
var phoneNumberDelimiters = "()- ";
var validUSPhoneChars = digits + phoneNumberDelimiters;
var validWorldPhoneChars = digits + phoneNumberDelimiters + "+";
var SSNDelimiters = "- ";
var validSSNChars = digits + SSNDelimiters;
var digitsInSocialSecurityNumber = 9;
var digitsInUSPhoneNumber = 10;
var ZIPCodeDelimiters = "-";
var ZIPCodeDelimeter = "-"
var validZIPCodeChars = digits + ZIPCodeDelimiters
var digitsInZIPCode1 = 5
var digitsInZIPCode2 = 9
var creditCardDelimiters = " "

var mPrefix = "You did not enter a value into the '"
var mSuffix = "' field. This is a required field. Please enter it now."
var sUSLastName = "Last Name"
var sUSFirstName = "First Name"
var sWorldLastName = "Family Name"
var sWorldFirstName = "Given Name"
var sTitle = "Title"
var sCompanyName = "Company Name"
var sUSAddress = "Street Address"
var sWorldAddress = "Address"
var sCity = "City"
var sStateCode = "State Code"
var sWorldState = "State, Province, or Prefecture"
var sCountry = "Country"
var sZIPCode = "ZIP Code"
var sWorldPostalCode = "Postal Code"
var sPhone = "Phone Number"
var sFax = "Fax Number"
var sDateOfBirth = "Date of Birth"
var sExpirationDate = "Expiration Date"
var sEmail = "Email"
var sSSN = "Social Security Number"
var sCreditCardNumber = "Credit Card Number"
var sOtherInfo = "Other Information"

var iStateCode = "This field must be a valid two character U.S. state abbreviation (like MA for Massachusetts). Please reenter it now."
var iZIPCode = "This field must be a 5 or 9 digit U.S. ZIP Code (like 02115). Please reenter it now."
var iUSPhone = "This field must be a 10 digit U.S. phone number (like 617 555 1212). Please reenter it now."
var iWorldPhone = "This field must be a valid international phone number. Please reenter it now."
var iSSN = "This field must be a 9 digit U.S. social security number (like 123 45 6789). Please reenter it now."
var iEmail = "This field must be a valid email address (like john@company.com). Please reenter it now."
var iCreditCardPrefix = "This is not a valid "
var iCreditCardSuffix = " credit card number. (Click the link on this form to see a list of sample numbers.) Please reenter it now."
var iDay = "This field must be a day number between 1 and 31.  Please reenter it now."
var iMonth = "This field must be a month number between 1 and 12.  Please reenter it now."
var iMonthPartial = "This field must be a month number between 0 and 12.  Please reenter it now."
var iYear = "This field must be a 2 or 4 digit year number. Please reenter it now."
var iDatePrefix = "The Month, Day, and Year for "
var iDateSuffix = " do not form a valid date.  Please reenter them now."
var iTimePrefix = "The Hours and Minutes for "
var iTimeSuffix = " do not form a valid time.  Please reenter them now."
var iHour = "This field must be an hour number between 1 and 12.  Please reenter it now."
var iMinute = "This field must be an minute number between 0 and 59.  Please reenter it now."
var iAmPm = "This field must be either 'am' or 'pm'.  Please reenter it now."
var iPositiveInteger = "This field must be a positive integer (greater than zero).  Please reenter it now."
var iNonNegativeInteger = "This field must be an integer greater than or equal to zero. Please reenter it now."
var iPositiveNumber = "This field must be a positive number (greater than zero).  Please reenter it now."
var iNonNegativeNumber = "This field must be a number greater than or equal to zero.  Please reenter it now."
var iNumber = "This field must be a number.  Please reenter it now."

var pEntryPrompt = "Please enter a "
var pStateCode = "2 character code (like MA)."
var pZIPCode = "5 or 9 digit U.S. ZIP Code (like 02115)."
var pUSPhone = "10 digit U.S. phone number (like 617 555 1212)."
var pWorldPhone = "international phone number."
var pSSN = "9 digit U.S. social security number (like 123 45 6789)."
var pEmail = "valid email address (like foo@bar.com)."
var pCreditCard = "valid credit card number."
var pDay = "day number between 1 and 31."
var pMonth = "month number between 1 and 12."
var pYear = "2 or 4 digit year number."

var defaultCurrencySymbol="$"
var defaultEmptyOK = false
function makeArray(n) {
//*** BUG: If I put this line in, I get two error messages:
//(1) Window.length can't be set by assignment
//(2) daysInMonth has no property indexed by 4
//If I leave it out, the code works fine.
//   this.length = n;
   for (var i = 1; i <= n; i++) {
      this[i] = 0
   } 
   return this
}


var daysInMonth = makeArray(12);
daysInMonth[1] = 31;
daysInMonth[2] = 29;   // must programmatically check this
daysInMonth[3] = 31;
daysInMonth[4] = 30;
daysInMonth[5] = 31;
daysInMonth[6] = 30;
daysInMonth[7] = 31;
daysInMonth[8] = 31;
daysInMonth[9] = 30;
daysInMonth[10] = 31;
daysInMonth[11] = 30;
daysInMonth[12] = 31;

// Valid U.S. Postal Codes for states, territories, armed forces, etc.
// See http://www.usps.gov/ncsc/lookups/abbr_state.txt.

var USStateCodeDelimiter = "|";
var USStateCodes = "AL|AK|AS|AZ|AR|CA|CO|CT|DE|DC|FM|FL|GA|GU|HI|ID|IL|IN|IA|KS|KY|LA|ME|MH|MD|MA|MI|MN|MS|MO|MT|NE|NV|NH|NJ|NM|NY|NC|ND|MP|OH|OK|OR|PW|PA|PR|RI|SC|SD|TN|TX|UT|VT|VI|VA|WA|WV|WI|WY|AE|AA|AE|AE|AP"


// Check whether string s is empty.

function isEmpty(s)
{   return ((s == null) || (s.length == 0))
}

// Returns true if string s is empty or 
// whitespace characters only.

function isWhitespace (s)

{   var i;

    // Is s empty?
    if (isEmpty(s)) return true;

    // Search through string's characters one by one
    // until we find a non-whitespace character.
    // When we do, return false; if we don't, return true.

    for (i = 0; i < s.length; i++)
    {   
        // Check that current character isn't whitespace.
        var c = s.charAt(i);

        if (whitespace.indexOf(c) == -1) return false;
    }

    // All characters are whitespace.
    return true;
}



// Removes all characters which appear in string bag from string s.

function stripCharsInBag (s, bag)

{   var i;
    var returnString = "";

    // Search through string's characters one by one.
    // If character is not in bag, append to returnString.

    for (i = 0; i < s.length; i++)
    {   
        // Check that current character isn't whitespace.
        var c = s.charAt(i);
        if (bag.indexOf(c) == -1) returnString += c;
    }

    return returnString;
}



// Removes all characters which do NOT appear in string bag 
// from string s.

function stripCharsNotInBag (s, bag)

{   var i;
    var returnString = "";

    // Search through string's characters one by one.
    // If character is in bag, append to returnString.

    for (i = 0; i < s.length; i++)
    {   
        // Check that current character isn't whitespace.
        var c = s.charAt(i);
        if (bag.indexOf(c) != -1) returnString += c;
    }

    return returnString;
}



// Removes all whitespace characters from s.
// Global variable whitespace (see above)
// defines which characters are considered whitespace.

function stripWhitespace (s)

{   return stripCharsInBag (s, whitespace)
}




// WORKAROUND FUNCTION FOR NAVIGATOR 2.0.2 COMPATIBILITY.
//
// The below function *should* be unnecessary.  In general,
// avoid using it.  Use the standard method indexOf instead.
//
// However, because of an apparent bug in indexOf on 
// Navigator 2.0.2, the below loop does not work as the
// body of stripInitialWhitespace:
//
// while ((i < s.length) && (whitespace.indexOf(s.charAt(i)) != -1))
//   i++;
//
// ... so we provide this workaround function charInString
// instead.
//
// charInString (CHARACTER c, STRING s)
//
// Returns true if single character c (actually a string)
// is contained within string s.

function charInString (c, s)
{   for (i = 0; i < s.length; i++)
    {   if (s.charAt(i) == c) return true;
    }
    return false
}



// Removes initial (leading) whitespace characters from s.
// Global variable whitespace (see above)
// defines which characters are considered whitespace.

function stripInitialWhitespace (s)

{   var i = 0;

    while ((i < s.length) && charInString (s.charAt(i), whitespace))
       i++;
    
    return s.substring (i, s.length);
}







// Returns true if character c is an English letter 
// (A .. Z, a..z).
//
// NOTE: Need i18n version to support European characters.
// This could be tricky due to different character
// sets and orderings for various languages and platforms.

function isLetter (c)
{   return ( ((c >= "a") && (c <= "z")) || ((c >= "A") && (c <= "Z")) )
}



// Returns true if character c is a digit 
// (0 .. 9).

function isDigit (c)
{   return ((c >= "0") && (c <= "9"))
}



// Returns true if character c is a letter or digit.

function isLetterOrDigit (c)
{   return (isLetter(c) || isDigit(c))
}



// isInteger (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if all characters in string s are numbers.
//
// Accepts non-signed integers only. Does not accept floating 
// point, exponential notation, etc.
//
// We don't use parseInt because that would accept a string
// with trailing non-numeric characters.
//
// By default, returns defaultEmptyOK if s is empty.
// There is an optional second argument called emptyOK.
// emptyOK is used to override for a single function call
//      the default behavior which is specified globally by
//      defaultEmptyOK.
// If emptyOK is false (or any value other than true), 
//      the function will return false if s is empty.
// If emptyOK is true, the function will return true if s is empty.
//
// EXAMPLE FUNCTION CALL:     RESULT:
// isInteger ("5")            true 
// isInteger ("")             defaultEmptyOK
// isInteger ("-5")           false
// isInteger ("", true)       true
// isInteger ("", false)      false
// isInteger ("5", false)     true

function isInteger (s)

{   var i;

    if (isEmpty(s)) 
       if (isInteger.arguments.length == 1) return defaultEmptyOK;
       else return (isInteger.arguments[1] == true);

    // Search through string's characters one by one
    // until we find a non-numeric character.
    // When we do, return false; if we don't, return true.

    for (i = 0; i < s.length; i++)
    {   
        // Check that current character is number.
        var c = s.charAt(i);

        if (!isDigit(c)) return false;
    }

    // All characters are numbers.
    return true;
}







// isSignedInteger (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if all characters are numbers; 
// first character is allowed to be + or - as well.
//
// Does not accept floating point, exponential notation, etc.
//
// We don't use parseInt because that would accept a string
// with trailing non-numeric characters.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.
//
// EXAMPLE FUNCTION CALL:          RESULT:
// isSignedInteger ("5")           true 
// isSignedInteger ("")            defaultEmptyOK
// isSignedInteger ("-5")          true
// isSignedInteger ("+5")          true
// isSignedInteger ("", false)     false
// isSignedInteger ("", true)      true

function isSignedInteger (s)

{   if (isEmpty(s)) 
       if (isSignedInteger.arguments.length == 1) return defaultEmptyOK;
       else return (isSignedInteger.arguments[1] == true);

    else {
        var startPos = 0;
        var secondArg = defaultEmptyOK;

        if (isSignedInteger.arguments.length > 1)
            secondArg = isSignedInteger.arguments[1];

        // skip leading + or -
        if ( (s.charAt(0) == "-") || (s.charAt(0) == "+") )
           startPos = 1;    
        return (isInteger(s.substring(startPos, s.length), secondArg))
    }
}


function isSignedNumber (s)

{   if (isEmpty(s)) 
       if (isSignedNumber.arguments.length == 1) return defaultEmptyOK;
       else return (isSignedNumber.arguments[1] == true);

    else {
        var startPos = 0;
        var secondArg = defaultEmptyOK;

        if (isSignedNumber.arguments.length > 1)
            secondArg = isSignedNumber.arguments[1];

        // skip leading + or -
        if ( (s.charAt(0) == "-") || (s.charAt(0) == "+") )
           startPos = 1;    
        return (isSignedFloat(s.substring(startPos, s.length), secondArg))
    }
}



// isPositiveInteger (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if string s is an integer > 0.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isPositiveInteger (s)
{   var secondArg = defaultEmptyOK;

    if (isPositiveInteger.arguments.length > 1)
        secondArg = isPositiveInteger.arguments[1];

    // The next line is a bit byzantine.  What it means is:
    // a) s must be a signed integer, AND
    // b) one of the following must be true:
    //    i)  s is empty and we are supposed to return true for
    //        empty strings
    //    ii) this is a positive, not negative, number

    return (isSignedInteger(s, secondArg)
         && ( (isEmpty(s) && secondArg)  || (parseInt (s,10) > 0) ) );
}


function isPositiveNumber (s)
{   var secondArg = defaultEmptyOK;

    if (isPositiveNumber.arguments.length > 1)
        secondArg = isPositiveNumber.arguments[1];

    // The next line is a bit byzantine.  What it means is:
    // a) s must be a signed integer, AND
    // b) one of the following must be true:
    //    i)  s is empty and we are supposed to return true for
    //        empty strings
    //    ii) this is a positive, not negative, number

    return (isSignedNumber(s, secondArg)
         && ( (isEmpty(s) && secondArg)  || (parseInt (s,10) > 0) ) );
}


// isNonNegativeInteger (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if string s is an integer >= 0.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isNonNegativeInteger (s)
{   var secondArg = defaultEmptyOK;

    if (isNonNegativeInteger.arguments.length > 1)
        secondArg = isNonNegativeInteger.arguments[1];

    // The next line is a bit byzantine.  What it means is:
    // a) s must be a signed integer, AND
    // b) one of the following must be true:
    //    i)  s is empty and we are supposed to return true for
    //        empty strings
    //    ii) this is a number >= 0

    return (isSignedInteger(s, secondArg)
         && ( (isEmpty(s) && secondArg)  || (parseInt (s,10) >= 0) ) );
}


function isNonNegativeNumber (s)
{   var secondArg = defaultEmptyOK;

    if (isNonNegativeNumber.arguments.length > 1)
        secondArg = isNonNegativeNumber.arguments[1];

    // The next line is a bit byzantine.  What it means is:
    // a) s must be a signed integer, AND
    // b) one of the following must be true:
    //    i)  s is empty and we are supposed to return true for
    //        empty strings
    //    ii) this is a number >= 0

    return (isSignedNumber(s, secondArg)
         && ( (isEmpty(s) && secondArg)  || (parseInt (s,10) >= 0) ) );
}



// isNegativeInteger (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if string s is an integer < 0.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isNegativeInteger (s)
{   var secondArg = defaultEmptyOK;

    if (isNegativeInteger.arguments.length > 1)
        secondArg = isNegativeInteger.arguments[1];

    // The next line is a bit byzantine.  What it means is:
    // a) s must be a signed integer, AND
    // b) one of the following must be true:
    //    i)  s is empty and we are supposed to return true for
    //        empty strings
    //    ii) this is a negative, not positive, number

    return (isSignedInteger(s, secondArg)
         && ( (isEmpty(s) && secondArg)  || (parseInt (s,10) < 0) ) );
}






// isNonpositiveInteger (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if string s is an integer <= 0.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isNonpositiveInteger (s)
{   var secondArg = defaultEmptyOK;

    if (isNonpositiveInteger.arguments.length > 1)
        secondArg = isNonpositiveInteger.arguments[1];

    // The next line is a bit byzantine.  What it means is:
    // a) s must be a signed integer, AND
    // b) one of the following must be true:
    //    i)  s is empty and we are supposed to return true for
    //        empty strings
    //    ii) this is a number <= 0

    return (isSignedInteger(s, secondArg)
         && ( (isEmpty(s) && secondArg)  || (parseInt (s,10) <= 0) ) );
}

function isNonpositiveNumber (s)
{   var secondArg = defaultEmptyOK;

    if (isNonpositiveNumber.arguments.length > 1)
        secondArg = isNonpositiveNumber.arguments[1];

    // The next line is a bit byzantine.  What it means is:
    // a) s must be a signed integer, AND
    // b) one of the following must be true:
    //    i)  s is empty and we are supposed to return true for
    //        empty strings
    //    ii) this is a number <= 0

    return (isSignedNumber(s, secondArg)
         && ( (isEmpty(s) && secondArg)  || (parseInt (s,10) <= 0) ) );
}


// isFloat (STRING s [, BOOLEAN emptyOK])
// 
// True if string s is an unsigned floating point (real) number. 
//
// Also returns true for unsigned integers. If you wish
// to distinguish between integers and floating point numbers,
// first call isInteger, then call isFloat.
//
// Does not accept exponential notation.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isFloat (s)

{   var i;
    var seenDecimalPoint = false;

    if (isEmpty(s)) 
       if (isFloat.arguments.length == 1) return defaultEmptyOK;
       else return (isFloat.arguments[1] == true);

    if (s == decimalPointDelimiter) return false;

    // Search through string's characters one by one
    // until we find a non-numeric character.
    // When we do, return false; if we don't, return true.

    for (i = 0; i < s.length; i++)
    {   
        // Check that current character is number.
        var c = s.charAt(i);

        if ((c == decimalPointDelimiter) && !seenDecimalPoint) seenDecimalPoint = true;
        else if (!isDigit(c)) return false;
    }

    // All characters are numbers.
    return true;
}


// isSignedFloat (STRING s [, BOOLEAN emptyOK])
// 
// True if string s is a signed or unsigned floating point 
// (real) number. First character is allowed to be + or -.
//
// Also returns true for unsigned integers. If you wish
// to distinguish between integers and floating point numbers,
// first call isSignedInteger, then call isSignedFloat.
//
// Does not accept exponential notation.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isSignedFloat (s)

{   if (isEmpty(s)) 
       if (isSignedFloat.arguments.length == 1) return defaultEmptyOK;
       else return (isSignedFloat.arguments[1] == true);

    else {
        var startPos = 0;
        var secondArg = defaultEmptyOK;

        if (isSignedFloat.arguments.length > 1)
            secondArg = isSignedFloat.arguments[1];

        // skip leading + or -
        if ( (s.charAt(0) == "-") || (s.charAt(0) == "+") )
           startPos = 1;    
        return (isFloat(s.substring(startPos, s.length), secondArg))
    }
}


function isDecimal (s, numberDecimals, eok)

{   var i;
	var decimalPosition=-1;
	
    var seenDecimalPoint = false;

    if (isEmpty(s)) 
       if (isDecimal.arguments.length == 2) return defaultEmptyOK;
       else return (isDecimal.arguments[2] == true);

    if (s == decimalPointDelimiter) return false;

    // Search through string's characters one by one
    // until we find a non-numeric character.
    // When we do, return false; if we don't, return true.

	
    for (i = 0; i < s.length; i++)
    {   
        // Check that current character is number.
        var c = s.charAt(i);

        if ((c == decimalPointDelimiter) && !seenDecimalPoint) 
				{
				decimalPosition = (i + 1)
				seenDecimalPoint = true;
				}

        else 
        	if (!isDigit(c)) 
				return false;
 	}

	x = (s.length - decimalPosition)
	if (x > numberDecimals)
		return false;
	
    // All characters are numbers.
    return true;
}

function isDate (year, month, day)
{   // catch invalid years (not 2- or 4-digit) and invalid months and days.

    if (! (isYear(year, false) && isMonth(month, false) && isDay(day, false))) return false;

    // Explicitly change type to integer to make code work in both
    // JavaScript 1.1 and JavaScript 1.2.
    var intYear = parseInt(year,10);
    var intMonth = parseInt(month,10);
    var intDay = parseInt(day,10);


    // catch invalid days, except for February
    if (intDay > daysInMonth[intMonth]) return false; 

    if ((intMonth == 2) && (intDay > daysInFebruary(intYear))) return false;

    return true;
}


// isAlphabetic (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if string s is English letters 
// (A .. Z, a..z) only.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.
//
// NOTE: Need i18n version to support European characters.
// This could be tricky due to different character
// sets and orderings for various languages and platforms.

function isAlphabetic (s)

{   var i;

    if (isEmpty(s)) 
       if (isAlphabetic.arguments.length == 1) return defaultEmptyOK;
       else return (isAlphabetic.arguments[1] == true);

    // Search through string's characters one by one
    // until we find a non-alphabetic character.
    // When we do, return false; if we don't, return true.

    for (i = 0; i < s.length; i++)
    {   
        // Check that current character is letter.
        var c = s.charAt(i);

        if (!isLetter(c))
        return false;
    }

    // All characters are letters.
    return true;
}




// isAlphanumeric (STRING s [, BOOLEAN emptyOK])
// 
// Returns true if string s is English letters 
// (A .. Z, a..z) and numbers only.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.
//
// NOTE: Need i18n version to support European characters.
// This could be tricky due to different character
// sets and orderings for various languages and platforms.

function isAlphanumeric (s)

{   var i;

    if (isEmpty(s)) 
       if (isAlphanumeric.arguments.length == 1) return defaultEmptyOK;
       else return (isAlphanumeric.arguments[1] == true);

    // Search through string's characters one by one
    // until we find a non-alphanumeric character.
    // When we do, return false; if we don't, return true.

    for (i = 0; i < s.length; i++)
    {   
        // Check that current character is number or letter.
        var c = s.charAt(i);

        if (! (isLetter(c) || isDigit(c) ) )
        return false;
    }

    // All characters are numbers or letters.
    return true;
}




// reformat (TARGETSTRING, STRING, INTEGER, STRING, INTEGER ... )       
//
// Handy function for arbitrarily inserting formatting characters
// or delimiters of various kinds within TARGETSTRING.
//
// reformat takes one named argument, a string s, and any number
// of other arguments.  The other arguments must be integers or
// strings.  These other arguments specify how string s is to be
// reformatted and how and where other strings are to be inserted
// into it.
//
// reformat processes the other arguments in order one by one.
// * If the argument is an integer, reformat appends that number 
//   of sequential characters from s to the resultString.
// * If the argument is a string, reformat appends the string
//   to the resultString.
//
// NOTE: The first argument after TARGETSTRING must be a string.
// (It can be empty.)  The second argument must be an integer.
// Thereafter, integers and strings must alternate.  This is to
// provide backward compatibility to Navigator 2.0.2 JavaScript
// by avoiding use of the typeof operator.
//
// It is the caller's responsibility to make sure that we do not
// try to copy more characters from s than s.length.
//
// EXAMPLES:
//
// * To reformat a 10-digit U.S. phone number from "1234567890"
//   to "(123) 456-7890" make this function call:
//   reformat("1234567890", "(", 3, ") ", 3, "-", 4)
//
// * To reformat a 9-digit U.S. Social Security number from
//   "123456789" to "123-45-6789" make this function call:
//   reformat("123456789", "", 3, "-", 2, "-", 4)
//
// HINT:
//
// If you have a string which is already delimited in one way
// (example: a phone number delimited with spaces as "123 456 7890")
// and you want to delimit it in another way using function reformat,
// call function stripCharsNotInBag to remove the unwanted 
// characters, THEN call function reformat to delimit as desired.
//
// EXAMPLE:
//
// reformat (stripCharsNotInBag ("123 456 7890", digits),
//           "(", 3, ") ", 3, "-", 4)

function reformat (s)

{   var arg;
    var sPos = 0;
    var resultString = "";

    for (var i = 1; i < reformat.arguments.length; i++) {
       arg = reformat.arguments[i];
       if (i % 2 == 1) resultString += arg;
       else {
           resultString += s.substring(sPos, sPos + arg);
           sPos += arg;
       }
    }
    return resultString;
}




// isSSN (STRING s [, BOOLEAN emptyOK])
// 
// isSSN returns true if string s is a valid U.S. Social
// Security Number.  Must be 9 digits.
//
// NOTE: Strip out any delimiters (spaces, hyphens, etc.)
// from string s before calling this function.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isSSN (s)
{   if (isEmpty(s)) 
       if (isSSN.arguments.length == 1) return defaultEmptyOK;
       else return (isSSN.arguments[1] == true);
    return (isInteger(s) && s.length == digitsInSocialSecurityNumber)
}




// isUSPhoneNumber (STRING s [, BOOLEAN emptyOK])
// 
// isUSPhoneNumber returns true if string s is a valid U.S. Phone
// Number.  Must be 10 digits.
//
// NOTE: Strip out any delimiters (spaces, hyphens, parentheses, etc.)
// from string s before calling this function.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isUSPhoneNumber (s)
{   if (isEmpty(s)) 
       if (isUSPhoneNumber.arguments.length == 1) return defaultEmptyOK;
       else return (isUSPhoneNumber.arguments[1] == true);
    return (isInteger(s) && s.length == digitsInUSPhoneNumber)
}




// isInternationalPhoneNumber (STRING s [, BOOLEAN emptyOK])
// 
// isInternationalPhoneNumber returns true if string s is a valid 
// international phone number.  Must be digits only; any length OK.
// May be prefixed by + character.
//
// NOTE: A phone number of all zeros would not be accepted.
// I don't think that is a valid phone number anyway.
//
// NOTE: Strip out any delimiters (spaces, hyphens, parentheses, etc.)
// from string s before calling this function.  You may leave in 
// leading + character if you wish.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isInternationalPhoneNumber (s)
{   if (isEmpty(s)) 
       if (isInternationalPhoneNumber.arguments.length == 1) return defaultEmptyOK;
       else return (isInternationalPhoneNumber.arguments[1] == true);
    return (isPositiveInteger(s))
}




// isZIPCode (STRING s [, BOOLEAN emptyOK])
// 
// isZIPCode returns true if string s is a valid 
// U.S. ZIP code.  Must be 5 or 9 digits only.
//
// NOTE: Strip out any delimiters (spaces, hyphens, etc.)
// from string s before calling this function.  
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isZIPCode (s)
{  if (isEmpty(s)) 
       if (isZIPCode.arguments.length == 1) return defaultEmptyOK;
       else return (isZIPCode.arguments[1] == true);
   return (isInteger(s) && 
            ((s.length == digitsInZIPCode1) ||
             (s.length == digitsInZIPCode2)))
}





// isStateCode (STRING s [, BOOLEAN emptyOK])
// 
// Return true if s is a valid U.S. Postal Code 
// (abbreviation for state).
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isStateCode(s)
{   if (isEmpty(s)) 
       if (isStateCode.arguments.length == 1) return defaultEmptyOK;
       else return (isStateCode.arguments[1] == true);
    return ( (USStateCodes.indexOf(s) != -1) &&
             (s.indexOf(USStateCodeDelimiter) == -1) )
}




// isEmail (STRING s [, BOOLEAN emptyOK])
// 
// Email address must be of form a@b.c -- in other words:
// * there must be at least one character before the @
// * there must be at least one character before and after the .
// * the characters @ and . are both required
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isEmail (s)
{   if (isEmpty(s)) 
       if (isEmail.arguments.length == 1) return defaultEmptyOK;
       else return (isEmail.arguments[1] == true);
   
    // is s whitespace?
    if (isWhitespace(s)) return false;
    
    // there must be >= 1 character before @, so we
    // start looking at character position 1 
    // (i.e. second character)
    var i = 1;
    var sLength = s.length;

    // look for @
    while ((i < sLength) && (s.charAt(i) != "@"))
    { i++
    }

    if ((i >= sLength) || (s.charAt(i) != "@")) return false;
    else i += 2;

    // look for .
    while ((i < sLength) && (s.charAt(i) != "."))
    { i++
    }

    // there must be at least one character after the .
    if ((i >= sLength - 1) || (s.charAt(i) != ".")) return false;
    else return true;
}





// isYear (STRING s [, BOOLEAN emptyOK])
// 
// isYear returns true if string s is a valid 
// Year number.  Must be 2 or 4 digits only.
// 
// For Year 2000 compliance, you are advised
// to use 4-digit year numbers everywhere.
//
// And yes, this function is not Year 10000 compliant, but 
// because I am giving you 8003 years of advance notice,
// I don't feel very guilty about this ...
//
// For B.C. compliance, write your own function. ;->
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isYear (s, emptyOK)
{   if (isEmpty(s)) 
       if (isYear.arguments.length ==1) return defaultEmptyOK;
       else return (isYear.arguments[1] == true);

    if (!isNonNegativeInteger(s)) return false;

    return ((s.length == 2) || (s.length == 4));
}



// isIntegerInRange (STRING s, INTEGER a, INTEGER b [, BOOLEAN emptyOK])
// 
// isIntegerInRange returns true if string s is an integer 
// within the range of integer arguments a and b, inclusive.
// 
// For explanation of optional argument emptyOK,
// see comments of function isInteger.


function isIntegerInRange (s, a, b)
{   if (isEmpty(s)) 
       if (isIntegerInRange.arguments.length == 1) return defaultEmptyOK;
       else return (isIntegerInRange.arguments[1] == true);

    // Catch non-integer strings to avoid creating a NaN below,
    // which isn't available on JavaScript 1.0 for Windows.
    if (!isInteger(s, false)) return false;

    // Now, explicitly change the type to integer via parseInt
    // so that the comparison code below will work both on 
    // JavaScript 1.2 (which typechecks in equality comparisons)
    // and JavaScript 1.1 and before (which doesn't).
	
    var num = parseInt (s,10);

    a = parseInt(a,10);
    b = parseInt(b,10);
    return ((num >= a) && (num <= b));
}

// isNumberInRange (STRING s, FLOAT a, FLOAT b [, BOOLEAN emptyOK])
// 
// isNumberInRange returns true if string s is a number
// within the range of integer arguments a and b, inclusive.
// 
// For explanation of optional argument emptyOK,
// see comments of function isInteger.


function isNumberInRange (s, a, b)
{   if (isEmpty(s)) 
       if (isNumberInRange.arguments.length == 1) return defaultEmptyOK;
       else return (isNumberInRange.arguments[1] == true);

    // Catch non-number strings to avoid creating a NaN below,
    // which isn't available on JavaScript 1.0 for Windows.
    //if (!isNonNegativeNumber(s, false)) return false;

	//st	
if (!isSignedInteger(s, false)) return false;


    // Now, explicitly change the type to integer via parseInt
    // so that the comparison code below will work both on 
    // JavaScript 1.2 (which typechecks in equality comparisons)
    // and JavaScript 1.1 and before (which doesn't).
    var num = parseFloat (s);
    return ((num >= a) && (num <= b));
}


// isMonth (STRING s [, BOOLEAN emptyOK])
// 
// isMonth returns true if string s is a valid 
// month number between 1 and 12.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isMonth (s, emptyOK)
{   if (isEmpty(s)) 
       if (isMonth.arguments.length == 1) return defaultEmptyOK;
       else return (isMonth.arguments[1] == true);

	//if (s.charAt(0)=="0")
	//	s=s.charAt(1)

    return isIntegerInRange (s, 1, 12);
}



// isDay (STRING s [, BOOLEAN emptyOK])
// 
// isDay returns true if string s is a valid 
// day number between 1 and 31.
// 
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function isDay (s, emptyOK)
{   if (isEmpty(s)) 
       if (isDay.arguments.length == 1) return defaultEmptyOK;
       else return (isDay.arguments[1] == true);   

    return isIntegerInRange (s,1,31)

}

function isHour (s, emptyOK)
{   if (isEmpty(s)) 
       if (isHour.arguments.length == 1) return defaultEmptyOK;
       else return (isHour.arguments[1] == true);   

    return isIntegerInRange (s, 0, 23);
}

function isMinute (s,emptyOK)
{   if (isEmpty(s)) 
       if (isMinute.arguments.length == 1) return defaultEmptyOK;
       else return (isMinute.arguments[1] == true);   

    return isIntegerInRange (s, 0, 59);
}

// daysInFebruary (INTEGER year)
// 
// Given integer argument year,
// returns number of days in February of that year.

function daysInFebruary (year)
{   // February has 29 days in any year evenly divisible by four,
    // EXCEPT for centurial years which are not also divisible by 400.
    return (  ((year % 4 == 0) && ( (!(year % 100 == 0)) || (year % 400 == 0) ) ) ? 29 : 28 );
}



// isDate (STRING year, STRING month, STRING day)
//
// isDate returns true if string arguments year, month, and day 
// form a valid date.
// 

/********************/




/* FUNCTIONS TO NOTIFY USER OF INPUT REQUIREMENTS OR MISTAKES. */


// Display prompt string s in status bar.

function prompt (s)
{   window.status = s
}



// Display data entry prompt string s in status bar.

function promptEntry (s)
{   window.status = pEntryPrompt + s
}




// Notify user that required field theField is empty.
// String s describes expected contents of theField.value.
// Put focus in theField and return false.

function warnEmpty (theField, s)
{   theField.focus()
    alert(mPrefix + s + mSuffix + theField.value)
    return false
}



// Notify user that contents of field theField are invalid.
// String s describes expected contents of theField.value.
// Put select theField, pu focus in it, and return false.

function warnInvalid (theField, s)
{   theField.focus()
    if (theField.type != "select-one")
	    theField.select()
    alert(s)
    return false
}




/* FUNCTIONS TO INTERACTIVELY CHECK VARIOUS FIELDS. */

// checkString (TEXTFIELD theField, STRING s, [, BOOLEAN emptyOK==false])
//
// Check that string theField.value is not all whitespace.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkString (theField, s, emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkString.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    if (isWhitespace(theField.value)) 
       return warnEmpty (theField, s);
    else return true;
}

//SJT 6/5/01
function checkMaxLength(theField, MaxLength)
{
	var fv=theField.value
	var vbcrlf = String.fromCharCode(10,13)
	
	if (fv.length>MaxLength)
		{
			fv=fv.substr(0,MaxLength) + vbcrlf + '»' + fv.substr(MaxLength,1000)
			theField.value=fv
			return warnInvalid(theField,"The text in this field is too long.  The part which does not fit begins with the '»' character.")
		}
	else
		return true
}

//SJT 4/4/02
function checkNumberDigits(theField,numDigits,emptyOK)
{
    if (checkNumberDigits.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
	
	if (!isInteger(theField.value))
		return warnInvalid (theField,"This field must be a " + numDigits + " digit number. Please enter the correct number of digits now.");

	if (theField.value.length != numDigits)
		return warnInvalid (theField,"This field must be a " + numDigits + " digit number. Please enter the correct number of digits now.");

	return true;

}


function checkNumberInRange (theField, a, b, emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkNumberInRange.arguments.length == 3) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    if (!isNumberInRange(theField.value, a, b)) 
       return warnInvalid (theField, "This field must be between " + a + " and " + b + " . Please reenter it now.");
    else return true;
}

function checkPositiveInteger (theField, s, emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkPositiveInteger.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
	if (theField.value==-9) return true;
    if (!isPositiveInteger(theField.value)) 
       return warnInvalid (theField, iPositiveInteger);
    else return true;
}

function checkNonNegativeInteger (theField, s,emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkNonNegativeInteger.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
	if (theField.value==-9) return true;
	
    if (!isNonNegativeInteger(theField.value)) 
       return warnInvalid (theField, iNonNegativeInteger);
    else return true;
}

function checkPositiveNumber (theField, s, emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkPositiveNumber.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
	if (theField.value==-9) return true;
    if (!isPositiveNumber(theField.value)) 
       return warnInvalid (theField, iPositiveNumber);
    else return true;
}

function checkNonNegativeNumber (theField, s,emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkNonNegativeNumber.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
	if (theField.value==-9) return true;
    if (!isNonNegativeNumber(theField.value)) 
       return warnInvalid (theField, iNonNegativeNumber);
    else return true;
}

function checkDecimal (theField, decimalPlaces, emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkDecimal.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
	if (theField.value==-9) return true;
    if (!isFloat(theField.value)) 
       return warnInvalid (theField, iPositiveInteger);
    else 
	return true;
}

function checkPercent (theField, numberDecimals, emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    if (checkPercent.arguments.length == 2) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    if (!isFloat(theField.value)) 
       return warnInvalid (theField, iPercent);
    else 
	{
	
	return true;
	}
}


// checkStateCode (TEXTFIELD theField [, BOOLEAN emptyOK==false])
//
// Check that string theField.value is a valid U.S. state code.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkStateCode (theField, emptyOK)
{   if (checkStateCode.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    else
    {  theField.value = theField.value.toUpperCase();
       if (!isStateCode(theField.value, false)) 
          return warnInvalid (theField, iStateCode);
       else return true;
    }
}



// takes ZIPString, a string of 5 or 9 digits;
// if 9 digits, inserts separator hyphen

function reformatZIPCode (ZIPString)
{   if (ZIPString.length == 5) return ZIPString;
    else return (reformat (ZIPString, "", 5, "-", 4));
}




// checkZIPCode (TEXTFIELD theField [, BOOLEAN emptyOK==false])
//
// Check that string theField.value is a valid ZIP code.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkZIPCode (theField, emptyOK)
{   if (checkZIPCode.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    else
    { var normalizedZIP = stripCharsInBag(theField.value, ZIPCodeDelimiters)
      if (!isZIPCode(normalizedZIP, false)) 
         return warnInvalid (theField, iZIPCode);
      else 
      {  // if you don't want to insert a hyphen, comment next line out
         theField.value = reformatZIPCode(normalizedZIP)
         return true;
      }
    }
}



// takes USPhone, a string of 10 digits
// and reformats as (123) 456-789

function reformatUSPhone (USPhone)
{   return (reformat (USPhone, "(", 3, ") ", 3, "-", 4))
}


//SJT ADDED THIS PHONE UTILITY
// checkPhone (TEXTFIELD theCountry, TEXTFIELD theField,  [, BOOLEAN emptyOK==false])

function checkPhone (theCountry, theField, emptyOK)
{
	if (theCountry.value == "US")
		return checkUSPhone (theField, emptyOK);

	else
		return checkInternationalPhone (theField, emptyOK);

}


// checkUSPhone (TEXTFIELD theField [, BOOLEAN emptyOK==false])
//
// Check that string theField.value is a valid US Phone.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkUSPhone (theField, emptyOK)
{   if (checkUSPhone.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    else
    {  var normalizedPhone = stripCharsInBag(theField.value, phoneNumberDelimiters)
       if (!isUSPhoneNumber(normalizedPhone, false)) 
          return warnInvalid (theField, iUSPhone);
       else 
       {  // if you don't want to reformat as (123) 456-789, comment next line out
          theField.value = reformatUSPhone(normalizedPhone)
          return true;
       }
    }
}



// checkInternationalPhone (TEXTFIELD theField [, BOOLEAN emptyOK==false])
//
// Check that string theField.value is a valid International Phone.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkInternationalPhone (theField, emptyOK)
{   if (checkInternationalPhone.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    else
    {  if (!isInternationalPhoneNumber(theField.value, false)) 
          return warnInvalid (theField, iWorldPhone);
       else return true;
    }
}



// checkEmail (TEXTFIELD theField [, BOOLEAN emptyOK==false])
//
// Check that string theField.value is a valid Email.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkEmail (theField, emptyOK)
{   if (checkEmail.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    else if (!isEmail(theField.value, false)) 
       return warnInvalid (theField, iEmail);
    else return true;
}



// takes SSN, a string of 9 digits
// and reformats as 123-45-6789

function reformatSSN (SSN)
{   return (reformat (SSN, "", 3, "-", 2, "-", 4))
}


// Check that string theField.value is a valid SSN.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkSSN (theField, emptyOK)
{   if (checkSSN.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    else
    {  var normalizedSSN = stripCharsInBag(theField.value, SSNDelimiters)
       if (!isSSN(normalizedSSN, false)) 
          return warnInvalid (theField, iSSN);
       else 
       {  // if you don't want to reformats as 123-456-7890, comment next line out
          theField.value = reformatSSN(normalizedSSN)
          return true;
       }
    }
}


function checkMinute (theField, emptyOK)
{   if (checkMinute.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    if (!isMinute(theField.value, false)) 
       return warnInvalid (theField, iMinute);
    else return true;
}

function checkYear (theField, emptyOK)
{   if (checkYear.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    if (!isYear(theField.value, false)) 
       return warnInvalid (theField, iYear);
    else return true;
}


// Check that string theField.value is a valid Month.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkMonth (theField, emptyOK)
{   if (checkMonth.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    if (!isMonth(theField.value, false)) 
       return warnInvalid (theField, iMonth);
    else return true;
}


// Check that string theField.value is a valid Day.
//
// For explanation of optional argument emptyOK,
// see comments of function isInteger.

function checkDay (theField, emptyOK)
{   if (checkDay.arguments.length == 1) emptyOK = defaultEmptyOK;
    if ((emptyOK == true) && (isEmpty(theField.value))) return true;
    if (!isDay(theField.value, false)) 
       return warnInvalid (theField, iDay);
    else return true;
}


// SJT ADDED
function checkTime (hrsField, minsField, ampmField, labelString, OKtoOmitTime)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.

    if (checkTime.arguments.length == 4) OKtoOmitTime = false;

	if ((OKtoOmitTime == true) && (isWhitespace(hrsField.value)) && (isWhitespace(minsField.value)))
		return true;

	if (!isHour(hrsField.value)) return warnInvalid(hrsField, iHour);
	if (!isMinute (minsField.value)) return warnInvalid(minsField, iMinute);
	if ((ampmField.value!="am") && (ampmField.value!="pm")) return warnInvalid (ampmField, iAmPm);
    if ((ampmField.value=="am") && (hrsField.value>12)) return warnInvalid (ampmField, "That is not a valid time.");
   return true;
}

//SJT 4/4/02

function checkDate (monthField, dayField, yearField, labelString, emptyOK)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
 	if (checkDate.arguments.length == 4) emptyOK = defaultEmptyOK;

    if (emptyOK == true)
	{
		if ((isWhitespace(monthField.value)) && (isWhitespace(dayField.value)))
				return true;
	}

    if (monthField.value=="-9") 
		return true;

	if (yearField.value=="-9") 
		return true;

	if (dayField.value=="-9")
		return true;

	if (yearField.value=="9999")
		{
		alert ("You must enter a value for the year field.")
		yearField.focus()
		return false;
		}

    if (!isMonth(monthField.value)) return warnInvalid (monthField, iMonth);

	    if (!isDay(dayField.value)) //&& (dayField.type<>'hidden')
       return warnInvalid (dayField, iDay);

    var yearValue=yearField.value
	if (yearValue.length==2)
		yearValue="20" + yearValue

    if (!isYear(yearValue)) return warnInvalid (yearField, iYear);

	var a=1900

	Today = new Date();
	var b = Today.getFullYear() + 10; //allow dates to be up to 10 years in the future
	
	var dt=parseInt(yearValue)
		
	if (dt<a)
		return warnInvalid(yearField,"Year cannot be prior to 1900.")

	if (dt>b)
		return warnInvalid(yearField,"Year cannot be greater than " + b + ".")


    if (isDate(yearField.value, monthField.value, dayField.value))
	{
	//yearField.value=yearValue
	return true;
	}

    alert (iDatePrefix + labelString.replace(/&nbsp;/gi," ") + iDateSuffix);
    return false;

}

function checkPartialDate (monthField, dayField, yearField, labelString, emptyOK, partialDateValue, allowFutureYears)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.

	//SJT 3/29/02
	partialDateValue="00"    
		
 	if (checkPartialDate.arguments.length == 4) emptyOK = defaultEmptyOK;
	if (checkPartialDate.arguments.length <7) allowFutureYears=false;

   	if (monthField.value=="-9") 
		return true;

	if (yearField.value=="-9") 
		return true;

	if (dayField.value=="-9")
		return true;

	if (yearField.value=="9999")
		{
		alert ("You must enter a value for the year field.")
		yearField.focus()
		return false;
		}

	if (emptyOK == true)
		{
		if (dayField.type == "hidden")
			if (isWhitespace(monthField.value))	
					return true;
			else
				if ((isWhitespace(monthField.value)) && (isWhitespace(dayField.value)))
					return true;
				
		}
	else
		{
		if (isWhitespace(monthField.value))
			{
			alert ("You must enter a value for the month field.")
			monthField.focus()
			return false;
			}

		if ((isWhitespace(dayField.value)) && (dayField.type != "hidden"))
				{
				alert ("You must enter a value for the day field.")
				dayField.focus()
				return false;
				}

		if (isWhitespace(yearField.value))
			{
			alert ("You must enter a value for the year field.")
			yearField.focus()
			return false;
			}
		}

	var monthValue=monthField.value
	var dayValue=dayField.value
	var yearValue=yearField.value

   if (monthValue==partialDateValue)
	monthValue=0

   if (dayValue==partialDateValue)
	dayValue=0

	Today = new Date();
	
	if (yearValue==(partialDateValue + partialDateValue))
	yearValue = "" + Today.getFullYear();

	if (yearValue=="")
		yearValue = "" + Today.getFullYear();
				
   if (monthValue!=0)
	    if (!isMonth(Number(monthValue))) return warnInvalid (monthField, iMonthPartial);

	if (dayValue!=0)
		if (!isDay(Number(dayValue))) return warnInvalid (dayField, "This field must be a day number between 1 and 31, or either '0' or blank if unknown.  Please reenter it now.");

   if (!isYear(yearValue, true)) return warnInvalid (yearField, iYear);

	if (yearValue.length==2)
		yearValue="20" + yearValue

	var a=1900

	if (allowFutureYears==true)
		var b=Today.getFullYear() + 50;
	else
		var b=Today.getFullYear();
				
	var dt=parseInt(yearValue)
		
	if (dt<a)
		return warnInvalid(yearField,"Year cannot be prior to 1900.")

	if (dt>b)
		return warnInvalid(yearField,"Year cannot be greater than " + b + ".")


    if ((monthValue==0) && (dayValue==0))
		return true

	if ((monthValue==0) && (dayValue!=0))
		return warnInvalid (monthField, "Cannot have an actual day with an unknown month.")

	if ((isMonth(monthValue)) && (dayValue==0))
		return true

	if ((monthValue!=0) && (dayValue!=0))
		if (isDate(yearValue, Number(monthValue), Number(dayValue))) 
			{
			//sjt 1/27/05 reinterpret 'Allow Future YEARS' as 'Allow Future DATES'
			var newDate = new Date(yearValue, Number(monthValue)-1, Number(dayValue))
			if (allowFutureYears==false)
				if (newDate>Today)
					return warnInvalid (yearField, "Date cannot be in the future.")
				else
					return true;
			else
				return true;
			}

    alert (iDatePrefix + "'" + labelString.replace(/&nbsp;/gi," ") + "'" + iDateSuffix);
    return false;

}


// Get checked value from radio button.

function getRadioButtonValue (radio)
{   for (var i = 0; i < radio.length; i++)
    {   if (radio[i].checked) { break }
    }
    return radio[i].value
}




function calcvalue(s)
{   
	var i;
	var x="";
    	var seenDecimalPoint = false;
	var decimalPointDelimiter = "."
	var currencySeparator=","
	var currencySymbol=defaultCurrencySymbol;	
		
    if (isEmpty(s)) 
		return 0		
    if (s == decimalPointDelimiter) 
		return 0		



    for (i = 0; i < s.length; i++)
    {   
        var c = s.charAt(i);

		if (c == currencySymbol)
			x = "";
		else if (c == currencySeparator)
			c = ""
       else if ((c == decimalPointDelimiter) && !seenDecimalPoint)
        	if (x == "")
        		{
        		x = "0" + decimalPointDelimiter
	        	seenDecimalPoint = true;
	        	}
	       else
	       	x = x + decimalPointDelimiter
	        	
        else if (!isDigit(c))
				return 0		
        	else 
        		x = x + c
    }

    // All characters are numbers.
    return Number(x);


}



function checkNum(data) 
{      // checks if all characters

	 
	var valid = "0123456789.,";     // are valid numbers or a "."
	var ok = 1; 
	var checktemp;
	var foundDecimal=0

    if (isEmpty(data)) 
		return 1;

	for (var i=0; i<data.length; i++) 
		{
		checktemp = "" + data.substring(i, i+1);
		if (valid.indexOf(checktemp) == "-1") 
			return 0;
			

		if (checktemp==".") 
			{			
 				if (foundDecimal==0)
 					foundDecimal=1;
 				else
 					return 0
 			}

		}
	
	return 1;
}

function checkCurrency(thefield , emptyOK)
{
	Num = thefield.value

	if ((emptyOK == true) && (isEmpty(thefield.value))) 
		return true;
    
	if (Num.charAt(0)=="$")
		Num=Num.substring(1);

	if (Num.length==0)
		return 1;

	if (checkNum(Num) == 0)
		{
		warnInvalid(thefield,"This does not appear to be a valid number.  Please try again.");
		return 0;
		}
		
	return 1
}
function dollarAmount(field, decimalPlaces, currencySymbol, width) 
{ // idea by David Turley

	Num = field.value
	dec = Num.indexOf(".");
	end=""

	if (Num.charAt(0)==currencySymbol)
		Num=Num.substring(1);

	if (Num.length==0)
		return;
		
		
	var secondArg = 2;
	var thirdArg=defaultCurrencySymbol
	var fourthArg=0

    if (dollarAmount.arguments.length > 1)
        secondArg = dollarAmount.arguments[1];

    if (dollarAmount.arguments.length > 2)
        thirdArg = dollarAmount.arguments[2];

    if (dollarAmount.arguments.length > 3)
        fourthArg = dollarAmount.arguments[3];

	decimalPlaces=secondArg
	currencySymbol=thirdArg
	width=fourthArg

	if (checkNum(Num) == 0) 
		{
		warnInvalid(field,"This does not appear to be a valid number.  Please try again.");
		return;
		}
	else 
		{ 

		field.value=currency(stripCharsNotInBag(Num,"01234567890."),decimalPlaces, currencySymbol, width)
		return;
	   }
}



//-- Shows number in $$xxx,xxx.xx format and pads left side with blanks.
function currency(anynum,decimalPlaces,currencySymbol,width) 
{
   anynum=eval(anynum)
   workNum=Math.abs((Math.round(anynum*100)/100));
   workStr=""+workNum
 
	if (workStr.indexOf(".")==-1)
   		{workStr+=".00"}
	
	dStr=workStr.substr(0,workStr.indexOf("."));
   	dNum=dStr-0
   	pStr=workStr.substr(workStr.indexOf("."))
	while (pStr.length<(decimalPlaces+1))
   		{pStr+="0"}
	   	

   //--- Adds comma in thousands place.
   if (dNum>=1000) {
      dLen=dStr.length
      dStr=parseInt(""+(dNum/1000))+","+dStr.substring(dLen-3,dLen)
   }

   //-- Adds comma in millions place.
   if (dNum>=1000000) {
      dLen=dStr.length
      dStr=parseInt(""+(dNum/1000000))+","+dStr.substring(dLen-7,dLen)
   }
   retval=dStr+pStr 
   if (anynum < 0) {
      retval=retval.substring(1,retval.length)
      retval="("+retval+")"        
   }
   retval = currencySymbol+retval
   //--Pad with leading blanks to better align numbers.
   while (retval.length<width)
   	{retval=" "+retval}

   return retval
}

function getRadioValue(radioName) {
  var collection;
  collection = document.all[radioName];
  for (i=0;i<collection.length;i++) {
    if (collection[i].checked)
    return(collection[i].value);
  }
}

function checkPartialTime (hoursField, minsField, labelString, emptyOK, partialDateValue, showAMPM, ampmField)
{   // Next line is needed on NN3 to avoid "undefined is not a number" error
    // in equality comparison below.
    

 	if (checkPartialTime.arguments.length == 3) emptyOK = defaultEmptyOK;


    	if (hoursField.value=="-9") 
		return true;


    if (emptyOK == true)
		{
		if ((isWhitespace(hoursField.value)) && (isWhitespace(minsField.value)))
				{
				if (showAMPM==true)
					ampmField.value="";
				return true;
				}
		}


		if (isWhitespace(hoursField.value))
			{
			alert ("You must enter a value for the hour field.")
			hoursField.focus()
			return false;
			}

		if (isWhitespace(minsField.value))
			{
			alert ("You must enter a value for the minute field.")
			minsField.focus()
			return false;
			}


	var hrsValue=hoursField.value
	var minsValue=minsField.value


   if (Number(hrsValue)!=0)
	    if (!isHour(Number(hrsValue))) return warnInvalid (hoursField, "This field must be an hour number between 0 and 23, or '-9' for a missing value.  Please reenter it now.");
	
	if (Number(minsValue)!=0)
	    if (!isMinute(Number(minsValue))) return warnInvalid (minsField, "This field must be a minute number between 0 and 59.  Please reenter it now.");
	
	if (showAMPM==true)
		if ((ampmField.value=="AM") && (Number(hrsValue)>12)) return warnInvalid (ampmField, "That is not a valid time.");
		
	if (showAMPM==true)
		if ((ampmField.value=="PM") && (Number(hrsValue)==0)) return warnInvalid (ampmField, "That is not a valid time.");

    return true;

}