kernel/eka/euser/us_parse.cpp
author Simon Howkins <simonh@symbian.org>
Fri, 05 Mar 2010 15:41:00 +0000
branchRCL_3
changeset 73 9c2a3e4960cf
parent 0 a41df078684a
permissions -rw-r--r--
Merge further Compiler Compatibility fixes onto RCL_3 branch.

// Copyright (c) 1996-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// e32\euser\us_parse.cpp
// 
//

#include "us_std.h"

class TStringToDateTime
	{
public:
	TStringToDateTime(const TDesC& aDes,TInt aCenturyOffset);
	TInt Parse(TTime& aTime);
	enum {ETimePresent=1,EDatePresent};
private:
// tokens
	enum {EDec=-12,ENov,EOct,ESep,EAug,EJul,EJun,EMay,EApr,EMar,EFeb,EJan};
	enum {ETokenAm=-19,ETokenPm};
	enum TDateSeparators{ESlash=-39,EDash,EComma,ESpace,EDateLocale1,EDateLocale2};
	enum TTimeSeparators{EColon=-49,EDot,ETimeLocale1,ETimeLocale2};
	enum TDecimalSeparators{EDecimalLocale=-59};
	enum {EErrorToken=-99,ENullToken=-100};
//
	enum {ENumberOfDateSep=6,ENumberOfTimeSep=4,EMaxTokens=27};
	enum {EFirstDateSep=ESlash,ELastDateSep=EDateLocale2,EFirstTimeSep=EColon,ELastTimeSep=ETimeLocale2};
private:
	TInt NextToken(TInt& aTokenLen);
	void StripSpaceTokens();
	TInt CrackTokenFormula();
	TInt GetDate(TInt aFormulaPos,TInt& aTokenCount);
	TInt GetTime(TInt aFormulaPos,TInt& aTokenCount);
//	
	TInt GetSeparatorToken(TChar aChar) const;	
	TBool IsTimeSeparator(TChar  aChar) const;
	TBool IsDateSeparator(TChar  aChar) const;
	TBool IsDecimalSeparator(TChar  aChar) const;
	TBool IsSeparator(TChar aChar) const;
	TBool IsSeparator(TInt aToken) const;
	inline TBool IsTimeSeparator(TInt aToken) const;
	inline TBool IsDateSeparator(TInt aToken) const;
	TBool IsDecimalSeparator(TInt aToken) const;
	inline TBool IsAmPm(TInt aToken) const;
	inline TBool IsAlphaMonth(TInt aToken) const;
private:
	TLex iLex;
	TInt iCenturyOffset;
	TDateFormat iDateFormat;
	TChar iTimeSepChars[ENumberOfTimeSep];
	TChar iDateSepChars[ENumberOfDateSep];
	TChar iDecSepChar;
	TDateTime iDateTime;
	TInt iCount;
	TInt iFormula[EMaxTokens];// 27 max possible with valid des (including spaces):" 10 : 00 : 00 . 000000 pm 6 / 12 / 99 "
	TUint8 iTokenLen[EMaxTokens];
	};

inline TBool TStringToDateTime::IsTimeSeparator(TInt aToken) const
	{
	return(aToken >= EFirstTimeSep && aToken <=ELastTimeSep);
	}
inline TBool TStringToDateTime::IsDateSeparator(TInt aToken) const
	{
	return(aToken>=EFirstDateSep && aToken<=ELastDateSep);
	}
inline TBool TStringToDateTime::IsAmPm(TInt aToken) const
	{
	return(aToken==ETokenAm || aToken==ETokenPm);
	}
inline TBool TStringToDateTime::IsAlphaMonth(TInt aToken) const
	{
	return (aToken>=EDec && aToken<=EJan);
	}

inline TBool TStringToDateTime::IsDecimalSeparator(TChar aChar) const
	{
	return(aChar==iDecSepChar);
	}
inline TBool TStringToDateTime::IsDecimalSeparator(TInt aToken) const
	{
	return(aToken==EDecimalLocale);
	}

TStringToDateTime::TStringToDateTime(const TDesC& aDes,TInt aCenturyOffset)
	: iLex(aDes),iCenturyOffset(aCenturyOffset),iDateTime(0,EJanuary,0,0,0,0,0)
	{

	__ASSERT_ALWAYS(aCenturyOffset>=0 && aCenturyOffset<100,Panic(ETTimeValueOutOfRange));
	TLocale locale;
	iDateFormat=locale.DateFormat();
	
	iTimeSepChars[0]=':';
	iTimeSepChars[1]='.';
	iTimeSepChars[2]=locale.TimeSeparator(1);
	iTimeSepChars[3]=locale.TimeSeparator(2);

	iDateSepChars[0]='/';
	iDateSepChars[1]='-';
	iDateSepChars[2]=',';
	iDateSepChars[3]=' ';
	iDateSepChars[4]=locale.DateSeparator(1);
	iDateSepChars[5]=locale.DateSeparator(2);
	iDecSepChar = locale.DecimalSeparator();
	}

TBool TStringToDateTime::IsTimeSeparator(TChar aChar) const
	{

	for (TInt ii=0;ii<ENumberOfTimeSep;++ii)
		if (aChar==iTimeSepChars[ii])
			return ETrue;
	return(EFalse);
	}

TBool TStringToDateTime::IsDateSeparator(TChar aChar) const
	{

	for (TInt ii=0;ii<ENumberOfDateSep;++ii)
		if (aChar==iDateSepChars[ii])
			return ETrue;
	return(EFalse);
	}

TBool TStringToDateTime::IsSeparator(TChar  aChar) const
	{

	return(IsTimeSeparator(aChar) || IsDateSeparator(aChar) || IsDecimalSeparator(aChar));
	}

TBool TStringToDateTime::IsSeparator(TInt aToken) const
	{

	return(IsTimeSeparator(aToken) || IsDateSeparator(aToken));
	}

TInt TStringToDateTime::GetSeparatorToken(TChar aChar) const
	{

	TInt ii=0;
	for (ii=0;ii<ENumberOfDateSep;++ii)
		if (aChar == iDateSepChars[ii])
			return(EFirstDateSep+ii);
	for (ii=0;ii<ENumberOfTimeSep;++ii)
		if (aChar == iTimeSepChars[ii])
			return(EFirstTimeSep+ii);
	if (aChar == iDecSepChar)
		return(EDecimalLocale);
	return(ENullToken);
	}

void TStringToDateTime::StripSpaceTokens()
// Removes excess space tokens from the formula
// The end of the formula is marked with a Null token
	{

	TInt t = 0;
	for (TInt s = 0 ; s < iCount ; ++s)
		{
		if (iFormula[s]==ESpace &&
			(IsSeparator(iFormula[s-1]) || s == iCount-1 || IsSeparator(iFormula[s+1]) || IsAmPm(iFormula[s+1])))
			continue;// Skip unwanted space token
		iFormula[t]=iFormula[s];
		iTokenLen[t]=iTokenLen[s];
		++t;
		}
	iCount=t;
	iFormula[t]=ENullToken;
	}

TInt TStringToDateTime::CrackTokenFormula()
	{

	if (iCount==0)
		return KErrArgument;// Nothing to read
	TInt token0=iFormula[0];
	TInt token1=iFormula[1];
	TInt numberOfTokens;
	TInt dummy=0;
	TInt error;
	if (IsDateSeparator(token1) || IsAlphaMonth(token0))
		{// Assume formula is a Date or DateTime
		if ((error=GetDate(0,numberOfTokens))!=EDatePresent)
			return error;
		numberOfTokens+=1;// Space char between the Date & Time
		return(GetTime(numberOfTokens,dummy));
		}
	else if (IsTimeSeparator(token1) || IsAmPm(token1))
		{// Assume formula is a Time or TimeDate
		if ((error=GetTime(0,numberOfTokens))!=ETimePresent)
			return error;
		numberOfTokens+=1;// Space char between the Time & Date
		return(GetDate(numberOfTokens,dummy));
		}
	else
		return(KErrArgument);
	}

TInt TStringToDateTime::GetDate(TInt aOffset,TInt& aTokenCount)
	// if aOffset == 0  then Date or DateTime format
	// if aOffset != 0  then TimeDate format
	{

	TInt relativeCount=iCount;
	if (aOffset!=0)// aFormat==ETimeDate
		{
		relativeCount-=aOffset;
		if (relativeCount<=-1)
			return(ETimePresent);
		}
	TInt numberOfDateFields=0;
	if (relativeCount==3)
		numberOfDateFields=2;
	else if (relativeCount==5)
		numberOfDateFields=3;
	else if (aOffset==0)
		{// DateTime
		if (IsTimeSeparator(iFormula[5]) || IsAmPm(iFormula[5]))
			numberOfDateFields=2;
		else
			numberOfDateFields=3;
		}
	else// (aOffset!=0)
		{// Date
		if (relativeCount==3)
			numberOfDateFields=2;
		else if (relativeCount==5)
			numberOfDateFields=3;
		else
			return(KErrArgument);
		}

	if (!IsDateSeparator(iFormula[1+aOffset])) 
		return(KErrArgument);
	if (numberOfDateFields==2)
		{
		if (aOffset!=0 && relativeCount!=3)// ie TimeDate
			return(KErrArgument);
		}
	if (numberOfDateFields==3)
		{
		if (aOffset!=0 && relativeCount!=5)// ie TimeDate
			return(KErrArgument);
		if (!IsDateSeparator(iFormula[3+aOffset]))
			return(KErrArgument);
		}

	// A month will always be in the first two fields // DMY MDY YMD
	TBool alphaMonth=(IsAlphaMonth(iFormula[0+aOffset]) || IsAlphaMonth(iFormula[2+aOffset]) );
	
	TInt dayIndex;
	TInt monthIndex;
	TInt yearIndex=4;// Reset if Japanese

	if (iDateFormat==EDateJapanese)
		{// 1996 feb 3
		if (numberOfDateFields==2)
			{
			monthIndex=0;
			dayIndex=2;
			}
		else
			{
			yearIndex=0;
			monthIndex=2;
			dayIndex=4;
			}
		}
	else if (IsAlphaMonth(iFormula[0+aOffset])
		|| (!alphaMonth && iDateFormat==EDateAmerican))// Amer Euro 
		{// feb 3 1996 valid Amer or Euro format // 2 3 1996 Amer
		monthIndex=0;
		dayIndex=2;
		}		
	else
		{// 3 feb 1996 valid Amer or Euro format // 3 2 1996 Euro
		__ASSERT_DEBUG(
			IsAlphaMonth(iFormula[2+aOffset]) || 
			(!alphaMonth && iDateFormat==EDateEuropean),User::Invariant());
		monthIndex=2;
		dayIndex=0;
		}
	
	TTime timeNow;
	timeNow.HomeTime();
	TDateTime now=timeNow.DateTime();
	TInt currentCentury=((now.Year()/100)*100);// Integer arithmetic
	TInt currentTwoDigitYear=now.Year()-currentCentury;

	TInt year=0;
	if (numberOfDateFields==3)// then year value exists
		{
		year=iFormula[yearIndex+aOffset];
		if (year<0)// ie a token has been returned as a year
			return(KErrArgument);
		else if (iTokenLen[yearIndex+aOffset]<=2)
			{
			if (currentTwoDigitYear>=iCenturyOffset)
				{
				if (year>=00 && year<iCenturyOffset) 
					year+=currentCentury+100;// next century
				else
					year+=currentCentury;
				}
			else
				{
				if (year>=00 && year<iCenturyOffset) 
					year+=currentCentury;
				else
					year+=currentCentury-100;// last century
				}
			}
		}

	TInt month=iFormula[monthIndex+aOffset];
	if (IsAlphaMonth(month))
		month=-month;// alphaMonth is -ve enum token
	month-=1;// months start at zero

	TInt error;// Set Year, Month and Day
	if ((error=iDateTime.SetYear(year))==KErrNone)
		if ((error=iDateTime.SetMonth((TMonth)month))==KErrNone)
			error=iDateTime.SetDay(iFormula[dayIndex+aOffset]-1);
	if (error!=KErrNone)
		return(error);


	if (numberOfDateFields==2)
		aTokenCount=3;
	else if (numberOfDateFields==3)
		aTokenCount=5;
	
	if (aOffset!=0)
		return(EDatePresent|ETimePresent);
	return(EDatePresent);
	}

TInt TStringToDateTime::GetTime(TInt aOffset,TInt& aTokenCount)
	// aFormulaPos == 0  Time format or TimeDate format with Time first
	// aFormulaPos != 0  Date preceeds Time i.e. DateTime format
	// 7 formats 10:00:00.012345 // 10:00:00.012345pm // 10:00:00pm // 10:00:00 // 10:00pm // 10:00 // 10pm
	// offset and relativeCount allow this function to check times 
	// when both the Time(10:00pm) and DateTime(3-feb-69 10:00pm) formats are used.
	{

	TInt relativeCount=iCount;
	if (aOffset!=0)// DateTime // else format==Time
		{
		relativeCount-=aOffset;
		if (relativeCount<=-1)
			return(EDatePresent);
		}
	TInt fields=0;

	if (IsTimeSeparator(iFormula[1+aOffset]) && IsTimeSeparator(iFormula[3+aOffset])&& 
		 (IsTimeSeparator(iFormula[5+aOffset]) || IsDecimalSeparator(iFormula[5+aOffset])))
		{
		fields=4;// 10:00:00.000000 (am)
		aTokenCount=7; 
		if (IsAmPm(iFormula[7+aOffset]))
			aTokenCount+=1;
		}
	else if (IsTimeSeparator(iFormula[1+aOffset]) && IsTimeSeparator(iFormula[3+aOffset]))
		{
		fields=3;// 10:00:00 (am)
		aTokenCount=5; 
		if (IsAmPm(iFormula[5+aOffset]))
			aTokenCount+=1;
		}
	else if (IsTimeSeparator(iFormula[1+aOffset]))
		{
		fields=2;// 10:00 (am)
		aTokenCount=3;
		if (IsAmPm(iFormula[3+aOffset]))
			aTokenCount+=1;
		}
	else if (IsAmPm(iFormula[1+aOffset]))
		{
		fields=1;// 10am
		aTokenCount=2;
		}
	if (fields==0 || (fields==4 && relativeCount==6) || (fields==3 && relativeCount==4) || (fields==2 && relativeCount==2))
		return(KErrArgument);// Colon\DecimalPoint in wrong place 10:00:00. || 10:00: || 10:
	
	TInt error;
	if ((error=iDateTime.SetHour(iFormula[0+aOffset]))!=KErrNone)
		return error;
	if (fields==2)
		error=iDateTime.SetMinute(iFormula[2+aOffset]);
	else if (fields==3)
		{
		if ((error=iDateTime.SetMinute(iFormula[2+aOffset]))==KErrNone)
			error=iDateTime.SetSecond(iFormula[4+aOffset]);
		}
	else if (fields==4)
		{
		if ((error=iDateTime.SetMinute(iFormula[2+aOffset]))==KErrNone)
			 if ((error=iDateTime.SetSecond(iFormula[4+aOffset]))==KErrNone)
				 error = iDateTime.SetMicroSecond(iFormula[6+aOffset]);
		}
	if (error!=KErrNone)
		return(error);

	TInt ampmIndex=2*fields-1;
	if (iFormula[ampmIndex+aOffset]==ETokenAm && iDateTime.Hour()==12)
		error=iDateTime.SetHour(00);// 12am->00 hrs. Ignore 13am
	else if (iFormula[ampmIndex+aOffset]==ETokenPm && iDateTime.Hour()<12)
		error=iDateTime.SetHour(iDateTime.Hour()+12);
	if (error!=KErrNone)
		return(error);

	if (aOffset!=0)
		return(ETimePresent|EDatePresent);
	return(ETimePresent);
	}

TInt TStringToDateTime::NextToken(TInt& aTokenLen)
	{
	if (iLex.Eos())
		return ENullToken;

	TChar ch=iLex.Peek();

	if (ch.IsDigit())
		{
		iLex.Mark();
		do iLex.Inc(); while (iLex.Peek().IsDigit());

		TPtrC des=iLex.MarkedToken();

		TInt digit;
		TLex lex(des);
		if (lex.Val(digit)!=KErrNone)
			return(EErrorToken);
		aTokenLen = des.Length();
		return(digit);
		}
	else if (IsSeparator(ch))
		{
		iLex.Inc();
		iLex.SkipSpace();
		aTokenLen = 1;
		return(GetSeparatorToken(ch));
		}
	else
		{
		iLex.Mark();
		do iLex.Inc(); while (iLex.Peek().IsAlpha() || iLex.Peek().IsDigit());

		TPtrC des=iLex.MarkedToken();
		aTokenLen = des.Length();

		for (TInt month=EJanuary; month<=EDecember; ++month)
			{
			// Abbreviated month name
			TMonthNameAbb nameAbb((TMonth)month);
			if (nameAbb.CompareF(des)==0)
				return(-(month+1)); // All values negative

			// Full month name
			TMonthName name((TMonth)month);
			if (name.CompareF(des)==0)
				return(-(month+1)); // All values negative
			}

		// Substring of am or pm
		TAmPmName am(EAm);
		TAmPmName pm(EPm);
			
		if (am.FindF(des)==0)
			return(ETokenAm);
		else if (pm.FindF(des)==0)
			return(ETokenPm);
		
		return(EErrorToken);
		}
	}


TInt TStringToDateTime::Parse(TTime& aTime)
	{

	iLex.SkipSpace();
	TInt i = 0;
	for (;;)
		{
		if (i==EMaxTokens-1)	// space left to append NullToken
			return KErrArgument;
		TInt len;
		TInt token=NextToken(len);// uses iLex
		if (token==EErrorToken)
			return KErrArgument;
		if (token==ENullToken)
			break;
		iFormula[i]=token;	// append token to formula
		iTokenLen[i]=(TUint8)Min(len, 255);
		++i;
		}
	iCount=i;

	StripSpaceTokens();// Uses then resets iCount
	TInt ret=CrackTokenFormula();
	if (ret<0)
		return(ret);
	if (iDateTime.Year()>9999)
		return KErrArgument;
	aTime=iDateTime;
	return(ret);
	}

EXPORT_C TInt TTime::Parse(const TDesC& aDes,TInt aCenturyOffset)
/**
Parses a descriptor containing either or both a date and time, and sets this 
TTime to the value of the parsed descriptor.
	
The descriptor may contain the date only, the time only, the date followed 
by the time, or the time followed by the date. When both the date and time 
are specified in the descriptor, they should be separated using one or more 
space characters. 
	
Leading zeros and spaces preceding any time or date components are discarded.
	
Dates may be specified either with all three components (day, month and year), 
or with just two components; for example month and day. The date suffix ("st" 
"nd" "rd" or "th") may not be included in the descriptor.
	
The date and its components may take different forms:
	
1. The month may be represented by text or by numbers.
	
2  European (DD/MM/YYYY), American (MM/DD/YYYY) and Japanese (YYYY/MM/DD) date 
   formats are supported. An exception to this ordering of date components occurs 
   when European or American formatting is used and the month is represented 
   by text. In this case, the month may be positioned in either the first or 
   second field. When using Japanese date format, the month, whether text or 
  numbers, must always be the second field.
	
3. The year may be two or four digits. When the year is a two digit number, (e.g. 
   97 rather than 1997), to resolve any confusion as to which century the year 
   falls in, the second argument determines the century. For example, if the 
   current year is 1997, a value for aCenturyOffset of 20 means that any two 
   digit year will resolve to a year in the range 1920 to 2019. In this case, 
   two digit years between 00 and 19 inclusive refer to the years between 2000 
   and 2019 and two digit years between 20 and 99 inclusive refer to the years 
   between 1920 and 1999. By default, two digit years are in the current century 
   (aCenturyOffset = 0).
	
4. Any of the following characters may be used as the date separator: /(slash) 
   - (dash) , (comma), spaces, or either of the date separator characters specified 
   in TLocale::SetDateSeparator() (at index 1 or 2). Other characters are illegal.
	
If a colon or a dot has been specified in TLocale as the date separator character, 
neither may be used as date separators in this function.
	
If specified, the time must include the hour, but both minutes and seconds, 
or seconds alone may be omitted. 
	
The time and its components may take different forms:
	
1. An am/pm time suffix may be appended to the time. If 24 hour clock format 
   is in use, this text will be ignored. 
	
2. The am/pm suffix may be abbreviated to "a" or "p".
	
3. Any of the following characters may be used as the time separator: :(colon) 
   .(dot) or either of the time separator characters specified in
   TLocale::SetDateSeparator() (at index 1 or 2). Other characters are illegal.
	
When a character can be interpreted as either a date or time separator character, 
this function will interpret it as a date separator.
	
Look out for cases in which wrongly interpreting the contents of a descriptor, 
based on the interpretation of separator characters, causes an error. For 
example, trying to interpret "5.6.1996" as a time is invalid and will return 
an error of -2 because 1,996 seconds is out of range.
	
Notes:
	
1. The entire content of the descriptor must be valid and syntactically correct, 
   or an error will be returned and the parse will fail. So, excepting whitespace, 
   which is discarded, any trailing characters within the descriptor which do 
   not form part of the date or time are illegal. 
	
2. If no time is specified in the descriptor, the hours, minutes and seconds 
   of this TTime are all set to zero, corresponding to midnight at the start 
   of the day specified in the date. If no date is specified, each of this TTime's 
   date components are set to zero.

@param aDes           Descriptor containing any combination of date and
                      time as text.
@param aCenturyOffset Offset between zero (the default) and 99. Allows a flexible 
                      interpretation of the century for two digit year values.
                      If less than zero, or greater than 99, a panic occurs.
                      
@return If equal to or greater than zero, the function completed successfully. 
        EParseDatePresent and/or EParseTimePresent indicate whether either or both 
        of the date or time are present.
        If less than zero, an error code.
        KErrGeneral indicates that the time or date value is out of range,
        e.g. if the hour is greater than 23 or if the minute is greater
        than 59.
        KErrNotSupported indicates that a two field date has been entered.
        KErrArgument indicates that the descriptor was syntactically incorrect.
        If the function fails, this TTime object remains unchanged.
*/
	{

	TStringToDateTime parse(aDes,aCenturyOffset);
	return parse.Parse(*this);
	}