View Javadoc
1   /**
2    * Copyright 2004-2015 The Kuali Foundation
3    *
4    * Licensed under the Educational Community 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.opensource.org/licenses/ecl2.php
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  package org.kuali.kpme.tklm.time.detail.validation;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.joda.time.*;
20  import org.kuali.kpme.core.api.assignment.Assignment;
21  import org.kuali.kpme.core.api.assignment.AssignmentDescriptionKey;
22  import org.kuali.kpme.core.api.calendar.entry.CalendarEntry;
23  import org.kuali.kpme.core.api.earncode.EarnCode;
24  import org.kuali.kpme.core.api.job.JobContract;
25  import org.kuali.kpme.core.calendar.entry.CalendarEntryBo;
26  import org.kuali.kpme.core.service.HrServiceLocator;
27  import org.kuali.kpme.core.util.HrConstants;
28  import org.kuali.kpme.core.util.HrContext;
29  import org.kuali.kpme.core.util.TKUtils;
30  import org.kuali.kpme.core.util.ValidationUtils;
31  import org.kuali.kpme.tklm.api.common.TkConstants;
32  import org.kuali.kpme.tklm.api.leave.block.LeaveBlock;
33  import org.kuali.kpme.tklm.api.leave.summary.LeaveSummaryContract;
34  import org.kuali.kpme.tklm.api.time.clocklog.ClockLog;
35  import org.kuali.kpme.tklm.api.time.timeblock.TimeBlock;
36  import org.kuali.kpme.tklm.common.CalendarValidationUtil;
37  import org.kuali.kpme.tklm.leave.calendar.validation.LeaveCalendarValidationUtil;
38  import org.kuali.kpme.tklm.leave.service.LmServiceLocator;
39  import org.kuali.kpme.tklm.time.clocklog.ClockLogBo;
40  import org.kuali.kpme.tklm.time.detail.web.TimeDetailActionFormBase;
41  import org.kuali.kpme.tklm.time.service.TkServiceLocator;
42  import org.kuali.kpme.tklm.time.timesheet.TimesheetDocument;
43  import org.kuali.rice.krad.util.ObjectUtils;
44  
45  import java.math.BigDecimal;
46  import java.util.ArrayList;
47  import java.util.List;
48  
49  public class TimeDetailValidationUtil extends CalendarValidationUtil {
50  
51      public static List<String> validateLeaveEntry(TimeDetailActionFormBase tdaf) throws Exception {
52      	// This validator could be shared between LeaveCalendarValidationUtil and TimeDetailValidationUtil
53      	// if the common parameters required by both are moved to a shared parent of TimeDetailActionFormBase and LeaveCalendarForm.
54      	List<String> errorMsgList = new ArrayList<String>();
55          CalendarEntry payCalendarEntry = tdaf.getCalendarEntry();
56      	if(ObjectUtils.isNotNull(payCalendarEntry)) {
57  			LeaveBlock lb = null;
58  			if(StringUtils.isNotEmpty(tdaf.getLmLeaveBlockId())) {
59  				lb = LmServiceLocator.getLeaveBlockService().getLeaveBlock(tdaf.getLmLeaveBlockId());
60  			}
61  			errorMsgList.addAll(CalendarValidationUtil.validateEarnCode(tdaf.getSelectedEarnCode(),tdaf.getStartDate(),tdaf.getEndDate()));
62  			if(errorMsgList.isEmpty()) {
63  				LeaveSummaryContract ls = LmServiceLocator.getLeaveSummaryService().getLeaveSummaryAsOfDate(HrContext.getTargetPrincipalId(), TKUtils.formatDateString(tdaf.getEndDate()));
64  				
65  				  BigDecimal leaveAmount = tdaf.getLeaveAmount();
66                    if(leaveAmount == null) {
67  	                  Long startTime = TKUtils.convertDateStringToDateTimeWithoutZone(tdaf.getStartDate(), tdaf.getStartTime()).getMillis();
68  	                  Long endTime = TKUtils.convertDateStringToDateTimeWithoutZone(tdaf.getEndDate(), tdaf.getEndTime()).getMillis();
69  	                  leaveAmount = TKUtils.getHoursBetween(startTime, endTime);
70                    }
71  				
72  				// Validate LeaveBlock timings and all that
73  				/**
74  				 * In all cases, TIME, AMOUNT, HOUR, DAY, we should validate usage and balance limits. But depending on earn code record method, the derivation of requested
75  				 * leave amount varies. It can be derived from endTime - startTime, or vary by units.
76  				 * 
77  				 * TimeDetailValidationUtil.validateLeaveParametersByEarnCodeRecordMethod offers a form of organization for these validations.
78  				 */
79  				errorMsgList.addAll(TimeDetailValidationUtil.validateLeaveParametersByEarnCodeRecordMethod(tdaf));
80  				errorMsgList.addAll(LeaveCalendarValidationUtil.validateAvailableLeaveBalanceForUsage(tdaf.getSelectedEarnCode(), 
81  						tdaf.getStartDate(), tdaf.getEndDate(), leaveAmount, lb));
82  				//Validate leave block does not exceed max usage. Leave Calendar Validators at this point rely on a leave summary.
83  		        errorMsgList.addAll(LeaveCalendarValidationUtil.validateLeaveAccrualRuleMaxUsage(ls, tdaf.getSelectedEarnCode(),
84  		                tdaf.getStartDate(), tdaf.getEndDate(), leaveAmount, lb));
85  		        errorMsgList.addAll(LeaveCalendarValidationUtil.validateHoursUnderTwentyFour(tdaf.getSelectedEarnCode(),
86  		        		tdaf.getStartDate(), tdaf.getEndDate(),leaveAmount));
87  			}
88  		}
89  		return errorMsgList;
90      }
91      
92      public static List<String> validateLeaveParametersByEarnCodeRecordMethod(TimeDetailActionFormBase lcf) {
93      	List<String> errors = new ArrayList<String>();
94      	if (StringUtils.isNotBlank(lcf.getSelectedEarnCode()) &&  lcf.getCalendarEntry() != null) {
95      		//earn code is validate through the span of the leave entry, could the earn code's record method change between then and the leave period end date?
96      		//Why not use endDateS to retrieve the earn code?
97              CalendarEntry calendarEntry = lcf.getCalendarEntry();
98      		EarnCode earnCode = HrServiceLocator.getEarnCodeService().getEarnCode(lcf.getSelectedEarnCode(), calendarEntry.getEndPeriodFullDateTime().toLocalDate());
99      		if(earnCode != null) {
100     			if(earnCode.getRecordMethod().equalsIgnoreCase(HrConstants.EARN_CODE_TIME)) {
101     		    	return LeaveCalendarValidationUtil.validateTimeParametersForLeaveEntry(earnCode, lcf.getCalendarEntry(), lcf.getStartDate(), lcf.getEndDate(), lcf.getStartTime(), lcf.getEndTime(), lcf.getSelectedAssignment(), lcf.getLmLeaveBlockId(), null);
102     			}
103     			// we should not have any leave earn codes with amount recording method
104 //    			else if (earnCode.getRecordMethod().equalsIgnoreCase(HrConstants.EARN_CODE_AMOUNT)) {
105 //    		    	return validateAmountParametersForLeaveEntry(earnCode, lcf.getCalendarEntry(), lcf.getStartDate(), lcf.getEndDate(), lcf.getSelectedAssignment(), lcf.getLmLeaveBlockId());
106 //    			}
107     			else if (earnCode.getRecordMethod().equalsIgnoreCase(HrConstants.EARN_CODE_HOUR)) {
108     		    	return validateHourParametersForLeaveEntry(earnCode, lcf.getCalendarEntry(), lcf.getStartDate(), lcf.getEndDate(), lcf.getLeaveAmount());
109     			}
110     			else if (earnCode.getRecordMethod().equalsIgnoreCase(HrConstants.EARN_CODE_DAY)) {
111     		    	return validateDayParametersForLeaveEntry(earnCode, lcf.getCalendarEntry(), lcf.getStartDate(), lcf.getEndDate(), lcf.getLeaveAmount());
112     			}
113     		}
114     	}
115     	return errors;
116     }
117 	 /**
118      * Validate the earn code exists on every day within the date rage
119      * @param earnCode
120      * @param startDateString
121      * @param endDateString
122      *
123      * @return A list of error strings.
124      */
125     @Deprecated
126     public static List<String> validateEarnCode(String earnCode, String startDateString, String endDateString) {
127     	List<String> errors = new ArrayList<String>();
128 
129     	LocalDate tempDate = TKUtils.formatDateTimeStringNoTimezone(startDateString).toLocalDate();
130     	LocalDate localEnd = TKUtils.formatDateTimeStringNoTimezone(endDateString).toLocalDate();
131 		// tempDate and localEnd could be the same day
132     	while(!localEnd.isBefore(tempDate)) {
133     		if(!ValidationUtils.validateEarnCode(earnCode, tempDate)) {
134     			 errors.add("Earn Code " + earnCode + " is not available for " + tempDate);
135     			 break;
136     		}
137     		tempDate = tempDate.plusDays(1);
138     	}
139     	
140     	return errors;
141     }
142 	
143     /**
144      * Convenience method for handling validation directly from the form object.
145      * @param tdaf The populated form.
146      *
147      * @return A list of error strings.
148      */
149     public static List<String> validateTimeEntryDetails(TimeDetailActionFormBase tdaf) {
150     	boolean spanningWeeks = false;
151     	boolean acrossDays = false;
152     	
153     	if(tdaf.getAcrossDays() != null) {
154     		acrossDays = tdaf.getAcrossDays().equalsIgnoreCase("y");
155     	}
156     	
157         return validateTimeEntryDetails(
158                 tdaf.getHours(), tdaf.getAmount(), tdaf.getStartTime(), tdaf.getEndTime(),
159                 tdaf.getStartDate(), tdaf.getEndDate(), tdaf.getTimesheetDocument(),
160                 tdaf.getSelectedEarnCode(), tdaf.getSelectedAssignment(),
161                 acrossDays, tdaf.getTkTimeBlockId(), tdaf.getOvertimePref()
162         );
163     }
164 
165     public static List<String> validateTimeEntryDetails(BigDecimal hours, BigDecimal amount, String startTimeS, String endTimeS, String startDateS, String endDateS, TimesheetDocument timesheetDocument, String selectedEarnCode, String selectedAssignment, boolean acrossDays, String timeblockId, String overtimePref) {
166         List<String> errors = new ArrayList<String>();
167         LocalDate savedStartDate = TKUtils.formatDateString(startDateS);
168         LocalDate savedEndDate = TKUtils.formatDateString(endDateS);
169 
170         if (timesheetDocument == null) {
171             errors.add("No timesheet document found.");
172         }
173         if (errors.size() > 0) return errors;
174 
175         CalendarEntry payCalEntry = timesheetDocument.getCalendarEntry();
176         EarnCode earnCode = null;
177         if (StringUtils.isNotBlank(selectedEarnCode)) {
178             earnCode = HrServiceLocator.getEarnCodeService().getEarnCode(selectedEarnCode, TKUtils.formatDateTimeStringNoTimezone(endDateS).toLocalDate());
179         }
180         boolean isTimeRecordMethod = earnCode != null && StringUtils.equalsIgnoreCase(earnCode.getRecordMethod(), HrConstants.EARN_CODE_TIME);
181 
182         errors.addAll(CalendarValidationUtil.validateDates(startDateS, endDateS));
183 
184         if (isTimeRecordMethod) {
185             errors.addAll(CalendarValidationUtil.validateTimes(startTimeS, endTimeS));
186         }
187         if (errors.size() > 0) return errors;
188 
189         Long startTime;
190         Long endTime;
191 
192         if (!isTimeRecordMethod) {
193             startTimeS = "0:0";
194             endTimeS = "0:0";
195         }
196 
197         if (acrossDays && !endTimeS.equals("0:00")) { endDateS = startDateS;}
198 
199         startTime = TKUtils.convertDateStringToDateTimeWithoutZone(startDateS, startTimeS).getMillis();
200         endTime = TKUtils.convertDateStringToDateTimeWithoutZone(endDateS, endTimeS).getMillis();
201 
202         errors.addAll(CalendarValidationUtil.validateInterval(payCalEntry, startTime, endTime));
203         if (errors.size() > 0) return errors;
204 
205         if (isTimeRecordMethod) {
206             if (startTimeS == null) errors.add("The start time is blank.");
207             if (endTimeS == null) errors.add("The end time is blank.");
208             if (startTime - endTime == 0) errors.add("Start time and end time cannot be equivalent");
209         }
210         if (errors.size() > 0) return errors;
211 
212         DateTime startTemp = new DateTime(startTime);
213         DateTime endTemp = new DateTime(endTime);
214 /*
215  * KPME-2687:
216  *
217  * Removed 24 hour limitation. System creates continuous sequence of time blocks when !accrossDays,
218  * hours between startTemp and endTemp may be over 24 hours.
219  *
220         if (errors.size() == 0 && !acrossDays && !StringUtils.equals(TkConstants.EARN_CODE_CPE, overtimePref)) {
221             Hours hrs = Hours.hoursBetween(startTemp, endTemp);
222             if (hrs.getHours() > 24) errors.add("One timeblock cannot exceed 24 hours");
223         }
224         if (errors.size() > 0) return errors;
225 
226  */
227 
228         //Check that assignment is valid within the timeblock span. 
229         AssignmentDescriptionKey assignKey = HrServiceLocator.getAssignmentService().getAssignmentDescriptionKey(selectedAssignment);
230         Assignment assign = HrServiceLocator.getAssignmentService().getAssignmentForTargetPrincipal(assignKey, startTemp.toLocalDate());
231         if (assign == null) errors.add("Assignment is not valid for start date " + TKUtils.formatDate(new LocalDate(startTime)));
232         assign = HrServiceLocator.getAssignmentService().getAssignmentForTargetPrincipal(assignKey, endTemp.toLocalDate());
233         if (assign == null) errors.add("Assignment is not valid for end date " + TKUtils.formatDate(new LocalDate(endTime)));
234         if (errors.size() > 0) return errors;
235 
236         //------------------------
237         // some of the simple validations are in the js side in order to reduce the server calls
238         // 1. check if the begin / end time is empty - tk.calenadr.js
239         // 2. check the time format - timeparse.js
240         // 3. only allows decimals to be entered in the hour field
241         //------------------------
242 
243         //------------------------
244         // check if the overnight shift is across days
245         //------------------------
246         if (acrossDays && hours == null && amount == null) {
247             if (savedEndDate.isAfter(savedStartDate)
248                     && startTemp.getHourOfDay() > endTemp.getHourOfDay()
249                     && !(endTemp.getDayOfYear() - startTemp.getDayOfYear() <= 1
250                     && endTemp.getHourOfDay() == 0)) {
251                 errors.add("The \"apply to each day\" box should not be checked.");
252             }
253         }
254         if (errors.size() > 0) return errors;
255 
256         //------------------------
257         // check if the begin / end time are valid
258         //------------------------
259         if ((startTime.compareTo(endTime) > 0 || endTime.compareTo(startTime) < 0)) {
260             errors.add("The time or date is not valid.");
261         }
262         if (errors.size() > 0) return errors;
263         
264         // KPME-1446 
265         // -------------------------------
266         // check if there is a weekend day when the include weekends flag is checked
267         //--------------------------------
268        
269         //------------------------
270         // Amount cannot be zero
271         //------------------------
272         if (amount != null && earnCode != null && StringUtils.equals(earnCode.getEarnCodeType(), HrConstants.EARN_CODE_AMOUNT)) {
273             if (amount.equals(BigDecimal.ZERO)) {
274                 errors.add("Amount cannot be zero.");
275             }
276             if (amount.scale() > 2) {
277                 errors.add("Amount cannot have more than two digits after decimal point.");
278             }
279         }
280         if (errors.size() > 0) return errors;
281 
282         //------------------------
283         // check if the hours entered for hourly earn code is greater than 24 hours per day
284         // Hours cannot be zero
285         //------------------------
286         if (hours != null && earnCode != null && StringUtils.equals(earnCode.getEarnCodeType(), HrConstants.EARN_CODE_HOUR)) {
287             if (hours.equals(BigDecimal.ZERO)) {
288                 errors.add("Hours cannot be zero.");
289             }
290             if (hours.scale() > 2) {
291                 errors.add("Hours cannot have more than two digits after decimal point.");
292             }
293 /*
294  * KPME-2671:
295  * 
296  * Replacing this conditional with the one below. Shouldn't matter if the date range spans more than one day,
297  * hours shouldn't exceed 24.
298  * 
299  *          int dayDiff = endTemp.getDayOfYear() - startTemp.getDayOfYear() + 1;
300             if (hours.compareTo(new BigDecimal(dayDiff * 24)) == 1) {
301                 errors.add("Cannot enter more than 24 hours per day.");
302             }
303  */
304         }
305         if (errors.size() > 0) return errors;
306 
307         /**
308          * KPME-2671:
309          * 
310          * Generalize 24 limit to hour field on time entry form.
311          * 
312          */
313     	if(hours != null && hours.compareTo(new BigDecimal(24.0)) > 0) {
314     		errors.add("Hours cannot exceed 24.");
315     	}
316     	//------------------------
317         // check if time blocks overlap with each other. Note that the tkTimeBlockId is used to
318         // determine is it's updating an existing time block or adding a new one
319         //------------------------
320 
321         boolean isRegularEarnCode = StringUtils.equals(assign.getJob().getPayTypeObj().getRegEarnCode(),selectedEarnCode);
322         startTime = TKUtils.convertDateStringToDateTime(startDateS, startTimeS).getMillis();
323         endTime = TKUtils.convertDateStringToDateTime(endDateS, endTimeS).getMillis();
324 
325         errors.addAll(validateOverlap(startTime, endTime, acrossDays, startDateS, endTimeS,startTemp, endTemp, timesheetDocument, timeblockId, isRegularEarnCode, selectedEarnCode));
326         if (errors.size() > 0) return errors;
327 
328         // Accrual Hour Limits Validation
329         //errors.addAll(TkServiceLocator.getTimeOffAccrualService().validateAccrualHoursLimitByEarnCode(timesheetDocument, selectedEarnCode));
330 
331         return errors;
332     }
333 
334     public static List<String> validateOverlap(Long startTime, Long endTime, boolean acrossDays, String startDateS, String endTimeS, DateTime startTemp, DateTime endTemp, TimesheetDocument timesheetDocument, String timeblockId, boolean isRegularEarnCode, String selectedEarnCode) {
335         List<String> errors = new ArrayList<String>();
336         Interval addedTimeblockInterval = new Interval(startTime, endTime);
337         List<Interval> dayInt = new ArrayList<Interval>();
338 
339         //if the user is clocked in, check if this time block overlaps with the clock action
340         ClockLog lastClockLog = TkServiceLocator.getClockLogService().getLastClockLog(HrContext.getTargetPrincipalId());
341         if(lastClockLog != null &&
342         		(lastClockLog.getClockAction().equals(TkConstants.CLOCK_IN) 
343         				|| lastClockLog.getClockAction().equals(TkConstants.LUNCH_IN))) {
344         	 DateTime lastClockDateTime = lastClockLog.getClockDateTime();
345              //String lastClockZone = lastClockLog.getClockTimestampTimezone();
346              //if (StringUtils.isEmpty(lastClockZone)) {
347              //    lastClockZone = TKUtils.getSystemTimeZone();
348              //}
349              //DateTimeZone zone = DateTimeZone.forID(lastClockZone);
350              //DateTime clockWithZone = lastClockDateTime.withZone(zone);
351              //DateTime currentTime = new DateTime(System.currentTimeMillis(), zone);
352             DateTime currentTime = DateTime.now();
353             if (lastClockDateTime.getMillis() > currentTime.getMillis()) {
354                 currentTime = new DateTime(lastClockDateTime.getMillis());
355             }
356             
357             Interval currentClockInInterval = new Interval(lastClockDateTime, currentTime);
358             if (isRegularEarnCode && addedTimeblockInterval.overlaps(currentClockInInterval)) {
359                  errors.add("The time block you are trying to add overlaps with the current clock action.");
360                  return errors;
361              }
362         }
363        
364         if (acrossDays) {
365         	DateTime start = new DateTime(startTime);
366         	// KPME-2720
367         	// The line below is necessary because "end" needs to have start date to construct the right end datetime to 
368         	// create an interval.  For example, if user selects from 8a 8/19 to 10a 8/21 and checks across days check box, 
369         	// startTime would have 8a 8/19 and endTime would have 10a 8/21.  With the line below, "end" would have 10a 8/19, and
370         	// an interval from 8a 8/19 to 10a 8/19 would be created.  However, since it was using convertDateStringToDateTime,
371         	// which takes into account user time zone, "end" was off.  Use convertDateStringToDateTimeWithoutZone isntead
372         	// so that "end" would have the right time in default time zone.
373         	//        	DateTime end = TKUtils.convertDateStringToDateTimeWithoutZone(startDateS, endTimeS);
374         	
375         	// kpme-3181
376         	// revert the following line of change from kpme-2720 since it's using timezone for starttime but not endtime
377         	// that's causing interval not being build correctly issue
378             DateTime end = TKUtils.convertDateStringToDateTime(startDateS, endTimeS);
379         	
380         	if (endTemp.getDayOfYear() - startTemp.getDayOfYear() < 1) {
381                 end = new DateTime(endTime);
382             }
383             DateTime groupEnd = new DateTime(endTime);
384             Long startLong = start.getMillis();
385             Long endLong = end.getMillis();
386             //create interval span if start is before the end and the end is after the start except
387             //for when the end is midnight ..that converts to midnight of next day
388             DateMidnight midNight = new DateMidnight(endLong);
389             while (start.isBefore(groupEnd.getMillis()) && ((endLong >= startLong) || end.isEqual(midNight))) {
390                 Interval tempInt = null;
391                 if (end.isEqual(midNight)) {
392                     tempInt = addedTimeblockInterval;
393                 } else {
394                     tempInt = new Interval(startLong, endLong);
395                 }
396                 dayInt.add(tempInt);
397                 start = start.plusDays(1);
398                 end = end.plusDays(1);
399                 startLong = start.getMillis();
400                 endLong = end.getMillis();
401             }
402         } else {
403             dayInt.add(addedTimeblockInterval);
404         }
405 
406         for (TimeBlock timeBlock : timesheetDocument.getTimeBlocks()) {
407             if (errors.size() == 0 && StringUtils.equals(timeBlock.getEarnCodeType(), HrConstants.EARN_CODE_TIME)) {
408             	// allow regular time blocks to be added with overlapping non-regular time blocks
409             	JobContract aJob = HrServiceLocator.getJobService().getJob(timeBlock.getPrincipalId(), timeBlock.getJobNumber(), timeBlock.getBeginDateTime().toLocalDate());
410             	if(aJob != null && aJob.getPayTypeObj() != null && isRegularEarnCode && !StringUtils.equals(aJob.getPayTypeObj().getRegEarnCode(),timeBlock.getEarnCode())) {
411             		continue;
412             	}
413             	
414                 Interval timeBlockInterval = new Interval(timeBlock.getBeginDateTime(), timeBlock.getEndDateTime());
415                 for (Interval intv : dayInt) {
416                 	// KPME-2720
417                 	// timeblockInterval above seems to have the server timezone (America/New York), for example, if you log in as iadetail1 (America/chicago) 
418                 	// and see a timeblock from 8a to 10a on the screen, that time block is actually stored as 9a to 11a in the table because 
419                 	// there is an hour difference.  So, timeblockInterval intervale above is 9a to 11a in America/New York timezone.  
420                 	// However, intv interval seems to have local time with the server timezone.  For example, if you create a timeblock from 10a to 12p
421                 	// as iadetail1, intv interval is 10a to 12p in America/New York timezone.  This is why it was giving the error below because it was
422                 	// comparing a time block with 9a-11a to a time block with 10a-12p (overlapping).  To fix this, we will create
423                 	// an interval with the right time in the server timezone.
424 //                    String start_datetime = TKUtils.formatDateTimeLong(intv.getStart());
425 //                	String start_date = TKUtils.formatDateTimeShort(intv.getStart());
426 //                	String start_time = TKUtils.formatTimeShort(start_datetime);
427 //                	String end_datetime = TKUtils.formatDateTimeLong(intv.getEnd());                	
428 //                	String end_date = TKUtils.formatDateTimeShort(intv.getEnd());
429 //                	String end_time =  TKUtils.formatTimeShort(end_datetime);
430                 	DateTime start_dt_timezone = new DateTime(startTime);
431                 	DateTime end_dt_timezone = new DateTime(endTime);
432                 	Interval converted_intv = new Interval(start_dt_timezone.getMillis(), end_dt_timezone.getMillis()); // interval with start/end datetime in server timezone
433                     List<Interval> intervals = new ArrayList<Interval>();
434                     if (acrossDays) {
435                         List<LocalDate> localDates = new ArrayList<LocalDate>();
436                         LocalDate startDay = new LocalDate(start_dt_timezone);
437                         DateTimeZone userTimeZone = DateTimeZone.forID(HrServiceLocator.getTimezoneService().getUserTimezone(timesheetDocument.getPrincipalId()));
438                         if (userTimeZone==null) {
439                             userTimeZone = HrServiceLocator.getTimezoneService().getTargetUserTimezoneWithFallback();
440                         }
441 
442 
443                         int days = end_dt_timezone.withZone(userTimeZone).toLocalTime().equals(new LocalTime(0,0,0)) ? Days.daysBetween(startDay, new LocalDate(end_dt_timezone)).getDays() : Days.daysBetween(startDay, new LocalDate(end_dt_timezone)).getDays()+1;
444                         for (int i=0; i < days; i++) {
445                             LocalDate d = startDay.withFieldAdded(DurationFieldType.days(), i);
446                             localDates.add(d);
447                         }
448                         for (LocalDate localDate : localDates) {
449                             DateTime startDateTime = localDate.toDateTime(start_dt_timezone.toLocalTime());
450                             DateTime endDateTime = localDate.toDateTime(end_dt_timezone.toLocalTime());
451                             endDateTime = endDateTime.isBefore(startDateTime) ? endDateTime.plusDays(1) : endDateTime;
452 
453                             intervals.add(new Interval(startDateTime,endDateTime));
454                         }
455 
456                     } else {
457                         intervals.add(converted_intv);
458                     }
459 
460                     for (Interval interval : intervals) {
461                         if (isRegularEarnCode && timeBlockInterval.overlaps(interval) && (timeblockId == null || timeblockId.compareTo(timeBlock.getTkTimeBlockId()) != 0)) {
462                         	errors.add("The time block you are trying to add overlaps with an existing time block.");
463                             return errors;
464                         }else if(timeBlockInterval.overlaps(interval) && (timeblockId == null || timeblockId.compareTo(timeBlock.getTkTimeBlockId()) != 0)){
465                         	errors.add("The time block you are trying to add overlaps with an existing time block.");
466                         	return errors;
467                         }
468                     }
469                 }
470             }
471         }
472 
473         return errors;
474     }
475     
476     /*
477      * Moving to CalendarValidationUtil
478      * @param startDateS
479      * @param endDateS
480      * @return
481      */
482     @Deprecated
483     public static List<String> validateDates(String startDateS, String endDateS) {
484         List<String> errors = new ArrayList<String>();
485         if (errors.size() == 0 && StringUtils.isEmpty(startDateS)) errors.add("The start date is blank.");
486         if (errors.size() == 0 && StringUtils.isEmpty(endDateS)) errors.add("The end date is blank.");
487         return errors;
488     }
489     
490     /*
491      * Moving to CalendarValidationUtil
492      * @param startTimeS
493      * @param endTimeS
494      * @return
495      */
496     @Deprecated
497     public static List<String> validateTimes(String startTimeS, String endTimeS) {
498         List<String> errors = new ArrayList<String>();
499         if (errors.size() == 0 && startTimeS == null) errors.add("The start time is blank.");
500         if (errors.size() == 0 && endTimeS == null) errors.add("The end time is blank.");
501         return errors;
502     }
503     
504     /*
505      * Moving to CalendarValidationUtil
506      * @param payCalEntry
507      * @param startTime
508      * @param endTime
509      * @return
510      */
511     @Deprecated
512     public static List<String> validateInterval(CalendarEntryBo payCalEntry, Long startTime, Long endTime) {
513         List<String> errors = new ArrayList<String>();
514         LocalDateTime pcb_ldt = payCalEntry.getBeginPeriodLocalDateTime();
515         LocalDateTime pce_ldt = payCalEntry.getEndPeriodLocalDateTime();
516 
517         DateTime p_cal_b_dt = pcb_ldt.toDateTime();
518         DateTime p_cal_e_dt = pce_ldt.toDateTime();
519 
520         Interval payInterval = new Interval(p_cal_b_dt, p_cal_e_dt);
521         if (errors.size() == 0 && !payInterval.contains(startTime)) {
522             errors.add("The start date/time is outside the pay period");
523         }
524         if (errors.size() == 0 && !payInterval.contains(endTime) && p_cal_e_dt.getMillis() != endTime) {
525             errors.add("The end date/time is outside the pay period");
526         }
527         return errors;
528     }
529 
530 }