35 36 /** 37 * Concrete implementation of the {@link java.util.spi.CalendarDataProvider 38 * CalendarDataProvider} class for the JRE LocaleProviderAdapter. 39 * 40 * @author Masayoshi Okutsu 41 * @author Naoto Sato 42 */ 43 public class CalendarNameProviderImpl extends CalendarNameProvider implements AvailableLanguageTags { 44 private final LocaleProviderAdapter.Type type; 45 private final Set<String> langtags; 46 47 public CalendarNameProviderImpl(LocaleProviderAdapter.Type type, Set<String> langtags) { 48 this.type = type; 49 this.langtags = langtags; 50 } 51 52 @Override 53 public String getDisplayName(String calendarType, int field, int value, int style, Locale locale) { 54 String name = null; 55 String key = getKey(calendarType, field, style); 56 if (key != null) { 57 ResourceBundle rb = LocaleProviderAdapter.forType(type).getLocaleData().getDateFormatData(locale); 58 if (rb.containsKey(key)) { 59 String[] strings = rb.getStringArray(key); 60 if (strings.length > 0) { 61 if (field == DAY_OF_WEEK || field == YEAR) { 62 --value; 63 } 64 name = strings[value]; 65 // If name is empty in standalone, try its `format' style. 66 if (name.length() == 0 67 && (style == SHORT_STANDALONE || style == LONG_STANDALONE)) { 68 name = getDisplayName(calendarType, field, value, 69 style == SHORT_STANDALONE ? SHORT_FORMAT : LONG_FORMAT, 70 locale); 71 } 72 } 73 } 74 } 75 return name; 76 } 77 78 @Override 79 public Map<String, Integer> getDisplayNames(String calendarType, int field, int style, Locale locale) { 80 Map<String, Integer> names; 81 if (style == ALL_STYLES) { 82 names = getDisplayNamesImpl(calendarType, field, SHORT_FORMAT, locale); 83 if (field != AM_PM) { 84 for (int st : new int[] { SHORT_STANDALONE, LONG_FORMAT, LONG_STANDALONE }) { 85 names.putAll(getDisplayNamesImpl(calendarType, field, st, locale)); 86 } 87 } 88 } else { 89 // specific style 90 names = getDisplayNamesImpl(calendarType, field, style, locale); 91 } 92 return names.isEmpty() ? null : names; 93 } 94 95 private Map<String, Integer> getDisplayNamesImpl(String calendarType, int field, 96 int style, Locale locale) { 97 String key = getKey(calendarType, field, style); 98 Map<String, Integer> map = new TreeMap<>(LengthBasedComparator.INSTANCE); 99 if (key != null) { 100 ResourceBundle rb = LocaleProviderAdapter.forType(type).getLocaleData().getDateFormatData(locale); 101 if (rb.containsKey(key)) { 102 String[] strings = rb.getStringArray(key); 103 if (field == YEAR) { 104 if (strings.length > 0) { 105 map.put(strings[0], 1); 106 } 107 } else { 108 int base = (field == DAY_OF_WEEK) ? 1 : 0; 109 for (int i = 0; i < strings.length; i++) { 110 String name = strings[i]; 111 // Ignore any empty string (some standalone month names 112 // are not defined) 113 if (name.length() == 0) { 114 continue; 115 } 116 map.put(name, base + i); 117 } 118 } 119 } 120 } 121 return map; 122 } 123 124 /** 125 * Comparator implementation for TreeMap which iterates keys from longest 126 * to shortest. 127 */ 128 private static class LengthBasedComparator implements Comparator<String> { 129 private static final LengthBasedComparator INSTANCE = new LengthBasedComparator(); 130 131 private LengthBasedComparator() { 132 } 133 134 @Override 135 public int compare(String o1, String o2) { 136 int n = o2.length() - o1.length(); 137 return (n == 0) ? o1.compareTo(o2) : n; 138 } 139 } 140 141 @Override 142 public Locale[] getAvailableLocales() { 143 return LocaleProviderAdapter.toLocaleArray(langtags); 163 default: 164 // Unknown calendar type 165 return false; 166 } 167 } 168 if (langtags.contains(locale.toLanguageTag())) { 169 return true; 170 } 171 if (type == LocaleProviderAdapter.Type.JRE) { 172 String oldname = locale.toString().replace('_', '-'); 173 return langtags.contains(oldname); 174 } 175 return false; 176 } 177 178 @Override 179 public Set<String> getAvailableLanguageTags() { 180 return langtags; 181 } 182 183 private int getIntData(String key, Locale locale) { 184 ResourceBundle rb = LocaleProviderAdapter.forType(type).getLocaleData().getCalendarData(locale); 185 if (rb.containsKey(key)) { 186 String firstday = rb.getString(key); 187 return Integer.parseInt(firstday); 188 } 189 // Note that the base bundle of CLDR doesn't have the Calendar week parameters. 190 return 0; 191 } 192 193 private String getKey(String type, int field, int style) { 194 boolean standalone = (style & 0x8000) != 0; 195 style &= ~0x8000; 196 197 if ("gregory".equals(type)) { 198 type = null; 199 } 200 201 StringBuilder key = new StringBuilder(); 202 switch (field) { 203 case ERA: 204 if (type != null) { 205 key.append(type).append('.'); 206 } 207 if (style == SHORT) { 208 key.append("short."); 209 } 210 key.append("Eras"); 211 break; 212 213 case YEAR: 214 key.append(type).append(".FirstYear"); 215 break; 216 217 case MONTH: 218 if (standalone) { 219 key.append("standalone."); 220 } 221 key.append(style == SHORT ? "MonthAbbreviations" : "MonthNames"); 222 break; 223 224 case DAY_OF_WEEK: 225 key.append(style == SHORT ? "DayAbbreviations" : "DayNames"); 226 break; 227 228 case AM_PM: 229 key.append("AmPmMarkers"); 230 break; 231 } 232 return key.length() > 0 ? key.toString() : null; 233 } 234 } | 35 36 /** 37 * Concrete implementation of the {@link java.util.spi.CalendarDataProvider 38 * CalendarDataProvider} class for the JRE LocaleProviderAdapter. 39 * 40 * @author Masayoshi Okutsu 41 * @author Naoto Sato 42 */ 43 public class CalendarNameProviderImpl extends CalendarNameProvider implements AvailableLanguageTags { 44 private final LocaleProviderAdapter.Type type; 45 private final Set<String> langtags; 46 47 public CalendarNameProviderImpl(LocaleProviderAdapter.Type type, Set<String> langtags) { 48 this.type = type; 49 this.langtags = langtags; 50 } 51 52 @Override 53 public String getDisplayName(String calendarType, int field, int value, int style, Locale locale) { 54 String name = null; 55 String key = getResourceKey(calendarType, field, style); 56 if (key != null) { 57 ResourceBundle rb = LocaleProviderAdapter.forType(type).getLocaleData().getDateFormatData(locale); 58 if (rb.containsKey(key)) { 59 String[] strings = rb.getStringArray(key); 60 if (strings.length > 0) { 61 if (field == DAY_OF_WEEK || field == YEAR) { 62 --value; 63 } 64 name = strings[value]; 65 // If name is empty in standalone, try its `format' style. 66 if (name.length() == 0 67 && (style == SHORT_STANDALONE || style == LONG_STANDALONE 68 || style == NARROW_STANDALONE)) { 69 name = getDisplayName(calendarType, field, value, 70 getBaseStyle(style), 71 locale); 72 } 73 } 74 } 75 } 76 return name; 77 } 78 79 private static int[] REST_OF_STYLES = { 80 SHORT_STANDALONE, LONG_FORMAT, LONG_STANDALONE, 81 NARROW_FORMAT, NARROW_STANDALONE 82 }; 83 @Override 84 public Map<String, Integer> getDisplayNames(String calendarType, int field, int style, Locale locale) { 85 Map<String, Integer> names; 86 if (style == ALL_STYLES) { 87 names = getDisplayNamesImpl(calendarType, field, SHORT_FORMAT, locale); 88 for (int st : REST_OF_STYLES) { 89 names.putAll(getDisplayNamesImpl(calendarType, field, st, locale)); 90 } 91 } else { 92 // specific style 93 names = getDisplayNamesImpl(calendarType, field, style, locale); 94 } 95 return names.isEmpty() ? null : names; 96 } 97 98 private Map<String, Integer> getDisplayNamesImpl(String calendarType, int field, 99 int style, Locale locale) { 100 String key = getResourceKey(calendarType, field, style); 101 Map<String, Integer> map = new TreeMap<>(LengthBasedComparator.INSTANCE); 102 if (key != null) { 103 ResourceBundle rb = LocaleProviderAdapter.forType(type).getLocaleData().getDateFormatData(locale); 104 if (rb.containsKey(key)) { 105 String[] strings = rb.getStringArray(key); 106 if (!hasDuplicates(strings)) { 107 if (field == YEAR) { 108 if (strings.length > 0) { 109 map.put(strings[0], 1); 110 } 111 } else { 112 int base = (field == DAY_OF_WEEK) ? 1 : 0; 113 for (int i = 0; i < strings.length; i++) { 114 String name = strings[i]; 115 // Ignore any empty string (some standalone month names 116 // are not defined) 117 if (name.length() == 0) { 118 continue; 119 } 120 map.put(name, base + i); 121 } 122 } 123 } 124 } 125 } 126 return map; 127 } 128 129 private int getBaseStyle(int style) { 130 return style & ~(SHORT_STANDALONE - SHORT_FORMAT); 131 } 132 133 /** 134 * Comparator implementation for TreeMap which iterates keys from longest 135 * to shortest. 136 */ 137 private static class LengthBasedComparator implements Comparator<String> { 138 private static final LengthBasedComparator INSTANCE = new LengthBasedComparator(); 139 140 private LengthBasedComparator() { 141 } 142 143 @Override 144 public int compare(String o1, String o2) { 145 int n = o2.length() - o1.length(); 146 return (n == 0) ? o1.compareTo(o2) : n; 147 } 148 } 149 150 @Override 151 public Locale[] getAvailableLocales() { 152 return LocaleProviderAdapter.toLocaleArray(langtags); 172 default: 173 // Unknown calendar type 174 return false; 175 } 176 } 177 if (langtags.contains(locale.toLanguageTag())) { 178 return true; 179 } 180 if (type == LocaleProviderAdapter.Type.JRE) { 181 String oldname = locale.toString().replace('_', '-'); 182 return langtags.contains(oldname); 183 } 184 return false; 185 } 186 187 @Override 188 public Set<String> getAvailableLanguageTags() { 189 return langtags; 190 } 191 192 private boolean hasDuplicates(String[] strings) { 193 int len = strings.length; 194 for (int i = 0; i < len - 1; i++) { 195 String a = strings[i]; 196 if (a != null) { 197 for (int j = i + 1; j < len; j++) { 198 if (a.equals(strings[j])) { 199 return true; 200 } 201 } 202 } 203 } 204 return false; 205 } 206 207 private String getResourceKey(String type, int field, int style) { 208 int baseStyle = getBaseStyle(style); 209 boolean isStandalone = (style != baseStyle); 210 211 if ("gregory".equals(type)) { 212 type = null; 213 } 214 boolean isNarrow = (baseStyle == NARROW_FORMAT); 215 StringBuilder key = new StringBuilder(); 216 switch (field) { 217 case ERA: 218 if (type != null) { 219 key.append(type).append('.'); 220 } 221 if (isNarrow) { 222 key.append("narrow."); 223 } else { 224 // JRE and CLDR use different resource key conventions 225 // due to historical reasons. (JRE DateFormatSymbols.getEras returns 226 // abbreviations while other getShort*() return abbreviations.) 227 if (this.type == LocaleProviderAdapter.Type.JRE) { 228 if (baseStyle == SHORT) { 229 key.append("short."); 230 } 231 } else { // CLDR 232 if (baseStyle == LONG) { 233 key.append("long."); 234 } 235 } 236 } 237 key.append("Eras"); 238 break; 239 240 case YEAR: 241 if (!isNarrow) { 242 key.append(type).append(".FirstYear"); 243 } 244 break; 245 246 case MONTH: 247 if (isStandalone) { 248 key.append("standalone."); 249 } 250 key.append("Month").append(toStyleName(baseStyle)); 251 break; 252 253 case DAY_OF_WEEK: 254 // support standalone narrow day names 255 if (isStandalone && isNarrow) { 256 key.append("standalone."); 257 } 258 key.append("Day").append(toStyleName(baseStyle)); 259 break; 260 261 case AM_PM: 262 if (isNarrow) { 263 key.append("narrow."); 264 } 265 key.append("AmPmMarkers"); 266 break; 267 } 268 return key.length() > 0 ? key.toString() : null; 269 } 270 271 private String toStyleName(int baseStyle) { 272 switch (baseStyle) { 273 case SHORT: 274 return "Abbreviations"; 275 case NARROW_FORMAT: 276 return "Narrows"; 277 } 278 return "Names"; 279 } 280 } |