engelsystem/includes/pages/user_shifts.php

392 lines
12 KiB
PHP
Raw 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;
2020-09-06 23:50:36 +02:00
use Engelsystem\Models\Room;
2016-10-02 23:00:01 +02:00
use Engelsystem\ShiftsFilter;
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()
{
return __('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
2018-10-10 03:10:28 +02:00
if (User_is_freeloader(auth()->user())) {
2019-09-08 02:25:49 +02:00
throw_redirect(page_link_to('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
*
* @param ShiftsFilter $shiftsFilter The shifts filter to update from request data
* @param boolean $user_shifts_admin Has the user user_shift_admin privilege?
* @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()));
$shiftsFilter->setRooms(check_request_int_array('rooms', $shiftsFilter->getRooms()));
$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
/**
2020-09-06 23:50:36 +02:00
* @return Room[]|Collection
2017-01-03 03:22:48 +01:00
*/
2017-01-02 03:57:23 +01:00
function load_rooms()
{
2020-09-06 23:50:36 +02:00
$rooms = Rooms();
if ($rooms->isEmpty()) {
error(__('The administration has not configured any rooms yet.'));
2019-09-08 02:25:49 +02:00
throw_redirect(page_link_to('/'));
2017-01-02 03:57:23 +01:00
}
2020-09-06 23:50:36 +02:00
2017-01-02 03:57:23 +01:00
return $rooms;
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
'
SELECT DISTINCT DATE(FROM_UNIXTIME(`start`)) AS `id`, DATE(FROM_UNIXTIME(`start`)) AS `name`
FROM `Shifts`
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') {
2019-09-08 02:25:49 +02:00
throw_redirect(page_link_to('/'));
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();
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.'));
2019-09-08 02:25:49 +02:00
throw_redirect(page_link_to('/'));
2017-01-02 03:57:23 +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 (
NOT `UserAngelTypes`.`confirm_user_id` IS NULL
OR `UserAngelTypes`.`id` IS NULL
)
) AS `enabled`
2022-11-09 00:02:30 +01:00
FROM `angel_types`
LEFT JOIN `UserAngelTypes`
ON (
2022-11-09 00:02:30 +01:00
`UserAngelTypes`.`angeltype_id`=`angel_types`.`id`
AND `UserAngelTypes`.`user_id`=?
2017-01-02 15:43:36 +01:00
)
2022-11-09 00:02:30 +01:00
ORDER BY `angel_types`.`name`
',
[
2018-10-08 21:15:56 +02:00
$user->id,
]
);
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
}
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();
$rooms = load_rooms();
$types = load_types();
$ownTypes = [];
2021-05-11 21:50:54 +02:00
foreach (UserAngelTypes_by_User($user->id, true) as $type) {
if (!$type['confirm_user_id'] && $type['restricted']) {
continue;
}
$ownTypes[] = (int)$type['angeltype_id'];
}
2017-01-02 15:43:36 +01:00
if (!$session->has('shifts-filter')) {
2020-09-06 23:50:36 +02:00
$room_ids = $rooms->pluck('id')->toArray();
$shiftsFilter = new ShiftsFilter(auth()->can('user_shifts_admin'), $room_ids, $ownTypes);
$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)) {
2017-01-02 03:57:23 +01:00
User_reset_api_key($user, false);
}
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
]
];
2017-01-03 14:12:17 +01:00
$start_day = date('Y-m-d', $shiftsFilter->getStartTime());
$start_time = date('H:i', $shiftsFilter->getStartTime());
$end_day = date('Y-m-d', $shiftsFilter->getEndTime());
$end_time = date('H:i', $shiftsFilter->getEndTime());
2017-01-02 15:43:36 +01:00
2018-10-10 03:10:28 +02:00
if (config('signup_requires_arrival') && !$user->state->arrived) {
info(render_user_arrived_hint());
}
2017-01-02 03:57:23 +01:00
return page([
2017-01-02 15:43:36 +01:00
div('col-md-12', [
msg(),
view(__DIR__ . '/../../resources/views/pages/user-shifts.html', [
2017-01-02 15:43:36 +01:00
'title' => shifts_title(),
'room_select' => make_select($rooms, $shiftsFilter->getRooms(), 'rooms', __('Rooms')),
2017-12-25 23:12:52 +01:00
'start_select' => html_select_key(
'start_day',
'start_day',
array_combine($days, $days),
$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',
array_combine($days, $days),
$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',
__('Angeltypes') . '<sup>1</sup>',
[
button(
'javascript:checkOwnTypes(\'selection_types\', ' . json_encode($ownTypes) . ')',
2019-07-24 20:57:27 +02:00
__('Own'),
'd-print-none'
),
]
2017-01-02 15:43:36 +01:00
),
'filled_select' => make_select($filled, $shiftsFilter->getFilled(), 'filled', __('Occupancy')),
2017-01-02 15:43:36 +01:00
'task_notice' =>
'<sup>1</sup>'
. __('The tasks shown here are influenced by the angeltypes you joined already!')
2017-08-28 16:21:10 +02:00
. ' <a href="' . page_link_to('angeltypes', ['action' => 'about']) . '">'
. __('Description of the jobs.')
2017-01-03 03:22:48 +01:00
. '</a>',
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'),
2017-12-25 23:12:52 +01:00
'buttons' => button(
public_dashboard_link(),
icon('speedometer2') . __('Public Dashboard')
2017-12-25 23:12:52 +01:00
)
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(
__('Export your own shifts. <a href="%s">iCal format</a> or <a href="%s">JSON format</a> available (please keep secret, otherwise <a href="%s">reset the api key</a>).'),
2018-10-08 21:15:56 +02:00
page_link_to('ical', ['key' => $user->api_key]),
page_link_to('shifts_json_export', ['key' => $user->api_key]),
2017-12-26 20:41:35 +01:00
page_link_to('user_myshifts', ['reset' => 1])
)
. ' <button class="btn btn-sm btn-danger" type="button"
data-bs-toggle="collapse" data-bs-target="#collapseApiKey"
aria-expanded="false" aria-controls="collapseApiKey">
2020-05-13 18:26:32 +02:00
' . __('Show API Key') . '
</button>'
. '</p>'
. '<p id="collapseApiKey" class="collapse"><code>' . $user->api_key . '</code></p>';
2017-12-26 20:41:35 +01:00
}
2017-01-03 03:22:48 +01:00
/**
* @param array $array
* @return array
*/
2017-01-02 03:57:23 +01:00
function get_ids_from_array($array)
{
2017-01-03 14:12:17 +01:00
return $array['id'];
2011-07-13 14:30:19 +02:00
}
/**
* @param array $items
* @param array $selected
* @param string $name
* @param string $title
* @param array $additionalButtons
* @return string
*/
function make_select($items, $selected, $name, $title = null, $additionalButtons = [])
2017-01-02 03:57:23 +01:00
{
2017-12-29 16:20:30 +01:00
$html = '';
$htmlItems = [];
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
2017-12-29 16:20:30 +01:00
$buttons = [];
$buttons[] = button('javascript:checkAll(\'selection_' . $name . '\', true)', __('All'), 'd-print-none');
$buttons[] = button('javascript:checkAll(\'selection_' . $name . '\', false)', __('None'), 'd-print-none');
2017-12-29 16:20:30 +01:00
$buttons = array_merge($buttons, $additionalButtons);
$html .= buttons($buttons);
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"' : '')
2022-12-03 23:49:55 +01:00
. '><label class="form-check-label" for="' . $id . '">' . $i['name'] . '</label>'
2021-09-10 14:30:16 +02:00
. (!isset($i['enabled']) || $i['enabled'] ? '' : icon('book'))
. '</div>';
2017-01-02 03:57:23 +01:00
}
2017-12-29 16:20:30 +01:00
$html .= '<div id="selection_' . $name . '" class="selection ' . $name . '">' . "\n";
$html .= implode("\n", $htmlItems);
$html .= buttons($buttons);
2017-01-02 03:57:23 +01:00
$html .= '</div>' . "\n";
return $html;
2011-07-13 14:30:19 +02:00
}