src/hbcore/i18n/hbstringutil.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Wed, 23 Jun 2010 18:33:25 +0300
changeset 6 c3690ec91ef8
parent 3 11d3954df52a
child 7 923ff622b8b9
permissions -rw-r--r--
Revision: 201023 Kit: 2010125

/****************************************************************************
**
** Copyright (C) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (developer.feedback@nokia.com)
**
** This file is part of the HbCore module of the UI Extensions for Mobile.
**
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser General Public
** License version 2.1 as published by the Free Software Foundation and
** appearing in the file LICENSE.LGPL included in the packaging of this file.
** Please review the following information to ensure the GNU Lesser General
** Public License version 2.1 requirements will be met:
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights.  These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you have questions regarding the use of this file, please contact
** Nokia at developer.feedback@nokia.com.
**
****************************************************************************/

#include <hbextendedlocale.h>
#include <QtAlgorithms>

#include "hbstringutil_p.h"
#include "hbstringutil.h"

#if defined(__SYMBIAN32__)
#include <e32std.h>
#include <e32def.h>
#include <e32des16.h>
#include <collate.h>
#define Q_OS_SYMBIAN
#endif

#ifndef QT_NO_REGEXP
#include <QRegExp>
#endif

/*!
    @stable
    @hbcore
    \class HbStringUtil
    \brief The HbStringUtil class can be used to execute operations on strings,
    such as comparisons and finding data sequences.

    \sa HbStringUtil
*/

/*!
    \enum HbStringUtil::Option

    This enum describes the way collation is done for matchC, compareC
    Pass one of these values to setReadChannel() to set the
    current read channel of QProcess.

    \value Default          Use the default System flags.
     
    \value IgnoreNone       Don't ignore anything.
    
    \value SwapCase         Reverse case ordering.
    
    \value AccentsBackwards Compare secondary keys which represent accents in reverse order.
    
    \value SwapKana         Reverse order for katakana/hiragana.
    
    \value FoldCase         Fold to lower case, file comparisons.
    
    \value MatchingTable    Table used for matching.
    
    \value IgnoreCombining  Ignore check for adjacent combining characters. 
    
    \sa compareC, matchC
*/

/*!
    Searches source string's collated data for a
    match with collated data supplied in pattern string
      
    \attention Cross-Platform API
    
    \param strFrom Source string.
    \param strToMatch Pattern string.
    \param maxLevel Determines the tightness of the collation.
    Level 0 - Character identity; 
    Level 1 - Character identity and accents; 
    Level 2 - Character identity, accents and case; 
    Level 3 - Character identity, accents, case and Unicode value; 
    \param flags The flags that will be used. Default value is Default.
    \param wildChar Wild card character.
    \param wildSequenceChar Wild card sequence character.
    \param escapeChar The escape character, for example,  '?', '*'.
    
    \return If a match is found the offset within source string's
    data where the match first occurs. -1 if match is not found.
    
    Example:
    \snippet{unittest_hbstringutil/unittest_hbstringutil.cpp,3}
 */
int HbStringUtil::matchC( const QString &strFrom, const QString &strToMatch,
                                    int maxLevel, Options flags,
                                    int wildChar, int wildSequenceChar, int escapeChar )
{
#if defined( Q_OS_SYMBIAN )
    TPtrC s1Ptr( strFrom.utf16() );
    TPtrC s2Ptr( strToMatch.utf16() );

	if ( (maxLevel < 0) || (maxLevel > 3) ) {
		maxLevel = 0;
	}
    if ( (flags < 0) || (flags > 127) ) {
        flags = Default;
    }

    TCollationMethod m = *Mem::GetDefaultMatchingTable();
    m.iFlags |= flags;

    return s1Ptr.MatchC(s2Ptr, wildChar, wildSequenceChar, escapeChar, maxLevel, &m);
#else
    Q_UNUSED(maxLevel);
    Q_UNUSED(flags);
    Q_UNUSED(wildChar);
    Q_UNUSED(wildSequenceChar);
    Q_UNUSED(escapeChar);
#ifdef QT_NO_REGEXP
    // if no regular expressions defined do standard MatchF
    return strFrom.indexOf( strToMatch, 0, Qt::CaseSensitive );
#else    
    // works with standard wildcards is not correct
    QRegExp locStrToMatch( strToMatch, Qt::CaseSensitive, QRegExp::Wildcard );    
    return strFrom.indexOf( locStrToMatch, 0 );
#endif    
    
#endif
}

/*!
    Compares source string's data with the other string's
    data using the specified collation method.
    
    \attention Cross-Platform API
    
    \param string1 Source string.
    \param string2 String whose data is to be compared with the source string.
    \param maxLevel Maximum level to use for comparing.
    Level 0 - Character identity; 
    Level 1 - Character identity and accents; 
    Level 2 - Character identity, accents and case; 
    Level 3 - Character identity, accents, case and Unicode value; 
    \param flags The flags that will be used. Default value is Default.
    
    \return Positive if source string is greater, negative if it is less and 
    zero	if the content of both strings match.
    
    Example:
    \snippet{unittest_hbstringutil/unittest_hbstringutil.cpp,1}
 */
int HbStringUtil::compareC( const QString &string1, const QString &string2,
                                        int maxLevel, Options flags )
{
#if defined( Q_OS_SYMBIAN )
    TPtrC s1Ptr(string1.utf16());
    TPtrC s2Ptr(string2.utf16());
   
	if ( (maxLevel < 0) || (maxLevel > 3) ) {
		maxLevel = 3;
	}
    if ( (flags < 0) || (flags > 127) ) {
        flags = Default;
    }
    
    TCollationMethod m = *Mem::CollationMethodByIndex( 0 );
    m.iFlags |= flags;

    return s1Ptr.CompareC( s2Ptr, maxLevel, &m);
#else
    Q_UNUSED(maxLevel);
    Q_UNUSED(flags);
    return string1.localeAwareCompare( string2 );
#endif    
}

/*!
    Searches for the first occurrence of the specified collated 
    data sequence in the aStrFrom to the specified maximum
    collation level.
    
    \attention Cross-Platform API
    
    \param strFrom Source string.
    \param strToFind String whose data is to be compared with the source string.
    \param maxLevel The maximum collation level.
    Level 0 - Character identity; 
    Level 1 - Character identity and accents; 
    Level 2 - Character identity, accents and case; 
    Level 3 - Character identity, accents, case and Unicode value; 
    
    \return Offset of the data sequence from the beginning of the
    aStrFrom. -1 if the data sequence cannot be found.
    
    Example:
    \snippet{unittest_hbstringutil/unittest_hbstringutil.cpp,5}
 */
int HbStringUtil::findC( const QString &strFrom,
                         const QString &strToFind,
                         int           maxLevel )
{
#if defined( Q_OS_SYMBIAN )
    TPtrC s1Ptr( strFrom.utf16() );
    TPtrC s2Ptr( strToFind.utf16() );
    
	if ( (maxLevel < 0) || (maxLevel > 3) ) {
		maxLevel = 0;
	}
    return s1Ptr.FindC( s2Ptr.Ptr(),
                        s2Ptr.Length(),
                        maxLevel );
#else
    Q_UNUSED(maxLevel);
    return strFrom.indexOf( strToFind, 0, Qt::CaseSensitive );
#endif 
}

/*!
    Searches source string's folded data for a
    match with folded data supplied in pattern string
    
    \attention Cross-Platform API
    
    \param strFrom Source string.
    \param strToMatch Pattern string.
    
    \return If a match is found the offset within source string's
    data where the match first occurs. -1 if match is not found.
    
    Example:
    \snippet{unittest_hbstringutil/unittest_hbstringutil.cpp,4}
 */
int HbStringUtil::matchF( const QString &strFrom,
                          const QString &strToMatch )
{
#if defined( Q_OS_SYMBIAN )
    TPtrC s1Ptr( strFrom.utf16() );
    TPtrC s2Ptr( strToMatch.utf16() );
    return s1Ptr.MatchF( s2Ptr );
#else
    // folding is just case insensitive      
#ifdef QT_NO_REGEXP
    // if no regular expressions defined do standard FindF
    return strFrom.indexOf( strToMatch, 0, Qt::CaseInsensitive );
#else    
    QRegExp locStrToMatch( strToMatch, Qt::CaseInsensitive, QRegExp::Wildcard );
    return strFrom.indexOf( locStrToMatch, 0 );
#endif
    
#endif      
}

/*!
    Searches for the first occurrence of the specified folded 
    data sequence in the strFrom.
    
    \attention Cross-Platform API
    
    \param strFrom Source string.
    \param strToFind String whose data is to be compared with the source string.
    
    \return Offset of the data sequence from the beginning of the
    strFrom. -1 if the data sequence cannot be found. Zero,
    if the length of search data sequence is zero.
    
    Example:
    \snippet{unittest_hbstringutil/unittest_hbstringutil.cpp,6}
 */
int HbStringUtil::findF( const QString &strFrom,
                         const QString &strToFind )
{
#if defined( Q_OS_SYMBIAN )
    TPtrC s1Ptr( strFrom.utf16() );
    TPtrC s2Ptr( strToFind.utf16() );
    return s1Ptr.FindF( s2Ptr );
#else
    // folding is just case insensitive    
    return strFrom.indexOf( strToFind, 0, Qt::CaseInsensitive );   
#endif 
}

/*!
    Compares source string's folded data with the other string's
    folded data.
    
    \attention Cross-Platform API
    
    \param string1 Source string.
    \param string2 String whose data is to be compared with the source string.
    
    \return Positive if source string is greater, negative if it is less and 
    zero if the content of both strings match.
    
    Example:
    \snippet{unittest_hbstringutil/unittest_hbstringutil.cpp,2}
 */
int HbStringUtil::compareF( const QString &string1,
                            const QString &string2 )
{
#if defined( Q_OS_SYMBIAN )
    TPtrC s1Ptr( string1.utf16() );
    TPtrC s2Ptr( string2.utf16() );
    return s1Ptr.CompareF( s2Ptr );
#else
    // folding is just case insensitive     
    return string1.compare( string2, Qt::CaseInsensitive );   
#endif 
}

/*!
    Returns the starting digit range of the native digit
    \param ch native digit
    \return starting digit range
 */
static QChar nativeDigitBase(QChar ch)
{
	DigitType d[] = { WesternDigit, ArabicIndicDigit, EasternArabicIndicDigit, DevanagariDigit, ThaiDigit };
    int i = 0;
    int num = sizeof(d)/sizeof(d[0]);
    while(i<num) {
        if (ch>QChar(d[i]) && ch<QChar(d[i]+10)) { return d[i]; }
        i++;
    }
    return ch;
}

/*!
    Converts digits to native digits based on current UI language.
    
    \attention Cross-Platform API
    
    \param str digits to be converted.
 */
QString HbStringUtil::convertDigits( const QString str ) 
{
	HbExtendedLocale locale = HbExtendedLocale::system();
	DigitType digitType = WesternDigit;
	if (locale.language() == HbExtendedLocale::Arabic) {
		digitType = ArabicIndicDigit;
	} else if (locale.language() == HbExtendedLocale::Persian || locale.language() == HbExtendedLocale::Urdu) {
		digitType = EasternArabicIndicDigit;
	}
	QString converted = HbStringUtil::convertDigitsTo(str, digitType);
	return converted;
}

/*!
    Converts the digit from Latin to native or native to latin or native to native
    
    \attention Cross-Platform API
    
    \param str digits to be converted.
    \param digitType type of the digit to be converted to
 */
QString HbStringUtil::convertDigitsTo( const QString str, const DigitType digitType ) 
{
    QString convDigit;
	int length = str.length();
    for(int i=0; i<length; i++) 
       {
       ushort digit = str[i].unicode();
       ushort digitBase = nativeDigitBase(str[i]).unicode();
       ushort convertedDigit = 0;
       switch (digitBase) 
           {
           case WesternDigit:
           case ArabicIndicDigit:
           case EasternArabicIndicDigit:
           case DevanagariDigit:
           case ThaiDigit:
               convertedDigit += digitType + digit - digitBase; 
               convDigit[i] = QChar(convertedDigit);
               break;
           default:
        	   convDigit[i] = QChar(digit);
               break;
           };
       }
    return convDigit;
}

/*!
    Sorts QStrings into alphabetically order (overwrites the strList's original content)
    
    \attention Cross-Platform API
    
    \param strList List of QStrings which need to be sorted.
    
    Example:
    \snippet{unittest_hbstringutil/unittest_hbstringutil.cpp,7}
 */
void HbStringUtil::sort( QStringList &strList )
{
	if ( strList.size() > 1 ) {
		qSort(strList.begin(), strList.end(), hbStringUtil_SortHelper);
	}
}

bool hbStringUtil_SortHelper( const QString &s1, const QString &s2 )
{
	if ( HbStringUtil::compareC(s1, s2) < 0 ) {
		// s1 is before s2
		return true;
	} else {
		// s1 is after s2 (or they are equal) 
		return false;
	}			
}