shiftsFilter = $shiftsFilter; $this->firstBlockStartTime = $this->calcFirstBlockStartTime($shifts); $this->lastBlockEndTime = $this->calcLastBlockEndTime($shifts); $this->lanes = $this->assignShiftsToLanes($shifts); } /** * Assigns the shifts to different lanes per room if they collide * * @param Shift[] $shifts The shifts to assign * @return array 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 = []; foreach ($shifts as $shift) { $room = $shift->room; $header = Room_name_render($room); if (!isset($lanes[$room->id])) { // initialize room with one lane $lanes[$room->id] = [ new ShiftCalendarLane($header), ]; } // Try to add the shift to the existing lanes for this room $shift_added = false; foreach ($lanes[$room->id] as $lane) { /** @var ShiftCalendarLane $lane */ if ($lane->shiftFits($shift)) { $lane->addShift($shift); $shift_added = true; break; } } // If all lanes for this room are busy, create a new lane and add shift to it if (!$shift_added) { $newLane = new ShiftCalendarLane($header); $newLane->addShift($shift); $lanes[$room->id][] = $newLane; } } return $lanes; } /** * @return int */ public function getFirstBlockStartTime() { return $this->firstBlockStartTime; } /** * @return int */ public function getLastBlockEndTime() { return $this->lastBlockEndTime; } /** * @return float */ public function getBlocksPerSlot() { if (is_null($this->blocksPerSlot)) { $this->blocksPerSlot = $this->calcBlocksPerSlot(); } return $this->blocksPerSlot; } /** * Renders the whole calendar * * @return string the generated html */ public function render() { if (count($this->lanes) == 0) { return info(__('No shifts found.'), true); } return div('shift-calendar table-responsive', [ $this->renderTimeLane(), $this->renderShiftLanes(), ]) . $this->renderLegend(); } /** * Renders the lanes containing the shifts * * @return string */ private function renderShiftLanes() { $html = ''; foreach ($this->lanes as $room_lanes) { foreach ($room_lanes as $lane) { $html .= $this->renderLane($lane); } } return $html; } /** * Renders a single lane * * @param ShiftCalendarLane $lane The lane to render * @return string */ private function renderLane(ShiftCalendarLane $lane) { $shift_renderer = new ShiftCalendarShiftRenderer(); $html = ''; $rendered_until = $this->getFirstBlockStartTime(); foreach ($lane->getShifts() as $shift) { while ($rendered_until + ShiftCalendarRenderer::SECONDS_PER_ROW <= $shift->start->timestamp) { $html .= $this->renderTick($rendered_until); $rendered_until += ShiftCalendarRenderer::SECONDS_PER_ROW; } list ($shift_height, $shift_html) = $shift_renderer->render( $shift, $this->needed_angeltypes[$shift->id], $this->shift_entries[$shift->id], auth()->user() ); $html .= $shift_html; $rendered_until += $shift_height * ShiftCalendarRenderer::SECONDS_PER_ROW; } while ($rendered_until < $this->getLastBlockEndTime()) { $html .= $this->renderTick($rendered_until); $rendered_until += ShiftCalendarRenderer::SECONDS_PER_ROW; } $bg = 'bg-' . theme_type(); return div('lane', [ div('header ' . $bg, $lane->getHeader()), $html, ]); } /** * Renders a tick/block for given time * * @param int $time unix timestamp * @param boolean $label Should time labels be generated? * @return string rendered tick html */ private function renderTick($time, $label = false) { $time = Carbon::createFromTimestamp($time); $class = $label ? 'tick bg-' . theme_type() : 'tick '; if ($time->isStartOfDay()) { if (!$label) { return div($class . ' day'); } return div($class . ' day', [ $time->format(__('m-d')) . '
' . $time->format(__('H:i')), ]); } elseif ($time->isStartOfHour()) { if (!$label) { return div($class . ' hour'); } return div($class . ' hour', [ $time->format(__('m-d')) . '
' . $time->format(__('H:i')), ]); } return div($class); } /** * Renders the left time lane including hour/day ticks * * @return string */ private function renderTimeLane() { $bg = 'bg-' . theme_type(); $time_slot = [ div('header ' . $bg, [ __('log.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); } /** * @param Shift[] $shifts * @return int */ private function calcFirstBlockStartTime($shifts) { $start_time = $this->shiftsFilter->getEndTime(); foreach ($shifts as $shift) { if ($shift->start->timestamp < $start_time) { $start_time = $shift->start->timestamp; } } return ShiftCalendarRenderer::SECONDS_PER_ROW * floor( ($start_time - ShiftCalendarRenderer::TIME_MARGIN) / ShiftCalendarRenderer::SECONDS_PER_ROW ); } /** * @param Shift[] $shifts * @return int */ private function calcLastBlockEndTime($shifts) { $end_time = $this->shiftsFilter->getStartTime(); foreach ($shifts as $shift) { if ($shift->end->timestamp > $end_time) { $end_time = $shift->end->timestamp; } } return ShiftCalendarRenderer::SECONDS_PER_ROW * ceil( ($end_time + ShiftCalendarRenderer::TIME_MARGIN) / ShiftCalendarRenderer::SECONDS_PER_ROW ); } /** * @return int */ private function calcBlocksPerSlot() { return ceil( ($this->getLastBlockEndTime() - $this->getFirstBlockStartTime()) / ShiftCalendarRenderer::SECONDS_PER_ROW ); } /** * Renders a legend explaining the shift coloring * * @return string */ private function renderLegend() { return div('legend mt-3', [ badge(__('Your shift'), 'primary'), badge(__('Help needed'), 'danger'), badge(__('Other angeltype needed / collides with my shifts'), 'warning'), badge(__('Shift is full'), 'success'), badge(__('Shift is running/ended or you have not arrived'), 'secondary'), ]); } }