/* * $Id: DateTime.java,v 1.12 2006/07/23 08:21:23 fortuna Exp $ * * Created on 26/06/2005 * * Copyright (c) 2005, Ben Fortuna * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * o Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * o Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * o Neither the name of Ben Fortuna nor the names of any other contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package net.fortuna.ical4j.model; import java.text.DateFormat; import java.text.ParseException; import java.text.SimpleDateFormat; import net.fortuna.ical4j.util.Dates; import net.fortuna.ical4j.util.TimeZones; /** * Represents a time of day on a specific date. * * @author Ben Fortuna */ public class DateTime extends Date { private static final long serialVersionUID = -6407231357919440387L; private static final String DEFAULT_PATTERN = "yyyyMMdd'T'HHmmss"; private static final String UTC_PATTERN = "yyyyMMdd'T'HHmmss'Z'"; /** * Used for parsing times in a UTC date-time representation. */ private static final DateFormat UTC_FORMAT = new SimpleDateFormat(UTC_PATTERN); static { UTC_FORMAT.setTimeZone(TimeZone.getTimeZone(TimeZones.UTC_ID)); UTC_FORMAT.setLenient(false); } /** * Used for parsing times in a local date-time representation. */ private static final DateFormat DEFAULT_FORMAT = new SimpleDateFormat(DEFAULT_PATTERN); static { DEFAULT_FORMAT.setLenient(false); } private static final DateFormat LENIENT_DEFAULT_FORMAT = new SimpleDateFormat(DEFAULT_PATTERN); private Time time; private TimeZone timezone; /** * Default constructor. */ public DateTime() { super(Dates.PRECISION_SECOND); this.time = new Time(System.currentTimeMillis(), getFormat().getTimeZone()); } /** * @param utc */ public DateTime(final boolean utc) { this(); setUtc(utc); } /** * @param time */ public DateTime(final long time) { super(time, Dates.PRECISION_SECOND); this.time = new Time(time, getFormat().getTimeZone()); } /** * @param date */ public DateTime(final java.util.Date date) { super(date.getTime(), Dates.PRECISION_SECOND); this.time = new Time(date.getTime(), getFormat().getTimeZone()); // copy timezone information if applicable.. if (date instanceof DateTime) { DateTime dateTime = (DateTime) date; if (dateTime.isUtc()) { setUtc(true); } else { setTimeZone(dateTime.getTimeZone()); } } } /** * Constructs a new DateTime instance from parsing the specified * string representation in the default (local) timezone. * @param value */ public DateTime(final String value) throws ParseException { this(value, null); /* long time = 0; try { synchronized (UTC_FORMAT) { time = UTC_FORMAT.parse(value).getTime(); } setUtc(true); } catch (ParseException pe) { synchronized (DEFAULT_FORMAT) { DEFAULT_FORMAT.setTimeZone(getFormat().getTimeZone()); time = DEFAULT_FORMAT.parse(value).getTime(); } this.time = new Time(time, getFormat().getTimeZone()); } setTime(time); */ } /** * Creates a new date-time instance from the specified value in the given * timezone. If a timezone is not specified, the default timezone (as * returned by {@link java.util.TimeZone#getDefault()}) is used. * @param value * @throws ParseException */ public DateTime(final String value, final TimeZone timezone) throws ParseException { this(); try { synchronized (UTC_FORMAT) { setTime(UTC_FORMAT.parse(value).getTime()); } setUtc(true); } catch (ParseException pe) { synchronized (DEFAULT_FORMAT) { if (timezone != null) { DEFAULT_FORMAT.setTimeZone(timezone); setTime(DEFAULT_FORMAT.parse(value).getTime()); } else { // Use lenient parsing for floating times. This is to overcome // the problem of parsing VTimeZone dates that specify dates // that the strict parser does not accept. LENIENT_DEFAULT_FORMAT.setTimeZone(getFormat().getTimeZone()); setTime(LENIENT_DEFAULT_FORMAT.parse(value).getTime()); } } setTimeZone(timezone); } } /* (non-Javadoc) * @see java.util.Date#setTime(long) */ public final void setTime(final long time) { super.setTime(time); this.time.setTime(time); } /** * @return Returns the utc. */ public final boolean isUtc() { return time.isUtc(); } /** * Updates this date-time to display in UTC time if the argument * is true. Otherwise, resets to the default timezone. * @param utc The utc to set. */ public final void setUtc(final boolean utc) { // reset the timezone associated with this instance.. setTimeZone(null); if (utc) { getFormat().setTimeZone(TimeZone.getTimeZone(TimeZones.UTC_ID)); time = new Time(time, getFormat().getTimeZone()); } } /** * Sets the timezone associated with this date-time instance. If the specified * timezone is null * @param timezone */ public final void setTimeZone(final TimeZone timezone) { this.timezone = timezone; if (timezone != null) { getFormat().setTimeZone(timezone); } else { // use GMT timezone to avoid daylight savings rules affecting floating // time values.. getFormat().setTimeZone(TimeZone.getDefault()); // getFormat().setTimeZone(TimeZone.getTimeZone(TimeZones.GMT_ID)); } time = new Time(time, getFormat().getTimeZone()); } /** * Returns the current timezone associated with this date-time value. * @return a Java timezone */ public final TimeZone getTimeZone() { return timezone; } /* (non-Javadoc) * @see java.lang.Object#toString() */ public final String toString() { StringBuffer b = new StringBuffer(super.toString()); b.append('T'); b.append(time.toString()); return b.toString(); } /* (non-Javadoc) * @see java.lang.Object#equals(java.lang.Object) */ public boolean equals(final Object arg0) { //TODO: what about compareTo, before, after, etc.? if (arg0 instanceof DateTime) { return time.equals(((DateTime) arg0).time) && super.equals(arg0); } return super.equals(arg0); } /* (non-Javadoc) * @see java.lang.Object#hashCode() */ public int hashCode() { return time.hashCode() + super.hashCode(); } }