Make grouping by location an option, and set grid rows of event nodes.

Event nodes used to only be sorted. This commit explictly set their CSS 
grid rows.
This commit is contained in:
Daru13 2018-11-26 18:37:52 +01:00
parent cd771c5ba5
commit b43422dc50
2 changed files with 58 additions and 29 deletions

View file

@ -36,6 +36,9 @@ class Calendar {
this.onlyDisplaySubscribedEvents = calendarParameters.onlyDisplaySubscribedEvents !== undefined this.onlyDisplaySubscribedEvents = calendarParameters.onlyDisplaySubscribedEvents !== undefined
? calendarParameters.onlyDisplaySubscribedEvents ? calendarParameters.onlyDisplaySubscribedEvents
: false; : false;
this.groupEventsByLocation = calendarParameters.groupEventsByLocation !== undefined
? calendarParameters.groupEventsByLocation
: true;
this.eventDetailURLFormat = calendarParameters.eventDetailURLFormat !== undefined this.eventDetailURLFormat = calendarParameters.eventDetailURLFormat !== undefined
? calendarParameters.eventDetailURLFormat ? calendarParameters.eventDetailURLFormat
@ -71,7 +74,7 @@ class Calendar {
this.updateEventLocationStyleID(); this.updateEventLocationStyleID();
//this.sortEventNodesByEndTimeAndLocation(); //this.sortEventNodesByEndTimeAndLocation();
this.sortEventNodesByLocationAndIntervalGraphColoring(); this.sortEventNodesByIntervalGraphColoring();
this.updateCalendarNodeHeight(); this.updateCalendarNodeHeight();
this.initEventOverflowTooltips(); this.initEventOverflowTooltips();
@ -89,6 +92,7 @@ class Calendar {
this.updateEventVisibilities(); this.updateEventVisibilities();
this.updateCalendarNodeHeight(); this.updateCalendarNodeHeight();
this.sortEventNodesByIntervalGraphColoring();
this.startShowingEventOverflowTooltips(); this.startShowingEventOverflowTooltips();
} }
@ -102,6 +106,7 @@ class Calendar {
this.updateEventVisibilities(); this.updateEventVisibilities();
this.updateCalendarNodeHeight(); this.updateCalendarNodeHeight();
this.sortEventNodesByIntervalGraphColoring();
this.startShowingEventOverflowTooltips(); this.startShowingEventOverflowTooltips();
} }
@ -337,29 +342,36 @@ class Calendar {
// The following method requires the IntervalColoration class, // The following method requires the IntervalColoration class,
// which provides the algorithm (see interval_coloring.js) // which provides the algorithm (see interval_coloring.js)
sortEventNodesByLocationAndIntervalGraphColoring () { sortEventNodesByIntervalGraphColoring () {
// Group events by location let eventGroupIterator = [this.events];
let locationsToEvents = new Map();
for (let event of this.events) {
let location = event.location;
if (! locationsToEvents.has(location)) { // If required, group events by location
locationsToEvents.set(location, [event]); if (this.groupEventsByLocation) {
} let locationsToEvents = new Map();
else {
locationsToEvents for (let event of this.events) {
.get(location) let location = event.location;
.push(event);
if (! locationsToEvents.has(location)) {
locationsToEvents.set(location, [event]);
}
else {
locationsToEvents
.get(location)
.push(event);
}
} }
eventGroupIterator = locationsToEvents.values();
} }
// Assign a color to all events, // Assign a color to all events and a grid row to each event node,
// by using the interval graph coloration algorithm // by applying the interval graph coloration algorithm
// on each subset of events with the same location // to each subset of events with the same location
let allEventColors = []; let currentLocationFirstGridRow = 1;
let nbAlreadyColoredEvents = 0; let eventsToColors = new Map();
for (let eventsAtSameLocation of locationsToEvents.values()) { for (let eventsAtSameLocation of eventGroupIterator) {
// Build intervals for each event // Build intervals for each event
let intervals = []; let intervals = [];
for (let event of eventsAtSameLocation) { for (let event of eventsAtSameLocation) {
@ -369,26 +381,42 @@ class Calendar {
]); ]);
} }
// Get the graph coloring, and assign each color to each event // Get the graph coloring
let intervalGraphColors = new IntervalColoration(intervals).colors; let intervalGraphColors = new IntervalColoration(intervals).colors;
let eventsToColors = new Map();
for (let i = 0; i < eventsAtSameLocation; i++) { // Assign a color to each event, and a grid row to each event node
eventsToColors.set(event, intervalGraphColors[i]); let maximumColor = 0;
for (let i = 0; i < eventsAtSameLocation.length; i++) {
let event = eventsAtSameLocation[i];
let color = intervalGraphColors[i];
eventsToColors.set(event, color);
if (color > maximumColor) {
maximumColor = color;
}
event.node.css({
"grid-row-start": `${currentLocationFirstGridRow + color}`,
"grid-row-end" : `${currentLocationFirstGridRow + color}`
});
} }
// Update the start row of the next location
currentLocationFirstGridRow += 1 + maximumColor;
// Sort the events by color // Sort the events by color
eventsAtSameLocation.sort((event1, event2) => { eventsAtSameLocation.sort((event1, event2) => {
return eventsToColors.get(event2) - eventsToColors.get(event1); return eventsToColors.get(event1) - eventsToColors.get(event2);
}); });
} }
// Finally sort all event nodes, // Finally sort all event nodes (by (1) location and (2) color)
// (1) by their location and (2) by their color
// Note: the container is detached from the DOM for better performances // Note: the container is detached from the DOM for better performances
this.eventContainerNode.detach(); this.eventContainerNode.detach();
for (let eventsAtSameLocation of locationsToEvents.values()) { for (let eventsAtSameLocation of eventGroupIterator) {
for (let event of eventsAtSameLocation) { for (let event of eventsAtSameLocation) {
this.eventContainerNode.prepend(event.node); this.eventContainerNode.prepend(event.node);
} }

View file

@ -36,7 +36,8 @@
endDate: new Date(2018, 11, 2, 6), endDate: new Date(2018, 11, 2, 6),
eventDetailURLFormat: "https://cof.ens.fr/poulpe/event/activity/999999", eventDetailURLFormat: "https://cof.ens.fr/poulpe/event/activity/999999",
subscriptionURLFormat: "{% url "event:enrol_activity" 999999 %}?ajax=json", subscriptionURLFormat: "{% url "event:enrol_activity" 999999 %}?ajax=json",
csrfToken: $(".planning [name=csrfmiddlewaretoken]").val() csrfToken: $(".planning [name=csrfmiddlewaretoken]").val(),
groupEventsByLocation: true
}); });
// TODO: move this elsewhere // TODO: move this elsewhere