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')) {
62$langs->loadLangs(array(
"main",
"other",
"dict",
"agenda",
"errors",
"companies"));
64$action =
GETPOST(
'action',
'aZ09');
66$id_availability =
GETPOSTINT(
'id_availability');
78$backtopage =
GETPOST(
"backtopage",
"alpha");
84if ($id_availability > 0) {
85 $result = $availability->fetch($id_availability);
90$nowyear = $nowarray[
'year'];
91$nowmonth = $nowarray[
'mon'];
92$nowday = $nowarray[
'mday'];
95$prev_year = $prev[
'year'];
96$prev_month = $prev[
'month'];
98$next_year = $next[
'year'];
99$next_month = $next[
'month'];
101$max_day_in_prev_month = idate(
"t",
dol_mktime(0, 0, 0, $prev_month, 1, $prev_year,
'gmt'));
102$max_day_in_month = idate(
"t",
dol_mktime(0, 0, 0, $month, 1, $year));
104$tmpday = - idate(
"w",
dol_mktime(12, 0, 0, $month, 1, $year,
'gmt')) + 2;
110$firstdaytoshow =
dol_mktime(0, 0, 0, $prev_month, $max_day_in_prev_month + $tmpday, $prev_year,
'tzuserrel');
111$next_day = 7 - ($max_day_in_month + 1 - $tmpday) % 7;
115$lastdaytoshow =
dol_mktime(0, 0, 0, $next_month, $next_day, $next_year,
'tzuserrel');
117$datechosen =
GETPOST(
'datechosen',
'alpha');
118$datetimechosen =
GETPOSTINT(
'datetimechosen');
119$isdatechosen =
false;
120$timebooking =
GETPOST(
"timebooking");
121$datetimebooking =
GETPOSTINT(
"datetimebooking");
122$durationbooking =
GETPOSTINT(
"durationbooking");
136function llxHeaderVierge($title, $head =
"", $disablejs = 0, $disablehead = 0, $arrayofjs = [], $arrayofcss = [])
138 global
$conf, $langs, $mysoc;
140 top_htmlhead($head, $title, $disablejs, $disablehead, $arrayofjs, $arrayofcss);
142 print
'<body id="mainbody" class="publicnewmemberform">';
150 $urllogo = DOL_URL_ROOT.
'/theme/common/login_logo.png';
152 if (!empty($mysoc->logo_small) && is_readable(
$conf->mycompany->dir_output.
'/logos/thumbs/'.$mysoc->logo_small)) {
153 $urllogo = DOL_URL_ROOT.
'/viewimage.php?modulepart=mycompany&entity='.
$conf->entity.
'&file='.urlencode(
'logos/thumbs/'.$mysoc->logo_small);
154 } elseif (!empty($mysoc->logo) && is_readable(
$conf->mycompany->dir_output.
'/logos/'.$mysoc->logo)) {
155 $urllogo = DOL_URL_ROOT.
'/viewimage.php?modulepart=mycompany&entity='.
$conf->entity.
'&file='.urlencode(
'logos/'.$mysoc->logo);
156 } elseif (is_readable(DOL_DOCUMENT_ROOT.
'/theme/dolibarr_logo.svg')) {
157 $urllogo = DOL_URL_ROOT.
'/theme/dolibarr_logo.svg';
162 print
'<div class="center">';
164 print
'<div class="backgreypublicpayment">';
165 print
'<div class="logopublicpayment">';
167 print
'<a href="'.(getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') ?
getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') :
dol_buildpath(
'/public/ticket/index.php?entity='.
$conf->entity, 1)).
'">';
168 print
'<img id="dolpaymentlogo" src="'.$urllogo.
'">';
172 print
'<div class="clearboth"></div><strong>'.(getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') ?
getDolGlobalString(
'BOOKCAL_PUBLIC_INTERFACE_TOPIC') : $langs->trans(
"BookCalSystem")).
'</strong>';
179 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>';
185 print
'<div class="divmainbodylarge">';
193if ($action ==
'add' ) {
199 $nb_post_max =
getDolGlobalInt(
"MAIN_SECURITY_MAX_POST_ON_PUBLIC_PAGES_BY_IP_ADDRESS", 200);
201 if (!is_object($user)) {
202 $user =
new User($db);
209 $errmsg .= $langs->trans(
"ErrorFieldRequired", $langs->transnoentitiesnoconv(
"Lastname")).
"<br>\n";
213 $errmsg .= $langs->trans(
"ErrorFieldRequired", $langs->transnoentitiesnoconv(
"Firstname")).
"<br>\n";
217 $errmsg .= $langs->trans(
"ErrorFieldRequired", $langs->transnoentitiesnoconv(
"Email")).
"<br>\n";
221 $sql =
"SELECT s.rowid";
222 $sql .=
" FROM ".MAIN_DB_PREFIX.
"socpeople as s";
223 $sql .=
" WHERE s.lastname = '".$db->escape(
GETPOST(
"lastname")).
"'";
224 $sql .=
" AND s.firstname = '".$db->escape(
GETPOST(
"firstname")).
"'";
225 $sql .=
" AND s.email = '".$db->escape(
GETPOST(
"email")).
"'";
226 $resql = $db->query($sql);
229 $num = $db->num_rows($resql);
231 $obj = $db->fetch_object($resql);
232 $idcontact = $obj->rowid;
233 $contact->fetch($idcontact);
235 $contact->lastname =
GETPOST(
"lastname");
236 $contact->firstname =
GETPOST(
"firstname");
237 $contact->email =
GETPOST(
"email");
242 $errmsg .= implode(
'<br>', $contact->errors);
244 $result = $contact->create($user);
247 $errmsg .= $contact->error.
" ".implode(
',', $contact->errors);
253 $errmsg .= $db->lasterror();
260 $actioncomm->label = $langs->trans(
"BookcalBookingTitle");
261 $actioncomm->type =
'AC_RDV';
262 $actioncomm->type_id = 5;
263 $actioncomm->datep =
GETPOSTINT(
"datetimebooking");
264 $actioncomm->datef = $dateend;
265 $actioncomm->note_private =
GETPOST(
"description");
266 $actioncomm->percentage = -1;
267 $actioncomm->fk_bookcal_calendar =
$id;
268 $actioncomm->userownerid = $calendar->visibility;
269 $actioncomm->contact_id = $contact->id;
270 $actioncomm->socpeopleassigned = [
272 'id' => $contact->id,
274 'answer_status' => 0,
281 $errmsg .= implode(
'<br>', $actioncomm->errors);
283 $result = $actioncomm->create($user);
286 $errmsg .= $actioncomm->error.
" ".implode(
',', $actioncomm->errors);
293 $action =
'afteradd';
305$form =
new Form($db);
313if ($action ==
'create') {
314 $backtopage = $_SERVER[
"PHP_SELF"].
'?id='.
$id.
'&datechosen='.$datechosen;
316 $backtopage = DOL_URL_ROOT.
'/public/bookcal/index.php?id='.
$id;
321print
'<div class="bookcalpublicarea centpercent center" style="min-width:30%;width:fit-content;height:70%;top:60%;left: 50%;">';
322print
'<div class="bookcalform" style="min-height:50%">';
323if ($action ==
'afteradd') {
325 print $langs->trans(
"BookingSuccessfullyBooked");
331 print
'<table class="centpercent">';
334 if ($action !=
'create') {
335 print
'<form name="formsearch" action="'.$_SERVER[
"PHP_SELF"].
'">';
336 print
'<input type="hidden" name="id" value="'.$id.
'">';
338 $nav =
'<a href="?id='.$id.
"&year=".$prev_year.
"&month=".$prev_month.$param.
'"><i class="fa fa-chevron-left"></i></a> '.
"\n";
339 $nav .=
' <span id="month_name">'.dol_print_date(
dol_mktime(0, 0, 0, $month, 1, $year),
"%b %Y");
340 $nav .=
" </span>\n";
341 $nav .=
' <a href="?id='.$id.
"&year=".$next_year.
"&month=".$next_month.$param.
'"><i class="fa fa-chevron-right"></i></a>'.
"\n";
342 if (empty(
$conf->dol_optimize_smallscreen)) {
343 $nav .=
' <a href="?id='.$id.
"&year=".$nowyear.
"&month=".$nowmonth.
"&day=".$nowday.$param.
'" class="datenowlink">'.$langs->trans(
"Today").
'</a> ';
345 $nav .= $form->selectDate($dateselect,
'dateselect', 0, 0, 1,
'', 1, 0);
346 $nav .=
'<button type="submit" class="liste_titre button_search valignmiddle" name="button_search_x" value="x"><span class="fa fa-search"></span></button>';
353 print
'<div class="bookingtab hidden" style="height:50%">';
354 print
'<div id="bookingtabspandate"></div>';
360 if ($action ==
"create") {
362 if (empty($datetimebooking)) {
363 $timebookingarray = explode(
" - ", $timebooking);
364 $timestartarray = explode(
":", $timebookingarray[0]);
365 $timeendarray = explode(
":", $timebookingarray[1]);
369 print
'<span>'.img_picto(
"",
"calendar").
" ".
dol_print_date($datetimebooking,
'dayhourtext').
'</span>';
370 print
'<div class="center"><a href="'.$_SERVER[
"PHP_SELF"].
'?id=1&year=2024&month=2" class="small">('.$langs->trans(
"SelectANewDate").
')</a></div>';
374 print
'<form method="POST" action="'.$_SERVER[
"PHP_SELF"].
'">';
375 print
'<table class="border" summary="form to subscribe" id="tablesubscribe">'.
"\n";
376 print
'<input type="hidden" name="token" value="'.newToken().
'">';
377 print
'<input type="hidden" name="action" value="add">';
378 print
'<input type="hidden" name="datetimebooking" value="'.$datetimebooking.
'">';
379 print
'<input type="hidden" name="datechosen" value="'.$datechosen.
'">';
380 print
'<input type="hidden" name="id" value="'.$id.
'">';
381 print
'<input type="hidden" name="durationbooking" value="'.$durationbooking.
'">';
384 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";
386 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";
388 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";
392 print
'<td class="tdtop">';
393 print $langs->trans(
"Message");
394 print
'<textarea name="description" id="description" wrap="soft" class="quatrevingtpercent" rows="'.ROWS_4.
'">'.
dol_escape_htmltag(
GETPOST(
'description',
'restricthtml'), 0, 1).
'</textarea></td>';
396 print
'</table>'.
"\n";
397 print
'<div class="center">';
398 print
'<input type="submit" value="'.$langs->trans(
"Submit").
'" id="submitsave" class="button">';
404 print
'<table class="centpercent noborder nocellnopadd cal_pannel cal_month">';
405 print
' <tr class="">';
407 print
' <td class="center hideonsmartphone">#</td>';
410 $numdayinweek = (($i + (isset(
$conf->global->MAIN_START_WEEK) ?
$conf->global->MAIN_START_WEEK : 1)) % 7);
411 if (!empty(
$conf->dol_optimize_smallscreen)) {
412 print
' <td class="center bold uppercase tdfordaytitle'.($i == 0 ?
' borderleft' :
'').
'">';
413 $labelshort = array(0 =>
'SundayMin', 1 =>
'MondayMin', 2 =>
'TuesdayMin', 3 =>
'WednesdayMin', 4 =>
'ThursdayMin', 5 =>
'FridayMin', 6 =>
'SaturdayMin');
414 print $langs->trans($labelshort[$numdayinweek]);
417 print
' <td class="center minwidth75 bold uppercase small tdoverflowmax50 tdfordaytitle'.($i == 0 ?
' borderleft' :
'').
'">';
419 $labelshort = array(0 =>
'Sunday', 1 =>
'Monday', 2 =>
'Tuesday', 3 =>
'Wednesday', 4 =>
'Thursday', 5 =>
'Friday', 6 =>
'Saturday');
420 print $langs->trans($labelshort[$numdayinweek]);
428 $todaytms =
dol_mktime(0, 0, 0, $todayarray[
'mon'], $todayarray[
'mday'], $todayarray[
'year']);
431 $arrayofavailabledays = array();
433 $arrayofavailabilities = $availability->fetchAll(
'',
'', 0, 0,
'(status:=:1) AND (fk_bookcal_calendar:=:'.((
int) $id).
')');
434 if ($arrayofavailabilities < 0) {
437 foreach ($arrayofavailabilities as $key => $value) {
440 for ($i = $startarray[
'mday']; $i <= $endarray[
'mday']; $i++) {
441 if ($todayarray[
'mon'] >= $startarray[
'mon'] && $todayarray[
'mon'] <= $endarray[
'mon']) {
442 $arrayofavailabledays[
dol_mktime(0, 0, 0, $todayarray[
'mon'], $i, $todayarray[
'year'])] =
dol_mktime(0, 0, 0, $todayarray[
'mon'], $i, $todayarray[
'year']);
448 for ($iter_week = 0; $iter_week < 6; $iter_week++) {
452 $currdate0 = sprintf(
"%04d", $prev_year).sprintf(
"%02d", $prev_month).sprintf(
"%02d", $max_day_in_prev_month + $tmpday);
453 } elseif ($tmpday <= $max_day_in_month) {
454 $currdate0 = sprintf(
"%04d", $year).sprintf(
"%02d", $month).sprintf(
"%02d", $tmpday);
456 $currdate0 = sprintf(
"%04d", $next_year).sprintf(
"%02d", $next_month).sprintf(
"%02d", $tmpday - $max_day_in_month);
459 $numweek0 = idate(
"W", strtotime(date($currdate0)));
461 echo
' <td class="center weeknumber opacitymedium hideonsmartphone" style="min-width: 40px">'.$numweek0.
'</td>';
463 for ($iter_day = 0; $iter_day < 7; $iter_day++) {
466 $style =
'cal_other_month cal_past';
467 if ($iter_day == 6) {
468 $style .=
' cal_other_month_right';
470 echo
' <td class="'.$style.
' nowrap tdtop" width="14%">';
473 } elseif ($tmpday <= $max_day_in_month) {
475 $curtime =
dol_mktime(0, 0, 0, $month, $tmpday, $year);
476 $style =
'cal_current_month';
477 if ($iter_day == 6) {
478 $style .=
' cal_current_month_right';
481 if ($todayarray[
'mday'] == $tmpday && $todayarray[
'mon'] == $month && $todayarray[
'year'] == $year) {
485 if ($curtime > $todaytms && in_array($curtime, $arrayofavailabledays)) {
486 $style .=
' cal_available cursorpointer';
488 if ($curtime < $todaytms) {
489 $style .=
' cal_past';
491 $dateint = sprintf(
"%04d", $year).
'_'.sprintf(
"%02d", $month).
'_'.sprintf(
"%02d", $tmpday);
492 if (!empty(explode(
'dayevent_', $datechosen)[1]) && explode(
'dayevent_', $datechosen)[1] == $dateint) {
493 $style .=
' cal_chosen';
494 $isdatechosen =
true;
496 echo
' <td class="'.$style.
' nowrap tdtop" width="14%">';
501 $style =
'cal_other_month';
502 if ($iter_day == 6) {
503 $style .=
' cal_other_month_right';
505 echo
' <td class="'.$style.
' nowrap tdtop" width="14%">';
517 print
'<div class="center bookingtab" style="height:50%">';
518 print
'<div style="height:100%">';
519 print
'<form id="formbooking" name="formbooking" method="POST" action="'.$_SERVER[
"PHP_SELF"].
'">';
520 print
'<input type="hidden" name="id" value="'.$id.
'">';
521 print
'<input type="hidden" name="token" value="'.newToken().
'">';
522 print
'<input type="hidden" name="action" value="create">';
523 print
'<input type="hidden" id="datechosen" name="datechosen" value="">';
524 print
'<input type="hidden" id="datetimechosen" name="datetimechosen" value="">';
525 print
'<input type="hidden" id="durationbooking" name="durationbooking" value="">';
527 print
'<div id="bookinghoursection">';
528 print
'<br><br><br><br><br><br><div class="opacitymedium center">'.$langs->trans(
"SelectADay").
'</div>';
543 function generateBookingButtons(timearray, datestring){
544 console.log("We generate all booking buttons of "+datestring);
547 for (index in timearray){
548 let hour = new Date("2000-01-01T" + index + ":00");
549 duration = timearray[index];
550 isalreadybooked = false;
553 isalreadybooked = true;
555 hour.setMinutes(hour.getMinutes() + duration);
557 let hours = hour.getHours().toString().padStart(2, "0"); // Formatter pour obtenir deux chiffres
558 let mins = hour.getMinutes().toString().padStart(2, "0"); // Formatter pour obtenir deux chiffres
560 timerange = index + " - " + `${hours}:${mins}`;
561 str += \'<input class="button btnsubmitbooking \'+(isalreadybooked == true ? "btnbookcalbooked" : "")+\'" type="submit" name="timebooking" value="\'+timerange+\'" data-duration="\'+duration+\'"><br>\';
564 $("#bookinghoursection").html(str);
565 $(".btnsubmitbooking").on("click", function(){
566 duration = $(this).data("duration");
567 $("#durationbooking").val(duration);
570 print
'$(document).ready(function() {
571 $(".cal_available").on("click", function(){
572 console.log("We click on cal_available");
573 $(".cal_chosen").removeClass("cal_chosen");
574 $(this).addClass("cal_chosen");
575 datestring = $(this).children("div").data("date");
578 url: "'.DOL_URL_ROOT.
'/public/bookcal/bookcalAjax.php",
580 action: "verifyavailability",
582 datetocheck: $(this).children("div").data("datetime"),
585 }).done(function (data) {
586 console.log("We show all booking");
587 if (data["code"] == "SUCCESS") {
588 /* TODO Replace this with a creating of allavailable hours button */
590 timearray = data["availability"];
591 console.log(timearray);
592 generateBookingButtons(timearray, datestring);
593 $(".btnbookcalbooked").prop("disabled", true);
595 if(data["code"] == "NO_DATA_FOUND"){
596 console.log("No booking to hide");
598 console.log(data["message"]);
602 $(".bookingtab").removeClass("hidden");
603 $("#bookingtabspandate").text($(this).children("div").data("date"));
604 $("#datechosen").val($(this).children("div").attr("id"));
605 $("#datetimechosen").val($(this).children("div").data("datetime"));
610 '.($datechosen ?
'$(".cal_chosen").trigger( "click" )' :
'').
'
630 if (
$conf->use_javascript_ajax) {
631 $conf->global->MAIN_JS_SWITCH_AGENDA = 1;
634 $dateint = sprintf(
"%04d", $year).
'_'.sprintf(
"%02d", $month).
'_'.sprintf(
"%02d", $day);
635 $eventdatetime =
dol_mktime(-1, -1, -1, $month, $day, $year);
640 $curtime =
dol_mktime(0, 0, 0, $month, $day, $year);
642 print
'<div id="dayevent_'.$dateint.
'" class="dayevent tagtable centpercent nobordernopadding" data-datetime="'.$eventdatetime.
'" data-date="'.
dol_print_date($eventdatetime,
"daytext").
'">'.
"\n";
646 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.
global $conf
The following vars must be defined: $type2label $form $conf, $lang, The following vars may also be de...
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.