30if (!defined(
'NOLOGIN')) {
33if (!defined(
'NOCSRFCHECK')) {
34 define(
"NOCSRFCHECK", 1);
36if (!defined(
'NOBROWSERNOTIF')) {
37 define(
'NOBROWSERNOTIF',
'1');
41require
'../../main.inc.php';
42require_once DOL_DOCUMENT_ROOT.
'/core/lib/company.lib.php';
43require_once DOL_DOCUMENT_ROOT.
'/core/lib/payments.lib.php';
44require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
45require_once DOL_DOCUMENT_ROOT.
'/core/lib/date.lib.php';
46require_once DOL_DOCUMENT_ROOT.
'/core/lib/functions2.lib.php';
47require_once DOL_DOCUMENT_ROOT.
'/bookcal/class/calendar.class.php';
48require_once DOL_DOCUMENT_ROOT.
'/bookcal/class/availabilities.class.php';
49require_once DOL_DOCUMENT_ROOT.
'/contact/class/contact.class.php';
50require_once DOL_DOCUMENT_ROOT.
'/comm/action/class/actioncomm.class.php';
51require_once DOL_DOCUMENT_ROOT.
'/core/lib/public.lib.php';
54if (!isModEnabled(
'bookcal')) {
58$langs->loadLangs(array(
"main",
"other",
"dict",
"agenda",
"errors",
"companies"));
60$action =
GETPOST(
'action',
'aZ09');
62$id_availability =
GETPOSTINT(
'id_availability');
74$backtopage =
GETPOST(
"backtopage",
"alpha");
80if ($id_availability > 0) {
81 $result = $availability->fetch($id_availability);
86$nowyear = $nowarray[
'year'];
87$nowmonth = $nowarray[
'mon'];
88$nowday = $nowarray[
'mday'];
91$prev_year = $prev[
'year'];
92$prev_month = $prev[
'month'];
94$next_year = $next[
'year'];
95$next_month = $next[
'month'];
97$max_day_in_prev_month = idate(
"t",
dol_mktime(0, 0, 0, $prev_month, 1, $prev_year,
'gmt'));
98$max_day_in_month = idate(
"t",
dol_mktime(0, 0, 0, $month, 1, $year));
100$tmpday = - idate(
"w",
dol_mktime(12, 0, 0, $month, 1, $year,
'gmt')) + 2;
106$firstdaytoshow =
dol_mktime(0, 0, 0, $prev_month, $max_day_in_prev_month + $tmpday, $prev_year,
'tzuserrel');
107$next_day = 7 - ($max_day_in_month + 1 - $tmpday) % 7;
111$lastdaytoshow =
dol_mktime(0, 0, 0, $next_month, $next_day, $next_year,
'tzuserrel');
113$datechosen =
GETPOST(
'datechosen',
'alpha');
114$datetimechosen =
GETPOSTINT(
'datetimechosen');
115$isdatechosen =
false;
116$timebooking =
GETPOST(
"timebooking");
117$datetimebooking =
GETPOSTINT(
"datetimebooking");
118$durationbooking =
GETPOSTINT(
"durationbooking");
132function llxHeaderVierge($title, $head =
"", $disablejs = 0, $disablehead = 0, $arrayofjs = [], $arrayofcss = [])
134 global $conf, $langs, $mysoc;
136 top_htmlhead($head, $title, $disablejs, $disablehead, $arrayofjs, $arrayofcss);
138 print
'<body id="mainbody" class="publicnewmemberform">';
146 $urllogo = DOL_URL_ROOT.
'/theme/common/login_logo.png';
148 if (!empty($mysoc->logo_small) && is_readable($conf->mycompany->dir_output.
'/logos/thumbs/'.$mysoc->logo_small)) {
149 $urllogo = DOL_URL_ROOT.
'/viewimage.php?modulepart=mycompany&entity='.$conf->entity.
'&file='.urlencode(
'logos/thumbs/'.$mysoc->logo_small);
150 } elseif (!empty($mysoc->logo) && is_readable($conf->mycompany->dir_output.
'/logos/'.$mysoc->logo)) {
151 $urllogo = DOL_URL_ROOT.
'/viewimage.php?modulepart=mycompany&entity='.$conf->entity.
'&file='.urlencode(
'logos/'.$mysoc->logo);
152 } elseif (is_readable(DOL_DOCUMENT_ROOT.
'/theme/dolibarr_logo.svg')) {
153 $urllogo = DOL_URL_ROOT.
'/theme/dolibarr_logo.svg';
158 print
'<div class="center">';
160 print
'<div class="backgreypublicpayment">';
161 print
'<div class="logopublicpayment">';
163 print
'<a href="'.(getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') ?
getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') :
dol_buildpath(
'/public/ticket/index.php?entity='.$conf->entity, 1)).
'">';
164 print
'<img id="dolpaymentlogo" src="'.$urllogo.
'">';
168 print
'<div class="clearboth"></div><strong>'.(getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') ?
getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') : $langs->trans(
"BookCalSystem")).
'</strong>';
175 print
'<div class="poweredbypublicpayment opacitymedium right hideonsmartphone"><a class="poweredbyhref" href="https://www.dolibarr.org?utm_medium=website&utm_source=poweredby" target="dolibarr" rel="noopener">'.$langs->trans(
"PoweredBy").
'<br><img src="'.DOL_URL_ROOT.
'/theme/dolibarr_logo.svg" width="80px"></a></div>';
181 print
'<div class="divmainbodylarge">';
189if ($action ==
'add' ) {
195 $nb_post_max =
getDolGlobalInt(
"MAIN_SECURITY_MAX_POST_ON_PUBLIC_PAGES_BY_IP_ADDRESS", 200);
197 if (!is_object($user)) {
198 $user =
new User($db);
205 $errmsg .= $langs->trans(
"ErrorFieldRequired", $langs->transnoentitiesnoconv(
"Lastname")).
"<br>\n";
209 $errmsg .= $langs->trans(
"ErrorFieldRequired", $langs->transnoentitiesnoconv(
"Firstname")).
"<br>\n";
213 $errmsg .= $langs->trans(
"ErrorFieldRequired", $langs->transnoentitiesnoconv(
"Email")).
"<br>\n";
217 $sql =
"SELECT s.rowid";
218 $sql .=
" FROM ".MAIN_DB_PREFIX.
"socpeople as s";
219 $sql .=
" WHERE s.lastname = '".$db->escape(
GETPOST(
"lastname")).
"'";
220 $sql .=
" AND s.firstname = '".$db->escape(
GETPOST(
"firstname")).
"'";
221 $sql .=
" AND s.email = '".$db->escape(
GETPOST(
"email")).
"'";
222 $resql = $db->query($sql);
225 $num = $db->num_rows($resql);
227 $obj = $db->fetch_object($resql);
228 $idcontact = $obj->rowid;
229 $contact->fetch($idcontact);
231 $contact->lastname =
GETPOST(
"lastname");
232 $contact->firstname =
GETPOST(
"firstname");
233 $contact->email =
GETPOST(
"email");
238 $errmsg .= implode(
'<br>', $contact->errors);
240 $result = $contact->create($user);
243 $errmsg .= $contact->error.
" ".implode(
',', $contact->errors);
249 $errmsg .= $db->lasterror();
256 $actioncomm->label = $langs->trans(
"BookcalBookingTitle");
257 $actioncomm->type =
'AC_RDV';
258 $actioncomm->type_id = 5;
259 $actioncomm->datep =
GETPOSTINT(
"datetimebooking");
260 $actioncomm->datef = $dateend;
261 $actioncomm->note_private =
GETPOST(
"description");
262 $actioncomm->percentage = -1;
263 $actioncomm->fk_bookcal_calendar =
$id;
264 $actioncomm->userownerid = $calendar->visibility;
265 $actioncomm->contact_id = $contact->id;
266 $actioncomm->socpeopleassigned = [
268 'id' => $contact->id,
270 'answer_status' => 0,
277 $errmsg .= implode(
'<br>', $actioncomm->errors);
279 $result = $actioncomm->create($user);
282 $errmsg .= $actioncomm->error.
" ".implode(
',', $actioncomm->errors);
289 $action =
'afteradd';
301$form =
new Form($db);
309if ($action ==
'create') {
310 $backtopage = $_SERVER[
"PHP_SELF"].
'?id='.
$id.
'&datechosen='.$datechosen;
312 $backtopage = DOL_URL_ROOT.
'/public/bookcal/index.php?id='.
$id;
317print
'<div class="bookcalpublicarea centpercent center" style="min-width:30%;width:fit-content;height:70%;top:60%;left: 50%;">';
318print
'<div class="bookcalform" style="min-height:50%">';
319if ($action ==
'afteradd') {
321 print $langs->trans(
"BookingSuccessfullyBooked");
327 print
'<table class="centpercent">';
330 if ($action !=
'create') {
331 print
'<form name="formsearch" action="'.$_SERVER[
"PHP_SELF"].
'">';
332 print
'<input type="hidden" name="id" value="'.$id.
'">';
334 $nav =
'<a href="?id='.$id.
"&year=".$prev_year.
"&month=".$prev_month.$param.
'"><i class="fa fa-chevron-left"></i></a> '.
"\n";
335 $nav .=
' <span id="month_name">'.dol_print_date(
dol_mktime(0, 0, 0, $month, 1, $year),
"%b %Y");
336 $nav .=
" </span>\n";
337 $nav .=
' <a href="?id='.$id.
"&year=".$next_year.
"&month=".$next_month.$param.
'"><i class="fa fa-chevron-right"></i></a>'.
"\n";
338 if (empty($conf->dol_optimize_smallscreen)) {
339 $nav .=
' <a href="?id='.$id.
"&year=".$nowyear.
"&month=".$nowmonth.
"&day=".$nowday.$param.
'" class="datenowlink">'.$langs->trans(
"Today").
'</a> ';
341 $nav .= $form->selectDate($dateselect,
'dateselect', 0, 0, 1,
'', 1, 0);
342 $nav .=
'<button type="submit" class="liste_titre button_search valignmiddle" name="button_search_x" value="x"><span class="fa fa-search"></span></button>';
349 print
'<div class="bookingtab hidden" style="height:50%">';
350 print
'<div id="bookingtabspandate"></div>';
356 if ($action ==
"create") {
358 if (empty($datetimebooking)) {
359 $timebookingarray = explode(
" - ", $timebooking);
360 $timestartarray = explode(
":", $timebookingarray[0]);
361 $timeendarray = explode(
":", $timebookingarray[1]);
365 print
'<span>'.img_picto(
"",
"calendar").
" ".
dol_print_date($datetimebooking,
'dayhourtext').
'</span>';
366 print
'<div class="center"><a href="'.$_SERVER[
"PHP_SELF"].
'?id=1&year=2024&month=2" class="small">('.$langs->trans(
"SelectANewDate").
')</a></div>';
370 print
'<form method="POST" action="'.$_SERVER[
"PHP_SELF"].
'">';
371 print
'<table class="border" summary="form to subscribe" id="tablesubscribe">'.
"\n";
372 print
'<input type="hidden" name="token" value="'.newToken().
'">';
373 print
'<input type="hidden" name="action" value="add">';
374 print
'<input type="hidden" name="datetimebooking" value="'.$datetimebooking.
'">';
375 print
'<input type="hidden" name="datechosen" value="'.$datechosen.
'">';
376 print
'<input type="hidden" name="id" value="'.$id.
'">';
377 print
'<input type="hidden" name="durationbooking" value="'.$durationbooking.
'">';
380 print
'<tr><td><input autofocus type="text" name="lastname" class="minwidth150" placeholder="'.dol_escape_htmltag($langs->trans(
"Lastname").
'*').
'" value="'.
dol_escape_htmltag(
GETPOST(
'lastname')).
'"></td></tr>'.
"\n";
382 print
'<tr><td><input type="text" name="firstname" class="minwidth150" placeholder="'.dol_escape_htmltag($langs->trans(
"Firstname").
'*').
'" value="'.
dol_escape_htmltag(
GETPOST(
'firstname')).
'"></td></tr>'.
"\n";
384 print
'<tr><td><input type="email" name="email" maxlength="255" class="minwidth150" placeholder="'.dol_escape_htmltag($langs->trans(
"Email").
'*').
'" value="'.
dol_escape_htmltag(
GETPOST(
'email')).
'"></td></tr>'.
"\n";
388 print
'<td class="tdtop">';
389 print $langs->trans(
"Message");
390 print
'<textarea name="description" id="description" wrap="soft" class="quatrevingtpercent" rows="'.ROWS_4.
'">'.
dol_escape_htmltag(
GETPOST(
'description',
'restricthtml'), 0, 1).
'</textarea></td>';
392 print
'</table>'.
"\n";
393 print
'<div class="center">';
394 print
'<input type="submit" value="'.$langs->trans(
"Submit").
'" id="submitsave" class="button">';
400 print
'<table class="centpercent noborder nocellnopadd cal_pannel cal_month">';
401 print
' <tr class="">';
403 print
' <td class="center hideonsmartphone">#</td>';
406 $numdayinweek = (($i + (isset($conf->global->MAIN_START_WEEK) ? $conf->global->MAIN_START_WEEK : 1)) % 7);
407 if (!empty($conf->dol_optimize_smallscreen)) {
408 print
' <td class="center bold uppercase tdfordaytitle'.($i == 0 ?
' borderleft' :
'').
'">';
409 $labelshort = array(0 =>
'SundayMin', 1 =>
'MondayMin', 2 =>
'TuesdayMin', 3 =>
'WednesdayMin', 4 =>
'ThursdayMin', 5 =>
'FridayMin', 6 =>
'SaturdayMin');
410 print $langs->trans($labelshort[$numdayinweek]);
413 print
' <td class="center minwidth75 bold uppercase small tdoverflowmax50 tdfordaytitle'.($i == 0 ?
' borderleft' :
'').
'">';
415 $labelshort = array(0 =>
'Sunday', 1 =>
'Monday', 2 =>
'Tuesday', 3 =>
'Wednesday', 4 =>
'Thursday', 5 =>
'Friday', 6 =>
'Saturday');
416 print $langs->trans($labelshort[$numdayinweek]);
424 $todaytms =
dol_mktime(0, 0, 0, $todayarray[
'mon'], $todayarray[
'mday'], $todayarray[
'year']);
427 $arrayofavailabledays = array();
429 $arrayofavailabilities = $availability->fetchAll(
'',
'', 0, 0,
'(status:=:1) AND (fk_bookcal_calendar:=:'.((
int) $id).
')');
430 if ($arrayofavailabilities < 0) {
433 foreach ($arrayofavailabilities as $key => $value) {
436 for ($i = $startarray[
'mday']; $i <= $endarray[
'mday']; $i++) {
437 if ($todayarray[
'mon'] >= $startarray[
'mon'] && $todayarray[
'mon'] <= $endarray[
'mon']) {
438 $arrayofavailabledays[
dol_mktime(0, 0, 0, $todayarray[
'mon'], $i, $todayarray[
'year'])] =
dol_mktime(0, 0, 0, $todayarray[
'mon'], $i, $todayarray[
'year']);
444 for ($iter_week = 0; $iter_week < 6; $iter_week++) {
448 $currdate0 = sprintf(
"%04d", $prev_year).sprintf(
"%02d", $prev_month).sprintf(
"%02d", $max_day_in_prev_month + $tmpday);
449 } elseif ($tmpday <= $max_day_in_month) {
450 $currdate0 = sprintf(
"%04d", $year).sprintf(
"%02d", $month).sprintf(
"%02d", $tmpday);
452 $currdate0 = sprintf(
"%04d", $next_year).sprintf(
"%02d", $next_month).sprintf(
"%02d", $tmpday - $max_day_in_month);
455 $numweek0 = idate(
"W", strtotime(date($currdate0)));
457 echo
' <td class="center weeknumber opacitymedium hideonsmartphone" style="min-width: 40px">'.$numweek0.
'</td>';
459 for ($iter_day = 0; $iter_day < 7; $iter_day++) {
462 $style =
'cal_other_month cal_past';
463 if ($iter_day == 6) {
464 $style .=
' cal_other_month_right';
466 echo
' <td class="'.$style.
' nowrap tdtop" width="14%">';
469 } elseif ($tmpday <= $max_day_in_month) {
471 $curtime =
dol_mktime(0, 0, 0, $month, $tmpday, $year);
472 $style =
'cal_current_month';
473 if ($iter_day == 6) {
474 $style .=
' cal_current_month_right';
477 if ($todayarray[
'mday'] == $tmpday && $todayarray[
'mon'] == $month && $todayarray[
'year'] == $year) {
481 if ($curtime > $todaytms && in_array($curtime, $arrayofavailabledays)) {
482 $style .=
' cal_available cursorpointer';
484 if ($curtime < $todaytms) {
485 $style .=
' cal_past';
487 $dateint = sprintf(
"%04d", $year).
'_'.sprintf(
"%02d", $month).
'_'.sprintf(
"%02d", $tmpday);
488 if (!empty(explode(
'dayevent_', $datechosen)[1]) && explode(
'dayevent_', $datechosen)[1] == $dateint) {
489 $style .=
' cal_chosen';
490 $isdatechosen =
true;
492 echo
' <td class="'.$style.
' nowrap tdtop" width="14%">';
497 $style =
'cal_other_month';
498 if ($iter_day == 6) {
499 $style .=
' cal_other_month_right';
501 echo
' <td class="'.$style.
' nowrap tdtop" width="14%">';
513 print
'<div class="center bookingtab" style="height:50%">';
514 print
'<div style="height:100%">';
515 print
'<form id="formbooking" name="formbooking" method="POST" action="'.$_SERVER[
"PHP_SELF"].
'">';
516 print
'<input type="hidden" name="id" value="'.$id.
'">';
517 print
'<input type="hidden" name="token" value="'.newToken().
'">';
518 print
'<input type="hidden" name="action" value="create">';
519 print
'<input type="hidden" id="datechosen" name="datechosen" value="">';
520 print
'<input type="hidden" id="datetimechosen" name="datetimechosen" value="">';
521 print
'<input type="hidden" id="durationbooking" name="durationbooking" value="">';
523 print
'<div id="bookinghoursection">';
524 print
'<br><br><br><br><br><br><div class="opacitymedium center">'.$langs->trans(
"SelectADay").
'</div>';
539 function generateBookingButtons(timearray, datestring){
540 console.log("We generate all booking buttons of "+datestring);
543 for (index in timearray){
544 let hour = new Date("2000-01-01T" + index + ":00");
545 duration = timearray[index];
546 isalreadybooked = false;
549 isalreadybooked = true;
551 hour.setMinutes(hour.getMinutes() + duration);
553 let hours = hour.getHours().toString().padStart(2, "0"); // Formatter pour obtenir deux chiffres
554 let mins = hour.getMinutes().toString().padStart(2, "0"); // Formatter pour obtenir deux chiffres
556 timerange = index + " - " + `${hours}:${mins}`;
557 str += \'<input class="button btnsubmitbooking \'+(isalreadybooked == true ? "btnbookcalbooked" : "")+\'" type="submit" name="timebooking" value="\'+timerange+\'" data-duration="\'+duration+\'"><br>\';
560 $("#bookinghoursection").html(str);
561 $(".btnsubmitbooking").on("click", function(){
562 duration = $(this).data("duration");
563 $("#durationbooking").val(duration);
566 print
'$(document).ready(function() {
567 $(".cal_available").on("click", function(){
568 console.log("We click on cal_available");
569 $(".cal_chosen").removeClass("cal_chosen");
570 $(this).addClass("cal_chosen");
571 datestring = $(this).children("div").data("date");
574 url: "'.DOL_URL_ROOT.
'/public/bookcal/bookcalAjax.php",
576 action: "verifyavailability",
578 datetocheck: $(this).children("div").data("datetime"),
581 }).done(function (data) {
582 console.log("We show all booking");
583 if (data["code"] == "SUCCESS") {
584 /* TODO Replace this with a creating of allavailable hours button */
586 timearray = data["availability"];
587 console.log(timearray);
588 generateBookingButtons(timearray, datestring);
589 $(".btnbookcalbooked").prop("disabled", true);
591 if(data["code"] == "NO_DATA_FOUND"){
592 console.log("No booking to hide");
594 console.log(data["message"]);
598 $(".bookingtab").removeClass("hidden");
599 $("#bookingtabspandate").text($(this).children("div").data("date"));
600 $("#datechosen").val($(this).children("div").attr("id"));
601 $("#datetimechosen").val($(this).children("div").data("datetime"));
606 '.($datechosen ?
'$(".cal_chosen").trigger( "click" )' :
'').
'
626 if ($conf->use_javascript_ajax) {
627 $conf->global->MAIN_JS_SWITCH_AGENDA = 1;
630 $dateint = sprintf(
"%04d", $year).
'_'.sprintf(
"%02d", $month).
'_'.sprintf(
"%02d", $day);
631 $eventdatetime =
dol_mktime(-1, -1, -1, $month, $day, $year);
636 $curtime =
dol_mktime(0, 0, 0, $month, $day, $year);
638 print
'<div id="dayevent_'.$dateint.
'" class="dayevent tagtable centpercent nobordernopadding" data-datetime="'.$eventdatetime.
'" data-date="'.
dol_print_date($eventdatetime,
"daytext").
'">'.
"\n";
642 print
img_picto(
'today',
'fontawesome_circle_fas_black_7px');
if( $user->socid > 0) if(! $user->hasRight('accounting', 'chartofaccount')) $object
Class to manage agenda events (actions)
Class for Availabilities.
Class to manage Dolibarr users.
dol_get_prev_month($month, $year)
Return previous month.
dol_get_next_month($month, $year)
Return next month.
dol_time_plus_duree($time, $duration_value, $duration_unit, $ruleforendofmonth=0)
Add a delay to a date.
dol_mktime($hour, $minute, $second, $month, $day, $year, $gm='auto', $check=1)
Return a timestamp date built from detailed information (by default a local PHP server timestamp) Rep...
setEventMessages($mesg, $mesgs, $style='mesgs', $messagekey='', $noduplicate=0, $attop=0)
Set event messages in dol_events session object.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
GETPOSTINT($paramname, $method=0)
Return the value of a $_GET or $_POST supervariable, converted into integer.
currentToken()
Return the value of token currently saved into session with name 'token'.
dol_now($mode='auto')
Return date for now.
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
dol_print_date($time, $format='', $tzoutput='auto', $outputlangs=null, $encodetooutput=false)
Output date in a string format according to outputlangs (or langs if not defined).
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
dol_buildpath($path, $type=0, $returnemptyifnotfound=0)
Return path of url or filesystem.
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
getUserRemoteIP()
Return the IP of remote user.
dol_htmloutput_errors($mesgstring='', $mesgarray=array(), $keepembedded=0)
Print formatted error messages to output (Used to show messages on html output).
dol_getdate($timestamp, $fast=false, $forcetimezone='')
Return an array with locale date info.
dol_escape_htmltag($stringtoescape, $keepb=0, $keepn=0, $noescapetags='', $escapeonlyhtmltags=0, $cleanalsojavascript=0)
Returns text escaped for inclusion in HTML alt or title or value tags, or into values of HTML input f...
top_htmlhead($head, $title='', $disablejs=0, $disablehead=0, $arrayofjs=array(), $arrayofcss=array(), $disableforlogin=0, $disablenofollow=0, $disablenoindex=0)
Output html header of a page.
show_bookcal_day_events($day, $month, $year, $today=0)
Show event of a particular day.
llxHeaderVierge($title, $head="", $disablejs=0, $disablehead=0, $arrayofjs=[], $arrayofcss=[])
Show header for booking.
checkNbPostsForASpeceificIp($object, $nb_post_max)
Check if the object exceeded the number of posts for a specific ip in the same week.
httponly_accessforbidden($message='1', $http_response_code=403, $stringalreadysanitized=0)
Show a message to say access is forbidden and stop program.