dolibarr 21.0.3
expensereport.class.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2011 Dimitri Mouillard <dmouillard@teclib.com>
3 * Copyright (C) 2015 Laurent Destailleur <eldy@users.sourceforge.net>
4 * Copyright (C) 2015 Alexandre Spangaro <aspangaro@open-dsi.fr>
5 * Copyright (C) 2018 Nicolas ZABOURI <info@inovea-conseil.com>
6 * Copyright (c) 2018-2024 Frédéric France <frederic.france@free.fr>
7 * Copyright (C) 2016-2020 Ferran Marcet <fmarcet@2byte.es>
8 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
9 * Copyright (C) 2024 Frédéric France <frederic.france@free.fr>
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
30require_once DOL_DOCUMENT_ROOT.'/core/class/commonobject.class.php';
31require_once DOL_DOCUMENT_ROOT.'/expensereport/class/expensereportline.class.php';
32require_once DOL_DOCUMENT_ROOT.'/expensereport/class/expensereport_ik.class.php';
33require_once DOL_DOCUMENT_ROOT.'/expensereport/class/expensereport_rule.class.php';
34
35
40{
44 public $element = 'expensereport';
45
49 public $table_element = 'expensereport';
50
54 public $table_element_line = 'expensereport_det';
55
59 public $fk_element = 'fk_expensereport';
60
64 public $picto = 'trip';
65
69 public $lines = array();
70
74 public $line;
75
79 public $date_debut;
80
84 public $date_fin;
85
89 public $date_approbation;
90
94 public $fk_user;
95
99 public $user_approve_id;
100
106 public $status;
107
114 public $fk_statut;
115
119 public $fk_c_paiement;
120
124 public $modepaymentid;
125
129 public $paid;
130
131 // Paiement
135 public $user_paid_infos;
136
140 public $user_author_infos;
141
145 public $user_validator_infos;
146
150 public $rule_warning_message;
151
152 // ACTIONS
153
154 // Create
158 public $date_create;
159
163 public $fk_user_creat;
164
168 public $fk_user_author; // Note fk_user_author is not the 'author' but the guy the expense report is for.
169
170 // Update
174 public $date_modif;
175
179 public $fk_user_modif;
180
181 // Refus
185 public $date_refuse;
186
190 public $detail_refuse;
191
195 public $fk_user_refuse;
196
197 // Annulation
201 public $date_cancel;
202
206 public $detail_cancel;
207
211 public $fk_user_cancel;
212
216 public $fk_user_validator;
217
224 public $datevalid;
225
230 public $date_valid;
231
235 public $fk_user_valid;
236
240 public $user_valid_infos;
241
242 // Approve
246 public $date_approve;
247
251 public $fk_user_approve;
252
257 public $localtax1; // for backward compatibility (real field should be total_localtax1 defined into CommonObject)
262 public $localtax2; // for backward compatibility (real field should be total_localtax2 defined into CommonObject)
263
267 const STATUS_DRAFT = 0;
268
273
278
283
287 const STATUS_CLOSED = 6;
288
292 const STATUS_REFUSED = 99;
293
294 public $fields = array(
295 'rowid' => array('type' => 'integer', 'label' => 'ID', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 10),
296 'ref' => array('type' => 'varchar(50)', 'label' => 'Ref', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'showoncombobox' => 1, 'position' => 15),
297 'entity' => array('type' => 'integer', 'label' => 'Entity', 'default' => '1', 'enabled' => 1, 'visible' => -2, 'notnull' => 1, 'position' => 20),
298 'ref_number_int' => array('type' => 'integer', 'label' => 'Ref number int', 'enabled' => 1, 'visible' => -1, 'position' => 25),
299 'ref_ext' => array('type' => 'integer', 'label' => 'RefExt', 'enabled' => 1, 'visible' => 0, 'position' => 30),
300 'total_ht' => array('type' => 'double(24,8)', 'label' => 'Total ht', 'enabled' => 1, 'visible' => -1, 'position' => 35),
301 'total_tva' => array('type' => 'double(24,8)', 'label' => 'Total tva', 'enabled' => 1, 'visible' => -1, 'position' => 40),
302 'localtax1' => array('type' => 'double(24,8)', 'label' => 'Localtax1', 'enabled' => 1, 'visible' => -1, 'position' => 45),
303 'localtax2' => array('type' => 'double(24,8)', 'label' => 'Localtax2', 'enabled' => 1, 'visible' => -1, 'position' => 50),
304 'total_ttc' => array('type' => 'double(24,8)', 'label' => 'Total ttc', 'enabled' => 1, 'visible' => -1, 'position' => 55),
305 'date_debut' => array('type' => 'date', 'label' => 'Date debut', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 60),
306 'date_fin' => array('type' => 'date', 'label' => 'Date fin', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 65),
307 'date_valid' => array('type' => 'datetime', 'label' => 'Date valid', 'enabled' => 1, 'visible' => -1, 'position' => 75),
308 'date_approve' => array('type' => 'datetime', 'label' => 'Date approve', 'enabled' => 1, 'visible' => -1, 'position' => 80),
309 'date_refuse' => array('type' => 'datetime', 'label' => 'Date refuse', 'enabled' => 1, 'visible' => -1, 'position' => 85),
310 'date_cancel' => array('type' => 'datetime', 'label' => 'Date cancel', 'enabled' => 1, 'visible' => -1, 'position' => 90),
311 'fk_user_author' => array('type' => 'integer', 'label' => 'Fk user author', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 100),
312 'fk_user_modif' => array('type' => 'integer', 'label' => 'Fk user modif', 'enabled' => 1, 'visible' => -1, 'position' => 105),
313 'fk_user_valid' => array('type' => 'integer', 'label' => 'Fk user valid', 'enabled' => 1, 'visible' => -1, 'position' => 110),
314 'fk_user_validator' => array('type' => 'integer', 'label' => 'Fk user validator', 'enabled' => 1, 'visible' => -1, 'position' => 115),
315 'fk_user_approve' => array('type' => 'integer', 'label' => 'Fk user approve', 'enabled' => 1, 'visible' => -1, 'position' => 120),
316 'fk_user_refuse' => array('type' => 'integer', 'label' => 'Fk user refuse', 'enabled' => 1, 'visible' => -1, 'position' => 125),
317 'fk_user_cancel' => array('type' => 'integer', 'label' => 'Fk user cancel', 'enabled' => 1, 'visible' => -1, 'position' => 130),
318 'fk_c_paiement' => array('type' => 'integer', 'label' => 'Fk c paiement', 'enabled' => 1, 'visible' => -1, 'position' => 140),
319 'paid' => array('type' => 'integer', 'label' => 'Paid', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 145),
320 'note_public' => array('type' => 'html', 'label' => 'Note public', 'enabled' => 1, 'visible' => 0, 'position' => 150),
321 'note_private' => array('type' => 'html', 'label' => 'Note private', 'enabled' => 1, 'visible' => 0, 'position' => 155),
322 'detail_refuse' => array('type' => 'varchar(255)', 'label' => 'Detail refuse', 'enabled' => 1, 'visible' => -1, 'position' => 160),
323 'detail_cancel' => array('type' => 'varchar(255)', 'label' => 'Detail cancel', 'enabled' => 1, 'visible' => -1, 'position' => 165),
324 'integration_compta' => array('type' => 'integer', 'label' => 'Integration compta', 'enabled' => 1, 'visible' => -1, 'position' => 170),
325 'fk_bank_account' => array('type' => 'integer', 'label' => 'Fk bank account', 'enabled' => 1, 'visible' => -1, 'position' => 175),
326 'fk_multicurrency' => array('type' => 'integer', 'label' => 'Fk multicurrency', 'enabled' => 1, 'visible' => -1, 'position' => 185),
327 'multicurrency_code' => array('type' => 'varchar(255)', 'label' => 'Multicurrency code', 'enabled' => 1, 'visible' => -1, 'position' => 190),
328 'multicurrency_tx' => array('type' => 'double(24,8)', 'label' => 'Multicurrency tx', 'enabled' => 1, 'visible' => -1, 'position' => 195),
329 'multicurrency_total_ht' => array('type' => 'double(24,8)', 'label' => 'Multicurrency total ht', 'enabled' => 1, 'visible' => -1, 'position' => 200),
330 'multicurrency_total_tva' => array('type' => 'double(24,8)', 'label' => 'Multicurrency total tva', 'enabled' => 1, 'visible' => -1, 'position' => 205),
331 'multicurrency_total_ttc' => array('type' => 'double(24,8)', 'label' => 'Multicurrency total ttc', 'enabled' => 1, 'visible' => -1, 'position' => 210),
332 'extraparams' => array('type' => 'varchar(255)', 'label' => 'Extraparams', 'enabled' => 1, 'visible' => -1, 'position' => 220),
333 'date_create' => array('type' => 'datetime', 'label' => 'Date create', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 300),
334 'tms' => array('type' => 'timestamp', 'label' => 'Tms', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 305),
335 'import_key' => array('type' => 'varchar(14)', 'label' => 'ImportId', 'enabled' => 1, 'visible' => -1, 'position' => 1000),
336 'model_pdf' => array('type' => 'varchar(255)', 'label' => 'Model pdf', 'enabled' => 1, 'visible' => 0, 'position' => 1010),
337 'fk_statut' => array('type' => 'integer', 'label' => 'Fk statut', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 500),
338 );
339
345 public function __construct($db)
346 {
347 $this->db = $db;
348 $this->total_ht = 0;
349 $this->total_ttc = 0;
350 $this->total_tva = 0;
351 $this->total_localtax1 = 0;
352 $this->total_localtax2 = 0;
353 $this->localtax1 = 0; // For backward compatibility
354 $this->localtax2 = 0; // For backward compatibility
355 $this->modepaymentid = 0;
356
357 // List of language codes for status
358 $this->labelStatusShort = array(0 => 'Draft', 2 => 'Validated', 4 => 'Canceled', 5 => 'Approved', 6 => 'Paid', 99 => 'Refused');
359 $this->labelStatus = array(0 => 'Draft', 2 => 'ValidatedWaitingApproval', 4 => 'Canceled', 5 => 'Approved', 6 => 'Paid', 99 => 'Refused');
360
361 $this->fields['ref_ext']['visible'] = getDolGlobalInt('MAIN_LIST_SHOW_REF_EXT');
362 }
363
371 public function create($user, $notrigger = 0)
372 {
373 global $conf, $langs;
374
375 $now = dol_now();
376
377 $error = 0;
378
379 // Check parameters
380 if (empty($this->date_debut) || empty($this->date_fin)) {
381 $this->error = $langs->trans('ErrorFieldRequired', $langs->transnoentitiesnoconv('Date'));
382 return -1;
383 }
384
385 $fuserid = $this->fk_user_author; // Note fk_user_author is not the 'author' but the guy the expense report is for.
386 if (empty($fuserid)) {
387 $fuserid = $user->id;
388 }
389
390 $this->db->begin();
391
392 $sql = "INSERT INTO ".MAIN_DB_PREFIX.$this->table_element." (";
393 $sql .= "ref";
394 $sql .= ",total_ht";
395 $sql .= ",total_ttc";
396 $sql .= ",total_tva";
397 $sql .= ",date_debut";
398 $sql .= ",date_fin";
399 $sql .= ",date_create";
400 $sql .= ",fk_user_creat";
401 $sql .= ",fk_user_author";
402 $sql .= ",fk_user_validator";
403 $sql .= ",fk_user_approve";
404 $sql .= ",fk_user_modif";
405 $sql .= ",fk_statut";
406 $sql .= ",fk_c_paiement";
407 $sql .= ",paid";
408 $sql .= ",note_public";
409 $sql .= ",note_private";
410 $sql .= ",entity";
411 $sql .= ") VALUES(";
412 $sql .= "'(PROV)'";
413 $sql .= ", ".price2num($this->total_ht, 'MT');
414 $sql .= ", ".price2num($this->total_ttc, 'MT');
415 $sql .= ", ".price2num($this->total_tva, 'MT');
416 $sql .= ", '".$this->db->idate($this->date_debut)."'";
417 $sql .= ", '".$this->db->idate($this->date_fin)."'";
418 $sql .= ", '".$this->db->idate($now)."'";
419 $sql .= ", ".((int) $user->id);
420 $sql .= ", ".((int) $fuserid);
421 $sql .= ", ".($this->fk_user_validator > 0 ? ((int) $this->fk_user_validator) : "null");
422 $sql .= ", ".($this->fk_user_approve > 0 ? ((int) $this->fk_user_approve) : "null");
423 $sql .= ", ".($this->fk_user_modif > 0 ? ((int) $this->fk_user_modif) : "null");
424 $sql .= ", ".($this->fk_statut > 1 ? ((int) $this->fk_statut) : 0);
425 $sql .= ", ".($this->modepaymentid ? ((int) $this->modepaymentid) : "null");
426 $sql .= ", 0";
427 $sql .= ", ".($this->note_public ? "'".$this->db->escape($this->note_public)."'" : "null");
428 $sql .= ", ".($this->note_private ? "'".$this->db->escape($this->note_private)."'" : "null");
429 $sql .= ", ".((int) $conf->entity);
430 $sql .= ")";
431
432 $result = $this->db->query($sql);
433 if ($result) {
434 $this->id = $this->db->last_insert_id(MAIN_DB_PREFIX.$this->table_element);
435 $this->ref = '(PROV'.$this->id.')';
436
437 $sql = 'UPDATE '.MAIN_DB_PREFIX.$this->table_element." SET ref='".$this->db->escape($this->ref)."' WHERE rowid=".((int) $this->id);
438 $resql = $this->db->query($sql);
439 if (!$resql) {
440 $this->error = $this->db->lasterror();
441 $error++;
442 }
443
444 if (!$error) {
445 if (is_array($this->lines) && count($this->lines) > 0) {
446 foreach ($this->lines as $line) {
447 // Test and convert into object this->lines[$i]. When coming from REST API, we may still have an array
448 //if (! is_object($line)) $line=json_decode(json_encode($line), false); // convert recursively array into object.
449 if (!is_object($line)) {
450 $line = (object) $line;
451 $newndfline = new ExpenseReportLine($this->db);
452 $newndfline->fk_expensereport = $line->fk_expensereport;
453 $newndfline->fk_c_type_fees = $line->fk_c_type_fees;
454 $newndfline->fk_project = $line->fk_project;
455 $newndfline->vatrate = $line->vatrate;
456 $newndfline->vat_src_code = $line->vat_src_code;
457 $newndfline->localtax1_tx = $line->localtax1_tx;
458 $newndfline->localtax2_tx = $line->localtax2_tx;
459 $newndfline->localtax1_type = $line->localtax1_type;
460 $newndfline->localtax2_type = $line->localtax2_type;
461 $newndfline->comments = $line->comments;
462 $newndfline->qty = $line->qty;
463 $newndfline->value_unit = $line->value_unit;
464 $newndfline->total_ht = $line->total_ht;
465 $newndfline->total_ttc = $line->total_ttc;
466 $newndfline->total_tva = $line->total_tva;
467 $newndfline->total_localtax1 = $line->total_localtax1;
468 $newndfline->total_localtax2 = $line->total_localtax2;
469 $newndfline->date = $line->date;
470 $newndfline->rule_warning_message = $line->rule_warning_message;
471 $newndfline->fk_c_exp_tax_cat = $line->fk_c_exp_tax_cat;
472 $newndfline->fk_ecm_files = $line->fk_ecm_files;
473 } else {
474 $newndfline = $line;
475 }
476 //$newndfline=new ExpenseReportLine($this->db);
477 $newndfline->fk_expensereport = $this->id;
478 $result = $newndfline->insert();
479 if ($result < 0) {
480 $this->error = $newndfline->error;
481 $this->errors = $newndfline->errors;
482 $error++;
483 break;
484 }
485 }
486 }
487 }
488
489 if (!$error) {
490 $result = $this->insertExtraFields();
491 if ($result < 0) {
492 $error++;
493 }
494 }
495
496 if (!$error) {
497 $result = $this->update_price(1);
498 if ($result > 0) {
499 if (!$notrigger) {
500 // Call trigger
501 $result = $this->call_trigger('EXPENSE_REPORT_CREATE', $user);
502
503 if ($result < 0) {
504 $error++;
505 }
506 // End call triggers
507 }
508
509 if (empty($error)) {
510 $this->db->commit();
511 return $this->id;
512 } else {
513 $this->db->rollback();
514 return -4;
515 }
516 } else {
517 $this->db->rollback();
518 return -3;
519 }
520 } else {
521 dol_syslog(get_class($this)."::create error ".$this->error, LOG_ERR);
522 $this->db->rollback();
523 return -2;
524 }
525 } else {
526 $this->error = $this->db->lasterror()." sql=".$sql;
527 $this->db->rollback();
528 return -1;
529 }
530 }
531
539 public function createFromClone(User $user, $fk_user_author)
540 {
541 global $hookmanager;
542
543 $error = 0;
544
545 if (empty($fk_user_author)) {
546 $fk_user_author = $user->id;
547 }
548
549 $this->db->begin();
550
551 // get extrafields so they will be clone
552 //foreach($this->lines as $line)
553 //$line->fetch_optionals();
554
555 // Load source object
556 $objFrom = clone $this;
557
558 $this->id = 0;
559 $this->ref = '';
560 $this->status = 0;
561 $this->fk_statut = 0; // deprecated
562
563 // Clear fields
564 $this->fk_user_creat = $user->id;
565 $this->fk_user_author = $fk_user_author; // Note fk_user_author is not the 'author' but the guy the expense report is for.
566 $this->fk_user_valid = 0;
567 $this->date_create = '';
568 $this->date_creation = '';
569 $this->date_validation = '';
570
571 // Remove link on lines to a joined file
572 if (is_array($this->lines) && count($this->lines) > 0) {
573 foreach ($this->lines as $key => $line) {
574 $this->lines[$key]->fk_ecm_files = 0;
575 }
576 }
577
578 // Create clone
579 $this->context['createfromclone'] = 'createfromclone';
580 $result = $this->create($user);
581 if ($result < 0) {
582 $error++;
583 }
584
585 if (!$error) {
586 // Hook of thirdparty module
587 if (is_object($hookmanager)) {
588 $parameters = array('objFrom' => $objFrom);
589 $action = '';
590 $reshook = $hookmanager->executeHooks('createFrom', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
591 if ($reshook < 0) {
592 $this->setErrorsFromObject($hookmanager);
593 $error++;
594 }
595 }
596 }
597
598 unset($this->context['createfromclone']);
599
600 // End
601 if (!$error) {
602 $this->db->commit();
603 return $this->id;
604 } else {
605 $this->db->rollback();
606 return -1;
607 }
608 }
609
610
619 public function update($user, $notrigger = 0, $userofexpensereport = null)
620 {
621 $error = 0;
622 $this->db->begin();
623
624 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element." SET";
625 $sql .= " total_ht = ".((float) $this->total_ht);
626 $sql .= " , total_ttc = ".((float) $this->total_ttc);
627 $sql .= " , total_tva = ".((float) $this->total_tva);
628 $sql .= " , date_debut = '".$this->db->idate($this->date_debut)."'";
629 $sql .= " , date_fin = '".$this->db->idate($this->date_fin)."'";
630 if ($userofexpensereport && is_object($userofexpensereport)) {
631 $sql .= " , fk_user_author = ".($userofexpensereport->id > 0 ? $userofexpensereport->id : "null"); // Note fk_user_author is not the 'author' but the guy the expense report is for.
632 }
633 $sql .= " , fk_user_validator = ".($this->fk_user_validator > 0 ? $this->fk_user_validator : "null");
634 $sql .= " , fk_user_valid = ".($this->fk_user_valid > 0 ? $this->fk_user_valid : "null");
635 $sql .= " , fk_user_approve = ".($this->fk_user_approve > 0 ? $this->fk_user_approve : "null");
636 $sql .= " , fk_user_modif = ".((int) $user->id);
637 $sql .= " , fk_statut = ".($this->fk_statut >= 0 ? $this->fk_statut : '0');
638 $sql .= " , fk_c_paiement = ".($this->fk_c_paiement > 0 ? $this->fk_c_paiement : "null");
639 $sql .= " , note_public = ".(!empty($this->note_public) ? "'".$this->db->escape($this->note_public)."'" : "''");
640 $sql .= " , note_private = ".(!empty($this->note_private) ? "'".$this->db->escape($this->note_private)."'" : "''");
641 $sql .= " , detail_refuse = ".(!empty($this->detail_refuse) ? "'".$this->db->escape($this->detail_refuse)."'" : "''");
642 $sql .= " WHERE rowid = ".((int) $this->id);
643
644 dol_syslog(get_class($this)."::update", LOG_DEBUG);
645 $result = $this->db->query($sql);
646 if ($result) {
647 $result = $this->insertExtraFields();
648 if ($result < 0) {
649 $error++;
650 }
651
652 if (!$error && !$notrigger) {
653 // Call trigger
654 $result = $this->call_trigger('EXPENSE_REPORT_MODIFY', $user);
655 if ($result < 0) {
656 $error++;
657 }
658 // End call triggers
659 }
660
661 if (!$error) {
662 $this->db->commit();
663 return 1;
664 } else {
665 $this->db->rollback();
666 $this->error = $this->db->error();
667 return -2;
668 }
669 } else {
670 $this->db->rollback();
671 $this->error = $this->db->error();
672 return -1;
673 }
674 }
675
683 public function fetch($id, $ref = '')
684 {
685 $sql = "SELECT d.rowid, d.entity, d.ref, d.note_public, d.note_private,"; // DEFAULT
686 $sql .= " d.detail_refuse, d.detail_cancel, d.fk_user_refuse, d.fk_user_cancel,"; // ACTIONS
687 $sql .= " d.date_refuse, d.date_cancel,"; // ACTIONS
688 $sql .= " d.total_ht, d.total_ttc, d.total_tva,";
689 $sql .= " d.localtax1 as total_localtax1, d.localtax2 as total_localtax2,";
690 $sql .= " d.date_debut, d.date_fin, d.date_create, d.tms as date_modif, d.date_valid, d.date_approve,"; // DATES (datetime)
691 $sql .= " d.fk_user_creat, d.fk_user_author, d.fk_user_modif, d.fk_user_validator,";
692 $sql .= " d.fk_user_valid, d.fk_user_approve,";
693 $sql .= " d.fk_statut as status, d.fk_c_paiement, d.paid";
694 $sql .= " FROM ".MAIN_DB_PREFIX.$this->table_element." as d";
695 if ($ref) {
696 $sql .= " WHERE d.ref = '".$this->db->escape($ref)."'";
697 $sql .= " AND d.entity IN (".getEntity('expensereport').")";
698 } else {
699 $sql .= " WHERE d.rowid = ".((int) $id);
700 }
701 //$sql.= $restrict;
702
703 dol_syslog(get_class($this)."::fetch", LOG_DEBUG);
704 $resql = $this->db->query($sql);
705 if ($resql) {
706 $obj = $this->db->fetch_object($resql);
707 if ($obj) {
708 $this->id = $obj->rowid;
709 $this->ref = $obj->ref;
710
711 $this->entity = $obj->entity;
712
713 $this->total_ht = $obj->total_ht;
714 $this->total_tva = $obj->total_tva;
715 $this->total_ttc = $obj->total_ttc;
716 $this->localtax1 = $obj->total_localtax1; // For backward compatibility
717 $this->localtax2 = $obj->total_localtax2; // For backward compatibility
718 $this->total_localtax1 = $obj->total_localtax1;
719 $this->total_localtax2 = $obj->total_localtax2;
720
721 $this->note_public = $obj->note_public;
722 $this->note_private = $obj->note_private;
723 $this->detail_refuse = $obj->detail_refuse;
724 $this->detail_cancel = $obj->detail_cancel;
725
726 $this->date_debut = $this->db->jdate($obj->date_debut);
727 $this->date_fin = $this->db->jdate($obj->date_fin);
728 $this->date_valid = $this->db->jdate($obj->date_valid);
729 $this->date_approve = $this->db->jdate($obj->date_approve);
730 $this->date_create = $this->db->jdate($obj->date_create);
731 $this->date_modif = $this->db->jdate($obj->date_modif);
732 $this->date_refuse = $this->db->jdate($obj->date_refuse);
733 $this->date_cancel = $this->db->jdate($obj->date_cancel);
734
735 $this->fk_user_creat = $obj->fk_user_creat;
736 $this->fk_user_author = $obj->fk_user_author; // Note fk_user_author is not the 'author' but the guy the expense report is for.
737 $this->fk_user_modif = $obj->fk_user_modif;
738 $this->fk_user_validator = $obj->fk_user_validator;
739 $this->fk_user_valid = $obj->fk_user_valid;
740 $this->fk_user_refuse = $obj->fk_user_refuse;
741 $this->fk_user_cancel = $obj->fk_user_cancel;
742 $this->fk_user_approve = $obj->fk_user_approve;
743
744 $user_author = new User($this->db);
745 if ($this->fk_user_author > 0) {
746 $user_author->fetch($this->fk_user_author);
747 }
748
749 $this->user_author_infos = dolGetFirstLastname($user_author->firstname, $user_author->lastname);
750
751 $user_approver = new User($this->db);
752 if ($this->fk_user_approve > 0) {
753 $user_approver->fetch($this->fk_user_approve);
754 } elseif ($this->fk_user_validator > 0) {
755 $user_approver->fetch($this->fk_user_validator); // For backward compatibility
756 }
757 $this->user_validator_infos = dolGetFirstLastname($user_approver->firstname, $user_approver->lastname);
758
759 $this->fk_statut = (int) $obj->status; // deprecated
760 $this->status = (int) $obj->status;
761 $this->fk_c_paiement = $obj->fk_c_paiement;
762 $this->paid = $obj->paid;
763
764 if ($this->status == self::STATUS_APPROVED || $this->status == self::STATUS_CLOSED) {
765 $user_valid = new User($this->db);
766 if ($this->fk_user_valid > 0) {
767 $user_valid->fetch($this->fk_user_valid);
768 }
769 $this->user_valid_infos = dolGetFirstLastname($user_valid->firstname, $user_valid->lastname);
770 }
771
772 $this->fetch_optionals();
773
774 $result = $this->fetch_lines();
775
776 return $result;
777 } else {
778 return 0;
779 }
780 } else {
781 $this->error = $this->db->lasterror();
782 return -1;
783 }
784 }
785
786 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
797 public function set_paid($id, $fuser, $notrigger = 0)
798 {
799 // phpcs:enable
800 dol_syslog(get_class($this)."::set_paid is deprecated, use setPaid instead", LOG_NOTICE);
801 return $this->setPaid($id, $fuser, $notrigger);
802 }
803
812 public function setPaid($id, $fuser, $notrigger = 0)
813 {
814 $error = 0;
815 $this->db->begin();
816
817 $sql = "UPDATE ".MAIN_DB_PREFIX."expensereport";
818 $sql .= " SET fk_statut = ".self::STATUS_CLOSED.", paid=1";
819 $sql .= " WHERE rowid = ".((int) $id)." AND fk_statut = ".self::STATUS_APPROVED;
820
821 dol_syslog(get_class($this)."::setPaid", LOG_DEBUG);
822 $resql = $this->db->query($sql);
823 if ($resql) {
824 if ($this->db->affected_rows($resql)) {
825 if (!$notrigger) {
826 // Call trigger
827 $result = $this->call_trigger('EXPENSE_REPORT_PAID', $fuser);
828
829 if ($result < 0) {
830 $error++;
831 }
832 // End call triggers
833 }
834
835 if (empty($error)) {
836 $this->db->commit();
837 return 1;
838 } else {
839 $this->db->rollback();
840 $this->error = $this->db->error();
841 return -2;
842 }
843 } else {
844 $this->db->commit();
845 return 0;
846 }
847 } else {
848 $this->db->rollback();
849 dol_print_error($this->db);
850 return -1;
851 }
852 }
853
860 public function getLibStatut($mode = 0)
861 {
862 return $this->LibStatut($this->status, $mode);
863 }
864
865 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
873 public function LibStatut($status, $mode = 0)
874 {
875 // phpcs:enable
876 global $langs;
877
878 $labelStatus = $langs->transnoentitiesnoconv($this->labelStatus[$status]);
879 $labelStatusShort = $langs->transnoentitiesnoconv($this->labelStatusShort[$status]);
880
881 $statuslogo = array(0 => 'status0', 2 => 'status1', 4 => 'status6', 5 => 'status4', 6 => 'status6', 99 => 'status5');
882
883 $statusType = $statuslogo[$status];
884
885 return dolGetStatus($labelStatus, $labelStatusShort, '', $statusType, $mode);
886 }
887
888
895 public function info($id)
896 {
897 global $conf;
898
899 $sql = "SELECT f.rowid,";
900 $sql .= " f.date_create as datec,";
901 $sql .= " f.tms as date_modification,";
902 $sql .= " f.date_valid as datev,";
903 $sql .= " f.date_approve as datea,";
904 $sql .= " f.fk_user_creat as fk_user_creation,";
905 $sql .= " f.fk_user_author as fk_user_author,";
906 $sql .= " f.fk_user_modif as fk_user_modification,";
907 $sql .= " f.fk_user_valid,";
908 $sql .= " f.fk_user_approve";
909 $sql .= " FROM ".MAIN_DB_PREFIX."expensereport as f";
910 $sql .= " WHERE f.rowid = ".((int) $id);
911 $sql .= " AND f.entity = ".$conf->entity;
912
913
914
915 $resql = $this->db->query($sql);
916 if ($resql) {
917 if ($this->db->num_rows($resql)) {
918 $obj = $this->db->fetch_object($resql);
919
920 $this->id = $obj->rowid;
921
922 $this->date_creation = $this->db->jdate($obj->datec);
923 $this->date_modification = $this->db->jdate($obj->date_modification);
924 $this->date_validation = $this->db->jdate($obj->datev);
925 $this->date_approbation = $this->db->jdate($obj->datea);
926
927 $this->user_creation_id = $obj->fk_user_author;
928 $this->user_creation_id = $obj->fk_user_creation;
929 $this->user_validation_id = $obj->fk_user_valid;
930 $this->user_modification_id = $obj->fk_user_modification;
931 $this->user_approve_id = $obj->fk_user_approve;
932 }
933 $this->db->free($resql);
934 } else {
935 dol_print_error($this->db);
936 }
937 }
938
939
940
948 public function initAsSpecimen()
949 {
950 global $user, $langs;
951
952 $now = dol_now();
953
954 // Initialise parameters
955 $this->id = 0;
956 $this->ref = 'SPECIMEN';
957 $this->specimen = 1;
958 $this->entity = 1;
959 $this->date_create = $now;
960 $this->date_debut = $now;
961 $this->date_fin = $now;
962 $this->date_valid = $now;
963 $this->date_approve = $now;
964
965 $type_fees_id = 2; // TF_TRIP
966
967 $this->status = 5;
968
969 $this->fk_user_author = $user->id;
970 $this->fk_user_validator = $user->id;
971 $this->fk_user_valid = $user->id;
972 $this->fk_user_approve = $user->id;
973
974 $this->note_private = 'Private note';
975 $this->note_public = 'SPECIMEN';
976 $nbp = min(1000, GETPOSTINT('nblines') ? GETPOSTINT('nblines') : 5); // We can force the nb of lines to test from command line (but not more than 1000)
977 $xnbp = 0;
978 while ($xnbp < $nbp) {
979 $line = new ExpenseReportLine($this->db);
980 $line->comments = $langs->trans("Comment")." ".$xnbp;
981 $line->date = ($now - 3600 * (1 + $xnbp));
982 $line->total_ht = 100;
983 $line->total_tva = 20;
984 $line->total_ttc = 120;
985 $line->qty = 1;
986 $line->vatrate = 20;
987 $line->value_unit = 120;
988 $line->fk_expensereport = 0;
989 $line->type_fees_code = 'TRA';
990 $line->fk_c_type_fees = $type_fees_id;
991
992 $line->projet_ref = 'ABC';
993
994 $this->lines[$xnbp] = $line;
995 $xnbp++;
996
997 $this->total_ht += $line->total_ht;
998 $this->total_tva += $line->total_tva;
999 $this->total_ttc += $line->total_ttc;
1000 }
1001
1002 return 1;
1003 }
1004
1005 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1013 public function fetch_line_by_project($projectid, $user)
1014 {
1015 // phpcs:enable
1016 global $langs;
1017
1018 $langs->load('trips');
1019
1020 if ($user->hasRight('expensereport', 'lire')) {
1021 $sql = "SELECT de.fk_expensereport, de.date, de.comments, de.total_ht, de.total_ttc";
1022 $sql .= " FROM ".MAIN_DB_PREFIX."expensereport_det as de";
1023 $sql .= " WHERE de.fk_projet = ".((int) $projectid);
1024
1025 dol_syslog(get_class($this)."::fetch", LOG_DEBUG);
1026 $result = $this->db->query($sql);
1027 if ($result) {
1028 $num = $this->db->num_rows($result);
1029 $i = 0;
1030 $total_HT = 0;
1031 $total_TTC = 0;
1032
1033 while ($i < $num) {
1034 $objp = $this->db->fetch_object($result);
1035
1036 $sql2 = "SELECT d.rowid, d.fk_user_author, d.ref, d.fk_statut as status";
1037 $sql2 .= " FROM ".MAIN_DB_PREFIX."expensereport as d";
1038 $sql2 .= " WHERE d.rowid = ".((int) $objp->fk_expensereport);
1039
1040 $result2 = $this->db->query($sql2);
1041 $obj = $this->db->fetch_object($result2);
1042
1043 $objp->fk_user_author = $obj->fk_user_author;
1044 $objp->ref = $obj->ref;
1045 $objp->fk_c_expensereport_status = $obj->status;
1046 $objp->rowid = $obj->rowid;
1047
1048 $total_HT += $objp->total_ht;
1049 $total_TTC += $objp->total_ttc;
1050 $author = new User($this->db);
1051 $author->fetch($objp->fk_user_author);
1052
1053 print '<tr>';
1054 print '<td><a href="'.DOL_URL_ROOT.'/expensereport/card.php?id='.$objp->rowid.'">'.$objp->ref_num.'</a></td>';
1055 print '<td class="center">'.dol_print_date($objp->date, 'day').'</td>';
1056 print '<td>'.$author->getNomUrl(1).'</td>';
1057 print '<td>'.$objp->comments.'</td>';
1058 print '<td class="right">'.price($objp->total_ht).'</td>';
1059 print '<td class="right">'.price($objp->total_ttc).'</td>';
1060 print '<td class="right">';
1061
1062 switch ($objp->fk_c_expensereport_status) {
1063 case 4:
1064 print img_picto($langs->trans('StatusOrderCanceled'), 'statut5');
1065 break;
1066 case 1:
1067 print $langs->trans('Draft').' '.img_picto($langs->trans('Draft'), 'statut0');
1068 break;
1069 case 2:
1070 print $langs->trans('TripForValid').' '.img_picto($langs->trans('TripForValid'), 'statut3');
1071 break;
1072 case 5:
1073 print $langs->trans('TripForPaid').' '.img_picto($langs->trans('TripForPaid'), 'statut3');
1074 break;
1075 case 6:
1076 print $langs->trans('TripPaid').' '.img_picto($langs->trans('TripPaid'), 'statut4');
1077 break;
1078 }
1079 /*
1080 if ($status==4) return img_picto($langs->trans('StatusOrderCanceled'),'statut5');
1081 if ($status==1) return img_picto($langs->trans('StatusOrderDraft'),'statut0');
1082 if ($status==2) return img_picto($langs->trans('StatusOrderValidated'),'statut1');
1083 if ($status==2) return img_picto($langs->trans('StatusOrderOnProcess'),'statut3');
1084 if ($status==5) return img_picto($langs->trans('StatusOrderToBill'),'statut4');
1085 if ($status==6) return img_picto($langs->trans('StatusOrderOnProcess'),'statut6');
1086 */
1087 print '</td>';
1088 print '</tr>';
1089
1090 $i++;
1091 }
1092
1093 print '<tr class="liste_total"><td colspan="4">'.$langs->trans("Number").': '.$i.'</td>';
1094 print '<td class="right" width="100">'.$langs->trans("TotalHT").' : '.price($total_HT).'</td>';
1095 print '<td class="right" width="100">'.$langs->trans("TotalTTC").' : '.price($total_TTC).'</td>';
1096 print '<td>&nbsp;</td>';
1097 print '</tr>';
1098 } else {
1099 $this->error = $this->db->lasterror();
1100 return -1;
1101 }
1102 }
1103
1104 return 0;
1105 }
1106
1107 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1113 public function fetch_lines()
1114 {
1115 // phpcs:enable
1116 $this->lines = array();
1117
1118 $sql = ' SELECT de.rowid, de.comments, de.qty, de.value_unit, de.date, de.rang,';
1119 $sql .= " de.".$this->fk_element.", de.fk_c_type_fees, de.fk_c_exp_tax_cat, de.fk_projet as fk_project,";
1120 $sql .= ' de.tva_tx, de.vat_src_code,';
1121 $sql .= ' de.localtax1_tx, de.localtax2_tx, de.localtax1_type, de.localtax2_type,';
1122 $sql .= ' de.fk_ecm_files,';
1123 $sql .= ' de.total_ht, de.total_tva, de.total_ttc,';
1124 $sql .= ' de.total_localtax1, de.total_localtax2, de.rule_warning_message,';
1125 $sql .= ' ctf.code as code_type_fees, ctf.label as label_type_fees, ctf.accountancy_code as accountancy_code_type_fees,';
1126 $sql .= ' p.ref as ref_projet, p.title as title_projet';
1127 $sql .= ' FROM '.MAIN_DB_PREFIX.$this->table_element_line.' as de';
1128 $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_type_fees as ctf ON de.fk_c_type_fees = ctf.id';
1129 $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'projet as p ON de.fk_projet = p.rowid';
1130 $sql .= " WHERE de.".$this->fk_element." = ".((int) $this->id);
1131 if (getDolGlobalString('EXPENSEREPORT_LINES_SORTED_BY_ROWID')) {
1132 $sql .= ' ORDER BY de.rang ASC, de.rowid ASC';
1133 } else {
1134 $sql .= ' ORDER BY de.rang ASC, de.date ASC';
1135 }
1136
1137 $resql = $this->db->query($sql);
1138 if ($resql) {
1139 $num = $this->db->num_rows($resql);
1140 $i = 0;
1141 while ($i < $num) {
1142 $objp = $this->db->fetch_object($resql);
1143
1144 $deplig = new ExpenseReportLine($this->db);
1145
1146 $deplig->rowid = $objp->rowid;
1147 $deplig->id = $objp->rowid;
1148 $deplig->comments = $objp->comments;
1149 $deplig->qty = $objp->qty;
1150 $deplig->value_unit = $objp->value_unit;
1151 $deplig->date = $objp->date;
1152 $deplig->dates = $this->db->jdate($objp->date);
1153
1154 $deplig->fk_expensereport = $objp->fk_expensereport;
1155 $deplig->fk_c_type_fees = $objp->fk_c_type_fees;
1156 $deplig->fk_c_exp_tax_cat = $objp->fk_c_exp_tax_cat;
1157 $deplig->fk_projet = $objp->fk_project; // deprecated
1158 $deplig->fk_project = $objp->fk_project;
1159 $deplig->fk_ecm_files = $objp->fk_ecm_files;
1160
1161 $deplig->total_ht = $objp->total_ht;
1162 $deplig->total_tva = $objp->total_tva;
1163 $deplig->total_ttc = $objp->total_ttc;
1164 $deplig->total_localtax1 = $objp->total_localtax1;
1165 $deplig->total_localtax2 = $objp->total_localtax2;
1166
1167 $deplig->type_fees_code = empty($objp->code_type_fees) ? 'TF_OTHER' : $objp->code_type_fees;
1168 $deplig->type_fees_libelle = $objp->label_type_fees;
1169 $deplig->type_fees_accountancy_code = $objp->accountancy_code_type_fees;
1170
1171 $deplig->tva_tx = $objp->tva_tx;
1172 $deplig->vatrate = $objp->tva_tx;
1173 $deplig->vat_src_code = $objp->vat_src_code;
1174 $deplig->localtax1_tx = $objp->localtax1_tx;
1175 $deplig->localtax2_tx = $objp->localtax2_tx;
1176 $deplig->localtax1_type = $objp->localtax1_type;
1177 $deplig->localtax2_type = $objp->localtax2_type;
1178
1179 $deplig->projet_ref = $objp->ref_projet;
1180 $deplig->projet_title = $objp->title_projet;
1181
1182 $deplig->rule_warning_message = $objp->rule_warning_message;
1183
1184 $deplig->rang = $objp->rang;
1185
1186 $this->lines[$i] = $deplig;
1187
1188 $i++;
1189 }
1190 $this->db->free($resql);
1191 return 1;
1192 } else {
1193 $this->error = $this->db->lasterror();
1194 dol_syslog(get_class($this)."::fetch_lines: Error ".$this->error, LOG_ERR);
1195 return -3;
1196 }
1197 }
1198
1199
1207 public function delete($user = null, $notrigger = 0)
1208 {
1209 global $conf;
1210 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
1211
1212 $error = 0;
1213
1214 $this->db->begin();
1215
1216 if (!$notrigger) {
1217 // Call trigger
1218 $result = $this->call_trigger('EXPENSE_REPORT_DELETE', $user);
1219 if ($result < 0) {
1220 $error++;
1221 }
1222 // End call triggers
1223 }
1224
1225 // Delete extrafields of lines and lines
1226 if (!$error && !empty($this->table_element_line)) {
1227 $tabletodelete = $this->table_element_line;
1228 //$sqlef = "DELETE FROM ".MAIN_DB_PREFIX.$tabletodelete."_extrafields WHERE fk_object IN (SELECT rowid FROM ".MAIN_DB_PREFIX.$tabletodelete." WHERE ".$this->fk_element." = ".((int) $this->id).")";
1229 $sql = "DELETE FROM ".MAIN_DB_PREFIX.$tabletodelete." WHERE ".$this->fk_element." = ".((int) $this->id);
1230 if (!$this->db->query($sql)) {
1231 $error++;
1232 $this->error = $this->db->lasterror();
1233 $this->errors[] = $this->error;
1234 dol_syslog(get_class($this)."::delete error ".$this->error, LOG_ERR);
1235 }
1236 }
1237
1238 if (!$error) {
1239 // Delete linked object
1240 $res = $this->deleteObjectLinked();
1241 if ($res < 0) {
1242 $error++;
1243 }
1244 }
1245
1246 if (!$error) {
1247 // Delete linked contacts
1248 $res = $this->delete_linked_contact();
1249 if ($res < 0) {
1250 $error++;
1251 }
1252 }
1253
1254 // Removed extrafields of object
1255 if (!$error) {
1256 $result = $this->deleteExtraFields();
1257 if ($result < 0) {
1258 $error++;
1259 dol_syslog(get_class($this)."::delete error ".$this->error, LOG_ERR);
1260 }
1261 }
1262
1263 // Delete main record
1264 if (!$error) {
1265 $sql = "DELETE FROM ".MAIN_DB_PREFIX.$this->table_element." WHERE rowid = ".((int) $this->id);
1266 $res = $this->db->query($sql);
1267 if (!$res) {
1268 $error++;
1269 $this->error = $this->db->lasterror();
1270 $this->errors[] = $this->error;
1271 dol_syslog(get_class($this)."::delete error ".$this->error, LOG_ERR);
1272 }
1273 }
1274
1275 // Delete record into ECM index and physically
1276 if (!$error) {
1277 $res = $this->deleteEcmFiles(0); // Deleting files physically is done later with the dol_delete_dir_recursive
1278 $res = $this->deleteEcmFiles(1); // Deleting files physically is done later with the dol_delete_dir_recursive
1279 if (!$res) {
1280 $error++;
1281 }
1282 }
1283
1284 if (!$error) {
1285 // We remove directory
1286 $ref = dol_sanitizeFileName($this->ref);
1287 if ($conf->expensereport->multidir_output[$this->entity] && !empty($this->ref)) {
1288 $dir = $conf->expensereport->multidir_output[$this->entity]."/".$ref;
1289 $file = $dir."/".$ref.".pdf";
1290 if (file_exists($file)) {
1291 dol_delete_preview($this);
1292
1293 if (!dol_delete_file($file, 0, 0, 0, $this)) {
1294 $this->error = 'ErrorFailToDeleteFile';
1295 $this->errors[] = $this->error;
1296 $this->db->rollback();
1297 return 0;
1298 }
1299 }
1300 if (file_exists($dir)) {
1301 $res = @dol_delete_dir_recursive($dir);
1302 if (!$res) {
1303 $this->error = 'ErrorFailToDeleteDir';
1304 $this->errors[] = $this->error;
1305 $this->db->rollback();
1306 return 0;
1307 }
1308 }
1309 }
1310 }
1311
1312 if (!$error) {
1313 dol_syslog(get_class($this)."::delete ".$this->id." by ".$user->id, LOG_DEBUG);
1314 $this->db->commit();
1315 return 1;
1316 } else {
1317 $this->db->rollback();
1318 return -1;
1319 }
1320 }
1321
1329 public function setValidate($fuser, $notrigger = 0)
1330 {
1331 global $conf, $langs, $user;
1332
1333 $error = 0;
1334 $now = dol_now();
1335
1336 // Protection
1337 if ($this->status == self::STATUS_VALIDATED) {
1338 dol_syslog(get_class($this)."::valid action abandoned: already validated", LOG_WARNING);
1339 return 0;
1340 }
1341
1342 $this->date_valid = $now; // Required for the getNextNum later.
1343
1344 // Define new ref
1345 if (!$error && (preg_match('/^[\‍(]?PROV/i', $this->ref) || empty($this->ref))) { // empty should not happened, but when it occurs, the test save life
1346 $num = $this->getNextNumRef();
1347 } else {
1348 $num = $this->ref;
1349 }
1350 if (empty($num) || $num < 0) {
1351 return -1;
1352 }
1353
1354 $this->newref = dol_sanitizeFileName($num);
1355
1356 $this->db->begin();
1357
1358 // Validate
1359 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
1360 $sql .= " SET ref = '".$this->db->escape($num)."',";
1361 $sql .= " fk_statut = ".self::STATUS_VALIDATED.",";
1362 $sql .= " date_valid = '".$this->db->idate($this->date_valid)."',";
1363 $sql .= " fk_user_valid = ".((int) $user->id);
1364 $sql .= " WHERE rowid = ".((int) $this->id);
1365
1366 $resql = $this->db->query($sql);
1367 if ($resql) {
1368 if (!$error && !$notrigger) {
1369 // Call trigger
1370 $result = $this->call_trigger('EXPENSE_REPORT_VALIDATE', $fuser);
1371 if ($result < 0) {
1372 $error++;
1373 }
1374 // End call triggers
1375 }
1376
1377 if (!$error) {
1378 $this->oldref = $this->ref;
1379
1380 // Rename directory if dir was a temporary ref
1381 if (preg_match('/^[\‍(]?PROV/i', $this->ref)) {
1382 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
1383
1384 // Now we rename also files into index
1385 $sql = 'UPDATE '.MAIN_DB_PREFIX."ecm_files set filename = CONCAT('".$this->db->escape($this->newref)."', SUBSTR(filename, ".(strlen($this->ref) + 1).")), filepath = 'expensereport/".$this->db->escape($this->newref)."'";
1386 $sql .= " WHERE filename LIKE '".$this->db->escape($this->ref)."%' AND filepath = 'expensereport/".$this->db->escape($this->ref)."' AND entity = ".((int) $this->entity);
1387 $resql = $this->db->query($sql);
1388 if (!$resql) {
1389 $error++;
1390 $this->error = $this->db->lasterror();
1391 }
1392 $sql = 'UPDATE '.MAIN_DB_PREFIX."ecm_files set filepath = 'expensereport/".$this->db->escape($this->newref)."'";
1393 $sql .= " WHERE filepath = 'expensereport/".$this->db->escape($this->ref)."' and entity = ".$conf->entity;
1394 $resql = $this->db->query($sql);
1395 if (!$resql) {
1396 $error++;
1397 $this->error = $this->db->lasterror();
1398 }
1399
1400 // We rename directory ($this->ref = old ref, $num = new ref) in order not to lose the attachments
1401 $oldref = dol_sanitizeFileName($this->ref);
1402 $newref = dol_sanitizeFileName($num);
1403 $dirsource = $conf->expensereport->multidir_output[$this->entity].'/'.$oldref;
1404 $dirdest = $conf->expensereport->multidir_output[$this->entity].'/'.$newref;
1405 if (!$error && file_exists($dirsource)) {
1406 dol_syslog(get_class($this)."::setValidate() rename dir ".$dirsource." into ".$dirdest);
1407
1408 if (@rename($dirsource, $dirdest)) {
1409 dol_syslog("Rename ok");
1410 // Rename docs starting with $oldref with $newref
1411 $listoffiles = dol_dir_list($dirdest, 'files', 1, '^'.preg_quote($oldref, '/'));
1412 foreach ($listoffiles as $fileentry) {
1413 $dirsource = $fileentry['name'];
1414 $dirdest = preg_replace('/^'.preg_quote($oldref, '/').'/', $newref, $dirsource);
1415 $dirsource = $fileentry['path'].'/'.$dirsource;
1416 $dirdest = $fileentry['path'].'/'.$dirdest;
1417 @rename($dirsource, $dirdest);
1418 }
1419 }
1420 }
1421 }
1422 }
1423
1424 // Set new ref and current status
1425 if (!$error) {
1426 $this->ref = $num;
1428 }
1429
1430 if (empty($error)) {
1431 $this->db->commit();
1432 return 1;
1433 } else {
1434 $this->db->rollback();
1435 $this->error = $this->db->error();
1436 return -2;
1437 }
1438 } else {
1439 $this->db->rollback();
1440 $this->error = $this->db->lasterror();
1441 return -1;
1442 }
1443 }
1444
1445 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1452 public function set_save_from_refuse($fuser)
1453 {
1454 // phpcs:enable
1455 // Sélection de la date de début de la NDF
1456 $sql = 'SELECT date_debut';
1457 $sql .= ' FROM '.MAIN_DB_PREFIX.$this->table_element;
1458 $sql .= " WHERE rowid = ".((int) $this->id);
1459
1460 $result = $this->db->query($sql);
1461
1462 $objp = $this->db->fetch_object($result);
1463
1464 $this->date_debut = $this->db->jdate($objp->date_debut);
1465
1466 if ($this->status != self::STATUS_VALIDATED) {
1467 $sql = 'UPDATE '.MAIN_DB_PREFIX.$this->table_element;
1468 $sql .= " SET fk_statut = ".self::STATUS_VALIDATED;
1469 $sql .= " WHERE rowid = ".((int) $this->id);
1470
1471 dol_syslog(get_class($this)."::set_save_from_refuse", LOG_DEBUG);
1472
1473 if ($this->db->query($sql)) {
1474 return 1;
1475 } else {
1476 $this->error = $this->db->lasterror();
1477 return -1;
1478 }
1479 } else {
1480 dol_syslog(get_class($this)."::set_save_from_refuse expensereport already with save status", LOG_WARNING);
1481 }
1482
1483 return 0;
1484 }
1485
1493 public function setApproved($fuser, $notrigger = 0)
1494 {
1495 $now = dol_now();
1496 $error = 0;
1497
1498 // date approval
1499 $this->date_approve = $now;
1500 if ($this->status != self::STATUS_APPROVED) {
1501 $this->db->begin();
1502
1503 $sql = 'UPDATE '.MAIN_DB_PREFIX.$this->table_element;
1504 $sql .= " SET ref = '".$this->db->escape($this->ref)."', fk_statut = ".self::STATUS_APPROVED.", fk_user_approve = ".((int) $fuser->id).",";
1505 $sql .= " date_approve='".$this->db->idate($this->date_approve)."'";
1506 $sql .= " WHERE rowid = ".((int) $this->id);
1507 if ($this->db->query($sql)) {
1508 if (!$notrigger) {
1509 // Call trigger
1510 $result = $this->call_trigger('EXPENSE_REPORT_APPROVE', $fuser);
1511
1512 if ($result < 0) {
1513 $error++;
1514 }
1515 // End call triggers
1516 }
1517
1518 if (empty($error)) {
1519 $this->db->commit();
1520 return 1;
1521 } else {
1522 $this->db->rollback();
1523 $this->error = $this->db->error();
1524 return -2;
1525 }
1526 } else {
1527 $this->db->rollback();
1528 $this->error = $this->db->lasterror();
1529 return -1;
1530 }
1531 } else {
1532 dol_syslog(get_class($this)."::setApproved expensereport already with approve status", LOG_WARNING);
1533 }
1534
1535 return 0;
1536 }
1537
1546 public function setDeny($fuser, $details, $notrigger = 0)
1547 {
1548 $now = dol_now();
1549 $error = 0;
1550
1551 // date de refus
1552 if ($this->status != self::STATUS_REFUSED) {
1553 $sql = 'UPDATE '.MAIN_DB_PREFIX.$this->table_element;
1554 $sql .= " SET ref = '".$this->db->escape($this->ref)."', fk_statut = ".self::STATUS_REFUSED.", fk_user_refuse = ".((int) $fuser->id).",";
1555 $sql .= " date_refuse='".$this->db->idate($now)."',";
1556 $sql .= " detail_refuse='".$this->db->escape($details)."',";
1557 $sql .= " fk_user_approve = NULL";
1558 $sql .= " WHERE rowid = ".((int) $this->id);
1559 if ($this->db->query($sql)) {
1560 $this->fk_statut = 99; // deprecated
1561 $this->status = 99;
1562 $this->fk_user_refuse = $fuser->id;
1563 $this->detail_refuse = $details;
1564 $this->date_refuse = $now;
1565
1566 if (!$notrigger) {
1567 // Call trigger
1568 $result = $this->call_trigger('EXPENSE_REPORT_DENY', $fuser);
1569
1570 if ($result < 0) {
1571 $error++;
1572 }
1573 // End call triggers
1574 }
1575
1576 if (empty($error)) {
1577 $this->db->commit();
1578 return 1;
1579 } else {
1580 $this->db->rollback();
1581 $this->error = $this->db->error();
1582 return -2;
1583 }
1584 } else {
1585 $this->db->rollback();
1586 $this->error = $this->db->lasterror();
1587 return -1;
1588 }
1589 } else {
1590 dol_syslog(get_class($this)."::setDeny expensereport already with refuse status", LOG_WARNING);
1591 }
1592
1593 return 0;
1594 }
1595
1596 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1606 public function set_unpaid($fuser, $notrigger = 0)
1607 {
1608 // phpcs:enable
1609 dol_syslog(get_class($this)."::set_unpaid is deprecated, use setUnpaid instead", LOG_NOTICE);
1610 return $this->setUnpaid($fuser, $notrigger);
1611 }
1612
1620 public function setUnpaid($fuser, $notrigger = 0)
1621 {
1622 $error = 0;
1623
1624 if ($this->paid) {
1625 $this->db->begin();
1626
1627 $sql = 'UPDATE '.MAIN_DB_PREFIX.$this->table_element;
1628 $sql .= " SET paid = 0, fk_statut = ".self::STATUS_APPROVED;
1629 $sql .= " WHERE rowid = ".((int) $this->id);
1630
1631 dol_syslog(get_class($this)."::set_unpaid", LOG_DEBUG);
1632
1633 if ($this->db->query($sql)) {
1634 if (!$notrigger) {
1635 // Call trigger
1636 $result = $this->call_trigger('EXPENSE_REPORT_UNPAID', $fuser);
1637
1638 if ($result < 0) {
1639 $error++;
1640 }
1641 // End call triggers
1642 }
1643
1644 if (empty($error)) {
1645 $this->db->commit();
1646 return 1;
1647 } else {
1648 $this->db->rollback();
1649 $this->error = $this->db->error();
1650 return -2;
1651 }
1652 } else {
1653 $this->db->rollback();
1654 $this->error = $this->db->error();
1655 return -1;
1656 }
1657 } else {
1658 dol_syslog(get_class($this)."::set_unpaid expensereport already with unpaid status", LOG_WARNING);
1659 }
1660
1661 return 0;
1662 }
1663
1664 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1673 public function set_cancel($fuser, $detail, $notrigger = 0)
1674 {
1675 // phpcs:enable
1676 $error = 0;
1677 $this->date_cancel = $this->db->idate(dol_now());
1678 if ($this->status != self::STATUS_CANCELED) {
1679 $this->db->begin();
1680
1681 $sql = 'UPDATE '.MAIN_DB_PREFIX.$this->table_element;
1682 $sql .= " SET fk_statut = ".self::STATUS_CANCELED.", fk_user_cancel = ".((int) $fuser->id);
1683 $sql .= ", date_cancel='".$this->db->idate($this->date_cancel)."'";
1684 $sql .= ", detail_cancel='".$this->db->escape($detail)."'";
1685 $sql .= " WHERE rowid = ".((int) $this->id);
1686
1687 dol_syslog(get_class($this)."::set_cancel", LOG_DEBUG);
1688
1689 if ($this->db->query($sql)) {
1690 if (!$notrigger) {
1691 // Call trigger
1692 $result = $this->call_trigger('EXPENSE_REPORT_CANCEL', $fuser);
1693
1694 if ($result < 0) {
1695 $error++;
1696 }
1697 // End call triggers
1698 }
1699
1700 if (empty($error)) {
1701 $this->db->commit();
1702 return 1;
1703 } else {
1704 $this->db->rollback();
1705 $this->error = $this->db->error();
1706 return -2;
1707 }
1708 } else {
1709 $this->db->rollback();
1710 $this->error = $this->db->error();
1711 return -1;
1712 }
1713 } else {
1714 dol_syslog(get_class($this)."::set_cancel expensereport already with cancel status", LOG_WARNING);
1715 }
1716 return 0;
1717 }
1718
1724 public function getNextNumRef()
1725 {
1726 global $langs, $conf;
1727 $langs->load("trips");
1728
1729 if (getDolGlobalString('EXPENSEREPORT_ADDON')) {
1730 $mybool = false;
1731
1732 $file = getDolGlobalString('EXPENSEREPORT_ADDON') . ".php";
1733 $classname = getDolGlobalString('EXPENSEREPORT_ADDON');
1734
1735 // Include file with class
1736 $dirmodels = array_merge(array('/'), (array) $conf->modules_parts['models']);
1737 foreach ($dirmodels as $reldir) {
1738 $dir = dol_buildpath($reldir."core/modules/expensereport/");
1739
1740 // Load file with numbering class (if found)
1741 $mybool = ((bool) @include_once $dir.$file) || $mybool;
1742 }
1743
1744 if (!$mybool) {
1745 dol_print_error(null, "Failed to include file ".$file);
1746 return '';
1747 }
1748
1749 $obj = new $classname();
1750 '@phan-var-force ModeleNumRefExpenseReport $obj';
1751 $numref = $obj->getNextValue($this);
1752
1753 if ($numref != "") {
1754 return $numref;
1755 } else {
1756 $this->error = $obj->error;
1757 $this->errors = $obj->errors;
1758 //dol_print_error($this->db,get_class($this)."::getNextNumRef ".$obj->error);
1759 return -1;
1760 }
1761 } else {
1762 $this->error = "Error_EXPENSEREPORT_ADDON_NotDefined";
1763 return -2;
1764 }
1765 }
1766
1773 public function getTooltipContentArray($params)
1774 {
1775 global $conf, $langs;
1776
1777 $langs->load('trips');
1778
1779 $nofetch = !empty($params['nofetch']);
1780 $moretitle = $params['moretitle'] ?? '';
1781
1782 $datas = array();
1783 $datas['picto'] = img_picto('', $this->picto).' <u class="paddingrightonly">'.$langs->trans("ExpenseReport").'</u>';
1784 if (isset($this->status)) {
1785 $datas['picto'] .= ' '.$this->getLibStatut(5);
1786 }
1787 if ($moretitle) {
1788 $datas['picto'] .= ' - '.$moretitle;
1789 }
1790 if (!empty($this->ref)) {
1791 $datas['ref'] = '<br><b>'.$langs->trans('Ref').':</b> '.$this->ref;
1792 }
1793 if (!empty($this->total_ht)) {
1794 $datas['total_ht'] = '<br><b>'.$langs->trans('AmountHT').':</b> '.price($this->total_ht, 0, $langs, 0, -1, -1, $conf->currency);
1795 }
1796 if (!empty($this->total_tva)) {
1797 $datas['total_tva'] = '<br><b>'.$langs->trans('VAT').':</b> '.price($this->total_tva, 0, $langs, 0, -1, -1, $conf->currency);
1798 }
1799 if (!empty($this->total_ttc)) {
1800 $datas['total_ttc'] = '<br><b>'.$langs->trans('AmountTTC').':</b> '.price($this->total_ttc, 0, $langs, 0, -1, -1, $conf->currency);
1801 }
1802
1803 return $datas;
1804 }
1805
1818 public function getNomUrl($withpicto = 0, $option = '', $max = 0, $short = 0, $moretitle = '', $notooltip = 0, $save_lastsearch_value = -1)
1819 {
1820 global $langs, $hookmanager;
1821
1822 $result = '';
1823
1824 $url = DOL_URL_ROOT.'/expensereport/card.php?id='.$this->id;
1825
1826 if ($short) {
1827 return $url;
1828 }
1829
1830 $params = [
1831 'id' => $this->id,
1832 'objecttype' => $this->element,
1833 'option' => $option,
1834 'moretitle' => $moretitle,
1835 'nofetch' => 1,
1836 ];
1837 $classfortooltip = 'classfortooltip';
1838 $dataparams = '';
1839 if (getDolGlobalInt('MAIN_ENABLE_AJAX_TOOLTIP')) {
1840 $classfortooltip = 'classforajaxtooltip';
1841 $dataparams = ' data-params="'.dol_escape_htmltag(json_encode($params)).'"';
1842 $label = '';
1843 } else {
1844 $label = implode($this->getTooltipContentArray($params));
1845 }
1846
1847 if ($option != 'nolink') {
1848 // Add param to save lastsearch_values or not
1849 $add_save_lastsearch_values = ($save_lastsearch_value == 1 ? 1 : 0);
1850 if ($save_lastsearch_value == -1 && isset($_SERVER["PHP_SELF"]) && preg_match('/list\.php/', $_SERVER["PHP_SELF"])) {
1851 $add_save_lastsearch_values = 1;
1852 }
1853 if ($add_save_lastsearch_values) {
1854 $url .= '&save_lastsearch_values=1';
1855 }
1856 }
1857
1858 $ref = $this->ref;
1859 if (empty($ref)) {
1860 $ref = $this->id;
1861 }
1862
1863 $linkclose = '';
1864 if (empty($notooltip)) {
1865 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
1866 $label = $langs->trans("ShowExpenseReport");
1867 $linkclose .= ' alt="'.dolPrintHTMLForAttribute($label).'"';
1868 }
1869 $linkclose .= ($label ? ' title="'.dolPrintHTMLForAttribute($label).'"' : ' title="tocomplete"');
1870 $linkclose .= $dataparams.' class="'.$classfortooltip.'"';
1871 }
1872
1873 $linkstart = '<a href="'.$url.'"';
1874 $linkstart .= $linkclose.'>';
1875 $linkend = '</a>';
1876
1877 $result .= $linkstart;
1878 if ($withpicto) {
1879 $result .= img_object(($notooltip ? '' : $label), ($this->picto ? $this->picto : 'generic'), ($notooltip ? (($withpicto != 2) ? 'class="paddingright"' : '') : 'class="'.(($withpicto != 2) ? 'paddingright ' : '').'"'), 0, 0, $notooltip ? 0 : 1);
1880 }
1881 if ($withpicto != 2) {
1882 $result .= ($max ? dol_trunc($ref, $max) : $ref);
1883 }
1884 $result .= $linkend;
1885
1886 global $action;
1887 $hookmanager->initHooks(array($this->element . 'dao'));
1888 $parameters = array('id' => $this->id, 'getnomurl' => &$result);
1889 $reshook = $hookmanager->executeHooks('getNomUrl', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
1890 if ($reshook > 0) {
1891 $result = $hookmanager->resPrint;
1892 } else {
1893 $result .= $hookmanager->resPrint;
1894 }
1895 return $result;
1896 }
1897
1898 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1906 public function update_totaux_add($ligne_total_ht, $ligne_total_tva)
1907 {
1908 // phpcs:enable
1909 $this->total_ht += (float) $ligne_total_ht;
1910 $this->total_tva += (float) $ligne_total_tva;
1911 $this->total_ttc += $this->total_tva;
1912
1913 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element." SET";
1914 $sql .= " total_ht = ".((float) $this->total_ht);
1915 $sql .= " , total_ttc = ".((float) $this->total_ttc);
1916 $sql .= " , total_tva = ".((float) $this->total_tva);
1917 $sql .= " WHERE rowid = ".((int) $this->id);
1918
1919 $result = $this->db->query($sql);
1920 if ($result) {
1921 return 1;
1922 } else {
1923 $this->error = $this->db->error();
1924 return -1;
1925 }
1926 }
1927
1943 public function addline($qty = 0, $up = 0, $fk_c_type_fees = 0, $vatrate = 0, $date = '', $comments = '', $fk_project = 0, $fk_c_exp_tax_cat = 0, $type = 0, $fk_ecm_files = 0)
1944 {
1945 global $langs, $mysoc;
1946
1947 dol_syslog(get_class($this)."::addline qty=$qty, up=$up, fk_c_type_fees=$fk_c_type_fees, vatrate=$vatrate, date=$date, fk_project=$fk_project, type=$type, comments=$comments", LOG_DEBUG);
1948
1949 if ($this->status == self::STATUS_DRAFT || $this->status == self::STATUS_REFUSED) {
1950 if (empty($qty)) {
1951 $qty = 0;
1952 }
1953 if (empty($fk_c_type_fees) || $fk_c_type_fees < 0) {
1954 $fk_c_type_fees = 0;
1955 }
1956 if (empty($fk_c_exp_tax_cat) || $fk_c_exp_tax_cat < 0) {
1957 $fk_c_exp_tax_cat = 0;
1958 }
1959 if (empty($vatrate) || $vatrate < 0) {
1960 $vatrate = 0;
1961 }
1962 if (empty($date)) {
1963 $date = '';
1964 }
1965 if (empty($fk_project)) {
1966 $fk_project = 0;
1967 }
1968
1969 $qty = (float) price2num($qty);
1970 if (!preg_match('/\s*\‍((.*)\‍)/', $vatrate)) {
1971 $vatrate = price2num($vatrate); // $txtva can have format '5.0 (XXX)' or '5'
1972 }
1973 $up = price2num($up);
1974
1975 $this->db->begin();
1976
1977 $this->line = new ExpenseReportLine($this->db);
1978
1979 // We don't know seller and buyer for expense reports
1980 $seller = $mysoc; // We use same than current company (expense report are often done in same country)
1981 $seller->tva_assuj = 1; // Most seller uses vat
1982 $buyer = new Societe($this->db);
1983
1984 $localtaxes_type = getLocalTaxesFromRate($vatrate, 0, $buyer, $seller);
1985
1986 $vat_src_code = '';
1987 $reg = array();
1988 if (preg_match('/\s*\‍((.*)\‍)/', $vatrate, $reg)) {
1989 $vat_src_code = $reg[1];
1990 $vatrate = preg_replace('/\s*\‍(.*\‍)/', '', $vatrate); // Remove code into vatrate.
1991 }
1992 $vatrate = preg_replace('/\*/', '', $vatrate);
1993
1994 $tmp = calcul_price_total($qty, $up, 0, (float) price2num($vatrate), -1, -1, 0, 'TTC', 0, $type, $seller, $localtaxes_type);
1995
1996 $this->line->value_unit = $up;
1997
1998 $this->line->vat_src_code = $vat_src_code;
1999 $this->line->vatrate = price2num($vatrate);
2000 $this->line->localtax1_tx = $localtaxes_type[1];
2001 $this->line->localtax2_tx = $localtaxes_type[3];
2002 $this->line->localtax1_type = $localtaxes_type[0];
2003 $this->line->localtax2_type = $localtaxes_type[2];
2004
2005 $this->line->total_ttc = (float) $tmp[2];
2006 $this->line->total_ht = (float) $tmp[0];
2007 $this->line->total_tva = (float) $tmp[1];
2008 $this->line->total_localtax1 = (float) $tmp[9];
2009 $this->line->total_localtax2 = (float) $tmp[10];
2010
2011 $this->line->fk_expensereport = $this->id;
2012 $this->line->qty = $qty;
2013 $this->line->date = $date;
2014 $this->line->fk_c_type_fees = $fk_c_type_fees;
2015 $this->line->fk_c_exp_tax_cat = $fk_c_exp_tax_cat;
2016 $this->line->comments = $comments;
2017 $this->line->fk_projet = $fk_project; // deprecated
2018 $this->line->fk_project = $fk_project;
2019
2020 $this->line->fk_ecm_files = $fk_ecm_files;
2021
2022 $this->applyOffset();
2023 $this->checkRules($type, $seller);
2024
2025 $result = $this->line->insert(0, true);
2026 if ($result > 0) {
2027 $result = $this->update_price(1); // This method is designed to add line from user input so total calculation must be done using 'auto' mode.
2028 if ($result > 0) {
2029 $this->db->commit();
2030 return $this->line->id;
2031 } else {
2032 $this->db->rollback();
2033 return -1;
2034 }
2035 } else {
2036 $this->error = $this->line->error;
2037 dol_syslog(get_class($this)."::addline error=".$this->error, LOG_ERR);
2038 $this->db->rollback();
2039 return -2;
2040 }
2041 } else {
2042 dol_syslog(get_class($this)."::addline status of expense report must be Draft to allow use of ->addline()", LOG_ERR);
2043 $this->error = 'ErrorExpenseNotDraftAndNotRefused';
2044 return -3;
2045 }
2046 }
2047
2055 public function checkRules($type = 0, $seller = '')
2056 {
2057 global $conf, $db, $langs, $mysoc;
2058
2059 $langs->load('trips');
2060
2061 // We don't know seller and buyer for expense reports
2062 if (!is_object($seller)) {
2063 $seller = $mysoc; // We use same than current company (expense report are often done in same country)
2064 $seller->tva_assuj = 1; // Most seller uses vat
2065 }
2066
2067 $expensereportrule = new ExpenseReportRule($db);
2068 $rulestocheck = $expensereportrule->getAllRule($this->line->fk_c_type_fees, $this->line->date, $this->fk_user_author);
2069
2070 $violation = 0;
2071 $rule_warning_message_tab = array();
2072
2073 $current_total_ttc = $this->line->total_ttc;
2074 $new_current_total_ttc = $this->line->total_ttc;
2075
2076 // check if one is violated
2077 foreach ($rulestocheck as $rule) {
2078 if (in_array($rule->code_expense_rules_type, array('EX_DAY', 'EX_MON', 'EX_YEA'))) {
2079 $amount_to_test = $this->line->getExpAmount($rule, $this->fk_user_author, $rule->code_expense_rules_type);
2080 } else {
2081 $amount_to_test = $current_total_ttc; // EX_EXP
2082 }
2083
2084 $amount_to_test = $amount_to_test - $current_total_ttc + $new_current_total_ttc; // if amount as been modified by a previous rule
2085
2086 if ($amount_to_test > $rule->amount) {
2087 $violation++;
2088
2089 if ($rule->restrictive) {
2090 $this->error = 'ExpenseReportConstraintViolationError';
2091 $this->errors[] = $this->error;
2092
2093 $new_current_total_ttc -= $amount_to_test - $rule->amount; // ex, entered 16€, limit 12€, subtracts 4€;
2094 $rule_warning_message_tab[] = $langs->trans('ExpenseReportConstraintViolationError', $rule->id, price($amount_to_test, 0, $langs, 1, -1, -1, $conf->currency), price($rule->amount, 0, $langs, 1, -1, -1, $conf->currency));
2095 } else {
2096 $this->error = 'ExpenseReportConstraintViolationWarning';
2097 $this->errors[] = $this->error;
2098
2099 $rule_warning_message_tab[] = $langs->trans('ExpenseReportConstraintViolationWarning', $rule->id, price($amount_to_test, 0, $langs, 1, -1, -1, $conf->currency), price($rule->amount, 0, $langs, 1, -1, -1, $conf->currency));
2100 }
2101
2102 // No break, we should test if another rule is violated
2103 }
2104 }
2105
2106 $this->line->rule_warning_message = implode('\n', $rule_warning_message_tab);
2107
2108 if ($violation > 0) {
2109 $tmp = calcul_price_total($this->line->qty, $new_current_total_ttc / $this->line->qty, 0, $this->line->vatrate, 0, 0, 0, 'TTC', 0, $type, $seller);
2110
2111 $this->line->value_unit = $tmp[5];
2112 $this->line->total_ttc = (float) $tmp[2];
2113 $this->line->total_ht = (float) $tmp[0];
2114 $this->line->total_tva = (float) $tmp[1];
2115 $this->line->total_localtax1 = (float) $tmp[9];
2116 $this->line->total_localtax2 = (float) $tmp[10];
2117
2118 return false;
2119 } else {
2120 return true;
2121 }
2122 }
2123
2131 public function applyOffset($type = 0, $seller = '')
2132 {
2133 global $mysoc;
2134
2135 if (!getDolGlobalString('MAIN_USE_EXPENSE_IK')) {
2136 return false;
2137 }
2138
2139 $userauthor = new User($this->db);
2140 if ($userauthor->fetch($this->fk_user_author) <= 0) {
2141 $this->error = 'ErrorCantFetchUser';
2142 $this->errors[] = 'ErrorCantFetchUser';
2143 return false;
2144 }
2145
2146 // We don't know seller and buyer for expense reports
2147 if (!is_object($seller)) {
2148 $seller = $mysoc; // We use same than current company (expense report are often done in same country)
2149 $seller->tva_assuj = 1; // Most seller uses vat
2150 }
2151
2152 $expenseik = new ExpenseReportIk($this->db);
2153 $range = $expenseik->getRangeByUser($userauthor, $this->line->fk_c_exp_tax_cat);
2154
2155 if (empty($range)) {
2156 $this->error = 'ErrorNoRangeAvailable';
2157 $this->errors[] = 'ErrorNoRangeAvailable';
2158 return false;
2159 }
2160
2161 if (getDolGlobalString('MAIN_EXPENSE_APPLY_ENTIRE_OFFSET')) {
2162 $ikoffset = $range->ikoffset;
2163 } else {
2164 $ikoffset = $range->ikoffset / 12; // The amount of offset is a global value for the year
2165 }
2166
2167 // Test if ikoffset has been applied for the current month
2168 if (!$this->offsetAlreadyGiven()) {
2169 $new_up = $range->coef + ($ikoffset / $this->line->qty);
2170 $tmp = calcul_price_total($this->line->qty, $new_up, 0, $this->line->vatrate, 0, 0, 0, 'TTC', 0, $type, $seller);
2171
2172 $this->line->value_unit = $tmp[5];
2173 $this->line->total_ttc = (float) $tmp[2];
2174 $this->line->total_ht = (float) $tmp[0];
2175 $this->line->total_tva = (float) $tmp[1];
2176 $this->line->total_localtax1 = (float) $tmp[9];
2177 $this->line->total_localtax2 = (float) $tmp[10];
2178
2179 return true;
2180 }
2181
2182 return false;
2183 }
2184
2190 public function offsetAlreadyGiven()
2191 {
2192 $sql = 'SELECT e.rowid FROM '.MAIN_DB_PREFIX.'expensereport e';
2193 $sql .= " INNER JOIN ".MAIN_DB_PREFIX."expensereport_det d ON (e.rowid = d.fk_expensereport)";
2194 $sql .= " INNER JOIN ".MAIN_DB_PREFIX."c_type_fees f ON (d.fk_c_type_fees = f.id AND f.code = 'EX_KME')";
2195 $sql .= " WHERE e.fk_user_author = ".(int) $this->fk_user_author;
2196 $sql .= " AND YEAR(d.date) = '".dol_print_date($this->line->date, '%Y')."' AND MONTH(d.date) = '".dol_print_date($this->line->date, '%m')."'";
2197 if (!empty($this->line->id)) {
2198 $sql .= ' AND d.rowid <> '.((int) $this->line->id);
2199 }
2200
2201 dol_syslog(get_class($this)."::offsetAlreadyGiven");
2202 $resql = $this->db->query($sql);
2203 if ($resql) {
2204 $num = $this->db->num_rows($resql);
2205 if ($num > 0) {
2206 return true;
2207 }
2208 } else {
2209 dol_print_error($this->db);
2210 }
2211
2212 return false;
2213 }
2214
2232 public function updateline($rowid, $type_fees_id, $projet_id, $vatrate, $comments, $qty, $value_unit, $date, $expensereport_id, $fk_c_exp_tax_cat = 0, $fk_ecm_files = 0, $notrigger = 0)
2233 {
2234 global $user, $mysoc;
2235
2236 if ($this->status == self::STATUS_DRAFT || $this->status == self::STATUS_REFUSED) {
2237 $this->db->begin();
2238
2239 $error = 0;
2240 $type = 0; // TODO What if type is service ?
2241
2242 // We don't know seller and buyer for expense reports
2243 $seller = $mysoc; // We use same than current company (expense report are often done in same country)
2244 $seller->tva_assuj = 1; // Most seller uses vat
2245 $seller->localtax1_assuj = $mysoc->localtax1_assuj; // We don't know, we reuse the state of company
2246 $seller->localtax2_assuj = $mysoc->localtax1_assuj; // We don't know, we reuse the state of company
2247 $buyer = new Societe($this->db);
2248
2249 $localtaxes_type = getLocalTaxesFromRate($vatrate, 0, $buyer, $seller);
2250
2251 // Clean vat code
2252 $reg = array();
2253 $vat_src_code = '';
2254 if (preg_match('/\‍((.*)\‍)/', (string) $vatrate, $reg)) {
2255 $vat_src_code = $reg[1];
2256 $vatrate = preg_replace('/\s*\‍(.*\‍)/', '', (string) $vatrate); // Remove code into vatrate.
2257 }
2258 $vatrate = preg_replace('/\*/', '', $vatrate);
2259
2260 $tmp = calcul_price_total($qty, $value_unit, 0, (float) price2num($vatrate), -1, -1, 0, 'TTC', 0, $type, $seller, $localtaxes_type);
2261 // calcul total of line
2262 // $total_ttc = price2num($qty*$value_unit, 'MT');
2263
2264 $tx_tva = 1 + (float) $vatrate / 100;
2265
2266 $this->line = new ExpenseReportLine($this->db);
2267 $this->line->comments = $comments;
2268 $this->line->qty = $qty;
2269 $this->line->value_unit = $value_unit;
2270 $this->line->date = $date;
2271
2272 $this->line->fk_expensereport = $expensereport_id;
2273 $this->line->fk_c_type_fees = $type_fees_id;
2274 $this->line->fk_c_exp_tax_cat = $fk_c_exp_tax_cat;
2275 $this->line->fk_projet = $projet_id; // deprecated
2276 $this->line->fk_project = $projet_id;
2277
2278 $this->line->vat_src_code = $vat_src_code;
2279 $this->line->vatrate = price2num($vatrate);
2280 $this->line->localtax1_tx = $localtaxes_type[1];
2281 $this->line->localtax2_tx = $localtaxes_type[3];
2282 $this->line->localtax1_type = $localtaxes_type[0];
2283 $this->line->localtax2_type = $localtaxes_type[2];
2284
2285 $this->line->total_ttc = (float) $tmp[2];
2286 $this->line->total_ht = (float) $tmp[0];
2287 $this->line->total_tva = (float) $tmp[1];
2288 $this->line->total_localtax1 = (float) $tmp[9];
2289 $this->line->total_localtax2 = (float) $tmp[10];
2290
2291 $this->line->fk_ecm_files = $fk_ecm_files;
2292
2293 $this->line->id = ((int) $rowid);
2294
2295 // Select des infos sur le type fees
2296 $sql = "SELECT c.code as code_type_fees, c.label as label_type_fees";
2297 $sql .= " FROM ".MAIN_DB_PREFIX."c_type_fees as c";
2298 $sql .= " WHERE c.id = ".((int) $type_fees_id);
2299 $resql = $this->db->query($sql);
2300 if ($resql) {
2301 $objp_fees = $this->db->fetch_object($resql);
2302 $this->line->type_fees_code = $objp_fees->code_type_fees;
2303 $this->line->type_fees_libelle = $objp_fees->label_type_fees;
2304 $this->db->free($resql);
2305 }
2306
2307 // Select des information du projet
2308 $sql = "SELECT p.ref as ref_projet, p.title as title_projet";
2309 $sql .= " FROM ".MAIN_DB_PREFIX."projet as p";
2310 $sql .= " WHERE p.rowid = ".((int) $projet_id);
2311 $resql = $this->db->query($sql);
2312 if ($resql) {
2313 $objp_projet = $this->db->fetch_object($resql);
2314 $this->line->projet_ref = $objp_projet->ref_projet;
2315 $this->line->projet_title = $objp_projet->title_projet;
2316 $this->db->free($resql);
2317 }
2318
2319 $this->applyOffset();
2320 $this->checkRules();
2321
2322 $result = $this->line->update($user, $notrigger);
2323 if ($result < 0) {
2324 $error++;
2325 }
2326
2327 if (!$error) {
2328 $this->db->commit();
2329 return 1;
2330 } else {
2331 $this->error = $this->line->error;
2332 $this->errors = $this->line->errors;
2333 $this->db->rollback();
2334 return -2;
2335 }
2336 }
2337
2338 return 0;
2339 }
2340
2349 public function deleteLine($rowid, $fuser = '', $notrigger = 0)
2350 {
2351 $error = 0;
2352
2353 $this->db->begin();
2354
2355 if (!$notrigger) {
2356 // Call triggers
2357 $result = $this->call_trigger('EXPENSE_REPORT_DET_DELETE', $fuser);
2358 if ($result < 0) {
2359 $error++;
2360 }
2361 // End call triggers
2362 }
2363
2364 $sql = ' DELETE FROM '.MAIN_DB_PREFIX.$this->table_element_line;
2365 $sql .= ' WHERE rowid = '.((int) $rowid);
2366
2367 dol_syslog(get_class($this)."::deleteline sql=".$sql);
2368 $result = $this->db->query($sql);
2369
2370 if (!$result || $error > 0) {
2371 $this->error = $this->db->error();
2372 dol_syslog(get_class($this)."::deleteline Error ".$this->error, LOG_ERR);
2373 $this->db->rollback();
2374 return -1;
2375 }
2376
2377 $this->update_price(1);
2378
2379 $this->db->commit();
2380
2381 return 1;
2382 }
2383
2392 public function periodExists($fuser, $date_debut, $date_fin)
2393 {
2394 global $conf;
2395
2396 $sql = "SELECT rowid, date_debut, date_fin";
2397 $sql .= " FROM ".MAIN_DB_PREFIX.$this->table_element;
2398 $sql .= " WHERE entity = ".((int) $conf->entity); // not shared, only for the current entity
2399 $sql .= " AND fk_user_author = ".((int) $fuser->id);
2400
2401 dol_syslog(get_class($this)."::periodExists sql=".$sql);
2402 $result = $this->db->query($sql);
2403 if ($result) {
2404 $num_rows = $this->db->num_rows($result);
2405 $i = 0;
2406
2407 if ($num_rows > 0) {
2408 $date_d_form = $date_debut;
2409 $date_f_form = $date_fin;
2410
2411 while ($i < $num_rows) {
2412 $objp = $this->db->fetch_object($result);
2413
2414 $date_d_req = $this->db->jdate($objp->date_debut); // 3
2415 $date_f_req = $this->db->jdate($objp->date_fin); // 4
2416
2417 if (!($date_f_form < $date_d_req || $date_d_form > $date_f_req)) {
2418 return $objp->rowid;
2419 }
2420
2421 $i++;
2422 }
2423
2424 return 0;
2425 } else {
2426 return 0;
2427 }
2428 } else {
2429 $this->error = $this->db->lasterror();
2430 dol_syslog(get_class($this)."::periodExists Error ".$this->error, LOG_ERR);
2431 return -1;
2432 }
2433 }
2434
2435
2436 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
2444 {
2445 // phpcs:enable
2446 $users_validator = array();
2447
2448 $sql = "SELECT DISTINCT ur.fk_user";
2449 $sql .= " FROM ".MAIN_DB_PREFIX."user_rights as ur, ".MAIN_DB_PREFIX."rights_def as rd";
2450 $sql .= " WHERE ur.fk_id = rd.id and rd.module = 'expensereport' AND rd.perms = 'approve'"; // Permission 'Approve';
2451 $sql .= " UNION";
2452 $sql .= " SELECT DISTINCT ugu.fk_user";
2453 $sql .= " FROM ".MAIN_DB_PREFIX."usergroup_user as ugu, ".MAIN_DB_PREFIX."usergroup_rights as ur, ".MAIN_DB_PREFIX."rights_def as rd";
2454 $sql .= " WHERE ugu.fk_usergroup = ur.fk_usergroup AND ur.fk_id = rd.id and rd.module = 'expensereport' AND rd.perms = 'approve'"; // Permission 'Approve';
2455 //print $sql;
2456
2457 dol_syslog(get_class($this)."::fetch_users_approver_expensereport sql=".$sql);
2458 $result = $this->db->query($sql);
2459 if ($result) {
2460 $num_rows = $this->db->num_rows($result);
2461 $i = 0;
2462 while ($i < $num_rows) {
2463 $objp = $this->db->fetch_object($result);
2464 array_push($users_validator, $objp->fk_user);
2465 $i++;
2466 }
2467 return $users_validator;
2468 } else {
2469 $this->error = $this->db->lasterror();
2470 dol_syslog(get_class($this)."::fetch_users_approver_expensereport Error ".$this->error, LOG_ERR);
2471 return -1;
2472 }
2473 }
2474
2486 public function generateDocument($modele, $outputlangs, $hidedetails = 0, $hidedesc = 0, $hideref = 0, $moreparams = null)
2487 {
2488 $outputlangs->load("trips");
2489
2490 if (!dol_strlen($modele)) {
2491 if (!empty($this->model_pdf)) {
2492 $modele = $this->model_pdf;
2493 } elseif (getDolGlobalString('EXPENSEREPORT_ADDON_PDF')) {
2494 $modele = getDolGlobalString('EXPENSEREPORT_ADDON_PDF');
2495 }
2496 }
2497
2498 if (!empty($modele)) {
2499 $modelpath = "core/modules/expensereport/doc/";
2500
2501 return $this->commonGenerateDocument($modelpath, $modele, $outputlangs, $hidedetails, $hidedesc, $hideref, $moreparams);
2502 } else {
2503 return 0;
2504 }
2505 }
2506
2513 public function listOfTypes($active = 1)
2514 {
2515 global $langs;
2516 $ret = array();
2517 $sql = "SELECT id, code, label";
2518 $sql .= " FROM ".MAIN_DB_PREFIX."c_type_fees";
2519 $sql .= " WHERE active = ".((int) $active);
2520 dol_syslog(get_class($this)."::listOfTypes", LOG_DEBUG);
2521 $result = $this->db->query($sql);
2522 if ($result) {
2523 $num = $this->db->num_rows($result);
2524 $i = 0;
2525 while ($i < $num) {
2526 $obj = $this->db->fetch_object($result);
2527 $ret[$obj->code] = (($langs->transnoentitiesnoconv($obj->code) != $obj->code) ? $langs->transnoentitiesnoconv($obj->code) : $obj->label);
2528 $i++;
2529 }
2530 } else {
2531 dol_print_error($this->db);
2532 }
2533 return $ret;
2534 }
2535
2541 public function loadStateBoard()
2542 {
2543 global $user;
2544
2545 $this->nb = array();
2546
2547 $sql = "SELECT count(ex.rowid) as nb";
2548 $sql .= " FROM ".MAIN_DB_PREFIX."expensereport as ex";
2549 $sql .= " WHERE ex.fk_statut > 0";
2550 $sql .= " AND ex.entity IN (".getEntity('expensereport').")";
2551 if (!$user->hasRight('expensereport', 'readall')) {
2552 $userchildids = $user->getAllChildIds(1);
2553 $sql .= " AND (ex.fk_user_author IN (".$this->db->sanitize(implode(',', $userchildids)).")";
2554 $sql .= " OR ex.fk_user_validator IN (".$this->db->sanitize(implode(',', $userchildids))."))";
2555 }
2556
2557 $resql = $this->db->query($sql);
2558 if ($resql) {
2559 while ($obj = $this->db->fetch_object($resql)) {
2560 $this->nb["expensereports"] = $obj->nb;
2561 }
2562 $this->db->free($resql);
2563 return 1;
2564 } else {
2565 dol_print_error($this->db);
2566 $this->error = $this->db->error();
2567 return -1;
2568 }
2569 }
2570
2571 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
2579 public function load_board($user, $option = 'topay')
2580 {
2581 // phpcs:enable
2582 global $conf, $langs;
2583
2584 if ($user->socid) {
2585 return -1; // protection pour eviter appel par utilisateur externe
2586 }
2587
2588 $now = dol_now();
2589
2590 $sql = "SELECT ex.rowid, ex.date_valid";
2591 $sql .= " FROM ".MAIN_DB_PREFIX."expensereport as ex";
2592 if ($option == 'toapprove') {
2593 $sql .= " WHERE ex.fk_statut = ".self::STATUS_VALIDATED;
2594 } else {
2595 $sql .= " WHERE ex.fk_statut = ".self::STATUS_APPROVED;
2596 }
2597 $sql .= " AND ex.entity IN (".getEntity('expensereport').")";
2598 if (!$user->hasRight('expensereport', 'readall')) {
2599 $userchildids = $user->getAllChildIds(1);
2600 $sql .= " AND (ex.fk_user_author IN (".$this->db->sanitize(implode(',', $userchildids)).")";
2601 $sql .= " OR ex.fk_user_validator IN (".$this->db->sanitize(implode(',', $userchildids))."))";
2602 }
2603
2604 $resql = $this->db->query($sql);
2605 if ($resql) {
2606 $langs->load("trips");
2607
2608 $response = new WorkboardResponse();
2609 if ($option == 'toapprove') {
2610 $response->warning_delay = $conf->expensereport->approve->warning_delay / 60 / 60 / 24;
2611 $response->label = $langs->trans("ExpenseReportsToApprove");
2612 $response->labelShort = $langs->trans("ToApprove");
2613 $response->url = DOL_URL_ROOT.'/expensereport/list.php?mainmenu=hrm&amp;statut='.self::STATUS_VALIDATED;
2614 } else {
2615 $response->warning_delay = $conf->expensereport->payment->warning_delay / 60 / 60 / 24;
2616 $response->label = $langs->trans("ExpenseReportsToPay");
2617 $response->labelShort = $langs->trans("StatusToPay");
2618 $response->url = DOL_URL_ROOT.'/expensereport/list.php?mainmenu=hrm&amp;statut='.self::STATUS_APPROVED;
2619 }
2620 $response->img = img_object('', "trip");
2621
2622 while ($obj = $this->db->fetch_object($resql)) {
2623 $response->nbtodo++;
2624
2625 if ($option == 'toapprove') {
2626 if ($this->db->jdate($obj->date_valid) < ($now - $conf->expensereport->approve->warning_delay)) {
2627 $response->nbtodolate++;
2628 }
2629 } else {
2630 if ($this->db->jdate($obj->date_valid) < ($now - $conf->expensereport->payment->warning_delay)) {
2631 $response->nbtodolate++;
2632 }
2633 }
2634 }
2635
2636 return $response;
2637 } else {
2638 dol_print_error($this->db);
2639 $this->error = $this->db->error();
2640 return -1;
2641 }
2642 }
2643
2650 public function hasDelay($option)
2651 {
2652 global $conf;
2653
2654 // Only valid expenses reports
2655 if ($option == 'toapprove' && $this->status != 2) {
2656 return false;
2657 }
2658 if ($option == 'topay' && $this->status != 5) {
2659 return false;
2660 }
2661
2662 $now = dol_now();
2663 if ($option == 'toapprove') {
2664 return (!empty($this->datevalid) ? $this->datevalid : $this->date_valid) < ($now - $conf->expensereport->approve->warning_delay);
2665 } else {
2666 return (!empty($this->datevalid) ? $this->datevalid : $this->date_valid) < ($now - $conf->expensereport->payment->warning_delay);
2667 }
2668 }
2669
2675 public function getVentilExportCompta()
2676 {
2677 $alreadydispatched = 0;
2678
2679 $type = 'expense_report';
2680
2681 $sql = " SELECT COUNT(ab.rowid) as nb FROM ".MAIN_DB_PREFIX."accounting_bookkeeping as ab WHERE ab.doc_type='".$this->db->escape($type)."' AND ab.fk_doc = ".((int) $this->id);
2682 $resql = $this->db->query($sql);
2683 if ($resql) {
2684 $obj = $this->db->fetch_object($resql);
2685 if ($obj) {
2686 $alreadydispatched = $obj->nb;
2687 }
2688 } else {
2689 $this->error = $this->db->lasterror();
2690 return -1;
2691 }
2692
2693 if ($alreadydispatched) {
2694 return 1;
2695 }
2696 return 0;
2697 }
2698
2704 public function getSumPayments()
2705 {
2706 $table = 'payment_expensereport';
2707 $field = 'fk_expensereport';
2708
2709 $sql = 'SELECT sum(amount) as amount';
2710 $sql .= ' FROM '.MAIN_DB_PREFIX.$table;
2711 $sql .= " WHERE ".$field." = ".((int) $this->id);
2712
2713 dol_syslog(get_class($this)."::getSumPayments", LOG_DEBUG);
2714 $resql = $this->db->query($sql);
2715 if ($resql) {
2716 $obj = $this->db->fetch_object($resql);
2717 $this->db->free($resql);
2718 return (empty($obj->amount) ? 0 : $obj->amount);
2719 } else {
2720 $this->error = $this->db->lasterror();
2721 return -1;
2722 }
2723 }
2724
2733 public function computeTotalKm($fk_cat, $qty, $tva)
2734 {
2735 global $langs, $db, $conf;
2736
2737 $cumulYearQty = 0;
2738 $ranges = array();
2739 $coef = 0;
2740
2741
2742 if ($fk_cat < 0) {
2743 $this->error = $langs->trans('ErrorBadParameterCat');
2744 return -1;
2745 }
2746
2747 if ($qty <= 0) {
2748 $this->error = $langs->trans('ErrorBadParameterQty');
2749 return -1;
2750 }
2751
2752 $currentUser = new User($db);
2753 $currentUser->fetch($this->fk_user);
2754 $currentUser->loadRights('expensereport');
2755 //Clean
2756 $qty = (float) price2num($qty);
2757
2758 $sql = " SELECT r.range_ik, t.ikoffset, t.coef";
2759 $sql .= " FROM ".MAIN_DB_PREFIX."expensereport_ik t";
2760 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."c_exp_tax_range r ON r.rowid = t.fk_range";
2761 $sql .= " WHERE t.fk_c_exp_tax_cat = ".(int) $fk_cat;
2762 $sql .= " ORDER BY r.range_ik ASC";
2763
2764 dol_syslog("expenseReport::computeTotalkm sql=".$sql, LOG_DEBUG);
2765
2766 $result = $this->db->query($sql);
2767
2768 if ($result) {
2769 if ($conf->global->EXPENSEREPORT_CALCULATE_MILEAGE_EXPENSE_COEFFICIENT_ON_CURRENT_YEAR) {
2770 $arrayDate = dol_getdate(dol_now());
2771 $sql = " SELECT count(n.qty) as cumul FROM ".MAIN_DB_PREFIX."expensereport_det n";
2772 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."expensereport e ON e.rowid = n.fk_expensereport";
2773 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."c_type_fees tf ON tf.id = n.fk_c_type_fees";
2774 $sql .= " WHERE e.fk_user_author = ".(int) $this->fk_user_author;
2775 $sql .= " AND YEAR(n.date) = ".(int) $arrayDate['year'];
2776 $sql .= " AND tf.code = 'EX_KME' ";
2777 $sql .= " AND e.fk_statut = ".(int) ExpenseReport::STATUS_VALIDATED;
2778
2779 $resql = $this->db->query($sql);
2780
2781 if ($resql) {
2782 $obj = $this->db->fetch_object($resql);
2783 $cumulYearQty = $obj->cumul;
2784 }
2785
2786 $qty += (float) $cumulYearQty;
2787 }
2788
2789 $num = $this->db->num_rows($result);
2790
2791 if ($num) {
2792 for ($i = 0; $i < $num; $i++) {
2793 $obj = $this->db->fetch_object($result);
2794
2795 $ranges[$i] = $obj;
2796 }
2797 '@phan-var-force Object[] $ranges';
2798
2799 for ($i = 0; $i < $num; $i++) {
2800 if ($i < ($num - 1)) {
2801 // @phan-suppress-next-line PhanTypeInvalidDimOffset
2802 if ($qty > $ranges[$i]->range_ik && $qty < $ranges[$i + 1]->range_ik) {
2803 $coef = $ranges[$i]->coef;
2804 $offset = $ranges[$i]->ikoffset;
2805 }
2806 } else {
2807 if ($qty > $ranges[$i]->range_ik) {
2808 $coef = $ranges[$i]->coef;
2809 $offset = $ranges[$i]->ikoffset;
2810 }
2811 }
2812 }
2813 $total_ht = $coef;
2814 return $total_ht;
2815 } else {
2816 $this->error = $langs->trans('TaxUndefinedForThisCategory');
2817 return 0;
2818 }
2819 } else {
2820 $this->error = $this->db->error()." sql=".$sql;
2821
2822 return -1;
2823 }
2824 }
2825
2833 public function getKanbanView($option = '', $arraydata = null)
2834 {
2835 global $langs;
2836
2837 $selected = (empty($arraydata['selected']) ? 0 : $arraydata['selected']);
2838
2839 $return = '<div class="box-flex-item box-flex-grow-zero">';
2840 $return .= '<div class="info-box info-box-sm">';
2841 $return .= '<span class="info-box-icon bg-infobox-action">';
2842 $return .= img_picto('', $this->picto);
2843 $return .= '</span>';
2844 $return .= '<div class="info-box-content">';
2845 $return .= '<span class="info-box-ref inline-block tdoverflowmax150 valignmiddle">'.(method_exists($this, 'getNomUrl') ? $this->getNomUrl(1) : $this->ref).'</span>';
2846 if ($selected >= 0) {
2847 $return .= '<input id="cb'.$this->id.'" class="flat checkforselect fright" type="checkbox" name="toselect[]" value="'.$this->id.'"'.($selected ? ' checked="checked"' : '').'>';
2848 }
2849 if (array_key_exists('userauthor', $arraydata) && $arraydata['userauthor'] instanceof User) {
2850 $return .= '<br><span class="info-box-label">'.$arraydata['userauthor']->getNomUrl(-1).'</span>';
2851 }
2852 if (property_exists($this, 'date_debut') && property_exists($this, 'date_fin')) {
2853 $return .= '<br><span class="info-box-label">'.dol_print_date($this->date_debut, 'day').'</span>';
2854 $return .= ' <span class="opacitymedium">'.$langs->trans("To").'</span> ';
2855 $return .= '<span class="info-box-label">'.dol_print_date($this->date_fin, 'day').'</span>';
2856 }
2857 if (method_exists($this, 'getLibStatut')) {
2858 $return .= '<br><div class="info-box-status">'.$this->getLibStatut(3).'</div>';
2859 }
2860 $return .= '</div>';
2861 $return .= '</div>';
2862 $return .= '</div>';
2863 return $return;
2864 }
2865}
$object ref
Definition info.php:89
Parent class of all other business classes (invoices, contracts, proposals, orders,...
fetch_optionals($rowid=null, $optionsArray=null)
Function to get extra fields of an object into $this->array_options This method is in most cases call...
deleteEcmFiles($mode=0)
Delete related files of object in database.
update_price($exclspec=0, $roundingadjust='auto', $nodatabaseupdate=0, $seller=null)
Update total_ht, total_ttc, total_vat, total_localtax1, total_localtax2 for an object (sum of lines).
commonGenerateDocument($modelspath, $modele, $outputlangs, $hidedetails, $hidedesc, $hideref, $moreparams=null)
Common function for all objects extending CommonObject for generating documents.
deleteObjectLinked($sourceid=null, $sourcetype='', $targetid=null, $targettype='', $rowid=0, $f_user=null, $notrigger=0)
Delete all links between an object $this.
setErrorsFromObject($object)
setErrorsFromObject
deleteExtraFields()
Delete all extra fields values for the current object.
insertExtraFields($trigger='', $userused=null)
Add/Update all extra fields values for the current object.
delete_linked_contact($source='', $code='')
Delete all links between an object $this and all its contacts in llx_element_contact.
call_trigger($triggerName, $user)
Call trigger based on this instance.
Class to manage Trips and Expenses.
setPaid($id, $fuser, $notrigger=0)
Classify the expense report as paid.
loadStateBoard()
Load the indicators this->nb for the state board.
__construct($db)
Constructor.
checkRules($type=0, $seller='')
Check constraint of rules and update price if needed.
updateline($rowid, $type_fees_id, $projet_id, $vatrate, $comments, $qty, $value_unit, $date, $expensereport_id, $fk_c_exp_tax_cat=0, $fk_ecm_files=0, $notrigger=0)
Update an expense report line.
getNextNumRef()
Return next reference of expense report not already used.
createFromClone(User $user, $fk_user_author)
Load an object from its id and create a new one in database.
addline($qty=0, $up=0, $fk_c_type_fees=0, $vatrate=0, $date='', $comments='', $fk_project=0, $fk_c_exp_tax_cat=0, $type=0, $fk_ecm_files=0)
Add expense report line.
const STATUS_DRAFT
Draft status.
computeTotalKm($fk_cat, $qty, $tva)
Compute the cost of the kilometers expense based on the number of kilometers and the vehicle category...
periodExists($fuser, $date_debut, $date_fin)
periodExists
offsetAlreadyGiven()
If the sql find any rows then the ikoffset is already given (ikoffset is applied at the first expense...
listOfTypes($active=1)
List of types.
const STATUS_APPROVED
Classified approved.
set_save_from_refuse($fuser)
set_save_from_refuse
setValidate($fuser, $notrigger=0)
Set to status validate.
getSumPayments()
Return amount of payments already done.
getLibStatut($mode=0)
Returns the label status.
set_cancel($fuser, $detail, $notrigger=0)
set_cancel
getNomUrl($withpicto=0, $option='', $max=0, $short=0, $moretitle='', $notooltip=0, $save_lastsearch_value=-1)
Return clickable name (with picto eventually)
set_unpaid($fuser, $notrigger=0)
set_unpaid
info($id)
Load information on object.
getTooltipContentArray($params)
getTooltipContentArray
hasDelay($option)
Return if an expense report is late or not.
applyOffset($type=0, $seller='')
Method to apply the offset if needed.
const STATUS_CANCELED
Classified canceled.
const STATUS_CLOSED
Classified paid.
const STATUS_REFUSED
Classified refused.
deleteLine($rowid, $fuser='', $notrigger=0)
deleteline
setDeny($fuser, $details, $notrigger=0)
setDeny
getVentilExportCompta()
Return if object was dispatched into bookkeeping.
getKanbanView($option='', $arraydata=null)
Return clickable link of object (with optional picto)
update($user, $notrigger=0, $userofexpensereport=null)
update
const STATUS_VALIDATED
Validated (need to be paid)
create($user, $notrigger=0)
Create object in database.
load_board($user, $option='topay')
Load indicators for dashboard (this->nbtodo and this->nbtodolate)
set_paid($id, $fuser, $notrigger=0)
Classify the expense report as paid.
generateDocument($modele, $outputlangs, $hidedetails=0, $hidedesc=0, $hideref=0, $moreparams=null)
Create a document onto disk according to template module.
initAsSpecimen()
Initialise an instance with random values.
fetch_users_approver_expensereport()
Return list of people with permission to validate expense reports.
setApproved($fuser, $notrigger=0)
Set status to approved.
LibStatut($status, $mode=0)
Returns the label of a status.
setUnpaid($fuser, $notrigger=0)
set_unpaid
fetch_line_by_project($projectid, $user)
fetch_line_by_project
update_totaux_add($ligne_total_ht, $ligne_total_tva)
Update total of an expense report when you add a line.
Class to manage inventories.
Class of expense report details lines.
Class to manage inventories.
Class to manage third parties objects (customers, suppliers, prospects...)
Class to manage Dolibarr users.
print $langs trans("Ref").' m titre as m m statut as status
Or an array listing all the potential status of the object: array: int of the status => translated la...
Definition index.php:171
dol_delete_file($file, $disableglob=0, $nophperrors=0, $nohook=0, $object=null, $allowdotdot=false, $indexdatabase=1, $nolog=0)
Remove a file or several files with a mask.
dol_delete_dir_recursive($dir, $count=0, $nophperrors=0, $onlysub=0, &$countdeleted=0, $indexdatabase=1, $nolog=0, $level=0)
Remove a directory $dir and its subdirectories (or only files and subdirectories)
dol_dir_list($utf8_path, $types="all", $recursive=0, $filter="", $excludefilter=null, $sortcriteria="name", $sortorder=SORT_ASC, $mode=0, $nohook=0, $relativename="", $donotfollowsymlinks=0, $nbsecondsold=0)
Scan a directory and return a list of files/directories.
Definition files.lib.php:63
dol_delete_preview($object)
Delete all preview files linked to object instance.
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.
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
img_object($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $allowothertags=array())
Show a picto called object_picto (generic function)
dol_strlen($string, $stringencoding='UTF-8')
Make a strlen call.
price($amount, $form=0, $outlangs='', $trunc=1, $rounding=-1, $forcerounding=-1, $currency_code='')
Function to format a value into an amount for visual output Function used into PDF and HTML pages.
dol_now($mode='auto')
Return date for now.
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
getLocalTaxesFromRate($vatrate, $local, $buyer, $seller, $firstparamisid=0)
Get type and rate of localtaxes for a particular vat rate/country of a thirdparty.
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).
dolGetFirstLastname($firstname, $lastname, $nameorder=-1)
Return firstname and lastname in correct order.
dolGetStatus($statusLabel='', $statusLabelShort='', $html='', $statusType='status0', $displayMode=0, $url='', $params=array())
Output the badge of a status.
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...
dol_trunc($string, $size=40, $trunc='right', $stringencoding='UTF-8', $nodot=0, $display=0)
Truncate a string to a particular length adding '…' if string larger than length.
dol_sanitizeFileName($str, $newstr='_', $unaccent=1, $includequotes=0)
Clean a string to use it as a file name.
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
dol_getdate($timestamp, $fast=false, $forcetimezone='')
Return an array with locale date info.
global $conf
The following vars must be defined: $type2label $form $conf, $lang, The following vars may also be de...
Definition member.php:79
calcul_price_total($qty, $pu, $remise_percent_ligne, $txtva, $uselocaltax1_rate, $uselocaltax2_rate, $remise_percent_global, $price_base_type, $info_bits, $type, $seller='', $localtaxes_array=[], $progress=100, $multicurrency_tx=1, $pu_devise=0, $multicurrency_code='')
Calculate totals (net, vat, ...) of a line.
Definition price.lib.php:90