001    /**
002     * Copyright 2004-2013 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.kuali.hr.time.detail.web;
017    
018    import java.sql.Date;
019    import java.text.DateFormat;
020    import java.text.SimpleDateFormat;
021    import java.util.ArrayList;
022    import java.util.Collections;
023    import java.util.HashMap;
024    import java.util.HashSet;
025    import java.util.LinkedHashMap;
026    import java.util.LinkedList;
027    import java.util.List;
028    import java.util.Map;
029    import java.util.Set;
030    
031    import org.apache.commons.collections.CollectionUtils;
032    import org.apache.commons.lang.StringUtils;
033    import org.apache.commons.lang.time.DateUtils;
034    import org.joda.time.DateTime;
035    import org.joda.time.format.ISODateTimeFormat;
036    import org.json.simple.JSONValue;
037    import org.kuali.hr.lm.accrual.AccrualCategory;
038    import org.kuali.hr.lm.leaveblock.LeaveBlock;
039    import org.kuali.hr.lm.leavecalendar.validation.LeaveCalendarValidationUtil;
040    import org.kuali.hr.lm.leaveplan.LeavePlan;
041    import org.kuali.hr.time.assignment.AssignmentDescriptionKey;
042    import org.kuali.hr.time.calendar.CalendarEntries;
043    import org.kuali.hr.time.earncode.EarnCode;
044    import org.kuali.hr.time.earncodegroup.EarnCodeGroup;
045    import org.kuali.hr.time.principal.PrincipalHRAttributes;
046    import org.kuali.hr.time.roles.TkUserRoles;
047    import org.kuali.hr.time.service.base.TkServiceLocator;
048    import org.kuali.hr.time.timeblock.TimeBlock;
049    import org.kuali.hr.time.timeblock.TimeHourDetail;
050    import org.kuali.hr.time.util.TKUser;
051    import org.kuali.hr.time.util.TKUtils;
052    import org.kuali.hr.time.util.TkConstants;
053    import org.kuali.hr.time.workarea.WorkArea;
054    import org.kuali.rice.krad.util.GlobalVariables;
055    
056    public class ActionFormUtils {
057    
058       // public static void validateHourLimit(TimeDetailActionFormBase tdaf) throws Exception {
059       //     List<String> warningMessages = TkServiceLocator.getTimeOffAccrualService().validateAccrualHoursLimit(tdaf.getTimesheetDocument());
060       //      addUniqueWarningsToForm(tdaf, warningMessages);
061       // }
062    
063        public static void addWarningTextFromEarnGroup(TimeDetailActionFormBase tdaf) throws Exception {
064            List<String> warningMessages = TkServiceLocator.getEarnCodeGroupService().warningTextFromEarnCodeGroupsOfDocument(tdaf.getTimesheetDocument());
065            addUniqueWarningsToForm(tdaf, warningMessages);
066        }
067    
068        public static void addUnapprovedIPWarningFromClockLog(TimeDetailActionFormBase tdaf) {
069            List<String> warningMessages = new ArrayList<String>();
070            Set<String> aSet = new HashSet<String>();
071            if(tdaf.getTimesheetDocument() != null) {
072                    List<TimeBlock> tbList = tdaf.getTimesheetDocument().getTimeBlocks();
073                    if(CollectionUtils.isNotEmpty(tbList)) {
074                             aSet.addAll(TkServiceLocator.getClockLogService().getUnapprovedIPWarning(tbList));
075                            
076                    }
077            }
078            warningMessages.addAll(aSet);
079            addUniqueWarningsToForm(tdaf, warningMessages);
080        }
081        
082        public static void addUniqueWarningsToForm(TimeDetailActionFormBase tdaf, List<String> warningMessages) {
083            if (!warningMessages.isEmpty()) {
084                Set<String> aSet = new HashSet<String>();
085                aSet.addAll(warningMessages);
086                aSet.addAll(tdaf.getWarningMessages()); //Only warnings. TODO: Do we need actions and info messages here?
087    
088                List<String> aList = new ArrayList<String>();
089                aList.addAll(aSet);
090                tdaf.setWarningMessages(aList);
091            }
092        }
093    
094    //    public static String getTimeBlockJSONMap(List<TimeBlock> blocks) {
095    //        List<Map<String, Object>> jsonList = getTimeBlocksJson(blocks, null);
096    //        Map<String, Map<String, Object>> jsonMappedList = new HashMap<String, Map<String, Object>>();
097    //        for (Map<String, Object> tbm : jsonList) {
098    //            String id = (String) tbm.get("id");
099    //            jsonMappedList.put(id, tbm);
100    //        }
101    //        return JSONValue.toJSONString(jsonMappedList);
102    //    }
103    
104        public static Map<String, String> buildAssignmentStyleClassMap(List<TimeBlock> timeBlocks, List<LeaveBlock> leaveBlocks) {
105            Map<String, String> aMap = new HashMap<String, String>();
106            List<String> assignmentKeys = new ArrayList<String>();
107    
108            for (TimeBlock tb : timeBlocks) {
109                if (!assignmentKeys.contains(tb.getAssignmentKey())) {
110                    assignmentKeys.add(tb.getAssignmentKey());
111                }
112            }
113            for(LeaveBlock lb : leaveBlocks) {
114                    if (!assignmentKeys.contains(lb.getAssignmentKey())) {
115                    assignmentKeys.add(lb.getAssignmentKey());
116                }
117            }
118            Collections.sort(assignmentKeys);
119    
120            for (int i = 0; i < assignmentKeys.size(); i++) {
121                // pick a color from a 15 color palette
122                aMap.put(assignmentKeys.get(i), "assignment" + Integer.toString(i % 15));
123            }
124    
125            return aMap;
126        }
127        
128        public static Map<String, String> buildAssignmentStyleClassMap(List<TimeBlock> timeBlocks) {
129            Map<String, String> aMap = new HashMap<String, String>();
130            List<String> assignmentKeys = new ArrayList<String>();
131    
132            for (TimeBlock tb : timeBlocks) {
133                if (!assignmentKeys.contains(tb.getAssignmentKey())) {
134                    assignmentKeys.add(tb.getAssignmentKey());
135                }
136            }
137    
138            Collections.sort(assignmentKeys);
139    
140            for (int i = 0; i < assignmentKeys.size(); i++) {
141                // pick a color from a five color palette
142                aMap.put(assignmentKeys.get(i), "assignment" + Integer.toString(i % 5));
143            }
144    
145            return aMap;
146        }
147    
148        /**
149         * This method will build the JSON data structure needed for calendar
150         * manipulation and processing on the client side. Start and End times here
151         * are based on the pre-timezone adjusted times startDisplayTime, and
152         * endDisplayTime.
153         *
154         * @param timeBlocks
155         * @return
156         */
157        public static String getTimeBlocksJson(List<TimeBlock> timeBlocks) {
158    
159            if (timeBlocks == null || timeBlocks.size() == 0) {
160                return "";
161            }
162    
163            List<Map<String, Object>> timeBlockList = new LinkedList<Map<String, Object>>();
164            String timezone = TkServiceLocator.getTimezoneService().getUserTimezone();
165    
166            for (TimeBlock timeBlock : timeBlocks) {
167                Map<String, Object> timeBlockMap = new LinkedHashMap<String, Object>();
168    
169                WorkArea workArea = TkServiceLocator.getWorkAreaService().getWorkArea(timeBlock.getWorkArea(), new java.sql.Date(timeBlock.getEndTimestamp().getTime()));
170                String workAreaDesc = workArea.getDescription();
171    
172                // Roles
173                Boolean isAnyApprover = TkUserRoles.getUserRoles(GlobalVariables.getUserSession().getPrincipalId()).isAnyApproverActive();
174                timeBlockMap.put("isApprover", isAnyApprover);
175                timeBlockMap.put("isSynchronousUser", timeBlock.getClockLogCreated());
176    
177                // Permissions
178                timeBlockMap.put("canEditTb", TkServiceLocator.getPermissionsService().canEditTimeBlock(timeBlock));
179                timeBlockMap.put("canEditTBOvt", TkServiceLocator.getPermissionsService().canEditOvertimeEarnCode(timeBlock));
180                timeBlockMap.put("canAddTB", TkServiceLocator.getPermissionsService().canAddTimeBlock());
181    
182                if (TkServiceLocator.getPermissionsService().canEditTimeBlockAllFields(timeBlock)) {
183                    timeBlockMap.put("canEditTBAll", true);
184                    timeBlockMap.put("canEditTBAssgOnly", false);
185                } else {
186                    timeBlockMap.put("canEditTBAll", false);
187                    timeBlockMap.put("canEditTBAssgOnly", true);
188                }
189    
190                //    tracking any kind of 'mutating' state with this object, it's just a one off modification under a specific circumstance.
191                DateTime start = timeBlock.getBeginTimeDisplay();
192                DateTime end = timeBlock.getEndTimeDisplay();
193    
194                /**
195                 * This is the timeblock backward pushing logic.
196                 * the purpose of this is to accommodate the virtual day mode where the start/end period time is not from 12a to 12a.
197                 * A timeblock will be pushed back if the timeblock is still within the previous interval
198                 */
199                if (timeBlock.isPushBackward()) {
200                    start = start.minusDays(1);
201                    end = end.minusDays(1);
202                }
203    
204                timeBlockMap.put("documentId", timeBlock.getDocumentId());
205                timeBlockMap.put("title", workAreaDesc);
206                EarnCode ec = TkServiceLocator.getEarnCodeService().getEarnCode(timeBlock.getEarnCode(), timeBlock.getBeginDate());
207                timeBlockMap.put("earnCode", timeBlock.getEarnCode());
208                timeBlockMap.put("earnCodeDesc", ec != null ? ec.getDescription() : StringUtils.EMPTY);
209                //TODO: need to cache this or pre-load it when the app boots up
210                // EarnCode earnCode = TkServiceLocator.getEarnCodeService().getEarnCode(timeBlock.getEarnCode(), new java.sql.Date(timeBlock.getBeginTimestamp().getTime()));
211                timeBlockMap.put("earnCodeType", timeBlock.getEarnCodeType());
212    
213                // TODO: Cleanup the start / end time related properties. We certainly don't need all of them.
214                // The ones which are used by the javascript are startDate, endDate, startTime, endTime, startTimeHourMinute, and endTimeHourMinute
215                timeBlockMap.put("start", start.toString(ISODateTimeFormat.dateTimeNoMillis()));
216                timeBlockMap.put("end", end.toString(ISODateTimeFormat.dateTimeNoMillis()));
217                timeBlockMap.put("startDate", start.toString(TkConstants.DT_BASIC_DATE_FORMAT));
218                timeBlockMap.put("endDate", end.toString(TkConstants.DT_BASIC_DATE_FORMAT));
219                timeBlockMap.put("startNoTz", start.toString(ISODateTimeFormat.dateHourMinuteSecond()));
220                timeBlockMap.put("endNoTz", end.toString(ISODateTimeFormat.dateHourMinuteSecond()));
221                // start / endTimeHourMinute fields are for only for the display purpose
222                timeBlockMap.put("startTimeHourMinute", start.toString(TkConstants.DT_BASIC_TIME_FORMAT));
223                timeBlockMap.put("endTimeHourMinute", end.toString(TkConstants.DT_BASIC_TIME_FORMAT));
224                // start / endTime are the actual fields used by the adding / editing timeblocks
225                timeBlockMap.put("startTime", start.toString(TkConstants.DT_MILITARY_TIME_FORMAT));
226                timeBlockMap.put("endTime", end.toString(TkConstants.DT_MILITARY_TIME_FORMAT));
227                timeBlockMap.put("id", timeBlock.getTkTimeBlockId() == null ? null : timeBlock.getTkTimeBlockId().toString());
228                timeBlockMap.put("hours", timeBlock.getHours());
229                timeBlockMap.put("amount", timeBlock.getAmount());
230                timeBlockMap.put("timezone", timezone);
231                timeBlockMap.put("assignment", new AssignmentDescriptionKey(timeBlock.getJobNumber(), timeBlock.getWorkArea(), timeBlock.getTask()).toAssignmentKeyString());
232                timeBlockMap.put("tkTimeBlockId", timeBlock.getTkTimeBlockId() != null ? timeBlock.getTkTimeBlockId() : "");
233                timeBlockMap.put("lunchDeleted", timeBlock.isLunchDeleted());
234    
235                List<Map<String, Object>> timeHourDetailList = new LinkedList<Map<String, Object>>();
236                for (TimeHourDetail timeHourDetail : timeBlock.getTimeHourDetails()) {
237                    Map<String, Object> timeHourDetailMap = new LinkedHashMap<String, Object>();
238                    timeHourDetailMap.put("earnCode", timeHourDetail.getEarnCode());
239                    timeHourDetailMap.put("hours", timeHourDetail.getHours());
240                    timeHourDetailMap.put("amount", timeHourDetail.getAmount());
241    
242                    // if there is a lunch hour deduction, add a flag to the timeBlockMap
243                    if (StringUtils.equals(timeHourDetail.getEarnCode(), "LUN")) {
244                        timeBlockMap.put("lunchDeduction", true);
245                    }
246    
247                    timeHourDetailList.add(timeHourDetailMap);
248                }
249                timeBlockMap.put("timeHourDetails", JSONValue.toJSONString(timeHourDetailList));
250    
251                timeBlockList.add(timeBlockMap);
252            }
253    
254    //        Map<String, Map<String, Object>> jsonMappedList = new HashMap<String, Map<String, Object>>();
255    //        for (Map<String, Object> tbm : timeBlockList) {
256    //            String id = (String) tbm.get("id");
257    //            jsonMappedList.put(id, tbm);
258    //        }
259            return JSONValue.toJSONString(timeBlockList);
260        }
261        
262        
263        /**
264         * This method will build the leave blocks JSON data structure needed for calendar
265         * manipulation and processing on the client side.
266         *
267         * @param leaveBlocks
268         * @return
269         */
270        public static String getLeaveBlocksJson(List<LeaveBlock> leaveBlocks) {
271            if (CollectionUtils.isEmpty(leaveBlocks)) {
272                return "";
273            }
274            List<Map<String, Object>> leaveBlockList = new LinkedList<Map<String, Object>>();
275            for (LeaveBlock leaveBlock : leaveBlocks) {
276                    Map<String, Object> leaveBlockMap = new LinkedHashMap<String, Object>();
277                    leaveBlockMap.put("title", leaveBlock.getAssignmentTitle());
278                    leaveBlockMap.put("assignment", leaveBlock.getAssignmentKey());
279                    leaveBlockMap.put("earnCode", leaveBlock.getEarnCode());
280                    leaveBlockMap.put("lmLeaveBlockId", leaveBlock.getLmLeaveBlockId());
281                    leaveBlockMap.put("leaveAmount", leaveBlock.getLeaveAmount().toString());
282                    DateTime leaveDate = new DateTime(leaveBlock.getLeaveDate());
283                    leaveBlockMap.put("leaveDate", leaveDate.toString(TkConstants.DT_BASIC_DATE_FORMAT));
284                    leaveBlockMap.put("id", leaveBlock.getLmLeaveBlockId());
285                    leaveBlockMap.put("canTransfer", TkServiceLocator.getPermissionsService().canTransferSSTOUsage(leaveBlock));
286                    leaveBlockMap.put("startDate", leaveDate.toString(TkConstants.DT_BASIC_DATE_FORMAT));
287                    leaveBlockMap.put("endDate", leaveDate.toString(TkConstants.DT_BASIC_DATE_FORMAT));
288                    
289                    if(leaveBlock.getBeginTimestamp() != null && leaveBlock.getEndTimestamp() != null) {
290                        DateTime start = new DateTime(leaveBlock.getBeginTimestamp().getTime());
291                            DateTime end = new DateTime(leaveBlock.getEndTimestamp().getTime());
292                            leaveBlockMap.put("startTimeHourMinute", start.toString(TkConstants.DT_BASIC_TIME_FORMAT));
293                            leaveBlockMap.put("endTimeHourMinute", end.toString(TkConstants.DT_BASIC_TIME_FORMAT));
294                            leaveBlockMap.put("startTime", start.toString(TkConstants.DT_MILITARY_TIME_FORMAT));
295                            leaveBlockMap.put("endTime", end.toString(TkConstants.DT_MILITARY_TIME_FORMAT));
296                            leaveBlockMap.put("startDate", start.toString(TkConstants.DT_BASIC_DATE_FORMAT));
297                            leaveBlockMap.put("endDate", end.toString(TkConstants.DT_BASIC_DATE_FORMAT));
298                }
299                    
300                    leaveBlockList.add(leaveBlockMap);
301            }
302            return JSONValue.toJSONString(leaveBlockList);
303        }
304        
305        public static Map<String, String> getPayPeriodsMap(List<CalendarEntries> payPeriods, String viewPrincipal) {
306            // use linked map to keep the order of the pay periods
307            Map<String, String> pMap = Collections.synchronizedMap(new LinkedHashMap<String, String>());
308            if (payPeriods == null || payPeriods.isEmpty()) {
309                return pMap;
310            }
311            payPeriods.removeAll(Collections.singletonList(null));
312            Collections.sort(payPeriods);  // sort the pay period list by getBeginPeriodDate
313            Collections.reverse(payPeriods);  // newest on top
314            SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy");
315            for (CalendarEntries pce : payPeriods) {
316                    
317                    // Check if service date of user is after the Calendar entry
318                Date asOfDate = new Date(DateUtils.addDays(pce.getEndPeriodDate(),-1).getTime());
319                    PrincipalHRAttributes principalHRAttributes = null;
320                    
321                    if(viewPrincipal != null) {
322                            principalHRAttributes = TkServiceLocator.getPrincipalHRAttributeService().getPrincipalCalendar(viewPrincipal, asOfDate);
323                    } else {
324                            pMap.put(pce.getHrCalendarEntriesId(), sdf.format(pce.getBeginPeriodDate()) + " - " + sdf.format((DateUtils.addMilliseconds(pce.getEndPeriodDate(),-1))));
325                    }
326                    
327                    if(principalHRAttributes != null && pce != null && pce.getHrCalendarEntriesId()!= null && pce.getBeginPeriodDate() != null && pce.getEndPeriodDate() != null ) {
328                            Date startCalDate = principalHRAttributes.getServiceDate();
329                            if(startCalDate != null) {
330                                    if(!(pce.getBeginPeriodDate().compareTo(startCalDate) < 0)) {
331                                    //pMap.put(pce.getHrCalendarEntriesId(), sdf.format(pce.getBeginPeriodDate()) + " - " + sdf.format(pce.getEndPeriodDate()));
332                            //getting one millisecond of the endperioddate to match the actual pay period. i.e. pay period end at the 11:59:59:59...PM of that day
333                            pMap.put(pce.getHrCalendarEntriesId(), sdf.format(pce.getBeginPeriodDate()) + " - " + sdf.format((DateUtils.addMilliseconds(pce.getEndPeriodDate(),-1))));
334                            } 
335                    } else {
336                            pMap.put(pce.getHrCalendarEntriesId(), sdf.format(pce.getBeginPeriodDate()) + " - " + sdf.format((DateUtils.addMilliseconds(pce.getEndPeriodDate(),-1))));                      
337                    }
338                    } 
339                    
340            }
341            
342            return pMap;
343        }
344        
345        // detect if the passed-in calendar entry is the current one
346        public static boolean getOnCurrentPeriodFlag(CalendarEntries pce) {
347            Date currentDate = TKUtils.getTimelessDate(null);
348            String viewPrincipal = TKUser.getCurrentTargetPersonId();
349            CalendarEntries calendarEntry = TkServiceLocator.getCalendarService().getCurrentCalendarDates(viewPrincipal,  currentDate);
350    
351            if(pce != null && calendarEntry != null && calendarEntry.equals(pce)) {
352                    return true;
353            }
354            return false;
355        }
356         
357        public static String getUnitOfTimeForEarnCode(EarnCode earnCode) {
358    //      AccrualCategory acObj = null;
359    //      if(earnCode.getAccrualCategory() != null) {
360    //              acObj = TkServiceLocator.getAccrualCategoryService().getAccrualCategory(earnCode.getAccrualCategory(), TKUtils.getCurrentDate());
361    //      }
362    //      String unitTime = (acObj!= null ? acObj.getUnitOfTime() : earnCode.getRecordMethod()) ;
363            String unitTime = earnCode.getRecordMethod() ;
364            return unitTime;
365        }
366        
367        public static int getPlanningMonthsForEmployee(String principalid) {
368                    int plannningMonths = 0;
369                    PrincipalHRAttributes principalHRAttributes = TkServiceLocator
370                                    .getPrincipalHRAttributeService().getPrincipalCalendar(
371                                                    principalid, TKUtils.getCurrentDate());
372                    if (principalHRAttributes != null
373                                    && principalHRAttributes.getLeavePlan() != null) {
374                            LeavePlan lp = TkServiceLocator.getLeavePlanService()
375                                            .getLeavePlan(principalHRAttributes.getLeavePlan(),
376                                                            TKUtils.getCurrentDate());
377                            if (lp != null && lp.getPlanningMonths() != null) {
378                                    plannningMonths = Integer.parseInt(lp.getPlanningMonths());
379                            }
380                    }
381                    return plannningMonths;
382        }
383        
384        public static List<CalendarEntries> getAllCalendarEntriesForYear(List<CalendarEntries> entries, String year) {
385            List<CalendarEntries> results = new ArrayList<CalendarEntries>();
386            DateFormat df = new SimpleDateFormat("yyyy");
387             
388            for(CalendarEntries ce : entries) {
389                    if(df.format(ce.getBeginPeriodDate()).equals(year)) {
390                            results.add(ce);
391                    }
392            }
393            return results;
394        }
395        
396    }
397