tor-browser

The Tor Browser
git clone https://git.dasho.dev/tor-browser.git
Log | Files | Refs | README | LICENSE

dtitvinf.h (18981B)


      1 // © 2016 and later: Unicode, Inc. and others.
      2 // License & terms of use: http://www.unicode.org/copyright.html
      3 /*
      4 *******************************************************************************
      5 * Copyright (C) 2008-2016, International Business Machines Corporation and
      6 * others. All Rights Reserved.
      7 *******************************************************************************
      8 *
      9 * File DTITVINF.H
     10 *
     11 *******************************************************************************
     12 */
     13 
     14 #ifndef __DTITVINF_H__
     15 #define __DTITVINF_H__
     16 
     17 #include "unicode/utypes.h"
     18 
     19 #if U_SHOW_CPLUSPLUS_API
     20 
     21 /**
     22 * \file
     23 * \brief C++ API: Date/Time interval patterns for formatting date/time interval
     24 */
     25 
     26 #if !UCONFIG_NO_FORMATTING
     27 
     28 #include "unicode/udat.h"
     29 #include "unicode/locid.h"
     30 #include "unicode/ucal.h"
     31 #include "unicode/dtptngen.h"
     32 
     33 U_NAMESPACE_BEGIN
     34 
     35 /**
     36 * DateIntervalInfo is a public class for encapsulating localizable
     37 * date time interval patterns. It is used by DateIntervalFormat.
     38 *
     39 * <P>
     40 * For most users, ordinary use of DateIntervalFormat does not need to create
     41 * DateIntervalInfo object directly.
     42 * DateIntervalFormat will take care of it when creating a date interval
     43 * formatter when user pass in skeleton and locale.
     44 *
     45 * <P>
     46 * For power users, who want to create their own date interval patterns,
     47 * or want to re-set date interval patterns, they could do so by
     48 * directly creating DateIntervalInfo and manipulating it.
     49 *
     50 * <P>
     51 * Logically, the interval patterns are mappings
     52 * from (skeleton, the_largest_different_calendar_field)
     53 * to (date_interval_pattern).
     54 *
     55 * <P>
     56 * A skeleton
     57 * <ol>
     58 * <li>
     59 * only keeps the field pattern letter and ignores all other parts
     60 * in a pattern, such as space, punctuations, and string literals.
     61 * <li>
     62 * hides the order of fields.
     63 * <li>
     64 * might hide a field's pattern letter length.
     65 *
     66 * For those non-digit calendar fields, the pattern letter length is
     67 * important, such as MMM, MMMM, and MMMMM; EEE and EEEE,
     68 * and the field's pattern letter length is honored.
     69 *
     70 * For the digit calendar fields,  such as M or MM, d or dd, yy or yyyy,
     71 * the field pattern length is ignored and the best match, which is defined
     72 * in date time patterns, will be returned without honor the field pattern
     73 * letter length in skeleton.
     74 * </ol>
     75 *
     76 * <P>
     77 * The calendar fields we support for interval formatting are:
     78 * year, month, date, day-of-week, am-pm, hour, hour-of-day, and minute.
     79 * Those calendar fields can be defined in the following order:
     80 * year >  month > date > am-pm > hour >  minute
     81 *
     82 * The largest different calendar fields between 2 calendars is the
     83 * first different calendar field in above order.
     84 *
     85 * For example: the largest different calendar fields between &quot;Jan 10, 2007&quot;
     86 * and &quot;Feb 20, 2008&quot; is year.
     87 *
     88 * <P>
     89 * There is a set of pre-defined static skeleton strings.
     90 * There are pre-defined interval patterns for those pre-defined skeletons
     91 * in locales' resource files.
     92 * For example, for a skeleton UDAT_YEAR_ABBR_MONTH_DAY, which is  &quot;yMMMd&quot;,
     93 * in  en_US, if the largest different calendar field between date1 and date2
     94 * is &quot;year&quot;, the date interval pattern  is &quot;MMM d, yyyy - MMM d, yyyy&quot;,
     95 * such as &quot;Jan 10, 2007 - Jan 10, 2008&quot;.
     96 * If the largest different calendar field between date1 and date2 is &quot;month&quot;,
     97 * the date interval pattern is &quot;MMM d - MMM d, yyyy&quot;,
     98 * such as &quot;Jan 10 - Feb 10, 2007&quot;.
     99 * If the largest different calendar field between date1 and date2 is &quot;day&quot;,
    100 * the date interval pattern is &quot;MMM d-d, yyyy&quot;, such as &quot;Jan 10-20, 2007&quot;.
    101 *
    102 * For date skeleton, the interval patterns when year, or month, or date is
    103 * different are defined in resource files.
    104 * For time skeleton, the interval patterns when am/pm, or hour, or minute is
    105 * different are defined in resource files.
    106 *
    107 *
    108 * <P>
    109 * There are 2 dates in interval pattern. For most locales, the first date
    110 * in an interval pattern is the earlier date. There might be a locale in which
    111 * the first date in an interval pattern is the later date.
    112 * We use fallback format for the default order for the locale.
    113 * For example, if the fallback format is &quot;{0} - {1}&quot;, it means
    114 * the first date in the interval pattern for this locale is earlier date.
    115 * If the fallback format is &quot;{1} - {0}&quot;, it means the first date is the
    116 * later date.
    117 * For a particular interval pattern, the default order can be overridden
    118 * by prefixing &quot;latestFirst:&quot; or &quot;earliestFirst:&quot; to the interval pattern.
    119 * For example, if the fallback format is &quot;{0}-{1}&quot;,
    120 * but for skeleton &quot;yMMMd&quot;, the interval pattern when day is different is
    121 * &quot;latestFirst:d-d MMM yy&quot;, it means by default, the first date in interval
    122 * pattern is the earlier date. But for skeleton &quot;yMMMd&quot;, when day is different,
    123 * the first date in &quot;d-d MMM yy&quot; is the later date.
    124 *
    125 * <P>
    126 * The recommended way to create a DateIntervalFormat object is to pass in
    127 * the locale.
    128 * By using a Locale parameter, the DateIntervalFormat object is
    129 * initialized with the pre-defined interval patterns for a given or
    130 * default locale.
    131 * <P>
    132 * Users can also create DateIntervalFormat object
    133 * by supplying their own interval patterns.
    134 * It provides flexibility for power users.
    135 *
    136 * <P>
    137 * After a DateIntervalInfo object is created, clients may modify
    138 * the interval patterns using setIntervalPattern function as so desired.
    139 * Currently, users can only set interval patterns when the following
    140 * calendar fields are different: ERA, YEAR, MONTH, DATE, DAY_OF_MONTH,
    141 * DAY_OF_WEEK, AM_PM, HOUR, HOUR_OF_DAY, MINUTE, SECOND, and MILLISECOND.
    142 * Interval patterns when other calendar fields are different is not supported.
    143 * <P>
    144 * DateIntervalInfo objects are cloneable.
    145 * When clients obtain a DateIntervalInfo object,
    146 * they can feel free to modify it as necessary.
    147 * <P>
    148 * DateIntervalInfo are not expected to be subclassed.
    149 * Data for a calendar is loaded out of resource bundles.
    150 * Through ICU 4.4, date interval patterns are only supported in the Gregorian
    151 * calendar; non-Gregorian calendars are supported from ICU 4.4.1.
    152 * @stable ICU 4.0
    153 **/
    154 class U_I18N_API DateIntervalInfo final : public UObject {
    155 public:
    156    /**
    157     * Default constructor.
    158     * It does not initialize any interval patterns except
    159     * that it initialize default fall-back pattern as "{0} - {1}",
    160     * which can be reset by setFallbackIntervalPattern().
    161     * It should be followed by setFallbackIntervalPattern() and
    162     * setIntervalPattern(),
    163     * and is recommended to be used only for power users who
    164     * wants to create their own interval patterns and use them to create
    165     * date interval formatter.
    166     * @param status   output param set to success/failure code on exit
    167     * @internal ICU 4.0
    168     */
    169    DateIntervalInfo(UErrorCode& status);
    170 
    171 
    172    /**
    173     * Construct DateIntervalInfo for the given locale,
    174     * @param locale  the interval patterns are loaded from the appropriate calendar
    175     *                data (specified calendar or default calendar) in this locale.
    176     * @param status  output param set to success/failure code on exit
    177     * @stable ICU 4.0
    178     */
    179    DateIntervalInfo(const Locale& locale, UErrorCode& status);
    180 
    181 
    182    /**
    183     * Copy constructor.
    184     * @stable ICU 4.0
    185     */
    186    DateIntervalInfo(const DateIntervalInfo&);
    187 
    188    /**
    189     * Assignment operator
    190     * @stable ICU 4.0
    191     */
    192    DateIntervalInfo& operator=(const DateIntervalInfo&);
    193 
    194    /**
    195     * Clone this object polymorphically.
    196     * The caller owns the result and should delete it when done.
    197     * @return   a copy of the object
    198     * @stable ICU 4.0
    199     */
    200    DateIntervalInfo* clone() const;
    201 
    202    /**
    203     * Destructor.
    204     * @stable ICU 4.0
    205     */
    206    ~DateIntervalInfo();
    207 
    208 
    209    /**
    210     * Return true if another object is semantically equal to this one.
    211     *
    212     * @param other    the DateIntervalInfo object to be compared with.
    213     * @return         true if other is semantically equal to this.
    214     * @stable ICU 4.0
    215     */
    216    bool operator==(const DateIntervalInfo& other) const;
    217 
    218    /**
    219     * Return true if another object is semantically unequal to this one.
    220     *
    221     * @param other    the DateIntervalInfo object to be compared with.
    222     * @return         true if other is semantically unequal to this.
    223     * @stable ICU 4.0
    224     */
    225    bool operator!=(const DateIntervalInfo& other) const;
    226 
    227 
    228 
    229    /**
    230     * Provides a way for client to build interval patterns.
    231     * User could construct DateIntervalInfo by providing a list of skeletons
    232     * and their patterns.
    233     * <P>
    234     * For example:
    235     * <pre>
    236     * UErrorCode status = U_ZERO_ERROR;
    237     * DateIntervalInfo dIntervalInfo = new DateIntervalInfo();
    238     * dIntervalInfo->setFallbackIntervalPattern("{0} ~ {1}");
    239     * dIntervalInfo->setIntervalPattern("yMd", UCAL_YEAR, "'from' yyyy-M-d 'to' yyyy-M-d", status);
    240     * dIntervalInfo->setIntervalPattern("yMMMd", UCAL_MONTH, "'from' yyyy MMM d 'to' MMM d", status);
    241     * dIntervalInfo->setIntervalPattern("yMMMd", UCAL_DAY, "yyyy MMM d-d", status, status);
    242     * </pre>
    243     *
    244     * Restriction:
    245     * Currently, users can only set interval patterns when the following
    246     * calendar fields are different: ERA, YEAR, MONTH, DATE,  DAY_OF_MONTH,
    247     * DAY_OF_WEEK, AM_PM,  HOUR, HOUR_OF_DAY, MINUTE, SECOND and MILLISECOND.
    248     * Interval patterns when other calendar fields are different are
    249     * not supported.
    250     *
    251     * @param skeleton         the skeleton on which interval pattern based
    252     * @param lrgDiffCalUnit   the largest different calendar unit.
    253     * @param intervalPattern  the interval pattern on the largest different
    254     *                         calendar unit.
    255     *                         For example, if lrgDiffCalUnit is
    256     *                         "year", the interval pattern for en_US when year
    257     *                         is different could be "'from' yyyy 'to' yyyy".
    258     * @param status           output param set to success/failure code on exit
    259     * @stable ICU 4.0
    260     */
    261    void setIntervalPattern(const UnicodeString& skeleton,
    262                            UCalendarDateFields lrgDiffCalUnit,
    263                            const UnicodeString& intervalPattern,
    264                            UErrorCode& status);
    265 
    266    /**
    267     * Get the interval pattern given skeleton and
    268     * the largest different calendar field.
    269     * @param skeleton   the skeleton
    270     * @param field      the largest different calendar field
    271     * @param result     output param to receive the pattern
    272     * @param status     output param set to success/failure code on exit
    273     * @return a reference to 'result'
    274     * @stable ICU 4.0
    275     */
    276    UnicodeString& getIntervalPattern(const UnicodeString& skeleton,
    277                                      UCalendarDateFields field,
    278                                      UnicodeString& result,
    279                                      UErrorCode& status) const;
    280 
    281    /**
    282     * Get the fallback interval pattern.
    283     * @param  result   output param to receive the pattern
    284     * @return a reference to 'result'
    285     * @stable ICU 4.0
    286     */
    287    UnicodeString& getFallbackIntervalPattern(UnicodeString& result) const;
    288 
    289 
    290    /**
    291     * Re-set the fallback interval pattern.
    292     *
    293     * In construction, default fallback pattern is set as "{0} - {1}".
    294     * And constructor taking locale as parameter will set the
    295     * fallback pattern as what defined in the locale resource file.
    296     *
    297     * This method provides a way for user to replace the fallback pattern.
    298     *
    299     * @param fallbackPattern  fall-back interval pattern.
    300     * @param status           output param set to success/failure code on exit
    301     * @stable ICU 4.0
    302     */
    303    void setFallbackIntervalPattern(const UnicodeString& fallbackPattern,
    304                                    UErrorCode& status);
    305 
    306 
    307    /** Get default order -- whether the first date in pattern is later date
    308                             or not.
    309     * return default date ordering in interval pattern. true if the first date
    310     *        in pattern is later date, false otherwise.
    311     * @stable ICU 4.0
    312     */
    313    UBool getDefaultOrder() const;
    314 
    315 
    316    /**
    317     * ICU "poor man's RTTI", returns a UClassID for the actual class.
    318     *
    319     * @stable ICU 4.0
    320     */
    321    virtual UClassID getDynamicClassID() const override;
    322 
    323    /**
    324     * ICU "poor man's RTTI", returns a UClassID for this class.
    325     *
    326     * @stable ICU 4.0
    327     */
    328    static UClassID U_EXPORT2 getStaticClassID();
    329 
    330 
    331 private:
    332    /**
    333     * DateIntervalFormat will need access to
    334     * getBestSkeleton(), parseSkeleton(), enum IntervalPatternIndex,
    335     * and calendarFieldToPatternIndex().
    336     *
    337     * Instead of making above public,
    338     * make DateIntervalFormat a friend of DateIntervalInfo.
    339     */
    340    friend class DateIntervalFormat;
    341 
    342    /**
    343     * Internal struct used to load resource bundle data.
    344     */
    345    struct U_HIDDEN DateIntervalSink;
    346 
    347    /**
    348     * Following is for saving the interval patterns.
    349     * We only support interval patterns on
    350     * ERA, YEAR, MONTH, DAY, AM_PM, HOUR, MINUTE, SECOND and MILLISECOND.
    351     */
    352    enum IntervalPatternIndex
    353    {
    354        kIPI_ERA,
    355        kIPI_YEAR,
    356        kIPI_MONTH,
    357        kIPI_DATE,
    358        kIPI_AM_PM,
    359        kIPI_HOUR,
    360        kIPI_MINUTE,
    361        kIPI_SECOND,
    362        kIPI_MILLISECOND,
    363        kIPI_MAX_INDEX
    364    };
    365 public:
    366 #ifndef U_HIDE_INTERNAL_API
    367    /**
    368     * Max index for stored interval patterns
    369     * @internal ICU 4.4
    370     */
    371     enum {
    372         kMaxIntervalPatternIndex = kIPI_MAX_INDEX
    373     };
    374 #endif  /* U_HIDE_INTERNAL_API */
    375 private:
    376 
    377 
    378    /**
    379     * Initialize the DateIntervalInfo from locale
    380     * @param locale   the given locale.
    381     * @param status   output param set to success/failure code on exit
    382     */
    383    void initializeData(const Locale& locale, UErrorCode& status);
    384 
    385 
    386    /* Set Interval pattern.
    387     *
    388     * It sets interval pattern into the hash map.
    389     *
    390     * @param skeleton         skeleton on which the interval pattern based
    391     * @param lrgDiffCalUnit   the largest different calendar unit.
    392     * @param intervalPattern  the interval pattern on the largest different
    393     *                         calendar unit.
    394     * @param status           output param set to success/failure code on exit
    395     */
    396    void setIntervalPatternInternally(const UnicodeString& skeleton,
    397                                      UCalendarDateFields lrgDiffCalUnit,
    398                                      const UnicodeString& intervalPattern,
    399                                      UErrorCode& status);
    400 
    401 
    402    /**given an input skeleton, get the best match skeleton
    403     * which has pre-defined interval pattern in resource file.
    404     * Also return the difference between the input skeleton
    405     * and the best match skeleton.
    406     *
    407     * TODO (xji): set field weight or
    408     *             isolate the functionality in DateTimePatternGenerator
    409     * @param  skeleton               input skeleton
    410     * @param  bestMatchDistanceInfo  the difference between input skeleton
    411     *                                and best match skeleton.
    412     *         0, if there is exact match for input skeleton
    413     *         1, if there is only field width difference between
    414     *            the best match and the input skeleton
    415     *         2, the only field difference is 'v' and 'z'
    416     *        -1, if there is calendar field difference between
    417     *            the best match and the input skeleton
    418     * @return                        best match skeleton
    419     */
    420    const UnicodeString* getBestSkeleton(const UnicodeString& skeleton,
    421                                         int8_t& bestMatchDistanceInfo) const;
    422 
    423 
    424    /**
    425     * Parse skeleton, save each field's width.
    426     * It is used for looking for best match skeleton,
    427     * and adjust pattern field width.
    428     * @param skeleton            skeleton to be parsed
    429     * @param skeletonFieldWidth  parsed skeleton field width
    430     */
    431    static void U_EXPORT2 parseSkeleton(const UnicodeString& skeleton,
    432                                        int32_t* skeletonFieldWidth);
    433 
    434 
    435    /**
    436     * Check whether one field width is numeric while the other is string.
    437     *
    438     * TODO (xji): make it general
    439     *
    440     * @param fieldWidth          one field width
    441     * @param anotherFieldWidth   another field width
    442     * @param patternLetter       pattern letter char
    443     * @return true if one field width is numeric and the other is string,
    444     *         false otherwise.
    445     */
    446    static UBool U_EXPORT2 stringNumeric(int32_t fieldWidth,
    447                                         int32_t anotherFieldWidth,
    448                                         char patternLetter);
    449 
    450 
    451    /**
    452     * Convert calendar field to the interval pattern index in
    453     * hash table.
    454     *
    455     * Since we only support the following calendar fields:
    456     * ERA, YEAR, MONTH, DATE, DAY_OF_MONTH, DAY_OF_WEEK,
    457     * AM_PM, HOUR, HOUR_OF_DAY, MINUTE, SECOND, and MILLISECOND.
    458     * We reserve only 4 interval patterns for a skeleton.
    459     *
    460     * @param field    calendar field
    461     * @param status   output param set to success/failure code on exit
    462     * @return  interval pattern index in hash table
    463     */
    464    static IntervalPatternIndex U_EXPORT2 calendarFieldToIntervalIndex(
    465                                                      UCalendarDateFields field,
    466                                                      UErrorCode& status);
    467 
    468 
    469    /**
    470     * delete hash table (of type fIntervalPatterns).
    471     *
    472     * @param hTable  hash table to be deleted
    473     */
    474    void deleteHash(Hashtable* hTable);
    475 
    476 
    477    /**
    478     * initialize hash table (of type fIntervalPatterns).
    479     *
    480     * @param status   output param set to success/failure code on exit
    481     * @return         hash table initialized
    482     */
    483    Hashtable* initHash(UErrorCode& status);
    484 
    485 
    486 
    487    /**
    488     * copy hash table (of type fIntervalPatterns).
    489     *
    490     * @param source   the source to copy from
    491     * @param target   the target to copy to
    492     * @param status   output param set to success/failure code on exit
    493     */
    494    void copyHash(const Hashtable* source, Hashtable* target, UErrorCode& status);
    495 
    496 
    497    // data members
    498    // fallback interval pattern
    499    UnicodeString fFallbackIntervalPattern;
    500    // default order
    501    UBool fFirstDateInPtnIsLaterDate;
    502 
    503    // HashMap<UnicodeString, UnicodeString[kIPI_MAX_INDEX]>
    504    // HashMap( skeleton, pattern[largest_different_field] )
    505    Hashtable* fIntervalPatterns;
    506 
    507 };// end class DateIntervalInfo
    508 
    509 
    510 inline bool
    511 DateIntervalInfo::operator!=(const DateIntervalInfo& other) const {
    512    return !operator==(other);
    513 }
    514 
    515 
    516 U_NAMESPACE_END
    517 
    518 #endif
    519 
    520 #endif /* U_SHOW_CPLUSPLUS_API */
    521 
    522 #endif