Home | History | Annotate | Download | only in util
      1 /*
      2  * Copyright (C) 2006 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package android.util;
     18 
     19 import android.content.res.Resources;
     20 import android.content.res.XmlResourceParser;
     21 import android.os.SystemClock;
     22 import android.text.format.DateUtils;
     23 
     24 import com.android.internal.util.XmlUtils;
     25 
     26 import org.xmlpull.v1.XmlPullParser;
     27 import org.xmlpull.v1.XmlPullParserException;
     28 
     29 import java.io.IOException;
     30 import java.io.PrintWriter;
     31 import java.util.ArrayList;
     32 import java.util.Calendar;
     33 import java.util.Collection;
     34 import java.util.Date;
     35 import java.util.TimeZone;
     36 
     37 import libcore.util.ZoneInfoDB;
     38 
     39 /**
     40  * A class containing utility methods related to time zones.
     41  */
     42 public class TimeUtils {
     43     /** @hide */ public TimeUtils() {}
     44     private static final boolean DBG = false;
     45     private static final String TAG = "TimeUtils";
     46 
     47     /** Cached results of getTineZones */
     48     private static final Object sLastLockObj = new Object();
     49     private static ArrayList<TimeZone> sLastZones = null;
     50     private static String sLastCountry = null;
     51 
     52     /** Cached results of getTimeZonesWithUniqueOffsets */
     53     private static final Object sLastUniqueLockObj = new Object();
     54     private static ArrayList<TimeZone> sLastUniqueZoneOffsets = null;
     55     private static String sLastUniqueCountry = null;
     56 
     57 
     58     /**
     59      * Tries to return a time zone that would have had the specified offset
     60      * and DST value at the specified moment in the specified country.
     61      * Returns null if no suitable zone could be found.
     62      */
     63     public static TimeZone getTimeZone(int offset, boolean dst, long when, String country) {
     64         TimeZone best = null;
     65         final Date d = new Date(when);
     66 
     67         TimeZone current = TimeZone.getDefault();
     68         String currentName = current.getID();
     69         int currentOffset = current.getOffset(when);
     70         boolean currentDst = current.inDaylightTime(d);
     71 
     72         for (TimeZone tz : getTimeZones(country)) {
     73             // If the current time zone is from the right country
     74             // and meets the other known properties, keep it
     75             // instead of changing to another one.
     76 
     77             if (tz.getID().equals(currentName)) {
     78                 if (currentOffset == offset && currentDst == dst) {
     79                     return current;
     80                 }
     81             }
     82 
     83             // Otherwise, take the first zone from the right
     84             // country that has the correct current offset and DST.
     85             // (Keep iterating instead of returning in case we
     86             // haven't encountered the current time zone yet.)
     87 
     88             if (best == null) {
     89                 if (tz.getOffset(when) == offset &&
     90                     tz.inDaylightTime(d) == dst) {
     91                     best = tz;
     92                 }
     93             }
     94         }
     95 
     96         return best;
     97     }
     98 
     99     /**
    100      * Return list of unique time zones for the country. Do not modify
    101      *
    102      * @param country to find
    103      * @return list of unique time zones, maybe empty but never null. Do not modify.
    104      * @hide
    105      */
    106     public static ArrayList<TimeZone> getTimeZonesWithUniqueOffsets(String country) {
    107         synchronized(sLastUniqueLockObj) {
    108             if ((country != null) && country.equals(sLastUniqueCountry)) {
    109                 if (DBG) {
    110                     Log.d(TAG, "getTimeZonesWithUniqueOffsets(" +
    111                             country + "): return cached version");
    112                 }
    113                 return sLastUniqueZoneOffsets;
    114             }
    115         }
    116 
    117         Collection<TimeZone> zones = getTimeZones(country);
    118         ArrayList<TimeZone> uniqueTimeZones = new ArrayList<TimeZone>();
    119         for (TimeZone zone : zones) {
    120             // See if we already have this offset,
    121             // Using slow but space efficient and these are small.
    122             boolean found = false;
    123             for (int i = 0; i < uniqueTimeZones.size(); i++) {
    124                 if (uniqueTimeZones.get(i).getRawOffset() == zone.getRawOffset()) {
    125                     found = true;
    126                     break;
    127                 }
    128             }
    129             if (found == false) {
    130                 if (DBG) {
    131                     Log.d(TAG, "getTimeZonesWithUniqueOffsets: add unique offset=" +
    132                             zone.getRawOffset() + " zone.getID=" + zone.getID());
    133                 }
    134                 uniqueTimeZones.add(zone);
    135             }
    136         }
    137 
    138         synchronized(sLastUniqueLockObj) {
    139             // Cache the last result
    140             sLastUniqueZoneOffsets = uniqueTimeZones;
    141             sLastUniqueCountry = country;
    142 
    143             return sLastUniqueZoneOffsets;
    144         }
    145     }
    146 
    147     /**
    148      * Returns the time zones for the country, which is the code
    149      * attribute of the timezone element in time_zones_by_country.xml. Do not modify.
    150      *
    151      * @param country is a two character country code.
    152      * @return TimeZone list, maybe empty but never null. Do not modify.
    153      * @hide
    154      */
    155     public static ArrayList<TimeZone> getTimeZones(String country) {
    156         synchronized (sLastLockObj) {
    157             if ((country != null) && country.equals(sLastCountry)) {
    158                 if (DBG) Log.d(TAG, "getTimeZones(" + country + "): return cached version");
    159                 return sLastZones;
    160             }
    161         }
    162 
    163         ArrayList<TimeZone> tzs = new ArrayList<TimeZone>();
    164 
    165         if (country == null) {
    166             if (DBG) Log.d(TAG, "getTimeZones(null): return empty list");
    167             return tzs;
    168         }
    169 
    170         Resources r = Resources.getSystem();
    171         XmlResourceParser parser = r.getXml(com.android.internal.R.xml.time_zones_by_country);
    172 
    173         try {
    174             XmlUtils.beginDocument(parser, "timezones");
    175 
    176             while (true) {
    177                 XmlUtils.nextElement(parser);
    178 
    179                 String element = parser.getName();
    180                 if (element == null || !(element.equals("timezone"))) {
    181                     break;
    182                 }
    183 
    184                 String code = parser.getAttributeValue(null, "code");
    185 
    186                 if (country.equals(code)) {
    187                     if (parser.next() == XmlPullParser.TEXT) {
    188                         String zoneIdString = parser.getText();
    189                         TimeZone tz = TimeZone.getTimeZone(zoneIdString);
    190                         if (tz.getID().startsWith("GMT") == false) {
    191                             // tz.getID doesn't start not "GMT" so its valid
    192                             tzs.add(tz);
    193                             if (DBG) {
    194                                 Log.d(TAG, "getTimeZone('" + country + "'): found tz.getID=="
    195                                     + ((tz != null) ? tz.getID() : "<no tz>"));
    196                             }
    197                         }
    198                     }
    199                 }
    200             }
    201         } catch (XmlPullParserException e) {
    202             Log.e(TAG, "Got xml parser exception getTimeZone('" + country + "'): e=", e);
    203         } catch (IOException e) {
    204             Log.e(TAG, "Got IO exception getTimeZone('" + country + "'): e=", e);
    205         } finally {
    206             parser.close();
    207         }
    208 
    209         synchronized(sLastLockObj) {
    210             // Cache the last result;
    211             sLastZones = tzs;
    212             sLastCountry = country;
    213             return sLastZones;
    214         }
    215     }
    216 
    217     /**
    218      * Returns a String indicating the version of the time zone database currently
    219      * in use.  The format of the string is dependent on the underlying time zone
    220      * database implementation, but will typically contain the year in which the database
    221      * was updated plus a letter from a to z indicating changes made within that year.
    222      *
    223      * <p>Time zone database updates should be expected to occur periodically due to
    224      * political and legal changes that cannot be anticipated in advance.  Therefore,
    225      * when computing the UTC time for a future event, applications should be aware that
    226      * the results may differ following a time zone database update.  This method allows
    227      * applications to detect that a database change has occurred, and to recalculate any
    228      * cached times accordingly.
    229      *
    230      * <p>The time zone database may be assumed to change only when the device runtime
    231      * is restarted.  Therefore, it is not necessary to re-query the database version
    232      * during the lifetime of an activity.
    233      */
    234     public static String getTimeZoneDatabaseVersion() {
    235         return ZoneInfoDB.getInstance().getVersion();
    236     }
    237 
    238     /** @hide Field length that can hold 999 days of time */
    239     public static final int HUNDRED_DAY_FIELD_LEN = 19;
    240 
    241     private static final int SECONDS_PER_MINUTE = 60;
    242     private static final int SECONDS_PER_HOUR = 60 * 60;
    243     private static final int SECONDS_PER_DAY = 24 * 60 * 60;
    244 
    245     /** @hide */
    246     public static final long NANOS_PER_MS = 1000000;
    247 
    248     private static final Object sFormatSync = new Object();
    249     private static char[] sFormatStr = new char[HUNDRED_DAY_FIELD_LEN+5];
    250 
    251     private static final long LARGEST_DURATION = (1000 * DateUtils.DAY_IN_MILLIS) - 1;
    252 
    253     static private int accumField(int amt, int suffix, boolean always, int zeropad) {
    254         if (amt > 99 || (always && zeropad >= 3)) {
    255             return 3+suffix;
    256         }
    257         if (amt > 9 || (always && zeropad >= 2)) {
    258             return 2+suffix;
    259         }
    260         if (always || amt > 0) {
    261             return 1+suffix;
    262         }
    263         return 0;
    264     }
    265 
    266     static private int printField(char[] formatStr, int amt, char suffix, int pos,
    267             boolean always, int zeropad) {
    268         if (always || amt > 0) {
    269             final int startPos = pos;
    270             if ((always && zeropad >= 3) || amt > 99) {
    271                 int dig = amt/100;
    272                 formatStr[pos] = (char)(dig + '0');
    273                 pos++;
    274                 amt -= (dig*100);
    275             }
    276             if ((always && zeropad >= 2) || amt > 9 || startPos != pos) {
    277                 int dig = amt/10;
    278                 formatStr[pos] = (char)(dig + '0');
    279                 pos++;
    280                 amt -= (dig*10);
    281             }
    282             formatStr[pos] = (char)(amt + '0');
    283             pos++;
    284             formatStr[pos] = suffix;
    285             pos++;
    286         }
    287         return pos;
    288     }
    289 
    290     private static int formatDurationLocked(long duration, int fieldLen) {
    291         if (sFormatStr.length < fieldLen) {
    292             sFormatStr = new char[fieldLen];
    293         }
    294 
    295         char[] formatStr = sFormatStr;
    296 
    297         if (duration == 0) {
    298             int pos = 0;
    299             fieldLen -= 1;
    300             while (pos < fieldLen) {
    301                 formatStr[pos++] = ' ';
    302             }
    303             formatStr[pos] = '0';
    304             return pos+1;
    305         }
    306 
    307         char prefix;
    308         if (duration > 0) {
    309             prefix = '+';
    310         } else {
    311             prefix = '-';
    312             duration = -duration;
    313         }
    314 
    315         if (duration > LARGEST_DURATION) {
    316             duration = LARGEST_DURATION;
    317         }
    318 
    319         int millis = (int)(duration%1000);
    320         int seconds = (int) Math.floor(duration / 1000);
    321         int days = 0, hours = 0, minutes = 0;
    322 
    323         if (seconds > SECONDS_PER_DAY) {
    324             days = seconds / SECONDS_PER_DAY;
    325             seconds -= days * SECONDS_PER_DAY;
    326         }
    327         if (seconds > SECONDS_PER_HOUR) {
    328             hours = seconds / SECONDS_PER_HOUR;
    329             seconds -= hours * SECONDS_PER_HOUR;
    330         }
    331         if (seconds > SECONDS_PER_MINUTE) {
    332             minutes = seconds / SECONDS_PER_MINUTE;
    333             seconds -= minutes * SECONDS_PER_MINUTE;
    334         }
    335 
    336         int pos = 0;
    337 
    338         if (fieldLen != 0) {
    339             int myLen = accumField(days, 1, false, 0);
    340             myLen += accumField(hours, 1, myLen > 0, 2);
    341             myLen += accumField(minutes, 1, myLen > 0, 2);
    342             myLen += accumField(seconds, 1, myLen > 0, 2);
    343             myLen += accumField(millis, 2, true, myLen > 0 ? 3 : 0) + 1;
    344             while (myLen < fieldLen) {
    345                 formatStr[pos] = ' ';
    346                 pos++;
    347                 myLen++;
    348             }
    349         }
    350 
    351         formatStr[pos] = prefix;
    352         pos++;
    353 
    354         int start = pos;
    355         boolean zeropad = fieldLen != 0;
    356         pos = printField(formatStr, days, 'd', pos, false, 0);
    357         pos = printField(formatStr, hours, 'h', pos, pos != start, zeropad ? 2 : 0);
    358         pos = printField(formatStr, minutes, 'm', pos, pos != start, zeropad ? 2 : 0);
    359         pos = printField(formatStr, seconds, 's', pos, pos != start, zeropad ? 2 : 0);
    360         pos = printField(formatStr, millis, 'm', pos, true, (zeropad && pos != start) ? 3 : 0);
    361         formatStr[pos] = 's';
    362         return pos + 1;
    363     }
    364 
    365     /** @hide Just for debugging; not internationalized. */
    366     public static void formatDuration(long duration, StringBuilder builder) {
    367         synchronized (sFormatSync) {
    368             int len = formatDurationLocked(duration, 0);
    369             builder.append(sFormatStr, 0, len);
    370         }
    371     }
    372 
    373     /** @hide Just for debugging; not internationalized. */
    374     public static void formatDuration(long duration, PrintWriter pw, int fieldLen) {
    375         synchronized (sFormatSync) {
    376             int len = formatDurationLocked(duration, fieldLen);
    377             pw.print(new String(sFormatStr, 0, len));
    378         }
    379     }
    380 
    381     /** @hide Just for debugging; not internationalized. */
    382     public static void formatDuration(long duration, PrintWriter pw) {
    383         formatDuration(duration, pw, 0);
    384     }
    385 
    386     /** @hide Just for debugging; not internationalized. */
    387     public static void formatDuration(long time, long now, PrintWriter pw) {
    388         if (time == 0) {
    389             pw.print("--");
    390             return;
    391         }
    392         formatDuration(time-now, pw, 0);
    393     }
    394 
    395     /** @hide Just for debugging; not internationalized. */
    396     public static String formatUptime(long time) {
    397         final long diff = time - SystemClock.uptimeMillis();
    398         if (diff > 0) {
    399             return time + " (in " + diff + " ms)";
    400         }
    401         if (diff < 0) {
    402             return time + " (" + -diff + " ms ago)";
    403         }
    404         return time + " (now)";
    405     }
    406 
    407     /**
    408      * Convert a System.currentTimeMillis() value to a time of day value like
    409      * that printed in logs. MM-DD HH:MM:SS.MMM
    410      *
    411      * @param millis since the epoch (1/1/1970)
    412      * @return String representation of the time.
    413      * @hide
    414      */
    415     public static String logTimeOfDay(long millis) {
    416         Calendar c = Calendar.getInstance();
    417         if (millis >= 0) {
    418             c.setTimeInMillis(millis);
    419             return String.format("%tm-%td %tH:%tM:%tS.%tL", c, c, c, c, c, c);
    420         } else {
    421             return Long.toString(millis);
    422         }
    423     }
    424 }
    425