dolibarr 21.0.0-alpha
main.inc.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2002-2007 Rodolphe Quiedeville <rodolphe@quiedeville.org>
3 * Copyright (C) 2003 Xavier Dutoit <doli@sydesy.com>
4 * Copyright (C) 2004-2021 Laurent Destailleur <eldy@users.sourceforge.net>
5 * Copyright (C) 2004 Sebastien Di Cintio <sdicintio@ressource-toi.org>
6 * Copyright (C) 2004 Benoit Mortier <benoit.mortier@opensides.be>
7 * Copyright (C) 2005-2021 Regis Houssin <regis.houssin@inodbox.com>
8 * Copyright (C) 2011-2014 Philippe Grand <philippe.grand@atoo-net.com>
9 * Copyright (C) 2008 Matteli
10 * Copyright (C) 2011-2016 Juanjo Menent <jmenent@2byte.es>
11 * Copyright (C) 2012 Christophe Battarel <christophe.battarel@altairis.fr>
12 * Copyright (C) 2014-2015 Marcos García <marcosgdf@gmail.com>
13 * Copyright (C) 2015 Raphaël Doursenaud <rdoursenaud@gpcsolutions.fr>
14 * Copyright (C) 2020 Demarest Maxime <maxime@indelog.fr>
15 * Copyright (C) 2020 Charlene Benke <charlie@patas-monkey.com>
16 * Copyright (C) 2021-2024 Frédéric France <frederic.france@free.fr>
17 * Copyright (C) 2021 Alexandre Spangaro <aspangaro@open-dsi.fr>
18 * Copyright (C) 2023 Joachim Küter <git-jk@bloxera.com>
19 * Copyright (C) 2023 Eric Seigne <eric.seigne@cap-rel.fr>
20 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
21 *
22 * This program is free software; you can redistribute it and/or modify
23 * it under the terms of the GNU General Public License as published by
24 * the Free Software Foundation; either version 3 of the License, or
25 * (at your option) any later version.
26 *
27 * This program is distributed in the hope that it will be useful,
28 * but WITHOUT ANY WARRANTY; without even the implied warranty of
29 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
30 * GNU General Public License for more details.
31 *
32 * You should have received a copy of the GNU General Public License
33 * along with this program. If not, see <https://www.gnu.org/licenses/>.
34 */
35
42//@ini_set('memory_limit', '128M'); // This may be useless if memory is hard limited by your PHP
43
44// For optional tuning. Enabled if environment variable MAIN_SHOW_TUNING_INFO is defined.
45$micro_start_time = 0;
46if (!empty($_SERVER['MAIN_SHOW_TUNING_INFO'])) {
47 list($usec, $sec) = explode(" ", microtime());
48 $micro_start_time = ((float) $usec + (float) $sec);
49 // Add Xdebug code coverage
50 //define('XDEBUGCOVERAGE',1);
51 if (defined('XDEBUGCOVERAGE')) {
52 xdebug_start_code_coverage();
53 }
54}
55
63{
64 $arrayofcommonemoji = array(
65 'misc' => array('2600', '26FF'), // Miscellaneous Symbols
66 'ding' => array('2700', '27BF'), // Dingbats
67 '????' => array('9989', '9989'), // Variation Selectors
68 'vars' => array('FE00', 'FE0F'), // Variation Selectors
69 'pict' => array('1F300', '1F5FF'), // Miscellaneous Symbols and Pictographs
70 'emot' => array('1F600', '1F64F'), // Emoticons
71 'tran' => array('1F680', '1F6FF'), // Transport and Map Symbols
72 'flag' => array('1F1E0', '1F1FF'), // Flags (note: may be 1F1E6 instead of 1F1E0)
73 'supp' => array('1F900', '1F9FF'), // Supplemental Symbols and Pictographs
74 );
75
76 return $arrayofcommonemoji;
77}
78
87{
88 $newstringnumentity = preg_replace('/;$/', '', $matches[1]);
89 //print ' $newstringnumentity='.$newstringnumentity;
90
91 if (preg_match('/^x/i', $newstringnumentity)) { // if numeric is hexadecimal
92 $newstringnumentity = hexdec(preg_replace('/^x/i', '', $newstringnumentity));
93 } else {
94 $newstringnumentity = (int) $newstringnumentity;
95 }
96
97 // The numeric values we don't want as entities because they encode ascii char, and why using html entities on ascii except for haking ?
98 if (($newstringnumentity >= 65 && $newstringnumentity <= 90) || ($newstringnumentity >= 97 && $newstringnumentity <= 122)) {
99 return chr((int) $newstringnumentity);
100 }
101
102 // The numeric values we want in UTF8 instead of entities because it is emoji
103 $arrayofemojis = getArrayOfEmoji();
104 foreach ($arrayofemojis as $valarray) {
105 if ($newstringnumentity >= hexdec($valarray[0]) && $newstringnumentity <= hexdec($valarray[1])) {
106 // This is a known emoji
107 return html_entity_decode($matches[0], ENT_COMPAT | ENT_HTML5, 'UTF-8');
108 }
109 }
110
111 return '&#'.$matches[1]; // Value will be unchanged because regex was /&#( )/
112}
113
123function testSqlAndScriptInject($val, $type)
124{
125 // Decode string first because a lot of things are obfuscated by encoding or multiple encoding.
126 // So <svg o&#110;load='console.log(&quot;123&quot;)' become <svg onload='console.log(&quot;123&quot;)'
127 // So "&colon;&apos;" become ":'" (due to ENT_HTML5)
128 // So "&Tab;&NewLine;" become ""
129 // So "&lpar;&rpar;" become "()"
130
131 // Loop to decode until no more things to decode.
132 //print "before decoding $val\n";
133 do {
134 $oldval = $val;
135 $val = html_entity_decode($val, ENT_QUOTES | ENT_HTML5); // Decode '&colon;', '&apos;', '&Tab;', '&NewLine', ...
136 // Sometimes we have entities without the ; at end so html_entity_decode does not work but entities is still interpreted by browser.
137 $val = preg_replace_callback(
138 '/&#(x?[0-9][0-9a-f]+;?)/i',
143 static function ($m) {
144 // Decode '&#110;', ...
146 },
147 $val
148 );
149
150 // We clean html comments because some hacks try to obfuscate evil strings by inserting HTML comments. Example: on<!-- -->error=alert(1)
151 $val = preg_replace('/<!--[^>]*-->/', '', $val);
152 $val = preg_replace('/[\r\n\t]/', '', $val);
153 } while ($oldval != $val);
154 //print "type = ".$type." after decoding: ".$val."\n";
155
156 $inj = 0;
157
158 // We check string because some hacks try to obfuscate evil strings by inserting non printable chars. Example: 'java(ascci09)scr(ascii00)ipt' is processed like 'javascript' (whatever is place of evil ascii char)
159 // We should use dol_string_nounprintableascii but function is not yet loaded/available
160 // Example of valid UTF8 chars:
161 // utf8 or utf8mb3: '\x09', '\x0A', '\x0D', '\x7E'
162 // utf8 or utf8mb3: '\xE0\xA0\x80'
163 // utf8mb4: '\xF0\x9D\x84\x9E' (so this may be refused by the database insert if pagecode is utf8=utf8mb3)
164 $newval = preg_replace('/[\x00-\x08\x0B-\x0C\x0E-\x1F\x7F]/u', '', $val); // /u operator makes UTF8 valid characters being ignored so are not included into the replace
165
166 // Note that $newval may also be completely empty '' when non valid UTF8 are found.
167 if ($newval != $val) {
168 // If $val has changed after removing non valid UTF8 chars, it means we have an evil string.
169 $inj += 1;
170 }
171 //print 'inj='.$inj.'-type='.$type.'-val='.$val.'-newval='.$newval."\n";
172
173 // For SQL Injection (only GET are used to scan for such injection strings)
174 if ($type == 1 || $type == 3) {
175 // Note the \s+ is replaced into \s* because some spaces may have been modified in previous loop
176 $inj += preg_match('/delete\s*from/i', $val);
177 $inj += preg_match('/create\s*table/i', $val);
178 $inj += preg_match('/insert\s*into/i', $val);
179 $inj += preg_match('/select\s*from/i', $val);
180 $inj += preg_match('/into\s*(outfile|dumpfile)/i', $val);
181 $inj += preg_match('/user\s*\‍(/i', $val); // avoid to use function user() or mysql_user() that return current database login
182 $inj += preg_match('/information_schema/i', $val); // avoid to use request that read information_schema database
183 $inj += preg_match('/<svg/i', $val); // <svg can be allowed in POST
184 $inj += preg_match('/update[^&=\w].*set.+=/i', $val); // the [^&=\w] test is to avoid error when request is like action=update&...set... or &updatemodule=...set...
185 $inj += preg_match('/union.+select/i', $val);
186 }
187 if ($type == 3) {
188 // Note the \s+ is replaced into \s* because some spaces may have been modified in previous loop
189 $inj += preg_match('/select|update|delete|truncate|replace|group\s*by|concat|count|from|union/i', $val);
190 }
191 if ($type != 2) { // Not common key strings, so we can check them both on GET and POST
192 $inj += preg_match('/updatexml\‍(/i', $val);
193 $inj += preg_match('/(\.\.%2f)+/i', $val);
194 $inj += preg_match('/\s@@/', $val);
195 }
196 // For XSS Injection done by closing textarea to execute content into a textarea field
197 $inj += preg_match('/<\/textarea/i', $val);
198 // For XSS Injection done by adding javascript with script
199 // This is all cases a browser consider text is javascript:
200 // When it found '<script', 'javascript:', '<style', 'onload\s=' on body tag, '="&' on a tag size with old browsers
201 // All examples on page: http://ha.ckers.org/xss.html#XSScalc
202 // More on https://www.owasp.org/index.php/XSS_Filter_Evasion_Cheat_Sheet
203 $inj += preg_match('/<audio/i', $val);
204 $inj += preg_match('/<embed/i', $val);
205 $inj += preg_match('/<iframe/i', $val);
206 $inj += preg_match('/<object/i', $val);
207 $inj += preg_match('/<script/i', $val);
208 $inj += preg_match('/Set\.constructor/i', $val); // ECMA script 6
209 if (!defined('NOSTYLECHECK')) {
210 $inj += preg_match('/<style/i', $val);
211 }
212 $inj += preg_match('/base\s+href/si', $val);
213 $inj += preg_match('/=data:/si', $val);
214 // List of dom events is on https://www.w3schools.com/jsref/dom_obj_event.asp and https://developer.mozilla.org/en-US/docs/Web/Events
215 $inj += preg_match('/on(mouse|drag|key|load|touch|pointer|select|transition)[a-z]*\s*=/i', $val); // onmousexxx can be set on img or any html tag like <img title='...' onmouseover=alert(1)>
216 $inj += preg_match('/on(abort|after|animation|auxclick|before|blur|cancel|canplay|canplaythrough|change|click|close|contextmenu|cuechange|copy|cut)[a-z]*\s*=/i', $val);
217 $inj += preg_match('/on(dblclick|drop|durationchange|emptied|end|ended|error|focus|focusin|focusout|formdata|gotpointercapture|hashchange|input|invalid)[a-z]*\s*=/i', $val);
218 $inj += preg_match('/on(lostpointercapture|offline|online|pagehide|pageshow)[a-z]*\s*=/i', $val);
219 $inj += preg_match('/on(paste|pause|play|playing|progress|ratechange|reset|resize|scroll|search|seeked|seeking|show|stalled|start|submit|suspend)[a-z]*\s*=/i', $val);
220 $inj += preg_match('/on(timeupdate|toggle|unload|volumechange|waiting|wheel)[a-z]*\s*=/i', $val);
221 // More not into the previous list
222
223 $inj += preg_match('/on(repeat|begin|finish|beforeinput)[a-z]*\s*=/i', $val);
224
225 // We refuse html into html because some hacks try to obfuscate evil strings by inserting HTML into HTML. Example: <img on<a>error=alert(1) to bypass test on onerror
226 $tmpval = preg_replace('/<[^<]+>/', '', $val);
227 // List of dom events is on https://www.w3schools.com/jsref/dom_obj_event.asp and https://developer.mozilla.org/en-US/docs/Web/Events
228 $inj += preg_match('/on(mouse|drag|key|load|touch|pointer|select|transition)[a-z]*\s*=/i', $tmpval); // onmousexxx can be set on img or any html tag like <img title='...' onmouseover=alert(1)>
229 $inj += preg_match('/on(abort|after|animation|auxclick|before|blur|cancel|canplay|canplaythrough|change|click|close|contextmenu|cuechange|copy|cut)[a-z]*\s*=/i', $tmpval);
230 $inj += preg_match('/on(dblclick|drop|durationchange|emptied|end|ended|error|focus|focusin|focusout|formdata|gotpointercapture|hashchange|input|invalid)[a-z]*\s*=/i', $tmpval);
231 $inj += preg_match('/on(lostpointercapture|offline|online|pagehide|pageshow)[a-z]*\s*=/i', $tmpval);
232 $inj += preg_match('/on(paste|pause|play|playing|progress|ratechange|reset|resize|scroll|search|seeked|seeking|show|stalled|start|submit|suspend)[a-z]*\s*=/i', $tmpval);
233 $inj += preg_match('/on(timeupdate|toggle|unload|volumechange|waiting|wheel)[a-z]*\s*=/i', $tmpval);
234 // More not into the previous list
235 $inj += preg_match('/on(repeat|begin|finish|beforeinput)[a-z]*\s*=/i', $tmpval);
236
237 //$inj += preg_match('/on[A-Z][a-z]+\*=/', $val); // To lock event handlers onAbort(), ...
238 $inj += preg_match('/&#58;|&#0000058|&#x3A/i', $val); // refused string ':' encoded (no reason to have it encoded) to lock 'javascript:...'
239 $inj += preg_match('/j\s*a\s*v\s*a\s*s\s*c\s*r\s*i\s*p\s*t\s*:/i', $val);
240 $inj += preg_match('/vbscript\s*:/i', $val);
241 // For XSS Injection done by adding javascript closing html tags like with onmousemove, etc... (closing a src or href tag with not cleaned param)
242 if ($type == 1 || $type == 3) {
243 $val = str_replace('enclosure="', 'enclosure=X', $val); // We accept enclosure=" for the export/import module
244 $inj += preg_match('/"/i', $val); // We refused " in GET parameters value.
245 }
246 if ($type == 2) {
247 $inj += preg_match('/[:;"\'<>\?\‍(\‍){}\$%]/', $val); // PHP_SELF is a file system (or url path without parameters). It can contains spaces.
248 }
249
250 return $inj;
251}
252
261function analyseVarsForSqlAndScriptsInjection(&$var, $type, $stopcode = 1)
262{
263 if (is_array($var)) {
264 foreach ($var as $key => $value) { // Warning, $key may also be used for attacks
265 // Exclude check for some variable keys
266 if ($type === 0 && defined('NOSCANPOSTFORINJECTION') && is_array(constant('NOSCANPOSTFORINJECTION')) && in_array($key, constant('NOSCANPOSTFORINJECTION'))) {
267 continue;
268 }
269
270 if (analyseVarsForSqlAndScriptsInjection($key, $type, $stopcode) && analyseVarsForSqlAndScriptsInjection($value, $type, $stopcode)) {
271 //$var[$key] = $value; // This is useless
272 } else {
273 http_response_code(403);
274
275 // Get remote IP: PS: We do not use getRemoteIP(), function is not yet loaded and we need a value that can't be spoofed
276 $ip = (empty($_SERVER['REMOTE_ADDR']) ? 'unknown' : $_SERVER['REMOTE_ADDR']);
277
278 if ($stopcode) {
279 $errormessage = 'Access refused to '.htmlentities($ip, ENT_COMPAT, 'UTF-8').' by SQL or Script injection protection in main.inc.php:analyseVarsForSqlAndScriptsInjection type='.htmlentities((string) $type, ENT_COMPAT, 'UTF-8');
280 //$errormessage .= ' paramkey='.htmlentities($key, ENT_COMPAT, 'UTF-8'); // Disabled to avoid text injection
281
282 $errormessage2 = 'page='.htmlentities((empty($_SERVER["REQUEST_URI"]) ? '' : $_SERVER["REQUEST_URI"]), ENT_COMPAT, 'UTF-8');
283 $errormessage2 .= ' paramtype='.htmlentities((string) $type, ENT_COMPAT, 'UTF-8');
284 $errormessage2 .= ' paramkey='.htmlentities($key, ENT_COMPAT, 'UTF-8');
285 $errormessage2 .= ' paramvalue='.htmlentities($value, ENT_COMPAT, 'UTF-8');
286
287 print $errormessage;
288 print "<br>\n";
289 print 'Try to go back, fix data of your form and resubmit it. You can contact also your technical support.';
290
291 print "\n".'<!--'."\n";
292 print $errormessage2;
293 print "\n".'-->';
294
295 // Add entry into the PHP server error log
296 if (function_exists('error_log')) {
297 error_log($errormessage.' '.substr($errormessage2, 2000));
298 }
299
300 // Note: No addition into security audit table is done because we don't want to execute code in such a case.
301 // Detection of too many such requests can be done with a fail2ban rule on 403 error code or into the PHP server error log.
302
303
304 if (class_exists('PHPUnit\Framework\TestSuite')) {
305 $message = $errormessage.' '.substr($errormessage2, 2000);
306 throw new Exception("Security injection exception: $message");
307 }
308 exit;
309 } else {
310 return false;
311 }
312 }
313 }
314 return true;
315 } else {
316 return (testSqlAndScriptInject($var, $type) <= 0);
317 }
318}
319
320// To disable the WAF for GET and POST and PHP_SELF, uncomment this
321//define('NOSCANPHPSELFFORINJECTION', 1);
322//define('NOSCANGETFORINJECTION', 1);
323//define('NOSCANPOSTFORINJECTION', 1 or 2);
324
325// Check consistency of NOREQUIREXXX DEFINES
326if ((defined('NOREQUIREDB') || defined('NOREQUIRETRAN')) && !defined('NOREQUIREMENU')) {
327 print 'If define NOREQUIREDB or NOREQUIRETRAN are set, you must also set NOREQUIREMENU or not set them.';
328 exit;
329}
330if (defined('NOREQUIREUSER') && !defined('NOREQUIREMENU')) {
331 print 'If define NOREQUIREUSER is set, you must also set NOREQUIREMENU or not set it.';
332 exit;
333}
334
335// Sanity check on URL
336if (!defined('NOSCANPHPSELFFORINJECTION') && !empty($_SERVER["PHP_SELF"])) {
337 $morevaltochecklikepost = array($_SERVER["PHP_SELF"]);
338 analyseVarsForSqlAndScriptsInjection($morevaltochecklikepost, 2);
339}
340// Sanity check on GET parameters
341if (!defined('NOSCANGETFORINJECTION') && !empty($_SERVER["QUERY_STRING"])) {
342 // Note: QUERY_STRING is url encoded, but $_GET and $_POST are already decoded
343 // Because the analyseVarsForSqlAndScriptsInjection is designed for already url decoded value, we must decode QUERY_STRING
344 // Another solution is to provide $_GET as parameter with analyseVarsForSqlAndScriptsInjection($_GET, 1);
345 $morevaltochecklikeget = array(urldecode($_SERVER["QUERY_STRING"]));
346 analyseVarsForSqlAndScriptsInjection($morevaltochecklikeget, 1);
347}
348// Sanity check on POST
349if (!defined('NOSCANPOSTFORINJECTION') || is_array(constant('NOSCANPOSTFORINJECTION'))) {
351}
352
353// This is to make Dolibarr working with Plesk
354if (!empty($_SERVER['DOCUMENT_ROOT']) && substr($_SERVER['DOCUMENT_ROOT'], -6) !== 'htdocs') {
355 set_include_path($_SERVER['DOCUMENT_ROOT'].'/htdocs');
356}
357
358// Include the conf.php and functions.lib.php and security.lib.php. This defined the constants like DOL_DOCUMENT_ROOT, DOL_DATA_ROOT, DOL_URL_ROOT...
359require_once 'filefunc.inc.php';
360
361// If there is a POST parameter to tell to save automatically some POST parameters into cookies, we do it.
362// This is used for example by form of boxes to save personalization of some options.
363// DOL_AUTOSET_COOKIE=cookiename:val1,val2 and cookiename_val1=aaa cookiename_val2=bbb will set cookie_name with value json_encode(array('val1'=> , ))
364if (GETPOST("DOL_AUTOSET_COOKIE")) {
365 $tmpautoset = explode(':', GETPOST("DOL_AUTOSET_COOKIE"), 2);
366 $tmplist = explode(',', $tmpautoset[1]);
367 $cookiearrayvalue = array();
368 foreach ($tmplist as $tmpkey) {
369 $postkey = $tmpautoset[0].'_'.$tmpkey;
370 //var_dump('tmpkey='.$tmpkey.' postkey='.$postkey.' value='.GETPOST($postkey);
371 if (GETPOST($postkey)) {
372 $cookiearrayvalue[$tmpkey] = GETPOST($postkey);
373 }
374 }
375 $cookiename = $tmpautoset[0];
376 $cookievalue = json_encode($cookiearrayvalue);
377 //var_dump('setcookie cookiename='.$cookiename.' cookievalue='.$cookievalue);
378 if (PHP_VERSION_ID < 70300) {
379 setcookie($cookiename, empty($cookievalue) ? '' : $cookievalue, empty($cookievalue) ? 0 : (time() + (86400 * 354)), '/', '', ((empty($dolibarr_main_force_https) && isHTTPS() === false) ? false : true), true); // keep cookie 1 year and add tag httponly
380 } else {
381 // Only available for php >= 7.3
382 $cookieparams = array(
383 'expires' => empty($cookievalue) ? 0 : (time() + (86400 * 354)),
384 'path' => '/',
385 //'domain' => '.mywebsite.com', // the dot at the beginning allows compatibility with subdomains
386 'secure' => ((empty($dolibarr_main_force_https) && isHTTPS() === false) ? false : true),
387 'httponly' => true,
388 'samesite' => 'Lax' // None || Lax || Strict
389 );
390 setcookie($cookiename, empty($cookievalue) ? '' : $cookievalue, $cookieparams);
391 }
392 if (empty($cookievalue)) {
393 unset($_COOKIE[$cookiename]);
394 }
395}
396
397// Set the handler of session
398// if (ini_get('session.save_handler') == 'user')
399if (!empty($php_session_save_handler) && $php_session_save_handler == 'db') {
400 require_once 'core/lib/phpsessionin'.$php_session_save_handler.'.lib.php';
401}
402
403// Init session. Name of session is specific to Dolibarr instance.
404// Must be done after the include of filefunc.inc.php so global variables of conf file are defined (like $dolibarr_main_instance_unique_id or $dolibarr_main_force_https).
405// Note: the function dol_getprefix() is defined into functions.lib.php but may have been defined to return a different key to manage another area to protect.
406$prefix = dol_getprefix('');
407$sessionname = 'DOLSESSID_'.$prefix;
408$sessiontimeout = 'DOLSESSTIMEOUT_'.$prefix;
409if (!empty($_COOKIE[$sessiontimeout])) {
410 ini_set('session.gc_maxlifetime', $_COOKIE[$sessiontimeout]);
411}
412
413// This create lock, released by session_write_close() or end of page.
414// We need this lock as long as we read/write $_SESSION ['vars']. We can remove lock when finished.
415if (!defined('NOSESSION')) {
416 if (PHP_VERSION_ID < 70300) {
417 session_set_cookie_params(0, '/', null, ((empty($dolibarr_main_force_https) && isHTTPS() === false) ? false : true), true); // Add tag secure and httponly on session cookie (same as setting session.cookie_httponly into php.ini). Must be called before the session_start.
418 } else {
419 // Only available for php >= 7.3
420 $sessioncookieparams = array(
421 'lifetime' => 0,
422 'path' => '/',
423 //'domain' => '.mywebsite.com', // the dot at the beginning allows compatibility with subdomains
424 'secure' => ((empty($dolibarr_main_force_https) && isHTTPS() === false) ? false : true),
425 'httponly' => true,
426 'samesite' => 'Lax' // None || Lax || Strict
427 );
428 session_set_cookie_params($sessioncookieparams);
429 }
430 session_name($sessionname);
431 dol_session_start(); // This call the open and read of session handler
432 //exit; // this exist generates a call to write and close
433}
434
435
436// Init the 6 global objects, this include will make the 'new Xxx()' and set properties for: $conf, $db, $langs, $user, $mysoc, $hookmanager
437require_once 'master.inc.php';
438
439// Uncomment this and set session.save_handler = user to use local session storing
440// include DOL_DOCUMENT_ROOT.'/core/lib/phpsessionindb.inc.php
441
442// If software has been locked. Only login $conf->global->MAIN_ONLY_LOGIN_ALLOWED is allowed.
443if (getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')) {
444 $ok = 0;
445 if ((!session_id() || !isset($_SESSION["dol_login"])) && !isset($_POST["username"]) && !empty($_SERVER["GATEWAY_INTERFACE"])) {
446 $ok = 1; // We let working pages if not logged and inside a web browser (login form, to allow login by admin)
447 } elseif (isset($_POST["username"]) && in_array($_POST["username"], explode(';', getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')))) {
448 $ok = 1; // We let working pages that is a login submission (login submit, to allow login by admin)
449 } elseif (defined('NOREQUIREDB')) {
450 $ok = 1; // We let working pages that don't need database access (xxx.css.php)
451 } elseif (defined('EVEN_IF_ONLY_LOGIN_ALLOWED')) {
452 $ok = 1; // We let working pages that ask to work even if only login enabled (logout.php)
453 } elseif (session_id() && isset($_SESSION["dol_login"]) && in_array($_SESSION["dol_login"], explode(';', getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')))) {
454 $ok = 1; // We let working if user is allowed admin
455 }
456 if (!$ok) {
457 if (session_id() && isset($_SESSION["dol_login"]) && !in_array($_SESSION["dol_login"], explode(';', getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')))) {
458 print 'Sorry, your application is offline.'."\n";
459 print 'You are logged with user "'.$_SESSION["dol_login"].'" and only administrator users (' . str_replace(';', ', ', getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')).') is allowed to connect for the moment.'."\n";
460 $nexturl = DOL_URL_ROOT.'/user/logout.php?token='.newToken();
461 print 'Please try later or <a href="'.$nexturl.'">click here to disconnect and change login user</a>...'."\n";
462 } else {
463 print 'Sorry, your application is offline. Only administrator users (' . str_replace(';', ', ', getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')).') is allowed to connect for the moment.'."\n";
464 $nexturl = DOL_URL_ROOT.'/';
465 print 'Please try later or <a href="'.$nexturl.'">click here to change login user</a>...'."\n";
466 }
467 exit;
468 }
469}
470
471
472// Activate end of page function
473register_shutdown_function('dol_shutdown');
474
475// Load debugbar
476if (isModEnabled('debugbar') && !GETPOST('dol_use_jmobile') && empty($_SESSION['dol_use_jmobile'])) {
477 global $debugbar;
478 include_once DOL_DOCUMENT_ROOT.'/debugbar/class/DebugBar.php';
479 $debugbar = new DolibarrDebugBar();
480 $renderer = $debugbar->getJavascriptRenderer();
481 if (!getDolGlobalString('MAIN_HTML_HEADER')) {
482 $conf->global->MAIN_HTML_HEADER = '';
483 }
484 $conf->global->MAIN_HTML_HEADER .= $renderer->renderHead();
485
486 $debugbar['time']->startMeasure('pageaftermaster', 'Page generation (after environment init)');
487}
488
489// Detection browser
490if (isset($_SERVER["HTTP_USER_AGENT"])) {
491 $tmp = getBrowserInfo($_SERVER["HTTP_USER_AGENT"]);
492 $conf->browser->name = $tmp['browsername'];
493 $conf->browser->os = $tmp['browseros'];
494 $conf->browser->version = $tmp['browserversion'];
495 $conf->browser->ua = $tmp['browserua'];
496 $conf->browser->layout = $tmp['layout']; // 'classic', 'phone', 'tablet'
497 //var_dump($conf->browser);
498
499 if ($conf->browser->layout == 'phone') {
500 $conf->dol_no_mouse_hover = 1;
501 }
502}
503
504// If theme is forced
505if (GETPOST('theme', 'aZ09')) {
506 $conf->theme = GETPOST('theme', 'aZ09');
507 $conf->css = "/theme/".$conf->theme."/style.css.php";
508}
509
510// Set global MAIN_OPTIMIZEFORTEXTBROWSER (must be before login part)
511if (GETPOSTINT('textbrowser') || (!empty($conf->browser->name) && $conf->browser->name == 'lynxlinks')) { // If we must enable text browser
512 $conf->global->MAIN_OPTIMIZEFORTEXTBROWSER = 2;
513}
514
515// Force HTTPS if required ($conf->file->main_force_https is 0/1 or 'https dolibarr root url')
516// $_SERVER["HTTPS"] is 'on' when link is https, otherwise $_SERVER["HTTPS"] is empty or 'off'
517if (!empty($conf->file->main_force_https) && !isHTTPS() && !defined('NOHTTPSREDIRECT')) {
518 $newurl = '';
519 if (is_numeric($conf->file->main_force_https)) {
520 if ($conf->file->main_force_https == '1' && !empty($_SERVER["SCRIPT_URI"])) { // If SCRIPT_URI supported by server
521 if (preg_match('/^http:/i', $_SERVER["SCRIPT_URI"]) && !preg_match('/^https:/i', $_SERVER["SCRIPT_URI"])) { // If link is http
522 $newurl = preg_replace('/^http:/i', 'https:', $_SERVER["SCRIPT_URI"]);
523 }
524 } else {
525 // If HTTPS is not defined in DOL_MAIN_URL_ROOT,
526 // Check HTTPS environment variable (Apache/mod_ssl only)
527 $newurl = preg_replace('/^http:/i', 'https:', DOL_MAIN_URL_ROOT).$_SERVER["REQUEST_URI"];
528 }
529 } else {
530 // Check HTTPS environment variable (Apache/mod_ssl only)
531 $newurl = $conf->file->main_force_https.$_SERVER["REQUEST_URI"];
532 }
533 // Start redirect
534 if ($newurl) {
535 header_remove(); // Clean header already set to be sure to remove any header like "Set-Cookie: DOLSESSID_..." from non HTTPS answers
536 dol_syslog("main.inc: dolibarr_main_force_https is on, we make a redirect to ".$newurl);
537 header("Location: ".$newurl);
538 exit;
539 } else {
540 dol_syslog("main.inc: dolibarr_main_force_https is on but we failed to forge new https url so no redirect is done", LOG_WARNING);
541 }
542}
543
544if (!defined('NOLOGIN') && !defined('NOIPCHECK') && !empty($dolibarr_main_restrict_ip)) {
545 $listofip = explode(',', $dolibarr_main_restrict_ip);
546 $found = false;
547 foreach ($listofip as $ip) {
548 $ip = trim($ip);
549 if ($ip == $_SERVER['REMOTE_ADDR']) {
550 $found = true;
551 break;
552 }
553 }
554 if (!$found) {
555 print 'Access refused by IP protection. Your detected IP is '.$_SERVER['REMOTE_ADDR'];
556 exit;
557 }
558}
559
560// Loading of additional presentation includes
561if (!defined('NOREQUIREHTML')) {
562 require_once DOL_DOCUMENT_ROOT.'/core/class/html.form.class.php'; // Need 660ko memory (800ko in 2.2)
563}
564if (!defined('NOREQUIREAJAX')) {
565 require_once DOL_DOCUMENT_ROOT.'/core/lib/ajax.lib.php'; // Need 22ko memory
566}
567
568// If install or upgrade process not done or not completely finished, we call the install page.
569if (getDolGlobalString('MAIN_NOT_INSTALLED') || getDolGlobalString('MAIN_NOT_UPGRADED')) {
570 dol_syslog("main.inc: A previous install or upgrade was not complete. Redirect to install page.", LOG_WARNING);
571 header("Location: ".DOL_URL_ROOT."/install/index.php");
572 exit;
573}
574// If an upgrade process is required, we call the install page.
575$checkifupgraderequired = false;
576if (getDolGlobalString('MAIN_VERSION_LAST_UPGRADE') && getDolGlobalString('MAIN_VERSION_LAST_UPGRADE') != DOL_VERSION) {
577 $checkifupgraderequired = true;
578}
579if (!getDolGlobalString('MAIN_VERSION_LAST_UPGRADE') && getDolGlobalString('MAIN_VERSION_LAST_INSTALL') && getDolGlobalString('MAIN_VERSION_LAST_INSTALL') != DOL_VERSION) {
580 $checkifupgraderequired = true;
581}
582if ($checkifupgraderequired) {
583 $versiontocompare = getDolGlobalString('MAIN_VERSION_LAST_UPGRADE', getDolGlobalString('MAIN_VERSION_LAST_INSTALL'));
584 require_once DOL_DOCUMENT_ROOT.'/core/lib/admin.lib.php';
585 $dolibarrversionlastupgrade = preg_split('/[.-]/', $versiontocompare);
586 $dolibarrversionprogram = preg_split('/[.-]/', DOL_VERSION);
587 $rescomp = versioncompare($dolibarrversionprogram, $dolibarrversionlastupgrade);
588 if ($rescomp > 0) { // Programs have a version higher than database.
589 if (!getDolGlobalString('MAIN_NO_UPGRADE_REDIRECT_ON_LEVEL_3_CHANGE') || $rescomp < 3) {
590 // We did not add "&& $rescomp < 3" because we want upgrade process for build upgrades
591 dol_syslog("main.inc: database version ".$versiontocompare." is lower than programs version ".DOL_VERSION.". Redirect to install/upgrade page.", LOG_WARNING);
592 if (php_sapi_name() === "cli") {
593 print "main.inc: database version ".$versiontocompare." is lower than programs version ".DOL_VERSION.". Try to run upgrade process.\n";
594 } else {
595 header("Location: ".DOL_URL_ROOT."/install/index.php");
596 }
597 exit;
598 }
599 }
600}
601
602// Creation of a token against CSRF vulnerabilities
603if (!defined('NOTOKENRENEWAL') && !defined('NOSESSION')) {
604 // No token renewal on .css.php, .js.php and .json.php (even if the NOTOKENRENEWAL was not provided)
605 if (!preg_match('/\.(css|js|json)\.php$/', $_SERVER["PHP_SELF"])) {
606 // Rolling token at each call ($_SESSION['token'] contains token of previous page)
607 if (isset($_SESSION['newtoken'])) {
608 $_SESSION['token'] = $_SESSION['newtoken'];
609 }
610
611 if (!isset($_SESSION['newtoken']) || getDolGlobalInt('MAIN_SECURITY_CSRF_TOKEN_RENEWAL_ON_EACH_CALL')) {
612 // Note: Using MAIN_SECURITY_CSRF_TOKEN_RENEWAL_ON_EACH_CALL is not recommended: if a user succeed in entering a data from
613 // a public page with a link that make a token regeneration, it can make use of the backoffice no more possible !
614 // Save in $_SESSION['newtoken'] what will be next token. Into forms, we will add param token = $_SESSION['newtoken']
615 $token = dol_hash(uniqid((string) mt_rand(), false), 'md5'); // Generates a hash of a random number. We don't need a secured hash, just a changing random value.
616 $_SESSION['newtoken'] = $token;
617 dol_syslog("NEW TOKEN generated by : ".$_SERVER['PHP_SELF'], LOG_DEBUG);
618 }
619 }
620}
621
622//dol_syslog("CSRF info: ".defined('NOCSRFCHECK')." - ".$dolibarr_nocsrfcheck." - ".$conf->global->MAIN_SECURITY_CSRF_WITH_TOKEN." - ".$_SERVER['REQUEST_METHOD']." - ".GETPOST('token', 'alpha'));
623
624// Check validity of token, only if option MAIN_SECURITY_CSRF_WITH_TOKEN enabled or if constant CSRFCHECK_WITH_TOKEN is set into page
625if ((!defined('NOCSRFCHECK') && empty($dolibarr_nocsrfcheck) && getDolGlobalInt('MAIN_SECURITY_CSRF_WITH_TOKEN')) || defined('CSRFCHECK_WITH_TOKEN')) {
626 // Array of action code where CSRFCHECK with token will be forced (so token must be provided on url request)
627 $sensitiveget = false;
628 if ((GETPOSTISSET('massaction') || GETPOST('action', 'aZ09')) && getDolGlobalInt('MAIN_SECURITY_CSRF_WITH_TOKEN') >= 3) {
629 // All GET actions (except the listed exceptions that are usually post for pre-actions and not real action) and mass actions are processed as sensitive.
630 if (GETPOSTISSET('massaction') || !in_array(GETPOST('action', 'aZ09'), array('create', 'createsite', 'createcard', 'edit', 'editvalidator', 'file_manager', 'presend', 'presend_addmessage', 'preview', 'specimen'))) { // We exclude some action that are not sensitive so legitimate
631 $sensitiveget = true;
632 }
633 } elseif (getDolGlobalInt('MAIN_SECURITY_CSRF_WITH_TOKEN') >= 2) {
634 // Few GET actions coded with a &token into url are also processed as sensitive.
635 $arrayofactiontoforcetokencheck = array(
636 'activate',
637 'doprev', 'donext', 'dvprev', 'dvnext',
638 'freezone', 'install',
639 'reopen'
640 );
641 if (in_array(GETPOST('action', 'aZ09'), $arrayofactiontoforcetokencheck)) {
642 $sensitiveget = true;
643 }
644 // We also need a valid token for actions matching one of these values
645 if (preg_match('/^(confirm_)?(add|classify|close|confirm|copy|del|disable|enable|remove|set|unset|update|save)/', GETPOST('action', 'aZ09'))) {
646 $sensitiveget = true;
647 }
648 }
649
650 // Check a token is provided for all cases that need a mandatory token
651 // (all POST actions + all sensitive GET actions + all mass actions + all login/actions/logout on pages with CSRFCHECK_WITH_TOKEN set)
652 if (
653 (!empty($_SERVER['REQUEST_METHOD']) && $_SERVER['REQUEST_METHOD'] == 'POST') ||
654 $sensitiveget ||
655 GETPOSTISSET('massaction') ||
656 ((GETPOSTISSET('actionlogin') || GETPOSTISSET('action')) && defined('CSRFCHECK_WITH_TOKEN'))
657 ) {
658 // If token is not provided or empty, error (we are in case it is mandatory)
659 if (!GETPOST('token', 'alpha') || GETPOST('token', 'alpha') == 'notrequired') {
660 top_httphead();
661 if (GETPOSTINT('uploadform')) {
662 dol_syslog("--- Access to ".(empty($_SERVER["REQUEST_METHOD"]) ? '' : $_SERVER["REQUEST_METHOD"].' ').$_SERVER["PHP_SELF"]." refused. File size too large or not provided.");
663 $langs->loadLangs(array("errors", "install"));
664 print $langs->trans("ErrorFileSizeTooLarge").' ';
665 print $langs->trans("ErrorGoBackAndCorrectParameters");
666 } else {
667 http_response_code(403);
668 if (defined('CSRFCHECK_WITH_TOKEN')) {
669 dol_syslog("--- Access to ".(empty($_SERVER["REQUEST_METHOD"]) ? '' : $_SERVER["REQUEST_METHOD"].' ').$_SERVER["PHP_SELF"]." refused by CSRF protection (CSRFCHECK_WITH_TOKEN protection) in main.inc.php. Token not provided.", LOG_WARNING);
670 print "Access to a page that needs a token (constant CSRFCHECK_WITH_TOKEN is defined) is refused by CSRF protection in main.inc.php. Token not provided.\n";
671 } else {
672 dol_syslog("--- Access to ".(empty($_SERVER["REQUEST_METHOD"]) ? '' : $_SERVER["REQUEST_METHOD"].' ').$_SERVER["PHP_SELF"]." refused by CSRF protection (POST method or GET with a sensible value for 'action' parameter) in main.inc.php. Token not provided.", LOG_WARNING);
673 print "Access to this page this way (POST method or GET with a sensible value for 'action' parameter) is refused by CSRF protection in main.inc.php. Token not provided.\n";
674 print "If you access your server behind a proxy using url rewriting and the parameter is provided by caller, you might check that all HTTP header are propagated (or add the line \$dolibarr_nocsrfcheck=1 into your conf.php file or MAIN_SECURITY_CSRF_WITH_TOKEN to 0";
675 if (getDolGlobalString('MAIN_SECURITY_CSRF_WITH_TOKEN')) {
676 print " instead of " . getDolGlobalString('MAIN_SECURITY_CSRF_WITH_TOKEN');
677 }
678 print " into setup).\n";
679 }
680 }
681 die;
682 }
683 }
684
685 $sessiontokenforthisurl = (empty($_SESSION['token']) ? '' : $_SESSION['token']);
686 // TODO Get the sessiontokenforthisurl into an array of session token (one array per base URL so we can use the CSRF per page and we keep ability for several tabs per url in a browser)
687 if (GETPOSTISSET('token') && GETPOST('token') != 'notrequired' && GETPOST('token', 'alpha') != $sessiontokenforthisurl) {
688 dol_syslog("--- Access to ".(empty($_SERVER["REQUEST_METHOD"]) ? '' : $_SERVER["REQUEST_METHOD"].' ').$_SERVER["PHP_SELF"]." refused by CSRF protection (invalid token), so we disable POST and some GET parameters - referrer=".(empty($_SERVER['HTTP_REFERER']) ? '' : $_SERVER['HTTP_REFERER']).", action=".GETPOST('action', 'aZ09').", _GET|POST['token']=".GETPOST('token', 'alpha'), LOG_WARNING);
689 //dol_syslog("_SESSION['token']=".$sessiontokenforthisurl, LOG_DEBUG);
690 // Do not output anything on standard output because this create problems when using the BACK button on browsers. So we just set a message into session.
691 if (!defined('NOTOKENRENEWAL')) {
692 // If the page is not a page that disable the token renewal, we report a warning message to explain token has epired.
693 setEventMessages('SecurityTokenHasExpiredSoActionHasBeenCanceledPleaseRetry', null, 'warnings', '', 1);
694 }
695 $savid = null;
696 if (isset($_POST['id'])) {
697 $savid = ((int) $_POST['id']);
698 }
699 unset($_POST);
700 unset($_GET['confirm']);
701 unset($_GET['action']);
702 unset($_GET['confirmmassaction']);
703 unset($_GET['massaction']);
704 unset($_GET['token']); // TODO Make a redirect if we have a token in url to remove it ?
705 if (isset($savid)) {
706 $_POST['id'] = ((int) $savid);
707 }
708 // So rest of code can know something was wrong here
709 $_GET['errorcode'] = 'InvalidToken';
710 }
711
712 // Note: There is another CSRF protection into the filefunc.inc.php
713}
714
715// Disable modules (this must be after session_start and after conf has been loaded)
716if (GETPOSTISSET('disablemodules')) {
717 $_SESSION["disablemodules"] = GETPOST('disablemodules', 'alpha');
718}
719if (!empty($_SESSION["disablemodules"])) {
720 $modulepartkeys = array('css', 'js', 'tabs', 'triggers', 'login', 'substitutions', 'menus', 'theme', 'sms', 'tpl', 'barcode', 'models', 'societe', 'hooks', 'dir', 'syslog', 'tpllinkable', 'contactelement', 'moduleforexternal', 'websitetemplates');
721
722 $disabled_modules = explode(',', $_SESSION["disablemodules"]);
723 foreach ($disabled_modules as $module) {
724 if ($module) {
725 if (empty($conf->$module)) {
726 $conf->$module = new stdClass(); // To avoid warnings
727 }
728 $conf->$module->enabled = false;
729 foreach ($modulepartkeys as $modulepartkey) {
730 unset($conf->modules_parts[$modulepartkey][$module]);
731 }
732 if ($module == 'fournisseur') { // Special case
733 $conf->supplier_order->enabled = 0;
734 $conf->supplier_invoice->enabled = 0;
735 }
736 }
737 }
738}
739
740// Set current modulepart
741$modulepart = explode("/", $_SERVER["PHP_SELF"]);
742if (is_array($modulepart) && count($modulepart) > 0) {
743 foreach ($conf->modules as $module) {
744 if (in_array($module, $modulepart)) {
745 $modulepart = $module;
746 break;
747 }
748 }
749}
750if (is_array($modulepart)) {
751 $modulepart = '';
752}
753
754
755/*
756 * Phase authentication / login
757 */
758
759$login = '';
760$error = 0;
761if (!defined('NOLOGIN')) {
762 // $authmode lists the different method of identification to be tested in order of preference.
763 // Example: 'http', 'dolibarr', 'ldap', 'http,forceuser', '...'
764
765 if (defined('MAIN_AUTHENTICATION_MODE')) {
766 $dolibarr_main_authentication = constant('MAIN_AUTHENTICATION_MODE');
767 } else {
768 // Authentication mode
769 if (empty($dolibarr_main_authentication)) {
770 $dolibarr_main_authentication = 'dolibarr';
771 }
772 // Authentication mode: forceuser
773 if ($dolibarr_main_authentication == 'forceuser' && empty($dolibarr_auto_user)) {
774 $dolibarr_auto_user = 'auto';
775 }
776 }
777 // Set authmode
778 $authmode = explode(',', $dolibarr_main_authentication);
779
780 // No authentication mode
781 if (!count($authmode)) {
782 $langs->load('main');
783 dol_print_error(null, $langs->trans("ErrorConfigParameterNotDefined", 'dolibarr_main_authentication'));
784 exit;
785 }
786
787 // If login request was already post, we retrieve login from the session
788 // Call module if not realized that his request.
789 // At the end of this phase, the variable $login is defined.
790 $resultFetchUser = '';
791 $test = true;
792 if (!isset($_SESSION["dol_login"])) {
793 // It is not already authenticated and it requests the login / password
794 include_once DOL_DOCUMENT_ROOT.'/core/lib/security2.lib.php';
795
796 $dol_dst_observed = GETPOSTINT("dst_observed", 3);
797 $dol_dst_first = GETPOSTINT("dst_first", 3);
798 $dol_dst_second = GETPOSTINT("dst_second", 3);
799 $dol_screenwidth = GETPOSTINT("screenwidth", 3);
800 $dol_screenheight = GETPOSTINT("screenheight", 3);
801 $dol_hide_topmenu = GETPOSTINT('dol_hide_topmenu', 3);
802 $dol_hide_leftmenu = GETPOSTINT('dol_hide_leftmenu', 3);
803 $dol_optimize_smallscreen = GETPOSTINT('dol_optimize_smallscreen', 3);
804 $dol_no_mouse_hover = GETPOSTINT('dol_no_mouse_hover', 3);
805 $dol_use_jmobile = GETPOSTINT('dol_use_jmobile', 3); // 0=default, 1=to say we use app from a webview app, 2=to say we use app from a webview app and keep ajax
806
807 // If in demo mode, we check we go to home page through the public/demo/index.php page
808 if (!empty($dolibarr_main_demo) && $_SERVER['PHP_SELF'] == DOL_URL_ROOT.'/index.php') { // We ask index page
809 if (empty($_SERVER['HTTP_REFERER']) || !preg_match('/public/', $_SERVER['HTTP_REFERER'])) {
810 dol_syslog("Call index page from another url than demo page (call is done from page ".(empty($_SERVER['HTTP_REFERER']) ? '' : $_SERVER['HTTP_REFER']).")");
811 $url = '';
812 $url .= ($url ? '&' : '').($dol_hide_topmenu ? 'dol_hide_topmenu='.$dol_hide_topmenu : '');
813 $url .= ($url ? '&' : '').($dol_hide_leftmenu ? 'dol_hide_leftmenu='.$dol_hide_leftmenu : '');
814 $url .= ($url ? '&' : '').($dol_optimize_smallscreen ? 'dol_optimize_smallscreen='.$dol_optimize_smallscreen : '');
815 $url .= ($url ? '&' : '').($dol_no_mouse_hover ? 'dol_no_mouse_hover='.$dol_no_mouse_hover : '');
816 $url .= ($url ? '&' : '').($dol_use_jmobile ? 'dol_use_jmobile='.$dol_use_jmobile : '');
817 $url = DOL_URL_ROOT.'/public/demo/index.php'.($url ? '?'.$url : '');
818 header("Location: ".$url);
819 exit;
820 }
821 }
822
823 // Hooks for security access
824 $action = '';
825 $hookmanager->initHooks(array('login'));
826 $parameters = array();
827 $reshook = $hookmanager->executeHooks('beforeLoginAuthentication', $parameters, $user, $action); // Note that $action and $object may have been modified by some hooks
828 if ($reshook < 0) {
829 $test = false;
830 $error++;
831 }
832
833 // Verification security graphic code
834 if ($test && GETPOST("username", "alpha", 2) && getDolGlobalString('MAIN_SECURITY_ENABLECAPTCHA') && !isset($_SESSION['dol_bypass_antispam'])) {
835 $sessionkey = 'dol_antispam_value';
836 $ok = (array_key_exists($sessionkey, $_SESSION) === true && (strtolower($_SESSION[$sessionkey]) === strtolower(GETPOST('code', 'restricthtml'))));
837
838 // Check code
839 if (!$ok) {
840 dol_syslog('Bad value for code, connection refused', LOG_NOTICE);
841 // Load translation files required by page
842 $langs->loadLangs(array('main', 'errors'));
843
844 $_SESSION["dol_loginmesg"] = $langs->transnoentitiesnoconv("ErrorBadValueForCode");
845 $test = false;
846
847 // Call trigger for the "security events" log
848 $user->context['audit'] = 'ErrorBadValueForCode - login='.GETPOST("username", "alpha", 2);
849
850 // Call trigger
851 $result = $user->call_trigger('USER_LOGIN_FAILED', $user);
852 if ($result < 0) {
853 $error++;
854 }
855 // End call triggers
856
857 // Hooks on failed login
858 $action = '';
859 $hookmanager->initHooks(array('login'));
860 $parameters = array('dol_authmode' => $authmode, 'dol_loginmesg' => $_SESSION["dol_loginmesg"]);
861 $reshook = $hookmanager->executeHooks('afterLoginFailed', $parameters, $user, $action); // Note that $action and $object may have been modified by some hooks
862 if ($reshook < 0) {
863 $error++;
864 }
865
866 // Note: exit is done later
867 }
868 }
869
870 $allowedmethodtopostusername = 3;
871 if (defined('MAIN_AUTHENTICATION_POST_METHOD')) {
872 $allowedmethodtopostusername = constant('MAIN_AUTHENTICATION_POST_METHOD'); // Note a value of 2 is not compatible with some authentication methods that put username as GET parameter
873 }
874 // TODO Remove use of $_COOKIE['login_dolibarr'] ? Replace $usertotest = with $usertotest = GETPOST("username", "alpha", $allowedmethodtopostusername);
875 $usertotest = (!empty($_COOKIE['login_dolibarr']) ? preg_replace('/[^a-zA-Z0-9_@\-\.]/', '', $_COOKIE['login_dolibarr']) : GETPOST("username", "alpha", $allowedmethodtopostusername));
876 $passwordtotest = GETPOST('password', 'password', $allowedmethodtopostusername);
877 $entitytotest = (GETPOSTINT('entity') ? GETPOSTINT('entity') : (!empty($conf->entity) ? $conf->entity : 1));
878
879 // Define if we received the correct data to go into the test of the login with the checkLoginPassEntity().
880 $goontestloop = false;
881 if (isset($_SERVER["REMOTE_USER"]) && in_array('http', $authmode)) { // For http basic login test
882 $goontestloop = true;
883 }
884 if ($dolibarr_main_authentication == 'forceuser' && !empty($dolibarr_auto_user)) { // For automatic login with a forced user
885 $goontestloop = true;
886 }
887 if (GETPOST("username", "alpha", $allowedmethodtopostusername)) { // For posting the login form
888 $goontestloop = true;
889 }
890 if (GETPOST('openid_mode', 'alpha')) { // For openid_connect ?
891 $goontestloop = true;
892 }
893 if (GETPOST('beforeoauthloginredirect') || GETPOST('afteroauthloginreturn')) { // For oauth login
894 $goontestloop = true;
895 }
896 if (!empty($_COOKIE['login_dolibarr'])) { // TODO For ? Remove this ?
897 $goontestloop = true;
898 }
899
900 if (!is_object($langs)) { // This can occurs when calling page with NOREQUIRETRAN defined, however we need langs for error messages.
901 include_once DOL_DOCUMENT_ROOT.'/core/class/translate.class.php';
902 $langs = new Translate("", $conf);
903 $langcode = (GETPOST('lang', 'aZ09', 1) ? GETPOST('lang', 'aZ09', 1) : getDolGlobalString('MAIN_LANG_DEFAULT', 'auto'));
904 if (defined('MAIN_LANG_DEFAULT')) {
905 $langcode = constant('MAIN_LANG_DEFAULT');
906 }
907 $langs->setDefaultLang($langcode);
908 }
909
910 // Validation of login/pass/entity
911 // If ok, the variable login will be returned
912 // If error, we will put error message in session under the name dol_loginmesg
913 if ($test && $goontestloop && (GETPOST('actionlogin', 'aZ09') == 'login' || $dolibarr_main_authentication != 'dolibarr')) {
914 // Loop on each test mode defined into $authmode
915 // $authmode is an array for example: array('0'=>'dolibarr', '1'=>'googleoauth');
916 $oauthmodetotestarray = array('google');
917 foreach ($oauthmodetotestarray as $oauthmodetotest) {
918 if (in_array($oauthmodetotest.'oauth', $authmode)) { // This is an authmode that is currently qualified. Do we have to remove it ?
919 // If we click on the link to use OAuth authentication or if we goes after callback return, we do nothing
920 if (GETPOST('beforeoauthloginredirect') == $oauthmodetotest || GETPOST('afteroauthloginreturn')) {
921 // TODO Use: if (GETPOST('beforeoauthloginredirect') == $oauthmodetotest || GETPOST('afteroauthloginreturn') == $oauthmodetotest) {
922 continue;
923 }
924 dol_syslog("User did not click on link for OAuth or is not on the OAuth return, so we disable check using ".$oauthmodetotest);
925 foreach ($authmode as $tmpkey => $tmpval) {
926 if ($tmpval == $oauthmodetotest.'oauth') {
927 unset($authmode[$tmpkey]);
928 break;
929 }
930 }
931 }
932 }
933
934 // Check login for all qualified modes in array $authmode.
935 $login = checkLoginPassEntity($usertotest, $passwordtotest, $entitytotest, $authmode);
936 if ($login === '--bad-login-validity--') {
937 $login = '';
938 }
939
940 $dol_authmode = '';
941
942 if ($login) {
943 $dol_authmode = $conf->authmode; // This properties is defined only when logged, to say what mode was successfully used
944 $dol_tz = empty($_POST["tz"]) ? (empty($_SESSION["tz"]) ? '' : $_SESSION["tz"]) : $_POST["tz"];
945 $dol_tz_string = empty($_POST["tz_string"]) ? (empty($_SESSION["tz_string"]) ? '' : $_SESSION["tz_string"]) : $_POST["tz_string"];
946 $dol_tz_string = preg_replace('/\s*\‍(.+\‍)$/', '', $dol_tz_string);
947 $dol_tz_string = preg_replace('/,/', '/', $dol_tz_string);
948 $dol_tz_string = preg_replace('/\s/', '_', $dol_tz_string);
949 $dol_dst = 0;
950 // Keep $_POST here. Do not use GETPOSTISSET
951 $dol_dst_first = empty($_POST["dst_first"]) ? (empty($_SESSION["dst_first"]) ? '' : $_SESSION["dst_first"]) : $_POST["dst_first"];
952 $dol_dst_second = empty($_POST["dst_second"]) ? (empty($_SESSION["dst_second"]) ? '' : $_SESSION["dst_second"]) : $_POST["dst_second"];
953 if ($dol_dst_first && $dol_dst_second) {
954 include_once DOL_DOCUMENT_ROOT.'/core/lib/date.lib.php';
955 $datenow = dol_now();
956 $datefirst = dol_stringtotime($dol_dst_first);
957 $datesecond = dol_stringtotime($dol_dst_second);
958 if ($datenow >= $datefirst && $datenow < $datesecond) {
959 $dol_dst = 1;
960 }
961 }
962 $dol_screenheight = empty($_POST["screenheight"]) ? (empty($_SESSION["dol_screenheight"]) ? '' : $_SESSION["dol_screenheight"]) : $_POST["screenheight"];
963 $dol_screenwidth = empty($_POST["screenwidth"]) ? (empty($_SESSION["dol_screenwidth"]) ? '' : $_SESSION["dol_screenwidth"]) : $_POST["screenwidth"];
964 //print $datefirst.'-'.$datesecond.'-'.$datenow.'-'.$dol_tz.'-'.$dol_tzstring.'-'.$dol_dst.'-'.sdol_screenheight.'-'.sdol_screenwidth; exit;
965 }
966
967 if (!$login) {
968 dol_syslog('Bad password, connection refused (see a previous notice message for more info)', LOG_NOTICE);
969 // Load translation files required by page
970 $langs->loadLangs(array('main', 'errors'));
971
972 // Bad password. No authmode has found a good password.
973 // We set a generic message if not defined inside function checkLoginPassEntity or subfunctions
974 if (empty($_SESSION["dol_loginmesg"])) {
975 $_SESSION["dol_loginmesg"] = $langs->transnoentitiesnoconv("ErrorBadLoginPassword");
976 }
977
978 // Call trigger for the "security events" log
979 $user->context['audit'] = $langs->trans("ErrorBadLoginPassword").' - login='.GETPOST("username", "alpha", 2);
980
981 // Call trigger
982 $result = $user->call_trigger('USER_LOGIN_FAILED', $user);
983 if ($result < 0) {
984 $error++;
985 }
986 // End call triggers
987
988 // Hooks on failed login
989 $action = '';
990 $hookmanager->initHooks(array('login'));
991 $parameters = array('dol_authmode' => $dol_authmode, 'dol_loginmesg' => $_SESSION["dol_loginmesg"]);
992 $reshook = $hookmanager->executeHooks('afterLoginFailed', $parameters, $user, $action); // Note that $action and $object may have been modified by some hooks
993 if ($reshook < 0) {
994 $error++;
995 }
996
997 // Note: exit is done in next chapter
998 }
999 }
1000
1001 // End test login / passwords
1002 if (!$login || (in_array('ldap', $authmode) && empty($passwordtotest))) { // With LDAP we refused empty password because some LDAP are "opened" for anonymous access so connection is a success.
1003 // No data to test login, so we show the login page.
1004 dol_syslog("--- Access to ".(empty($_SERVER["REQUEST_METHOD"]) ? '' : $_SERVER["REQUEST_METHOD"].' ').$_SERVER["PHP_SELF"]." - action=".GETPOST('action', 'aZ09')." - actionlogin=".GETPOST('actionlogin', 'aZ09')." - showing the login form and exit", LOG_NOTICE);
1005 if (defined('NOREDIRECTBYMAINTOLOGIN')) {
1006 // When used with NOREDIRECTBYMAINTOLOGIN set, the http header must already be set when including the main.
1007 // See example with selectsearchbox.php. This case is reserved for the selectesearchbox.php so we can
1008 // report a message to ask to login when search ajax component is used after a timeout.
1009 //top_httphead();
1010 return 'ERROR_NOT_LOGGED';
1011 } else {
1012 if (!empty($_SERVER["HTTP_USER_AGENT"]) && $_SERVER["HTTP_USER_AGENT"] == 'securitytest') {
1013 http_response_code(401); // It makes easier to understand if session was broken during security tests
1014 }
1015 dol_loginfunction($langs, $conf, (!empty($mysoc) ? $mysoc : '')); // This include http headers
1016 }
1017 exit;
1018 }
1019
1020 $resultFetchUser = $user->fetch('', $login, '', 1, ($entitytotest > 0 ? $entitytotest : -1)); // value for $login was retrieved previously when checking password.
1021 if ($resultFetchUser <= 0 || $user->isNotIntoValidityDateRange()) {
1022 dol_syslog('User not found or not valid, connection refused');
1023 session_destroy();
1024 session_set_cookie_params(0, '/', null, (empty($dolibarr_main_force_https) ? false : true), true); // Add tag secure and httponly on session cookie
1025 session_name($sessionname);
1027
1028 if ($resultFetchUser == 0) {
1029 // Load translation files required by page
1030 $langs->loadLangs(array('main', 'errors'));
1031
1032 $_SESSION["dol_loginmesg"] = $langs->transnoentitiesnoconv("ErrorCantLoadUserFromDolibarrDatabase", $login);
1033
1034 $user->context['audit'] = 'ErrorCantLoadUserFromDolibarrDatabase - login='.$login;
1035 } elseif ($resultFetchUser < 0) {
1036 $_SESSION["dol_loginmesg"] = $user->error;
1037
1038 $user->context['audit'] = $user->error;
1039 } else {
1040 // Load translation files required by the page
1041 $langs->loadLangs(array('main', 'errors'));
1042
1043 $_SESSION["dol_loginmesg"] = $langs->transnoentitiesnoconv("ErrorLoginDateValidity");
1044
1045 $user->context['audit'] = $langs->trans("ErrorLoginDateValidity").' - login='.$login;
1046 }
1047
1048 // Call trigger
1049 $result = $user->call_trigger('USER_LOGIN_FAILED', $user);
1050 if ($result < 0) {
1051 $error++;
1052 }
1053 // End call triggers
1054
1055
1056 // Hooks on failed login
1057 $action = '';
1058 $hookmanager->initHooks(array('login'));
1059 $parameters = array('dol_authmode' => $dol_authmode, 'dol_loginmesg' => $_SESSION["dol_loginmesg"]);
1060 $reshook = $hookmanager->executeHooks('afterLoginFailed', $parameters, $user, $action); // Note that $action and $object may have been modified by some hooks
1061 if ($reshook < 0) {
1062 $error++;
1063 }
1064
1065 $paramsurl = array();
1066 if (GETPOSTINT('textbrowser')) {
1067 $paramsurl[] = 'textbrowser='.GETPOSTINT('textbrowser');
1068 }
1069 if (GETPOSTINT('nojs')) {
1070 $paramsurl[] = 'nojs='.GETPOSTINT('nojs');
1071 }
1072 if (GETPOST('lang', 'aZ09')) {
1073 $paramsurl[] = 'lang='.GETPOST('lang', 'aZ09');
1074 }
1075 header('Location: '.DOL_URL_ROOT.'/index.php'.(count($paramsurl) ? '?'.implode('&', $paramsurl) : ''));
1076 exit;
1077 } else {
1078 // User is loaded, we may need to change language for him according to its choice
1079 if (!empty($user->conf->MAIN_LANG_DEFAULT)) {
1080 $langs->setDefaultLang($user->conf->MAIN_LANG_DEFAULT);
1081 }
1082 }
1083 } else {
1084 // We are already into an authenticated session
1085 $login = $_SESSION["dol_login"];
1086 $entity = isset($_SESSION["dol_entity"]) ? $_SESSION["dol_entity"] : 0;
1087 dol_syslog("- This is an already logged session. _SESSION['dol_login']=".$login." _SESSION['dol_entity']=".$entity, LOG_DEBUG);
1088
1089 $resultFetchUser = $user->fetch('', $login, '', 1, ($entity > 0 ? $entity : -1));
1090
1091 //var_dump(dol_print_date($user->flagdelsessionsbefore, 'dayhour', 'gmt')." ".dol_print_date($_SESSION["dol_logindate"], 'dayhour', 'gmt'));
1092
1093 if ($resultFetchUser <= 0
1094 || ($user->flagdelsessionsbefore && !empty($_SESSION["dol_logindate"]) && $user->flagdelsessionsbefore > $_SESSION["dol_logindate"])
1095 || ($user->status != $user::STATUS_ENABLED)
1096 || ($user->isNotIntoValidityDateRange())) {
1097 if ($resultFetchUser <= 0) {
1098 // Account has been removed after login
1099 dol_syslog("Can't load user even if session logged. _SESSION['dol_login']=".$login, LOG_WARNING);
1100 } elseif ($user->flagdelsessionsbefore && !empty($_SESSION["dol_logindate"]) && $user->flagdelsessionsbefore > $_SESSION["dol_logindate"]) {
1101 // Session is no more valid
1102 dol_syslog("The user has a date for session invalidation = ".$user->flagdelsessionsbefore." and a session date = ".$_SESSION["dol_logindate"].". We must invalidate its sessions.");
1103 } elseif ($user->status != $user::STATUS_ENABLED) {
1104 // User is not enabled
1105 dol_syslog("The user login is disabled");
1106 } else {
1107 // User validity dates are no more valid
1108 dol_syslog("The user login has a validity between [".$user->datestartvalidity." and ".$user->dateendvalidity."], current date is ".dol_now());
1109 }
1110 session_destroy();
1111 session_set_cookie_params(0, '/', null, (empty($dolibarr_main_force_https) ? false : true), true); // Add tag secure and httponly on session cookie
1112 session_name($sessionname);
1114
1115 if ($resultFetchUser == 0) {
1116 $langs->loadLangs(array('main', 'errors'));
1117
1118 $_SESSION["dol_loginmesg"] = $langs->transnoentitiesnoconv("ErrorCantLoadUserFromDolibarrDatabase", $login);
1119
1120 $user->context['audit'] = 'ErrorCantLoadUserFromDolibarrDatabase - login='.$login;
1121 } elseif ($resultFetchUser < 0) {
1122 $_SESSION["dol_loginmesg"] = $user->error;
1123
1124 $user->context['audit'] = $user->error;
1125 } else {
1126 $langs->loadLangs(array('main', 'errors'));
1127
1128 $_SESSION["dol_loginmesg"] = $langs->transnoentitiesnoconv("ErrorSessionInvalidatedAfterPasswordChange");
1129
1130 $user->context['audit'] = 'ErrorUserSessionWasInvalidated - login='.$login;
1131 }
1132
1133 // Call trigger
1134 $result = $user->call_trigger('USER_LOGIN_FAILED', $user);
1135 if ($result < 0) {
1136 $error++;
1137 }
1138 // End call triggers
1139
1140 // Hooks on failed login
1141 $action = '';
1142 $hookmanager->initHooks(array('login'));
1143 $parameters = array('dol_authmode' => (isset($dol_authmode) ? $dol_authmode : ''), 'dol_loginmesg' => $_SESSION["dol_loginmesg"]);
1144 $reshook = $hookmanager->executeHooks('afterLoginFailed', $parameters, $user, $action); // Note that $action and $object may have been modified by some hooks
1145 if ($reshook < 0) {
1146 $error++;
1147 }
1148
1149 $paramsurl = array();
1150 if (GETPOSTINT('textbrowser')) {
1151 $paramsurl[] = 'textbrowser='.GETPOSTINT('textbrowser');
1152 }
1153 if (GETPOSTINT('nojs')) {
1154 $paramsurl[] = 'nojs='.GETPOSTINT('nojs');
1155 }
1156 if (GETPOST('lang', 'aZ09')) {
1157 $paramsurl[] = 'lang='.GETPOST('lang', 'aZ09');
1158 }
1159
1160 header('Location: '.DOL_URL_ROOT.'/index.php'.(count($paramsurl) ? '?'.implode('&', $paramsurl) : ''));
1161 exit;
1162 } else {
1163 // Initialize a technical object to manage hooks of page. Note that conf->hooks_modules contains an array of hook context
1164 $hookmanager->initHooks(array('main'));
1165
1166 // Code for search criteria persistence.
1167 if (!empty($_GET['save_lastsearch_values']) && !empty($_SERVER["HTTP_REFERER"])) { // We must use $_GET here
1168 $relativepathstring = preg_replace('/\?.*$/', '', $_SERVER["HTTP_REFERER"]);
1169 $relativepathstring = preg_replace('/^https?:\/\/[^\/]*/', '', $relativepathstring); // Get full path except host server
1170 // Clean $relativepathstring
1171 if (constant('DOL_URL_ROOT')) {
1172 $relativepathstring = preg_replace('/^'.preg_quote(constant('DOL_URL_ROOT'), '/').'/', '', $relativepathstring);
1173 }
1174 $relativepathstring = preg_replace('/^\//', '', $relativepathstring);
1175 $relativepathstring = preg_replace('/^custom\//', '', $relativepathstring);
1176 //var_dump($relativepathstring);
1177
1178 // We click on a link that leave a page we have to save search criteria, contextpage, limit and page and mode. We save them from tmp to no tmp
1179 if (!empty($_SESSION['lastsearch_values_tmp_'.$relativepathstring])) {
1180 $_SESSION['lastsearch_values_'.$relativepathstring] = $_SESSION['lastsearch_values_tmp_'.$relativepathstring];
1181 unset($_SESSION['lastsearch_values_tmp_'.$relativepathstring]);
1182 }
1183 if (!empty($_SESSION['lastsearch_contextpage_tmp_'.$relativepathstring])) {
1184 $_SESSION['lastsearch_contextpage_'.$relativepathstring] = $_SESSION['lastsearch_contextpage_tmp_'.$relativepathstring];
1185 unset($_SESSION['lastsearch_contextpage_tmp_'.$relativepathstring]);
1186 }
1187 if (!empty($_SESSION['lastsearch_limit_tmp_'.$relativepathstring]) && $_SESSION['lastsearch_limit_tmp_'.$relativepathstring] != $conf->liste_limit) {
1188 $_SESSION['lastsearch_limit_'.$relativepathstring] = $_SESSION['lastsearch_limit_tmp_'.$relativepathstring];
1189 unset($_SESSION['lastsearch_limit_tmp_'.$relativepathstring]);
1190 }
1191 if (!empty($_SESSION['lastsearch_page_tmp_'.$relativepathstring]) && $_SESSION['lastsearch_page_tmp_'.$relativepathstring] > 0) {
1192 $_SESSION['lastsearch_page_'.$relativepathstring] = $_SESSION['lastsearch_page_tmp_'.$relativepathstring];
1193 unset($_SESSION['lastsearch_page_tmp_'.$relativepathstring]);
1194 }
1195 if (!empty($_SESSION['lastsearch_mode_tmp_'.$relativepathstring])) {
1196 $_SESSION['lastsearch_mode_'.$relativepathstring] = $_SESSION['lastsearch_mode_tmp_'.$relativepathstring];
1197 unset($_SESSION['lastsearch_mode_tmp_'.$relativepathstring]);
1198 }
1199 }
1200 if (!empty($_GET['save_pageforbacktolist']) && !empty($_SERVER["HTTP_REFERER"])) { // We must use $_GET here
1201 if (empty($_SESSION['pageforbacktolist'])) {
1202 $pageforbacktolistarray = array();
1203 } else {
1204 $pageforbacktolistarray = $_SESSION['pageforbacktolist'];
1205 }
1206 $tmparray = explode(':', $_GET['save_pageforbacktolist'], 2);
1207 if (!empty($tmparray[0]) && !empty($tmparray[1])) {
1208 $pageforbacktolistarray[$tmparray[0]] = $tmparray[1];
1209 $_SESSION['pageforbacktolist'] = $pageforbacktolistarray;
1210 }
1211 }
1212
1213 $action = '';
1214 $parameters = array();
1215 $reshook = $hookmanager->executeHooks('updateSession', $parameters, $user, $action);
1216 if ($reshook < 0) {
1217 setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
1218 }
1219 }
1220 }
1221
1222 // Is it a new session that has started ?
1223 // If we are here, this means authentication was successful.
1224 if (!isset($_SESSION["dol_login"])) {
1225 // New session for this login has started.
1226 $error = 0;
1227
1228 // Store value into session (values always stored)
1229 $_SESSION["dol_login"] = $user->login;
1230 $_SESSION["dol_logindate"] = dol_now('gmt');
1231 $_SESSION["dol_authmode"] = isset($dol_authmode) ? $dol_authmode : '';
1232 $_SESSION["dol_tz"] = isset($dol_tz) ? $dol_tz : '';
1233 $_SESSION["dol_tz_string"] = isset($dol_tz_string) ? $dol_tz_string : '';
1234 $_SESSION["dol_dst"] = isset($dol_dst) ? $dol_dst : '';
1235 $_SESSION["dol_dst_observed"] = isset($dol_dst_observed) ? $dol_dst_observed : '';
1236 $_SESSION["dol_dst_first"] = isset($dol_dst_first) ? $dol_dst_first : '';
1237 $_SESSION["dol_dst_second"] = isset($dol_dst_second) ? $dol_dst_second : '';
1238 $_SESSION["dol_screenwidth"] = isset($dol_screenwidth) ? $dol_screenwidth : '';
1239 $_SESSION["dol_screenheight"] = isset($dol_screenheight) ? $dol_screenheight : '';
1240 $_SESSION["dol_company"] = getDolGlobalString("MAIN_INFO_SOCIETE_NOM");
1241 $_SESSION["dol_entity"] = $conf->entity;
1242 // Store value into session (values stored only if defined)
1243 if (!empty($dol_hide_topmenu)) {
1244 $_SESSION['dol_hide_topmenu'] = $dol_hide_topmenu;
1245 }
1246 if (!empty($dol_hide_leftmenu)) {
1247 $_SESSION['dol_hide_leftmenu'] = $dol_hide_leftmenu;
1248 }
1249 if (!empty($dol_optimize_smallscreen)) {
1250 $_SESSION['dol_optimize_smallscreen'] = $dol_optimize_smallscreen;
1251 }
1252 if (!empty($dol_no_mouse_hover)) {
1253 $_SESSION['dol_no_mouse_hover'] = $dol_no_mouse_hover;
1254 }
1255 if (!empty($dol_use_jmobile)) {
1256 $_SESSION['dol_use_jmobile'] = $dol_use_jmobile;
1257 }
1258
1259 dol_syslog("This is a new started user session. _SESSION['dol_login']=".$_SESSION["dol_login"]." Session id=".session_id());
1260
1261 $db->begin();
1262
1263 $user->update_last_login_date();
1264
1265 $loginfo = 'TZ='.$_SESSION["dol_tz"].';TZString='.$_SESSION["dol_tz_string"].';Screen='.$_SESSION["dol_screenwidth"].'x'.$_SESSION["dol_screenheight"];
1266 $loginfo .= ' - authmode='.$dol_authmode.' - entity='.$conf->entity;
1267
1268 // Call triggers for the "security events" log
1269 $user->context['audit'] = $loginfo;
1270 $user->context['authentication_method'] = $dol_authmode;
1271
1272 // Call trigger
1273 $result = $user->call_trigger('USER_LOGIN', $user);
1274 if ($result < 0) {
1275 $error++;
1276 }
1277 // End call triggers
1278
1279 // Hooks on successful login
1280 $action = '';
1281 $hookmanager->initHooks(array('login'));
1282 $parameters = array('dol_authmode' => $dol_authmode, 'dol_loginfo' => $loginfo);
1283 $reshook = $hookmanager->executeHooks('afterLogin', $parameters, $user, $action); // Note that $action and $object may have been modified by some hooks
1284 if ($reshook < 0) {
1285 $error++;
1286 }
1287
1288 if ($error) {
1289 $db->rollback();
1290 session_destroy();
1291 dol_print_error($db, 'Error in some triggers USER_LOGIN or in some hooks afterLogin');
1292 exit;
1293 } else {
1294 $db->commit();
1295 }
1296
1297 // Change landing page if defined.
1298 $landingpage = (empty($user->conf->MAIN_LANDING_PAGE) ? (!getDolGlobalString('MAIN_LANDING_PAGE') ? '' : $conf->global->MAIN_LANDING_PAGE) : $user->conf->MAIN_LANDING_PAGE);
1299 if (!empty($landingpage)) { // Example: /index.php
1300 $newpath = dol_buildpath($landingpage, 1);
1301 if ($_SERVER["PHP_SELF"] != $newpath) { // not already on landing page (avoid infinite loop)
1302 header('Location: '.$newpath);
1303 exit;
1304 }
1305 }
1306 }
1307
1308
1309 // If user admin, we force the rights-based modules
1310 if ($user->admin) {
1311 $user->rights->user->user->lire = 1;
1312 $user->rights->user->user->creer = 1;
1313 $user->rights->user->user->password = 1;
1314 $user->rights->user->user->supprimer = 1;
1315 $user->rights->user->self->creer = 1;
1316 $user->rights->user->self->password = 1;
1317
1318 //Required if advanced permissions are used with MAIN_USE_ADVANCED_PERMS
1319 if (getDolGlobalString('MAIN_USE_ADVANCED_PERMS')) {
1320 if (!$user->hasRight('user', 'user_advance')) {
1321 $user->rights->user->user_advance = new stdClass(); // To avoid warnings
1322 }
1323 if (!$user->hasRight('user', 'self_advance')) {
1324 $user->rights->user->self_advance = new stdClass(); // To avoid warnings
1325 }
1326 if (!$user->hasRight('user', 'group_advance')) {
1327 $user->rights->user->group_advance = new stdClass(); // To avoid warnings
1328 }
1329
1330 $user->rights->user->user_advance->readperms = 1;
1331 $user->rights->user->user_advance->write = 1;
1332 $user->rights->user->self_advance->readperms = 1;
1333 $user->rights->user->self_advance->writeperms = 1;
1334 $user->rights->user->group_advance->read = 1;
1335 $user->rights->user->group_advance->readperms = 1;
1336 $user->rights->user->group_advance->write = 1;
1337 $user->rights->user->group_advance->delete = 1;
1338 }
1339 }
1340
1341 /*
1342 * Overwrite some configs globals (try to avoid this and have code to use instead $user->conf->xxx)
1343 */
1344
1345 // Set liste_limit from user setup
1346 if (isset($user->conf->MAIN_SIZE_LISTE_LIMIT)) { // If a user setup exists
1347 $conf->liste_limit = getDolUserInt('MAIN_SIZE_LISTE_LIMIT'); // Can be 0
1348 }
1349 if ((int) $conf->liste_limit <= 0) {
1350 // Mode automatic.
1351 $conf->liste_limit = 15;
1352 if (!empty($_SESSION['dol_screenheight']) && $_SESSION['dol_screenheight'] < 910) {
1353 $conf->liste_limit = 10;
1354 } elseif (!empty($_SESSION['dol_screenheight']) && $_SESSION['dol_screenheight'] > 1130) {
1355 $conf->liste_limit = 20;
1356 }
1357 }
1358
1359 // Replace conf->css by personalized value if theme not forced
1360 if (!getDolGlobalString('MAIN_FORCETHEME') && getDolUserString('MAIN_THEME')) {
1361 $conf->theme = getDolUserString('MAIN_THEME');
1362 $conf->css = "/theme/".$conf->theme."/style.css.php";
1363 }
1364} else {
1365 // We may have NOLOGIN set, but NOREQUIREUSER not
1366 if (!empty($user) && method_exists($user, 'loadDefaultValues') && !defined('NODEFAULTVALUES')) {
1367 $user->loadDefaultValues(); // Load default values for everybody (works even if $user->id = 0
1368 }
1369}
1370
1371
1372// Case forcing style from url
1373if (GETPOST('theme', 'aZ09')) {
1374 $conf->theme = GETPOST('theme', 'aZ09', 1);
1375 $conf->css = "/theme/".$conf->theme."/style.css.php";
1376}
1377
1378// Set javascript option
1379if (GETPOSTINT('nojs')) { // If javascript was not disabled on URL
1380 $conf->use_javascript_ajax = 0;
1381} else {
1382 if (getDolUserString('MAIN_DISABLE_JAVASCRIPT')) {
1383 $conf->use_javascript_ajax = !getDolUserString('MAIN_DISABLE_JAVASCRIPT');
1384 }
1385}
1386
1387// Set MAIN_OPTIMIZEFORTEXTBROWSER for user (must be after login part)
1388if (!getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER') && getDolUserString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
1389 $conf->global->MAIN_OPTIMIZEFORTEXTBROWSER = getDolUserString('MAIN_OPTIMIZEFORTEXTBROWSER');
1390 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER') == 1) {
1391 $conf->global->THEME_TOPMENU_DISABLE_IMAGE = 1;
1392 }
1393}
1394//var_dump($conf->global->THEME_TOPMENU_DISABLE_IMAGE);
1395//var_dump($user->conf->THEME_TOPMENU_DISABLE_IMAGE);
1396
1397// set MAIN_OPTIMIZEFORCOLORBLIND for user
1398$conf->global->MAIN_OPTIMIZEFORCOLORBLIND = getDolUserString('MAIN_OPTIMIZEFORCOLORBLIND');
1399
1400// Set terminal output option according to conf->browser.
1401if (GETPOSTINT('dol_hide_leftmenu') || !empty($_SESSION['dol_hide_leftmenu'])) {
1402 $conf->dol_hide_leftmenu = 1;
1403}
1404if (GETPOSTINT('dol_hide_topmenu') || !empty($_SESSION['dol_hide_topmenu'])) {
1405 $conf->dol_hide_topmenu = 1;
1406}
1407if (GETPOSTINT('dol_optimize_smallscreen') || !empty($_SESSION['dol_optimize_smallscreen'])) {
1408 $conf->dol_optimize_smallscreen = 1;
1409}
1410if (GETPOSTINT('dol_no_mouse_hover') || !empty($_SESSION['dol_no_mouse_hover'])) {
1411 $conf->dol_no_mouse_hover = 1;
1412}
1413if (GETPOSTINT('dol_use_jmobile') || !empty($_SESSION['dol_use_jmobile'])) {
1414 $conf->dol_use_jmobile = 1;
1415}
1416// If not on Desktop
1417if (!empty($conf->browser->layout) && $conf->browser->layout != 'classic') {
1418 $conf->dol_no_mouse_hover = 1;
1419}
1420
1421// If on smartphone or optimized for small screen
1422if ((!empty($conf->browser->layout) && $conf->browser->layout == 'phone')
1423 || (!empty($_SESSION['dol_screenwidth']) && $_SESSION['dol_screenwidth'] < 400)
1424 || (!empty($_SESSION['dol_screenheight']) && $_SESSION['dol_screenheight'] < 400
1425 || getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER'))
1426) {
1427 $conf->dol_optimize_smallscreen = 1;
1428
1429 if (getDolGlobalInt('PRODUIT_DESC_IN_FORM') == 1) {
1430 $conf->global->PRODUIT_DESC_IN_FORM_ACCORDING_TO_DEVICE = 0;
1431 }
1432}
1433// Replace themes bugged with jmobile with eldy
1434if (!empty($conf->dol_use_jmobile) && in_array($conf->theme, array('bureau2crea', 'cameleo', 'amarok'))) {
1435 $conf->theme = 'eldy';
1436 $conf->css = "/theme/".$conf->theme."/style.css.php";
1437}
1438
1439if (!defined('NOREQUIRETRAN')) {
1440 if (!GETPOST('lang', 'aZ09')) { // If language was not forced on URL
1441 // If user has chosen its own language
1442 if (!empty($user->conf->MAIN_LANG_DEFAULT)) {
1443 // If different than current language
1444 //print ">>>".$langs->getDefaultLang()."-".$user->conf->MAIN_LANG_DEFAULT;
1445 if ($langs->getDefaultLang() != $user->conf->MAIN_LANG_DEFAULT) {
1446 $langs->setDefaultLang($user->conf->MAIN_LANG_DEFAULT);
1447 }
1448 }
1449 }
1450}
1451
1452if (!defined('NOLOGIN')) {
1453 // If the login is not recovered, it is identified with an account that does not exist.
1454 // Hacking attempt?
1455 if (!$user->login) {
1457 }
1458
1459 // Check if user is active
1460 if ($user->statut < 1) {
1461 // If not active, we refuse the user
1462 $langs->loadLangs(array("errors", "other"));
1463 dol_syslog("Authentication KO as login is disabled", LOG_NOTICE);
1464 accessforbidden("ErrorLoginDisabled");
1465 }
1466
1467 // Load permissions
1468 $user->getrights();
1469}
1470
1471dol_syslog("--- Access to ".(empty($_SERVER["REQUEST_METHOD"]) ? '' : $_SERVER["REQUEST_METHOD"].' ').$_SERVER["PHP_SELF"].' - action='.GETPOST('action', 'aZ09').', massaction='.GETPOST('massaction', 'aZ09').(defined('NOTOKENRENEWAL') ? ' NOTOKENRENEWAL='.constant('NOTOKENRENEWAL') : ''), LOG_NOTICE);
1472//Another call for easy debug
1473//dol_syslog("Access to ".$_SERVER["PHP_SELF"].' '.$_SERVER["HTTP_REFERER"].' GET='.join(',',array_keys($_GET)).'->'.join(',',$_GET).' POST:'.join(',',array_keys($_POST)).'->'.join(',',$_POST));
1474
1475// Load main languages files
1476if (!defined('NOREQUIRETRAN')) {
1477 // Load translation files required by page
1478 $langs->loadLangs(array('main', 'dict'));
1479}
1480
1481// Define some constants used for style of arrays
1482$bc = array(0 => 'class="impair"', 1 => 'class="pair"');
1483$bcdd = array(0 => 'class="drag drop oddeven"', 1 => 'class="drag drop oddeven"');
1484$bcnd = array(0 => 'class="nodrag nodrop nohover"', 1 => 'class="nodrag nodrop nohoverpair"'); // Used for tr to add new lines
1485$bctag = array(0 => 'class="impair tagtr"', 1 => 'class="pair tagtr"');
1486
1487// Define messages variables
1488$mesg = '';
1489$warning = '';
1490$error = 0;
1491// deprecated, see setEventMessages() and dol_htmloutput_events()
1492$mesgs = array();
1493$warnings = array();
1494$errors = array();
1495
1496// Constants used to defined number of lines in textarea
1497if (empty($conf->browser->firefox)) {
1498 define('ROWS_1', 1);
1499 define('ROWS_2', 2);
1500 define('ROWS_3', 3);
1501 define('ROWS_4', 4);
1502 define('ROWS_5', 5);
1503 define('ROWS_6', 6);
1504 define('ROWS_7', 7);
1505 define('ROWS_8', 8);
1506 define('ROWS_9', 9);
1507} else {
1508 define('ROWS_1', 0);
1509 define('ROWS_2', 1);
1510 define('ROWS_3', 2);
1511 define('ROWS_4', 3);
1512 define('ROWS_5', 4);
1513 define('ROWS_6', 5);
1514 define('ROWS_7', 6);
1515 define('ROWS_8', 7);
1516 define('ROWS_9', 8);
1517}
1518
1519$heightforframes = 50;
1520
1521// Init menu manager
1522if (!defined('NOREQUIREMENU')) {
1523 if (empty($user->socid)) { // If internal user or not defined
1524 $conf->standard_menu = getDolGlobalString('MAIN_MENU_STANDARD_FORCED', getDolGlobalString('MAIN_MENU_STANDARD', 'eldy_menu.php'));
1525 } else {
1526 // If external user
1527 $conf->standard_menu = getDolGlobalString('MAIN_MENUFRONT_STANDARD_FORCED', getDolGlobalString('MAIN_MENUFRONT_STANDARD', 'eldy_menu.php'));
1528 }
1529
1530 // Load the menu manager (only if not already done)
1531 $file_menu = $conf->standard_menu;
1532 if (GETPOST('menu', 'alpha')) {
1533 $file_menu = GETPOST('menu', 'alpha'); // example: menu=eldy_menu.php
1534 }
1535 if (!class_exists('MenuManager')) {
1536 $menufound = 0;
1537 $dirmenus = array_merge(array("/core/menus/"), (array) $conf->modules_parts['menus']);
1538 foreach ($dirmenus as $dirmenu) {
1539 $menufound = dol_include_once($dirmenu."standard/".$file_menu);
1540 if (class_exists('MenuManager')) {
1541 break;
1542 }
1543 }
1544 if (!class_exists('MenuManager')) { // If failed to include, we try with standard eldy_menu.php
1545 dol_syslog("You define a menu manager '".$file_menu."' that can not be loaded.", LOG_WARNING);
1546 $file_menu = 'eldy_menu.php';
1547 include_once DOL_DOCUMENT_ROOT."/core/menus/standard/".$file_menu;
1548 }
1549 }
1550 $menumanager = new MenuManager($db, empty($user->socid) ? 0 : 1);
1551 $menumanager->loadMenu();
1552}
1553
1554if (!empty(GETPOST('seteventmessages', 'alpha'))) {
1555 $message = GETPOST('seteventmessages', 'alpha');
1556 $messages = explode(',', $message);
1557 foreach ($messages as $key => $msg) {
1558 $tmp = explode(':', $msg);
1559 setEventMessages($tmp[0], null, !empty($tmp[1]) ? $tmp[1] : 'mesgs');
1560 }
1561}
1562
1563// Functions
1564
1565if (!function_exists("llxHeader")) {
1586 function llxHeader($head = '', $title = '', $help_url = '', $target = '', $disablejs = 0, $disablehead = 0, $arrayofjs = '', $arrayofcss = '', $morequerystring = '', $morecssonbody = '', $replacemainareaby = '', $disablenofollow = 0, $disablenoindex = 0)
1587 {
1588 global $conf, $hookmanager;
1589
1590 $parameters = array(
1591 'head' => & $head,
1592 'title' => & $title,
1593 'help_url' => & $help_url,
1594 'target' => & $target,
1595 'disablejs' => & $disablejs,
1596 'disablehead' => & $disablehead,
1597 'arrayofjs' => & $arrayofjs,
1598 'arrayofcss' => & $arrayofcss,
1599 'morequerystring' => & $morequerystring,
1600 'morecssonbody' => & $morecssonbody,
1601 'replacemainareaby' => & $replacemainareaby,
1602 'disablenofollow' => & $disablenofollow,
1603 'disablenoindex' => & $disablenoindex
1604
1605 );
1606 $reshook = $hookmanager->executeHooks('llxHeader', $parameters);
1607 if ($reshook > 0) {
1608 print $hookmanager->resPrint;
1609 return;
1610 }
1611
1612 // html header
1613 top_htmlhead($head, $title, $disablejs, $disablehead, $arrayofjs, $arrayofcss, 0, $disablenofollow, $disablenoindex);
1614
1615 $tmpcsstouse = 'sidebar-collapse'.($morecssonbody ? ' '.$morecssonbody : '');
1616 // If theme MD and classic layer, we open the menulayer by default.
1617 if ($conf->theme == 'md' && !in_array($conf->browser->layout, array('phone', 'tablet')) && !getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
1618 global $mainmenu;
1619 if ($mainmenu != 'website') {
1620 $tmpcsstouse = $morecssonbody; // We do not use sidebar-collpase by default to have menuhider open by default.
1621 }
1622 }
1623
1624 if (getDolGlobalString('MAIN_OPTIMIZEFORCOLORBLIND')) {
1625 $tmpcsstouse .= ' colorblind-'.strip_tags($conf->global->MAIN_OPTIMIZEFORCOLORBLIND);
1626 }
1627
1628 print '<body id="mainbody" class="'.$tmpcsstouse.'">'."\n";
1629
1630 // top menu and left menu area
1631 if ((empty($conf->dol_hide_topmenu) || GETPOSTINT('dol_invisible_topmenu')) && !GETPOSTINT('dol_openinpopup')) {
1632 top_menu($head, $title, $target, $disablejs, $disablehead, $arrayofjs, $arrayofcss, $morequerystring, $help_url);
1633 }
1634
1635 if (empty($conf->dol_hide_leftmenu) && !GETPOST('dol_openinpopup', 'aZ09')) {
1636 left_menu(array(), $help_url, '', array(), 1, $title, 1); // $menumanager is retrieved with a global $menumanager inside this function
1637 }
1638
1639 // main area
1640 if ($replacemainareaby) {
1641 print $replacemainareaby;
1642 return;
1643 }
1644
1645 main_area($title);
1646 }
1647}
1648
1649
1657function top_httphead($contenttype = 'text/html', $forcenocache = 0)
1658{
1659 global $db, $conf, $hookmanager;
1660
1661 if ($contenttype == 'text/html') {
1662 header("Content-Type: text/html; charset=".$conf->file->character_set_client);
1663 } else {
1664 header("Content-Type: ".$contenttype);
1665 }
1666
1667 // Security options
1668
1669 // X-Content-Type-Options
1670 header("X-Content-Type-Options: nosniff"); // With the nosniff option, if the server says the content is text/html, the browser will render it as text/html (note that most browsers now force this option to on)
1671
1672 // X-Frame-Options
1673 if (!defined('XFRAMEOPTIONS_ALLOWALL')) {
1674 header("X-Frame-Options: SAMEORIGIN"); // By default, frames allowed only if on same domain (stop some XSS attacks)
1675 } else {
1676 header("X-Frame-Options: ALLOWALL");
1677 }
1678
1679 if (getDolGlobalString('MAIN_SECURITY_FORCE_ACCESS_CONTROL_ALLOW_ORIGIN')) {
1680 $tmpurl = constant('DOL_MAIN_URL_ROOT');
1681 $tmpurl = preg_replace('/^(https?:\/\/[^\/]+)\/.*$/', '\1', $tmpurl);
1682 header('Access-Control-Allow-Origin: '.$tmpurl);
1683 header('Vary: Origin');
1684 }
1685
1686 // X-XSS-Protection
1687 //header("X-XSS-Protection: 1"); // XSS filtering protection of some browsers (note: use of Content-Security-Policy is more efficient). Disabled as deprecated.
1688
1689 // Content-Security-Policy-Report-Only
1690 if (!defined('MAIN_SECURITY_FORCECSPRO')) {
1691 // If CSP not forced from the page
1692
1693 // A default security policy that keep usage of js external component like ckeditor, stripe, google, working
1694 // For example: to restrict to only local resources, except for css (cloudflare+google), and js (transifex + google tags) and object/iframe (youtube)
1695 // default-src 'self'; style-src: https://cdnjs.cloudflare.com https://fonts.googleapis.com; script-src: https://cdn.transifex.com https://www.googletagmanager.com; object-src https://youtube.com; frame-src https://youtube.com; img-src: *;
1696 // For example, to restrict everything to itself except img that can be on other servers:
1697 // default-src 'self'; img-src *;
1698 // Pre-existing site that uses too much js code to fix but wants to ensure resources are loaded only over https and disable plugins:
1699 // default-src https: 'unsafe-inline' 'unsafe-eval'; object-src 'none'
1700 //
1701 // $contentsecuritypolicy = "frame-ancestors 'self'; img-src * data:; font-src *; default-src 'self' 'unsafe-inline' 'unsafe-eval' *.paypal.com *.stripe.com *.google.com *.googleapis.com *.google-analytics.com *.googletagmanager.com;";
1702 // $contentsecuritypolicy = "frame-ancestors 'self'; img-src * data:; font-src *; default-src *; script-src 'self' 'unsafe-inline' *.paypal.com *.stripe.com *.google.com *.googleapis.com *.google-analytics.com *.googletagmanager.com; style-src 'self' 'unsafe-inline'; connect-src 'self';";
1703 $contentsecuritypolicy = getDolGlobalString('MAIN_SECURITY_FORCECSPRO');
1704
1705 if (!is_object($hookmanager)) {
1706 include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
1707 $hookmanager = new HookManager($db);
1708 }
1709 $hookmanager->initHooks(array("main"));
1710
1711 $parameters = array('contentsecuritypolicy' => $contentsecuritypolicy, 'mode' => 'reportonly');
1712 $result = $hookmanager->executeHooks('setContentSecurityPolicy', $parameters); // Note that $action and $object may have been modified by some hooks
1713 if ($result > 0) {
1714 $contentsecuritypolicy = $hookmanager->resPrint; // Replace CSP
1715 } else {
1716 $contentsecuritypolicy .= $hookmanager->resPrint; // Concat CSP
1717 }
1718
1719 if (!empty($contentsecuritypolicy)) {
1720 header("Content-Security-Policy-Report-Only: ".$contentsecuritypolicy);
1721 }
1722 } else {
1723 header("Content-Security-Policy: ".constant('MAIN_SECURITY_FORCECSPRO'));
1724 }
1725
1726 // Content-Security-Policy
1727 if (!defined('MAIN_SECURITY_FORCECSP')) {
1728 // If CSP not forced from the page
1729
1730 // A default security policy that keep usage of js external component like ckeditor, stripe, google, working
1731 // For example: to restrict to only local resources, except for css (cloudflare+google), and js (transifex + google tags) and object/iframe (youtube)
1732 // default-src 'self'; style-src: https://cdnjs.cloudflare.com https://fonts.googleapis.com; script-src: https://cdn.transifex.com https://www.googletagmanager.com; object-src https://youtube.com; frame-src https://youtube.com; img-src: *;
1733 // For example, to restrict everything to itself except img that can be on other servers:
1734 // default-src 'self'; img-src *;
1735 // Pre-existing site that uses too much js code to fix but wants to ensure resources are loaded only over https and disable plugins:
1736 // default-src https: 'unsafe-inline' 'unsafe-eval'; object-src 'none'
1737 //
1738 // $contentsecuritypolicy = "frame-ancestors 'self'; img-src * data:; font-src *; default-src 'self' 'unsafe-inline' 'unsafe-eval' *.paypal.com *.stripe.com *.google.com *.googleapis.com *.google-analytics.com *.googletagmanager.com;";
1739 // $contentsecuritypolicy = "frame-ancestors 'self'; img-src * data:; font-src *; default-src *; script-src 'self' 'unsafe-inline' *.paypal.com *.stripe.com *.google.com *.googleapis.com *.google-analytics.com *.googletagmanager.com; style-src 'self' 'unsafe-inline'; connect-src 'self';";
1740 $contentsecuritypolicy = getDolGlobalString('MAIN_SECURITY_FORCECSP');
1741
1742 if (!is_object($hookmanager)) {
1743 include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
1744 $hookmanager = new HookManager($db);
1745 }
1746 $hookmanager->initHooks(array("main"));
1747
1748 $parameters = array('contentsecuritypolicy' => $contentsecuritypolicy, 'mode' => 'active');
1749 $result = $hookmanager->executeHooks('setContentSecurityPolicy', $parameters); // Note that $action and $object may have been modified by some hooks
1750 if ($result > 0) {
1751 $contentsecuritypolicy = $hookmanager->resPrint; // Replace CSP
1752 } else {
1753 $contentsecuritypolicy .= $hookmanager->resPrint; // Concat CSP
1754 }
1755
1756 if (!empty($contentsecuritypolicy)) {
1757 header("Content-Security-Policy: ".$contentsecuritypolicy);
1758 }
1759 } else {
1760 header("Content-Security-Policy: ".constant('MAIN_SECURITY_FORCECSP'));
1761 }
1762
1763 // Referrer-Policy
1764 // Say if we must provide the referrer when we jump onto another web page.
1765 // Default browser are 'strict-origin-when-cross-origin' (only domain is sent on other domain switching), we want more so we use 'same-origin' so browser doesn't send any referrer at all when going into another web site domain.
1766 // Note that we do not use 'strict-origin' as this breaks feature to restore filters when clicking on "back to page" link on some cases.
1767 if (!defined('MAIN_SECURITY_FORCERP')) {
1768 $referrerpolicy = getDolGlobalString('MAIN_SECURITY_FORCERP', "same-origin");
1769
1770 header("Referrer-Policy: ".$referrerpolicy);
1771 }
1772
1773 if ($forcenocache) {
1774 header("Cache-Control: no-cache, no-store, must-revalidate, max-age=0");
1775 }
1776
1777 // No need to add this token in header, we use instead the one into the forms.
1778 //header("anti-csrf-token: ".newToken());
1779}
1780
1796function top_htmlhead($head, $title = '', $disablejs = 0, $disablehead = 0, $arrayofjs = array(), $arrayofcss = array(), $disableforlogin = 0, $disablenofollow = 0, $disablenoindex = 0)
1797{
1798 global $db, $conf, $langs, $user, $mysoc, $hookmanager;
1799
1800 top_httphead();
1801
1802 if (empty($conf->css)) {
1803 $conf->css = '/theme/eldy/style.css.php'; // If not defined, eldy by default
1804 }
1805
1806 print '<!doctype html>'."\n";
1807
1808 print '<html lang="'.substr($langs->defaultlang, 0, 2).'">'."\n";
1809
1810 //print '<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="fr">'."\n";
1811 if (empty($disablehead)) {
1812 if (!is_object($hookmanager)) {
1813 include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
1814 $hookmanager = new HookManager($db);
1815 }
1816 $hookmanager->initHooks(array("main"));
1817
1818 $ext = 'layout='.(empty($conf->browser->layout) ? '' : $conf->browser->layout).'&amp;version='.urlencode(DOL_VERSION);
1819
1820 print "<head>\n";
1821
1822 if (GETPOST('dol_basehref', 'alpha')) {
1823 print '<base href="'.dol_escape_htmltag(GETPOST('dol_basehref', 'alpha')).'">'."\n";
1824 }
1825
1826 // Displays meta
1827 print '<meta charset="utf-8">'."\n";
1828 print '<meta name="robots" content="'.($disablenoindex ? 'index' : 'noindex').($disablenofollow ? ',follow' : ',nofollow').'">'."\n"; // Do not index
1829 print '<meta name="viewport" content="width=device-width, initial-scale=1.0">'."\n"; // Scale for mobile device
1830 print '<meta name="author" content="Dolibarr Development Team">'."\n";
1831 print '<meta name="anti-csrf-newtoken" content="'.newToken().'">'."\n";
1832 print '<meta name="anti-csrf-currenttoken" content="'.currentToken().'">'."\n";
1833 if (getDolGlobalInt('MAIN_FEATURES_LEVEL')) {
1834 print '<meta name="MAIN_FEATURES_LEVEL" content="'.getDolGlobalInt('MAIN_FEATURES_LEVEL').'">'."\n";
1835 }
1836 // Favicon
1837 $favicon = DOL_URL_ROOT.'/theme/dolibarr_256x256_color.png';
1838 if (!empty($mysoc->logo_squarred_mini)) {
1839 $favicon = DOL_URL_ROOT.'/viewimage.php?cache=1&modulepart=mycompany&file='.urlencode('logos/thumbs/'.$mysoc->logo_squarred_mini);
1840 }
1841 if (getDolGlobalString('MAIN_FAVICON_URL')) {
1842 $favicon = getDolGlobalString('MAIN_FAVICON_URL');
1843 }
1844 if (empty($conf->dol_use_jmobile)) {
1845 print '<link rel="shortcut icon" type="image/x-icon" href="'.$favicon.'"/>'."\n"; // Not required into an Android webview
1846 }
1847
1848 // Mobile appli like icon
1849 $manifest = DOL_URL_ROOT.'/theme/'.$conf->theme.'/manifest.json.php';
1850 $parameters = array('manifest' => $manifest);
1851 $resHook = $hookmanager->executeHooks('hookSetManifest', $parameters); // Note that $action and $object may have been modified by some hooks
1852 if ($resHook > 0) {
1853 $manifest = $hookmanager->resPrint; // Replace manifest.json
1854 } else {
1855 $manifest .= $hookmanager->resPrint; // Concat to actual manifest declaration
1856 }
1857 if (!empty($manifest)) {
1858 print '<link rel="manifest" href="'.$manifest.'" />'."\n";
1859 }
1860
1861 if (getDolGlobalString('THEME_ELDY_TOPMENU_BACK1')) {
1862 // TODO: use auto theme color switch
1863 print '<meta name="theme-color" content="rgb(' . getDolGlobalString('THEME_ELDY_TOPMENU_BACK1').')">'."\n";
1864 }
1865
1866 // Auto refresh page
1867 if (GETPOSTINT('autorefresh') > 0) {
1868 print '<meta http-equiv="refresh" content="'.GETPOSTINT('autorefresh').'">';
1869 }
1870
1871 // Displays title
1872 $appli = constant('DOL_APPLICATION_TITLE');
1873 if (getDolGlobalString('MAIN_APPLICATION_TITLE')) {
1874 $appli = getDolGlobalString('MAIN_APPLICATION_TITLE');
1875 }
1876
1877 print '<title>';
1878 $titletoshow = '';
1879 if ($title && preg_match('/showapp/', getDolGlobalString('MAIN_HTML_TITLE'))) {
1880 $titletoshow = dol_htmlentities($appli.' - '.$title);
1881 } elseif ($title) {
1882 $titletoshow = dol_htmlentities($title);
1883 } else {
1884 $titletoshow = dol_htmlentities($appli);
1885 }
1886
1887 $parameters = array('title' => $titletoshow);
1888 $result = $hookmanager->executeHooks('setHtmlTitle', $parameters); // Note that $action and $object may have been modified by some hooks
1889 if ($result > 0) {
1890 $titletoshow = $hookmanager->resPrint; // Replace Title to show
1891 } else {
1892 $titletoshow .= $hookmanager->resPrint; // Concat to Title to show
1893 }
1894
1895 print $titletoshow;
1896 print '</title>';
1897
1898 print "\n";
1899
1900 if (GETPOSTINT('version')) {
1901 $ext = 'version='.GETPOSTINT('version'); // useful to force no cache on css/js
1902 }
1903 // Refresh value of MAIN_IHM_PARAMS_REV before forging the parameter line.
1904 if (GETPOST('dol_resetcache')) {
1905 include_once DOL_DOCUMENT_ROOT.'/core/lib/admin.lib.php';
1906 dolibarr_set_const($db, "MAIN_IHM_PARAMS_REV", getDolGlobalInt('MAIN_IHM_PARAMS_REV') + 1, 'chaine', 0, '', $conf->entity);
1907 }
1908
1909 $themeparam = '?lang='.$langs->defaultlang.'&amp;theme='.$conf->theme.(GETPOST('optioncss', 'aZ09') ? '&amp;optioncss='.GETPOST('optioncss', 'aZ09', 1) : '').(empty($user->id) ? '' : ('&amp;userid='.$user->id)).'&amp;entity='.$conf->entity;
1910
1911 $themeparam .= ($ext ? '&amp;'.$ext : '').'&amp;revision='.getDolGlobalInt("MAIN_IHM_PARAMS_REV");
1912 if (GETPOSTISSET('dol_hide_topmenu')) {
1913 $themeparam .= '&amp;dol_hide_topmenu='.GETPOSTINT('dol_hide_topmenu');
1914 }
1915 if (GETPOSTISSET('dol_hide_leftmenu')) {
1916 $themeparam .= '&amp;dol_hide_leftmenu='.GETPOSTINT('dol_hide_leftmenu');
1917 }
1918 if (GETPOSTISSET('dol_optimize_smallscreen')) {
1919 $themeparam .= '&amp;dol_optimize_smallscreen='.GETPOSTINT('dol_optimize_smallscreen');
1920 }
1921 if (GETPOSTISSET('dol_no_mouse_hover')) {
1922 $themeparam .= '&amp;dol_no_mouse_hover='.GETPOSTINT('dol_no_mouse_hover');
1923 }
1924 if (GETPOSTISSET('dol_use_jmobile')) {
1925 $themeparam .= '&amp;dol_use_jmobile='.GETPOSTINT('dol_use_jmobile');
1926 $conf->dol_use_jmobile = GETPOSTINT('dol_use_jmobile');
1927 }
1928 if (GETPOSTISSET('THEME_DARKMODEENABLED')) {
1929 $themeparam .= '&amp;THEME_DARKMODEENABLED='.GETPOSTINT('THEME_DARKMODEENABLED');
1930 }
1931 if (GETPOSTISSET('THEME_SATURATE_RATIO')) {
1932 $themeparam .= '&amp;THEME_SATURATE_RATIO='.GETPOSTINT('THEME_SATURATE_RATIO');
1933 }
1934
1935 if (getDolGlobalString('MAIN_ENABLE_FONT_ROBOTO')) {
1936 print '<link rel="preconnect" href="https://fonts.gstatic.com">'."\n";
1937 print '<link href="https://fonts.googleapis.com/css2?family=Roboto:wght@200;300;400;500;600&display=swap" rel="stylesheet">'."\n";
1938 }
1939
1940 if (!defined('DISABLE_JQUERY') && !$disablejs && $conf->use_javascript_ajax) {
1941 print '<!-- Includes CSS for JQuery (Ajax library) -->'."\n";
1942 $jquerytheme = 'base';
1943 if (getDolGlobalString('MAIN_USE_JQUERY_THEME')) {
1944 $jquerytheme = getDolGlobalString('MAIN_USE_JQUERY_THEME');
1945 }
1946 if (constant('JS_JQUERY_UI')) {
1947 print '<link rel="stylesheet" type="text/css" href="'.JS_JQUERY_UI.'css/'.$jquerytheme.'/jquery-ui.min.css'.($ext ? '?'.$ext : '').'">'."\n"; // Forced JQuery
1948 } else {
1949 print '<link rel="stylesheet" type="text/css" href="'.DOL_URL_ROOT.'/includes/jquery/css/'.$jquerytheme.'/jquery-ui.css'.($ext ? '?'.$ext : '').'">'."\n"; // JQuery
1950 }
1951 if (!defined('DISABLE_JQUERY_JNOTIFY')) {
1952 print '<link rel="stylesheet" type="text/css" href="'.DOL_URL_ROOT.'/includes/jquery/plugins/jnotify/jquery.jnotify-alt.min.css'.($ext ? '?'.$ext : '').'">'."\n"; // JNotify
1953 }
1954 if (!defined('DISABLE_SELECT2') && (getDolGlobalString('MAIN_USE_JQUERY_MULTISELECT') || defined('REQUIRE_JQUERY_MULTISELECT'))) { // jQuery plugin "mutiselect", "multiple-select", "select2"...
1955 $tmpplugin = !getDolGlobalString('MAIN_USE_JQUERY_MULTISELECT') ? constant('REQUIRE_JQUERY_MULTISELECT') : $conf->global->MAIN_USE_JQUERY_MULTISELECT;
1956 print '<link rel="stylesheet" type="text/css" href="'.DOL_URL_ROOT.'/includes/jquery/plugins/'.$tmpplugin.'/dist/css/'.$tmpplugin.'.css'.($ext ? '?'.$ext : '').'">'."\n";
1957 }
1958 }
1959
1960 if (!defined('DISABLE_FONT_AWSOME')) {
1961 print '<!-- Includes CSS for font awesome -->'."\n";
1962 $fontawesome_directory = getDolGlobalString('MAIN_FONTAWESOME_DIRECTORY', '/theme/common/fontawesome-5');
1963 print '<link rel="stylesheet" type="text/css" href="'.DOL_URL_ROOT.$fontawesome_directory.'/css/all.min.css'.($ext ? '?'.$ext : '').'">'."\n";
1964 }
1965
1966 print '<!-- Includes CSS for Dolibarr theme -->'."\n";
1967 // Output style sheets (optioncss='print' or ''). Note: $conf->css looks like '/theme/eldy/style.css.php'
1968 $themepath = dol_buildpath($conf->css, 1);
1969 $themesubdir = '';
1970 if (!empty($conf->modules_parts['theme'])) { // This slow down
1971 foreach ($conf->modules_parts['theme'] as $reldir) {
1972 if (file_exists(dol_buildpath($reldir.$conf->css, 0))) {
1973 $themepath = dol_buildpath($reldir.$conf->css, 1);
1974 $themesubdir = $reldir;
1975 break;
1976 }
1977 }
1978 }
1979
1980 //print 'themepath='.$themepath.' themeparam='.$themeparam;exit;
1981 print '<link rel="stylesheet" type="text/css" href="'.$themepath.$themeparam.'">'."\n";
1982 if (getDolGlobalString('MAIN_FIX_FLASH_ON_CHROME')) {
1983 print '<!-- Includes CSS that does not exists as a workaround of flash bug of chrome -->'."\n".'<link rel="stylesheet" type="text/css" href="filethatdoesnotexiststosolvechromeflashbug">'."\n";
1984 }
1985
1986 // LEAFLET AND GEOMAN
1987 if (getDolGlobalString('MAIN_USE_GEOPHP')) {
1988 print '<link rel="stylesheet" href="'.DOL_URL_ROOT.'/includes/leaflet/leaflet.css'.($ext ? '?'.$ext : '')."\">\n";
1989 print '<link rel="stylesheet" href="'.DOL_URL_ROOT.'/includes/leaflet/leaflet-geoman.css'.($ext ? '?'.$ext : '')."\">\n";
1990 }
1991
1992 // CSS forced by modules (relative url starting with /)
1993 if (!empty($conf->modules_parts['css'])) {
1994 $arraycss = (array) $conf->modules_parts['css'];
1995 foreach ($arraycss as $modcss => $filescss) {
1996 $filescss = (array) $filescss; // To be sure filecss is an array
1997 foreach ($filescss as $cssfile) {
1998 if (empty($cssfile)) {
1999 dol_syslog("Warning: module ".$modcss." declared a css path file into its descriptor that is empty.", LOG_WARNING);
2000 }
2001 // cssfile is a relative path
2002 $urlforcss = dol_buildpath($cssfile, 1);
2003 if ($urlforcss && $urlforcss != '/') {
2004 print '<!-- Includes CSS added by module '.$modcss.' -->'."\n".'<link rel="stylesheet" type="text/css" href="'.$urlforcss;
2005 // We add params only if page is not static, because some web server setup does not return content type text/css if url has parameters, so browser cache is not used.
2006 if (!preg_match('/\.css$/i', $cssfile)) {
2007 print $themeparam;
2008 }
2009 print '">'."\n";
2010 } else {
2011 dol_syslog("Warning: module ".$modcss." declared a css path file for a file we can't find.", LOG_WARNING);
2012 }
2013 }
2014 }
2015 }
2016 // CSS forced by page in top_htmlhead call (relative url starting with /)
2017 if (is_array($arrayofcss)) {
2018 foreach ($arrayofcss as $cssfile) {
2019 if (preg_match('/^(http|\/\/)/i', $cssfile)) {
2020 $urltofile = $cssfile;
2021 } else {
2022 $urltofile = dol_buildpath($cssfile, 1);
2023 }
2024 print '<!-- Includes CSS added by page -->'."\n".'<link rel="stylesheet" type="text/css" title="default" href="'.$urltofile;
2025 // We add params only if page is not static, because some web server setup does not return content type text/css if url has parameters and browser cache is not used.
2026 if (!preg_match('/\.css$/i', $cssfile)) {
2027 print $themeparam;
2028 }
2029 print '">'."\n";
2030 }
2031 }
2032
2033 // Custom CSS
2034 if (getDolGlobalString('MAIN_IHM_CUSTOM_CSS')) {
2035 // If a custom CSS was set, we add link to the custom css php file
2036 print '<link rel="stylesheet" type="text/css" href="'.DOL_URL_ROOT.'/theme/custom.css.php'.($ext ? '?'.$ext : '').'&amp;revision='.getDolGlobalInt("MAIN_IHM_PARAMS_REV").'">'."\n";
2037 }
2038
2039 // Output standard javascript links
2040 if (!defined('DISABLE_JQUERY') && !$disablejs && !empty($conf->use_javascript_ajax)) {
2041 // JQuery. Must be before other includes
2042 print '<!-- Includes JS for JQuery -->'."\n";
2043 if (defined('JS_JQUERY') && constant('JS_JQUERY')) {
2044 print '<script nonce="'.getNonce().'" src="'.JS_JQUERY.'jquery.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2045 } else {
2046 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/js/jquery.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2047 }
2048 if (defined('JS_JQUERY_UI') && constant('JS_JQUERY_UI')) {
2049 print '<script nonce="'.getNonce().'" src="'.JS_JQUERY_UI.'jquery-ui.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2050 } else {
2051 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/js/jquery-ui.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2052 }
2053 // jQuery jnotify
2054 if (!getDolGlobalString('MAIN_DISABLE_JQUERY_JNOTIFY') && !defined('DISABLE_JQUERY_JNOTIFY')) {
2055 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/jnotify/jquery.jnotify.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2056 }
2057 // Table drag and drop lines
2058 if (empty($disableforlogin) && !defined('DISABLE_JQUERY_TABLEDND')) {
2059 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/tablednd/jquery.tablednd.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2060 }
2061 // Chart
2062 if (empty($disableforlogin) && (!getDolGlobalString('MAIN_JS_GRAPH') || getDolGlobalString('MAIN_JS_GRAPH') == 'chart') && !defined('DISABLE_JS_GRAPH')) {
2063 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/nnnick/chartjs/dist/chart.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2064 }
2065
2066 // jQuery jeditable for Edit In Place features
2067 if (getDolGlobalString('MAIN_USE_JQUERY_JEDITABLE') && !defined('DISABLE_JQUERY_JEDITABLE')) {
2068 print '<!-- JS to manage editInPlace feature -->'."\n";
2069 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/jeditable/jquery.jeditable.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2070 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/jeditable/jquery.jeditable.ui-datepicker.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2071 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/jeditable/jquery.jeditable.ui-autocomplete.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2072 print '<script>'."\n";
2073 print 'var urlSaveInPlace = \''.DOL_URL_ROOT.'/core/ajax/saveinplace.php\';'."\n";
2074 print 'var urlLoadInPlace = \''.DOL_URL_ROOT.'/core/ajax/loadinplace.php\';'."\n";
2075 print 'var tooltipInPlace = \''.$langs->transnoentities('ClickToEdit').'\';'."\n"; // Added in title attribute of span
2076 print 'var placeholderInPlace = \'&nbsp;\';'."\n"; // If we put another string than $langs->trans("ClickToEdit") here, nothing is shown. If we put empty string, there is error, Why ?
2077 print 'var cancelInPlace = \''.$langs->trans("Cancel").'\';'."\n";
2078 print 'var submitInPlace = \''.$langs->trans('Ok').'\';'."\n";
2079 print 'var indicatorInPlace = \'<img src="'.DOL_URL_ROOT."/theme/".$conf->theme."/img/working.gif".'">\';'."\n";
2080 print 'var withInPlace = 300;'; // width in pixel for default string edit
2081 print '</script>'."\n";
2082 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/core/js/editinplace.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2083 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/jeditable/jquery.jeditable.ckeditor.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2084 }
2085 // jQuery Timepicker
2086 if (getDolGlobalString('MAIN_USE_JQUERY_TIMEPICKER') || defined('REQUIRE_JQUERY_TIMEPICKER')) {
2087 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/timepicker/jquery-ui-timepicker-addon.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2088 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/core/js/timepicker.js.php?lang='.$langs->defaultlang.($ext ? '&amp;'.$ext : '').'"></script>'."\n";
2089 }
2090 if (!defined('DISABLE_SELECT2') && (getDolGlobalString('MAIN_USE_JQUERY_MULTISELECT') || defined('REQUIRE_JQUERY_MULTISELECT'))) {
2091 // jQuery plugin "mutiselect", "multiple-select", "select2", ...
2092 $tmpplugin = !getDolGlobalString('MAIN_USE_JQUERY_MULTISELECT') ? constant('REQUIRE_JQUERY_MULTISELECT') : $conf->global->MAIN_USE_JQUERY_MULTISELECT;
2093 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/'.$tmpplugin.'/dist/js/'.$tmpplugin.'.full.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n"; // We include full because we need the support of containerCssClass
2094 }
2095 if (!defined('DISABLE_MULTISELECT')) { // jQuery plugin "mutiselect" to select with checkboxes. Can be removed once we have an enhanced search tool
2096 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/jquery/plugins/multiselect/jquery.multi-select.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2097 }
2098 }
2099
2100 if (!$disablejs && !empty($conf->use_javascript_ajax)) {
2101 // CKEditor
2102 if (empty($disableforlogin) && (isModEnabled('fckeditor') && (!getDolGlobalString('FCKEDITOR_EDITORNAME') || getDolGlobalString('FCKEDITOR_EDITORNAME') == 'ckeditor') && !defined('DISABLE_CKEDITOR')) || defined('FORCE_CKEDITOR')) {
2103 print '<!-- Includes JS for CKEditor -->'."\n";
2104 $pathckeditor = DOL_URL_ROOT.'/includes/ckeditor/ckeditor/';
2105 $jsckeditor = 'ckeditor.js';
2106 if (constant('JS_CKEDITOR')) {
2107 // To use external ckeditor 4 js lib
2108 $pathckeditor = constant('JS_CKEDITOR');
2109 }
2110 print '<script nonce="'.getNonce().'">';
2111 print '/* enable ckeditor by main.inc.php */';
2112 print 'var CKEDITOR_BASEPATH = \''.dol_escape_js($pathckeditor).'\';'."\n";
2113 print 'var ckeditorConfig = \''.dol_escape_js(dol_buildpath($themesubdir.'/theme/'.$conf->theme.'/ckeditor/config.js'.($ext ? '?'.$ext : ''), 1)).'\';'."\n"; // $themesubdir='' in standard usage
2114 print 'var ckeditorFilebrowserBrowseUrl = \''.DOL_URL_ROOT.'/core/filemanagerdol/browser/default/browser.php?Connector='.DOL_URL_ROOT.'/core/filemanagerdol/connectors/php/connector.php\';'."\n";
2115 print 'var ckeditorFilebrowserImageBrowseUrl = \''.DOL_URL_ROOT.'/core/filemanagerdol/browser/default/browser.php?Type=Image&Connector='.DOL_URL_ROOT.'/core/filemanagerdol/connectors/php/connector.php\';'."\n";
2116 print '</script>'."\n";
2117 print '<script src="'.$pathckeditor.$jsckeditor.($ext ? '?'.$ext : '').'"></script>'."\n";
2118 print '<script>';
2119 if (GETPOST('mode', 'aZ09') == 'Full_inline') {
2120 print 'CKEDITOR.disableAutoInline = false;'."\n";
2121 } else {
2122 print 'CKEDITOR.disableAutoInline = true;'."\n";
2123 }
2124 print '</script>'."\n";
2125 }
2126
2127 // Browser notifications (if NOREQUIREMENU is on, it is mostly a page for popup, so we do not enable notif too. We hide also for public pages).
2128 if (!defined('NOBROWSERNOTIF') && !defined('NOREQUIREMENU') && !defined('NOLOGIN')) {
2129 $enablebrowsernotif = false;
2130 if (isModEnabled('agenda') && getDolGlobalString('AGENDA_REMINDER_BROWSER')) {
2131 $enablebrowsernotif = true;
2132 }
2133 if ($conf->browser->layout == 'phone') {
2134 $enablebrowsernotif = false;
2135 }
2136 if ($enablebrowsernotif) {
2137 print '<!-- Includes JS of Dolibarr (browser layout = '.$conf->browser->layout.')-->'."\n";
2138 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/core/js/lib_notification.js.php'.($ext ? '?'.$ext : '').'"></script>'."\n";
2139 }
2140 }
2141
2142 // Global js function
2143 print '<!-- Includes JS of Dolibarr -->'."\n";
2144 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/core/js/lib_head.js.php?lang='.$langs->defaultlang.($ext ? '&amp;'.$ext : '').'"></script>'."\n";
2145
2146 // Leaflet TODO use dolibarr files
2147 if (getDolGlobalString('MAIN_USE_GEOPHP')) {
2148 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/leaflet/leaflet.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2149 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/includes/leaflet/leaflet-geoman.min.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2150 }
2151
2152 // JS forced by modules (relative url starting with /)
2153 if (!empty($conf->modules_parts['js'])) { // $conf->modules_parts['js'] is array('module'=>array('file1','file2'))
2154 $arrayjs = (array) $conf->modules_parts['js'];
2155 foreach ($arrayjs as $modjs => $filesjs) {
2156 $filesjs = (array) $filesjs; // To be sure filejs is an array
2157 foreach ($filesjs as $jsfile) {
2158 // jsfile is a relative path
2159 $urlforjs = dol_buildpath($jsfile, 1);
2160 if ($urlforjs && $urlforjs != '/') {
2161 print '<!-- Include JS added by module '.$modjs.'-->'."\n";
2162 print '<script nonce="'.getNonce().'" src="'.$urlforjs.((strpos($jsfile, '?') === false) ? '?' : '&amp;').'lang='.$langs->defaultlang.'"></script>'."\n";
2163 } else {
2164 dol_syslog("Warning: module ".$modjs." declared a js path file for a file we can't find.", LOG_WARNING);
2165 }
2166 }
2167 }
2168 }
2169 // JS forced by page in top_htmlhead (relative url starting with /)
2170 if (is_array($arrayofjs)) {
2171 print '<!-- Includes JS added by page -->'."\n";
2172 foreach ($arrayofjs as $jsfile) {
2173 if (preg_match('/^(http|\/\/)/i', $jsfile)) {
2174 print '<script nonce="'.getNonce().'" src="'.$jsfile.((strpos($jsfile, '?') === false) ? '?' : '&amp;').'lang='.$langs->defaultlang.'"></script>'."\n";
2175 } else {
2176 print '<script nonce="'.getNonce().'" src="'.dol_buildpath($jsfile, 1).((strpos($jsfile, '?') === false) ? '?' : '&amp;').'lang='.$langs->defaultlang.'"></script>'."\n";
2177 }
2178 }
2179 }
2180 }
2181
2182 //If you want to load custom javascript file from your selected theme directory
2183 if (getDolGlobalString('ALLOW_THEME_JS')) {
2184 $theme_js = dol_buildpath('/theme/'.$conf->theme.'/'.$conf->theme.'.js', 0);
2185 if (file_exists($theme_js)) {
2186 print '<script nonce="'.getNonce().'" src="'.DOL_URL_ROOT.'/theme/'.$conf->theme.'/'.$conf->theme.'.js'.($ext ? '?'.$ext : '').'"></script>'."\n";
2187 }
2188 }
2189
2190 if (!empty($head)) {
2191 print $head."\n";
2192 }
2193 if (getDolGlobalString('MAIN_HTML_HEADER')) {
2194 print getDolGlobalString('MAIN_HTML_HEADER') . "\n";
2195 }
2196
2197 $parameters = array();
2198 $result = $hookmanager->executeHooks('addHtmlHeader', $parameters); // Note that $action and $object may have been modified by some hooks
2199 print $hookmanager->resPrint; // Replace Title to show
2200
2201 print "</head>\n\n";
2202 }
2203
2204 $conf->headerdone = 1; // To tell header was output
2205}
2206
2207
2224function top_menu($head, $title = '', $target = '', $disablejs = 0, $disablehead = 0, $arrayofjs = array(), $arrayofcss = array(), $morequerystring = '', $helppagename = '')
2225{
2226 global $user, $conf, $langs, $db, $form;
2227 global $dolibarr_main_authentication, $dolibarr_main_demo;
2228 global $hookmanager, $menumanager;
2229
2230 $searchform = '';
2231
2232 // Instantiate hooks for external modules
2233 $hookmanager->initHooks(array('toprightmenu'));
2234
2235 $toprightmenu = '';
2236
2237 // For backward compatibility with old modules
2238 if (empty($conf->headerdone)) {
2239 $disablenofollow = 0;
2240 top_htmlhead($head, $title, $disablejs, $disablehead, $arrayofjs, $arrayofcss, 0, $disablenofollow);
2241 print '<body id="mainbody">';
2242 }
2243
2244 /*
2245 * Top menu
2246 */
2247 if ((empty($conf->dol_hide_topmenu) || GETPOSTINT('dol_invisible_topmenu')) && (!defined('NOREQUIREMENU') || !constant('NOREQUIREMENU'))) {
2248 if (!isset($form) || !is_object($form)) {
2249 include_once DOL_DOCUMENT_ROOT.'/core/class/html.form.class.php';
2250 $form = new Form($db);
2251 }
2252
2253 print "\n".'<!-- Start top horizontal -->'."\n";
2254
2255 print '<header id="id-top" class="side-nav-vert'.(GETPOSTINT('dol_invisible_topmenu') ? ' hidden' : '').'">'; // dol_invisible_topmenu differs from dol_hide_topmenu: dol_invisible_topmenu means we output menu but we make it invisible.
2256
2257 // Show menu entries
2258 print '<div id="tmenu_tooltip'.(!getDolGlobalString('MAIN_MENU_INVERT') ? '' : 'invert').'" class="tmenu">'."\n";
2259 $menumanager->atarget = $target;
2260 $menumanager->showmenu('top', array('searchform' => $searchform)); // This contains a \n
2261 print "</div>\n";
2262
2263 // Define link to login card
2264 $appli = constant('DOL_APPLICATION_TITLE');
2265 if (getDolGlobalString('MAIN_APPLICATION_TITLE')) {
2266 $appli = getDolGlobalString('MAIN_APPLICATION_TITLE');
2267 if (preg_match('/\d\.\d/', $appli)) {
2268 if (!preg_match('/'.preg_quote(DOL_VERSION).'/', $appli)) {
2269 $appli .= " (".DOL_VERSION.")"; // If new title contains a version that is different than core
2270 }
2271 } else {
2272 $appli .= " ".DOL_VERSION;
2273 }
2274 } else {
2275 $appli .= " ".DOL_VERSION;
2276 }
2277
2278 if (getDolGlobalInt('MAIN_FEATURES_LEVEL')) {
2279 $appli .= "<br>".$langs->trans("LevelOfFeature").': '.getDolGlobalInt('MAIN_FEATURES_LEVEL');
2280 }
2281
2282 $logouttext = '';
2283 $logouthtmltext = '';
2284 if (!getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2285 //$logouthtmltext=$appli.'<br>';
2286 $stringforfirstkey = $langs->trans("KeyboardShortcut");
2287 if ($conf->browser->name == 'chrome') {
2288 $stringforfirstkey .= ' ALT +';
2289 } elseif ($conf->browser->name == 'firefox') {
2290 $stringforfirstkey .= ' ALT + SHIFT +';
2291 } else {
2292 $stringforfirstkey .= ' CTL +';
2293 }
2294 if ($_SESSION["dol_authmode"] != 'forceuser' && $_SESSION["dol_authmode"] != 'http') {
2295 $logouthtmltext .= $langs->trans("Logout").'<br>';
2296 $logouttext .= '<a accesskey="l" href="'.DOL_URL_ROOT.'/user/logout.php?token='.newToken().'">';
2297 $logouttext .= img_picto($langs->trans('Logout').' ('.$stringforfirstkey.' l)', 'sign-out', '', false, 0, 0, '', 'atoplogin valignmiddle');
2298 $logouttext .= '</a>';
2299 } else {
2300 $logouthtmltext .= $langs->trans("NoLogoutProcessWithAuthMode", $_SESSION["dol_authmode"]);
2301 $logouttext .= img_picto($langs->trans('Logout').' ('.$stringforfirstkey.' l)', 'sign-out', '', false, 0, 0, '', 'atoplogin valignmiddle opacitymedium');
2302 }
2303 }
2304
2305 print '<div class="login_block usedropdown">'."\n";
2306
2307 $toprightmenu .= '<div class="login_block_other">';
2308
2309 // Execute hook printTopRightMenu (hooks should output string like '<div class="login"><a href="">mylink</a></div>')
2310 $parameters = array();
2311 $result = $hookmanager->executeHooks('printTopRightMenu', $parameters); // Note that $action and $object may have been modified by some hooks
2312 if (is_numeric($result)) {
2313 if ($result == 0) {
2314 $toprightmenu .= $hookmanager->resPrint; // add
2315 } else {
2316 $toprightmenu = $hookmanager->resPrint; // replace
2317 }
2318 } else {
2319 $toprightmenu .= $result; // For backward compatibility
2320 }
2321
2322 // Link to module builder
2323 if (isModEnabled('modulebuilder')) {
2324 $text = '<a href="'.DOL_URL_ROOT.'/modulebuilder/index.php?mainmenu=home&leftmenu=admintools" target="modulebuilder">';
2325 //$text.= img_picto(":".$langs->trans("ModuleBuilder"), 'printer_top.png', 'class="printer"');
2326 $text .= '<span class="fa fa-bug atoplogin valignmiddle"></span>';
2327 $text .= '</a>';
2328 // @phan-suppress-next-line PhanPluginSuspiciousParamPosition
2329 $toprightmenu .= $form->textwithtooltip('', $langs->trans("ModuleBuilder"), 2, 1, $text, 'login_block_elem', 2);
2330 }
2331
2332 // Link to print main content area (optioncss=print)
2333 if (!getDolGlobalString('MAIN_PRINT_DISABLELINK') && !getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2334 $qs = dol_escape_htmltag($_SERVER["QUERY_STRING"]);
2335
2336 if (isset($_POST) && is_array($_POST)) {
2337 foreach ($_POST as $key => $value) {
2338 $key = preg_replace('/[^a-z0-9_\.\-\[\]]/i', '', $key);
2339 if (in_array($key, array('action', 'massaction', 'password'))) {
2340 continue;
2341 }
2342 if (!is_array($value)) {
2343 if ($value !== '') {
2344 $qs .= '&'.urlencode($key).'='.urlencode($value);
2345 }
2346 } else {
2347 foreach ($value as $value2) {
2348 if (($value2 !== '') && (!is_array($value2))) {
2349 $qs .= '&'.urlencode($key).'[]='.urlencode($value2);
2350 }
2351 }
2352 }
2353 }
2354 }
2355 $qs .= (($qs && $morequerystring) ? '&' : '').$morequerystring;
2356 $text = '<a href="'.dol_escape_htmltag($_SERVER["PHP_SELF"]).'?'.$qs.($qs ? '&' : '').'optioncss=print" target="_blank" rel="noopener noreferrer">';
2357 //$text.= img_picto(":".$langs->trans("PrintContentArea"), 'printer_top.png', 'class="printer"');
2358 $text .= '<span class="fa fa-print atoplogin valignmiddle"></span>';
2359 $text .= '</a>';
2360 // @phan-suppress-next-line PhanPluginSuspiciousParamPosition
2361 $toprightmenu .= $form->textwithtooltip('', $langs->trans("PrintContentArea"), 2, 1, $text, 'login_block_elem', 2);
2362 }
2363
2364 // Link to Dolibarr wiki pages
2365 if (!getDolGlobalString('MAIN_HELP_DISABLELINK') && !getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2366 $langs->load("help");
2367
2368 $helpbaseurl = '';
2369 $helppage = '';
2370 $mode = '';
2371 $helppresent = '';
2372
2373 if (empty($helppagename)) {
2374 $helppagename = 'EN:User_documentation|FR:Documentation_utilisateur|ES:Documentación_usuarios|DE:Benutzerdokumentation';
2375 } else {
2376 $helppresent = 'helppresent';
2377 }
2378
2379 // Get helpbaseurl, helppage and mode from helppagename and langs
2380 $arrayres = getHelpParamFor($helppagename, $langs);
2381 $helpbaseurl = $arrayres['helpbaseurl'];
2382 $helppage = $arrayres['helppage'];
2383 $mode = $arrayres['mode'];
2384
2385 // Link to help pages
2386 if ($helpbaseurl && $helppage) {
2387 $text = '';
2388 $title = $langs->trans($mode == 'wiki' ? 'GoToWikiHelpPage' : 'GoToHelpPage').', ';
2389 if ($mode == 'wiki') {
2390 $title .= '<br>'.img_picto('', 'globe', 'class="pictofixedwidth"').$langs->trans("PageWiki").' '.dol_escape_htmltag('"'.strtr($helppage, '_', ' ').'"');
2391 if ($helppresent) {
2392 $title .= ' <span class="opacitymedium">('.$langs->trans("DedicatedPageAvailable").')</span>';
2393 } else {
2394 $title .= ' <span class="opacitymedium">('.$langs->trans("HomePage").')</span>';
2395 }
2396 }
2397 $text .= '<a class="help" target="_blank" rel="noopener noreferrer" href="';
2398 if ($mode == 'wiki') {
2399 // @phan-suppress-next-line PhanPluginPrintfVariableFormatString
2400 $text .= sprintf($helpbaseurl, urlencode(html_entity_decode($helppage)));
2401 } else {
2402 // @phan-suppress-next-line PhanPluginPrintfVariableFormatString
2403 $text .= sprintf($helpbaseurl, $helppage);
2404 }
2405 $text .= '">';
2406 $text .= '<span class="fa fa-question-circle atoplogin valignmiddle'.($helppresent ? ' '.$helppresent : '').'"></span>';
2407 $text .= '<span class="fa fa-long-arrow-alt-up helppresentcircle'.($helppresent ? '' : ' unvisible').'"></span>';
2408 $text .= '</a>';
2409 // @phan-suppress-next-line PhanPluginSuspiciousParamPosition
2410 $toprightmenu .= $form->textwithtooltip('', $title, 2, 1, $text, 'login_block_elem', 2);
2411 }
2412
2413 // Version
2414 if (getDolGlobalString('MAIN_SHOWDATABASENAMEINHELPPAGESLINK')) {
2415 $langs->load('admin');
2416 $appli .= '<br>'.$langs->trans("Database").': '.$db->database_name;
2417 }
2418 }
2419
2420 if (!getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2421 $text = '<span class="aversion"><span class="hideonsmartphone small">'.DOL_VERSION.'</span></span>';
2422 // @phan-suppress-next-line PhanPluginSuspiciousParamPosition
2423 $toprightmenu .= $form->textwithtooltip('', $appli, 2, 1, $text, 'login_block_elem', 2);
2424 }
2425
2426 // Logout link
2427 $toprightmenu .= $form->textwithtooltip('', $logouthtmltext, 2, 1, $logouttext, 'login_block_elem logout-btn', 2);
2428
2429 $toprightmenu .= '</div>'; // end div class="login_block_other"
2430
2431
2432 // Add login user link
2433 $toprightmenu .= '<div class="login_block_user">';
2434
2435 // Login name with photo and tooltip
2436 $mode = -1;
2437 $toprightmenu .= '<div class="inline-block login_block_elem login_block_elem_name nowrap centpercent" style="padding: 0px;">';
2438
2439 if (getDolGlobalString('MAIN_USE_TOP_MENU_SEARCH_DROPDOWN')) {
2440 // Add search dropdown
2441 $toprightmenu .= top_menu_search();
2442 }
2443
2444 if (getDolGlobalString('MAIN_USE_TOP_MENU_QUICKADD_DROPDOWN')) {
2445 // Add search dropdown
2446 $toprightmenu .= top_menu_quickadd();
2447 }
2448
2449 // Add bookmark dropdown
2450 $toprightmenu .= top_menu_bookmark();
2451
2452 // Add user dropdown
2453 $toprightmenu .= top_menu_user();
2454
2455 $toprightmenu .= '</div>';
2456
2457 $toprightmenu .= '</div>'."\n";
2458
2459
2460 print $toprightmenu;
2461
2462 print "</div>\n"; // end div class="login_block"
2463
2464 print '</header>';
2465 //print '<header class="header2">&nbsp;</header>';
2466
2467 print '<div style="clear: both;"></div>';
2468 print "<!-- End top horizontal menu -->\n\n";
2469 }
2470
2471 if (empty($conf->dol_hide_leftmenu) && empty($conf->dol_use_jmobile)) {
2472 print '<!-- Begin div id-container --><div id="id-container" class="id-container">';
2473 }
2474}
2475
2476
2484function top_menu_user($hideloginname = 0, $urllogout = '')
2485{
2486 global $langs, $conf, $db, $hookmanager, $user, $mysoc;
2487 global $dolibarr_main_authentication, $dolibarr_main_demo;
2488 global $menumanager;
2489
2490 $langs->load('companies');
2491
2492 $userImage = $userDropDownImage = '';
2493 if (!empty($user->photo)) {
2494 $userImage = Form::showphoto('userphoto', $user, 0, 0, 0, 'photouserphoto userphoto', 'small', 0, 1);
2495 $userDropDownImage = Form::showphoto('userphoto', $user, 0, 0, 0, 'dropdown-user-image', 'small', 0, 1);
2496 } else {
2497 $nophoto = '/public/theme/common/user_anonymous.png';
2498 if ($user->gender == 'man') {
2499 $nophoto = '/public/theme/common/user_man.png';
2500 }
2501 if ($user->gender == 'woman') {
2502 $nophoto = '/public/theme/common/user_woman.png';
2503 }
2504
2505 $userImage = '<img class="photo photouserphoto userphoto" alt="" src="'.DOL_URL_ROOT.$nophoto.'">';
2506 $userDropDownImage = '<img class="photo dropdown-user-image" alt="" src="'.DOL_URL_ROOT.$nophoto.'">';
2507 }
2508
2509 $dropdownBody = '';
2510 $dropdownBody .= '<span id="topmenulogincompanyinfo-btn"><i class="fa fa-caret-right"></i> '.$langs->trans("ShowCompanyInfos").'</span>';
2511 $dropdownBody .= '<div id="topmenulogincompanyinfo" >';
2512
2513 $dropdownBody .= '<br><b>'.$langs->trans("Company").'</b>: <span>'.dol_escape_htmltag($mysoc->name).'</span>';
2514 if ($langs->transcountry("ProfId1", $mysoc->country_code) != '-') {
2515 $dropdownBody .= '<br><b>'.$langs->transcountry("ProfId1", $mysoc->country_code).'</b>: <span>'.dol_print_profids(getDolGlobalString("MAIN_INFO_SIREN"), 1).'</span>';
2516 }
2517 if ($langs->transcountry("ProfId2", $mysoc->country_code) != '-') {
2518 $dropdownBody .= '<br><b>'.$langs->transcountry("ProfId2", $mysoc->country_code).'</b>: <span>'.dol_print_profids(getDolGlobalString("MAIN_INFO_SIRET"), 2).'</span>';
2519 }
2520 if ($langs->transcountry("ProfId3", $mysoc->country_code) != '-') {
2521 $dropdownBody .= '<br><b>'.$langs->transcountry("ProfId3", $mysoc->country_code).'</b>: <span>'.dol_print_profids(getDolGlobalString("MAIN_INFO_APE"), 3).'</span>';
2522 }
2523 if ($langs->transcountry("ProfId4", $mysoc->country_code) != '-') {
2524 $dropdownBody .= '<br><b>'.$langs->transcountry("ProfId4", $mysoc->country_code).'</b>: <span>'.dol_print_profids(getDolGlobalString("MAIN_INFO_RCS"), 4).'</span>';
2525 }
2526 if ($langs->transcountry("ProfId5", $mysoc->country_code) != '-') {
2527 $dropdownBody .= '<br><b>'.$langs->transcountry("ProfId5", $mysoc->country_code).'</b>: <span>'.dol_print_profids(getDolGlobalString("MAIN_INFO_PROFID5"), 5).'</span>';
2528 }
2529 if ($langs->transcountry("ProfId6", $mysoc->country_code) != '-') {
2530 $dropdownBody .= '<br><b>'.$langs->transcountry("ProfId6", $mysoc->country_code).'</b>: <span>'.dol_print_profids(getDolGlobalString("MAIN_INFO_PROFID6"), 6).'</span>';
2531 }
2532 $dropdownBody .= '<br><b>'.$langs->trans("VATIntraShort").'</b>: <span>'.dol_print_profids(getDolGlobalString("MAIN_INFO_TVAINTRA"), 'VAT').'</span>';
2533 $dropdownBody .= '<br><b>'.$langs->trans("Country").'</b>: <span>'.($mysoc->country_code ? $langs->trans("Country".$mysoc->country_code) : '').'</span>';
2534 if (isModEnabled('multicurrency')) {
2535 $dropdownBody .= '<br><b>'.$langs->trans("Currency").'</b>: <span>'.$conf->currency.'</span>';
2536 }
2537 $dropdownBody .= '</div>';
2538
2539 $dropdownBody .= '<br>';
2540 $dropdownBody .= '<span id="topmenuloginmoreinfo-btn"><i class="fa fa-caret-right"></i> '.$langs->trans("ShowMoreInfos").'</span>';
2541 $dropdownBody .= '<div id="topmenuloginmoreinfo" >';
2542
2543 // login infos
2544 if (!empty($user->admin)) {
2545 $dropdownBody .= '<br><b>'.$langs->trans("Administrator").'</b>: '.yn($user->admin);
2546 }
2547 if (!empty($user->socid)) { // Add thirdparty for external users
2548 $thirdpartystatic = new Societe($db);
2549 $thirdpartystatic->fetch($user->socid);
2550 $companylink = ' '.$thirdpartystatic->getNomUrl(2); // picto only of company
2551 $company = ' ('.$langs->trans("Company").': '.$thirdpartystatic->name.')';
2552 }
2553 $type = ($user->socid ? $langs->trans("External").$company : $langs->trans("Internal"));
2554 $dropdownBody .= '<br><b>'.$langs->trans("Type").':</b> '.$type;
2555 $dropdownBody .= '<br><b>'.$langs->trans("Status").'</b>: '.$user->getLibStatut(0);
2556 $dropdownBody .= '<br>';
2557
2558 $dropdownBody .= '<br><u>'.$langs->trans("Session").'</u>';
2559 $dropdownBody .= '<br><b>'.$langs->trans("IPAddress").'</b>: '.dol_escape_htmltag($_SERVER["REMOTE_ADDR"]);
2560 if (getDolGlobalString('MAIN_MODULE_MULTICOMPANY')) {
2561 $dropdownBody .= '<br><b>'.$langs->trans("ConnectedOnMultiCompany").':</b> '.$conf->entity.' (user entity '.$user->entity.')';
2562 }
2563 $dropdownBody .= '<br><b>'.$langs->trans("AuthenticationMode").':</b> '.$_SESSION["dol_authmode"].(empty($dolibarr_main_demo) ? '' : ' (demo)');
2564 $dropdownBody .= '<br><b>'.$langs->trans("ConnectedSince").':</b> '.dol_print_date($user->datelastlogin, "dayhour", 'tzuser');
2565 $dropdownBody .= '<br><b>'.$langs->trans("PreviousConnexion").':</b> '.dol_print_date($user->datepreviouslogin, "dayhour", 'tzuser');
2566 $dropdownBody .= '<br><b>'.$langs->trans("CurrentTheme").':</b> '.$conf->theme;
2567 $dropdownBody .= '<br><b>'.$langs->trans("CurrentMenuManager").':</b> '.(isset($menumanager) ? $menumanager->name : 'unknown');
2568 $langFlag = picto_from_langcode($langs->getDefaultLang());
2569 $dropdownBody .= '<br><b>'.$langs->trans("CurrentUserLanguage").':</b> '.($langFlag ? $langFlag.' ' : '').$langs->getDefaultLang();
2570
2571 $tz = (int) $_SESSION['dol_tz'] + (int) $_SESSION['dol_dst'];
2572 $dropdownBody .= '<br><b>'.$langs->trans("ClientTZ").':</b> '.($tz ? ($tz >= 0 ? '+' : '').$tz : '');
2573 $dropdownBody .= ' ('.$_SESSION['dol_tz_string'].')';
2574 //$dropdownBody .= ' &nbsp; &nbsp; &nbsp; '.$langs->trans("DaylingSavingTime").': ';
2575 //if ($_SESSION['dol_dst'] > 0) $dropdownBody .= yn(1);
2576 //else $dropdownBody .= yn(0);
2577
2578 $dropdownBody .= '<br><b>'.$langs->trans("Browser").':</b> '.$conf->browser->name.($conf->browser->version ? ' '.$conf->browser->version : '').' <small class="opacitymedium">('.dol_escape_htmltag($_SERVER['HTTP_USER_AGENT']).')</small>';
2579 $dropdownBody .= '<br><b>'.$langs->trans("Layout").':</b> '.$conf->browser->layout;
2580 $dropdownBody .= '<br><b>'.$langs->trans("Screen").':</b> '.$_SESSION['dol_screenwidth'].' x '.$_SESSION['dol_screenheight'];
2581 if ($conf->browser->layout == 'phone') {
2582 $dropdownBody .= '<br><b>'.$langs->trans("Phone").':</b> '.$langs->trans("Yes");
2583 }
2584 if (!empty($_SESSION["disablemodules"])) {
2585 $dropdownBody .= '<br><b>'.$langs->trans("DisabledModules").':</b> <br>'.implode(', ', explode(',', $_SESSION["disablemodules"]));
2586 }
2587 $dropdownBody .= '</div>';
2588
2589 // Execute hook
2590 $parameters = array('user' => $user, 'langs' => $langs);
2591 $result = $hookmanager->executeHooks('printTopRightMenuLoginDropdownBody', $parameters); // Note that $action and $object may have been modified by some hooks
2592 if (is_numeric($result)) {
2593 if ($result == 0) {
2594 $dropdownBody .= $hookmanager->resPrint; // add
2595 } else {
2596 $dropdownBody = $hookmanager->resPrint; // replace
2597 }
2598 }
2599
2600 if (empty($urllogout)) {
2601 $urllogout = DOL_URL_ROOT.'/user/logout.php?token='.newToken();
2602 }
2603
2604 // accesskey is for Windows or Linux: ALT + key for chrome, ALT + SHIFT + KEY for firefox
2605 // accesskey is for Mac: CTRL + key for all browsers
2606 $stringforfirstkey = $langs->trans("KeyboardShortcut");
2607 if ($conf->browser->name == 'chrome') {
2608 $stringforfirstkey .= ' ALT +';
2609 } elseif ($conf->browser->name == 'firefox') {
2610 $stringforfirstkey .= ' ALT + SHIFT +';
2611 } else {
2612 $stringforfirstkey .= ' CTL +';
2613 }
2614
2615 // Defined the links for bottom of card
2616 $profilLink = '<a accesskey="u" href="'.DOL_URL_ROOT.'/user/card.php?id='.$user->id.'" class="button-top-menu-dropdown" title="'.dol_escape_htmltag($langs->trans("YourUserFile").' ('.$stringforfirstkey.' u)').'"><i class="fa fa-user"></i> '.$langs->trans("Card").'</a>';
2617 $urltovirtualcard = '/user/virtualcard.php?id='.((int) $user->id);
2618 $virtuelcardLink = dolButtonToOpenUrlInDialogPopup('publicvirtualcardmenu', $langs->transnoentitiesnoconv("PublicVirtualCardUrl").(is_object($user) ? ' - '.$user->getFullName($langs) : '').' ('.$stringforfirstkey.' v)', img_picto($langs->trans("PublicVirtualCardUrl").' ('.$stringforfirstkey.' v)', 'card', ''), $urltovirtualcard, '', 'button-top-menu-dropdown marginleftonly nohover', "closeTopMenuLoginDropdown()", '', 'v');
2619 $logoutLink = '<a accesskey="l" href="'.$urllogout.'" class="button-top-menu-dropdown" title="'.dol_escape_htmltag($langs->trans("Logout").' ('.$stringforfirstkey.' l)').'"><i class="fa fa-sign-out-alt padingright"></i><span class="hideonsmartphone">'.$langs->trans("Logout").'</span></a>';
2620
2621 $profilName = $user->getFullName($langs).' ('.$user->login.')';
2622 if (!empty($user->admin)) {
2623 $profilName = '<i class="far fa-star classfortooltip" title="'.$langs->trans("Administrator").'" ></i> '.$profilName;
2624 }
2625
2626 // Define version to show
2627 $appli = constant('DOL_APPLICATION_TITLE');
2628 if (getDolGlobalString('MAIN_APPLICATION_TITLE')) {
2629 $appli = getDolGlobalString('MAIN_APPLICATION_TITLE');
2630 if (preg_match('/\d\.\d/', $appli)) {
2631 if (!preg_match('/'.preg_quote(DOL_VERSION).'/', $appli)) {
2632 $appli .= " (".DOL_VERSION.")"; // If new title contains a version that is different than core
2633 }
2634 } else {
2635 $appli .= " ".DOL_VERSION;
2636 }
2637 } else {
2638 $appli .= " ".DOL_VERSION;
2639 }
2640
2641 if (!getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2642 $btnUser = '<!-- div for user link -->
2643 <div id="topmenu-login-dropdown" class="userimg atoplogin dropdown user user-menu inline-block">
2644 <a href="'.DOL_URL_ROOT.'/user/card.php?id='.$user->id.'" class="dropdown-toggle login-dropdown-a valignmiddle" data-toggle="dropdown">
2645 '.$userImage.(empty($user->photo) ? '<!-- no photo so show also the login --><span class="hidden-xs maxwidth200 atoploginusername hideonsmartphone paddingleft valignmiddle small">'.dol_trunc($user->firstname ? $user->firstname : $user->login, 10).'</span>' : '').'
2646 </a>
2647 <div class="dropdown-menu">
2648 <!-- User image -->
2649 <div class="user-header">
2650 '.$userDropDownImage.'
2651 <p>
2652 '.$profilName.'<br>';
2653 if ($user->datelastlogin) {
2654 $title = $langs->trans("ConnectedSince").' : '.dol_print_date($user->datelastlogin, "dayhour", 'tzuser');
2655 if ($user->datepreviouslogin) {
2656 $title .= '<br>'.$langs->trans("PreviousConnexion").' : '.dol_print_date($user->datepreviouslogin, "dayhour", 'tzuser');
2657 }
2658 }
2659 $btnUser .= '<small class="classfortooltip" title="'.dol_escape_htmltag($title).'" ><i class="fa fa-user-clock"></i> '.dol_print_date($user->datelastlogin, "dayhour", 'tzuser').'</small><br>';
2660 if ($user->datepreviouslogin) {
2661 $btnUser .= '<small class="classfortooltip" title="'.dol_escape_htmltag($title).'" ><i class="fa fa-user-clock opacitymedium"></i> '.dol_print_date($user->datepreviouslogin, "dayhour", 'tzuser').'</small><br>';
2662 }
2663
2664 //$btnUser .= '<small class="classfortooltip"><i class="fa fa-cog"></i> '.$langs->trans("Version").' '.$appli.'</small>';
2665 $btnUser .= '
2666 </p>
2667 </div>
2668
2669 <!-- Menu Body user-->
2670 <div class="user-body">'.$dropdownBody.'</div>
2671
2672 <!-- Menu Footer-->
2673 <div class="user-footer">
2674 <div class="pull-left">
2675 '.$profilLink.'
2676 </div>
2677 <div class="pull-left">
2678 '.$virtuelcardLink.'
2679 </div>
2680 <div class="pull-right">
2681 '.$logoutLink.'
2682 </div>
2683 <div class="clearboth"></div>
2684 </div>
2685
2686 </div>
2687 </div>';
2688 } else {
2689 $btnUser = '<!-- div for user link text browser -->
2690 <div id="topmenu-login-dropdown" class="userimg atoplogin dropdown user user-menu inline-block">
2691 <a href="'.DOL_URL_ROOT.'/user/card.php?id='.$user->id.'" class="valignmiddle" alt="'.$langs->trans("MyUserCard").'">
2692 '.$userImage.(empty($user->photo) ? '<span class="hidden-xs maxwidth200 atoploginusername hideonsmartphone paddingleft small">'.dol_trunc($user->firstname ? $user->firstname : $user->login, 10).'</span>' : '').'
2693 </a>
2694 </div>';
2695 }
2696
2697 if (!defined('JS_JQUERY_DISABLE_DROPDOWN') && !empty($conf->use_javascript_ajax)) { // This may be set by some pages that use different jquery version to avoid errors
2698 $btnUser .= '
2699 <!-- Code to show/hide the user drop-down -->
2700 <script>
2701 function closeTopMenuLoginDropdown() {
2702 //console.log("close login dropdown"); // This is call at each click on page, so we disable the log
2703 // Hide the menus.
2704 jQuery("#topmenu-login-dropdown").removeClass("open");
2705 }
2706 jQuery(document).ready(function() {
2707 jQuery(document).on("click", function(event) {
2708 // console.log("Click somewhere on screen");
2709 if (!$(event.target).closest("#topmenu-login-dropdown").length) {
2710 closeTopMenuLoginDropdown();
2711 }
2712 });
2713 ';
2714
2715
2716 //if ($conf->theme != 'md') {
2717 $btnUser .= '
2718 jQuery("#topmenu-login-dropdown .dropdown-toggle").on("click", function(event) {
2719 console.log("Click on #topmenu-login-dropdown .dropdown-toggle");
2720 event.preventDefault();
2721 jQuery("#topmenu-login-dropdown").toggleClass("open");
2722 });
2723
2724 jQuery("#topmenulogincompanyinfo-btn").on("click", function() {
2725 console.log("Click on #topmenulogincompanyinfo-btn");
2726 jQuery("#topmenulogincompanyinfo").slideToggle();
2727 });
2728
2729 jQuery("#topmenuloginmoreinfo-btn").on("click", function() {
2730 console.log("Click on #topmenuloginmoreinfo-btn");
2731 jQuery("#topmenuloginmoreinfo").slideToggle();
2732 });';
2733 //}
2734
2735 $btnUser .= '
2736 });
2737 </script>
2738 ';
2739 }
2740
2741 return $btnUser;
2742}
2743
2750{
2751 global $conf, $langs;
2752
2753 // Button disabled on text browser
2754 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2755 return '';
2756 }
2757
2758 $html = '';
2759
2760 // accesskey is for Windows or Linux: ALT + key for chrome, ALT + SHIFT + KEY for firefox
2761 // accesskey is for Mac: CTRL + key for all browsers
2762 $stringforfirstkey = $langs->trans("KeyboardShortcut");
2763 if ($conf->browser->os === 'macintosh') {
2764 $stringforfirstkey .= ' CTL +';
2765 } else {
2766 if ($conf->browser->name == 'chrome') {
2767 $stringforfirstkey .= ' ALT +';
2768 } elseif ($conf->browser->name == 'firefox') {
2769 $stringforfirstkey .= ' ALT + SHIFT +';
2770 } else {
2771 $stringforfirstkey .= ' CTL +';
2772 }
2773 }
2774
2775 if (!empty($conf->use_javascript_ajax)) {
2776 $html .= '<!-- div for quick add link -->
2777 <div id="topmenu-quickadd-dropdown" class="atoplogin dropdown inline-block">
2778 <a accesskey="a" class="dropdown-toggle login-dropdown-a nofocusvisible" data-toggle="dropdown" href="#" title="'.$langs->trans('QuickAdd').' ('.$stringforfirstkey.' a)"><i class="fa fa-plus-circle"></i></a>
2779 <div class="dropdown-menu">'.printDropdownQuickadd().'</div>
2780 </div>';
2781 if (!defined('JS_JQUERY_DISABLE_DROPDOWN')) { // This may be set by some pages that use different jquery version to avoid errors
2782 $html .= '
2783 <!-- Code to show/hide the user drop-down for the quick add -->
2784 <script>
2785 jQuery(document).ready(function() {
2786 jQuery(document).on("click", function(event) {
2787 if (!$(event.target).closest("#topmenu-quickadd-dropdown").length) {
2788 // Hide the menus.
2789 $("#topmenu-quickadd-dropdown").removeClass("open");
2790 }
2791 });
2792 $("#topmenu-quickadd-dropdown .dropdown-toggle").on("click", function(event) {
2793 console.log("Click on #topmenu-quickadd-dropdown .dropdown-toggle");
2794 openQuickAddDropDown(event);
2795 });
2796
2797 // Key map shortcut
2798 $(document).keydown(function(event){
2799 var ostype = \''.dol_escape_js($conf->browser->os).'\';
2800 if (ostype === "macintosh") {
2801 if ( event.which === 65 && event.ctrlKey ) {
2802 console.log(\'control + a : trigger open quick add dropdown\');
2803 openQuickAddDropDown(event);
2804 }
2805 } else {
2806 if ( event.which === 65 && event.ctrlKey && event.shiftKey ) {
2807 console.log(\'control + shift + a : trigger open quick add dropdown\');
2808 openQuickAddDropDown(event);
2809 }
2810 }
2811 });
2812
2813 var openQuickAddDropDown = function(event) {
2814 event.preventDefault();
2815 $("#topmenu-quickadd-dropdown").toggleClass("open");
2816 //$("#top-quickadd-search-input").focus();
2817 }
2818 });
2819 </script>
2820 ';
2821 }
2822 }
2823
2824 return $html;
2825}
2826
2833{
2834 global $user, $langs, $hookmanager;
2835
2836 $items = array(
2837 'items' => array(
2838 array(
2839 "url" => "/adherents/card.php?action=create&amp;mainmenu=members",
2840 "title" => "MenuNewMember@members",
2841 "name" => "Adherent@members",
2842 "picto" => "object_member",
2843 "activation" => isModEnabled('member') && $user->hasRight("adherent", "write"), // vs hooking
2844 "position" => 5,
2845 ),
2846 array(
2847 "url" => "/societe/card.php?action=create&amp;mainmenu=companies",
2848 "title" => "MenuNewThirdParty@companies",
2849 "name" => "ThirdParty@companies",
2850 "picto" => "object_company",
2851 "activation" => isModEnabled("societe") && $user->hasRight("societe", "write"), // vs hooking
2852 "position" => 10,
2853 ),
2854 array(
2855 "url" => "/contact/card.php?action=create&amp;mainmenu=companies",
2856 "title" => "NewContactAddress@companies",
2857 "name" => "Contact@companies",
2858 "picto" => "object_contact",
2859 "activation" => isModEnabled("societe") && $user->hasRight("societe", "contact", "write"), // vs hooking
2860 "position" => 20,
2861 ),
2862 array(
2863 "url" => "/comm/propal/card.php?action=create&amp;mainmenu=commercial",
2864 "title" => "NewPropal@propal",
2865 "name" => "Proposal@propal",
2866 "picto" => "object_propal",
2867 "activation" => isModEnabled("propal") && $user->hasRight("propal", "write"), // vs hooking
2868 "position" => 30,
2869 ),
2870
2871 array(
2872 "url" => "/commande/card.php?action=create&amp;mainmenu=commercial",
2873 "title" => "NewOrder@orders",
2874 "name" => "Order@orders",
2875 "picto" => "object_order",
2876 "activation" => isModEnabled('order') && $user->hasRight("commande", "write"), // vs hooking
2877 "position" => 40,
2878 ),
2879 array(
2880 "url" => "/compta/facture/card.php?action=create&amp;mainmenu=billing",
2881 "title" => "NewBill@bills",
2882 "name" => "Bill@bills",
2883 "picto" => "object_bill",
2884 "activation" => isModEnabled('invoice') && $user->hasRight("facture", "write"), // vs hooking
2885 "position" => 50,
2886 ),
2887 array(
2888 "url" => "/contrat/card.php?action=create&amp;mainmenu=commercial",
2889 "title" => "NewContractSubscription@contracts",
2890 "name" => "Contract@contracts",
2891 "picto" => "object_contract",
2892 "activation" => isModEnabled('contract') && $user->hasRight("contrat", "write"), // vs hooking
2893 "position" => 60,
2894 ),
2895 array(
2896 "url" => "/supplier_proposal/card.php?action=create&amp;mainmenu=commercial",
2897 "title" => "SupplierProposalNew@supplier_proposal",
2898 "name" => "SupplierProposal@supplier_proposal",
2899 "picto" => "supplier_proposal",
2900 "activation" => isModEnabled('supplier_proposal') && $user->hasRight("supplier_invoice", "write"), // vs hooking
2901 "position" => 70,
2902 ),
2903 array(
2904 "url" => "/fourn/commande/card.php?action=create&amp;mainmenu=commercial",
2905 "title" => "NewSupplierOrderShort@orders",
2906 "name" => "SupplierOrder@orders",
2907 "picto" => "supplier_order",
2908 "activation" => (isModEnabled("fournisseur") && !getDolGlobalString('MAIN_USE_NEW_SUPPLIERMOD') && $user->hasRight("fournisseur", "commande", "write")) || (isModEnabled("supplier_order") && $user->hasRight("supplier_invoice", "write")), // vs hooking
2909 "position" => 80,
2910 ),
2911 array(
2912 "url" => "/fourn/facture/card.php?action=create&amp;mainmenu=billing",
2913 "title" => "NewBill@bills",
2914 "name" => "SupplierBill@bills",
2915 "picto" => "supplier_invoice",
2916 "activation" => (isModEnabled("fournisseur") && !getDolGlobalString('MAIN_USE_NEW_SUPPLIERMOD') && $user->hasRight("fournisseur", "facture", "write")) || (isModEnabled("supplier_invoice") && $user->hasRight("supplier_invoice", "write")), // vs hooking
2917 "position" => 90,
2918 ),
2919 array(
2920 "url" => "/ticket/card.php?action=create&amp;mainmenu=ticket",
2921 "title" => "NewTicket@ticket",
2922 "name" => "Ticket@ticket",
2923 "picto" => "ticket",
2924 "activation" => isModEnabled('ticket') && $user->hasRight("ticket", "write"), // vs hooking
2925 "position" => 100,
2926 ),
2927 array(
2928 "url" => "/fichinter/card.php?action=create&mainmenu=commercial",
2929 "title" => "NewIntervention@interventions",
2930 "name" => "Intervention@interventions",
2931 "picto" => "intervention",
2932 "activation" => isModEnabled('intervention') && $user->hasRight("ficheinter", "creer"), // vs hooking
2933 "position" => 110,
2934 ),
2935 array(
2936 "url" => "/product/card.php?action=create&amp;type=0&amp;mainmenu=products",
2937 "title" => "NewProduct@products",
2938 "name" => "Product@products",
2939 "picto" => "object_product",
2940 "activation" => isModEnabled("product") && $user->hasRight("produit", "write"), // vs hooking
2941 "position" => 400,
2942 ),
2943 array(
2944 "url" => "/product/card.php?action=create&amp;type=1&amp;mainmenu=products",
2945 "title" => "NewService@products",
2946 "name" => "Service@products",
2947 "picto" => "object_service",
2948 "activation" => isModEnabled("service") && $user->hasRight("service", "write"), // vs hooking
2949 "position" => 410,
2950 ),
2951 array(
2952 "url" => "/user/card.php?action=create&amp;type=1&amp;mainmenu=home",
2953 "title" => "AddUser@users",
2954 "name" => "User@users",
2955 "picto" => "user",
2956 "activation" => $user->hasRight("user", "user", "write"), // vs hooking
2957 "position" => 500,
2958 ),
2959 ),
2960 );
2961
2962 $dropDownQuickAddHtml = '';
2963
2964 // Define $dropDownQuickAddHtml
2965 $dropDownQuickAddHtml .= '<div class="quickadd-body dropdown-body">';
2966 $dropDownQuickAddHtml .= '<div class="dropdown-quickadd-list">';
2967
2968 // Allow the $items of the menu to be manipulated by modules
2969 $parameters = array();
2970 $hook_items = $items;
2971 $reshook = $hookmanager->executeHooks('menuDropdownQuickaddItems', $parameters, $hook_items); // Note that $action and $object may have been modified by some hooks
2972 if (is_numeric($reshook) && !empty($hookmanager->resArray) && is_array($hookmanager->resArray)) {
2973 if ($reshook == 0) {
2974 $items['items'] = array_merge($items['items'], $hookmanager->resArray); // add
2975 } else {
2976 $items = $hookmanager->resArray; // replace
2977 }
2978
2979 // Sort menu items by 'position' value
2980 $position = array();
2981 foreach ($items['items'] as $key => $row) {
2982 $position[$key] = $row['position'];
2983 }
2984 $array1_sort_order = SORT_ASC;
2985 array_multisort($position, $array1_sort_order, $items['items']);
2986 }
2987
2988 foreach ($items['items'] as $item) {
2989 if (!$item['activation']) {
2990 continue;
2991 }
2992 $langs->load(explode('@', $item['title'])[1]);
2993 $langs->load(explode('@', $item['name'])[1]);
2994 $dropDownQuickAddHtml .= '
2995 <a class="dropdown-item quickadd-item" href="'.DOL_URL_ROOT.$item['url'].'" title="'.$langs->trans(explode('@', $item['title'])[0]).'">
2996 '. img_picto('', $item['picto'], 'style="width:18px;"') . ' ' . $langs->trans(explode('@', $item['name'])[0]) . '</a>
2997 ';
2998 }
2999
3000 $dropDownQuickAddHtml .= '</div>';
3001 $dropDownQuickAddHtml .= '</div>';
3002
3003 return $dropDownQuickAddHtml;
3004}
3005
3012{
3013 global $langs, $conf, $db, $user;
3014
3015 $html = '';
3016
3017 // Define $bookmarks
3018 if (!isModEnabled('bookmark') || !$user->hasRight('bookmark', 'lire')) {
3019 return $html;
3020 }
3021
3022 // accesskey is for Windows or Linux: ALT + key for chrome, ALT + SHIFT + KEY for firefox
3023 // accesskey is for Mac: CTRL + key for all browsers
3024 $stringforfirstkey = $langs->trans("KeyboardShortcut");
3025 if ($conf->browser->os === 'macintosh') {
3026 $stringforfirstkey .= ' CTL +';
3027 } else {
3028 if ($conf->browser->name == 'chrome') {
3029 $stringforfirstkey .= ' ALT +';
3030 } elseif ($conf->browser->name == 'firefox') {
3031 $stringforfirstkey .= ' ALT + SHIFT +';
3032 } else {
3033 $stringforfirstkey .= ' CTL +';
3034 }
3035 }
3036
3037 if (!defined('JS_JQUERY_DISABLE_DROPDOWN') && !empty($conf->use_javascript_ajax)) { // This may be set by some pages that use different jquery version to avoid errors
3038 include_once DOL_DOCUMENT_ROOT.'/bookmarks/bookmarks.lib.php';
3039 $langs->load("bookmarks");
3040
3041 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
3042 $html .= '<div id="topmenu-bookmark-dropdown" class="dropdown inline-block">';
3043 $html .= printDropdownBookmarksList();
3044 $html .= '</div>';
3045 } else {
3046 $html .= '<!-- div for bookmark link -->
3047 <div id="topmenu-bookmark-dropdown" class="dropdown inline-block">
3048 <a accesskey="b" class="dropdown-toggle login-dropdown-a nofocusvisible" data-toggle="dropdown" href="#" title="'.$langs->trans('Bookmarks').' ('.$stringforfirstkey.' b)"><i class="fa fa-star"></i></a>
3049 <div class="dropdown-menu">
3051 </div>
3052 </div>';
3053
3054 $html .= '
3055 <!-- Code to show/hide the bookmark drop-down -->
3056 <script>
3057 jQuery(document).ready(function() {
3058 jQuery(document).on("click", function(event) {
3059 if (!$(event.target).closest("#topmenu-bookmark-dropdown").length) {
3060 //console.log("close bookmark dropdown - we click outside");
3061 // Hide the menus.
3062 $("#topmenu-bookmark-dropdown").removeClass("open");
3063 }
3064 });
3065
3066 jQuery("#topmenu-bookmark-dropdown .dropdown-toggle").on("click", function(event) {
3067 console.log("Click on #topmenu-bookmark-dropdown .dropdown-toggle");
3068 openBookMarkDropDown(event);
3069 });
3070
3071 // Key map shortcut
3072 jQuery(document).keydown(function(event) {
3073 var ostype = \''.dol_escape_js($conf->browser->os).'\';
3074 if (ostype === "macintosh") {
3075 if ( event.which === 66 && event.ctrlKey ) {
3076 console.log("Click on control + b : trigger open bookmark dropdown");
3077 openBookMarkDropDown(event);
3078 }
3079 } else {
3080 if ( event.which === 66 && event.ctrlKey && event.shiftKey ) {
3081 console.log("Click on control + shift + b : trigger open bookmark dropdown");
3082 openBookMarkDropDown(event);
3083 }
3084 }
3085 });
3086
3087 var openBookMarkDropDown = function(event) {
3088 event.preventDefault();
3089 jQuery("#topmenu-bookmark-dropdown").toggleClass("open");
3090 jQuery("#top-bookmark-search-input").focus();
3091 }
3092
3093 });
3094 </script>
3095 ';
3096 }
3097 }
3098 return $html;
3099}
3100
3106function top_menu_search()
3107{
3108 global $langs, $conf, $db, $user, $hookmanager; // used by htdocs/core/ajax/selectsearchbox.php
3109
3110 $html = '';
3111
3112 $usedbyinclude = 1;
3113 $arrayresult = array();
3114 include DOL_DOCUMENT_ROOT.'/core/ajax/selectsearchbox.php'; // This sets $arrayresult
3115
3116 // accesskey is for Windows or Linux: ALT + key for chrome, ALT + SHIFT + KEY for firefox
3117 // accesskey is for Mac: CTRL + key for all browsers
3118 $stringforfirstkey = $langs->trans("KeyboardShortcut");
3119 if ($conf->browser->name == 'chrome') {
3120 $stringforfirstkey .= ' ALT +';
3121 } elseif ($conf->browser->name == 'firefox') {
3122 $stringforfirstkey .= ' ALT + SHIFT +';
3123 } else {
3124 $stringforfirstkey .= ' CTL +';
3125 }
3126
3127 $searchInput = '<input type="search" name="search_all"'.($stringforfirstkey ? ' title="'.dol_escape_htmltag($stringforfirstkey.' s').'"' : '').' id="top-global-search-input" class="dropdown-search-input search_component_input" placeholder="'.$langs->trans('Search').'" autocomplete="off">';
3128
3129 $defaultAction = '';
3130 $buttonList = '<div class="dropdown-global-search-button-list" >';
3131 // Menu with all searchable items
3132 foreach ($arrayresult as $keyItem => $item) {
3133 if (empty($defaultAction)) {
3134 $defaultAction = $item['url'];
3135 }
3136 $buttonList .= '<button class="dropdown-item global-search-item tdoverflowmax300" data-target="'.dol_escape_htmltag($item['url']).'" >';
3137 $buttonList .= $item['text'];
3138 $buttonList .= '</button>';
3139 }
3140 $buttonList .= '</div>';
3141
3142 $dropDownHtml = '<form role="search" id="top-menu-action-search" name="actionsearch" method="GET" action="'.$defaultAction.'">';
3143
3144 $dropDownHtml .= '
3145 <!-- search input -->
3146 <div class="dropdown-header search-dropdown-header">
3147 ' . $searchInput.'
3148 </div>
3149 ';
3150
3151 $dropDownHtml .= '
3152 <!-- Menu Body search -->
3153 <div class="dropdown-body search-dropdown-body">
3154 '.$buttonList.'
3155 </div>
3156 ';
3157
3158 $dropDownHtml .= '</form>';
3159
3160 // accesskey is for Windows or Linux: ALT + key for chrome, ALT + SHIFT + KEY for firefox
3161 // accesskey is for Mac: CTRL + key for all browsers
3162 $stringforfirstkey = $langs->trans("KeyboardShortcut");
3163 if ($conf->browser->name == 'chrome') {
3164 $stringforfirstkey .= ' ALT +';
3165 } elseif ($conf->browser->name == 'firefox') {
3166 $stringforfirstkey .= ' ALT + SHIFT +';
3167 } else {
3168 $stringforfirstkey .= ' CTL +';
3169 }
3170
3171 $html .= '<!-- div for Global Search -->
3172 <div id="topmenu-global-search-dropdown" class="atoplogin dropdown inline-block">
3173 <a accesskey="s" class="dropdown-toggle login-dropdown-a nofocusvisible" data-toggle="dropdown" href="#" title="'.$langs->trans('Search').' ('.$stringforfirstkey.' s)">
3174 <i class="fa fa-search" aria-hidden="true" ></i>
3175 </a>
3176 <div class="dropdown-menu dropdown-search">
3177 '.$dropDownHtml.'
3178 </div>
3179 </div>';
3180
3181 $html .= '
3182 <!-- Code to show/hide the user drop-down -->
3183 <script>
3184 jQuery(document).ready(function() {
3185
3186 // prevent submitting form on press ENTER
3187 jQuery("#top-global-search-input").keydown(function (e) {
3188 if (e.keyCode == 13 || e.keyCode == 40) {
3189 var inputs = $(this).parents("form").eq(0).find(":button");
3190 if (inputs[inputs.index(this) + 1] != null) {
3191 inputs[inputs.index(this) + 1].focus();
3192 if (e.keyCode == 13){
3193 inputs[inputs.index(this) + 1].trigger("click");
3194 }
3195
3196 }
3197 e.preventDefault();
3198 return false;
3199 }
3200 });
3201
3202 // arrow key nav
3203 jQuery(document).keydown(function(e) {
3204 // Get the focused element:
3205 var $focused = $(":focus");
3206 if($focused.length && $focused.hasClass("global-search-item")){
3207
3208 // UP - move to the previous line
3209 if (e.keyCode == 38) {
3210 e.preventDefault();
3211 $focused.prev().focus();
3212 }
3213
3214 // DOWN - move to the next line
3215 if (e.keyCode == 40) {
3216 e.preventDefault();
3217 $focused.next().focus();
3218 }
3219 }
3220 });
3221
3222
3223 // submit form action
3224 jQuery(".dropdown-global-search-button-list .global-search-item").on("click", function(event) {
3225 jQuery("#top-menu-action-search").attr("action", $(this).data("target"));
3226 jQuery("#top-menu-action-search").submit();
3227 });
3228
3229 // close drop down
3230 jQuery(document).on("click", function(event) {
3231 if (!$(event.target).closest("#topmenu-global-search-dropdown").length) {
3232 console.log("click close search - we click outside");
3233 // Hide the menus.
3234 jQuery("#topmenu-global-search-dropdown").removeClass("open");
3235 }
3236 });
3237
3238 // Open drop down
3239 jQuery("#topmenu-global-search-dropdown .dropdown-toggle").on("click", function(event) {
3240 console.log("click on toggle #topmenu-global-search-dropdown .dropdown-toggle");
3241 openGlobalSearchDropDown();
3242 });
3243
3244 // Key map shortcut
3245 jQuery(document).keydown(function(e){
3246 if ( e.which === 70 && e.ctrlKey && e.shiftKey ) {
3247 console.log(\'control + shift + f : trigger open global-search dropdown\');
3248 openGlobalSearchDropDown();
3249 }
3250 if ( e.which === 70 && e.alKey ) {
3251 console.log(\'alt + f : trigger open global-search dropdown\');
3252 openGlobalSearchDropDown();
3253 }
3254 });
3255
3256 var openGlobalSearchDropDown = function() {
3257 jQuery("#topmenu-global-search-dropdown").toggleClass("open");
3258 jQuery("#top-global-search-input").focus();
3259 }
3260
3261 });
3262 </script>
3263 ';
3264
3265 return $html;
3266}
3267
3282function left_menu($menu_array_before, $helppagename = '', $notused = '', $menu_array_after = array(), $leftmenuwithoutmainarea = 0, $title = '', $acceptdelayedhtml = 0)
3283{
3284 global $user, $conf, $langs, $db, $form;
3285 global $hookmanager, $menumanager;
3286
3287 $searchform = '';
3288
3289 if (!empty($menu_array_before)) {
3290 dol_syslog("Deprecated parameter menu_array_before was used when calling main::left_menu function. Menu entries of module should now be defined into module descriptor and not provided when calling left_menu.", LOG_WARNING);
3291 }
3292
3293 if (empty($conf->dol_hide_leftmenu) && (!defined('NOREQUIREMENU') || !constant('NOREQUIREMENU'))) {
3294 // Instantiate hooks for external modules
3295 $hookmanager->initHooks(array('leftblock'));
3296
3297 print "\n".'<!-- Begin side-nav id-left -->'."\n".'<div class="side-nav"><div id="id-left">'."\n";
3298 print "\n";
3299
3300 if (!is_object($form)) {
3301 $form = new Form($db);
3302 }
3303 $selected = -1;
3304 if (!getDolGlobalString('MAIN_USE_TOP_MENU_SEARCH_DROPDOWN')) {
3305 // Select with select2 is awful on smartphone. TODO Is this still true with select2 v4 ?
3306 if ($conf->browser->layout == 'phone') {
3307 $conf->global->MAIN_USE_OLD_SEARCH_FORM = 1;
3308 }
3309
3310 $usedbyinclude = 1;
3311 $arrayresult = array();
3312 include DOL_DOCUMENT_ROOT.'/core/ajax/selectsearchbox.php'; // This make initHooks('searchform') then set $arrayresult
3313
3314 if ($conf->use_javascript_ajax && !getDolGlobalString('MAIN_USE_OLD_SEARCH_FORM')) {
3315 // accesskey is for Windows or Linux: ALT + key for chrome, ALT + SHIFT + KEY for firefox
3316 // accesskey is for Mac: CTRL + key for all browsers
3317 $stringforfirstkey = $langs->trans("KeyboardShortcut");
3318 if ($conf->browser->name == 'chrome') {
3319 $stringforfirstkey .= ' ALT +';
3320 } elseif ($conf->browser->name == 'firefox') {
3321 $stringforfirstkey .= ' ALT + SHIFT +';
3322 } else {
3323 $stringforfirstkey .= ' CTL +';
3324 }
3325
3326 //$textsearch = $langs->trans("Search");
3327 $textsearch = '<span class="fa fa-search paddingright pictofixedwidth"></span>'.$langs->trans("Search");
3328 $searchform .= $form->selectArrayFilter('searchselectcombo', $arrayresult, $selected, 'accesskey="s"', 1, 0, (!getDolGlobalString('MAIN_SEARCHBOX_CONTENT_LOADED_BEFORE_KEY') ? 1 : 0), 'vmenusearchselectcombo', 1, $textsearch, 1, $stringforfirstkey.' s');
3329 } else {
3330 if (is_array($arrayresult)) {
3331 foreach ($arrayresult as $key => $val) {
3332 $searchform .= printSearchForm($val['url'], $val['url'], $val['label'], 'maxwidth125', 'search_all', (empty($val['shortcut']) ? '' : $val['shortcut']), 'searchleft'.$key, $val['img']);
3333 }
3334 }
3335 }
3336
3337 // Execute hook printSearchForm
3338 $parameters = array('searchform' => $searchform);
3339 $reshook = $hookmanager->executeHooks('printSearchForm', $parameters); // Note that $action and $object may have been modified by some hooks
3340 if (empty($reshook)) {
3341 $searchform .= $hookmanager->resPrint;
3342 } else {
3343 $searchform = $hookmanager->resPrint;
3344 }
3345
3346 // Force special value for $searchform for text browsers or very old search form
3347 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER') || empty($conf->use_javascript_ajax)) {
3348 $urltosearch = DOL_URL_ROOT.'/core/search_page.php?showtitlebefore=1';
3349 $searchform = '<div class="blockvmenuimpair blockvmenusearchphone"><div id="divsearchforms1"><a href="'.$urltosearch.'" accesskey="s" alt="'.dol_escape_htmltag($langs->trans("ShowSearchFields")).'">'.$langs->trans("Search").'...</a></div></div>';
3350 } elseif ($conf->use_javascript_ajax && getDolGlobalString('MAIN_USE_OLD_SEARCH_FORM')) {
3351 $searchform = '<div class="blockvmenuimpair blockvmenusearchphone"><div id="divsearchforms1"><a href="#" alt="'.dol_escape_htmltag($langs->trans("ShowSearchFields")).'">'.$langs->trans("Search").'...</a></div><div id="divsearchforms2" style="display: none">'.$searchform.'</div>';
3352 $searchform .= '<script>
3353 jQuery(document).ready(function () {
3354 jQuery("#divsearchforms1").click(function(){
3355 jQuery("#divsearchforms2").toggle();
3356 });
3357 });
3358 </script>' . "\n";
3359 $searchform .= '</div>';
3360 }
3361
3362 // Key map shortcut
3363 $searchform .= '<script>
3364 jQuery(document).keydown(function(e){
3365 if( e.which === 70 && e.ctrlKey && e.shiftKey ){
3366 console.log(\'control + shift + f : trigger open global-search dropdown\');
3367 openGlobalSearchDropDown();
3368 }
3369 if( (e.which === 83 || e.which === 115) && e.altKey ){
3370 console.log(\'alt + s : trigger open global-search dropdown\');
3371 openGlobalSearchDropDown();
3372 }
3373 });
3374
3375 var openGlobalSearchDropDown = function() {
3376 jQuery("#searchselectcombo").select2(\'open\');
3377 }
3378 </script>';
3379 }
3380
3381 // Left column
3382 print '<!-- Begin left menu -->'."\n";
3383
3384 print '<div class="vmenu"'.(getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER') ? ' alt="Left menu"' : '').'>'."\n\n";
3385
3386 // Show left menu with other forms
3387 $menumanager->menu_array = $menu_array_before;
3388 $menumanager->menu_array_after = $menu_array_after;
3389 if (getDolGlobalInt('MAIN_MENU_LEFT_DROPDOWN')) {
3390 $menumanager->showmenu('leftdropdown', array('searchform' => $searchform)); // output menu_array and menu found in database
3391 } else {
3392 $menumanager->showmenu('left', array('searchform' => $searchform)); // output menu_array and menu found in database
3393 }
3394
3395 // Dolibarr version + help + bug report link
3396 print "\n";
3397 print "<!-- Begin Help Block-->\n";
3398 print '<div id="blockvmenuhelp" class="blockvmenuhelp">'."\n";
3399
3400 // Version
3401 if (getDolGlobalString('MAIN_SHOW_VERSION')) { // Version is already on help picto and on login page.
3402 $doliurl = 'https://www.dolibarr.org';
3403 //local communities
3404 if (preg_match('/fr/i', $langs->defaultlang)) {
3405 $doliurl = 'https://www.dolibarr.fr';
3406 }
3407 if (preg_match('/es/i', $langs->defaultlang)) {
3408 $doliurl = 'https://www.dolibarr.es';
3409 }
3410 if (preg_match('/de/i', $langs->defaultlang)) {
3411 $doliurl = 'https://www.dolibarr.de';
3412 }
3413 if (preg_match('/it/i', $langs->defaultlang)) {
3414 $doliurl = 'https://www.dolibarr.it';
3415 }
3416 if (preg_match('/gr/i', $langs->defaultlang)) {
3417 $doliurl = 'https://www.dolibarr.gr';
3418 }
3419
3420 $appli = constant('DOL_APPLICATION_TITLE');
3421 if (getDolGlobalString('MAIN_APPLICATION_TITLE')) {
3422 $appli = getDolGlobalString('MAIN_APPLICATION_TITLE');
3423 $doliurl = '';
3424 if (preg_match('/\d\.\d/', $appli)) {
3425 if (!preg_match('/'.preg_quote(DOL_VERSION).'/', $appli)) {
3426 $appli .= " (".DOL_VERSION.")"; // If new title contains a version that is different than core
3427 }
3428 } else {
3429 $appli .= " ".DOL_VERSION;
3430 }
3431 } else {
3432 $appli .= " ".DOL_VERSION;
3433 }
3434 print '<div id="blockvmenuhelpapp" class="blockvmenuhelp">';
3435 if ($doliurl) {
3436 print '<a class="help" target="_blank" rel="noopener noreferrer" href="'.$doliurl.'">';
3437 } else {
3438 print '<span class="help">';
3439 }
3440 print $appli;
3441 if ($doliurl) {
3442 print '</a>';
3443 } else {
3444 print '</span>';
3445 }
3446 print '</div>'."\n";
3447 }
3448
3449 // Link to bugtrack
3450 if (getDolGlobalString('MAIN_BUGTRACK_ENABLELINK')) {
3451 require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
3452
3453 if (getDolGlobalString('MAIN_BUGTRACK_ENABLELINK') == 'github') {
3454 $bugbaseurl = 'https://github.com/Dolibarr/dolibarr/issues/new?labels=Bug';
3455 $bugbaseurl .= '&title=';
3456 $bugbaseurl .= urlencode("Bug: ");
3457 $bugbaseurl .= '&body=';
3458 $bugbaseurl .= urlencode("# Instructions\n");
3459 $bugbaseurl .= urlencode("*This is a template to help you report good issues. You may use [Github Markdown](https://help.github.com/articles/getting-started-with-writing-and-formatting-on-github/) syntax to format your issue report.*\n");
3460 $bugbaseurl .= urlencode("*Please:*\n");
3461 $bugbaseurl .= urlencode("- *replace the bracket enclosed texts with meaningful information*\n");
3462 $bugbaseurl .= urlencode("- *remove any unused sub-section*\n");
3463 $bugbaseurl .= urlencode("\n");
3464 $bugbaseurl .= urlencode("\n");
3465 $bugbaseurl .= urlencode("# Bug\n");
3466 $bugbaseurl .= urlencode("[*Short description*]\n");
3467 $bugbaseurl .= urlencode("\n");
3468 $bugbaseurl .= urlencode("## Environment\n");
3469 $bugbaseurl .= urlencode("- **Version**: ".DOL_VERSION."\n");
3470 $bugbaseurl .= urlencode("- **OS**: ".php_uname('s')."\n");
3471 $bugbaseurl .= urlencode("- **Web server**: ".$_SERVER["SERVER_SOFTWARE"]."\n");
3472 $bugbaseurl .= urlencode("- **PHP**: ".php_sapi_name().' '.phpversion()."\n");
3473 $bugbaseurl .= urlencode("- **Database**: ".$db::LABEL.' '.$db->getVersion()."\n");
3474 $bugbaseurl .= urlencode("- **URL(s)**: ".$_SERVER["REQUEST_URI"]."\n");
3475 $bugbaseurl .= urlencode("\n");
3476 $bugbaseurl .= urlencode("## Expected and actual behavior\n");
3477 $bugbaseurl .= urlencode("[*Verbose description*]\n");
3478 $bugbaseurl .= urlencode("\n");
3479 $bugbaseurl .= urlencode("## Steps to reproduce the behavior\n");
3480 $bugbaseurl .= urlencode("[*Verbose description*]\n");
3481 $bugbaseurl .= urlencode("\n");
3482 $bugbaseurl .= urlencode("## [Attached files](https://help.github.com/articles/issue-attachments) (Screenshots, screencasts, dolibarr.log, debugging information…)\n");
3483 $bugbaseurl .= urlencode("[*Files*]\n");
3484 $bugbaseurl .= urlencode("\n");
3485
3486 $bugbaseurl .= urlencode("\n");
3487 $bugbaseurl .= urlencode("## Report\n");
3488 } elseif (getDolGlobalString('MAIN_BUGTRACK_ENABLELINK')) {
3489 $bugbaseurl = getDolGlobalString('MAIN_BUGTRACK_ENABLELINK');
3490 } else {
3491 $bugbaseurl = "";
3492 }
3493
3494 // Execute hook printBugtrackInfo
3495 $parameters = array('bugbaseurl' => $bugbaseurl);
3496 $reshook = $hookmanager->executeHooks('printBugtrackInfo', $parameters); // Note that $action and $object may have been modified by some hooks
3497 if (empty($reshook)) {
3498 $bugbaseurl .= $hookmanager->resPrint;
3499 } else {
3500 $bugbaseurl = $hookmanager->resPrint;
3501 }
3502
3503 print '<div id="blockvmenuhelpbugreport" class="blockvmenuhelp">';
3504 print '<a class="help" target="_blank" rel="noopener noreferrer" href="'.$bugbaseurl.'"><i class="fas fa-bug"></i> '.$langs->trans("FindBug").'</a>';
3505 print '</div>';
3506 }
3507
3508 print "</div>\n";
3509 print "<!-- End Help Block-->\n";
3510 print "\n";
3511
3512 print "</div>\n";
3513 print "<!-- End left menu -->\n";
3514 print "\n";
3515
3516 // Execute hook printLeftBlock
3517 $parameters = array();
3518 $reshook = $hookmanager->executeHooks('printLeftBlock', $parameters); // Note that $action and $object may have been modified by some hooks
3519 print $hookmanager->resPrint;
3520
3521 print '</div></div> <!-- End side-nav id-left -->'; // End div id="side-nav" div id="id-left"
3522 }
3523
3524 print "\n";
3525 print '<!-- Begin right area -->'."\n";
3526
3527 if (empty($leftmenuwithoutmainarea)) {
3528 main_area($title);
3529 }
3530}
3531
3532
3539function main_area($title = '')
3540{
3541 global $conf, $langs, $hookmanager;
3542
3543 if (empty($conf->dol_hide_leftmenu) && !GETPOST('dol_openinpopup')) {
3544 print '<div id="id-right">';
3545 }
3546
3547 print "\n";
3548
3549 print '<!-- Begin div class="fiche" -->'."\n".'<div class="fiche">'."\n";
3550
3551 $hookmanager->initHooks(array('main'));
3552 $parameters = array();
3553 $reshook = $hookmanager->executeHooks('printMainArea', $parameters); // Note that $action and $object may have been modified by some hooks
3554 print $hookmanager->resPrint;
3555
3556 if (getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')) {
3557 print info_admin($langs->trans("WarningYouAreInMaintenanceMode", getDolGlobalString('MAIN_ONLY_LOGIN_ALLOWED')), 0, 0, 1, 'warning maintenancemode');
3558 }
3559
3560 // Permit to add user company information on each printed document by setting SHOW_SOCINFO_ON_PRINT
3561 if (getDolGlobalString('SHOW_SOCINFO_ON_PRINT') && GETPOST('optioncss', 'aZ09') == 'print' && empty(GETPOST('disable_show_socinfo_on_print', 'aZ09'))) {
3562 $parameters = array();
3563 $reshook = $hookmanager->executeHooks('showSocinfoOnPrint', $parameters);
3564 if (empty($reshook)) {
3565 print '<!-- Begin show mysoc info header -->'."\n";
3566 print '<div id="mysoc-info-header">'."\n";
3567 print '<table class="centpercent div-table-responsive">'."\n";
3568 print '<tbody>';
3569 print '<tr><td rowspan="0" class="width20p">';
3570 if (getDolGlobalString('MAIN_SHOW_LOGO') && !getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER') && getDolGlobalString('MAIN_INFO_SOCIETE_LOGO')) {
3571 print '<img id="mysoc-info-header-logo" style="max-width:100%" alt="" src="'.DOL_URL_ROOT.'/viewimage.php?cache=1&modulepart=mycompany&file='.urlencode('logos/'.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_LOGO'))).'">';
3572 }
3573 print '</td><td rowspan="0" class="width50p"></td></tr>'."\n";
3574 print '<tr><td class="titre bold">'.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_NOM')).'</td></tr>'."\n";
3575 print '<tr><td>'.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_ADDRESS')).'<br>'.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_ZIP')).' '.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_TOWN')).'</td></tr>'."\n";
3576 if (getDolGlobalString('MAIN_INFO_SOCIETE_TEL')) {
3577 print '<tr><td style="padding-left: 1em" class="small">'.$langs->trans("Phone").' : '.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_TEL')).'</td></tr>';
3578 }
3579 if (getDolGlobalString('MAIN_INFO_SOCIETE_MAIL')) {
3580 print '<tr><td style="padding-left: 1em" class="small">'.$langs->trans("Email").' : '.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_MAIL')).'</td></tr>';
3581 }
3582 if (getDolGlobalString('MAIN_INFO_SOCIETE_WEB')) {
3583 print '<tr><td style="padding-left: 1em" class="small">'.$langs->trans("Web").' : '.dol_escape_htmltag(getDolGlobalString('MAIN_INFO_SOCIETE_WEB')).'</td></tr>';
3584 }
3585 print '</tbody>';
3586 print '</table>'."\n";
3587 print '</div>'."\n";
3588 print '<!-- End show mysoc info header -->'."\n";
3589 }
3590 }
3591}
3592
3593
3601function getHelpParamFor($helppagename, $langs)
3602{
3603 $helpbaseurl = '';
3604 $helppage = '';
3605 $mode = '';
3606
3607 if (preg_match('/^http/i', $helppagename)) {
3608 // If complete URL
3609 $helpbaseurl = '%s';
3610 $helppage = $helppagename;
3611 $mode = 'local';
3612 } else {
3613 // If WIKI URL
3614 $reg = array();
3615 if (preg_match('/^es/i', $langs->defaultlang)) {
3616 $helpbaseurl = 'http://wiki.dolibarr.org/index.php/%s';
3617 if (preg_match('/ES:([^|]+)/i', $helppagename, $reg)) {
3618 $helppage = $reg[1];
3619 }
3620 }
3621 if (preg_match('/^fr/i', $langs->defaultlang)) {
3622 $helpbaseurl = 'http://wiki.dolibarr.org/index.php/%s';
3623 if (preg_match('/FR:([^|]+)/i', $helppagename, $reg)) {
3624 $helppage = $reg[1];
3625 }
3626 }
3627 if (preg_match('/^de/i', $langs->defaultlang)) {
3628 $helpbaseurl = 'http://wiki.dolibarr.org/index.php/%s';
3629 if (preg_match('/DE:([^|]+)/i', $helppagename, $reg)) {
3630 $helppage = $reg[1];
3631 }
3632 }
3633 if (empty($helppage)) { // If help page not already found
3634 $helpbaseurl = 'http://wiki.dolibarr.org/index.php/%s';
3635 if (preg_match('/EN:([^|]+)/i', $helppagename, $reg)) {
3636 $helppage = $reg[1];
3637 }
3638 }
3639 $mode = 'wiki';
3640 }
3641 return array('helpbaseurl' => $helpbaseurl, 'helppage' => $helppage, 'mode' => $mode);
3642}
3643
3644
3661function printSearchForm($urlaction, $urlobject, $title, $htmlmorecss, $htmlinputname, $accesskey = '', $prefhtmlinputname = '', $img = '', $showtitlebefore = 0, $autofocus = 0)
3662{
3663 global $langs, $user;
3664
3665 $ret = '';
3666 $ret .= '<form action="'.$urlaction.'" method="post" class="searchform nowraponall tagtr">';
3667 $ret .= '<input type="hidden" name="token" value="'.newToken().'">';
3668 $ret .= '<input type="hidden" name="savelogin" value="'.dol_escape_htmltag($user->login).'">';
3669 if ($showtitlebefore) {
3670 $ret .= '<div class="tagtd left">'.$title.'</div> ';
3671 }
3672 $ret .= '<div class="tagtd">';
3673 $ret .= img_picto('', $img, '', false, 0, 0, '', 'paddingright width20');
3674 $ret .= '<input type="text" class="flat '.$htmlmorecss.'"';
3675 $ret .= ' style="background-repeat: no-repeat; background-position: 3px;"';
3676 $ret .= ($accesskey ? ' accesskey="'.$accesskey.'"' : '');
3677 $ret .= ' placeholder="'.strip_tags($title).'"';
3678 $ret .= ($autofocus ? ' autofocus' : '');
3679 $ret .= ' name="'.$htmlinputname.'" id="'.$prefhtmlinputname.$htmlinputname.'" />';
3680 $ret .= '<button type="submit" class="button bordertransp" style="padding-top: 4px; padding-bottom: 4px; padding-left: 6px; padding-right: 6px">';
3681 $ret .= '<span class="fa fa-search"></span>';
3682 $ret .= '</button>';
3683 $ret .= '</div>';
3684 $ret .= "</form>\n";
3685 return $ret;
3686}
3687
3688
3689if (!function_exists("llxFooter")) {
3700 function llxFooter($comment = '', $zone = 'private', $disabledoutputofmessages = 0)
3701 {
3702 global $conf, $db, $langs, $user, $mysoc, $object, $hookmanager, $action;
3703 global $delayedhtmlcontent;
3704 global $contextpage, $page, $limit, $mode;
3705 global $dolibarr_distrib;
3706
3707 $ext = 'layout='.urlencode($conf->browser->layout).'&version='.urlencode(DOL_VERSION);
3708
3709 // Hook to add more things on all pages within fiche DIV
3710 $llxfooter = '';
3711 $parameters = array();
3712 $reshook = $hookmanager->executeHooks('llxFooter', $parameters, $object, $action); // Note that $action and $object may have been modified by hook
3713 if (empty($reshook)) {
3714 $llxfooter .= $hookmanager->resPrint;
3715 } elseif ($reshook > 0) {
3716 $llxfooter = $hookmanager->resPrint;
3717 }
3718 if ($llxfooter) {
3719 print $llxfooter;
3720 }
3721
3722 // Global html output events ($mesgs, $errors, $warnings)
3723 dol_htmloutput_events($disabledoutputofmessages);
3724
3725 // Code for search criteria persistence.
3726 // $user->lastsearch_values was set by the GETPOST when form field search_xxx exists
3727 if (is_object($user) && !empty($user->lastsearch_values_tmp) && is_array($user->lastsearch_values_tmp)) {
3728 // Clean and save data
3729 foreach ($user->lastsearch_values_tmp as $key => $val) {
3730 unset($_SESSION['lastsearch_values_tmp_'.$key]); // Clean array to rebuild it just after
3731 if (count($val) && empty($_POST['button_removefilter']) && empty($_POST['button_removefilter_x'])) {
3732 if (empty($val['sortfield'])) {
3733 unset($val['sortfield']);
3734 }
3735 if (empty($val['sortorder'])) {
3736 unset($val['sortorder']);
3737 }
3738 dol_syslog('Save lastsearch_values_tmp_'.$key.'='.json_encode($val, 0)." (systematic recording of last search criteria)");
3739 $_SESSION['lastsearch_values_tmp_'.$key] = json_encode($val);
3740 unset($_SESSION['lastsearch_values_'.$key]);
3741 }
3742 }
3743 }
3744
3745
3746 $relativepathstring = $_SERVER["PHP_SELF"];
3747 // Clean $relativepathstring
3748 if (constant('DOL_URL_ROOT')) {
3749 $relativepathstring = preg_replace('/^'.preg_quote(constant('DOL_URL_ROOT'), '/').'/', '', $relativepathstring);
3750 }
3751 $relativepathstring = preg_replace('/^\//', '', $relativepathstring);
3752 $relativepathstring = preg_replace('/^custom\//', '', $relativepathstring);
3753 if (preg_match('/list\.php$/', $relativepathstring)) {
3754 unset($_SESSION['lastsearch_contextpage_tmp_'.$relativepathstring]);
3755 unset($_SESSION['lastsearch_page_tmp_'.$relativepathstring]);
3756 unset($_SESSION['lastsearch_limit_tmp_'.$relativepathstring]);
3757 unset($_SESSION['lastsearch_mode_tmp_'.$relativepathstring]);
3758
3759 if (!empty($contextpage)) {
3760 $_SESSION['lastsearch_contextpage_tmp_'.$relativepathstring] = $contextpage;
3761 }
3762 if (!empty($page) && $page > 0) {
3763 $_SESSION['lastsearch_page_tmp_'.$relativepathstring] = $page;
3764 }
3765 if (!empty($limit) && $limit != $conf->liste_limit) {
3766 $_SESSION['lastsearch_limit_tmp_'.$relativepathstring] = $limit;
3767 }
3768 if (!empty($mode)) {
3769 $_SESSION['lastsearch_mode_tmp_'.$relativepathstring] = $mode;
3770 }
3771
3772 unset($_SESSION['lastsearch_contextpage_'.$relativepathstring]);
3773 unset($_SESSION['lastsearch_page_'.$relativepathstring]);
3774 unset($_SESSION['lastsearch_limit_'.$relativepathstring]);
3775 unset($_SESSION['lastsearch_mode_'.$relativepathstring]);
3776 }
3777
3778 // Core error message
3779 if (getDolGlobalString('MAIN_CORE_ERROR')) {
3780 // Ajax version
3781 if ($conf->use_javascript_ajax) {
3782 $title = img_warning().' '.$langs->trans('CoreErrorTitle');
3783 print ajax_dialog($title, $langs->trans('CoreErrorMessage'));
3784 } else {
3785 // html version
3786 $msg = img_warning().' '.$langs->trans('CoreErrorMessage');
3787 print '<div class="error">'.$msg.'</div>';
3788 }
3789
3790 //define("MAIN_CORE_ERROR",0); // Constant was defined and we can't change value of a constant
3791 }
3792
3793 print "\n\n";
3794
3795 print '</div> <!-- End div class="fiche" -->'."\n"; // End div fiche
3796
3797 if (empty($conf->dol_hide_leftmenu) && !GETPOST('dol_openinpopup')) {
3798 print '</div> <!-- End div id-right -->'."\n"; // End div id-right
3799 }
3800
3801 if (empty($conf->dol_hide_leftmenu) && empty($conf->dol_use_jmobile)) {
3802 print '</div> <!-- End div id-container -->'."\n"; // End div container
3803 }
3804
3805 print "\n";
3806 if ($comment) {
3807 print '<!-- '.$comment.' -->'."\n";
3808 }
3809
3810 printCommonFooter($zone);
3811
3812 if (!empty($delayedhtmlcontent)) {
3813 print $delayedhtmlcontent;
3814 }
3815
3816 if (!empty($conf->use_javascript_ajax)) {
3817 print "\n".'<!-- Includes JS Footer of Dolibarr -->'."\n";
3818 print '<script src="'.DOL_URL_ROOT.'/core/js/lib_foot.js.php?lang='.$langs->defaultlang.($ext ? '&'.$ext : '').'"></script>'."\n";
3819 }
3820
3821 // Wrapper to add log when clicking on download or preview
3822 if (isModEnabled('blockedlog') && is_object($object) && !empty($object->id) && $object->id > 0) {
3823 if (in_array($object->element, array('facture')) && $object->statut > 0) { // Restrict for the moment to element 'facture'
3824 print "\n<!-- JS CODE TO ENABLE log when making a download or a preview of a document -->\n";
3825 ?>
3826 <script>
3827 jQuery(document).ready(function () {
3828 $('a.documentpreview').click(function() {
3829 console.log("Call /blockedlog/ajax/block-add on a.documentpreview");
3830 $.post('<?php echo DOL_URL_ROOT."/blockedlog/ajax/block-add.php" ?>'
3831 , {
3832 id:<?php echo $object->id; ?>
3833 , element:'<?php echo dol_escape_js($object->element) ?>'
3834 , action:'DOC_PREVIEW'
3835 , token: '<?php echo currentToken(); ?>'
3836 }
3837 );
3838 });
3839 $('a.documentdownload').click(function() {
3840 console.log("Call /blockedlog/ajax/block-add a.documentdownload");
3841 $.post('<?php echo DOL_URL_ROOT."/blockedlog/ajax/block-add.php" ?>'
3842 , {
3843 id:<?php echo $object->id; ?>
3844 , element:'<?php echo dol_escape_js($object->element) ?>'
3845 , action:'DOC_DOWNLOAD'
3846 , token: '<?php echo currentToken(); ?>'
3847 }
3848 );
3849 });
3850 });
3851 </script>
3852 <?php
3853 }
3854 }
3855
3856 // A div for the address popup
3857 print "\n<!-- A div to allow dialog popup by jQuery('#dialogforpopup').dialog() -->\n";
3858 print '<div id="dialogforpopup" style="display: none;"></div>'."\n";
3859
3860 // Add code for the asynchronous anonymous first ping (for telemetry)
3861 // You can use &forceping=1 in parameters to force the ping if the ping was already sent.
3862 $forceping = GETPOST('forceping', 'alpha');
3863 if (($_SERVER["PHP_SELF"] == DOL_URL_ROOT.'/index.php') || $forceping) {
3864 //print '<!-- instance_unique_id='.$conf->file->instance_unique_id.' MAIN_FIRST_PING_OK_ID='.$conf->global->MAIN_FIRST_PING_OK_ID.' -->';
3865 $hash_unique_id = dol_hash('dolibarr'.$conf->file->instance_unique_id, 'sha256'); // Note: if the global salt changes, this hash changes too so ping may be counted twice. We don't mind. It is for statistics purpose only.
3866
3867 if (!getDolGlobalString('MAIN_FIRST_PING_OK_DATE')
3868 || (!empty($conf->file->instance_unique_id) && ($hash_unique_id != $conf->global->MAIN_FIRST_PING_OK_ID) && (getDolGlobalString('MAIN_FIRST_PING_OK_ID') != 'disabled'))
3869 || $forceping) {
3870 // No ping done if we are into an alpha version
3871 if (strpos('alpha', DOL_VERSION) > 0 && !$forceping) {
3872 print "\n<!-- NO JS CODE TO ENABLE the anonymous Ping. It is an alpha version -->\n";
3873 } elseif (empty($_COOKIE['DOLINSTALLNOPING_'.$hash_unique_id]) || $forceping) { // Cookie is set when we uncheck the checkbox in the installation wizard.
3874 // MAIN_LAST_PING_KO_DATE
3875 // Disable ping if MAIN_LAST_PING_KO_DATE is set and is recent (this month)
3876 if (getDolGlobalString('MAIN_LAST_PING_KO_DATE') && substr($conf->global->MAIN_LAST_PING_KO_DATE, 0, 6) == dol_print_date(dol_now(), '%Y%m') && !$forceping) {
3877 print "\n<!-- NO JS CODE TO ENABLE the anonymous Ping. An error already occurred this month, we will try later. -->\n";
3878 } else {
3879 include_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
3880
3881 print "\n".'<!-- Includes JS for Ping of Dolibarr forceping='.$forceping.' MAIN_FIRST_PING_OK_DATE='.getDolGlobalString("MAIN_FIRST_PING_OK_DATE").' MAIN_FIRST_PING_OK_ID='.getDolGlobalString("MAIN_FIRST_PING_OK_ID").' MAIN_LAST_PING_KO_DATE='.getDolGlobalString("MAIN_LAST_PING_KO_DATE").' -->'."\n";
3882 print "\n<!-- JS CODE TO ENABLE the anonymous Ping -->\n";
3883 $url_for_ping = getDolGlobalString('MAIN_URL_FOR_PING', "https://ping.dolibarr.org/");
3884 // Try to guess the distrib used
3885 $distrib = 'standard';
3886 if ($_SERVER["SERVER_ADMIN"] == 'doliwamp@localhost') {
3887 $distrib = 'doliwamp';
3888 }
3889 if (!empty($dolibarr_distrib)) {
3890 $distrib = $dolibarr_distrib;
3891 }
3892 ?>
3893 <script>
3894 jQuery(document).ready(function (tmp) {
3895 console.log("Try Ping with hash_unique_id is dol_hash('dolibarr'+instance_unique_id, 'sha256')");
3896 $.ajax({
3897 method: "POST",
3898 url: "<?php echo $url_for_ping ?>",
3899 timeout: 500, // timeout milliseconds
3900 cache: false,
3901 data: {
3902 hash_algo: 'dol_hash-sha256',
3903 hash_unique_id: '<?php echo dol_escape_js($hash_unique_id); ?>',
3904 action: 'dolibarrping',
3905 version: '<?php echo (float) DOL_VERSION; ?>',
3906 entity: '<?php echo (int) $conf->entity; ?>',
3907 dbtype: '<?php echo dol_escape_js($db->type); ?>',
3908 country_code: '<?php echo $mysoc->country_code ? dol_escape_js($mysoc->country_code) : 'unknown'; ?>',
3909 php_version: '<?php echo dol_escape_js(phpversion()); ?>',
3910 os_version: '<?php echo dol_escape_js(version_os('smr')); ?>',
3911 db_version: '<?php echo dol_escape_js(version_db()); ?>',
3912 distrib: '<?php echo $distrib ? dol_escape_js($distrib) : 'unknown'; ?>',
3913 token: 'notrequired'
3914 },
3915 success: function (data, status, xhr) { // success callback function (data contains body of response)
3916 console.log("Ping ok");
3917 $.ajax({
3918 method: 'GET',
3919 url: '<?php echo DOL_URL_ROOT.'/core/ajax/pingresult.php'; ?>',
3920 timeout: 500, // timeout milliseconds
3921 cache: false,
3922 data: { hash_algo: 'dol_hash-sha256', hash_unique_id: '<?php echo dol_escape_js($hash_unique_id); ?>', action: 'firstpingok', token: '<?php echo currentToken(); ?>' }, // for update
3923 });
3924 },
3925 error: function (data,status,xhr) { // error callback function
3926 console.log("Ping ko: " + data);
3927 $.ajax({
3928 method: 'GET',
3929 url: '<?php echo DOL_URL_ROOT.'/core/ajax/pingresult.php'; ?>',
3930 timeout: 500, // timeout milliseconds
3931 cache: false,
3932 data: { hash_algo: 'dol_hash-sha256', hash_unique_id: '<?php echo dol_escape_js($hash_unique_id); ?>', action: 'firstpingko', token: '<?php echo currentToken(); ?>' },
3933 });
3934 }
3935 });
3936 });
3937 </script>
3938 <?php
3939 }
3940 } else {
3941 $now = dol_now();
3942 print "\n<!-- NO JS CODE TO ENABLE the anonymous Ping. It was disabled -->\n";
3943 include_once DOL_DOCUMENT_ROOT.'/core/lib/admin.lib.php';
3944 dolibarr_set_const($db, 'MAIN_FIRST_PING_OK_DATE', dol_print_date($now, 'dayhourlog', 'gmt'), 'chaine', 0, '', $conf->entity);
3945 dolibarr_set_const($db, 'MAIN_FIRST_PING_OK_ID', 'disabled', 'chaine', 0, '', $conf->entity);
3946 }
3947 }
3948 }
3949
3950 $parameters = array();
3951 $reshook = $hookmanager->executeHooks('beforeBodyClose', $parameters); // Note that $action and $object may have been modified by some hooks
3952 if ($reshook > 0) {
3953 print $hookmanager->resPrint;
3954 }
3955
3956 print "</body>\n";
3957 print "</html>\n";
3958 }
3959}
if( $user->socid > 0) if(! $user->hasRight('accounting', 'chartofaccount')) $object
Definition card.php:58
print $langs trans("AuditedSecurityEvents").'</strong >< span class="opacitymedium"></span >< br > status
Or an array listing all the potential status of the object: array: int of the status => translated la...
Definition security.php:626
dolibarr_set_const($db, $name, $value, $type='chaine', $visible=0, $note='', $entity=1)
Insert a parameter (key,value) into database (delete old key then insert it again).
versioncompare($versionarray1, $versionarray2)
Compare 2 versions (stored into 2 arrays).
Definition admin.lib.php:69
ajax_dialog($title, $message, $w=350, $h=150)
Show an ajax dialog.
Definition ajax.lib.php:413
if(!defined('NOREQUIRESOC')) if(!defined( 'NOREQUIRETRAN')) if(!defined('NOTOKENRENEWAL')) if(!defined( 'NOREQUIREMENU')) if(!defined('NOREQUIREHTML')) if(!defined( 'NOREQUIREAJAX')) llxHeader($head='', $title='', $help_url='', $target='', $disablejs=0, $disablehead=0, $arrayofjs='', $arrayofcss='', $morequerystring='', $morecssonbody='', $replacemainareaby='', $disablenofollow=0, $disablenoindex=0)
Empty header.
Definition wrapper.php:70
printDropdownBookmarksList()
Add area with bookmarks in top menu.
DolibarrDebugBar class.
Definition DebugBar.php:47
Class to manage generation of HTML components Only common components must be here.
static showphoto($modulepart, $object, $width=100, $height=0, $caneditfield=0, $cssclass='photowithmargin', $imagesize='', $addlinktofullsize=1, $cache=0, $forcecapture='', $noexternsourceoverwrite=0)
Return HTML code to output a photo.
Class to manage hooks.
Class to manage left menus.
Class to manage menu Auguria.
loadMenu($forcemainmenu='', $forceleftmenu='')
Load this->tabMenu.
Class to manage third parties objects (customers, suppliers, prospects...)
Class to manage translations.
dol_stringtotime($string, $gm=1)
Convert a string date into a GM Timestamps date Warning: YYYY-MM-DDTHH:MM:SS+02:00 (RFC3339) is not s...
Definition date.lib.php:426
llxFooter()
Footer empty.
Definition document.php:107
if(!defined( 'DOL_APPLICATION_TITLE')) if(!defined('DOL_VERSION')) if(!defined( 'EURO')) if(!defined('LOG_DEBUG')) if(defined( 'DOL_INC_FOR_VERSION_ERROR')) dol_session_start()
Replace session_start()
picto_from_langcode($codelang, $moreatt='', $notitlealt=0)
Return img flag of country for a language code or country code.
img_warning($titlealt='default', $moreatt='', $morecss='pictowarning')
Show warning logo.
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.
getDolUserInt($key, $default=0, $tmpuser=null)
Return Dolibarr user constant int value.
yn($yesno, $case=1, $color=0)
Return yes or no in current language.
dolButtonToOpenUrlInDialogPopup($name, $label, $buttonstring, $url, $disabled='', $morecss='classlink button bordertransp', $jsonopen='', $backtopagejsfields='', $accesskey='')
Return HTML code to output a button to open a dialog popup box.
isHTTPS()
Return if we are using a HTTPS connection Check HTTPS (no way to be modified by user but may be empty...
printCommonFooter($zone='private')
Print common footer : conf->global->MAIN_HTML_FOOTER js for switch of menu hider js for conf->global-...
getDolUserString($key, $default='', $tmpuser=null)
Return Dolibarr user constant string value.
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).
if(!function_exists( 'dol_getprefix')) dol_include_once($relpath, $classname='')
Make an include_once using default root and alternate root if it fails.
dol_htmlentities($string, $flags=ENT_QUOTES|ENT_SUBSTITUTE, $encoding='UTF-8', $double_encode=false)
Replace htmlentities functions.
getBrowserInfo($user_agent)
Return information about user browser.
dol_htmloutput_events($disabledoutputofmessages=0)
Print formatted messages to output (Used to show messages on html output).
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
setEventMessages($mesg, $mesgs, $style='mesgs', $messagekey='', $noduplicate=0)
Set event messages in dol_events session object.
dol_print_profids($profID, $profIDtype, $countrycode='', $addcpButton=1)
Format professional IDs according to their country.
dol_buildpath($path, $type=0, $returnemptyifnotfound=0)
Return path of url or filesystem.
dol_print_error($db=null, $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
getDolGlobalString($key, $default='')
Return dolibarr global constant string value.
info_admin($text, $infoonimgalt=0, $nodiv=0, $admin='1', $morecss='hideonsmartphone', $textfordropdown='', $picto='')
Show information in HTML for admin users or standard users.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
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...
ui state ui widget content ui state ui widget header ui state a ui button
0 = Do not include form tag and submit button -1 = Do not include form tag but include submit button
conf($dolibarr_main_document_root)
Load conf file (file must exists)
Definition inc.php:420
if(!empty( $_SERVER[ 'MAIN_SHOW_TUNING_INFO'])) getArrayOfEmoji()
Return array of Emojis.
Definition main.inc.php:62
top_menu_quickadd()
Build the tooltip on top menu quick add.
top_htmlhead($head, $title='', $disablejs=0, $disablehead=0, $arrayofjs=array(), $arrayofcss=array(), $disableforlogin=0, $disablenofollow=0, $disablenoindex=0)
Output html header of a page.
top_menu_user($hideloginname=0, $urllogout='')
Build the tooltip on user login.
testSqlAndScriptInject($val, $type)
Security: WAF layer for SQL Injection and XSS Injection (scripts) protection (Filters on GET,...
Definition main.inc.php:123
left_menu($menu_array_before, $helppagename='', $notused='', $menu_array_after=array(), $leftmenuwithoutmainarea=0, $title='', $acceptdelayedhtml=0)
Show left menu bar.
if(!defined( 'NOREQUIREMENU')) if(!empty(GETPOST('seteventmessages', 'alpha'))) if(!function_exists("llxHeader")) top_httphead($contenttype='text/html', $forcenocache=0)
Show HTTP header.
main_area($title='')
Begin main area.
realCharForNumericEntities($matches)
Return the real char for a numeric entities.
Definition main.inc.php:86
getHelpParamFor($helppagename, $langs)
Return helpbaseurl, helppage and mode.
printSearchForm($urlaction, $urlobject, $title, $htmlmorecss, $htmlinputname, $accesskey='', $prefhtmlinputname='', $img='', $showtitlebefore=0, $autofocus=0)
Show a search area.
top_menu($head, $title='', $target='', $disablejs=0, $disablehead=0, $arrayofjs=array(), $arrayofcss=array(), $morequerystring='', $helppagename='')
Show an HTML header + a BODY + The top menu bar.
top_menu_search()
Build the tooltip on top menu tsearch.
printDropdownQuickadd()
Generate list of quickadd items.
analyseVarsForSqlAndScriptsInjection(&$var, $type, $stopcode=1)
Return true if security check on parameters are OK, false otherwise.
Definition main.inc.php:261
top_menu_bookmark()
Build the tooltip on top menu bookmark.
$conf db user
Active Directory does not allow anonymous connections.
Definition repair.php:143
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition repair.php:139
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition repair.php:142
checkLoginPassEntity($usertotest, $passwordtotest, $entitytotest, $authmode, $context='')
Return a login if login/pass was successful.
dol_hash($chain, $type='0', $nosalt=0)
Returns a hash (non reversible encryption) of a string.
accessforbidden($message='', $printheader=1, $printfooter=1, $showonlymessage=0, $params=null)
Show a message to say access is forbidden and stop program.