First public contribution.
2 * Copyright (c) 2002-2004 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of the License "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
24 #include "SortUtilJapan.h"
25 #include "SortUtilImpl.h"
26 #include "SortUtilImplExport.h"
27 #include <jplangutil.h>
29 // Unnamed namespace for local definitions
34 * Character classes for Japanese sorting, in sorting order.
36 enum TJapaneseCharClass
46 * Returns the Japanese character class of aChar.
48 TJapaneseCharClass CharClass(TUint aChar)
50 const TChar ch(aChar);
52 if (ch.IsSpace() || ch.IsPunctuation() || ch.IsControl()
53 || aChar >= 0xF0000 && aChar <= 0xFFFD)
55 return ECharClassSpecial;
58 else if (aChar >= '0' && aChar <= '9' ||
59 aChar >= 0xFF10 && aChar <= 0xFF19)
61 return ECharClassDigit;
64 else if (aChar >= 'A' && aChar <= 'Z' ||
65 aChar >= 'a' && aChar <= 'z' ||
66 aChar >= 0x00C0 && aChar <= 0x1EF3 && aChar != 0x00D7 && aChar != 0x00F7 ||
67 aChar >= 0xFF21 && aChar <= 0xFF3A ||
68 aChar >= 0xFF41 && aChar <= 0xFF5A)
70 return ECharClassLatin;
73 else if (JPLangUtil::IsKatakana(static_cast<TText>(aChar)) ||
74 JPLangUtil::IsHiragana(static_cast<TText>(aChar)))
76 return ECharClassKana;
79 else if (JPLangUtil::IsKanji(static_cast<TText>(aChar)))
81 return ECharClassKanji;
83 // All other charcters are "special"
86 return ECharClassSpecial;
91 * Compares two characters with Japanese comparison rules.
93 TInt JapaneseCompareC(TUint aLhs, TUint aRhs)
95 // First compare character classes
96 const TInt lhsCls = CharClass(aLhs);
97 const TInt rhsCls = CharClass(aRhs);
98 return lhsCls - rhsCls;
102 * Compares two strings with Japanese comparison rules.
104 TInt JapaneseCompareS(const TDesC& aLhs, const TDesC& aRhs)
108 const TInt minLen = Min(aLhs.Length(), aRhs.Length());
111 for (i = 0; i < minLen && result == 0; ++i)
113 result = JapaneseCompareC(aLhs[i], aRhs[i]);
116 TInt orgClass = CharClass(aLhs[i]);
119 if (orgClass == ECharClassKana)
122 for (; j < aLhs.Length(); ++j)
124 if (orgClass != CharClass(aLhs[j]))
129 lPart.Set(aLhs.Mid(prevCompPos, j - prevCompPos));
130 for (j = i; j < aRhs.Length(); ++j)
132 if (orgClass != CharClass(aRhs[j]))
137 rPart.Set(aRhs.Mid(prevCompPos, j - prevCompPos));
138 result = lPart.CompareC(rPart);
141 const TInt minKanaLen = Min(lPart.Length(), rPart.Length());
142 TPtrC lKanaPart(lPart.Left(minKanaLen));
143 TPtrC rKanaPart(rPart.Left(minKanaLen));
144 TInt resultKana = lKanaPart.MatchC(rKanaPart);
147 TInt maxKanaLen = i+minKanaLen;
148 if (i+minKanaLen+1 >= minLen)
150 if (maxKanaLen < aLhs.Length()
151 || maxKanaLen < aRhs.Length())
158 result = lKanaPart.CompareC(rKanaPart);;
172 lPart.Set(aLhs.Mid(i, 1));
173 rPart.Set(aRhs.Mid(i, 1));
174 result = lPart.CompareC(rPart);
184 if (i < aLhs.Length())
186 // aLhs is longer than aRhs
187 result += User::Collate(aLhs[i]);
189 if (i < aRhs.Length())
191 // aRhs is longer that aLhs
192 result -= User::Collate(aRhs[i]);
199 // Only export in DLL
200 EXPORT_C MSortUtil* SortUtilFactoryFunctionL()
202 MSortUtil* util = new (ELeave) TSortUtilJapan;
206 inline TSortUtilJapan::TSortUtilJapan()
210 TSortUtilJapan::~TSortUtilJapan()
214 TInt TSortUtilJapan::CompareItems
215 (const MSortKeyArray& aLhs,
216 const MSortKeyArray& aRhs) const
222 // Compare only pronounciation keys in the first pass
225 const TDesC& lhsText =
226 SortUtilImpl::FindNextNonEmptyKey(aLhs, ESortKeyPronounciation, lhsIndex);
227 const TDesC& rhsText =
228 SortUtilImpl::FindNextNonEmptyKey(aRhs, ESortKeyPronounciation, rhsIndex);
229 if (lhsText.Length() > 0)
231 // lhs has pronounciation key
232 if (rhsText.Length() > 0)
234 // Both lhs and rhs have pronounciation key: compare the key texts
235 result = JapaneseCompareS(lhsText, rhsText);
239 // Only lhs has pronounciation key
245 // lhs does not have pronounciation key
246 if (rhsText.Length() > 0)
248 // Only rhs has pronounciation key
258 while (lhsIndex <= aLhs.SortKeyCount() && rhsIndex < aRhs.SortKeyCount());
260 // No difference found with pronounciation keys: compare basic keys
265 const TDesC& lhsText =
266 SortUtilImpl::FindNextNonEmptyKey(aLhs, ESortKeyBasic, lhsIndex);
267 const TDesC& rhsText =
268 SortUtilImpl::FindNextNonEmptyKey(aRhs, ESortKeyBasic, rhsIndex);
269 result = JapaneseCompareS(lhsText, rhsText);
275 while (lhsIndex <= aLhs.SortKeyCount() && rhsIndex < aRhs.SortKeyCount());