dolibarr 24.0.0-beta
compare.php
1<?php
2/* Copyright (C) 2017 Laurent Destailleur <eldy@users.sourceforge.net>
3 * Copyright (C) 2021 Gauthier VERDOL <gauthier.verdol@atm-consulting.fr>
4 * Copyright (C) 2021 Greg Rastklan <greg.rastklan@atm-consulting.fr>
5 * Copyright (C) 2021 Jean-Pascal BOUDET <jean-pascal.boudet@atm-consulting.fr>
6 * Copyright (C) 2021 Grégory BLEMAND <gregory.blemand@atm-consulting.fr>
7 * Copyright (C) 2024-2025 Frédéric France <frederic.france@free.fr>
8 * Copyright (C) 2024-2025 MDW <mdeweerd@users.noreply.github.com>
9 * Copyright (C) 2024 Alexandre Spangaro <alexandre@inovea-conseil.com>
10 *
11 * This program is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 3 of the License, or
14 * (at your option) any later version.
15 *
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with this program. If not, see <https://www.gnu.org/licenses/>.
23
24 * \file htdocs/hrm/compare.php
25 * \ingroup hrm
26 * \brief This file compares skills of user groups
27 *
28 * Displays a table in three parts.
29 * 1- the left part displays the list of users for the selected group 1.
30 *
31 * 2- the central part displays the skills. Display of the maximum score for this group and the number of occurrences.
32 *
33 * 3- the right part displays the members of group 2 or the job to be compared
34 */
35
36
37// Load Dolibarr environment
38require_once '../main.inc.php';
47require_once DOL_DOCUMENT_ROOT . '/core/lib/functions.lib.php';
48require_once DOL_DOCUMENT_ROOT . '/core/lib/functions2.lib.php';
49require_once DOL_DOCUMENT_ROOT . '/hrm/class/skill.class.php';
50require_once DOL_DOCUMENT_ROOT . '/hrm/class/job.class.php';
51require_once DOL_DOCUMENT_ROOT . '/hrm/class/evaluation.class.php';
52require_once DOL_DOCUMENT_ROOT . '/hrm/class/position.class.php';
53require_once DOL_DOCUMENT_ROOT . '/hrm/lib/hrm.lib.php';
54
55$action = GETPOST('action');
56
57// Load translation files required by the page
58$langs->load('hrm');
59
60$job = new Job($db);
61
62// Permissions
63$permissiontoread = $user->hasRight('hrm', 'evaluation', 'read') || $user->hasRight('hrm', 'compare_advance', 'read');
64$permissiontoadd = 0;
65
66if (!isModEnabled('hrm')) {
68}
69if (!$permissiontoread || ($action === 'create' && !$permissiontoadd)) {
71}
72
73
74/*
75 * View
76 */
77
78$css = array('/hrm/css/style.css');
79
80llxHeader('', $langs->trans('SkillComparison'), '', '', 0, 0, '', $css);
81
82$head = array();
83
84$h = 0;
85$head[$h][0] = DOL_URL_ROOT.'/hrm/compare.php';
86$head[$h][1] = $langs->trans("SkillComparison");
87$head[$h][2] = 'compare';
88
89print dol_get_fiche_head($head, 'compare', '', 1);
90
91?>
92 <script type="text/javascript">
93
94 $(document).ready(function () {
95
96 $("li[fk_user]").click(function () {
97
98 if ($(this).hasClass('disabled')) {
99 $(this).removeClass('disabled');
100 } else {
101 $(this).addClass('disabled');
102 }
103
104
105 var $userl = $(this).closest('ul');
106 var listname = $userl.attr('name');
107
108 var TId = [];
109
110 $userl.find('li').each(function (i, item) {
111
112 if ($(item).hasClass('disabled')) {
113 TId.push($(item).attr('fk_user'));
114 }
115
116 });
117
118 $('#' + listname + '_excluded_id').val(TId.join(','));
119
120 });
121
122 });
123
124
125 </script>
126
127
128<?php
129
130$fk_usergroup2 = 0;
131$fk_job = (int) GETPOST('fk_job');
132if ($fk_job <= 0) {
133 $fk_usergroup2 = GETPOSTINT('fk_usergroup2');
134}
135
136$fk_usergroup1 = GETPOSTINT('fk_usergroup1');
137
138?>
139
140
141 <div class="fichecenter">
142 <form action="<?php echo $_SERVER['PHP_SELF'] ?>">
143
144 <div class="tabBar tabBarWithBottom">
145 <div class="fichehalfleft">
146 <table class="border tableforfield" width="100%">
147 <tr>
148 <td><?php
149 print $langs->trans('group1ToCompare').'</td><td>';
150 print img_picto('', 'group', 'class="pictofixedwidth"');
151 print $form->select_dolgroups($fk_usergroup1, 'fk_usergroup1', 1);
152 ?></td>
153 </tr>
154 <tr><td></td><td></td></tr>
155 <tr>
156 <td><?php
157 print $langs->trans('group2ToCompare').'</td><td>';
158 print img_picto('', 'group', 'class="pictofixedwidth"');
159 print $form->select_dolgroups($fk_usergroup2, 'fk_usergroup2', 1);
160 ?></td>
161 </tr>
162 <tr>
163 <td><b><?php print $langs->trans('or'); ?></b></td><td></td>
164 </tr>
165 <tr>
166 <td><?php
167 echo $langs->trans('OrJobToCompare') . '</td><td>';
168 $j = new Job($db);
169 $jobs = $j->fetchAll();
170 $TJobs = array();
171
172 foreach ($jobs as &$j) {
173 $TJobs[$j->id] = $j->label;
174 }
175
176 print img_picto('', 'jobprofile', 'class="pictofixedwidth"').$form->selectarray('fk_job', $TJobs, $fk_job, 1);
177 ?></td>
178 </tr>
179 </table>
180 </div>
181
182 <div style="background:#eee;border-radius:5px 0;margin:0px 0 10px;font-style:italic;padding:5px;" class="fichehalfright">
183 <!--<h4><?php echo $langs->trans('legend'); ?></h4>-->
184 <table class="border" width="100%">
185 <tr>
186 <td><span style="vertical-align:middle" class="toohappy diffnote little"></span>
187 <?php echo $langs->trans('CompetenceAcquiredByOneOrMore'); ?></td>
188 </tr>
189 <tr>
190 <td><span style="vertical-align:middle" class="veryhappy diffnote little"></span>
191 <?php echo $langs->trans('MaxlevelGreaterThan'); ?></td>
192 </tr>
193 <tr>
194 <td><span style="vertical-align:middle" class="happy diffnote little"></span>
195 <?php echo $langs->trans('MaxLevelEqualTo'); ?></td>
196 </tr>
197 <tr>
198 <td><span style="vertical-align:middle" class="sad diffnote little"></span>
199 <?php echo $langs->trans('MaxLevelLowerThan'); ?></td>
200 </tr>
201 <tr>
202 <td><span style="vertical-align:middle" class="toosad diffnote little"></span>
203 <?php echo $langs->trans('SkillNotAcquired'); ?></td>
204 </tr>
205 </table>
206
207 </div>
208
209 <div class="clearboth"></div>
210
211 </div>
212
213 <br><br>
214 <div class="center">
215 <input class="button" type="SUBMIT" name="bt1" VALUE="<?php print $langs->trans('Refresh'); ?>">
216 </div>
217 <br><br>
218
219 <div id="compare" class="centpercent" style="position:relative;">
220
221 <?php if ($fk_usergroup1 > 0 || $fk_usergroup2 > 0 || $fk_job > 0) {
222 $TUser1 = $TUser2 = array();
223
224 $userlist1 = displayUsersListWithPicto($TUser1, $fk_usergroup1, 'list1'); // This fill also the $TUser1
225
226 $TSkill1 = getSkillForUsers($TUser1);
227
228 if ($fk_job > 0) {
229 $TSkill2 = getSkillForJob($fk_job);
230
231 $job = new Job($db);
232 $job->fetch($fk_job);
233 $userlist2 = '<ul>
234 <li>
235 <h3>' . $job->label . '</h3>
236 <p>' . $job->description . '</p>
237 </li>
238 </ul>';
239 } else {
240 $userlist2 = displayUsersListWithPicto($TUser2, $fk_usergroup2, 'list2');
241 $TSkill2 = getSkillForUsers($TUser2);
242 }
243
244 $TMergedSkills = mergeSkills($TSkill1, $TSkill2);
245 ?>
246 <table class="centpercent">
247 <tr>
248 <th class="left"><?php print $langs->trans('Employees'); ?></th>
249 <th class="left" style="padding-left: 10px;"><?php print $langs->trans('Skill'); ?></th>
250 <th><?php print $langs->trans('HighestRank'); ?></th>
251 <th><?php print $langs->trans('difference'); ?></th>
252 <th><?php print $langs->trans($fk_job > 0 ? 'ExpectedRank' : 'HighestRank'); ?></th>
253 <th></th>
254 </tr>
255
256 <?php
257 echo '<tr>';
258
259 echo '<td id="list-user-left" style="width:25%; padding-right: 10px; border-right: 1px solid #ccc" class="valigntop">';
260 echo $userlist1;
261 echo '</td>';
262
263 echo '<td id="" style="width:20%; padding-left: 10px;" valign="top">' . skillList($TMergedSkills) . '</td>';
264
265 echo '<td id="" style="width:10%" valign="top">' . rate($TMergedSkills, 'rate1') . '</td>';
266
267 echo '<td id="" style="width:10%" valign="top">' . diff($TMergedSkills) . '</td>';
268
269 echo '<td id="" style="width:10%; padding-right: 10px;" valign="top">' . rate($TMergedSkills, 'rate2') . '</td>';
270
271 echo '<td id="list-user-right" style="width:25%; padding-left: 10px; border-left: 1px solid #ccc;" class="valigntop">';
272 echo $userlist2;
273 echo '</td>';
274
275 echo '</tr>';
276
277 ?>
278
279 </table>
280
281 <?php } ?>
282
283 </div>
284
285 </form>
286
287 </div>
288
289<?php
290
291print dol_get_fiche_end();
292
293llxFooter();
294$db->close();
295
296
297
304function diff(&$TMergedSkills)
305{
306 $out = '<ul class="diff">';
307
308 foreach ($TMergedSkills as $id => &$sk) {
309 $class = 'diffnote';
310
311 if (empty($sk->rate2)) {
312 $class .= ' toohappy';
313 } elseif (empty($sk->rate1)) {
314 $class .= ' toosad';
315 } elseif ($sk->rate1 == $sk->rate2) {
316 $class .= ' happy';
317 } elseif ($sk->rate2 < $sk->rate1) {
318 $class .= ' veryhappy';
319 } elseif ($sk->rate2 > $sk->rate1) {
320 $class .= ' sad';
321 }
322
323 $out .= '<li fk_skill="' . $id . '" class="' . $class . '" style="text-align:center;">
324 <span class="' . $class . '">&nbsp;</span>
325 </li>';
326 }
327
328 $out .= '</ul>';
329
330 return $out;
331}
332
340function rate(&$TMergedSkills, $field)
341{
342 global $langs, $fk_job;
343
344 $out = '<ul class="competence">';
345
346 foreach ($TMergedSkills as $id => &$sk) {
347 $class = "note";
348 $how_many = 0;
349 if (empty($sk->$field)) {
350 $note = 'x';
351 $class .= ' none';
352 } else {
353 $note = $sk->$field < 0 ? $langs->trans("NA") : $sk->$field;
354 $how_many = ($field === 'rate1') ? $sk->how_many_max1 : $sk->how_many_max2;
355 }
356
357 if ($field === 'rate2' && $fk_job > 0) {
358 $trad = $langs->trans('RequiredRank');
359 } else {
360 $trad = $langs->trans('HighestRank');
361 }
362
363 $out .= '<li fk_skill="' . $id . '" class="center">
364 <p class="nowraponall"><span class="' . $class . ' classfortooltip" title="' . $trad . '">' . $note . '</span>' . ($how_many > 0 ? '<span class="bubble classfortooltip" title="' . $langs->trans('HowManyUserWithThisMaxNote') . '">' . $how_many . '</span>' : '') . '</p>
365 </li>';
366 }
367
368 $out .= '</ul>';
369
370 return $out;
371}
372
379function skillList(&$TMergedSkills)
380{
381 $out = '<ul class="competence">';
382
383 foreach ($TMergedSkills as $id => &$sk) {
384 $out .= '<li fk_skill="' . $id . '">
385 <h3>' . $sk->label . '</h3>
386 <p>' . $sk->description . '</p>
387 </li>';
388 }
389
390 $out .= '</ul>';
391
392 return $out;
393}
394
402function mergeSkills($TSkill1, $TSkill2)
403{
404 $Tab = array();
405
406 foreach ($TSkill1 as &$sk) {
407 if (empty($Tab[$sk->fk_skill])) {
408 $Tab[$sk->fk_skill] = new stdClass();
409 }
410
411 $Tab[$sk->fk_skill]->rate1 = $sk->rankorder;
412 $Tab[$sk->fk_skill]->how_many_max1 = $sk->how_many_max;
413
414 $Tab[$sk->fk_skill]->label = $sk->label;
415 $Tab[$sk->fk_skill]->description = $sk->description;
416 }
417
418 foreach ($TSkill2 as &$sk) {
419 if (empty($Tab[$sk->fk_skill])) {
420 $Tab[$sk->fk_skill] = new stdClass();
421 }
422 $Tab[$sk->fk_skill]->rate2 = $sk->rankorder;
423 $Tab[$sk->fk_skill]->how_many_max2 = $sk->how_many_max;
424
425 $Tab[$sk->fk_skill]->label = $sk->label;
426 $Tab[$sk->fk_skill]->description = $sk->description;
427 }
428
429 return $Tab;
430}
431
440function displayUsersListWithPicto(&$TUser, $fk_usergroup = 0, $namelist = 'list-user')
441{
442 global $db, $langs, $conf, $form;
443
444 $out = '';
445 if ($fk_usergroup > 0) {
446 $list = $namelist . '_excluded_id';
447
448 $excludedIdsList = GETPOST($list);
449
450 $sql = "SELECT u.rowid FROM " . MAIN_DB_PREFIX . "user u
451 LEFT JOIN " . MAIN_DB_PREFIX . "usergroup_user as ugu ON (u.rowid = ugu.fk_user)
452 WHERE u.statut > 0 AND ugu.entity = ".((int) $conf->entity);
453 $sql .= " AND ugu.fk_usergroup=" . ((int) $fk_usergroup);
454
455 $res = $db->query($sql);
456 $out .= '<ul name="' . $namelist . '">';
457
458 $TExcludedId = explode(',', $excludedIdsList);
459
460 $out .= '<input id="'.$list.'" type="hidden" name="'.$list.'" value="'.$excludedIdsList.'"> ';
461
462 $job = new Job($db);
463
464 while ($obj = $db->fetch_object($res)) {
465 $class = '';
466
467 $user = new User($db);
468 $user->fetch($obj->rowid);
469
470 $name = $user->getFullName($langs);
471 if (empty($name)) {
472 $name = $user->login;
473 }
474
475 if (in_array($user->id, $TExcludedId)) {
476 $class .= ' disabled';
477 } else {
478 if (!in_array($user->id, $TUser)) {
479 $TUser[] = $user->id;
480 }
481 }
482
483 $desc = '';
484
485 $jobstring = $job->getLastJobForUser($user->id);
486 $desc .= $jobstring;
487
488 $static_eval = new Evaluation($db);
489 $evaluation = $static_eval->getLastEvaluationForUser($user->id);
490
491 if (!empty($evaluation) && !empty($evaluation->date_eval)) {
492 $desc .= $langs->trans('DateLastEval') . ' : ' . dol_print_date($evaluation->date_eval);
493 } else {
494 $desc .= $langs->trans('NoEval');
495 }
496
497 if (!empty($user->array_options['options_DDA'])) {
498 $desc .= '<br>' . $langs->trans('Seniority') . ' : ' . dol_print_date(strtotime($user->array_options['options_DDA']));
499 }
500
501 $out .= '<li fk_user="' . $user->id . '" class="' . $class . '">
502 ' . $form->showphoto('userphoto', $user, 0, 0, 0, 'photoref', 'small', 1, 0, '', 1) . '
503 <h3>' . $name . '</h3>
504 <p>' . $desc . '</p>
505 </li>';
506 }
507
508 $out .= '</ul>';
509 }
510
511 return $out;
512}
513
514
521function getSkillForUsers($TUser)
522{
523 global $db;
524
525 // I go back to the user with the highest score in a given group for all the skills assessed in that group
526 if (empty($TUser)) {
527 return array();
528 }
529
530 $sql = 'SELECT sk.rowid, sk.label, sk.description, sk.skill_type,';
531 $sql .= ' MAX(sr.rankorder) as rankorder';
532 $sql .= ' FROM '.MAIN_DB_PREFIX.'hrm_skill sk';
533 $sql .= ' INNER JOIN '.MAIN_DB_PREFIX.'hrm_skillrank sr';
534 $sql .= " WHERE sk.rowid = sr.fk_skill AND sr.objecttype = '".$db->escape(SkillRank::SKILLRANK_TYPE_USER)."'";
535 $sql .= ' AND sr.fk_object IN ('.$db->sanitize(implode(',', $TUser)).')';
536 $sql .= ' AND rankorder >= 0';
537 $sql .= " GROUP BY sk.rowid, sk.label, sk.description, sk.skill_type"; // group by skill
538 $sql .= " ORDER BY sk.rowid ASC";
539
540 $resql = $db->query($sql);
541 $Tab = array();
542
543 if ($resql) {
544 // For each skill, we count the number of times that the max score has been reached within a given group
545 $num = 0;
546 while ($obj = $db->fetch_object($resql)) {
547 $Tab[$num] = new stdClass();
548
549 $Tab[$num]->fk_skill = $obj->rowid;
550 $Tab[$num]->label = $obj->label;
551 $Tab[$num]->description = $obj->description;
552 $Tab[$num]->skill_type = $obj->skill_type;
553 //$Tab[$num]->fk_object = $obj->fk_object;
554 $Tab[$num]->objectType = SkillRank::SKILLRANK_TYPE_USER;
555
556 $Tab[$num]->rankorder = $obj->rankorder;
557
558 // Get how_many_max
559 $sql1 = "SELECT COUNT(rowid) as how_many_max FROM ".MAIN_DB_PREFIX."hrm_skillrank as sr";
560 $sql1 .= " WHERE sr.rankorder = ".((int) $obj->rankorder);
561 $sql1 .= " AND sr.objecttype = '".$db->escape(SkillRank::SKILLRANK_TYPE_USER)."'";
562 $sql1 .= " AND sr.fk_skill = ".((int) $obj->rowid);
563 $sql1 .= " AND sr.fk_object IN (".$db->sanitize(implode(',', $TUser)).")";
564 $resql1 = $db->query($sql1);
565
566 $objMax = $db->fetch_object($resql1);
567
568 $Tab[$num]->how_many_max = $objMax->how_many_max;
569
570 $db->free($resql1);
571
572 $num++;
573 }
574 } else {
576 }
577
578 return $Tab;
579}
580
587function getSkillForJob($fk_job)
588{
589 global $db;
590
591 if (empty($fk_job)) {
592 return array();
593 }
594
595 $sql = 'SELECT sk.rowid, sk.label, sk.description, sk.skill_type, sr.fk_object, sr.objecttype, sr.fk_skill,';
596 $sql .= " MAX(sr.rankorder) as rankorder";
597 $sql .= ' FROM '.MAIN_DB_PREFIX.'hrm_skill as sk';
598 $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'hrm_skillrank as sr ON (sk.rowid = sr.fk_skill)';
599 $sql .= " WHERE sr.objecttype = '".SkillRank::SKILLRANK_TYPE_JOB."'";
600 $sql .= ' AND sr.fk_object = '.((int) $fk_job);
601 $sql .= ' GROUP BY sk.rowid, sk.label, sk.description, sk.skill_type, sr.fk_object, sr.objecttype, sr.fk_skill'; // group par competence*/
602
603 $resql = $db->query($sql);
604 $Tab = array();
605
606 if ($resql) {
607 $num = 0;
608 while ($obj = $db->fetch_object($resql)) {
609 $Tab[$num] = new stdClass();
610 $Tab[$num]->fk_skill = $obj->fk_skill;
611 $Tab[$num]->label = $obj->label;
612 $Tab[$num]->description = $obj->description;
613 $Tab[$num]->skill_type = $obj->skill_type;
614 //$Tab[$num]->date_start = '';// du poste
615 //$Tab[$num]->date_end = ''; // du poste
616 $Tab[$num]->fk_object = $obj->fk_object;
617 $Tab[$num]->objectType = SkillRank::SKILLRANK_TYPE_JOB;
618 $Tab[$num]->rankorder = $obj->rankorder;
619 $Tab[$num]->how_many_max = $obj->how_many_max;
620
621 $num++;
622 }
623 } else {
625 }
626
627 return $Tab;
628}
$id
Support class for third parties, contacts, members, users or resources.
Definition account.php:47
llxFooter($comment='', $zone='private', $disabledoutputofmessages=0)
Empty footer.
Definition wrapper.php:91
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:73
Class for Evaluation.
Class for Job.
Definition job.class.php:38
Class to manage Dolibarr users.
if(!isModEnabled('ai')||!getDolGlobalString('AI_ASSISTANT_ENABLED')) global $conf
The main.inc.php has been included so the following variable are now defined:
if(!isModEnabled('ai')||!getDolGlobalString('AI_ASSISTANT_ENABLED')) global $db
API class for accounts.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2, $allowothertags=array())
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.
dol_get_fiche_head($links=array(), $active='', $title='', $notab=0, $picto='', $pictoisfullpath=0, $morehtmlright='', $morecss='', $limittoshow=0, $moretabssuffix='', $dragdropfile=0, $morecssdiv='')
Show tabs of a record.
dol_get_fiche_end($notab=0)
Return tab footer of a card.
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
dol_print_date($time, $format='', $tzoutput='auto', $outputlangs=null, $encodetooutput=false, $decorate=0)
Output date in a string format according to outputlangs (or langs if not defined).
dol_print_error($db=null, $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
isModEnabled($module)
Is Dolibarr module enabled.
treeview li table
No Email.
if(preg_match('/(crypted|dolcrypt):/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
'integer', 'integer:ObjectClass:PathToClass[:AddCreateButtonOrNot[:Filter[:Sortfield]]]',...
Definition repair.php:130
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition repair.php:133
accessforbidden($message='', $printheader=1, $printfooter=1, $showonlymessage=0, $params=null)
Show a message to say access is forbidden and stop program.