2016-10-03 17:41:14 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Engelsystem;
|
|
|
|
|
|
|
|
class ShiftCalendarRenderer {
|
|
|
|
|
2016-10-05 22:28:39 +02:00
|
|
|
/**
|
|
|
|
* 15m * 60s/m = 900s
|
|
|
|
*/
|
2016-11-05 10:06:06 +01:00
|
|
|
const SECONDS_PER_ROW = 900;
|
2016-10-05 22:28:39 +02:00
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
/**
|
|
|
|
* Height of a block in pixel.
|
|
|
|
* Do not change - corresponds with theme/css
|
|
|
|
*/
|
|
|
|
const BLOCK_HEIGHT = 30;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Distance between two shifts in pixels
|
|
|
|
*/
|
|
|
|
const MARGIN = 5;
|
2016-11-15 21:33:54 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Seconds added to the start and end time
|
|
|
|
*/
|
|
|
|
const TIME_MARGIN = 1800;
|
2016-10-07 17:22:48 +02:00
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
private $lanes;
|
2016-10-05 22:28:39 +02:00
|
|
|
|
|
|
|
private $shiftsFilter;
|
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
private $firstBlockStartTime = null;
|
2016-11-14 17:58:15 +01:00
|
|
|
|
2016-11-14 17:40:24 +01:00
|
|
|
private $lastBlockEndTime = null;
|
2016-11-05 10:06:06 +01:00
|
|
|
|
|
|
|
private $blocksPerSlot = null;
|
|
|
|
|
2016-10-05 22:28:39 +02:00
|
|
|
public function __construct($shifts, ShiftsFilter $shiftsFilter) {
|
|
|
|
$this->shiftsFilter = $shiftsFilter;
|
2016-11-05 10:06:06 +01:00
|
|
|
$this->firstBlockStartTime = $this->calcFirstBlockStartTime($shifts);
|
2016-11-14 17:40:24 +01:00
|
|
|
$this->lastBlockEndTime = $this->calcLastBlockEndTime($shifts);
|
2016-11-05 10:06:06 +01:00
|
|
|
$this->lanes = $this->assignShiftsToLanes($shifts);
|
2016-10-05 22:28:39 +02:00
|
|
|
}
|
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
/**
|
|
|
|
* Assigns the shifts to different lanes per room if they collide
|
|
|
|
*
|
|
|
|
* @param Shift[] $shifts
|
|
|
|
* The shifts to assign
|
|
|
|
*
|
|
|
|
* @return Returns an array that assigns a room_id to an array of ShiftCalendarLane containing the shifts
|
|
|
|
*/
|
|
|
|
private function assignShiftsToLanes($shifts) {
|
|
|
|
// array that assigns a room id to a list of lanes (per room)
|
|
|
|
$lanes = [];
|
2016-10-07 17:22:48 +02:00
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
foreach ($shifts as $shift) {
|
|
|
|
$room_id = $shift['RID'];
|
|
|
|
if (! isset($lanes[$room_id])) {
|
|
|
|
// initialize room with one lane
|
|
|
|
$header = Room_name_render([
|
|
|
|
'RID' => $room_id,
|
|
|
|
'Name' => $shift['room_name']
|
|
|
|
]);
|
|
|
|
$lanes[$room_id] = [
|
|
|
|
new ShiftCalendarLane($header, $this->getFirstBlockStartTime(), $this->getBlocksPerSlot())
|
|
|
|
];
|
|
|
|
}
|
|
|
|
// Try to add the shift to the existing lanes for this room
|
|
|
|
$shift_added = false;
|
|
|
|
foreach ($lanes[$room_id] as $lane) {
|
|
|
|
$shift_added = $lane->addShift($shift);
|
|
|
|
if ($shift_added == true) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If all lanes for this room are busy, create a new lane and add shift to it
|
|
|
|
if ($shift_added == false) {
|
2016-11-15 21:33:54 +01:00
|
|
|
$newLane = new ShiftCalendarLane($header, $this->getFirstBlockStartTime(), $this->getBlocksPerSlot());
|
2016-11-05 10:06:06 +01:00
|
|
|
if (! $newLane->addShift($shift)) {
|
|
|
|
engelsystem_error("Unable to add shift to new lane.");
|
|
|
|
}
|
|
|
|
$lanes[$room_id][] = $newLane;
|
|
|
|
}
|
|
|
|
}
|
2016-10-07 17:22:48 +02:00
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
return $lanes;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getFirstBlockStartTime() {
|
|
|
|
return $this->firstBlockStartTime;
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
2016-11-14 17:58:15 +01:00
|
|
|
|
2016-11-14 17:40:24 +01:00
|
|
|
public function getLastBlockEndTime() {
|
|
|
|
return $this->lastBlockEndTime;
|
|
|
|
}
|
2016-10-07 17:22:48 +02:00
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
public function getBlocksPerSlot() {
|
|
|
|
if ($this->blocksPerSlot == null) {
|
|
|
|
$this->blocksPerSlot = $this->calcBlocksPerSlot();
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
2016-11-05 10:06:06 +01:00
|
|
|
return $this->blocksPerSlot;
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
/**
|
|
|
|
* Renders the whole calendar
|
|
|
|
*
|
|
|
|
* @return the generated html
|
|
|
|
*/
|
|
|
|
public function render() {
|
|
|
|
return div('shift-calendar', [
|
|
|
|
$this->renderTimeLane(),
|
|
|
|
$this->renderShiftLanes()
|
2016-11-14 17:58:15 +01:00
|
|
|
]) . $this->renderLegend();
|
2016-11-05 10:06:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders the lanes containing the shifts
|
|
|
|
*/
|
|
|
|
private function renderShiftLanes() {
|
|
|
|
$html = "";
|
2016-11-07 20:50:15 +01:00
|
|
|
foreach ($this->lanes as $room_lanes) {
|
2016-11-05 10:06:06 +01:00
|
|
|
foreach ($room_lanes as $lane) {
|
|
|
|
$html .= $this->renderLane($lane);
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
|
|
|
}
|
2016-11-05 10:06:06 +01:00
|
|
|
return $html;
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
/**
|
|
|
|
* Renders a single lane
|
|
|
|
*
|
|
|
|
* @param ShiftCalendarLane $lane
|
|
|
|
* The lane to render
|
|
|
|
*/
|
|
|
|
private function renderLane(ShiftCalendarLane $lane) {
|
2016-11-12 23:00:46 +01:00
|
|
|
global $user;
|
|
|
|
|
2016-11-07 20:50:15 +01:00
|
|
|
$shift_renderer = new ShiftCalendarShiftRenderer();
|
2016-11-05 10:06:06 +01:00
|
|
|
$html = "";
|
|
|
|
$rendered_until = $this->getFirstBlockStartTime();
|
|
|
|
foreach ($lane->getShifts() as $shift) {
|
|
|
|
while ($rendered_until + ShiftCalendarRenderer::SECONDS_PER_ROW <= $shift['start']) {
|
|
|
|
$html .= $this->renderTick($rendered_until);
|
|
|
|
$rendered_until += ShiftCalendarRenderer::SECONDS_PER_ROW;
|
|
|
|
}
|
|
|
|
|
2016-11-12 23:00:46 +01:00
|
|
|
list($shift_height, $shift_html) = $shift_renderer->render($shift, $user);
|
2016-11-05 10:06:06 +01:00
|
|
|
$html .= $shift_html;
|
|
|
|
$rendered_until += $shift_height * ShiftCalendarRenderer::SECONDS_PER_ROW;
|
|
|
|
}
|
2016-11-14 17:40:24 +01:00
|
|
|
while ($rendered_until < $this->getLastBlockEndTime()) {
|
2016-11-05 10:06:06 +01:00
|
|
|
$html .= $this->renderTick($rendered_until);
|
|
|
|
$rendered_until += ShiftCalendarRenderer::SECONDS_PER_ROW;
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
return div('lane', [
|
|
|
|
div('header', $lane->getHeader()),
|
|
|
|
$html
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders a tick/block for given time
|
|
|
|
*
|
|
|
|
* @param int $time
|
|
|
|
* unix timestamp
|
|
|
|
* @param boolean $label
|
|
|
|
* Should time labels be generated?
|
|
|
|
* @return rendered tick html
|
|
|
|
*/
|
|
|
|
private function renderTick($time, $label = false) {
|
|
|
|
if ($time % (24 * 60 * 60) == 23 * 60 * 60) {
|
|
|
|
if (! $label) {
|
|
|
|
return div('tick day');
|
|
|
|
}
|
|
|
|
return div('tick day', [
|
2016-11-15 21:33:54 +01:00
|
|
|
date('m-d<b\r />H:i', $time)
|
2016-11-05 10:06:06 +01:00
|
|
|
]);
|
|
|
|
} elseif ($time % (60 * 60) == 0) {
|
|
|
|
if (! $label) {
|
|
|
|
return div('tick hour');
|
|
|
|
}
|
|
|
|
return div('tick hour', [
|
|
|
|
date('H:i', $time)
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
return div('tick');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders the left time lane including hour/day ticks
|
|
|
|
*/
|
|
|
|
private function renderTimeLane() {
|
|
|
|
$time_slot = [
|
|
|
|
div('header', [
|
|
|
|
_("Time")
|
|
|
|
])
|
|
|
|
];
|
|
|
|
for ($block = 0; $block < $this->getBlocksPerSlot(); $block ++) {
|
|
|
|
$thistime = $this->getFirstBlockStartTime() + ($block * ShiftCalendarRenderer::SECONDS_PER_ROW);
|
|
|
|
$time_slot[] = $this->renderTick($thistime, true);
|
|
|
|
}
|
|
|
|
return div('lane time', $time_slot);
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
private function calcFirstBlockStartTime($shifts) {
|
2016-10-07 17:22:48 +02:00
|
|
|
$start_time = $this->shiftsFilter->getEndTime();
|
2016-11-05 10:06:06 +01:00
|
|
|
foreach ($shifts as $shift) {
|
2016-10-07 17:22:48 +02:00
|
|
|
if ($shift['start'] < $start_time) {
|
|
|
|
$start_time = $shift['start'];
|
|
|
|
}
|
|
|
|
}
|
2016-11-15 21:33:54 +01:00
|
|
|
return ShiftCalendarRenderer::SECONDS_PER_ROW * floor(($start_time - ShiftCalendarRenderer::TIME_MARGIN) / ShiftCalendarRenderer::SECONDS_PER_ROW);
|
2016-10-07 17:22:48 +02:00
|
|
|
}
|
|
|
|
|
2016-11-14 17:40:24 +01:00
|
|
|
private function calcLastBlockEndTime($shifts) {
|
|
|
|
$end_time = $this->shiftsFilter->getStartTime();
|
|
|
|
foreach ($shifts as $shift) {
|
|
|
|
if ($shift['end'] > $end_time) {
|
|
|
|
$end_time = $shift['end'];
|
|
|
|
}
|
|
|
|
}
|
2016-11-15 21:33:54 +01:00
|
|
|
return ShiftCalendarRenderer::SECONDS_PER_ROW * ceil(($end_time + ShiftCalendarRenderer::TIME_MARGIN) / ShiftCalendarRenderer::SECONDS_PER_ROW);
|
2016-11-14 17:40:24 +01:00
|
|
|
}
|
|
|
|
|
2016-11-05 10:06:06 +01:00
|
|
|
private function calcBlocksPerSlot() {
|
2016-11-14 17:40:24 +01:00
|
|
|
return ceil(($this->getLastBlockEndTime() - $this->getFirstBlockStartTime()) / ShiftCalendarRenderer::SECONDS_PER_ROW);
|
2016-10-03 17:41:14 +02:00
|
|
|
}
|
2016-11-14 17:58:15 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders a legend explaining the shift coloring
|
|
|
|
*/
|
|
|
|
private function renderLegend() {
|
|
|
|
return div('legend', [
|
|
|
|
label(_('Your shift'), 'primary'),
|
|
|
|
label(_('Help needed'), 'danger'),
|
|
|
|
label(_('Other angeltype needed / collides with my shifts'), 'warning'),
|
|
|
|
label(_('Shift is full'), 'success'),
|
|
|
|
label(_('Shift running/ended'), 'default')
|
|
|
|
]);
|
|
|
|
}
|
2016-10-03 17:41:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
?>
|