001/* 002 * #%L 003 * HAPI FHIR - Core Library 004 * %% 005 * Copyright (C) 2014 - 2025 Smile CDR, Inc. 006 * %% 007 * Licensed under the Apache License, Version 2.0 (the "License"); 008 * you may not use this file except in compliance with the License. 009 * You may obtain a copy of the License at 010 * 011 * http://www.apache.org/licenses/LICENSE-2.0 012 * 013 * Unless required by applicable law or agreed to in writing, software 014 * distributed under the License is distributed on an "AS IS" BASIS, 015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 * #L% 019 */ 020package ca.uhn.fhir.model.primitive; 021 022import ca.uhn.fhir.model.api.TemporalPrecisionEnum; 023import ca.uhn.fhir.model.api.annotation.DatatypeDef; 024import ca.uhn.fhir.model.api.annotation.SimpleSetter; 025import ca.uhn.fhir.parser.DataFormatException; 026 027import java.util.Calendar; 028import java.util.Date; 029import java.util.GregorianCalendar; 030import java.util.TimeZone; 031 032/** 033 * Represents a FHIR date datatype. Valid precisions values for this type are: 034 * <ul> 035 * <li>{@link TemporalPrecisionEnum#YEAR} 036 * <li>{@link TemporalPrecisionEnum#MONTH} 037 * <li>{@link TemporalPrecisionEnum#DAY} 038 * </ul> 039 * 040 * <p> 041 * <b>Note on using Java Date objects:</b> This type stores the date as a Java Date. Note that 042 * the Java Date has more precision (millisecond precision), and does not store a timezone. As such, 043 * it could potentially cause issues. For example, if a Date contains the number of milliseconds at 044 * midnight in a timezone across the date line from your location, it might refer to a different date than 045 * intended. 046 * </p> 047 * <p> 048 * As such, it is recommended to use the <code>Calendar<code> or <code>int,int,int</code> constructors 049 * </p> 050 */ 051@DatatypeDef(name = "date") 052public class DateDt extends BaseDateTimeDt { 053 054 /** 055 * The default precision for this type 056 */ 057 public static final TemporalPrecisionEnum DEFAULT_PRECISION = TemporalPrecisionEnum.DAY; 058 059 /** 060 * Constructor 061 */ 062 public DateDt() { 063 super(); 064 } 065 066 /** 067 * Constructor which accepts a date value and uses the {@link #DEFAULT_PRECISION} for this type. 068 */ 069 public DateDt(Calendar theCalendar) { 070 super(theCalendar.getTime(), DEFAULT_PRECISION); 071 setTimeZone(theCalendar.getTimeZone()); 072 } 073 074 /** 075 * Constructor which accepts a date value and uses the {@link #DEFAULT_PRECISION} for this type. 076 * <b>Please see the note on timezones</b> on the {@link DateDt class documentation} for considerations 077 * when using this constructor! 078 */ 079 @SimpleSetter(suffix = "WithDayPrecision") 080 public DateDt(@SimpleSetter.Parameter(name = "theDate") Date theDate) { 081 super(theDate, DEFAULT_PRECISION); 082 } 083 084 /** 085 * Constructor which accepts a date value and a precision value. Valid precisions values for this type are: 086 * <ul> 087 * <li>{@link TemporalPrecisionEnum#YEAR} 088 * <li>{@link TemporalPrecisionEnum#MONTH} 089 * <li>{@link TemporalPrecisionEnum#DAY} 090 * </ul> 091 * <b>Please see the note on timezones</b> on the {@link DateDt class documentation} for considerations 092 * when using this constructor! 093 * 094 * @throws DataFormatException 095 * If the specified precision is not allowed for this type 096 */ 097 @SimpleSetter 098 public DateDt( 099 @SimpleSetter.Parameter(name = "theDate") Date theDate, 100 @SimpleSetter.Parameter(name = "thePrecision") TemporalPrecisionEnum thePrecision) { 101 super(theDate, thePrecision); 102 } 103 104 /** 105 * Constructor which accepts a date value and uses the {@link #DEFAULT_PRECISION} for this type. 106 * 107 * @param theYear The year, e.g. 2015 108 * @param theMonth The month, e.g. 0 for January 109 * @param theDay The day (1 indexed) e.g. 1 for the first day of the month 110 */ 111 public DateDt(int theYear, int theMonth, int theDay) { 112 this(toCalendarZulu(theYear, theMonth, theDay)); 113 } 114 115 /** 116 * Constructor which accepts a date as a string in FHIR format 117 * 118 * @throws DataFormatException 119 * If the precision in the date string is not allowed for this type 120 */ 121 public DateDt(String theDate) { 122 super(theDate); 123 } 124 125 /** 126 * Returns the default precision for this datatype 127 * 128 * @see #DEFAULT_PRECISION 129 */ 130 @Override 131 protected TemporalPrecisionEnum getDefaultPrecisionForDatatype() { 132 return DEFAULT_PRECISION; 133 } 134 135 @Override 136 protected boolean isPrecisionAllowed(TemporalPrecisionEnum thePrecision) { 137 switch (thePrecision) { 138 case YEAR: 139 case MONTH: 140 case DAY: 141 return true; 142 default: 143 return false; 144 } 145 } 146 147 private static GregorianCalendar toCalendarZulu(int theYear, int theMonth, int theDay) { 148 GregorianCalendar retVal = new GregorianCalendar(TimeZone.getTimeZone("GMT")); 149 retVal.set(Calendar.YEAR, theYear); 150 retVal.set(Calendar.MONTH, theMonth); 151 retVal.set(Calendar.DATE, theDay); 152 return retVal; 153 } 154}