dolibarr 23.0.3
asset.class.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2017 Laurent Destailleur <eldy@users.sourceforge.net>
3 * Copyright (C) 2018-2025 Alexandre Spangaro <alexandre@inovea-conseil.com>
4 * Copyright (C) 2024-2025 Frédéric France <frederic.france@free.fr>
5 * Copyright (C) 2024-2025 MDW <mdeweerd@users.noreply.github.com>
6 * Copyright (C) 2024 Jose MARTINEZ <jose.martinez@pichinov.com>
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 3 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program. If not, see <https://www.gnu.org/licenses/>.
20 */
21
28require_once DOL_DOCUMENT_ROOT.'/core/class/commonobject.class.php';
29
33class Asset extends CommonObject
34{
38 public $module = 'asset';
39
43 public $element = 'asset';
44
48 public $table_element = 'asset';
49
53 public $picto = 'asset';
54
59 public $fk_element = 'fk_asset';
60
67 protected $childtablesoncascade = array(
68 '@AssetAccountancyCodes:/asset/class/assetaccountancycodes.class.php:fk_asset',
69 '@AssetAccountancyCodesFiscal:/asset/class/assetaccountancycodesfiscal.class.php:fk_asset'
70 );
71
72
73 const STATUS_DRAFT = 0; // Draft
74 const STATUS_VALIDATED = 1; // In progress
75 const STATUS_DISPOSED = 9; // Disposed
76
108 public $fields = array(
109 'rowid' => array('type' => 'integer', 'label' => 'TechnicalID', 'enabled' => 1, 'position' => 1, 'notnull' => 1, 'visible' => 0, 'noteditable' => 1, 'index' => 1, 'css' => 'left', 'comment' => "Id"),
110 'entity' => array('type' => 'integer', 'label' => 'Entity', 'enabled' => 1, 'visible' => 0, 'notnull' => 1, 'default' => '1', 'index' => 1, 'position' => 5),
111 'ref' => array('type' => 'varchar(128)', 'label' => 'Ref', 'enabled' => 1, 'position' => 20, 'notnull' => 1, 'visible' => 4, 'noteditable' => 0, 'default' => '(PROV)', 'index' => 1, 'searchall' => 1, 'showoncombobox' => 1, 'validate' => 1, 'comment' => "Reference of object", "css" => "maxwidth150"),
112 'label' => array('type' => 'varchar(255)', 'label' => 'Label', 'enabled' => 1, 'position' => 30, 'notnull' => 1, 'visible' => 1, 'searchall' => 1, 'csslist' => 'tdoverflowmax125', 'css' => 'minwidth300', 'cssview' => 'wordbreak', 'showoncombobox' => 2, 'validate' => 1,),
113 'fk_asset_model' => array('type' => 'integer:AssetModel:asset/class/assetmodel.class.php:1:((status:=:1) and (entity:IN:__SHARED_ENTITIES__))', 'label' => 'AssetModel', 'enabled' => 1, 'position' => 40, 'notnull' => 0, 'visible' => 1, 'index' => 1, 'validate' => 1, 'csslist' => 'tdoverflowmax75', 'css' => 'maxwidth300'),
114 'qty' => array('type' => 'real', 'label' => 'Qty', 'enabled' => 1, 'position' => 50, 'notnull' => 1, 'visible' => 0, 'default' => '1', 'isameasure' => 1, 'css' => 'maxwidth75imp', 'validate' => 1,),
115 'acquisition_type' => array('type' => 'smallint', 'label' => 'AssetAcquisitionType', 'enabled' => 1, 'position' => 60, 'notnull' => 1, 'visible' => 1, 'arrayofkeyval' => array(0 => 'AssetAcquisitionTypeNew', 1 => 'AssetAcquisitionTypeOccasion'), 'validate' => 1, 'csslist' => 'tdoverflowmax75'),
116 'asset_type' => array('type' => 'smallint', 'label' => 'AssetType', 'enabled' => 1, 'position' => 70, 'notnull' => 1, 'visible' => 1, 'arrayofkeyval' => array(0 => 'AssetTypeIntangible', 1 => 'AssetTypeTangible', 2 => 'AssetTypeInProgress', 3 => 'AssetTypeFinancial'), 'validate' => 1, 'csslist' => 'tdoverflowmax75'),
117 'not_depreciated' => array('type' => 'boolean', 'label' => 'AssetNotDepreciated', 'enabled' => 1, 'position' => 80, 'notnull' => 0, 'default' => '0', 'visible' => 1, 'validate' => 1, 'csslist' => 'maxwidth50 tdoverflowmax50', 'css' => 'maxwidth50imp'),
118 'date_acquisition' => array('type' => 'date', 'label' => 'AssetDateAcquisition', 'enabled' => 1, 'position' => 90, 'notnull' => 1, 'visible' => 1,),
119 'date_start' => array('type' => 'date', 'label' => 'AssetDateStart', 'enabled' => 1, 'position' => 100, 'notnull' => 0, 'visible' => -1,),
120 'acquisition_value_ht' => array('type' => 'price', 'label' => 'AssetAcquisitionValueHT', 'enabled' => 1, 'position' => 110, 'notnull' => 1, 'visible' => 1, 'isameasure' => 1, 'validate' => 1,),
121 'recovered_vat' => array('type' => 'price', 'label' => 'AssetRecoveredVAT', 'enabled' => 1, 'position' => 120, 'notnull' => 0, 'visible' => 1, 'isameasure' => 1, 'validate' => 1,),
122 'reversal_date' => array('type' => 'date', 'label' => 'AssetReversalDate', 'enabled' => 1, 'position' => 130, 'notnull' => 0, 'visible' => 1,),
123 'reversal_amount_ht' => array('type' => 'price', 'label' => 'AssetReversalAmountHT', 'enabled' => 1, 'position' => 140, 'notnull' => 0, 'visible' => 1, 'isameasure' => 1, 'validate' => 1, 'csslist' => 'maxwidth50'),
124 'disposal_date' => array('type' => 'date', 'label' => 'AssetDisposalDate', 'enabled' => 1, 'position' => 200, 'notnull' => 0, 'visible' => -2,),
125 'disposal_amount_ht' => array('type' => 'price', 'label' => 'AssetDisposalAmount', 'enabled' => 1, 'position' => 210, 'notnull' => 0, 'visible' => -2, 'default' => '0', 'isameasure' => 1, 'validate' => 1,),
126 'fk_disposal_type' => array('type' => 'sellist:c_asset_disposal_type:label:rowid::active=1', 'label' => 'AssetDisposalType', 'enabled' => 1, 'position' => 220, 'notnull' => 0, 'visible' => -2, 'index' => 1, 'validate' => 1,),
127 'disposal_depreciated' => array('type' => 'boolean', 'label' => 'AssetDisposalDepreciated', 'enabled' => 1, 'position' => 230, 'notnull' => 0, 'default' => '0', 'visible' => -2, 'validate' => 1,),
128 'disposal_subject_to_vat' => array('type' => 'boolean', 'label' => 'AssetDisposalSubjectToVat', 'enabled' => 1, 'position' => 240, 'notnull' => 0, 'default' => '0', 'visible' => -2, 'validate' => 1,),
129 'note_public' => array('type' => 'html', 'label' => 'NotePublic', 'enabled' => 1, 'position' => 300, 'notnull' => 0, 'visible' => 0, 'validate' => 1,),
130 'note_private' => array('type' => 'html', 'label' => 'NotePrivate', 'enabled' => 1, 'position' => 301, 'notnull' => 0, 'visible' => 0, 'validate' => 1,),
131 'date_creation' => array('type' => 'datetime', 'label' => 'DateCreation', 'enabled' => 1, 'position' => 500, 'notnull' => 1, 'visible' => -2,),
132 'tms' => array('type' => 'timestamp', 'label' => 'DateModification', 'enabled' => 1, 'position' => 501, 'notnull' => 0, 'visible' => -2,),
133 'date_valid' => array('type' => 'datetime', 'label' => 'DateValidation', 'enabled' => 1, 'visible' => -2, 'position' => 502, 'notnull' => 0,),
134 'fk_user_creat' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserAuthor', 'enabled' => 1, 'position' => 510, 'notnull' => 1, 'visible' => -2, 'foreignkey' => 'user.rowid',),
135 'fk_user_modif' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserModif', 'enabled' => 1, 'position' => 511, 'notnull' => -1, 'visible' => -2,),
136 'fk_user_valid' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserValidation', 'picto' => 'user', 'enabled' => 1, 'visible' => -2, 'position' => 512, 'notnull' => 0, 'csslist' => 'tdoverflowmax100'),
137 'last_main_doc' => array('type' => 'varchar(255)', 'label' => 'LastMainDoc', 'enabled' => 1, 'position' => 600, 'notnull' => 0, 'visible' => 0,),
138 'import_key' => array('type' => 'varchar(14)', 'label' => 'ImportId', 'enabled' => 1, 'position' => 1000, 'notnull' => -1, 'visible' => -2,),
139 'model_pdf' => array('type' => 'varchar(255)', 'label' => 'Model pdf', 'enabled' => 1, 'position' => 1010, 'notnull' => -1, 'visible' => 0,),
140 'status' => array('type' => 'smallint', 'label' => 'Status', 'enabled' => 1, 'position' => 1000, 'notnull' => 1, 'default' => '0', 'visible' => 2, 'index' => 1, 'arrayofkeyval' => array(0 => 'Draft', 1 => 'Validated', 9 => 'Canceled'), 'validate' => 1,),
141 );
142
146 public $rowid;
150 public $ref;
154 public $label;
158 public $fk_asset_model;
162 public $reversal_amount_ht;
166 public $acquisition_value_ht;
170 public $recovered_vat;
174 public $reversal_date;
178 public $date_acquisition;
182 public $date_start;
186 public $qty;
190 public $acquisition_type;
194 public $asset_type;
198 public $not_depreciated;
202 public $disposal_date;
206 public $disposal_amount_ht;
210 public $fk_disposal_type;
214 public $disposal_depreciated;
218 public $disposal_subject_to_vat;
222 public $supplier_invoice_id;
226 public $note_public;
230 public $note_private;
234 public $date_valid;
238 public $fk_user_creat;
242 public $fk_user_modif;
246 public $fk_user_valid;
250 public $last_main_doc;
251
255 public $import_key;
259 public $model_pdf;
260
264 public $status;
265
269 public $asset_depreciation_options;
270
274 public $asset_accountancy_codes;
275
279 public $depreciation_lines = array();
280
281
287 public function __construct(DoliDB $db)
288 {
289 global $langs;
290
291 $this->db = $db;
292
293 $this->ismultientitymanaged = 1;
294 $this->isextrafieldmanaged = 1;
295
296 if (!getDolGlobalString('MAIN_SHOW_TECHNICAL_ID') && isset($this->fields['rowid'])) {
297 $this->fields['rowid']['visible'] = 0;
298 }
299 if (!isModEnabled('multicompany') && isset($this->fields['entity'])) {
300 $this->fields['entity']['enabled'] = 0;
301 }
302
303 // Unset fields that are disabled
304 foreach ($this->fields as $key => $val) {
305 if (isset($val['enabled']) && empty($val['enabled'])) {
306 unset($this->fields[$key]);
307 }
308 }
309
310 // Translate some data of arrayofkeyval
311 if (is_object($langs)) {
312 foreach ($this->fields as $key => $val) {
313 if (!empty($val['arrayofkeyval']) && is_array($val['arrayofkeyval'])) {
314 foreach ($val['arrayofkeyval'] as $key2 => $val2) {
315 $this->fields[$key]['arrayofkeyval'][$key2] = $langs->trans($val2);
316 }
317 }
318 }
319 }
320 }
321
329 public function create(User $user, $notrigger = 0)
330 {
331 if (!isset($this->date_start) || $this->date_start === "") {
332 $this->date_start = $this->date_acquisition;
333 }
334
335 $this->db->begin();
336
337 $result = $result_create = $this->createCommon($user, $notrigger);
338 if ($result > 0 && $this->fk_asset_model > 0) {
339 $result = $this->setDataFromAssetModel($user, $notrigger);
340 }
341 if ($result > 0) {
342 if ($this->supplier_invoice_id > 0) {
343 $this->add_object_linked('invoice_supplier', $this->supplier_invoice_id);
344 }
345 }
346
347 if ($result < 0) {
348 $this->db->rollback();
349 } else {
350 $this->db->commit();
351 }
352
353 return $result > 0 ? $result_create : $result;
354 }
355
363 public function createFromClone(User $user, $fromid)
364 {
365 //global $langs, $extrafields;
366 //$error = 0;
367
368 dol_syslog(__METHOD__, LOG_DEBUG);
369
370 // $object = new self($this->db);
371 //
372 // $this->db->begin();
373 //
374 // // Load source object
375 // $result = $object->fetchCommon($fromid);
376 //
377 //
378 // // Reset some properties
379 // unset($object->id);
380 // unset($object->fk_user_creat);
381 // unset($object->import_key);
382 //
383 // // Clear fields
384 // if (property_exists($object, 'ref')) {
385 // $object->ref = empty($this->fields['ref']['default']) ? "Copy_Of_".$object->ref : $this->fields['ref']['default'];
386 // }
387 // if (property_exists($object, 'label')) {
388 // $object->label = empty($this->fields['label']['default']) ? $langs->trans("CopyOf")." ".$object->label : $this->fields['label']['default'];
389 // }
390 // if (property_exists($object, 'status')) {
391 // $object->status = self::STATUS_DRAFT;
392 // }
393 // if (property_exists($object, 'date_creation')) {
394 // $object->date_creation = dol_now();
395 // }
396 // if (property_exists($object, 'date_modification')) {
397 // $object->date_modification = null;
398 // }
399 // // ...
400 // // Clear extrafields that are unique
401 // if (is_array($object->array_options) && count($object->array_options) > 0) {
402 // $extrafields->fetch_name_optionals_label($this->table_element);
403 // foreach ($object->array_options as $key => $option) {
404 // $shortkey = preg_replace('/options_/', '', $key);
405 // if (!empty($extrafields->attributes[$this->table_element]['unique'][$shortkey])) {
406 // //var_dump($key); var_dump($clonedObj->array_options[$key]); exit;
407 // unset($object->array_options[$key]);
408 // }
409 // }
410 // }
411 //
412 // // Create clone
413 // $object->context['createfromclone'] = 'createfromclone';
414 // $result = $object->createCommon($user);
415 // if ($result < 0) {
416 // $error++;
417 // $this->error = $object->error;
418 // $this->errors = $object->errors;
419 // }
420 //
421 // if (!$error) {
422 // // copy internal contacts
423 // if ($this->copy_linked_contact($object, 'internal') < 0) {
424 // $error++;
425 // }
426 // }
427 //
428 // if (!$error) {
429 // // copy external contacts if same company
430 // if (property_exists($this, 'fk_soc') && $this->fk_soc == $object->socid) {
431 // if ($this->copy_linked_contact($object, 'external') < 0) {
432 // $error++;
433 // }
434 // }
435 // }
436 //
437 // unset($object->context['createfromclone']);
438 //
439 // // End
440 // if (!$error) {
441 // $this->db->commit();
442 // return $object;
443 // } else {
444 // $this->db->rollback();
445 // return -1;
446 // }
447
448 return -1;
449 }
450
458 public function fetch($id, $ref = null)
459 {
460 $result = $this->fetchCommon($id, $ref);
461 if ($result > 0) {
462 $res = $this->hasDepreciationLinesInBookkeeping();
463 if ($res < 0) {
464 return -1;
465 } elseif ($res > 0) {
466 $this->fields['date_acquisition']['noteditable'] = 1;
467 $this->fields['date_start']['noteditable'] = 1;
468 $this->fields['acquisition_value_ht']['noteditable'] = 1;
469 $this->fields['recovered_vat']['noteditable'] = 1;
470 $this->fields['reversal_date']['noteditable'] = 1;
471 $this->fields['reversal_amount_ht']['noteditable'] = 1;
472 }
473 }
474
475 return $result;
476 }
477
478
491 public function fetchAll($sortorder = '', $sortfield = '', $limit = 0, $offset = 0, $filter = '', $filtermode = 'AND')
492 {
493 dol_syslog(__METHOD__, LOG_DEBUG);
494
495 $records = array();
496
497 $sql = "SELECT ";
498 $sql .= $this->getFieldList('t');
499 $sql .= " FROM ".MAIN_DB_PREFIX.$this->table_element." as t";
500 if (isset($this->ismultientitymanaged) && $this->ismultientitymanaged == 1) {
501 $sql .= " WHERE t.entity IN (".getEntity($this->element).")";
502 } else {
503 $sql .= " WHERE 1 = 1";
504 }
505
506 // Manage filter
507 $errormessage = '';
508 $sql .= forgeSQLFromUniversalSearchCriteria($filter, $errormessage);
509 if ($errormessage) {
510 $this->errors[] = $errormessage;
511 dol_syslog(__METHOD__.' '.implode(',', $this->errors), LOG_ERR);
512
513 return -1;
514 }
515
516 if (!empty($sortfield)) {
517 $sql .= $this->db->order($sortfield, $sortorder);
518 }
519 if (!empty($limit)) {
520 $sql .= $this->db->plimit($limit, $offset);
521 }
522
523 $resql = $this->db->query($sql);
524 if ($resql) {
525 $num = $this->db->num_rows($resql);
526 $i = 0;
527 while ($i < ($limit ? min($limit, $num) : $num)) {
528 $obj = $this->db->fetch_object($resql);
529
530 $record = new self($this->db);
531 $record->setVarsFromFetchObj($obj);
532
533 $records[$record->id] = $record;
534
535 $i++;
536 }
537 $this->db->free($resql);
538
539 return $records;
540 } else {
541 $this->errors[] = 'Error '.$this->db->lasterror();
542 dol_syslog(__METHOD__.' '.implode(',', $this->errors), LOG_ERR);
543
544 return -1;
545 }
546 }
547
555 public function update(User $user, $notrigger = 0)
556 {
557 if (!isset($this->date_start) || $this->date_start === "") {
558 $this->date_start = $this->date_acquisition;
559 }
560
561 $this->db->begin();
562
563 $result = $this->updateCommon($user, $notrigger);
564 if ($result > 0 && $this->fk_asset_model > 0 && $this->fk_asset_model != $this->oldcopy->fk_asset_model) {
565 $result = $this->setDataFromAssetModel($user, $notrigger);
566 }
567 if ($result > 0 && (
568 $this->date_start != $this->oldcopy->date_start ||
569 $this->acquisition_value_ht != $this->oldcopy->acquisition_value_ht ||
570 $this->reversal_date != $this->oldcopy->reversal_date ||
571 $this->reversal_amount_ht != $this->oldcopy->reversal_amount_ht ||
572 ($this->fk_asset_model > 0 && $this->fk_asset_model != $this->oldcopy->fk_asset_model)
573 )
574 ) {
575 $result = $this->calculationDepreciation();
576 }
577
578 if ($result < 0) {
579 $this->db->rollback();
580 } else {
581 $this->db->commit();
582 }
583
584 return $result;
585 }
586
594 public function delete(User $user, $notrigger = 0)
595 {
596 $sql = 'DELETE FROM '.MAIN_DB_PREFIX.'asset_depreciation_codes_economic WHERE fk_asset = '.((int) $this->id);
597 $this->db->query($sql);
598
599 $sql = 'DELETE FROM '.MAIN_DB_PREFIX.'asset_depreciation_codes_fiscal WHERE fk_asset = '.((int) $this->id);
600 $this->db->query($sql);
601
602 $sql = 'DELETE FROM '.MAIN_DB_PREFIX.'asset_depreciation_options_fiscal WHERE fk_asset = '.((int) $this->id);
603 $this->db->query($sql);
604
605 $sql = 'DELETE FROM '.MAIN_DB_PREFIX.'asset_depreciation_options_economic WHERE fk_asset = '.((int) $this->id);
606 $this->db->query($sql);
607
608 $sql = 'DELETE FROM '.MAIN_DB_PREFIX.'asset_depreciation WHERE fk_asset = '.((int) $this->id);
609 $this->db->query($sql);
610
611 return $this->deleteCommon($user, $notrigger);
612 //return $this->deleteCommon($user, $notrigger, 1);
613 }
614
622 public function setDataFromAssetModel(User $user, $notrigger = 0)
623 {
624 global $langs;
625 $langs->load('assets');
626
627 // Clean parameters
628 $this->id = $this->id > 0 ? $this->id : 0;
629 $this->fk_asset_model = $this->fk_asset_model > 0 ? $this->fk_asset_model : 0;
630
631 // Check parameters
632 $error = 0;
633 if (empty($this->id)) {
634 $this->errors[] = $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("Asset") . ' (' . $langs->transnoentitiesnoconv("TechnicalID") . ')');
635 $error++;
636 }
637 if (empty($this->fk_asset_model)) {
638 $this->errors[] = $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("AssetModel") . ' (' . $langs->transnoentitiesnoconv("TechnicalID") . ')');
639 $error++;
640 }
641 if ($error) {
642 return -1;
643 }
644
645 $this->db->begin();
646
647 // Get depreciation options
648 //---------------------------
649 require_once DOL_DOCUMENT_ROOT . '/asset/class/assetdepreciationoptions.class.php';
650 $options_model = new AssetDepreciationOptions($this->db);
651 $result = $options_model->fetchDeprecationOptions(0, $this->fk_asset_model);
652 if ($result < 0) {
653 $this->error = $options_model->error;
654 $this->errors = $options_model->errors;
655 $error++;
656 } elseif ($result > 0) {
657 $options = new AssetDepreciationOptions($this->db);
658 $result = $options->fetchDeprecationOptions($this->id);
659 if ($result < 0) {
660 $this->error = $options->error;
661 $this->errors = $options->errors;
662 $error++;
663 }
664
665 if (!$error) {
666 foreach ($options_model->deprecation_options as $mode_key => $fields) {
667 foreach ($fields as $field_key => $value) {
668 $options->deprecation_options[$mode_key][$field_key] = $value;
669 }
670 }
671
672 $result = $options->updateDeprecationOptions($user, $this->id, 0, $notrigger);
673 if ($result < 0) {
674 $this->error = $options->error;
675 $this->errors = $options->errors;
676 $error++;
677 }
678 }
679 }
680
681 // Get accountancy codes
682 //---------------------------
683 if (!$error) {
684 require_once DOL_DOCUMENT_ROOT . '/asset/class/assetaccountancycodes.class.php';
685 $accountancy_codes_model = new AssetAccountancyCodes($this->db);
686 $result = $accountancy_codes_model->fetchAccountancyCodes(0, $this->fk_asset_model);
687 if ($result < 0) {
688 $this->error = $accountancy_codes_model->error;
689 $this->errors = $accountancy_codes_model->errors;
690 $error++;
691 } elseif ($result > 0) {
692 $accountancy_codes = new AssetAccountancyCodes($this->db);
693 $result = $accountancy_codes->fetchAccountancyCodes($this->id);
694 if ($result < 0) {
695 $this->error = $accountancy_codes->error;
696 $this->errors = $accountancy_codes->errors;
697 $error++;
698 }
699
700 if (!$error) {
701 foreach ($accountancy_codes_model->accountancy_codes as $mode_key => $fields) {
702 foreach ($fields as $field_key => $value) {
703 $accountancy_codes->accountancy_codes[$mode_key][$field_key] = $value;
704 }
705 }
706
707 $result = $accountancy_codes->updateAccountancyCodes($user, $this->id, 0, $notrigger);
708 if ($result < 0) {
709 $this->error = $accountancy_codes->error;
710 $this->errors = $accountancy_codes->errors;
711 $error++;
712 }
713 }
714 }
715 }
716
717 if ($error) {
718 $this->db->rollback();
719 return -1;
720 } else {
721 $this->db->commit();
722 return 1;
723 }
724 }
725
731 public function fetchDepreciationLines()
732 {
733 global $langs;
734 $langs->load('assets');
735 $this->depreciation_lines = array();
736
737 // Clean parameters
738 $this->id = $this->id > 0 ? $this->id : 0;
739
740 // Check parameters
741 $error = 0;
742 if (empty($this->id)) {
743 $this->errors[] = $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("Asset") . ' (' . $langs->transnoentitiesnoconv("TechnicalID") . ')');
744 $error++;
745 }
746 if ($error) {
747 return -1;
748 }
749
750 $sql = "SELECT ad.rowid, ad.depreciation_mode, ad.ref, ad.depreciation_date, ad.depreciation_ht, ad.cumulative_depreciation_ht,";
751 $sql .= " " . $this->db->ifsql('EXISTS (SELECT fk_docdet FROM ' . MAIN_DB_PREFIX . 'accounting_bookkeeping as ab WHERE ab.doc_type = \'asset\' AND ab.fk_docdet = ad.rowid)', '1', '0') . " AS bookkeeping";
752 $sql .= " FROM " . MAIN_DB_PREFIX . "asset_depreciation AS ad";
753 $sql .= " WHERE ad.fk_asset = " . (int) $this->id;
754 $sql .= " ORDER BY ad.depreciation_date ASC";
755
756 $resql = $this->db->query($sql);
757 if (!$resql) {
758 $this->errors[] = $langs->trans('AssetErrorFetchDepreciationLines') . ': ' . $this->db->lasterror();
759 return -1;
760 }
761
762 while ($obj = $this->db->fetch_object($resql)) {
763 if (!isset($this->depreciation_lines[$obj->depreciation_mode])) {
764 $this->depreciation_lines[$obj->depreciation_mode] = array();
765 }
766 $this->depreciation_lines[$obj->depreciation_mode][] = array(
767 'id' => $obj->rowid,
768 'ref' => $obj->ref,
769 'depreciation_date' => $this->db->jdate($obj->depreciation_date),
770 'depreciation_ht' => $obj->depreciation_ht,
771 'cumulative_depreciation_ht' => $obj->cumulative_depreciation_ht,
772 'bookkeeping' => $obj->bookkeeping,
773 );
774 }
775
776 return 1;
777 }
778
785 {
786 global $langs;
787 $langs->load('assets');
788
789 // Clean parameters
790 $this->id = $this->id > 0 ? $this->id : 0;
791
792 // Check parameters
793 $error = 0;
794 if (empty($this->id)) {
795 $this->errors[] = $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("Asset") . ' (' . $langs->transnoentitiesnoconv("TechnicalID") . ')');
796 $error++;
797 }
798 if ($error) {
799 return -1;
800 }
801
802 // Old request with 'WITH'
803 /*
804 $sql = "WITH in_accounting_bookkeeping(fk_docdet) AS (";
805 $sql .= " SELECT DISTINCT fk_docdet";
806 $sql .= " FROM " . MAIN_DB_PREFIX . "accounting_bookkeeping";
807 $sql .= " WHERE doc_type = 'asset'";
808 $sql .= ")";
809 $sql .= "SELECT COUNT(*) AS has_bookkeeping";
810 $sql .= " FROM " . MAIN_DB_PREFIX . "asset_depreciation AS ad";
811 $sql .= " LEFT JOIN in_accounting_bookkeeping as iab ON iab.fk_docdet = ad.rowid";
812 $sql .= " WHERE ad.fk_asset = " . (int) $this->id;
813 $sql .= " AND iab.fk_docdet IS NOT NULL";
814 */
815
816 $sql = "SELECT COUNT(*) AS has_bookkeeping";
817 $sql .= " FROM " . MAIN_DB_PREFIX . "asset_depreciation AS ad";
818 $sql .= " LEFT JOIN (SELECT DISTINCT fk_docdet FROM " . MAIN_DB_PREFIX . "accounting_bookkeeping WHERE doc_type = 'asset') AS iab ON iab.fk_docdet = ad.rowid";
819 $sql .= " WHERE ad.fk_asset = " . (int) $this->id;
820 $sql .= " AND iab.fk_docdet IS NOT NULL";
821
822 $resql = $this->db->query($sql);
823 if (!$resql) {
824 $this->errors[] = $langs->trans('AssetErrorFetchDepreciationLines') . ': ' . $this->db->lasterror();
825 return -1;
826 }
827
828 if ($obj = $this->db->fetch_object($resql)) {
829 return $obj->has_bookkeeping > 0 ? 1 : 0;
830 }
831
832 return 0;
833 }
834
847 public function addDepreciationLine($mode, $ref, $depreciation_date, $depreciation_ht, $cumulative_depreciation_ht, $accountancy_code_debit, $accountancy_code_credit)
848 {
849 global $langs;
850 $langs->load('assets');
851
852 // Clean parameters
853 $this->id = $this->id > 0 ? $this->id : 0;
854 $mode = strtolower(trim($mode));
855 $ref = trim($ref);
856 $accountancy_code_debit = trim($accountancy_code_debit);
857 $accountancy_code_credit = trim($accountancy_code_credit);
858
859 // Check parameters
860 $error = 0;
861 if (empty($this->id)) {
862 $this->errors[] = $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("Asset") . ' (' . $langs->transnoentitiesnoconv("TechnicalID") . ')');
863 $error++;
864 }
865 if ($error) {
866 return -1;
867 }
868
869 $sql = "INSERT INTO " . MAIN_DB_PREFIX . "asset_depreciation(fk_asset, depreciation_mode, ref, depreciation_date, depreciation_ht, cumulative_depreciation_ht, accountancy_code_debit, accountancy_code_credit)";
870 $sql .= " VALUES ( ";
871 $sql .= " " . (int) $this->id;
872 $sql .= ", '" . $this->db->escape($mode) . "'";
873 $sql .= ", '" . $this->db->escape($ref) . "'";
874 $sql .= ", '" . $this->db->idate($depreciation_date) . "'";
875 $sql .= ", " . (float) $depreciation_ht;
876 $sql .= ", " . (float) $cumulative_depreciation_ht;
877 $sql .= ", '" . $this->db->escape($accountancy_code_debit) . "'";
878 $sql .= ", '" . $this->db->escape($accountancy_code_credit) . "'";
879 $sql .= ")";
880
881 $resql = $this->db->query($sql);
882 if (!$resql) {
883 $this->errors[] = $langs->trans('AssetErrorAddDepreciationLine') . ': ' . $this->db->lasterror();
884 return -1;
885 }
886
887 return 1;
888 }
889
895 public function calculationDepreciation()
896 {
897 global $conf, $langs;
898 $langs->load('assets');
899
900 // Clean parameters
901 $this->id = $this->id > 0 ? $this->id : 0;
902
903 // Check parameters
904 $error = 0;
905 if (empty($this->id)) {
906 $this->errors[] = $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("Asset") . ' (' . $langs->transnoentitiesnoconv("TechnicalID") . ')');
907 $error++;
908 }
909 if ($error) {
910 return -1;
911 }
912
913 if (! empty($this->not_depreciated)) {
914 return 1;
915 }
916
917 // Get depreciation options
918 //---------------------------
919 require_once DOL_DOCUMENT_ROOT . '/asset/class/assetdepreciationoptions.class.php';
920 $options = new AssetDepreciationOptions($this->db);
921 $result = $options->fetchDeprecationOptions($this->id);
922 if ($result < 0) {
923 $this->error = $options->error;
924 $this->errors = $options->errors;
925 return -1;
926 }
927
928 // Get accountancy codes
929 //---------------------------
930 require_once DOL_DOCUMENT_ROOT . '/asset/class/assetaccountancycodes.class.php';
931 $accountancy_codes = new AssetAccountancyCodes($this->db);
932 $result = $accountancy_codes->fetchAccountancyCodes($this->id);
933 if ($result < 0) {
934 $this->error = $accountancy_codes->error;
935 $this->errors = $accountancy_codes->errors;
936 return -1;
937 }
938
939 $this->db->begin();
940
941 // Delete old lines
942 $modes = array();
943 foreach ($options->deprecation_options as $mode_key => $fields) {
944 $modes[$mode_key] = $this->db->escape($mode_key);
945 }
946 $sql = "DELETE FROM " . MAIN_DB_PREFIX . "asset_depreciation";
947 $sql .= " WHERE fk_asset = " . (int) $this->id;
948 $sql .= " AND depreciation_mode NOT IN ('" . $this->db->sanitize(implode("', '", $modes)) . "')";
949
950 $resql = $this->db->query($sql);
951 if (!$resql) {
952 $this->errors[] = $langs->trans('AssetErrorClearDepreciationLines') . ': ' . $this->db->lasterror();
953 $error++;
954 }
955
956 if (!$error) {
957 // Get fiscal period
958 require_once DOL_DOCUMENT_ROOT . '/core/lib/date.lib.php';
959 require_once DOL_DOCUMENT_ROOT . '/core/lib/accounting.lib.php';
960
961 // @FIXME getCurrentPeriodOfFiscalYear return the first period found. What if there is several ? And what if not closed ? And what if end date not yet defined.
962 $dates = getCurrentPeriodOfFiscalYear($this->db, $conf, $this->date_start > $this->date_acquisition ? $this->date_start : $this->date_acquisition);
963 $init_fiscal_period_start = $dates['date_start'];
964 $init_fiscal_period_end = $dates['date_end'];
965 /*
966 if (empty($init_fiscal_period_start) || empty($init_fiscal_period_end)) {
967 $pastmonthyear = $dates['pastmonthyear'];
968 $pastmonth = $dates['pastmonth'];
969 $init_fiscal_period_start = dol_get_first_day((int) $pastmonthyear, (int) $pastmonth, false);
970 $init_fiscal_period_end = dol_get_last_day((int) $pastmonthyear, (int) $pastmonth, false);
971 }
972 */
973
974 foreach ($options->deprecation_options as $mode_key => $fields) {
975 // Get last depreciation lines save in bookkeeping
976 //-----------------------------------------------------
977
978 // Old request with 'WITH'
979 /*
980 $sql = "WITH in_accounting_bookkeeping(fk_docdet) AS (";
981 $sql .= " SELECT fk_docdet";
982 $sql .= " FROM " . MAIN_DB_PREFIX . "accounting_bookkeeping";
983 $sql .= " WHERE doc_type = 'asset'";
984 $sql .= ")";
985 $sql .= "SELECT ad.depreciation_date, ad.cumulative_depreciation_ht";
986 $sql .= " FROM " . MAIN_DB_PREFIX . "asset_depreciation AS ad";
987 $sql .= " LEFT JOIN in_accounting_bookkeeping as iab ON iab.fk_docdet = ad.rowid";
988 $sql .= " WHERE ad.fk_asset = " . (int) $this->id;
989 $sql .= " AND ad.depreciation_mode = '" . $this->db->escape($mode_key) . "'";
990 $sql .= " AND iab.fk_docdet IS NOT NULL";
991 $sql .= " ORDER BY ad.depreciation_date DESC";
992 $sql .= " LIMIT 1";
993 */
994
995 $sql = "SELECT ad.depreciation_date, ad.cumulative_depreciation_ht";
996 $sql .= " FROM " . MAIN_DB_PREFIX . "asset_depreciation AS ad";
997 $sql .= " LEFT JOIN (SELECT DISTINCT fk_docdet FROM " . MAIN_DB_PREFIX . "accounting_bookkeeping WHERE doc_type = 'asset') AS iab ON iab.fk_docdet = ad.rowid";
998 $sql .= " WHERE ad.fk_asset = " . (int) $this->id;
999 $sql .= " AND ad.depreciation_mode = '" . $this->db->escape($mode_key) . "'";
1000 $sql .= " AND iab.fk_docdet IS NOT NULL";
1001 $sql .= " ORDER BY ad.depreciation_date DESC";
1002 $sql .= " LIMIT 1";
1003
1004 $resql = $this->db->query($sql);
1005 if (!$resql) {
1006 $this->errors[] = $langs->trans('AssetErrorFetchMaxDepreciationDateForMode', $mode_key) . ': ' . $this->db->lasterror();
1007 $error++;
1008 break;
1009 }
1010 $last_depreciation_date = '';
1011 $last_cumulative_depreciation_ht = $this->reversal_amount_ht;
1012 if ($obj = $this->db->fetch_object($resql)) {
1013 $last_depreciation_date = $this->db->jdate($obj->depreciation_date);
1014 $last_cumulative_depreciation_ht = $obj->cumulative_depreciation_ht;
1015 }
1016
1017 // Set last cumulative depreciation
1018 $sql = "UPDATE " . MAIN_DB_PREFIX . $options->deprecation_options_fields[$mode_key]['table'];
1019 $sql .= " SET total_amount_last_depreciation_ht = " . (empty($last_cumulative_depreciation_ht) ? 0 : $last_cumulative_depreciation_ht);
1020 $sql .= " WHERE fk_asset = " . (int) $this->id;
1021 $resql = $this->db->query($sql);
1022 if (!$resql) {
1023 $this->errors[] = $langs->trans('AssetErrorSetLastCumulativeDepreciation') . ': ' . $this->db->lasterror();
1024 $error++;
1025 break;
1026 }
1027
1028 // Delete old lines
1029 $sql = "DELETE FROM " . MAIN_DB_PREFIX . "asset_depreciation";
1030 $sql .= " WHERE fk_asset = " . (int) $this->id;
1031 $sql .= " AND depreciation_mode = '" . $this->db->escape($mode_key) . "'";
1032 $sql .= " AND NOT EXISTS (SELECT fk_docdet FROM " . MAIN_DB_PREFIX . "accounting_bookkeeping WHERE doc_type = 'asset' AND fk_docdet = " . MAIN_DB_PREFIX . "asset_depreciation.rowid)";
1033 if ($last_depreciation_date !== "") {
1034 $sql .= " AND ref <> ''";
1035 }
1036 $resql = $this->db->query($sql);
1037 if (!$resql) {
1038 $this->errors[] = $langs->trans('AssetErrorClearDepreciationLines') . ': ' . $this->db->lasterror();
1039 $error++;
1040 break;
1041 }
1042
1043 // Get depreciation period
1044 $depreciation_date_start = $this->date_start > $this->date_acquisition ? $this->date_start : $this->date_acquisition;
1045 $depreciation_date_end = dol_time_plus_duree(dol_time_plus_duree((int) $depreciation_date_start, (float) $fields['duration'], $fields['duration_type'] == 1 ? 'm' : ($fields['duration_type'] == 2 ? 'd' : 'y')), -1, 'd');
1046 $depreciation_amount = $fields['amount_base_depreciation_ht'];
1047 if ($fields['duration_type'] == 2) { // Daily
1048 $fiscal_period_start = $depreciation_date_start;
1049 $fiscal_period_end = $depreciation_date_start;
1050 } elseif ($fields['duration_type'] == 1) { // Monthly
1051 $date_temp = dol_getdate((int) $depreciation_date_start);
1052 $fiscal_period_start = dol_get_first_day($date_temp['year'], $date_temp['mon'], false);
1053 $fiscal_period_end = dol_get_last_day($date_temp['year'], $date_temp['mon'], false);
1054 } else { // Annually
1055 $fiscal_period_start = $init_fiscal_period_start;
1056 $fiscal_period_end = $init_fiscal_period_end;
1057 }
1058 $cumulative_depreciation_ht = (float) $last_cumulative_depreciation_ht;
1059 $depreciation_period_amount = $depreciation_amount - (float) $this->reversal_amount_ht;
1060 $start_date = $depreciation_date_start;
1061 $disposal_date = isset($this->disposal_date) && $this->disposal_date !== "" ? $this->disposal_date : "";
1062 $finish_date = $disposal_date !== "" ? $disposal_date : $depreciation_date_end;
1063 $accountancy_code_depreciation_debit_key = $accountancy_codes->accountancy_codes_fields[$mode_key]['depreciation_debit'];
1064 $accountancy_code_depreciation_debit = $accountancy_codes->accountancy_codes[$mode_key][$accountancy_code_depreciation_debit_key];
1065 $accountancy_code_depreciation_credit_key = $accountancy_codes->accountancy_codes_fields[$mode_key]['depreciation_credit'];
1066 $accountancy_code_credit = $accountancy_codes->accountancy_codes[$mode_key][$accountancy_code_depreciation_credit_key];
1067
1068 // Reversal depreciation line
1069 //-----------------------------------------------------
1070 if ($last_depreciation_date === "" && ($depreciation_date_start < $fiscal_period_start || is_numeric($this->reversal_date))) {
1071 if (is_numeric($this->reversal_date)) {
1072 if ($this->reversal_date < $fiscal_period_start) {
1073 $this->errors[] = $langs->trans('AssetErrorReversalDateNotGreaterThanCurrentBeginFiscalDateForMode', $mode_key);
1074 $error++;
1075 break;
1076 }
1077
1078 if (empty($this->reversal_amount_ht)) {
1079 $this->errors[] = $langs->trans('AssetErrorReversalAmountNotProvidedForMode', $mode_key);
1080 $error++;
1081 break;
1082 }
1083
1084 $start_date = $this->reversal_date;
1085 $result = $this->addDepreciationLine($mode_key, '', $start_date, (float) $this->reversal_amount_ht, (float) $this->reversal_amount_ht, $accountancy_code_depreciation_debit, $accountancy_code_credit);
1086 if ($result < 0) {
1087 $error++;
1088 break;
1089 }
1090 } else {
1091 $this->errors[] = $langs->trans('AssetErrorReversalDateNotProvidedForMode', $mode_key);
1092 $error++;
1093 break;
1094 }
1095 }
1096
1097 // futures depreciation lines
1098 //-----------------------------------------------------
1099 $nb_days_in_year = getDolGlobalInt('ASSET_DEPRECIATION_DURATION_PER_YEAR', 360);
1100 $nb_days_in_month = getDolGlobalInt('ASSET_DEPRECIATION_DURATION_PER_MONTH', 30);
1101 $period_amount = (float) ($fields['duration'] > 0 ? price2num($depreciation_period_amount / $fields['duration'], 'MT') : 0);
1102 $first_period_found = false;
1103
1104 //$first_period_date = isset($begin_period) && $begin_period > $fiscal_period_start ? $begin_period : $fiscal_period_start;
1105 $first_period_date = $fiscal_period_start;
1106
1107 $ref_date_format = "%Y" . ($fields['duration_type'] == 1 || $fields['duration_type'] == 2 ? '-%m' : '') . ($fields['duration_type'] == 2 ? '-%d' : '');
1108
1109 // Loop security
1110 $idx_loop = 0;
1111 $max_loop = $fields['duration'] + 2;
1112 do {
1113 // Loop security
1114 $idx_loop++;
1115 if ($idx_loop > $max_loop) {
1116 break;
1117 }
1118
1119 if ($last_depreciation_date < $fiscal_period_end && ($first_period_date <= $start_date || $first_period_found)) {
1120 // Disposal not depreciated
1121 if ($fiscal_period_start <= $disposal_date && $disposal_date <= $fiscal_period_end && empty($this->disposal_depreciated)) {
1122 break;
1123 }
1124
1125 $first_period_found = true;
1126
1127 $period_begin = dol_print_date($fiscal_period_start, $ref_date_format);
1128 $period_end = dol_print_date($fiscal_period_end, $ref_date_format);
1129 $ref = $period_begin . ($period_begin != $period_end ? ' - ' . $period_end : '');
1130 if ($fiscal_period_start <= $disposal_date && $disposal_date <= $fiscal_period_end) {
1131 $ref .= ' - ' . $langs->transnoentitiesnoconv('AssetDisposal');
1132 }
1133
1134 $begin_date = $fiscal_period_start < $start_date && $start_date <= $fiscal_period_end ? $start_date : $fiscal_period_start;
1135 $end_date = $fiscal_period_start < $finish_date && $finish_date <= $fiscal_period_end ? $finish_date : $fiscal_period_end;
1136 if ($fields['duration_type'] == 2) { // Daily
1137 $depreciation_ht = $period_amount;
1138 } elseif ($fields['duration_type'] == 1) { // Monthly
1139 $nb_days = min($nb_days_in_month, num_between_day($begin_date, $end_date, 1));
1140 if ($nb_days >= 28) {
1141 $date_temp = dol_getdate($begin_date);
1142 if ($date_temp['mon'] == 2) {
1143 $nb_days = 30;
1144 }
1145 }
1146 $depreciation_ht = (float) price2num($period_amount * $nb_days / $nb_days_in_month, 'MT');
1147 } else { // Annually, taking care for adjustments to shortened or extended periods (e.g., fiscal years of 9 or 15 months)
1148 $nb_days_real = num_between_day($begin_date, $end_date, 1);
1149 if (($nb_days_real > 366) || (num_between_day($fiscal_period_start, $fiscal_period_end, 1) < $nb_days_in_year)) { // FY Period changed
1150 $nb_days = $nb_days_real;
1151 } else {
1152 $nb_days = min($nb_days_in_year, $nb_days_real);
1153 }
1154 $depreciation_ht = (float) price2num($period_amount * $nb_days / $nb_days_in_year, 'MT');
1155 }
1156 if (getDolGlobalInt('ASSET_ROUND_INTEGER_NUMBER_UPWARDS') == 1) {
1157 if ($idx_loop < $max_loop) { // avoid last depreciation value
1158 $depreciation_ht = ceil($depreciation_ht);
1159 }
1160 }
1161
1162 if ($fiscal_period_start <= $depreciation_date_end && $depreciation_date_end <= $fiscal_period_end) { // last period
1163 $depreciation_ht = (float) price2num($depreciation_amount - (float) $cumulative_depreciation_ht, 'MT');
1164 $cumulative_depreciation_ht = $depreciation_amount;
1165 } else {
1166 $cumulative_depreciation_ht += $depreciation_ht;
1167 }
1168
1169 $result = $this->addDepreciationLine($mode_key, $ref, $fiscal_period_end, $depreciation_ht, (float) $cumulative_depreciation_ht, $accountancy_code_depreciation_debit, $accountancy_code_credit);
1170 if ($result < 0) {
1171 $error++;
1172 break;
1173 }
1174 }
1175
1176 // Next fiscal period (+1 day/month/year)
1177 $fiscal_period_start = dol_time_plus_duree($fiscal_period_end, 1, 'd');
1178
1179 if ($fields['duration_type'] == 0) { // Annually - use fiscal year-end
1180 // Search for the NEXT fiscal year
1181 $next_fy = getNextFiscalYear($this->db, $fiscal_period_end, 'gmt');
1182
1183 if ($next_fy !== null) {
1184 // Exercise found in database
1185 $fiscal_period_start = $next_fy['date_start'];
1186 $fiscal_period_end = $next_fy['date_end'];
1187 } else {
1188 // If there is no subsequent fiscal year in the database, a one-year fiscal year is automatically generated.
1189 // based on the structure of the current fiscal year
1190 $current_fy_start_parts = dol_getdate($fiscal_period_start);
1191
1192 // Calculate the end: 1 year minus 1 day from the start
1193 $fiscal_period_end = dol_time_plus_duree(dol_time_plus_duree($fiscal_period_start, 1, 'y'), -1, 'd');
1194
1195 // If the amortization end date is exceeded, there is no need to continue after that.
1196 // The loop will end naturally with the while condition.
1197 }
1198 } elseif ($fields['duration_type'] == 1) { // Monthly
1199 $fiscal_period_end = dol_time_plus_duree(dol_time_plus_duree($fiscal_period_start, 1, 'm'), -1, 'd');
1200 } else { // Daily
1201 $fiscal_period_end = $fiscal_period_start;
1202 }
1203 $last_period_date = $disposal_date !== "" && $disposal_date < $depreciation_date_end ? $disposal_date : $depreciation_date_end;
1204 } while ($fiscal_period_start < $last_period_date);
1205
1206 if ($error) {
1207 break;
1208 }
1209 }
1210 }
1211
1212 if ($error) {
1213 $this->db->rollback();
1214 return -1;
1215 } else {
1216 $this->db->commit();
1217 return 1;
1218 }
1219 }
1220
1227 public function setLastCumulativeDepreciation($asset_depreciation_id)
1228 {
1229 global $langs;
1230 $langs->load('assets');
1231
1232 // Clean parameters
1233 $asset_depreciation_id = $asset_depreciation_id > 0 ? $asset_depreciation_id : 0;
1234
1235 // Check parameters
1236 $error = 0;
1237 if (empty($asset_depreciation_id)) {
1238 $this->errors[] = $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("AssetDepreciation") . ' (' . $langs->transnoentitiesnoconv("TechnicalID") . ')');
1239 $error++;
1240 }
1241 if ($error) {
1242 return -1;
1243 }
1244
1245 $this->db->begin();
1246
1247 require_once DOL_DOCUMENT_ROOT . '/asset/class/assetdepreciationoptions.class.php';
1248 $options = new AssetDepreciationOptions($this->db);
1249
1250 // Get last depreciation lines save in bookkeeping
1251 //-----------------------------------------------------
1252 $sql = "SELECT fk_asset, depreciation_mode, cumulative_depreciation_ht";
1253 $sql .= " FROM " . MAIN_DB_PREFIX . "asset_depreciation";
1254 $sql .= " WHERE rowid = " . (int) $asset_depreciation_id;
1255 $resql = $this->db->query($sql);
1256 if (!$resql) {
1257 $this->errors[] = $langs->trans('AssetErrorFetchCumulativeDepreciation') . ': ' . $this->db->lasterror();
1258 $error++;
1259 } else {
1260 if ($obj = $this->db->fetch_object($resql)) {
1261 $mode_key = $obj->depreciation_mode;
1262 if (!empty($options->deprecation_options_fields[$mode_key])) {
1263 $sql = "UPDATE " . MAIN_DB_PREFIX . $options->deprecation_options_fields[$mode_key]['table'];
1264 $sql .= " SET total_amount_last_depreciation_ht = " . $obj->cumulative_depreciation_ht;
1265 $sql .= " WHERE fk_asset = " . (int) $obj->fk_asset;
1266 $resql = $this->db->query($sql);
1267 if (!$resql) {
1268 $this->errors[] = $langs->trans('AssetErrorSetLastCumulativeDepreciation') . ': ' . $this->db->lasterror();
1269 $error++;
1270 }
1271 }
1272 }
1273 }
1274
1275 if ($error) {
1276 $this->db->rollback();
1277 return -1;
1278 } else {
1279 $this->db->commit();
1280 return 1;
1281 }
1282 }
1283
1292 public function dispose($user, $disposal_invoice_id, $notrigger = 0)
1293 {
1294 global $conf, $langs;
1295
1296 // Protection
1297 if ($this->status == self::STATUS_DISPOSED) {
1298 return 0;
1299 }
1300
1301 $this->db->begin();
1302
1303 $required_fields = array('disposal_date', 'disposal_date', 'fk_disposal_type');
1304 foreach ($required_fields as $field) {
1305 $this->fields[$field]['notnull'] = 1;
1306 }
1307 $result = $this->update($user, 1);
1308 foreach ($required_fields as $field) {
1309 $this->fields[$field]['notnull'] = 0;
1310 }
1311 if ($result > 0) {
1312 if ($disposal_invoice_id > 0) {
1313 $this->add_object_linked('facture', $disposal_invoice_id);
1314 }
1315 $result = $this->setStatusCommon($user, self::STATUS_DISPOSED, $notrigger, 'ASSET_DISPOSED');
1316 }
1317 if ($result > 0) {
1318 $result = $this->calculationDepreciation();
1319 }
1320
1321 if ($result < 0) {
1322 $this->db->rollback();
1323 } else {
1324 $this->db->commit();
1325 }
1326
1327 // Define output language
1328 if ($result > 0 && !getDolGlobalString('MAIN_DISABLE_PDF_AUTOUPDATE')) {
1329 if (method_exists($this, 'generateDocument')) {
1330 global $hidedetails, $hidedesc, $hideref;
1331 $outputlangs = $langs;
1332 $newlang = '';
1333 if (getDolGlobalInt('MAIN_MULTILANGS') /* && empty($newlang) */ && GETPOST('lang_id', 'aZ09')) {
1334 $newlang = GETPOST('lang_id', 'aZ09');
1335 }
1336 if (getDolGlobalInt('MAIN_MULTILANGS') && empty($newlang)) {
1337 $newlang = $this->thirdparty->default_lang;
1338 }
1339 if (!empty($newlang)) {
1340 $outputlangs = new Translate("", $conf);
1341 $outputlangs->setDefaultLang($newlang);
1342 }
1343 $model = $this->model_pdf;
1344 $ret = $this->fetch($this->id); // Reload to get new records
1345
1346 $this->generateDocument($model, $outputlangs, $hidedetails, $hidedesc, $hideref);
1347 }
1348 }
1349
1350 return $result;
1351 }
1352
1360 public function reopen($user, $notrigger = 0)
1361 {
1362 global $conf, $langs;
1363
1364 // Protection
1365 if ($this->status != self::STATUS_DISPOSED) {
1366 return 0;
1367 }
1368
1369
1370 $this->db->begin();
1371
1372 $this->disposal_date = null;
1373 $this->disposal_amount_ht = null;
1374 $this->fk_disposal_type = null;
1375 $this->disposal_depreciated = 0;
1376 $this->disposal_subject_to_vat = 0;
1377 $result = $this->update($user, 1);
1378 if ($result > 0) {
1379 $this->deleteObjectLinked(null, 'facture');
1380 $result = $this->setStatusCommon($user, self::STATUS_VALIDATED, $notrigger, 'ASSET_REOPEN');
1381 }
1382 if ($result > 0) {
1383 $result = $this->calculationDepreciation();
1384 }
1385
1386 if ($result < 0) {
1387 $this->db->rollback();
1388 } else {
1389 $this->db->commit();
1390 }
1391
1392 // Define output language
1393 if ($result > 0 && !getDolGlobalString('MAIN_DISABLE_PDF_AUTOUPDATE')) {
1394 if (method_exists($this, 'generateDocument')) {
1395 global $hidedetails, $hidedesc, $hideref;
1396 $outputlangs = $langs;
1397 $newlang = '';
1398 if (getDolGlobalInt('MAIN_MULTILANGS') /* && empty($newlang) */ && GETPOST('lang_id', 'aZ09')) {
1399 $newlang = GETPOST('lang_id', 'aZ09');
1400 }
1401 if (getDolGlobalInt('MAIN_MULTILANGS') && empty($newlang)) {
1402 $newlang = $this->thirdparty->default_lang;
1403 }
1404 if (!empty($newlang)) {
1405 $outputlangs = new Translate("", $conf);
1406 $outputlangs->setDefaultLang($newlang);
1407 }
1408 $model = $this->model_pdf;
1409 $ret = $this->fetch($this->id); // Reload to get new records
1410
1411 $this->generateDocument($model, $outputlangs, $hidedetails, $hidedesc, $hideref);
1412 }
1413 }
1414
1415 return $result;
1416 }
1417
1429 public function getNomUrl($withpicto = 0, $option = '', $maxlen = 0, $notooltip = 0, $morecss = '', $save_lastsearch_value = -1)
1430 {
1431 global $db, $conf, $langs, $hookmanager;
1432 global $dolibarr_main_authentication, $dolibarr_main_demo;
1433 global $menumanager;
1434
1435 if (!empty($conf->dol_no_mouse_hover)) {
1436 $notooltip = 1; // Force disable tooltips
1437 }
1438
1439 $result = '';
1440
1441 $label = img_picto('', $this->picto).' <u>'.$langs->trans("Asset").'</u>';
1442 if (isset($this->status)) {
1443 $label .= ' '.$this->getLibStatut(5);
1444 }
1445 $label .= '<br>';
1446 $label .= '<b>'.$langs->trans('Ref').':</b> '.$this->ref;
1447
1448 $url = dol_buildpath('/asset/card.php', 1).'?id='.$this->id;
1449
1450 if ($option != 'nolink') {
1451 // Add param to save lastsearch_values or not
1452 $add_save_lastsearch_values = ($save_lastsearch_value == 1 ? 1 : 0);
1453 if ($save_lastsearch_value == -1 && isset($_SERVER["PHP_SELF"]) && preg_match('/list\.php/', $_SERVER["PHP_SELF"])) {
1454 $add_save_lastsearch_values = 1;
1455 }
1456 if ($add_save_lastsearch_values) {
1457 $url .= '&save_lastsearch_values=1';
1458 }
1459 }
1460
1461 $linkclose = '';
1462 if (empty($notooltip)) {
1463 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
1464 $label = $langs->trans("ShowAsset");
1465 $linkclose .= ' alt="'.dolPrintHTMLForAttribute($label).'"';
1466 }
1467 $linkclose .= ' title="'.dolPrintHTMLForAttribute($label).'"';
1468 $linkclose .= ' class="classfortooltip'.($morecss ? ' '.$morecss : '').'"';
1469 } else {
1470 $linkclose = ($morecss ? ' class="'.$morecss.'"' : '');
1471 }
1472
1473 if ($option == 'nolink') {
1474 $linkstart = '<span';
1475 } else {
1476 $linkstart = '<a href="'.$url.'"';
1477 }
1478 $linkstart .= $linkclose.'>';
1479 if ($option == 'nolink') {
1480 $linkend = '</span>';
1481 } else {
1482 $linkend = '</a>';
1483 }
1484
1485 $result .= $linkstart;
1486
1487 if (empty($this->showphoto_on_popup)) {
1488 if ($withpicto) {
1489 $result .= img_object(($notooltip ? '' : $label), ($this->picto ? $this->picto : 'generic'), ($notooltip ? (($withpicto != 2) ? 'class="paddingright"' : '') : 'class="'.(($withpicto != 2) ? 'paddingright ' : '').'classfortooltip"'), 0, 0, $notooltip ? 0 : 1);
1490 }
1491 } else {
1492 if ($withpicto) {
1493 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
1494
1495 list($class, $module) = explode('@', $this->picto);
1496 $upload_dir = $conf->$module->multidir_output[$conf->entity]."/$class/".dol_sanitizeFileName($this->ref);
1497 $filearray = dol_dir_list($upload_dir, "files");
1498 $filename = $filearray[0]['name'];
1499 if (!empty($filename)) {
1500 $pospoint = strpos($filearray[0]['name'], '.');
1501
1502 $pathtophoto = $class.'/'.$this->ref.'/thumbs/'.substr($filename, 0, $pospoint).'_mini'.substr($filename, $pospoint);
1503 if (!getDolGlobalString(strtoupper($module.'_'.$class).'_FORMATLISTPHOTOSASUSERS')) {
1504 $result .= '<div class="floatleft inline-block valignmiddle divphotoref"><div class="photoref"><img class="photo'.$module.'" alt="No photo" border="0" src="'.DOL_URL_ROOT.'/viewimage.php?modulepart='.$module.'&entity='.$conf->entity.'&file='.urlencode($pathtophoto).'"></div></div>';
1505 } else {
1506 $result .= '<div class="floatleft inline-block valignmiddle divphotoref"><img class="photouserphoto userphoto" alt="No photo" border="0" src="'.DOL_URL_ROOT.'/viewimage.php?modulepart='.$module.'&entity='.$conf->entity.'&file='.urlencode($pathtophoto).'"></div>';
1507 }
1508
1509 $result .= '</div>';
1510 } else {
1511 $result .= img_object(($notooltip ? '' : $label), ($this->picto ? $this->picto : 'generic'), ($notooltip ? (($withpicto != 2) ? 'class="paddingright"' : '') : 'class="'.(($withpicto != 2) ? 'paddingright ' : '').'classfortooltip"'), 0, 0, $notooltip ? 0 : 1);
1512 }
1513 }
1514 }
1515
1516 if ($withpicto != 2) {
1517 $name = $this->ref;
1518 if ($option == 'label') {
1519 $name = $this->label;
1520 } elseif ($option == 'with_label') {
1521 $name .= ' - ' . $this->label;
1522 }
1523 $result .= dol_escape_htmltag($maxlen ? dol_trunc($name, $maxlen) : $name);
1524 }
1525
1526 $result .= $linkend;
1527 //if ($withpicto != 2) $result.=(($addlabel && $this->label) ? $sep . dol_trunc($this->label, ($addlabel > 1 ? $addlabel : 0)) : '');
1528
1529 global $action;
1530 $hookmanager->initHooks(array($this->element . 'dao'));
1531 $parameters = array('id' => $this->id, 'getnomurl' => &$result);
1532 $reshook = $hookmanager->executeHooks('getNomUrl', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
1533 if ($reshook > 0) {
1534 $result = $hookmanager->resPrint;
1535 } else {
1536 $result .= $hookmanager->resPrint;
1537 }
1538 return $result;
1539 }
1540
1547 public function getLabelStatus($mode = 0)
1548 {
1549 return $this->LibStatut((int) $this->status, $mode);
1550 }
1551
1558 public function getLibStatut($mode = 0)
1559 {
1560 return $this->LibStatut((int) $this->status, $mode);
1561 }
1562
1563 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1571 public function LibStatut($status, $mode = 0)
1572 {
1573 // phpcs:enable
1574 if (empty($this->labelStatus) || empty($this->labelStatusShort)) {
1575 global $langs;
1576 //$langs->load("assets");
1577 $this->labelStatus[self::STATUS_DRAFT] = $langs->transnoentitiesnoconv('AssetInDraft');
1578 $this->labelStatus[self::STATUS_VALIDATED] = $langs->transnoentitiesnoconv('AssetInProgress');
1579 $this->labelStatus[self::STATUS_DISPOSED] = $langs->transnoentitiesnoconv('AssetDisposed');
1580 $this->labelStatusShort[self::STATUS_DRAFT] = $langs->transnoentitiesnoconv('AssetInDraft');
1581 $this->labelStatusShort[self::STATUS_VALIDATED] = $langs->transnoentitiesnoconv('AssetInProgress');
1582 $this->labelStatusShort[self::STATUS_DISPOSED] = $langs->transnoentitiesnoconv('AssetDisposed');
1583 }
1584
1585 $statusType = 'status'.$status;
1586 if ($status == self::STATUS_VALIDATED) {
1587 $statusType = 'status4';
1588 }
1589 if ($status == self::STATUS_DISPOSED) {
1590 $statusType = 'status6';
1591 }
1592
1593 return dolGetStatus($this->labelStatus[$status], $this->labelStatusShort[$status], '', $statusType, $mode);
1594 }
1595
1602 public function info($id)
1603 {
1604 $sql = "SELECT rowid, date_creation as datec, tms as datem, date_valid as datev,";
1605 $sql .= " fk_user_creat, fk_user_modif, fk_user_valid";
1606 $sql .= " FROM ".MAIN_DB_PREFIX.$this->table_element." as t";
1607 $sql .= " WHERE t.rowid = ".((int) $id);
1608
1609 $result = $this->db->query($sql);
1610 if ($result) {
1611 if ($this->db->num_rows($result)) {
1612 $obj = $this->db->fetch_object($result);
1613 $this->id = $obj->rowid;
1614
1615 $this->user_creation_id = $obj->fk_user_creat;
1616 $this->user_modification_id = $obj->fk_user_modif;
1617 $this->user_validation_id = $obj->fk_user_valid;
1618 $this->date_creation = $this->db->jdate($obj->datec);
1619 $this->date_modification = $this->db->jdate($obj->datem);
1620 $this->date_validation = $this->db->jdate($obj->datev);
1621 }
1622
1623 $this->db->free($result);
1624 } else {
1625 dol_print_error($this->db);
1626 }
1627 }
1628
1635 public function initAsSpecimen()
1636 {
1637 // Set here init that are not commonf fields
1638 // $this->property1 = ...
1639 // $this->property2 = ...
1640
1641 return $this->initAsSpecimenCommon();
1642 }
1643
1649 public function getNextNumRef()
1650 {
1651 global $langs, $conf;
1652 $langs->load("assets");
1653
1654 if (!getDolGlobalString('ASSET_ASSET_ADDON')) {
1655 $conf->global->ASSET_ASSET_ADDON = 'mod_asset_standard';
1656 }
1657
1658 if (getDolGlobalString('ASSET_ASSET_ADDON')) {
1659 $mybool = false;
1660
1661 $file = getDolGlobalString('ASSET_ASSET_ADDON') . ".php";
1662 $classname = getDolGlobalString('ASSET_ASSET_ADDON');
1663
1664 // Include file with class
1665 $dirmodels = array_merge(array('/'), (array) $conf->modules_parts['models']);
1666 foreach ($dirmodels as $reldir) {
1667 $dir = dol_buildpath($reldir."core/modules/asset/");
1668
1669 // Load file with numbering class (if found)
1670 $mybool = ((bool) @include_once $dir.$file) || $mybool;
1671 }
1672
1673 if (!$mybool) {
1674 dol_print_error(null, "Failed to include file ".$file);
1675 return '';
1676 }
1677
1678 if (class_exists($classname)) {
1679 $obj = new $classname();
1680
1681 '@phan-var-force ModeleNumRefAsset $obj';
1684 $numref = $obj->getNextValue($this);
1685
1686 if ($numref != '' && $numref != '-1') {
1687 return $numref;
1688 } else {
1689 $this->error = $obj->error;
1690 //dol_print_error($this->db,get_class($this)."::getNextNumRef ".$obj->error);
1691 return "";
1692 }
1693 } else {
1694 print $langs->trans("Error")." ".$langs->trans("ClassNotFound").' '.$classname;
1695 return "";
1696 }
1697 } else {
1698 print $langs->trans("ErrorNumberingModuleNotSetup", $this->element);
1699 return "";
1700 }
1701 }
1702
1710 public function validate($user, $notrigger = 0)
1711 {
1712 global $conf;
1713
1714 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
1715
1716 $error = 0;
1717
1718 // Protection
1719 if ($this->status == self::STATUS_VALIDATED) {
1720 dol_syslog(get_class($this)."::validate action abandoned: already validated", LOG_WARNING);
1721 return 0;
1722 }
1723
1724 $now = dol_now();
1725
1726 $this->db->begin();
1727
1728 // Define new ref
1729 if (/* !$error && */ (preg_match('/^[\‍(]?PROV/i', $this->ref) || empty($this->ref))) { // empty should not happened, but when it occurs, the test save life
1730 $num = $this->getNextNumRef();
1731 } else {
1732 $num = (string) $this->ref;
1733 }
1734 $this->newref = dol_sanitizeFileName($num);
1735
1736 // Validate
1737 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
1738 $sql .= " SET ref = '".$this->db->escape($num)."',";
1739 $sql .= " status = ".self::STATUS_VALIDATED.",";
1740 $sql .= " date_valid='".$this->db->idate($now)."',";
1741 $sql .= " fk_user_valid = ".((int) $user->id);
1742 $sql .= " WHERE rowid = ".((int) $this->id);
1743
1744 dol_syslog(get_class($this)."::validate()", LOG_DEBUG);
1745 $resql = $this->db->query($sql);
1746 if (!$resql) {
1747 dol_print_error($this->db);
1748 $this->error = $this->db->lasterror();
1749 $error++;
1750 }
1751
1752 if (!$error && !$notrigger) {
1753 // Call trigger
1754 $result = $this->call_trigger('ASSET_VALIDATE', $user);
1755 if ($result < 0) {
1756 $error++;
1757 }
1758 // End call triggers
1759 }
1760
1761 if (!$error) {
1762 $this->oldref = $this->ref;
1763
1764 // Rename directory if dir was a temporary ref
1765 if (preg_match('/^[\‍(]?PROV/i', $this->ref)) {
1766 // Now we rename also files into index
1767 $sql = 'UPDATE '.MAIN_DB_PREFIX."ecm_files set filename = CONCAT('".$this->db->escape($this->newref)."', SUBSTR(filename, ".(strlen($this->ref) + 1).")), filepath = 'asset/".$this->db->escape($this->newref)."'";
1768 $sql .= " WHERE filename LIKE '".$this->db->escape($this->ref)."%' AND filepath = 'bom/".$this->db->escape($this->ref)."' and entity = ".$conf->entity;
1769 $resql = $this->db->query($sql);
1770 if (!$resql) {
1771 $error++;
1772 $this->error = $this->db->lasterror();
1773 }
1774 $sql = 'UPDATE '.MAIN_DB_PREFIX."ecm_files set filepath = 'asset/".$this->db->escape($this->newref)."'";
1775 $sql .= " WHERE filepath = 'asset/".$this->db->escape($this->ref)."' and entity = ".$conf->entity;
1776 $resql = $this->db->query($sql);
1777 if (!$resql) {
1778 $error++;
1779 $this->error = $this->db->lasterror();
1780 }
1781
1782 // We rename directory ($this->ref = old ref, $num = new ref) in order not to lose the attachments
1783 $oldref = dol_sanitizeFileName($this->ref);
1784 $newref = dol_sanitizeFileName($num);
1785 $dirsource = $conf->asset->dir_output.'/'.$oldref;
1786 $dirdest = $conf->asset->dir_output.'/'.$newref;
1787 if (!$error && file_exists($dirsource)) {
1788 dol_syslog(get_class($this)."::validate() rename dir ".$dirsource." into ".$dirdest);
1789
1790 if (@rename($dirsource, $dirdest)) {
1791 dol_syslog("Rename ok");
1792 // Rename docs starting with $oldref with $newref
1793 $listoffiles = dol_dir_list($conf->asset->dir_output.'/'.$newref, 'files', 1, '^'.preg_quote($oldref, '/'));
1794 foreach ($listoffiles as $fileentry) {
1795 $dirsource = $fileentry['name'];
1796 $dirdest = preg_replace('/^'.preg_quote($oldref, '/').'/', $newref, $dirsource);
1797 $dirsource = $fileentry['path'].'/'.$dirsource;
1798 $dirdest = $fileentry['path'].'/'.$dirdest;
1799 @rename($dirsource, $dirdest);
1800 }
1801 }
1802 }
1803 }
1804 }
1805
1806 // Set new ref and current status
1807 if (!$error) {
1808 $this->ref = $num;
1809 $this->status = self::STATUS_VALIDATED;
1810 }
1811
1812 if (!$error) {
1813 $this->db->commit();
1814 return 1;
1815 } else {
1816 $this->db->rollback();
1817 return -1;
1818 }
1819 }
1820
1828 public function setDraft($user, $notrigger = 0)
1829 {
1830 // Protection
1831 if ($this->status <= self::STATUS_DRAFT) {
1832 return 0;
1833 }
1834
1835 return $this->setStatusCommon($user, self::STATUS_DRAFT, $notrigger, 'ASSET_UNVALIDATE');
1836 }
1837}
getCurrentPeriodOfFiscalYear($db, $conf, $from_time=null, $gm='tzserver', $withenddateonly=1)
Get current period of fiscal year?
getNextFiscalYear($db, $after_date, $gm='tzserver')
Get next fiscal year period after a given date.
$object ref
Definition info.php:90
Class for AssetAccountancyCodes.
Class for AssetDepreciationOptions.
Class for Asset.
getNomUrl($withpicto=0, $option='', $maxlen=0, $notooltip=0, $morecss='', $save_lastsearch_value=-1)
Return a link to the object card (with optionally the picto)
hasDepreciationLinesInBookkeeping()
If has depreciation lines in bookkeeping.
setDraft($user, $notrigger=0)
Set draft status.
dispose($user, $disposal_invoice_id, $notrigger=0)
Set dispose status.
validate($user, $notrigger=0)
Validate asset.
createFromClone(User $user, $fromid)
Clone an object into another one.
calculationDepreciation()
Calculation depreciation lines (reversal and future) for each mode.
__construct(DoliDB $db)
Constructor.
create(User $user, $notrigger=0)
Create object into database.
info($id)
Load the info information in the object.
getLabelStatus($mode=0)
Return the label of the status.
fetchAll($sortorder='', $sortfield='', $limit=0, $offset=0, $filter='', $filtermode='AND')
Load list of objects in memory from the database.
fetch($id, $ref=null)
Load object in memory from the database.
LibStatut($status, $mode=0)
Return the status.
reopen($user, $notrigger=0)
Set back to validated status if disposed status.
getLibStatut($mode=0)
Return the label of the status.
initAsSpecimen()
Initialise object with example values Id must be 0 if object instance is a specimen.
addDepreciationLine($mode, $ref, $depreciation_date, $depreciation_ht, $cumulative_depreciation_ht, $accountancy_code_debit, $accountancy_code_credit)
Add depreciation line for a mode.
fetchDepreciationLines()
Fetch depreciation lines for each mode in $this->depreciation_lines (sort by depreciation date)
setLastCumulativeDepreciation($asset_depreciation_id)
Set last cumulative depreciation for each mode.
update(User $user, $notrigger=0)
Update object into database.
setDataFromAssetModel(User $user, $notrigger=0)
Set asset model.
Parent class of all other business classes (invoices, contracts, proposals, orders,...
add_object_linked($origin=null, $origin_id=null, $f_user=null, $notrigger=0)
Add an object link into llx_element_element.
deleteObjectLinked($sourceid=null, $sourcetype='', $targetid=null, $targettype='', $rowid=0, $f_user=null, $notrigger=0)
Delete all links between an object $this.
createCommon(User $user, $notrigger=0)
Create object in the database.
getFieldList($alias='', $excludefields=array())
Function to concat keys of fields.
updateCommon(User $user, $notrigger=0)
Update object into database.
setStatusCommon($user, $status, $notrigger=0, $triggercode='')
Set to a status.
initAsSpecimenCommon()
Initialise object with example values Id must be 0 if object instance is a specimen.
fetchCommon($id, $ref=null, $morewhere='', $noextrafields=0)
Load object in memory from the database.
deleteCommon(User $user, $notrigger=0, $forcechilddeletion=0)
Delete object in database.
Class to manage Dolibarr database access.
Class to manage translations.
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_get_first_day($year, $month=1, $gm=false)
Return GMT time for first day of a month or year.
Definition date.lib.php:603
num_between_day($timestampStart, $timestampEnd, $lastday=0)
Function to return number of days between two dates (date must be UTC date !) Example: 2012-01-01 201...
dol_time_plus_duree($time, $duration_value, $duration_unit, $ruleforendofmonth=0)
Add a delay to a date.
Definition date.lib.php:125
dol_get_last_day($year, $month=12, $gm=false)
Return GMT time for last day of a month or year.
Definition date.lib.php:622
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:64
dol_now($mode='gmt')
Return date for now.
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)
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_sanitizeFileName($str, $newstr='_', $unaccent=1, $includequotes=0, $allowdash=0)
Clean a string to use it as a file name.
forgeSQLFromUniversalSearchCriteria($filter, &$errorstr='', $noand=0, $nopar=0, $noerror=0)
forgeSQLFromUniversalSearchCriteria
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
dolGetStatus($statusLabel='', $statusLabelShort='', $html='', $statusType='status0', $displayMode=0, $url='', $params=array())
Output the badge of a status.
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
dol_buildpath($path, $type=0, $returnemptyifnotfound=0)
Return path of url or filesystem.
dol_print_date($time, $format='', $tzoutput='auto', $outputlangs=null, $encodetooutput=false, $decorate=0)
Output date in a string format according to outputlangs (or langs if not defined).
dol_print_error($db=null, $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
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.
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
isModEnabled($module)
Is Dolibarr module enabled.
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.
dol_escape_htmltag($stringtoescape, $keepb=0, $keepn=0, $noescapetags='', $escapeonlyhtmltags=0, $cleanalsojavascript=0)
Returns text escaped for inclusion in HTML alt or title or value tags, or into values of HTML input f...