mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-08-14 17:58:59 +00:00
Start on the refactoring changes for ALF-11687, to support the repeating=all flag on the dashlet event listing too
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@32421 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
@@ -0,0 +1,178 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2005-2011 Alfresco Software Limited.
|
||||||
|
*
|
||||||
|
* This file is part of Alfresco
|
||||||
|
*
|
||||||
|
* Alfresco is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Alfresco is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public License
|
||||||
|
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
package org.alfresco.repo.web.scripts.calendar;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Calendar;
|
||||||
|
import java.util.Comparator;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.alfresco.service.cmr.calendar.CalendarEntry;
|
||||||
|
import org.alfresco.service.cmr.calendar.CalendarRecurrenceHelper;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class provides functionality common across the webscripts
|
||||||
|
* which list events.
|
||||||
|
*
|
||||||
|
* @author Nick Burch
|
||||||
|
* @since 4.0
|
||||||
|
*/
|
||||||
|
public abstract class AbstractCalendarListingWebScript extends AbstractCalendarWebScript
|
||||||
|
{
|
||||||
|
protected static final String RESULT_NAME = "name";
|
||||||
|
protected static final String RESULT_TITLE = "title";
|
||||||
|
protected static final String RESULT_START = "start";
|
||||||
|
protected static final String RESULT_END = "end";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a Comparator for (re-)sorting events, typically used after
|
||||||
|
* expanding out recurring instances.
|
||||||
|
*/
|
||||||
|
protected static Comparator<Map<String, Object>> getEventDetailsSorter()
|
||||||
|
{
|
||||||
|
return new Comparator<Map<String, Object>>()
|
||||||
|
{
|
||||||
|
public int compare(Map<String, Object> resultA,
|
||||||
|
Map<String, Object> resultB)
|
||||||
|
{
|
||||||
|
Date startA = (Date)resultA.get(RESULT_START);
|
||||||
|
Date startB = (Date)resultB.get(RESULT_START);
|
||||||
|
|
||||||
|
int cmp = startA.compareTo(startB);
|
||||||
|
if (cmp == 0)
|
||||||
|
{
|
||||||
|
Date endA = (Date)resultA.get(RESULT_END);
|
||||||
|
Date endB = (Date)resultB.get(RESULT_END);
|
||||||
|
cmp = endA.compareTo(endB);
|
||||||
|
if (cmp == 0)
|
||||||
|
{
|
||||||
|
String nameA = (String)resultA.get(RESULT_NAME);
|
||||||
|
String nameB = (String)resultB.get(RESULT_NAME);
|
||||||
|
return nameA.compareTo(nameB);
|
||||||
|
}
|
||||||
|
return cmp;
|
||||||
|
}
|
||||||
|
return cmp;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Do what's needed for recurring events.
|
||||||
|
*
|
||||||
|
* @return If dates have been tweaked, and a sort may be required
|
||||||
|
*/
|
||||||
|
protected boolean handleRecurring(CalendarEntry entry, Map<String, Object> entryResult,
|
||||||
|
List<Map<String, Object>> allResults, Date from, Date until, boolean repeatingFirstOnly)
|
||||||
|
{
|
||||||
|
if (entry.getRecurrenceRule() == null)
|
||||||
|
{
|
||||||
|
// Nothing to do
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If no date is given, start looking for occurrences from the event itself
|
||||||
|
if (from == null)
|
||||||
|
{
|
||||||
|
from = entry.getStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do we need to limit ourselves?
|
||||||
|
// Should we limit ourselves?
|
||||||
|
if (!repeatingFirstOnly)
|
||||||
|
{
|
||||||
|
if (until == null)
|
||||||
|
{
|
||||||
|
// If no end date was given, only allow repeating instances
|
||||||
|
// for next 60 days, to keep the list sane
|
||||||
|
// (It's normally only used for a month view anyway)
|
||||||
|
Calendar c = Calendar.getInstance();
|
||||||
|
c.setTime(from);
|
||||||
|
c.add(Calendar.DATE, 60);
|
||||||
|
until = c.getTime();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// How long is it?
|
||||||
|
long duration = entry.getEnd().getTime() - entry.getStart().getTime();
|
||||||
|
|
||||||
|
// Get it's recurring instances
|
||||||
|
List<Date> dates = CalendarRecurrenceHelper.getRecurrencesOnOrAfter(
|
||||||
|
entry, from, until, repeatingFirstOnly);
|
||||||
|
if (dates == null)
|
||||||
|
{
|
||||||
|
dates = new ArrayList<Date>();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add on the original event time itself if needed
|
||||||
|
if (entry.getStart().getTime() >= from.getTime())
|
||||||
|
{
|
||||||
|
if (dates.size() == 0 || dates.get(0).getTime() != entry.getStart().getTime())
|
||||||
|
{
|
||||||
|
// Original event is after the start time, and not on the recurring list
|
||||||
|
dates.add(0, entry.getStart());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we got no dates, then no recurrences in the period so zap
|
||||||
|
if (dates.size() == 0)
|
||||||
|
{
|
||||||
|
allResults.remove(entryResult);
|
||||||
|
return false; // Remains sorted despite delete
|
||||||
|
}
|
||||||
|
|
||||||
|
// Always update the live entry
|
||||||
|
updateRepeatingStartEnd(dates.get(0), duration, entryResult);
|
||||||
|
|
||||||
|
// If first result only, alter title and finish
|
||||||
|
if (repeatingFirstOnly)
|
||||||
|
{
|
||||||
|
entryResult.put(RESULT_TITLE, entry.getTitle() + " (Repeating)");
|
||||||
|
return true; // Date has been changed
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise generate one entry per extra date
|
||||||
|
for (int i=1; i<dates.size(); i++)
|
||||||
|
{
|
||||||
|
// Clone the properties
|
||||||
|
Map<String, Object> newResult = new HashMap<String, Object>(entryResult);
|
||||||
|
|
||||||
|
// Generate start and end based on this date
|
||||||
|
updateRepeatingStartEnd(dates.get(i), duration, newResult);
|
||||||
|
|
||||||
|
// Save as a new event
|
||||||
|
allResults.add(newResult);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Skip ignored instances
|
||||||
|
|
||||||
|
// New dates have been added
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateRepeatingStartEnd(Date newStart, long duration, Map<String, Object> result)
|
||||||
|
{
|
||||||
|
Date newEnd = new Date(newStart.getTime() + duration);
|
||||||
|
result.put(RESULT_START, newStart);
|
||||||
|
result.put(RESULT_END, newEnd);
|
||||||
|
}
|
||||||
|
}
|
@@ -41,6 +41,7 @@ import org.springframework.extensions.webscripts.WebScriptRequest;
|
|||||||
* This class is the controller for the slingshot calendar eventList.get webscript.
|
* This class is the controller for the slingshot calendar eventList.get webscript.
|
||||||
*
|
*
|
||||||
* TODO Improve what we give to the FTL
|
* TODO Improve what we give to the FTL
|
||||||
|
* TODO Switch to using {@link AbstractCalendarListingWebScript}
|
||||||
*
|
*
|
||||||
* @author Nick Burch
|
* @author Nick Burch
|
||||||
* @since 4.0
|
* @since 4.0
|
||||||
|
@@ -19,9 +19,7 @@
|
|||||||
package org.alfresco.repo.web.scripts.calendar;
|
package org.alfresco.repo.web.scripts.calendar;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Calendar;
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Comparator;
|
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -34,7 +32,6 @@ import org.alfresco.repo.security.authentication.AuthenticationUtil;
|
|||||||
import org.alfresco.repo.security.permissions.AccessDeniedException;
|
import org.alfresco.repo.security.permissions.AccessDeniedException;
|
||||||
import org.alfresco.service.cmr.calendar.CalendarEntry;
|
import org.alfresco.service.cmr.calendar.CalendarEntry;
|
||||||
import org.alfresco.service.cmr.calendar.CalendarEntryDTO;
|
import org.alfresco.service.cmr.calendar.CalendarEntryDTO;
|
||||||
import org.alfresco.service.cmr.calendar.CalendarRecurrenceHelper;
|
|
||||||
import org.alfresco.service.cmr.repository.NodeRef;
|
import org.alfresco.service.cmr.repository.NodeRef;
|
||||||
import org.alfresco.service.cmr.site.SiteInfo;
|
import org.alfresco.service.cmr.site.SiteInfo;
|
||||||
import org.json.simple.JSONObject;
|
import org.json.simple.JSONObject;
|
||||||
@@ -48,13 +45,8 @@ import org.springframework.extensions.webscripts.WebScriptRequest;
|
|||||||
* @author Nick Burch
|
* @author Nick Burch
|
||||||
* @since 4.0
|
* @since 4.0
|
||||||
*/
|
*/
|
||||||
public class UserCalendarEntriesGet extends AbstractCalendarWebScript
|
public class UserCalendarEntriesGet extends AbstractCalendarListingWebScript
|
||||||
{
|
{
|
||||||
private static final String RESULT_NAME = "name";
|
|
||||||
private static final String RESULT_TITLE = "title";
|
|
||||||
private static final String RESULT_START = "start";
|
|
||||||
private static final String RESULT_END = "end";
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Map<String, Object> executeImpl(WebScriptRequest req,
|
protected Map<String, Object> executeImpl(WebScriptRequest req,
|
||||||
Status status, Cache cache)
|
Status status, Cache cache)
|
||||||
@@ -206,31 +198,7 @@ public class UserCalendarEntriesGet extends AbstractCalendarWebScript
|
|||||||
// If the recurring events meant dates changed, re-sort
|
// If the recurring events meant dates changed, re-sort
|
||||||
if (resortNeeded)
|
if (resortNeeded)
|
||||||
{
|
{
|
||||||
Collections.sort(results, new Comparator<Map<String, Object>>()
|
Collections.sort(results, getEventDetailsSorter());
|
||||||
{
|
|
||||||
public int compare(Map<String, Object> resultA,
|
|
||||||
Map<String, Object> resultB)
|
|
||||||
{
|
|
||||||
Date startA = (Date)resultA.get(RESULT_START);
|
|
||||||
Date startB = (Date)resultB.get(RESULT_START);
|
|
||||||
|
|
||||||
int cmp = startA.compareTo(startB);
|
|
||||||
if (cmp == 0)
|
|
||||||
{
|
|
||||||
Date endA = (Date)resultA.get(RESULT_END);
|
|
||||||
Date endB = (Date)resultB.get(RESULT_END);
|
|
||||||
cmp = endA.compareTo(endB);
|
|
||||||
if (cmp == 0)
|
|
||||||
{
|
|
||||||
String nameA = (String)resultA.get(RESULT_NAME);
|
|
||||||
String nameB = (String)resultB.get(RESULT_NAME);
|
|
||||||
return nameA.compareTo(nameB);
|
|
||||||
}
|
|
||||||
return cmp;
|
|
||||||
}
|
|
||||||
return cmp;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// All done
|
// All done
|
||||||
@@ -298,102 +266,4 @@ public class UserCalendarEntriesGet extends AbstractCalendarWebScript
|
|||||||
|
|
||||||
return duration.toString();
|
return duration.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Do what's needed for recurring events.
|
|
||||||
*
|
|
||||||
* @return If dates have been tweaked, and a sort may be required
|
|
||||||
*/
|
|
||||||
private boolean handleRecurring(CalendarEntry entry, Map<String, Object> entryResult,
|
|
||||||
List<Map<String, Object>> allResults, Date from, Date until, boolean repeatingFirstOnly)
|
|
||||||
{
|
|
||||||
if (entry.getRecurrenceRule() == null)
|
|
||||||
{
|
|
||||||
// Nothing to do
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If no date is given, start looking for occurrences from the event itself
|
|
||||||
if (from == null)
|
|
||||||
{
|
|
||||||
from = entry.getStart();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Do we nee dto limit ourselves?
|
|
||||||
// Should we limit ourselves?
|
|
||||||
if (!repeatingFirstOnly)
|
|
||||||
{
|
|
||||||
if (until == null)
|
|
||||||
{
|
|
||||||
// If no end date was given, only allow repeating instances
|
|
||||||
// for next 60 days, to keep the list sane
|
|
||||||
// (It's normally only used for a month view anyway)
|
|
||||||
Calendar c = Calendar.getInstance();
|
|
||||||
c.setTime(from);
|
|
||||||
c.add(Calendar.DATE, 60);
|
|
||||||
until = c.getTime();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// How long is it?
|
|
||||||
long duration = entry.getEnd().getTime() - entry.getStart().getTime();
|
|
||||||
|
|
||||||
// Get it's recurring instances
|
|
||||||
List<Date> dates = CalendarRecurrenceHelper.getRecurrencesOnOrAfter(
|
|
||||||
entry, from, until, repeatingFirstOnly);
|
|
||||||
if (dates == null)
|
|
||||||
{
|
|
||||||
dates = new ArrayList<Date>();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add on the original event time itself if needed
|
|
||||||
if (entry.getStart().getTime() >= from.getTime())
|
|
||||||
{
|
|
||||||
if (dates.size() == 0 || dates.get(0).getTime() != entry.getStart().getTime())
|
|
||||||
{
|
|
||||||
// Original event is after the start time, and not on the recurring list
|
|
||||||
dates.add(0, entry.getStart());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we got no dates, then no recurrences in the period so zap
|
|
||||||
if (dates.size() == 0)
|
|
||||||
{
|
|
||||||
allResults.remove(entryResult);
|
|
||||||
return false; // Remains sorted despite delete
|
|
||||||
}
|
|
||||||
|
|
||||||
// Always update the live entry
|
|
||||||
updateRepeatingStartEnd(dates.get(0), duration, entryResult);
|
|
||||||
|
|
||||||
// If first result only, alter title and finish
|
|
||||||
if (repeatingFirstOnly)
|
|
||||||
{
|
|
||||||
entryResult.put(RESULT_TITLE, entry.getTitle() + " (Repeating)");
|
|
||||||
return true; // Date has been changed
|
|
||||||
}
|
|
||||||
|
|
||||||
// Otherwise generate one entry per extra date
|
|
||||||
for (int i=1; i<dates.size(); i++)
|
|
||||||
{
|
|
||||||
// Clone the properties
|
|
||||||
Map<String, Object> newResult = new HashMap<String, Object>(entryResult);
|
|
||||||
|
|
||||||
// Generate start and end based on this date
|
|
||||||
updateRepeatingStartEnd(dates.get(i), duration, newResult);
|
|
||||||
|
|
||||||
// Save as a new event
|
|
||||||
allResults.add(newResult);
|
|
||||||
}
|
|
||||||
|
|
||||||
// New dates have been added
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void updateRepeatingStartEnd(Date newStart, long duration, Map<String, Object> result)
|
|
||||||
{
|
|
||||||
Date newEnd = new Date(newStart.getTime() + duration);
|
|
||||||
result.put(RESULT_START, newStart);
|
|
||||||
result.put(RESULT_END, newEnd);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user