engelsystem/includes/pages/user_shifts.php

427 lines
14 KiB
PHP
Raw Permalink Normal View History

2011-07-13 14:30:19 +02:00
<?php
use Engelsystem\Database\Db;
use Engelsystem\Helpers\Carbon;
2022-11-09 00:02:30 +01:00
use Engelsystem\Models\AngelType;
2023-10-15 19:25:55 +02:00
use Engelsystem\Models\Location;
use Engelsystem\Models\Shifts\NeededAngelType;
use Engelsystem\Models\Shifts\Shift;
2022-12-03 00:57:04 +01:00
use Engelsystem\Models\UserAngelType;
2016-10-02 23:00:01 +02:00
use Engelsystem\ShiftsFilter;
2022-12-03 00:57:04 +01:00
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Collection as EloquentCollection;
2020-09-06 23:50:36 +02:00
use Illuminate\Support\Collection;
2014-09-28 19:44:53 +02:00
2017-01-03 03:22:48 +01:00
/**
* @return string
*/
2017-01-02 03:57:23 +01:00
function shifts_title()
{
2024-03-24 15:05:24 +01:00
return __('general.shifts');
2013-11-25 21:04:58 +01:00
}
/**
* Start different controllers for deleting shifts and shift_entries, edit shifts and add shift entries.
* FIXME:
* Transform into shift controller and shift entry controller.
* Split actions into shift edit, shift delete, shift entry edit, shift entry delete
* Introduce simpler and beautiful actions for shift entry join/leave for users
2017-01-03 03:22:48 +01:00
*
* @return string
*/
2017-01-02 03:57:23 +01:00
function user_shifts()
{
$request = request();
2017-01-02 15:43:36 +01:00
2023-01-18 13:02:11 +01:00
if (auth()->user()->isFreeloader()) {
2023-11-13 16:56:52 +01:00
throw_redirect(url('/user-myshifts'));
2017-01-02 03:57:23 +01:00
}
2017-01-02 15:43:36 +01:00
if ($request->has('edit_shift')) {
2017-01-02 15:43:36 +01:00
return shift_edit_controller();
} elseif ($request->has('delete_shift')) {
2017-01-02 15:43:36 +01:00
return shift_delete_controller();
}
2017-01-02 03:57:23 +01:00
return view_user_shifts();
2011-12-28 14:45:49 +01:00
}
2016-10-04 18:36:57 +02:00
/**
* Helper function that updates the start and end time from request data.
* Use update_ShiftsFilter().
*
2017-01-03 03:22:48 +01:00
* @param ShiftsFilter $shiftsFilter The shiftfilter to update.
* @param string[] $days
2016-10-04 18:36:57 +02:00
*/
2017-01-02 03:57:23 +01:00
function update_ShiftsFilter_timerange(ShiftsFilter $shiftsFilter, $days)
{
$start_time = $shiftsFilter->getStartTime();
2018-01-14 17:47:26 +01:00
if (is_null($start_time)) {
$now = (new DateTime())->format('Y-m-d');
$first_day = DateTime::createFromFormat(
'Y-m-d',
in_array($now, $days) ? $now : ($days[0] ?? (new DateTime())->format('Y-m-d'))
)->getTimestamp();
2022-04-13 01:02:37 +02:00
if (time() < $first_day) {
$start_time = $first_day;
} else {
$start_time = time();
}
2017-01-02 03:57:23 +01:00
}
2017-01-02 15:43:36 +01:00
2017-01-02 03:57:23 +01:00
$end_time = $shiftsFilter->getEndTime();
if (is_null($end_time)) {
2017-01-02 03:57:23 +01:00
$end_time = $start_time + 24 * 60 * 60;
$end = Carbon::createFromTimestamp($end_time);
if (!in_array($end->format('Y-m-d'), $days)) {
$end->startOfDay()->subSecond(); // the day before
$end_time = $end->timestamp;
}
2017-01-02 03:57:23 +01:00
}
2017-01-02 15:43:36 +01:00
2017-12-25 23:12:52 +01:00
$shiftsFilter->setStartTime(check_request_datetime(
'start_day',
'start_time',
$days,
$start_time
));
$shiftsFilter->setEndTime(check_request_datetime(
'end_day',
'end_time',
$days,
$end_time
));
2017-01-02 15:43:36 +01:00
2017-01-02 03:57:23 +01:00
if ($shiftsFilter->getStartTime() > $shiftsFilter->getEndTime()) {
$shiftsFilter->setEndTime($shiftsFilter->getStartTime() + 24 * 60 * 60);
}
2016-10-04 18:36:57 +02:00
}
2016-10-02 23:00:01 +02:00
/**
* Update given ShiftsFilter with filter params from user input
*
2022-12-03 00:57:04 +01:00
* @param ShiftsFilter $shiftsFilter The shifts filter to update from request data
* @param boolean $user_shifts_admin Has the user user_shift_admin privilege?
2022-12-03 00:57:04 +01:00
* @param string[] $days An array of available filter days
2016-10-02 23:00:01 +02:00
*/
2017-01-02 03:57:23 +01:00
function update_ShiftsFilter(ShiftsFilter $shiftsFilter, $user_shifts_admin, $days)
{
$shiftsFilter->setUserShiftsAdmin($user_shifts_admin);
$shiftsFilter->setFilled(check_request_int_array('filled', $shiftsFilter->getFilled()));
2023-10-15 19:25:55 +02:00
$shiftsFilter->setLocations(check_request_int_array('locations', $shiftsFilter->getLocations()));
2017-01-02 03:57:23 +01:00
$shiftsFilter->setTypes(check_request_int_array('types', $shiftsFilter->getTypes()));
update_ShiftsFilter_timerange($shiftsFilter, $days);
2016-10-02 23:00:01 +02:00
}
2017-01-03 03:22:48 +01:00
/**
2023-10-15 19:25:55 +02:00
* @return Location[]|Collection
2017-01-03 03:22:48 +01:00
*/
2023-10-15 19:25:55 +02:00
function load_locations(bool $onlyWithActiveShifts = false)
2017-01-02 03:57:23 +01:00
{
2023-10-15 19:25:55 +02:00
$locations = Location::orderBy('name');
if ($onlyWithActiveShifts) {
2023-10-15 19:25:55 +02:00
$locationIdsFromAngelType = NeededAngelType::query()
->whereNotNull('location_id')
->select('location_id');
2023-10-15 19:25:55 +02:00
$locationIdsFromShift = Shift::query()
->leftJoin('needed_angel_types', 'shifts.id', 'needed_angel_types.shift_id')
->leftJoin('needed_angel_types AS nast', 'shifts.shift_type_id', 'nast.shift_type_id')
->whereNotNull('needed_angel_types.id')
->orWhereNotNull('nast.id')
2023-10-15 19:25:55 +02:00
->select('shifts.location_id');
2023-10-15 19:25:55 +02:00
$locations->whereIn('id', $locationIdsFromAngelType)
->orWhereIn('id', $locationIdsFromShift);
}
2023-10-15 19:25:55 +02:00
$locations = $locations->get();
2023-10-15 19:25:55 +02:00
if ($locations->isEmpty()) {
2023-10-13 11:53:13 +02:00
error(__('The administration has not configured any locations yet.'));
2023-11-13 16:56:52 +01:00
throw_redirect(url('/'));
2017-01-02 03:57:23 +01:00
}
2020-09-06 23:50:36 +02:00
2023-10-15 19:25:55 +02:00
return $locations;
2016-10-04 18:36:57 +02:00
}
2017-01-03 03:22:48 +01:00
/**
* @return array
*/
2017-01-02 03:57:23 +01:00
function load_days()
{
2022-06-16 22:50:52 +02:00
$days = (new Collection(Db::select(
2021-01-03 01:47:39 +01:00
'
2023-01-03 22:19:03 +01:00
SELECT DISTINCT DATE(`start`) AS `id`, DATE(`start`) AS `name`
FROM `shifts`
2021-01-03 01:47:39 +01:00
ORDER BY `id`, `name`
'
)))
->pluck('id')
->toArray();
if (empty($days)) {
error(__('The administration has not configured any shifts yet.'));
2019-03-07 13:01:52 +01:00
// Do not try to redirect to the current page
if (config('home_site') != 'user_shifts') {
2023-11-13 16:56:52 +01:00
throw_redirect(url('/'));
2019-03-07 13:01:52 +01:00
}
2017-01-02 03:57:23 +01:00
}
return $days;
2016-10-04 18:36:57 +02:00
}
2017-01-03 03:22:48 +01:00
/**
2017-12-25 23:12:52 +01:00
* @return array[]|false
2017-01-03 03:22:48 +01:00
*/
2017-01-02 03:57:23 +01:00
function load_types()
{
2018-10-31 12:48:22 +01:00
$user = auth()->user();
$isShico = auth()->can('admin_shifts');
2017-01-02 15:43:36 +01:00
2022-11-09 00:02:30 +01:00
if (!AngelType::count()) {
error(__('The administration has not configured any angeltypes yet - or you are not subscribed to any angeltype.'));
2023-11-13 16:56:52 +01:00
throw_redirect(url('/'));
2017-01-02 03:57:23 +01:00
}
2022-12-03 00:57:04 +01:00
2022-10-18 19:15:22 +02:00
$types = Db::select(
'
SELECT
2022-11-09 00:02:30 +01:00
`angel_types`.`id`,
`angel_types`.`name`,
(
2022-11-09 00:02:30 +01:00
`angel_types`.`restricted`=0
OR (
2022-12-03 00:57:04 +01:00
NOT `user_angel_type`.`confirm_user_id` IS NULL
OR `user_angel_type`.`id` IS NULL
)
) AS `enabled`
2022-11-09 00:02:30 +01:00
FROM `angel_types`
2022-12-03 00:57:04 +01:00
LEFT JOIN `user_angel_type`
ON (
2022-12-03 00:57:04 +01:00
`user_angel_type`.`angel_type_id`=`angel_types`.`id`
AND `user_angel_type`.`user_id`=?
)'
. ($isShico ? '' :
'WHERE angel_types.hide_on_shift_view = 0
OR user_angel_type.user_id IS NOT NULL ') .
'ORDER BY `angel_types`.`name`
',
[
2018-10-08 21:15:56 +02:00
$user->id,
]
);
2022-12-03 00:57:04 +01:00
2017-01-02 03:57:23 +01:00
if (empty($types)) {
2020-11-24 01:18:05 +01:00
return unrestricted_angeltypes();
2017-01-02 03:57:23 +01:00
}
2022-12-03 00:57:04 +01:00
2017-01-02 03:57:23 +01:00
return $types;
2016-10-04 18:36:57 +02:00
}
2020-11-24 01:18:05 +01:00
/**
* @return array[]
*/
function unrestricted_angeltypes()
{
2022-11-09 00:02:30 +01:00
return AngelType::whereRestricted(0)->get(['id', 'name'])->toArray();
2020-11-24 01:18:05 +01:00
}
2017-01-03 03:22:48 +01:00
/**
* @return string
*/
2017-01-02 03:57:23 +01:00
function view_user_shifts()
{
2018-10-10 03:10:28 +02:00
$user = auth()->user();
2017-01-02 15:43:36 +01:00
2017-08-30 19:57:01 +02:00
$session = session();
2017-01-02 03:57:23 +01:00
$days = load_days();
2023-10-15 19:25:55 +02:00
$locations = load_locations(true);
2017-01-02 03:57:23 +01:00
$types = load_types();
2023-04-01 15:14:32 +02:00
$ownAngelTypes = [];
2022-12-03 00:57:04 +01:00
/** @var EloquentCollection|UserAngelType[] $userAngelTypes */
$userAngelTypes = UserAngelType::whereUserId($user->id)
->leftJoin('angel_types', 'user_angel_type.angel_type_id', 'angel_types.id')
->where(function (Builder $query) {
$query->whereNotNull('user_angel_type.confirm_user_id')
->orWhere('angel_types.restricted', false);
})
->get();
foreach ($userAngelTypes as $type) {
2023-04-01 15:14:32 +02:00
$ownAngelTypes[] = $type->angel_type_id;
}
2017-01-02 15:43:36 +01:00
if (!$session->has('shifts-filter')) {
2023-10-15 19:25:55 +02:00
$location_ids = $locations->pluck('id')->toArray();
$shiftsFilter = new ShiftsFilter(auth()->can('user_shifts_admin'), $location_ids, $ownAngelTypes);
$session->set('shifts-filter', $shiftsFilter->sessionExport());
2017-01-02 03:57:23 +01:00
}
2017-08-30 19:57:01 +02:00
$shiftsFilter = new ShiftsFilter();
$shiftsFilter->sessionImport($session->get('shifts-filter'));
update_ShiftsFilter($shiftsFilter, auth()->can('user_shifts_admin'), $days);
$session->set('shifts-filter', $shiftsFilter->sessionExport());
2017-01-02 15:43:36 +01:00
2017-01-02 03:57:23 +01:00
$shiftCalendarRenderer = shiftCalendarRendererByShiftFilter($shiftsFilter);
2017-01-02 15:43:36 +01:00
2018-10-10 03:10:28 +02:00
if (empty($user->api_key)) {
auth()->resetApiKey($user);
2017-01-02 03:57:23 +01:00
}
2017-01-02 15:43:36 +01:00
2017-01-02 03:57:23 +01:00
$filled = [
2017-01-02 15:43:36 +01:00
[
'id' => '1',
'name' => __('occupied'),
2017-01-02 15:43:36 +01:00
],
[
'id' => '0',
'name' => __('free'),
],
2017-01-02 15:43:36 +01:00
];
2023-01-03 22:19:03 +01:00
$start_day = $shiftsFilter->getStart()->format('Y-m-d');
$start_time = $shiftsFilter->getStart()->format('H:i');
$end_day = $shiftsFilter->getEnd()->format('Y-m-d');
$end_time = $shiftsFilter->getEnd()->format('H:i');
2017-01-02 15:43:36 +01:00
$canSignUpForShifts = true;
2018-10-10 03:10:28 +02:00
if (config('signup_requires_arrival') && !$user->state->arrived) {
$canSignUpForShifts = false;
info(render_user_arrived_hint());
}
2023-02-04 02:43:47 +01:00
$formattedDays = collect($days)->map(function ($value) {
return dateWithEventDay(Carbon::make($value)->format('Y-m-d'));
2023-02-04 02:43:47 +01:00
})->toArray();
2023-11-13 16:56:52 +01:00
$link = button(url('/admin-shifts'), icon('plus-lg'), 'add');
2017-01-02 03:57:23 +01:00
return page([
2017-01-02 15:43:36 +01:00
div('col-md-12', [
view(__DIR__ . '/../../resources/views/pages/user-shifts.html', [
2017-01-02 15:43:36 +01:00
'title' => shifts_title(),
'add_link' => auth()->can('admin_shifts') ? $link : '',
2023-10-15 19:25:55 +02:00
'location_select' => make_select(
$locations,
$shiftsFilter->getLocations(),
'locations',
2023-10-13 11:53:13 +02:00
icon('pin-map-fill') . __('Locations')
2022-12-02 23:03:23 +01:00
),
2017-12-25 23:12:52 +01:00
'start_select' => html_select_key(
'start_day',
'start_day',
2023-02-04 02:43:47 +01:00
array_combine($days, $formattedDays),
2017-12-25 23:12:52 +01:00
$start_day
),
2017-01-02 15:43:36 +01:00
'start_time' => $start_time,
2017-12-25 23:12:52 +01:00
'end_select' => html_select_key(
'end_day',
'end_day',
2023-02-04 02:43:47 +01:00
array_combine($days, $formattedDays),
2017-12-25 23:12:52 +01:00
$end_day
),
2017-01-02 15:43:36 +01:00
'end_time' => $end_time,
'type_select' => make_select(
$types,
$shiftsFilter->getTypes(),
2017-01-03 14:12:17 +01:00
'types',
icon('person-lines-fill') . __('angeltypes.angeltypes')
. ' <small><span class="bi bi-info-circle-fill text-info" data-bs-toggle="tooltip" title="'
. __('The tasks shown here are influenced by the angeltypes you joined already!')
. '"></span></small>',
2023-04-01 15:14:32 +02:00
$ownAngelTypes
2017-01-02 15:43:36 +01:00
),
2022-12-02 23:03:23 +01:00
'filled_select' => make_select(
$filled,
$shiftsFilter->getFilled(),
'filled',
icon('person-fill-slash') . __('Occupancy')
),
2017-01-02 15:43:36 +01:00
'shifts_table' => msg() . $shiftCalendarRenderer->render(),
'ical_text' => div('mt-3', ical_hint()),
'filter' => __('Filter'),
'filter_toggle' => __('shifts.filter.toggle'),
'set_yesterday' => __('Yesterday'),
'set_today' => __('Today'),
'set_tomorrow' => __('Tomorrow'),
'set_last_8h' => __('last 8h'),
'set_last_4h' => __('last 4h'),
'set_next_4h' => __('next 4h'),
'set_next_8h' => __('next 8h'),
'random' => auth()->can('user_shifts') && $canSignUpForShifts ? button(
url('/shifts/random'),
icon('dice-4-fill') . __('shifts.random')
) : '',
'dashboard' => button(
2017-12-25 23:12:52 +01:00
public_dashboard_link(),
icon('speedometer2') . __('Public Dashboard')
),
]),
]),
2017-01-02 15:43:36 +01:00
]);
2011-12-28 14:45:49 +01:00
}
2017-12-26 20:41:35 +01:00
/**
* Returns a hint for the user how the ical feature works.
*
* @return string
2017-12-26 20:41:35 +01:00
*/
function ical_hint()
{
2018-10-31 12:48:22 +01:00
$user = auth()->user();
2022-10-18 19:15:22 +02:00
if (!auth()->can('ical')) {
return '';
}
2017-12-26 20:41:35 +01:00
return heading(__('iCal export and API') . ' ' . button_help('user/ical'), 2)
2017-12-27 15:06:39 +01:00
. '<p>' . sprintf(
2024-04-07 19:27:46 +02:00
__('Export your own shifts formatted as <a href="%s" target="_blank">iCal</a> or <a href="%s" target="_blank">JSON</a> (please keep the link secret, otherwise you have to reset the api key <a href="%s">in your settings</a>).'),
2023-11-13 16:56:52 +01:00
url('/ical', ['key' => $user->api_key]),
url('/shifts-json-export', ['key' => $user->api_key]),
2024-04-07 19:27:46 +02:00
url('/settings/api')
) . '</p>';
2017-12-26 20:41:35 +01:00
}
/**
* @param array $items
* @param array $selected
* @param string $name
* @param string $title
2023-04-01 15:14:32 +02:00
* @param int[] $ownSelect
* @return string
*/
2023-04-01 15:14:32 +02:00
function make_select($items, $selected, $name, $title = null, $ownSelect = [])
2017-01-02 03:57:23 +01:00
{
2017-12-29 16:20:30 +01:00
$html = '';
2017-01-02 03:57:23 +01:00
if (isset($title)) {
2017-12-29 16:20:30 +01:00
$html .= '<h4>' . $title . '</h4>' . "\n";
2017-01-02 03:57:23 +01:00
}
2017-01-02 15:43:36 +01:00
2023-04-01 15:14:32 +02:00
$buttons = [
button_checkbox_selection($name, __('All'), 'true'),
button_checkbox_selection($name, __('None'), 'false'),
];
if (count($ownSelect) > 0) {
$buttons[] = button_checkbox_selection($name, __('Own'), json_encode($ownSelect));
}
2017-12-29 16:20:30 +01:00
$html .= buttons($buttons);
$html .= '<div id="selection_' . $name . '" class="mb-3 selection ' . $name . '">' . "\n";
2017-12-29 16:20:30 +01:00
$htmlItems = [];
2017-01-02 03:57:23 +01:00
foreach ($items as $i) {
$id = $name . '_' . $i['id'];
$htmlItems[] = '<div class="form-check">'
. '<input class="form-check-input" type="checkbox" id="' . $id . '" name="' . $name . '[]" value="' . $i['id'] . '" '
2017-01-02 15:43:36 +01:00
. (in_array($i['id'], $selected) ? ' checked="checked"' : '')
2023-12-04 23:33:07 +01:00
. '><label class="form-check-label" for="' . $id . '">' . htmlspecialchars($i['name']) . '</label>'
2022-12-02 23:03:23 +01:00
. (!isset($i['enabled']) || $i['enabled'] ? '' : icon('mortarboard-fill'))
. '</div>';
2017-01-02 03:57:23 +01:00
}
$html .= implode("\n", $htmlItems);
$html .= '</div>' . "\n";
$html .= buttons($buttons);
2017-01-02 03:57:23 +01:00
return $html;
2011-07-13 14:30:19 +02:00
}