haikal/templates/administration/staff_index.html
Marwan Alwali 1ed04ec706 update
2025-01-27 19:24:42 +03:00

410 lines
16 KiB
HTML

{% extends BASE_TEMPLATE %}
{% load i18n %}
{% load static %}
{% block customMetaTag %}
<meta name="csrf-token" content="{{ csrf_token }}">
{% endblock %}
{% block customCSS %}
<link rel="stylesheet" type="text/css" href="{% static 'css/appt-common.css' %}"/>
<link rel="stylesheet" type="text/css" href="{% static 'css/app_admin/admin.css' %}"/>
<style>
@media (max-width: 991px) {
body {
font-size: 15px; /* Adjust font size for tablets */
}
#calendar {
max-width: 800px; /* Adjust calendar width for tablets */
}
}
/* Mobile Styles */
@media (max-width: 767px) {
body {
margin: 20px 5px; /* Reduce margins on mobile */
font-size: 14px; /* Adjust font size for mobiles */
}
#calendar {
max-width: 100%; /* Let the calendar take the full width on mobile */
}
.fc, .fc-toolbar-title, h2 {
font-size: 15px !important; /* Adjust as needed */
margin: 0; /* Remove any default margin */
}
}
.calendar-wrapper {
display: flex;
transition: all 0.3s; /* smooth transition for resizing */
width: 100%;
flex-direction: column;
margin-top: 25px;
}
#calendar {
flex-grow: 1; /* Makes the calendar grow to occupy all available space */
transition: all 0.3s; /* smooth transition for resizing */
width: 100%;
}
#event-details {
overflow-y: auto;
width: 0; /* initially hidden */
transition: all 0.3s; /* smooth transition for resizing */
border-left: 1px solid #ccc;
flex-shrink: 0; /* Prevents the div from shrinking beyond its content's size */
}
.modal-content {
background-color: #fff;
margin: 10% auto; /* Centering the modal vertically */
width: 100%;
position: relative; /* For positioning the close button */
box-shadow: 0 0 20px rgba(0, 0, 0, 0.2); /* A little shadow for depth */
}
.modal-footer {
text-align: right;
}
.close-button {
position: absolute;
top: 10px;
right: 10px;
background: none;
border: none;
font-size: 24px;
cursor: pointer;
}
.body-blur {
filter: blur(2px); /* This will blur the whole body when modal is open */
}
#eventDetailsModal .modal-content {
background-color: #f4f4f4;
border-radius: 10px;
}
#eventDetailsModal .modal-header,
#eventDetailsModal .modal-footer {
background-color: #2c3e50;
color: #ecf0f1;
}
#eventDetailsModal .modal-title {
font-weight: bold;
}
#eventDetailsModal label {
display: block;
margin-top: 10px;
}
#eventDetailsModal input {
width: 100%;
padding: 5px;
margin: 5px 0;
border: 1px solid #ddd;
border-radius: 5px;
}
#eventDetailsModal .btn {
margin-right: 10px;
}
#serviceSelect, #staffSelect {
width: 100%;
padding: 8px 12px;
border: 1px solid #ccc;
border-radius: 4px;
background-color: #f9f9f9;
font-size: 14px;
transition: background-color 0.3s ease;
}
#serviceSelect:disabled, #staffSelect:disabled {
background-color: #e9ecef;
}
#customContextMenu ul {
list-style: none;
margin: 0;
padding: 0;
}
#customContextMenu ul li a {
padding: 5px 10px;
display: block;
text-decoration: none;
color: black;
}
#customContextMenu ul li a:hover {
background-color: #f0f0f0;
}
#customContextMenu {
border: 1px solid #ccc;
background-color: #fff;
box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.2);
padding: 5px 0;
position: absolute;
z-index: 1000;
display: none;
}
#eventDetailsModal input[type="checkbox"] {
width: auto;
margin: 0 5px 0 0;
}
/* General styles */
.flex-container-appt {
display: flex;
align-items: center;
margin-top: 5px;
}
.flex-container-appt label {
flex: 0 0 auto; /* Do not grow, do not shrink, and base width on content */
margin-right: 5px;
white-space: nowrap; /* Prevent label from wrapping */
}
.flex-container-appt input,
.flex-container-appt select {
flex: 1 1 auto; /* Grow and shrink based on available space */
}
/* Larger Modal on Desktop */
@media (min-width: 850px) {
.modal-content {
/* centering the modal horizontally */
margin: 10% auto;
left: 0;
right: 0;
}
}
/* Stack label and input on mobile */
@media (max-width: 767px) {
.flex-container-appt {
flex-direction: column;
align-items: flex-start;
}
.flex-container-appt label {
margin-right: 0;
margin-bottom: 5px;
}
}
.highlight-weekend {
background-color: rgba(155, 155, 155, 0.03); /* Light gray background for weekends */
}
.event-dot {
height: 5px;
width: 5px;
border-radius: 50%;
display: inline-block;
margin: 2px;
}
/* Hide event list container on larger screens */
@media (min-width: 768px) {
#event-list-container {
display: none; /* Hide the event list container */
}
}
/* Show event list container on smaller screens */
@media (max-width: 767px) {
#event-list-container {
/* Styles to display and position the event list container */
display: block; /* Show the event list container */
margin-top: 20px; /* Space from the calendar */
}
.event-list-item {
/* Styles for individual event list items */
padding: 10px;
border-bottom: 1px solid #ccc;
}
}
@media (max-width: 450px) {
.fc-daygrid-event-harness {
width: 5px;
}
.fc td, .fc th {
}
}
.event-list-item-appt {
/* pointer hand cursor */
cursor: pointer;
}
</style>
{% endblock %}
{% block title %}
{{ page_title }}
{% endblock %}
{% block description %}
{{ page_description }}
{% endblock %}
{% block body %}
<section class="content content-wrapper">
<div class="container">
<div class="calendar-wrapper">
<div id="calendar" class="calendarbox"></div>
<div id="event-list-container" class="event-list-container"></div>
</div>
{% include 'modal/event_details_modal.html' %}
{% include 'modal/error_modal.html' %}
<div class="messages" style="margin: 20px 0">
{% if messages %}
{% for message in messages %}
<div class="alert alert-dismissible {% if message.tags %}alert-{% if message.level == DEFAULT_MESSAGE_LEVELS.ERROR %}danger{% else %}{{ message.tags }}{% endif %}{% endif %}"
role="alert">{{ message }}</div>
{% endfor %}
{% endif %}
</div>
</div>
</section>
<div id="customContextMenu" style="display: none; position: absolute; z-index: 1000;">
<ul>
<li id="newAppointmentOption"><a href="#">New Appointment</a></li>
</ul>
</div>
{% include 'modal/confirm_modal.html' %}
{% endblock %}
{% block customJS %}
<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.30.1/moment.js"
integrity="sha512-3CuraBvy05nIgcoXjVN33mACRyI89ydVHg7y/HMN9wcTVbHeur0SeBzweSd/rxySapO7Tmfu68+JlKkLTnDFNg=="
crossorigin="anonymous" referrerpolicy="no-referrer"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.45/moment-timezone-with-data.min.js"
integrity="sha512-t/mY3un180WRfsSkWy4Yi0tAxEDGcY2rAEx873hb5BrkvLA0QLk54+SjfYgFBBoCdJDV1H86M8uyZdJhAOHeyA=="
crossorigin="anonymous" referrerpolicy="no-referrer"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/fullcalendar/6.1.10/index.global.min.js"
integrity="sha512-JCQkxdym6GmQ+AFVioDUq8dWaWN6tbKRhRyHvYZPupQ6DxpXzkW106FXS1ORgo/m3gxtt5lHRMqSdm2OfPajtg=="
crossorigin="anonymous" referrerpolicy="no-referrer"></script>
<script>
const timezone = "{{ timezone }}";
const locale = "{{ locale }}";
const availableSlotsAjaxURL = "{% url 'appointment:available_slots_ajax' %}";
const requestNextAvailableSlotURLTemplate = "{% url 'appointment:request_next_available_slot' service_id=0 %}";
const deleteAppointmentURLTemplate = "{% url 'appointment:delete_appointment_ajax' %}";
const getNonWorkingDaysURL = "{% url 'appointment:get_non_working_days_ajax' %}";
const serviceId = "{{ service.id }}";
const serviceDuration = parseInt("{{ service.duration.total_seconds }}") / 60;
let appointments = {{ appointments|safe }};
const fetchServiceListForStaffURL = "{% url 'appointment:fetch_service_list_for_staff' %}";
const fetchStaffListURL = "{% url 'appointment:fetch_staff_list' %}";
const updateApptMinInfoURL = "{% url 'appointment:update_appt_min_info' %}";
const updateApptDateURL = "{% url 'appointment:update_appt_date_time' %}";
const validateApptDateURL = "{% url 'appointment:validate_appointment_date' %}";
const isUserStaffAdminURL = "{% url 'appointment:is_user_staff_admin' %}";
const isUserSuperUser = "{{ is_superuser }}" === "True";
</script>
<script>
{# Text for translation #}
const confirmDeletionTitleTxt = "{% trans 'Confirm Deletion' %}";
const confirmDeletionTxt = "{% trans 'Are you sure you want to delete this appointment?' %}";
const deleteBtnTxt = "{% trans 'Delete' %}";
const eventsOnTxt = "{% trans 'Events on' %}";
const noEventTxt = "{% trans 'No events for this day.' %}";
const newEventTxt = "{% trans 'New Event' %}";
const successTxt = "{% trans 'Success' %}";
const errorTxt = "{% trans 'Error' %}";
const updateApptErrorTitleTxt = "{% trans 'Error: Unable to delete appointment.' %}";
const apptNotFoundTxt = "{% trans 'Appointment not found.' %}";
const notStaffMemberTxt = "{% trans "You're not a staff member. Can't perform this action !" %}";
const noServiceOfferedTxt = "{% trans "You don't offer any service. Add new service from your profile." %}";
const noStaffMemberTxt = "{% trans "No staff members found." %}";
</script>
<script src="{% static 'js/modal/error_modal.js' %}"></script>
<script src="{% static 'js/app_admin/staff_index.js' %}"></script>
<script src="{% static 'js/modal/show_modal.js' %}"></script>
<script src="{% static 'js/js-utils.js' %}"></script>
<script>
function createCommonInputFields(appointment, servicesDropdown, isEditMode, defaultStartTime, staffDropdown) {
const startTimeValue = isEditMode ? moment(appointment.start_time).format('HH:mm:ss') : defaultStartTime;
const disabledAttribute = isEditMode ? '' : 'disabled';
let superuserInputField = '';
if (isUserSuperUser) {
superuserInputField = `
<div class="flex-container-appt">
<label>{% trans 'Staff Member' %}:</label>
${staffDropdown.outerHTML}
</div>
`;
}
return `
${superuserInputField}
<div class="flex-container-appt">
<label>{% trans 'Service Name' %}:</label>
${servicesDropdown.outerHTML}
</div>
<label for="clientName">{% trans 'Client Name' %}:</label>
<input type="text" name="clientName" value="${appointment.client_name || ''}" ${disabledAttribute} id="clientName" placeholder="John Doe">
<label for="clientEmail">{% trans 'Client Email' %}:</label>
<input type="email" name="clientEmail" value="${appointment.client_email || ''}" ${disabledAttribute} id="clientEmail" placeholder="john.doe@example.com">
<span id="emailError" style="display:none;"></span>
<label for="clientPhone">{% trans 'Phone Number' %}:</label>
<input type="tel" name="clientPhone" value="${appointment.client_phone || ''}" ${disabledAttribute} id="clientPhone" placeholder="+12392350799">
<label for="clientAddress">{% trans 'Client address' %}:</label>
<input type="text" name="clientAddress" value="${appointment.client_address || ''}" ${disabledAttribute} id="clientAddress" placeholder="Naples, Florida">
<div class="flex-container-appt">
<label for="want_reminder">{% trans 'Wants reminder' %}:</label>
<input type="checkbox" name="want_reminder" id="want_reminder" value="true" ${appointment.want_reminder ? 'checked' : ''} ${disabledAttribute}>
</div>
<label for="additional_info">{% trans 'Additional Information' %}:</label>
<input type="text" name="additional_info" value="${appointment.additional_info || ''}" ${disabledAttribute} placeholder="{% trans 'Client wants this and that' %}" id="additional_info">
<div class="flex-container-appt">
<label for="startTime">{% trans 'Start time' %}:</label>
<input type="time" name="startTime" value="${startTimeValue}" ${disabledAttribute}>
</div>
`;
}
function generateModalContent(appointment, servicesDropdown, isEditMode, staffDropdown) {
const startTimeValue = moment(appointment.start_time).format('HH:mm:ss');
const commonInputs = createCommonInputFields(appointment, servicesDropdown, isEditMode, startTimeValue, staffDropdown);
return `
${commonInputs}
<div class="flex-container-appt">
<label for="endTime">{% trans 'End time' %}:</label>
<input type="time" name="endTime" value="${moment(appointment.end_time).format('HH:mm:ss')}" ${!isEditMode ? 'disabled' : ''}>
</div>
`;
}
function prepareCreateAppointmentModalContent(servicesDropdown, staffDropdown, defaultStartTime, formattedDate) {
const appointment = {client_name: '', client_email: '', client_phone: '', client_address: ''};
const commonInputs = createCommonInputFields(appointment, servicesDropdown, true, defaultStartTime, staffDropdown);
return `
${commonInputs}
<label for="date" style="display: none"></label>
<input type="hidden" name="date" value="${formattedDate}">
`;
}
</script>
{% endblock %}