1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*******************************************************************************
4 * Copyright (C) 2008-2016, International Business Machines Corporation and
5 * others. All Rights Reserved.
6 *******************************************************************************
10 *******************************************************************************
13 #include "unicode/dtitvinf.h"
16 #if !UCONFIG_NO_FORMATTING
18 //TODO: define it in compiler time
19 //#define DTITVINF_DEBUG 1
28 #include "unicode/msgfmt.h"
29 #include "unicode/uloc.h"
30 #include "unicode/ures.h"
31 #include "dtitv_impl.h"
45 #define PRINTMESG(msg) { std::cout << "(" << __FILE__ << ":" << __LINE__ << ") " << msg << "\n"; }
48 UOBJECT_DEFINE_RTTI_IMPLEMENTATION(DateIntervalInfo
)
50 static const char gCalendarTag
[]="calendar";
51 static const char gGregorianTag
[]="gregorian";
52 static const char gIntervalDateTimePatternTag
[]="intervalFormats";
53 static const char gFallbackPatternTag
[]="fallback";
56 static const UChar gFirstPattern
[] = {LEFT_CURLY_BRACKET
, DIGIT_ZERO
, RIGHT_CURLY_BRACKET
};
58 static const UChar gSecondPattern
[] = {LEFT_CURLY_BRACKET
, DIGIT_ONE
, RIGHT_CURLY_BRACKET
};
61 static const UChar gDefaultFallbackPattern
[] = {LEFT_CURLY_BRACKET
, DIGIT_ZERO
, RIGHT_CURLY_BRACKET
, SPACE
, EN_DASH
, SPACE
, LEFT_CURLY_BRACKET
, DIGIT_ONE
, RIGHT_CURLY_BRACKET
, 0};
63 DateIntervalInfo::DateIntervalInfo(UErrorCode
& status
)
64 : fFallbackIntervalPattern(gDefaultFallbackPattern
),
65 fFirstDateInPtnIsLaterDate(false),
66 fIntervalPatterns(NULL
)
68 fIntervalPatterns
= initHash(status
);
73 DateIntervalInfo::DateIntervalInfo(const Locale
& locale
, UErrorCode
& status
)
74 : fFallbackIntervalPattern(gDefaultFallbackPattern
),
75 fFirstDateInPtnIsLaterDate(false),
76 fIntervalPatterns(NULL
)
78 initializeData(locale
, status
);
84 DateIntervalInfo::setIntervalPattern(const UnicodeString
& skeleton
,
85 UCalendarDateFields lrgDiffCalUnit
,
86 const UnicodeString
& intervalPattern
,
89 if ( lrgDiffCalUnit
== UCAL_HOUR_OF_DAY
) {
90 setIntervalPatternInternally(skeleton
, UCAL_AM_PM
, intervalPattern
, status
);
91 setIntervalPatternInternally(skeleton
, UCAL_HOUR
, intervalPattern
, status
);
92 } else if ( lrgDiffCalUnit
== UCAL_DAY_OF_MONTH
||
93 lrgDiffCalUnit
== UCAL_DAY_OF_WEEK
) {
94 setIntervalPatternInternally(skeleton
, UCAL_DATE
, intervalPattern
, status
);
96 setIntervalPatternInternally(skeleton
, lrgDiffCalUnit
, intervalPattern
, status
);
102 DateIntervalInfo::setFallbackIntervalPattern(
103 const UnicodeString
& fallbackPattern
,
104 UErrorCode
& status
) {
105 if ( U_FAILURE(status
) ) {
108 int32_t firstPatternIndex
= fallbackPattern
.indexOf(gFirstPattern
,
109 UPRV_LENGTHOF(gFirstPattern
), 0);
110 int32_t secondPatternIndex
= fallbackPattern
.indexOf(gSecondPattern
,
111 UPRV_LENGTHOF(gSecondPattern
), 0);
112 if ( firstPatternIndex
== -1 || secondPatternIndex
== -1 ) {
113 status
= U_ILLEGAL_ARGUMENT_ERROR
;
116 if ( firstPatternIndex
> secondPatternIndex
) {
117 fFirstDateInPtnIsLaterDate
= true;
119 fFallbackIntervalPattern
= fallbackPattern
;
124 DateIntervalInfo::DateIntervalInfo(const DateIntervalInfo
& dtitvinf
)
126 fIntervalPatterns(NULL
)
134 DateIntervalInfo::operator=(const DateIntervalInfo
& dtitvinf
) {
135 if ( this == &dtitvinf
) {
139 UErrorCode status
= U_ZERO_ERROR
;
140 deleteHash(fIntervalPatterns
);
141 fIntervalPatterns
= initHash(status
);
142 copyHash(dtitvinf
.fIntervalPatterns
, fIntervalPatterns
, status
);
143 if ( U_FAILURE(status
) ) {
147 fFallbackIntervalPattern
= dtitvinf
.fFallbackIntervalPattern
;
148 fFirstDateInPtnIsLaterDate
= dtitvinf
.fFirstDateInPtnIsLaterDate
;
154 DateIntervalInfo::clone() const {
155 return new DateIntervalInfo(*this);
159 DateIntervalInfo::~DateIntervalInfo() {
160 deleteHash(fIntervalPatterns
);
161 fIntervalPatterns
= NULL
;
166 DateIntervalInfo::operator==(const DateIntervalInfo
& other
) const {
168 fFallbackIntervalPattern
== other
.fFallbackIntervalPattern
&&
169 fFirstDateInPtnIsLaterDate
== other
.fFirstDateInPtnIsLaterDate
);
171 if ( equal
== TRUE
) {
172 equal
= fIntervalPatterns
->equals(*(other
.fIntervalPatterns
));
180 DateIntervalInfo::getIntervalPattern(const UnicodeString
& skeleton
,
181 UCalendarDateFields field
,
182 UnicodeString
& result
,
183 UErrorCode
& status
) const {
184 if ( U_FAILURE(status
) ) {
188 const UnicodeString
* patternsOfOneSkeleton
= (UnicodeString
*) fIntervalPatterns
->get(skeleton
);
189 if ( patternsOfOneSkeleton
!= NULL
) {
190 IntervalPatternIndex index
= calendarFieldToIntervalIndex(field
, status
);
191 if ( U_FAILURE(status
) ) {
194 const UnicodeString
& intervalPattern
= patternsOfOneSkeleton
[index
];
195 if ( !intervalPattern
.isEmpty() ) {
196 result
= intervalPattern
;
204 DateIntervalInfo::getDefaultOrder() const {
205 return fFirstDateInPtnIsLaterDate
;
210 DateIntervalInfo::getFallbackIntervalPattern(UnicodeString
& result
) const {
211 result
= fFallbackIntervalPattern
;
215 #define ULOC_LOCALE_IDENTIFIER_CAPACITY (ULOC_FULLNAME_CAPACITY + 1 + ULOC_KEYWORD_AND_VALUES_CAPACITY)
218 static const int32_t PATH_PREFIX_LENGTH
= 17;
219 static const UChar PATH_PREFIX
[] = {SOLIDUS
, CAP_L
, CAP_O
, CAP_C
, CAP_A
, CAP_L
, CAP_E
, SOLIDUS
,
220 LOW_C
, LOW_A
, LOW_L
, LOW_E
, LOW_N
, LOW_D
, LOW_A
, LOW_R
, SOLIDUS
};
221 static const int32_t PATH_SUFFIX_LENGTH
= 16;
222 static const UChar PATH_SUFFIX
[] = {SOLIDUS
, LOW_I
, LOW_N
, LOW_T
, LOW_E
, LOW_R
, LOW_V
, LOW_A
,
223 LOW_L
, CAP_F
, LOW_O
, LOW_R
, LOW_M
, LOW_A
, LOW_T
, LOW_S
};
226 * Sink for enumerating all of the date interval skeletons.
228 struct DateIntervalInfo::DateIntervalSink
: public ResourceSink
{
231 DateIntervalInfo
&dateIntervalInfo
;
233 // Next calendar type
234 UnicodeString nextCalendarType
;
236 DateIntervalSink(DateIntervalInfo
&diInfo
, const char *currentCalendarType
)
237 : dateIntervalInfo(diInfo
), nextCalendarType(currentCalendarType
, -1, US_INV
) { }
238 virtual ~DateIntervalSink();
240 virtual void put(const char *key
, ResourceValue
&value
, UBool
/*noFallback*/, UErrorCode
&errorCode
) {
241 if (U_FAILURE(errorCode
)) { return; }
243 // Iterate over all the calendar entries and only pick the 'intervalFormats' table.
244 ResourceTable dateIntervalData
= value
.getTable(errorCode
);
245 if (U_FAILURE(errorCode
)) { return; }
246 for (int32_t i
= 0; dateIntervalData
.getKeyAndValue(i
, key
, value
); i
++) {
247 if (uprv_strcmp(key
, gIntervalDateTimePatternTag
) != 0) {
251 // Handle aliases and tables. Ignore the rest.
252 if (value
.getType() == URES_ALIAS
) {
253 // Get the calendar type for the alias path.
254 const UnicodeString
&aliasPath
= value
.getAliasUnicodeString(errorCode
);
255 if (U_FAILURE(errorCode
)) { return; }
257 nextCalendarType
.remove();
258 getCalendarTypeFromPath(aliasPath
, nextCalendarType
, errorCode
);
260 if (U_FAILURE(errorCode
)) {
261 resetNextCalendarType();
265 } else if (value
.getType() == URES_TABLE
) {
266 // Iterate over all the skeletons in the 'intervalFormat' table.
267 ResourceTable skeletonData
= value
.getTable(errorCode
);
268 if (U_FAILURE(errorCode
)) { return; }
269 for (int32_t j
= 0; skeletonData
.getKeyAndValue(j
, key
, value
); j
++) {
270 if (value
.getType() == URES_TABLE
) {
271 // Process the skeleton
272 processSkeletonTable(key
, value
, errorCode
);
273 if (U_FAILURE(errorCode
)) { return; }
282 * Processes the patterns for a skeleton table
284 void processSkeletonTable(const char *key
, ResourceValue
&value
, UErrorCode
&errorCode
) {
285 if (U_FAILURE(errorCode
)) { return; }
287 // Iterate over all the patterns in the current skeleton table
288 const char *currentSkeleton
= key
;
289 ResourceTable patternData
= value
.getTable(errorCode
);
290 if (U_FAILURE(errorCode
)) { return; }
291 for (int32_t k
= 0; patternData
.getKeyAndValue(k
, key
, value
); k
++) {
292 if (value
.getType() == URES_STRING
) {
294 UCalendarDateFields calendarField
= validateAndProcessPatternLetter(key
);
296 // If the calendar field has a valid value
297 if (calendarField
< UCAL_FIELD_COUNT
) {
298 // Set the interval pattern
299 setIntervalPatternIfAbsent(currentSkeleton
, calendarField
, value
, errorCode
);
300 if (U_FAILURE(errorCode
)) { return; }
307 * Extracts the calendar type from the path.
309 static void getCalendarTypeFromPath(const UnicodeString
&path
, UnicodeString
&calendarType
,
310 UErrorCode
&errorCode
) {
311 if (U_FAILURE(errorCode
)) { return; }
313 if (!path
.startsWith(PATH_PREFIX
, PATH_PREFIX_LENGTH
) || !path
.endsWith(PATH_SUFFIX
, PATH_SUFFIX_LENGTH
)) {
314 errorCode
= U_INVALID_FORMAT_ERROR
;
318 path
.extractBetween(PATH_PREFIX_LENGTH
, path
.length() - PATH_SUFFIX_LENGTH
, calendarType
);
322 * Validates and processes the pattern letter
324 UCalendarDateFields
validateAndProcessPatternLetter(const char *patternLetter
) {
325 // Check that patternLetter is just one letter
327 if ((c0
= patternLetter
[0]) != 0 && patternLetter
[1] == 0) {
328 // Check that the pattern letter is accepted
331 } else if (c0
== 'M') {
333 } else if (c0
== 'd') {
335 } else if (c0
== 'a') {
337 } else if (c0
== 'h' || c0
== 'H') {
339 } else if (c0
== 'm') {
341 }// TODO(ticket:12190): Why icu4c doesn't accept the calendar field "s" but icu4j does?
343 return UCAL_FIELD_COUNT
;
347 * Stores the interval pattern for the current skeleton in the internal data structure
348 * if it's not present.
350 void setIntervalPatternIfAbsent(const char *currentSkeleton
, UCalendarDateFields lrgDiffCalUnit
,
351 const ResourceValue
&value
, UErrorCode
&errorCode
) {
352 // Check if the pattern has already been stored on the data structure
353 IntervalPatternIndex index
=
354 dateIntervalInfo
.calendarFieldToIntervalIndex(lrgDiffCalUnit
, errorCode
);
355 if (U_FAILURE(errorCode
)) { return; }
357 UnicodeString
skeleton(currentSkeleton
, -1, US_INV
);
358 UnicodeString
* patternsOfOneSkeleton
=
359 (UnicodeString
*)(dateIntervalInfo
.fIntervalPatterns
->get(skeleton
));
361 if (patternsOfOneSkeleton
== NULL
|| patternsOfOneSkeleton
[index
].isEmpty()) {
362 UnicodeString pattern
= value
.getUnicodeString(errorCode
);
363 dateIntervalInfo
.setIntervalPatternInternally(skeleton
, lrgDiffCalUnit
,
368 const UnicodeString
&getNextCalendarType() {
369 return nextCalendarType
;
372 void resetNextCalendarType() {
373 nextCalendarType
.setToBogus();
377 // Virtual destructors must be defined out of line.
378 DateIntervalInfo::DateIntervalSink::~DateIntervalSink() {}
383 DateIntervalInfo::initializeData(const Locale
& locale
, UErrorCode
& status
)
385 fIntervalPatterns
= initHash(status
);
386 if (U_FAILURE(status
)) {
389 const char *locName
= locale
.getName();
391 // Get the correct calendar type
392 const char * calendarTypeToUse
= gGregorianTag
; // initial default
393 char calendarType
[ULOC_KEYWORDS_CAPACITY
]; // to be filled in with the type to use, if all goes well
394 char localeWithCalendarKey
[ULOC_LOCALE_IDENTIFIER_CAPACITY
];
395 // obtain a locale that always has the calendar key value that should be used
396 (void)ures_getFunctionalEquivalent(localeWithCalendarKey
, ULOC_LOCALE_IDENTIFIER_CAPACITY
, NULL
,
397 "calendar", "calendar", locName
, NULL
, FALSE
, &status
);
398 localeWithCalendarKey
[ULOC_LOCALE_IDENTIFIER_CAPACITY
-1] = 0; // ensure null termination
399 // now get the calendar key value from that locale
400 int32_t calendarTypeLen
= uloc_getKeywordValue(localeWithCalendarKey
, "calendar", calendarType
,
401 ULOC_KEYWORDS_CAPACITY
, &status
);
402 if (U_SUCCESS(status
) && calendarTypeLen
< ULOC_KEYWORDS_CAPACITY
) {
403 calendarTypeToUse
= calendarType
;
405 status
= U_ZERO_ERROR
;
407 // Instantiate the resource bundles
408 UResourceBundle
*rb
, *calBundle
;
409 rb
= ures_open(NULL
, locName
, &status
);
410 if (U_FAILURE(status
)) {
413 calBundle
= ures_getByKeyWithFallback(rb
, gCalendarTag
, NULL
, &status
);
416 if (U_SUCCESS(status
)) {
417 UResourceBundle
*calTypeBundle
, *itvDtPtnResource
;
419 // Get the fallback pattern
421 int32_t resStrLen
= 0;
422 calTypeBundle
= ures_getByKeyWithFallback(calBundle
, calendarTypeToUse
, NULL
, &status
);
423 itvDtPtnResource
= ures_getByKeyWithFallback(calTypeBundle
,
424 gIntervalDateTimePatternTag
, NULL
, &status
);
425 resStr
= ures_getStringByKeyWithFallback(itvDtPtnResource
, gFallbackPatternTag
,
426 &resStrLen
, &status
);
427 if ( U_SUCCESS(status
) ) {
428 UnicodeString pattern
= UnicodeString(TRUE
, resStr
, resStrLen
);
429 setFallbackIntervalPattern(pattern
, status
);
431 ures_close(itvDtPtnResource
);
432 ures_close(calTypeBundle
);
435 // Instantiate the sink
436 DateIntervalSink
sink(*this, calendarTypeToUse
);
437 const UnicodeString
&calendarTypeToUseUString
= sink
.getNextCalendarType();
439 // Already loaded calendar types
440 Hashtable
loadedCalendarTypes(FALSE
, status
);
442 if (U_SUCCESS(status
)) {
443 while (!calendarTypeToUseUString
.isBogus()) {
444 // Set an error when a loop is detected
445 if (loadedCalendarTypes
.geti(calendarTypeToUseUString
) == 1) {
446 status
= U_INVALID_FORMAT_ERROR
;
450 // Register the calendar type to avoid loops
451 loadedCalendarTypes
.puti(calendarTypeToUseUString
, 1, status
);
452 if (U_FAILURE(status
)) { break; }
454 // Get the calendar string
455 CharString calTypeBuffer
;
456 calTypeBuffer
.appendInvariantChars(calendarTypeToUseUString
, status
);
457 if (U_FAILURE(status
)) { break; }
458 const char *calType
= calTypeBuffer
.data();
460 // Reset the next calendar type to load.
461 sink
.resetNextCalendarType();
463 // Get all resources for this calendar type
464 ures_getAllItemsWithFallback(calBundle
, calType
, sink
, status
);
469 // Close the opened resource bundles
470 ures_close(calBundle
);
475 DateIntervalInfo::setIntervalPatternInternally(const UnicodeString
& skeleton
,
476 UCalendarDateFields lrgDiffCalUnit
,
477 const UnicodeString
& intervalPattern
,
478 UErrorCode
& status
) {
479 IntervalPatternIndex index
= calendarFieldToIntervalIndex(lrgDiffCalUnit
,status
);
480 if ( U_FAILURE(status
) ) {
483 UnicodeString
* patternsOfOneSkeleton
= (UnicodeString
*)(fIntervalPatterns
->get(skeleton
));
484 UBool emptyHash
= false;
485 if ( patternsOfOneSkeleton
== NULL
) {
486 patternsOfOneSkeleton
= new UnicodeString
[kIPI_MAX_INDEX
];
490 patternsOfOneSkeleton
[index
] = intervalPattern
;
491 if ( emptyHash
== TRUE
) {
492 fIntervalPatterns
->put(skeleton
, patternsOfOneSkeleton
, status
);
499 DateIntervalInfo::parseSkeleton(const UnicodeString
& skeleton
,
500 int32_t* skeletonFieldWidth
) {
501 const int8_t PATTERN_CHAR_BASE
= 0x41;
503 for ( i
= 0; i
< skeleton
.length(); ++i
) {
504 // it is an ASCII char in skeleton
505 int8_t ch
= (int8_t)skeleton
.charAt(i
);
506 ++skeletonFieldWidth
[ch
- PATTERN_CHAR_BASE
];
513 DateIntervalInfo::stringNumeric(int32_t fieldWidth
, int32_t anotherFieldWidth
,
514 char patternLetter
) {
515 if ( patternLetter
== 'M' ) {
516 if ( (fieldWidth
<= 2 && anotherFieldWidth
> 2) ||
517 (fieldWidth
> 2 && anotherFieldWidth
<= 2 )) {
527 DateIntervalInfo::getBestSkeleton(const UnicodeString
& skeleton
,
528 int8_t& bestMatchDistanceInfo
) const {
529 #ifdef DTITVINF_DEBUG
533 skeleton
.extract(0, skeleton
.length(), result
, "UTF-8");
534 sprintf(mesg
, "in getBestSkeleton: skeleton: %s; \n", result
);
539 int32_t inputSkeletonFieldWidth
[] =
541 // A B C D E F G H I J K L M N O
542 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
543 // P Q R S T U V W X Y Z
544 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
545 // a b c d e f g h i j k l m n o
546 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
547 // p q r s t u v w x y z
548 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
551 int32_t skeletonFieldWidth
[] =
553 // A B C D E F G H I J K L M N O
554 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
555 // P Q R S T U V W X Y Z
556 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
557 // a b c d e f g h i j k l m n o
558 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
559 // p q r s t u v w x y z
560 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
563 const int32_t DIFFERENT_FIELD
= 0x1000;
564 const int32_t STRING_NUMERIC_DIFFERENCE
= 0x100;
565 const int32_t BASE
= 0x41;
566 const UChar CHAR_V
= 0x0076;
567 const UChar CHAR_Z
= 0x007A;
569 // hack for 'v' and 'z'.
570 // resource bundle only have time skeletons ending with 'v',
571 // but not for time skeletons ending with 'z'.
572 UBool replaceZWithV
= false;
573 const UnicodeString
* inputSkeleton
= &skeleton
;
574 UnicodeString copySkeleton
;
575 if ( skeleton
.indexOf(CHAR_Z
) != -1 ) {
576 copySkeleton
= skeleton
;
577 copySkeleton
.findAndReplace(UnicodeString(CHAR_Z
), UnicodeString(CHAR_V
));
578 inputSkeleton
= ©Skeleton
;
579 replaceZWithV
= true;
582 parseSkeleton(*inputSkeleton
, inputSkeletonFieldWidth
);
583 int32_t bestDistance
= MAX_POSITIVE_INT
;
584 const UnicodeString
* bestSkeleton
= NULL
;
586 // 0 means exact the same skeletons;
587 // 1 means having the same field, but with different length,
588 // 2 means only z/v differs
589 // -1 means having different field.
590 bestMatchDistanceInfo
= 0;
591 int8_t fieldLength
= UPRV_LENGTHOF(skeletonFieldWidth
);
593 int32_t pos
= UHASH_FIRST
;
594 const UHashElement
* elem
= NULL
;
595 while ( (elem
= fIntervalPatterns
->nextElement(pos
)) != NULL
) {
596 const UHashTok keyTok
= elem
->key
;
597 UnicodeString
* skeleton
= (UnicodeString
*)keyTok
.pointer
;
598 #ifdef DTITVINF_DEBUG
599 skeleton
->extract(0, skeleton
->length(), result
, "UTF-8");
600 sprintf(mesg
, "available skeletons: skeleton: %s; \n", result
);
604 // clear skeleton field width
606 for ( i
= 0; i
< fieldLength
; ++i
) {
607 skeletonFieldWidth
[i
] = 0;
609 parseSkeleton(*skeleton
, skeletonFieldWidth
);
610 // calculate distance
611 int32_t distance
= 0;
612 int8_t fieldDifference
= 1;
613 for ( i
= 0; i
< fieldLength
; ++i
) {
614 int32_t inputFieldWidth
= inputSkeletonFieldWidth
[i
];
615 int32_t fieldWidth
= skeletonFieldWidth
[i
];
616 if ( inputFieldWidth
== fieldWidth
) {
619 if ( inputFieldWidth
== 0 ) {
620 fieldDifference
= -1;
621 distance
+= DIFFERENT_FIELD
;
622 } else if ( fieldWidth
== 0 ) {
623 fieldDifference
= -1;
624 distance
+= DIFFERENT_FIELD
;
625 } else if (stringNumeric(inputFieldWidth
, fieldWidth
,
627 distance
+= STRING_NUMERIC_DIFFERENCE
;
629 distance
+= (inputFieldWidth
> fieldWidth
) ?
630 (inputFieldWidth
- fieldWidth
) :
631 (fieldWidth
- inputFieldWidth
);
634 if ( distance
< bestDistance
) {
635 bestSkeleton
= skeleton
;
636 bestDistance
= distance
;
637 bestMatchDistanceInfo
= fieldDifference
;
639 if ( distance
== 0 ) {
640 bestMatchDistanceInfo
= 0;
644 if ( replaceZWithV
&& bestMatchDistanceInfo
!= -1 ) {
645 bestMatchDistanceInfo
= 2;
652 DateIntervalInfo::IntervalPatternIndex
653 DateIntervalInfo::calendarFieldToIntervalIndex(UCalendarDateFields field
,
654 UErrorCode
& status
) {
655 if ( U_FAILURE(status
) ) {
656 return kIPI_MAX_INDEX
;
658 IntervalPatternIndex index
= kIPI_MAX_INDEX
;
670 case UCAL_DAY_OF_WEEK
:
671 //case UCAL_DAY_OF_MONTH:
678 case UCAL_HOUR_OF_DAY
:
688 status
= U_ILLEGAL_ARGUMENT_ERROR
;
696 DateIntervalInfo::deleteHash(Hashtable
* hTable
)
698 if ( hTable
== NULL
) {
701 int32_t pos
= UHASH_FIRST
;
702 const UHashElement
* element
= NULL
;
703 while ( (element
= hTable
->nextElement(pos
)) != NULL
) {
704 const UHashTok valueTok
= element
->value
;
705 const UnicodeString
* value
= (UnicodeString
*)valueTok
.pointer
;
708 delete fIntervalPatterns
;
715 * set hash table value comparator
717 * @param val1 one value in comparison
718 * @param val2 the other value in comparison
719 * @return TRUE if 2 values are the same, FALSE otherwise
721 static UBool U_CALLCONV
dtitvinfHashTableValueComparator(UHashTok val1
, UHashTok val2
);
724 U_CALLCONV
dtitvinfHashTableValueComparator(UHashTok val1
, UHashTok val2
) {
725 const UnicodeString
* pattern1
= (UnicodeString
*)val1
.pointer
;
726 const UnicodeString
* pattern2
= (UnicodeString
*)val2
.pointer
;
729 for ( i
= 0; i
< DateIntervalInfo::kMaxIntervalPatternIndex
&& ret
== TRUE
; ++i
) {
730 ret
= (pattern1
[i
] == pattern2
[i
]);
739 DateIntervalInfo::initHash(UErrorCode
& status
) {
740 if ( U_FAILURE(status
) ) {
744 if ( (hTable
= new Hashtable(FALSE
, status
)) == NULL
) {
745 status
= U_MEMORY_ALLOCATION_ERROR
;
748 if ( U_FAILURE(status
) ) {
752 hTable
->setValueComparator(dtitvinfHashTableValueComparator
);
758 DateIntervalInfo::copyHash(const Hashtable
* source
,
760 UErrorCode
& status
) {
761 if ( U_FAILURE(status
) ) {
764 int32_t pos
= UHASH_FIRST
;
765 const UHashElement
* element
= NULL
;
767 while ( (element
= source
->nextElement(pos
)) != NULL
) {
768 const UHashTok keyTok
= element
->key
;
769 const UnicodeString
* key
= (UnicodeString
*)keyTok
.pointer
;
770 const UHashTok valueTok
= element
->value
;
771 const UnicodeString
* value
= (UnicodeString
*)valueTok
.pointer
;
772 UnicodeString
* copy
= new UnicodeString
[kIPI_MAX_INDEX
];
774 for ( i
= 0; i
< kIPI_MAX_INDEX
; ++i
) {
777 target
->put(UnicodeString(*key
), copy
, status
);
778 if ( U_FAILURE(status
) ) {