HEX
Server: Apache
System: Linux srv13.cpanelhost.cl 3.10.0-962.3.2.lve1.5.38.el7.x86_64 #1 SMP Thu Jun 18 05:28:41 EDT 2020 x86_64
User: cca63905 (4205)
PHP: 7.3.20
Disabled: NONE
Upload Files
File: /home4/cca63905/www/guiaweb/htdocs/subtotals/class/commonsubtotal.class.php
<?php
/* Copyright (C) 2014-2017  Laurent Destailleur     <eldy@users.sourceforge.net>
 * Copyright (C) 2024		MDW						<mdeweerd@users.noreply.github.com>
 * Copyright (C) 2024-2025  Frédéric France         <frederic.france@free.fr>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program. If not, see <https://www.gnu.org/licenses/>.
 * or see https://www.gnu.org/
 */

if (!defined('SUBTOTALS_SPECIAL_CODE')) {
	define('SUBTOTALS_SPECIAL_CODE', 81);
}

/**
 *
 * Trait CommonSubtotal
 *
 * Add subtotals lines
 */
trait CommonSubtotal
{
	/**
	 * @var int
	 * Type for subtotals module lines
	 */
	public static $PRODUCT_TYPE = 9;

	/**
	 * @var array<string>
	 * Options for subtotals module title lines
	 */
	public static $TITLE_OPTIONS = ['titleshowuponpdf', 'titleshowtotalexludingvatonpdf', 'titleforcepagebreak'];

	/**
	 * @var array<string>
	 * Options for subtotals module subtotal lines
	 */
	public static $SUBTOTAL_OPTIONS = ['subtotalshowtotalexludingvatonpdf'];

	/**
	 * Adds a subtotals line to a document.
	 * This function inserts a subtotal line based on the given parameters.
	 *
	 * @param Translate						$langs  		Translation.
	 * @param string						$desc			Description of the line.
	 * @param int							$depth			Level of the line (>0 for title lines, <0 for subtotal lines)
	 * @param array<string,string>|string	$options		Subtotal options for pdf view
	 * @param int							$parent_line	ID of the parent line for shipments
	 * @return int									ID of the added line if successful, 0 on warning, -1 on error
	 *
	 * @phan-suppress PhanUndeclaredMethod
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function addSubtotalLine($langs, $desc, $depth, $options = array(), $parent_line = 0)
	{
		if (empty($desc)) {
			if (isset($this->errors)) {
				$this->errors[] = $langs->trans("TitleNeedDesc");
			}
			return -1;
		}
		$current_module = $this->element;
		// Ensure the object is one of the supported types
		$allowed_types = array('propal', 'commande', 'facture', 'facturerec', 'shipping');
		if (!in_array($current_module, $allowed_types)) {
			if (isset($this->errors)) {
				$this->errors[] = $langs->trans("UnsupportedModuleError");
			}
			return -1; // Unsupported type
		}
		$error = 0;
		$desc = dol_html_entity_decode($desc, ENT_QUOTES);
		$rang = -1;
		$next_line = false;
		$result = 0;

		if ($depth < 0 && $current_module != 'shipping') {
			foreach ($this->lines as $line) {
				if (!$next_line && $line->desc == $desc && $line->qty == -$depth) {
					$next_line = true;
					continue;
				}
				if ($next_line && $line->desc == $desc && $line->qty == $depth) {
					$next_line = false;
					continue;
				}
				if ($next_line && $line->special_code == SUBTOTALS_SPECIAL_CODE && abs($line->qty) <= abs($depth)) {
					$rang = $line->rang;
					break;
				}
			}
		}

		if ($depth > 0 && $current_module != 'shipping') {
			$max_existing_level = 0;

			foreach ($this->lines as $line) {
				if ($line->special_code == SUBTOTALS_SPECIAL_CODE && $line->qty > $max_existing_level) {
					$max_existing_level = $line->qty;
				}
			}

			if ($max_existing_level+1 < $depth) {
				$depth = $max_existing_level+1;
				if (isset($this->errors)) {
					$this->errors[] = $langs->trans("TitleAddedLevelTooHigh", $depth);
				}

				$error ++;
			}
		}

		// Add the line calling the right module
		if ($current_module == 'facture') {
			$result = $this->addline( // @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				0,						// FK product @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				'',						// Date start @phpstan-ignore-line
				'',						// Date end @phpstan-ignore-line
				0,						// FK code ventilation @phpstan-ignore-line
				0,						// Info bits @phpstan-ignore-line
				0,						// FK remise except @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0,						// PU ttc @phpstan-ignore-line
				self::$PRODUCT_TYPE,	// Type @phpstan-ignore-line
				$rang,					// Rang @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE	// Special code @phpstan-ignore-line
			);
		} elseif ($current_module == 'propal') {
			$result = $this->addline( // @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				0,						// FK product @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0,						// PU ttc @phpstan-ignore-line
				0,						// Info bits @phpstan-ignore-line
				self::$PRODUCT_TYPE,	// Type @phpstan-ignore-line
				$rang,					// Rang @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE	// Special code @phpstan-ignore-line
			);
		} elseif ($current_module == 'commande') {
			$result = $this->addline( // @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				0,						// FK product @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				0,						// Info bits @phpstan-ignore-line
				0,						// FK remise except @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0,						// PU ttc @phpstan-ignore-line
				'',						// Date start @phpstan-ignore-line
				'',						// Date end @phpstan-ignore-line
				self::$PRODUCT_TYPE,	// Type @phpstan-ignore-line
				$rang,					// Rang @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE	// Special code @phpstan-ignore-line
			);
		} elseif ($current_module == 'shipping') {
			$result = $this->addline( // @phpstan-ignore-line
				'',						// Warehouse ID @phpstan-ignore-line
				(int) $parent_line,		// Source line @phpstan-ignore-line
				$depth					// Quantity @phpstan-ignore-line
			);
		} elseif ($current_module == 'facturerec') {
			$rang = $rang == -1 ? $rang : $rang-1;
			$result = $this->addline( // @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				0,						// FK product @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0,						// Info bits @phpstan-ignore-line
				0,						// FK remise except @phpstan-ignore-line
				0,						// PU ttc @phpstan-ignore-line
				self::$PRODUCT_TYPE,	// Type @phpstan-ignore-line
				$rang,					// Rang @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE	// Special code @phpstan-ignore-line
			);
			$this->fetch_lines();
		}

		if ($current_module != 'shipping') {
			foreach ($this->lines as $line) {
				'@phan-var-force CommonObjectLine $line';
				if ($line->id == $result) {
					$line->extraparams["subtotal"] = $options;
					$line->setExtraParameters();
				}
			}
		}

		if ($result < 0) {
			return $result;
		}

		return $error > 0 ? 0 : $result;
	}

	/**
	 * Deletes a subtotal or a title line from a document.
	 * If the corresponding subtotal line exists and second parameter true, it will also be deleted.
	 *
	 * @param Translate	$langs					Translation.
	 * @param int		$id						ID of the line to delete
	 * @param bool		$correspondingstline	If true, also deletes the corresponding subtotal line
	 * @param User		$user					performing the deletion (used for permissions in some modules)
	 * @return int								ID of deleted line if successful, -1 on error
	 *
	 * @phan-suppress PhanUndeclaredMethod
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function deleteSubtotalLine($langs, $id, $correspondingstline = false, $user = null)
	{
		$current_module = $this->element;
		// Ensure the object is one of the supported types
		$allowed_types = array('propal', 'commande', 'facture', 'facturerec', 'shipping');
		if (!in_array($current_module, $allowed_types)) {
			if (isset($this->errors)) {
				$this->errors[] = $langs->trans("UnsupportedModuleError");
			}
			return -1; // Unsupported type
		}

		$result = 0;

		if ($correspondingstline) {
			$oldDesc = "";
			$oldDepth =  0;
			foreach ($this->lines as $line) {
				if ($line->id == $id) {
					$oldDesc = $line->desc;
					$oldDepth = $line->qty;
				}
				if ($line->special_code == SUBTOTALS_SPECIAL_CODE && $line->qty == -$oldDepth && $line->desc == $oldDesc) {
					$this->deleteSubtotalLine($langs, $line->id, false, $user);
					break;
				}
			}
		}

		// Add the line calling the right module
		if ($current_module == 'facture') {
			$result = $this->deleteLine($id); // @phpstan-ignore-line
		} elseif ($current_module == 'propal') {
			$result = $this->deleteLine($id); // @phpstan-ignore-line
		} elseif ($current_module == 'commande') {
			$result = $this->deleteLine($user, $id); // @phpstan-ignore-line
		} elseif ($current_module == 'facturerec') {
			$line = new FactureLigneRec($this->db);
			$line->id = $id;
			$result = $line->delete($user); // @phpstan-ignore-line
		} elseif ($current_module == 'shipping') {
			$line = new ExpeditionLigne($this->db);
			$line->id = $id;
			$result = $line->delete($user); // @phpstan-ignore-line
		}

		return $result >= 0 ? $result : -1; // Return line ID or false
	}

	/**
	 * Updates a subtotal line of a document.
	 * This function updates a subtotals line based on its id and the given parameters.
	 * Updating a title line updates the corresponding subtotal line except options.
	 *
	 * @param Translate						$langs  	Translation.
	 * @param int							$lineid  	ID of the line to update.
	 * @param string						$desc		Description of the line.
	 * @param int							$depth		Level of the line (>0 for title lines, <0 for subtotal lines)
	 * @param array<string,string>|string	$options	Subtotal options for pdf view
	 * @return int									ID of the added line if successful, 0 on warning, -1 on error
	 *
	 * @phan-suppress PhanUndeclaredMethod
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function updateSubtotalLine($langs, $lineid, $desc, $depth, $options) // @phpstan-ignore-line
	{
		$current_module = $this->element;
		// Ensure the object is one of the supported types
		$allowed_types = array('propal', 'commande', 'facture', 'facturerec', 'shipping');
		if (!in_array($current_module, $allowed_types)) {
			if (isset($this->errors)) {
				$this->errors[] = $langs->trans("UnsupportedModuleError");
			}
			return -1; // Unsupported type
		}

		$result = 0;
		$error = 0;

		$max_existing_level = 0;

		if ($depth>0) {
			foreach ($this->lines as $line) {
				if ($line->special_code == SUBTOTALS_SPECIAL_CODE && $line->qty > $max_existing_level && $line->id != $lineid) {
					$max_existing_level = $line->qty;
				}
			}
		}

		if ($max_existing_level+1 < $depth) {
			$depth = $max_existing_level+1;
			if (isset($this->errors)) {
				$this->errors[] = $langs->trans("TitleEditedLevelTooHigh");
			}
			$error ++;
		}

		if ($depth>0) {
			$oldDesc = "";
			$oldDepth =  0;
			foreach ($this->lines as $line) {
				if ($line->id == $lineid) {
					$oldDesc = $line->desc;
					$oldDepth = $line->qty;
				}
				if ($line->special_code == SUBTOTALS_SPECIAL_CODE && $line->qty == -$oldDepth && $line->desc == $oldDesc) {
					$this->updateSubtotalLine($langs, $line->id, $desc, -$depth, !empty($line->extraparams["subtotal"]) ? $line->extraparams["subtotal"] : array());
					break;
				}
			}
		}

		// Update the line calling the right module
		if ($current_module == 'facture') {
			$result = $this->updateline( // @phpstan-ignore-line
				$lineid, 				// ID of line to change @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				'',						// Date start @phpstan-ignore-line
				'',						// Date end @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0, 						// Info bits @phpstan-ignore-line
				self::$PRODUCT_TYPE,	// Type @phpstan-ignore-line
				0,						// FK parent line @phpstan-ignore-line
				0,						// Skip update total @phpstan-ignore-line
				0,						// FK fournprice @phpstan-ignore-line
				0,						// PA ht @phpstan-ignore-line
				'',						// Label @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE	// Special code @phpstan-ignore-line
			);
		} elseif ($current_module == 'propal') {
			$result = $this->updateline( // @phpstan-ignore-line
				$lineid, 				// ID of line to change @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0,						// Info bits @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE, // Special code @phpstan-ignore-line
				0, 						// FK parent line @phpstan-ignore-line
				0, 						// Skip update total @phpstan-ignore-line
				0, 						// FK fournprice @phpstan-ignore-line
				0, 						// PA ht @phpstan-ignore-line
				'',						// Label @phpstan-ignore-line
				self::$PRODUCT_TYPE		// Type @phpstan-ignore-line
			);
		} elseif ($current_module == 'commande') {
			$result = $this->updateline( // @phpstan-ignore-line
				$lineid, 				// ID of line to change @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0,						// Info bits @phpstan-ignore-line
				'',						// Date start @phpstan-ignore-line
				'',						// Date end @phpstan-ignore-line
				self::$PRODUCT_TYPE,	// Type @phpstan-ignore-line
				0, 						// FK parent line @phpstan-ignore-line
				0, 						// Skip update total @phpstan-ignore-line
				0, 						// FK fournprice @phpstan-ignore-line
				0, 						// PA ht @phpstan-ignore-line
				'',						// Label @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE	// Special code @phpstan-ignore-line
			);
		} elseif ($current_module == 'facturerec') {
			$objectline = new FactureLigneRec($this->db);
			$objectline->fetch($lineid);
			$line_rang = $objectline->rang;
			$result = $this->updateline( // @phpstan-ignore-line
				$lineid,				// ID of line to change @phpstan-ignore-line
				$desc,					// Description @phpstan-ignore-line
				0,						// Unit price @phpstan-ignore-line
				$depth,					// Quantity @phpstan-ignore-line
				0,						// VAT rate @phpstan-ignore-line
				0,						// Local tax 1 @phpstan-ignore-line
				0,						// Local tax 2 @phpstan-ignore-line
				0,						// FK parent line @phpstan-ignore-line
				0,						// Discount percentage @phpstan-ignore-line
				'',						// Price base type @phpstan-ignore-line
				0,						// Info bits @phpstan-ignore-line
				0,						// FK parent line @phpstan-ignore-line
				0,						// PU ttc @phpstan-ignore-line
				self::$PRODUCT_TYPE,	// Type @phpstan-ignore-line
				$line_rang,				// Rang @phpstan-ignore-line
				SUBTOTALS_SPECIAL_CODE	// Special code @phpstan-ignore-line
			);
		}

		foreach ($this->lines as $line) {
			'@phan-var-force CommonObjectLine $line';
			/** @var CommonObjectLine $line */
			if ($line->id == $lineid) {
				$line->extraparams["subtotal"] = $options;
				$line->setExtraParameters();
			}
		}

		if ($result < 0) {
			return $result;
		}

		return $error > 0 ? 0 : $result;
	}

	/**
	 * Updates a block of lines of a document.
	 *
	 * @param Translate	$langs  	Translation.
	 * @param int		$linerang	Rang of the line to start from.
	 * @param string	$mode		Column to change (discount or vat).
	 * @param int		$value		Value of the change.
	 * @return int					Return integer < 0 if KO, 1 if OK
	 *
	 * @phan-suppress PhanUndeclaredMethod
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function updateSubtotalLineBlockLines($langs, $linerang, $mode, $value) // @phpstan-ignore-line
	{
		$current_module = $this->element;
		// Ensure the object is one of the supported types
		$allowed_types = array('propal', 'commande', 'facture', 'facturerec', 'shipping');
		if (!in_array($current_module, $allowed_types)) {
			if (isset($this->errors)) {
				$this->errors[] = $langs->trans("UnsupportedModuleError");
			}
			return -1; // Unsupported type
		}

		$result = 0;
		$linerang -= 1;

		$nb_lines = count($this->lines)+1;

		for ($i = $linerang+1; $i < $nb_lines; $i++) {
			if ($this->lines[$i]->special_code == SUBTOTALS_SPECIAL_CODE) {
				if (abs($this->lines[$i]->qty) <= (int) $this->lines[$linerang]->qty) {
					return 1;
				}
			} else {
				if ($current_module == 'facture') {
					$result = $this->updateline( // @phpstan-ignore-line
						$this->lines[$i]->id, // @phpstan-ignore-line
						$this->lines[$i]->desc, // @phpstan-ignore-line
						$this->lines[$i]->subprice, // @phpstan-ignore-line
						$this->lines[$i]->qty, // @phpstan-ignore-line
						$mode == 'discount' ? $value : $this->lines[$i]->remise_percent, // @phpstan-ignore-line
						$this->lines[$i]->date_start, // @phpstan-ignore-line
						$this->lines[$i]->date_end, // @phpstan-ignore-line
						$mode == 'tva' ? $value : $this->lines[$i]->tva_tx, // @phpstan-ignore-line
						$this->lines[$i]->localtax1_tx, // @phpstan-ignore-line
						$this->lines[$i]->localtax2_tx, // @phpstan-ignore-line
						'HT', // @phpstan-ignore-line
						$this->lines[$i]->info_bits, // @phpstan-ignore-line
						$this->lines[$i]->product_type, // @phpstan-ignore-line
						$this->lines[$i]->fk_parent_line, 0, // @phpstan-ignore-line
						$this->lines[$i]->fk_fournprice, // @phpstan-ignore-line
						$this->lines[$i]->pa_ht, // @phpstan-ignore-line
						$this->lines[$i]->label, // @phpstan-ignore-line
						$this->lines[$i]->special_code, // @phpstan-ignore-line
						$this->lines[$i]->array_options, // @phpstan-ignore-line
						$this->lines[$i]->situation_percent, // @phpstan-ignore-line
						$this->lines[$i]->fk_unit, // @phpstan-ignore-line
						$this->lines[$i]->multicurrency_subprice // @phpstan-ignore-line
					);
				} elseif ($current_module == 'commande') {
					$result = $this->updateline( // @phpstan-ignore-line
						$this->lines[$i]->id, // @phpstan-ignore-line
						$this->lines[$i]->desc, // @phpstan-ignore-line
						$this->lines[$i]->subprice, // @phpstan-ignore-line
						$this->lines[$i]->qty, // @phpstan-ignore-line
						$mode == 'discount' ? $value : $this->lines[$i]->remise_percent, // @phpstan-ignore-line
						$mode == 'tva' ? $value : $this->lines[$i]->tva_tx, // @phpstan-ignore-line
						$this->lines[$i]->localtax1_rate, // @phpstan-ignore-line
						$this->lines[$i]->localtax2_rate, // @phpstan-ignore-line
						'HT', // @phpstan-ignore-line
						$this->lines[$i]->info_bits, // @phpstan-ignore-line
						$this->lines[$i]->date_start, // @phpstan-ignore-line
						$this->lines[$i]->date_end, // @phpstan-ignore-line
						$this->lines[$i]->product_type, // @phpstan-ignore-line
						$this->lines[$i]->fk_parent_line, 0, // @phpstan-ignore-line
						$this->lines[$i]->fk_fournprice, // @phpstan-ignore-line
						$this->lines[$i]->pa_ht, // @phpstan-ignore-line
						$this->lines[$i]->label, // @phpstan-ignore-line
						$this->lines[$i]->special_code, // @phpstan-ignore-line
						$this->lines[$i]->array_options, // @phpstan-ignore-line
						$this->lines[$i]->fk_unit, // @phpstan-ignore-line
						$this->lines[$i]->multicurrency_subprice // @phpstan-ignore-line
					);
				} elseif ($current_module == 'propal') {
					$result = $this->updateline( // @phpstan-ignore-line
						$this->lines[$i]->id, // @phpstan-ignore-line
						$this->lines[$i]->subprice, // @phpstan-ignore-line
						$this->lines[$i]->qty, // @phpstan-ignore-line
						$mode == 'discount' ? $value : $this->lines[$i]->remise_percent, // @phpstan-ignore-line
						$mode == 'tva' ? $value : $this->lines[$i]->tva_tx, // @phpstan-ignore-line
						$this->lines[$i]->localtax1_rate, // @phpstan-ignore-line
						$this->lines[$i]->localtax2_rate, // @phpstan-ignore-line
						$this->lines[$i]->desc, // @phpstan-ignore-line
						'HT', // @phpstan-ignore-line
						$this->lines[$i]->info_bits, // @phpstan-ignore-line
						$this->lines[$i]->special_code, // @phpstan-ignore-line
						$this->lines[$i]->fk_parent_line, 0, // @phpstan-ignore-line
						$this->lines[$i]->fk_fournprice, // @phpstan-ignore-line
						$this->lines[$i]->pa_ht, // @phpstan-ignore-line
						$this->lines[$i]->label, // @phpstan-ignore-line
						$this->lines[$i]->product_type, // @phpstan-ignore-line
						$this->lines[$i]->date_start, // @phpstan-ignore-line
						$this->lines[$i]->date_end, // @phpstan-ignore-line
						$this->lines[$i]->array_options, // @phpstan-ignore-line
						$this->lines[$i]->fk_unit, // @phpstan-ignore-line
						$this->lines[$i]->multicurrency_subprice // @phpstan-ignore-line
					);
				}
				if ($result < 0) {
					return $result;
				}
			}
		}
		return 1;
	}

	/**
	 * Return the total_ht of lines that are above the current line (excluded) and that are not a subtotal line
	 * until a title line of the same level is found
	 *
	 * @param object	$line	Line that needs the subtotal amount.
	 * @return string	$total_ht
	 *
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function getSubtotalLineAmount($line)
	{
		$final_amount = 0;
		for ($i = $line->rang-1; $i > 0; $i--) {
			if (is_null($this->lines[$i-1]) || $this->lines[$i-1]->rang >= $line->rang) {
				continue;
			}
			if ($this->lines[$i-1]->special_code == SUBTOTALS_SPECIAL_CODE && $this->lines[$i-1]->qty > 0) {
				if ($this->lines[$i-1]->qty <= abs($line->qty)) {
					return price($final_amount);
				}
			} else {
				$final_amount += $this->lines[$i-1]->total_ht;
			}
		}
		return price($final_amount);
	}

	/**
	 * Return the multicurrency_total_ht of lines that are above the current line (excluded) and that are not a subtotal line
	 * until a title line of the same level is found
	 *
	 * @param object	$line	Line that needs the subtotal amount with multicurrency mod activated.
	 * @return string	$total_ht
	 *
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function getSubtotalLineMulticurrencyAmount($line)
	{
		$final_amount = 0;
		for ($i = $line->rang-1; $i > 0; $i--) {
			if (is_null($this->lines[$i-1]) || $this->lines[$i-1]->rang >= $line->rang) {
				continue;
			}
			if ($this->lines[$i-1]->special_code == SUBTOTALS_SPECIAL_CODE && $this->lines[$i-1]->qty>0) {
				if ($this->lines[$i-1]->qty <= abs($line->qty)) {
					return price($final_amount);
				}
			} else {
				$final_amount += $this->lines[$i-1]->multicurrency_total_ht;
			}
		}
		return price($final_amount);
	}

	/**
	 * Retrieve the background color associated with a specific subtotal level.
	 *
	 * @param int|float $level The level of the subtotal for which the color is requested.
	 * @return string|null The background color in hexadecimal format or null if not set.
	 */
	public function getSubtotalColors($level)
	{
		return getDolGlobalString('SUBTOTAL_BACK_COLOR_LEVEL_'.abs($level));
	}

	/**
	 * Retrieve current object possible titles to choose from
	 *
	 * @return array<string,string> The set of titles, empty if no title line set.
	 *
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function getPossibleTitles()
	{
		$titles = array();
		foreach ($this->lines as $line) {
			if ($line->special_code == SUBTOTALS_SPECIAL_CODE && $line->qty > 0) {
				$titles[$line->desc] = $line->desc;
			}
			if ($line->special_code == SUBTOTALS_SPECIAL_CODE && $line->qty < 0) {
				unset($titles[$line->desc]);
			}
		}
		return $titles;
	}

	/**
	 * Retrieve the current object possible levels (defined in admin page)
	 *
	 * @param Translate $langs 		Translations.
	 * @return array<int,string>	The set of possible levels, empty if not defined correctly.
	 *
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function getPossibleLevels($langs)
	{
		$depth_array = array();
		$max_depth = getDolGlobalString('SUBTOTAL_'.strtoupper($this->element).'_MAX_DEPTH', 2);
		for ($i = 0; $i < $max_depth; $i++) {
			$depth_array[$i + 1] = $langs->trans("SubtotalLevel", $i + 1);
		}
		return $depth_array;
	}

	/**
	 * Returns an array with the IDs of the line that we don't need to show to avoid empty blocks
	 *
	 * @return array<int>	$total_ht
	 *
	 * @phan-suppress PhanUndeclaredProperty
	 */
	public function getDisabledShippmentSubtotalLines()
	{
		$toDisableLines = array();
		$toDisable = true;
		$oldDesc = "";
		$oldDepth =  0;

		foreach ($this->lines as $titleLine) {
			if ($titleLine->special_code != SUBTOTALS_SPECIAL_CODE || $titleLine->qty <= 0) {
				continue;
			}
			foreach ($this->lines as $line) {
				if ($line->id == $titleLine->id) {
					$oldDesc = $line->desc;
					$oldDepth = $line->qty;
				}
				if ($line->special_code != SUBTOTALS_SPECIAL_CODE && $line->fk_product_type == 0 && !empty($oldDesc) && !empty($oldDepth)) {
					$toDisable = false;
				}
				if ($line->special_code == SUBTOTALS_SPECIAL_CODE && $line->qty == -$oldDepth && $line->desc == $oldDesc) {
					if ($toDisable) {
						$toDisableLines = array_merge($toDisableLines, array($titleLine->id, $line->id));
					}
					$oldDesc = "";
					$oldDepth =  0;
					$toDisable = true;
					break;
				}
			}
		}
		return $toDisableLines;
	}
}