dolibarr/htdocs/core/modules/facture/modules_facture.php
2024-09-06 20:28:06 +08:00

288 lines
9.8 KiB
PHP

<?php
/* Copyright (C) 2003-2005 Rodolphe Quiedeville <rodolphe@quiedeville.org>
* Copyright (C) 2004-2011 Laurent Destailleur <eldy@users.sourceforge.net>
* Copyright (C) 2004 Eric Seigne <eric.seigne@ryxeo.com>
* Copyright (C) 2005-2012 Regis Houssin <regis.houssin@inodbox.com>
* Copyright (C) 2014 Marcos García <marcosgdf@gmail.com>
* Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
*
* 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/
*/
/**
* \file htdocs/core/modules/facture/modules_facture.php
* \ingroup invoice
* \brief File that contains parent class for invoices models
* and parent class for invoices numbering models
*/
require_once DOL_DOCUMENT_ROOT.'/core/class/commondocgenerator.class.php';
require_once DOL_DOCUMENT_ROOT.'/core/class/commonnumrefgenerator.class.php';
require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php';
require_once DOL_DOCUMENT_ROOT.'/compta/bank/class/account.class.php'; // Required because used in classes that inherit
// For the experimental feature using swiss QR invoice generated by composer lib sparin/swiss-qr-bill
use Sprain\SwissQrBill;
/**
* Parent class of invoice document generators
*/
abstract class ModelePDFFactures extends CommonDocGenerator
{
public $posxpicture;
public $posxtva;
public $posxup;
public $posxqty;
public $posxunit;
public $posxdesc;
public $posxdiscount;
public $postotalht;
public $tva;
public $tva_array;
public $localtax1;
public $localtax2;
public $atleastonediscount = 0;
public $atleastoneratenotnull = 0;
// phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
/**
* Return list of active generation modules
*
* @param DoliDB $db Database handler
* @param int<0,max> $maxfilenamelength Max length of value to show
* @return string[]|int<-1,0> List of templates
*/
public static function liste_modeles($db, $maxfilenamelength = 0)
{
// phpcs:enable
$type = 'invoice';
$list = array();
include_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
$list = getListOfModels($db, $type, $maxfilenamelength);
return $list;
}
// phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
/**
* Function to build pdf onto disk
*
* @param Facture $object Object to generate
* @param Translate $outputlangs Lang output object
* @param string $srctemplatepath Full path of source filename for generator using a template file
* @param int<0,1> $hidedetails Do not show line details
* @param int<0,1> $hidedesc Do not show desc
* @param int<0,1> $hideref Do not show ref
* @return int<-1,1> 1=OK, <=0=KO
*/
abstract public function write_file($object, $outputlangs, $srctemplatepath = '', $hidedetails = 0, $hidedesc = 0, $hideref = 0);
// phpcs:enable
/**
* Get the SwissQR object, including validation
*
* @param Facture $object Invoice object
* @param Translate $langs Translation object
* @return SwissQrBill\QrBill|bool The valid SwissQR object, or false
*/
private function getSwissQrBill(Facture $object, Translate $langs)
{
global $conf;
if (getDolGlobalString('INVOICE_ADD_SWISS_QR_CODE') != 'bottom') {
return false;
}
if ($object->mode_reglement_code != 'VIR') {
$this->error = $langs->transnoentities("SwissQrOnlyVIR");
return false;
}
if (empty($object->fk_account)) {
$this->error = 'Bank account must be defined to use this experimental feature';
return false;
}
// Load the autoload file generated by composer
if (file_exists(DOL_DOCUMENT_ROOT.'/includes/sprain/swiss-qr-bill/autoload.php')) {
require_once DOL_DOCUMENT_ROOT.'/includes/sprain/swiss-qr-bill/autoload.php';
} elseif (file_exists(DOL_DOCUMENT_ROOT.'/includes/autoload.php')) {
require_once DOL_DOCUMENT_ROOT.'/includes/autoload.php';
} else {
$this->error = 'PHP library sprain/swiss-qr-bill was not found. Please install it with:<br>cd '.dirname(DOL_DOCUMENT_ROOT).'; cp composer.json.disabled composer.json; composer require sprain/swiss-qr-bill;';
return false;
}
// Create a new instance of SwissQrBill, containing default headers with fixed values
$qrBill = SwissQrBill\QrBill::create();
// First, set creditor address
$address = SwissQrBill\DataGroup\Element\CombinedAddress::create(
$this->emetteur->name,
$this->emetteur->address,
$this->emetteur->zip . " " . $this->emetteur->town,
$this->emetteur->country_code
);
if (!$address->isValid()) {
$this->error = $langs->transnoentities("SwissQrCreditorAddressInvalid", (string) $address->getViolations());
return false;
}
$qrBill->setCreditor($address);
// Get IBAN from account.
$account = new Account($this->db);
$account->fetch($object->fk_account);
$creditorInformation = SwissQrBill\DataGroup\Element\CreditorInformation::create($account->iban);
if (!$creditorInformation->isValid()) {
$langs->load("errors");
$this->error = $langs->transnoentities("SwissQrCreditorInformationInvalid", $account->iban, (string) $creditorInformation->getViolations());
return false;
}
$qrBill->setCreditorInformation($creditorInformation);
if ($creditorInformation->containsQrIban()) {
$this->error = $langs->transnoentities("SwissQrIbanNotImplementedYet", $account->iban);
return false;
}
// Add payment reference CLASSIC-IBAN
// This is what you will need to identify incoming payments.
$qrBill->setPaymentReference(
SwissQrBill\DataGroup\Element\PaymentReference::create(
SwissQrBill\DataGroup\Element\PaymentReference::TYPE_NON
)
);
$currencyinvoicecode = $object->multicurrency_code ? $object->multicurrency_code : $conf->currency;
// Add payment amount, with currency
$pai = SwissQrBill\DataGroup\Element\PaymentAmountInformation::create($currencyinvoicecode, $object->total_ttc);
if (!$pai->isValid()) {
$this->error = $langs->transnoentities("SwissQrPaymentInformationInvalid", $object->total_ttc, (string) $pai->getViolations());
return false;
}
$qrBill->setPaymentAmountInformation($pai);
// Add some human-readable information about what the bill is for.
$qrBill->setAdditionalInformation(
SwissQrBill\DataGroup\Element\AdditionalInformation::create(
$object->ref
)
);
// Set debtor address; We _know_ zip&town have to be filled, so skip that if unfilled.
if (!empty($object->thirdparty->zip) && !empty($object->thirdparty->town)) {
$address = SwissQrBill\DataGroup\Element\CombinedAddress::create(
$object->thirdparty->name,
$object->thirdparty->address,
$object->thirdparty->zip . " " . $object->thirdparty->town,
$object->thirdparty->country_code
);
if (!$address->isValid()) {
$this->error = $langs->transnoentities("SwissQrDebitorAddressInvalid", (string) $address->getViolations());
return false;
}
$qrBill->setUltimateDebtor($address);
}
return $qrBill;
}
/**
* Get the height for bottom-page QR invoice in mm, depending on the page number.
*
* @param int $pagenbr Page number
* @param Facture $object Invoice object
* @param Translate $langs Translation object
* @return int Height in mm of the bottom-page QR invoice. Can be zero if not on right page; not enabled
*/
protected function getHeightForQRInvoice(int $pagenbr, Facture $object, Translate $langs)
{
if (getDolGlobalString('INVOICE_ADD_SWISS_QR_CODE') == 'bottom') {
// Keep it, to reset it after QRinvoice getter
$error = $this->error;
if (!$this->getSwissQrBill($object, $langs)) {
// Reset error to previous one if exists
if (!empty($error)) {
$this->error = $error;
}
return 0;
}
// SWIFT's requirementis 105, but we get more room with 100 and the page number is in a nice place.
return $pagenbr == 1 ? 100 : 0;
}
return 0;
}
/**
* Add SwissQR invoice at bottom of page 1
*
* @param TCPDF $pdf TCPDF object
* @param Facture $object Invoice object
* @param Translate $langs Translation object
* @return bool True for for success
*/
public function addBottomQRInvoice(TCPDF $pdf, Facture $object, Translate $langs): bool
{
if (!($qrBill = $this->getSwissQrBill($object, $langs))) {
return false;
}
try {
$pdf->startTransaction();
$pdf->setPage(1);
$pdf->SetTextColor(0, 0, 0);
$output = new SwissQrBill\PaymentPart\Output\TcPdfOutput\TcPdfOutput($qrBill, in_array($langs->shortlang, ['de', 'fr', 'it']) ? $langs->shortlang : 'en', $pdf);
$output->setPrintable(false)->getPaymentPart();
} catch (Exception $e) {
$pdf->rollbackTransaction(true);
return false;
}
return true;
}
}
/**
* Parent class of invoice reference numbering templates
*/
abstract class ModeleNumRefFactures extends CommonNumRefGenerator
{
/**
* Return next value not used or last value used
*
* @param Societe $objsoc Object third party
* @param ?Facture $invoice Object invoice
* @param string $mode 'next' for next value or 'last' for last value
* @return string|int<-1,0> Value if OK, <=0 if KO
*/
abstract public function getNextValue($objsoc, $invoice, $mode = 'next');
/**
* Return an example of numbering
*
* @return string Example
*/
abstract public function getExample();
}