1 /* GENERATED SOURCE. DO NOT MODIFY. */
2 // © 2016 and later: Unicode, Inc. and others.
3 // License & terms of use: http://www.unicode.org/copyright.html
4 /*
5  *******************************************************************************
6  * Copyright (C) 2007-2014, International Business Machines Corporation and    *
7  * others. All Rights Reserved.                                                *
8  *******************************************************************************
9  */
10 package android.icu.util;
11 
12 import java.util.BitSet;
13 import java.util.Date;
14 import java.util.LinkedList;
15 import java.util.List;
16 
17 import android.icu.impl.Grego;
18 
19 /**
20  * <strong>[icu]</strong> BasicTimeZone extends <code>TimeZone</code> with additional methods to access
21  * time zone transitions and rules.  All ICU <code>TimeZone</code> concrete subclasses
22  * extend this class. APIs added to <code>java.util.TimeZone</code> by
23  * <code>BasicTimeZone</code> are annotated with <strong>'<font
24  * style="color:red">[icu]</font>'</strong>.
25  *
26  * @see android.icu.util.TimeZoneRule
27  * @see android.icu.util.TimeZoneTransition
28  *
29  * @hide Only a subset of ICU is exposed in Android
30  */
31 @libcore.api.CorePlatformApi
32 public abstract class BasicTimeZone extends TimeZone {
33 
34     private static final long serialVersionUID = -3204278532246180932L;
35 
36     private static final long MILLIS_PER_YEAR = 365*24*60*60*1000L;
37 
38     /**
39      * <strong>[icu]</strong> Returns the first time zone transition after the base time.
40      * <p>Example code:{{@literal @}.jcite  android.icu.samples.util.timezone.BasicTimeZoneExample:---getNextTransitionExample}
41      *
42      * @param base      The base time.
43      * @param inclusive Whether the base time is inclusive or not.
44      *
45      * @return  A <code>Date</code> holding the first time zone transition time
46      *          after the given base time, or null if no time zone transitions
47      *          are available after the base time.
48      */
49     @libcore.api.CorePlatformApi
getNextTransition(long base, boolean inclusive)50     public abstract TimeZoneTransition getNextTransition(long base, boolean inclusive);
51 
52     /**
53      * <strong>[icu]</strong> Returns the last time zone transition before the base time.
54      * <p>Example code:{{@literal @}.jcite  android.icu.samples.util.timezone.BasicTimeZoneExample:---getPreviousTransitionExample}
55      *
56      * @param base      The base time.
57      * @param inclusive Whether the base time is inclusive or not.
58      *
59      * @return  A <code>Date</code> holding the last time zone transition time
60      *          before the given base time, or null if no time zone transitions
61      *          are available before the base time.
62      */
getPreviousTransition(long base, boolean inclusive)63     public abstract TimeZoneTransition getPreviousTransition(long base, boolean inclusive);
64 
65     /**
66      * <strong>[icu]</strong> Checks if the time zone has equivalent transitions in the time range.
67      * This method returns true when all of transition times, from/to standard
68      * offsets and DST savings used by this time zone match the other in the
69      * time range.
70      * <p>Example code:{{@literal @}.jcite  android.icu.samples.util.timezone.BasicTimeZoneExample:---hasEquivalentTransitionsExample}
71      *
72      * @param tz    The instance of <code>TimeZone</code>
73      * @param start The start time of the evaluated time range (inclusive)
74      * @param end   The end time of the evaluated time range (inclusive)
75      *
76      * @return true if the other time zone has the equivalent transitions in the
77      * time range.  When tz is not a <code>BasicTimeZone</code>, this method
78      * returns false.
79      */
hasEquivalentTransitions(TimeZone tz, long start, long end)80     public boolean hasEquivalentTransitions(TimeZone tz, long start, long end) {
81         return hasEquivalentTransitions(tz, start, end, false);
82     }
83 
84     /**
85      * <strong>[icu]</strong> Checks if the time zone has equivalent transitions in the time range.
86      * This method returns true when all of transition times, from/to standard
87      * offsets and DST savings used by this time zone match the other in the
88      * time range.
89      *
90      * @param tz    The instance of <code>TimeZone</code>
91      * @param start The start time of the evaluated time range (inclusive)
92      * @param end   The end time of the evaluated time range (inclusive)
93      * @param ignoreDstAmount When true, any transitions with only daylight saving amount
94      * changes will be ignored, except either of them is zero. For example, a transition
95      * from rawoffset 3:00/dstsavings 1:00 to rawoffset 2:00/dstsavings 2:00 is excluded
96      * from the comparison, but a transtion from rawoffset 2:00/dstsavings 1:00 to
97      * rawoffset 3:00/dstsavings 0:00 is included.
98      *
99      * @return true if the other time zone has the equivalent transitions in the
100      * time range.  When tz is not a <code>BasicTimeZone</code>, this method
101      * returns false.
102      */
hasEquivalentTransitions(TimeZone tz, long start, long end, boolean ignoreDstAmount)103     public boolean hasEquivalentTransitions(TimeZone tz, long start, long end,
104                                             boolean ignoreDstAmount) {
105         if (this == tz) {
106             return true;
107         }
108 
109         if (!(tz instanceof BasicTimeZone)) {
110             return false;
111         }
112 
113         // Check the offsets at the start time
114         int[] offsets1 = new int[2];
115         int[] offsets2 = new int[2];
116 
117         getOffset(start, false, offsets1);
118         tz.getOffset(start, false, offsets2);
119 
120         if (ignoreDstAmount) {
121             if ((offsets1[0] + offsets1[1] != offsets2[0] + offsets2[1])
122                 || (offsets1[1] != 0 && offsets2[1] == 0)
123                 || (offsets1[1] == 0 && offsets2[1] != 0)) {
124                 return false;
125             }
126         } else {
127             if (offsets1[0] != offsets2[0] || offsets1[1] != offsets2[1]) {
128                 return false;
129             }
130         }
131 
132         // Check transitions in the range
133         long time = start;
134         while (true) {
135             TimeZoneTransition tr1 = getNextTransition(time, false);
136             TimeZoneTransition tr2 = ((BasicTimeZone)tz).getNextTransition(time, false);
137 
138             if (ignoreDstAmount) {
139                 // Skip a transition which only differ the amount of DST savings
140                 while (true) {
141                     if (tr1 != null
142                             && tr1.getTime() <= end
143                             && (tr1.getFrom().getRawOffset() + tr1.getFrom().getDSTSavings()
144                                     == tr1.getTo().getRawOffset() + tr1.getTo().getDSTSavings())
145                             && (tr1.getFrom().getDSTSavings() != 0 && tr1.getTo().getDSTSavings() != 0)) {
146                         tr1 = getNextTransition(tr1.getTime(), false);
147                     } else {
148                         break;
149                     }
150                 }
151                 while (true) {
152                     if (tr2 != null
153                             && tr2.getTime() <= end
154                             && (tr2.getFrom().getRawOffset() + tr2.getFrom().getDSTSavings()
155                                     == tr2.getTo().getRawOffset() + tr2.getTo().getDSTSavings())
156                             && (tr2.getFrom().getDSTSavings() != 0 && tr2.getTo().getDSTSavings() != 0)) {
157                         tr2 = ((BasicTimeZone)tz).getNextTransition(tr2.getTime(), false);
158                     } else {
159                         break;
160                     }
161                 }
162             }
163 
164             boolean inRange1 = false;
165             boolean inRange2 = false;
166             if (tr1 != null) {
167                 if (tr1.getTime() <= end) {
168                     inRange1 = true;
169                 }
170             }
171             if (tr2 != null) {
172                 if (tr2.getTime() <= end) {
173                     inRange2 = true;
174                 }
175             }
176             if (!inRange1 && !inRange2) {
177                 // No more transition in the range
178                 break;
179             }
180             if (!inRange1 || !inRange2) {
181                 return false;
182             }
183             if (tr1.getTime() != tr2.getTime()) {
184                 return false;
185             }
186             if (ignoreDstAmount) {
187                 if (tr1.getTo().getRawOffset() + tr1.getTo().getDSTSavings()
188                             != tr2.getTo().getRawOffset() + tr2.getTo().getDSTSavings()
189                         || tr1.getTo().getDSTSavings() != 0 &&  tr2.getTo().getDSTSavings() == 0
190                         || tr1.getTo().getDSTSavings() == 0 &&  tr2.getTo().getDSTSavings() != 0) {
191                     return false;
192                 }
193             } else {
194                 if (tr1.getTo().getRawOffset() != tr2.getTo().getRawOffset() ||
195                     tr1.getTo().getDSTSavings() != tr2.getTo().getDSTSavings()) {
196                     return false;
197                 }
198             }
199             time = tr1.getTime();
200         }
201         return true;
202     }
203 
204     /**
205      * <strong>[icu]</strong> Returns the array of <code>TimeZoneRule</code> which represents the rule
206      * of this time zone object.  The first element in the result array will
207      * be the <code>InitialTimeZoneRule</code> instance for the initial rule.
208      * The rest will be either <code>AnnualTimeZoneRule</code> or
209      * <code>TimeArrayTimeZoneRule</code> instances representing transitions.
210      *
211      * @return  The array of <code>TimeZoneRule</code> which represents this
212      *          time zone.
213      */
getTimeZoneRules()214     public abstract TimeZoneRule[] getTimeZoneRules();
215 
216     /**
217      * <strong>[icu]</strong> Returns the array of <code>TimeZoneRule</code> which represents the rule
218      * of this time zone object since the specified start time.  The first
219      * element in the result array will be the <code>InitialTimeZoneRule</code>
220      * instance for the initial rule.  The rest will be either
221      * <code>AnnualTimeZoneRule</code> or <code>TimeArrayTimeZoneRule</code>
222      * instances representing transitions.
223      * <p>Example code:{{@literal @}.jcite  android.icu.samples.util.timezone.BasicTimeZoneExample:---getTimeZoneRulesExample}
224      *
225      * @param start The start time (inclusive).
226      * @return  The array of <code>TimeZoneRule</code> which represents this
227      *          time zone since the start time.
228      */
getTimeZoneRules(long start)229     public TimeZoneRule[] getTimeZoneRules(long start) {
230         TimeZoneRule[] all = getTimeZoneRules();
231         TimeZoneTransition tzt = getPreviousTransition(start, true);
232         if (tzt == null) {
233             // No need to filter out rules only applicable to time before the start
234             return all;
235         }
236 
237         BitSet isProcessed = new BitSet(all.length);
238         List<TimeZoneRule> filteredRules = new LinkedList<TimeZoneRule>();
239 
240         // Create initial rule
241         TimeZoneRule initial = new InitialTimeZoneRule(tzt.getTo().getName(),
242                 tzt.getTo().getRawOffset(), tzt.getTo().getDSTSavings());
243         filteredRules.add(initial);
244         isProcessed.set(0);
245 
246         // Mark rules which does not need to be processed
247         for (int i = 1; i < all.length; i++) {
248             Date d = all[i].getNextStart(start, initial.getRawOffset(),
249                     initial.getDSTSavings(), false);
250             if (d == null) {
251                 isProcessed.set(i);
252             }
253         }
254 
255         long time = start;
256         boolean bFinalStd = false, bFinalDst = false;
257         while(!bFinalStd || !bFinalDst) {
258             tzt = getNextTransition(time, false);
259             if (tzt == null) {
260                 break;
261             }
262             time = tzt.getTime();
263 
264             TimeZoneRule toRule = tzt.getTo();
265             int ruleIdx = 1;
266             for (; ruleIdx < all.length; ruleIdx++) {
267                 if (all[ruleIdx].equals(toRule)) {
268                     break;
269                 }
270             }
271             if (ruleIdx >= all.length) {
272                 throw new IllegalStateException("The rule was not found");
273             }
274             if (isProcessed.get(ruleIdx)) {
275                 continue;
276             }
277             if (toRule instanceof TimeArrayTimeZoneRule) {
278                 TimeArrayTimeZoneRule tar = (TimeArrayTimeZoneRule)toRule;
279 
280                 // Get the previous raw offset and DST savings before the very first start time
281                 long t = start;
282                 while(true) {
283                     tzt = getNextTransition(t, false);
284                     if (tzt == null) {
285                         break;
286                     }
287                     if (tzt.getTo().equals(tar)) {
288                         break;
289                     }
290                     t = tzt.getTime();
291                 }
292                 if (tzt != null) {
293                     // Check if the entire start times to be added
294                     Date firstStart = tar.getFirstStart(tzt.getFrom().getRawOffset(),
295                             tzt.getFrom().getDSTSavings());
296                     if (firstStart.getTime() > start) {
297                         // Just add the rule as is
298                         filteredRules.add(tar);
299                     } else {
300                         // Collect transitions after the start time
301                         long[] times = tar.getStartTimes();
302                         int timeType = tar.getTimeType();
303                         int idx;
304                         for (idx = 0; idx < times.length; idx++) {
305                             t = times[idx];
306                             if (timeType == DateTimeRule.STANDARD_TIME) {
307                                 t -= tzt.getFrom().getRawOffset();
308                             }
309                             if (timeType == DateTimeRule.WALL_TIME) {
310                                 t -= tzt.getFrom().getDSTSavings();
311                             }
312                             if (t > start) {
313                                 break;
314                             }
315                         }
316                         int asize = times.length - idx;
317                         if (asize > 0) {
318                             long[] newtimes = new long[asize];
319                             System.arraycopy(times, idx, newtimes, 0, asize);
320                             TimeArrayTimeZoneRule newtar = new TimeArrayTimeZoneRule(
321                                     tar.getName(), tar.getRawOffset(), tar.getDSTSavings(),
322                                     newtimes, tar.getTimeType());
323                             filteredRules.add(newtar);
324                         }
325                     }
326                 }
327             } else if (toRule instanceof AnnualTimeZoneRule) {
328                 AnnualTimeZoneRule ar = (AnnualTimeZoneRule)toRule;
329                 Date firstStart = ar.getFirstStart(tzt.getFrom().getRawOffset(),
330                         tzt.getFrom().getDSTSavings());
331                 if (firstStart.getTime() == tzt.getTime()) {
332                     // Just add the rule as is
333                     filteredRules.add(ar);
334                 } else {
335                     // Calculate the transition year
336                     int[] dfields = new int[6];
337                     Grego.timeToFields(tzt.getTime(), dfields);
338                     // Recreate the rule
339                     AnnualTimeZoneRule newar = new AnnualTimeZoneRule(ar.getName(),
340                             ar.getRawOffset(), ar.getDSTSavings(),
341                             ar.getRule(), dfields[0], ar.getEndYear());
342                     filteredRules.add(newar);
343                 }
344                 // Check if this is a final rule
345                 if (ar.getEndYear() == AnnualTimeZoneRule.MAX_YEAR) {
346                     // After both final standard and dst rule are processed,
347                     // exit this while loop.
348                     if (ar.getDSTSavings() == 0) {
349                         bFinalStd = true;
350                     } else {
351                         bFinalDst = true;
352                     }
353                 }
354             }
355             isProcessed.set(ruleIdx);
356         }
357         TimeZoneRule[] rules = filteredRules.toArray(new TimeZoneRule[filteredRules.size()]);
358         return rules;
359     }
360 
361     /**
362      * <strong>[icu]</strong> Returns the array of <code>TimeZoneRule</code> which represents the rule of
363      * this time zone object near the specified date.  Some applications are not
364      * capable to handle historic time zone rule changes.  Also some applications
365      * can only handle certain type of rule definitions.  This method returns
366      * either a single <code>InitialTimeZoneRule</code> if this time zone does not
367      * have any daylight saving time within 1 year from the specified time, or a
368      * pair of <code>AnnualTimeZoneRule</code> whose rule type is
369      * <code>DateTimeRule.DOW</code> for date and <code>DateTimeRule.WALL_TIME</code>
370      * for time with a single <code>InitialTimeZoneRule</code> representing the
371      * initial time, when this time zone observes daylight saving time near the
372      * specified date.  Thus, the result may be only valid for dates around the
373      * specified date.
374      *
375      * @param date The date to be used for <code>TimeZoneRule</code> extraction.
376      * @return The array of <code>TimeZoneRule</code>, either a single
377      * <code>InitialTimeZoneRule</code> object, or a pair of <code>AnnualTimeZoneRule</code>
378      * with a single <code>InitialTimeZoneRule</code>.  The first element in the
379      * array is always a <code>InitialTimeZoneRule</code>.
380      */
getSimpleTimeZoneRulesNear(long date)381     public TimeZoneRule[] getSimpleTimeZoneRulesNear(long date) {
382         AnnualTimeZoneRule[] annualRules = null;
383         TimeZoneRule initialRule = null;
384         // Get the next transition
385         TimeZoneTransition tr = getNextTransition(date, false);
386         if (tr != null) {
387             String initialName = tr.getFrom().getName();
388             int initialRaw = tr.getFrom().getRawOffset();
389             int initialDst = tr.getFrom().getDSTSavings();
390 
391             // Check if the next transition is either DST->STD or STD->DST and
392             // within roughly 1 year from the specified date
393             long nextTransitionTime = tr.getTime();
394             if (((tr.getFrom().getDSTSavings() == 0 && tr.getTo().getDSTSavings() != 0)
395                     || (tr.getFrom().getDSTSavings() != 0 && tr.getTo().getDSTSavings() == 0))
396                         && date + MILLIS_PER_YEAR > nextTransitionTime) {
397                 annualRules = new AnnualTimeZoneRule[2];
398                 // Get local wall time for the transition time
399                 int dtfields[] = Grego.timeToFields(nextTransitionTime
400                         + tr.getFrom().getRawOffset() + tr.getFrom().getDSTSavings(), null);
401                 int weekInMonth = Grego.getDayOfWeekInMonth(dtfields[0], dtfields[1], dtfields[2]);
402                 // Create DOW rule
403                 DateTimeRule dtr = new DateTimeRule(dtfields[1], weekInMonth, dtfields[3],
404                         dtfields[5], DateTimeRule.WALL_TIME);
405 
406                 AnnualTimeZoneRule secondRule = null;
407 
408                 // Note:  SimpleTimeZone does not support raw offset change.
409                 // So we always use raw offset of the given time for the rule,
410                 // even raw offset is changed.  This will result that the result
411                 // zone to return wrong offset after the transition.
412                 // When we encounter such case, we do not inspect next next
413                 // transition for another rule.
414                 annualRules[0] = new AnnualTimeZoneRule(tr.getTo().getName(),
415                         initialRaw, tr.getTo().getDSTSavings(),
416                         dtr, dtfields[0], AnnualTimeZoneRule.MAX_YEAR);
417 
418                 if (tr.getTo().getRawOffset() == initialRaw) {
419 
420                     // Get the next next transition
421                     tr = getNextTransition(nextTransitionTime, false);
422                     if (tr != null) {
423                         // Check if the next next transition is either DST->STD or STD->DST
424                         // and within roughly 1 year from the next transition
425                         if (((tr.getFrom().getDSTSavings() == 0 && tr.getTo().getDSTSavings() != 0)
426                                 || (tr.getFrom().getDSTSavings() != 0
427                                     && tr.getTo().getDSTSavings() == 0))
428                             && nextTransitionTime + MILLIS_PER_YEAR > tr.getTime()) {
429                             // Generate another DOW rule
430                             dtfields = Grego.timeToFields(tr.getTime()
431                                     + tr.getFrom().getRawOffset() + tr.getFrom().getDSTSavings(),
432                                                           dtfields);
433                             weekInMonth = Grego.getDayOfWeekInMonth(dtfields[0], dtfields[1],
434                                                                     dtfields[2]);
435                             dtr = new DateTimeRule(dtfields[1], weekInMonth, dtfields[3],
436                                                    dtfields[5], DateTimeRule.WALL_TIME);
437                             secondRule = new AnnualTimeZoneRule(tr.getTo().getName(),
438                                     tr.getTo().getRawOffset(), tr.getTo().getDSTSavings(),
439                                     dtr, dtfields[0] - 1, AnnualTimeZoneRule.MAX_YEAR);
440                             // Make sure this rule can be applied to the specified date
441                             Date d = secondRule.getPreviousStart(date, tr.getFrom().getRawOffset(),
442                                     tr.getFrom().getDSTSavings(), true);
443                             if (d != null && d.getTime() <= date
444                                     && initialRaw == tr.getTo().getRawOffset()
445                                     && initialDst == tr.getTo().getDSTSavings()) {
446                                 // We can use this rule as the second transition rule
447                                 annualRules[1] = secondRule;
448                             }
449                         }
450                     }
451                 }
452 
453                 if (annualRules[1] == null) {
454                     // Try previous transition
455                     tr = getPreviousTransition(date, true);
456                     if (tr != null) {
457                         // Check if the previous transition is either DST->STD or STD->DST.
458                         // The actual transition time does not matter here.
459                         if ((tr.getFrom().getDSTSavings() == 0 && tr.getTo().getDSTSavings() != 0)
460                                 || (tr.getFrom().getDSTSavings() != 0
461                                     && tr.getTo().getDSTSavings() == 0)) {
462                             // Generate another DOW rule
463                             dtfields = Grego.timeToFields(tr.getTime()
464                                     + tr.getFrom().getRawOffset() + tr.getFrom().getDSTSavings(),
465                                                           dtfields);
466                             weekInMonth = Grego.getDayOfWeekInMonth(dtfields[0], dtfields[1],
467                                                                     dtfields[2]);
468                             dtr = new DateTimeRule(dtfields[1], weekInMonth, dtfields[3],
469                                                    dtfields[5], DateTimeRule.WALL_TIME);
470 
471                             // second rule raw/dst offsets should match raw/dst offsets
472                             // at the given time
473                             secondRule = new AnnualTimeZoneRule(
474                                 tr.getTo().getName(), initialRaw, initialDst, dtr,
475                                 annualRules[0].getStartYear() - 1, AnnualTimeZoneRule.MAX_YEAR);
476 
477                             // Check if this rule start after the first rule after the
478                             // specified date
479                             Date d = secondRule.getNextStart(date, tr.getFrom().getRawOffset(),
480                                                              tr.getFrom().getDSTSavings(), false);
481                             if (d.getTime() > nextTransitionTime) {
482                                 // We can use this rule as the second transition rule
483                                 annualRules[1] = secondRule;
484                             }
485                         }
486                     }
487                 }
488                 if (annualRules[1] == null) {
489                     // Cannot generate a good pair of AnnualTimeZoneRule
490                     annualRules = null;
491                 } else {
492                     // The initial rule should represent the rule before the previous transition
493                     initialName = annualRules[0].getName();
494                     initialRaw = annualRules[0].getRawOffset();
495                     initialDst = annualRules[0].getDSTSavings();
496                 }
497             }
498             initialRule = new InitialTimeZoneRule(initialName, initialRaw, initialDst);
499         } else {
500             // Try the previous one
501             tr = getPreviousTransition(date, true);
502             if (tr != null) {
503                 initialRule = new InitialTimeZoneRule(tr.getTo().getName(),
504                         tr.getTo().getRawOffset(), tr.getTo().getDSTSavings());
505             } else {
506                 // No transitions in the past.  Just use the current offsets
507                 int[] offsets = new int[2];
508                 getOffset(date, false, offsets);
509                 initialRule = new InitialTimeZoneRule(getID(), offsets[0], offsets[1]);
510             }
511         }
512 
513         TimeZoneRule[] result = null;
514         if (annualRules == null) {
515             result = new TimeZoneRule[1];
516             result[0] = initialRule;
517         } else {
518             result = new TimeZoneRule[3];
519             result[0] = initialRule;
520             result[1] = annualRules[0];
521             result[2] = annualRules[1];
522         }
523 
524         return result;
525     }
526 
527     /**
528      * <strong>[icu]</strong> The time type option for standard time used by
529      * {@link #getOffsetFromLocal(long, int, int, int[])}
530      * @deprecated This API is ICU internal only.
531      * @hide draft / provisional / internal are hidden on Android
532      */
533     @Deprecated
534     public static final int LOCAL_STD = 0x01;
535 
536     /**
537      * <strong>[icu]</strong> The time type option for daylight saving time used by
538      * {@link #getOffsetFromLocal(long, int, int, int[])}
539      * @deprecated This API is ICU internal only.
540      * @hide draft / provisional / internal are hidden on Android
541      */
542     @Deprecated
543     public static final int LOCAL_DST = 0x03;
544 
545     /**
546      * <strong>[icu]</strong> The option designate former time to be used by
547      * {@link #getOffsetFromLocal(long, int, int, int[])}
548      * @deprecated This API is ICU internal only.
549      * @hide draft / provisional / internal are hidden on Android
550      */
551     @Deprecated
552     public static final int LOCAL_FORMER = 0x04;
553 
554     /**
555      * <strong>[icu]</strong> The option designate latter time to be used by
556      * {@link #getOffsetFromLocal(long, int, int, int[])}
557      * @deprecated This API is ICU internal only.
558      * @hide draft / provisional / internal are hidden on Android
559      */
560     @Deprecated
561     public static final int LOCAL_LATTER = 0x0C;
562 
563     /**
564      * <strong>[icu]</strong> The bit mask for the time type option used by
565      * {@link #getOffsetFromLocal(long, int, int, int[])}
566      * @deprecated This API is ICU internal only.
567      * @hide draft / provisional / internal are hidden on Android
568      */
569     @Deprecated
570     protected static final int STD_DST_MASK = 0x03;
571 
572     /**
573      * <strong>[icu]</strong> The bit mask for the former/latter option used by
574      * {@link #getOffsetFromLocal(long, int, int, int[])}
575      * @deprecated This API is ICU internal only.
576      * @hide draft / provisional / internal are hidden on Android
577      */
578     @Deprecated
579     protected static final int FORMER_LATTER_MASK = 0x0C;
580 
581     /**
582      * <strong>[icu]</strong> Returns time zone offsets from local wall time.
583      * @deprecated This API is ICU internal only.
584      * @hide draft / provisional / internal are hidden on Android
585      */
586     @Deprecated
getOffsetFromLocal(long date, int nonExistingTimeOpt, int duplicatedTimeOpt, int[] offsets)587     public void getOffsetFromLocal(long date,
588             int nonExistingTimeOpt, int duplicatedTimeOpt, int[] offsets) {
589         throw new IllegalStateException("Not implemented");
590     }
591 
592     /**
593      * Protected no arg constructor.
594      */
BasicTimeZone()595     protected BasicTimeZone() {
596     }
597 
598     /**
599      * Constructing a BasicTimeZone with the given time zone ID.
600      * @param ID the time zone ID.
601      * @deprecated This API is ICU internal only.
602      * @hide draft / provisional / internal are hidden on Android
603      */
604     @Deprecated
BasicTimeZone(String ID)605     protected BasicTimeZone(String ID) {
606         super(ID);
607     }
608 }
609