dolibarr.bankimport/card.php
data b552b90303 feat: PaymentVarious - Zahlung anlegen für Transaktionen ohne Rechnung
Neuer Button "Zahlung anlegen" auf der Transaktions-Detailseite (card.php)
für Bankbuchungen ohne zugehörige Rechnung (z.B. Steuer-Erstattungen,
private Umbuchungen, sonstige Zahlungen).

- Inline-Formular mit Buchungskonto (Select2), Nebenbuchkonto,
  Buchungsseite (Soll/Haben) und Label
- Buchungsseite wird automatisch aus Vorzeichen ermittelt
- Erstellt PaymentVarious mit Bank-Eintrag und bank_url-Verknüpfung
- Transaktion wird auf MATCHED gesetzt
- Anzeige der sonstigen Zahlung mit Link bei gematchten Transaktionen
- Fix: update() speichert jetzt fk_user_match und date_match korrekt
- Fix: Leeres Nebenbuchkonto wird nicht mehr als -1 gespeichert

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-10 15:52:20 +01:00

1429 lines
54 KiB
PHP
Executable file

<?php
/* Copyright (C) 2026 Eduard Wisch <data@data-it-solution.de>
*
* 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.
*/
/**
* \file bankimport/card.php
* \ingroup bankimport
* \brief Card page for a single bank transaction
*/
// Load Dolibarr environment
$res = 0;
if (!$res && !empty($_SERVER["CONTEXT_DOCUMENT_ROOT"])) {
$res = @include $_SERVER["CONTEXT_DOCUMENT_ROOT"]."/main.inc.php";
}
$tmp = empty($_SERVER['SCRIPT_FILENAME']) ? '' : $_SERVER['SCRIPT_FILENAME'];
$tmp2 = realpath(__FILE__);
$i = strlen($tmp) - 1;
$j = strlen($tmp2) - 1;
while ($i > 0 && $j > 0 && isset($tmp[$i]) && isset($tmp2[$j]) && $tmp[$i] == $tmp2[$j]) {
$i--;
$j--;
}
if (!$res && $i > 0 && file_exists(substr($tmp, 0, ($i + 1))."/main.inc.php")) {
$res = @include substr($tmp, 0, ($i + 1))."/main.inc.php";
}
if (!$res && $i > 0 && file_exists(dirname(substr($tmp, 0, ($i + 1)))."/main.inc.php")) {
$res = @include dirname(substr($tmp, 0, ($i + 1)))."/main.inc.php";
}
if (!$res && file_exists("../main.inc.php")) {
$res = @include "../main.inc.php";
}
if (!$res && file_exists("../../main.inc.php")) {
$res = @include "../../main.inc.php";
}
if (!$res) {
die("Include of main fails");
}
require_once DOL_DOCUMENT_ROOT.'/core/lib/date.lib.php';
require_once DOL_DOCUMENT_ROOT.'/core/class/html.formother.class.php';
dol_include_once('/bankimport/class/banktransaction.class.php');
dol_include_once('/bankimport/lib/bankimport.lib.php');
/**
* @var Conf $conf
* @var DoliDB $db
* @var Translate $langs
* @var User $user
*/
$langs->loadLangs(array("bankimport@bankimport", "banks", "bills"));
$id = GETPOSTINT('id');
$ref = GETPOST('ref', 'alpha');
$action = GETPOST('action', 'aZ09');
$confirm = GETPOST('confirm', 'alpha');
// Security check
if (!$user->hasRight('bankimport', 'read')) {
accessforbidden();
}
/*
* Actions
*/
$object = new BankImportTransaction($db);
if ($id > 0 || !empty($ref)) {
$result = $object->fetch($id, $ref);
if ($result <= 0) {
setEventMessages($langs->trans("RecordNotFound"), null, 'errors');
}
}
// Set status
if ($action == 'setstatus' && $object->id > 0) {
$newstatus = GETPOSTINT('status');
$result = $object->setStatus($newstatus, $user);
if ($result > 0) {
setEventMessages($langs->trans("StatusUpdated"), null, 'mesgs');
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
} else {
setEventMessages($object->error, null, 'errors');
}
}
// Unlink payment (reset to NEW status)
if ($action == 'unlink' && $object->id > 0) {
if ($object->status == BankImportTransaction::STATUS_MATCHED) {
$result = $object->unlinkPayment($user);
if ($result > 0) {
setEventMessages($langs->trans("PaymentUnlinked"), null, 'mesgs');
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
} else {
setEventMessages($object->error, null, 'errors');
}
} else {
setEventMessages($langs->trans("CannotUnlinkThisStatus"), null, 'warnings');
}
}
// Find matches
if ($action == 'findmatches' && $object->id > 0) {
$matches = $object->findMatches();
if (count($matches) > 0) {
$_SESSION['bankimport_matches_'.$object->id] = $matches;
setEventMessages($langs->trans("MatchesFound", count($matches)), null, 'mesgs');
} else {
setEventMessages($langs->trans("NoMatchesFound"), null, 'warnings');
}
}
// Link to object (old method - just link without payment)
if ($action == 'linkto' && $object->id > 0) {
$linktype = GETPOST('linktype', 'alpha');
$linkid = GETPOSTINT('linkid');
if ($linktype && $linkid > 0) {
$result = $object->linkTo($linktype, $linkid, $user);
if ($result > 0) {
setEventMessages($langs->trans("LinkCreated"), null, 'mesgs');
unset($_SESSION['bankimport_matches_'.$object->id]);
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
} else {
setEventMessages($object->error, null, 'errors');
}
}
}
// Confirm payment (creates payment in Dolibarr)
$bankAccountId = getDolGlobalInt('BANKIMPORT_BANK_ACCOUNT_ID');
if ($action == 'confirmpayment' && $object->id > 0 && !empty($bankAccountId)) {
$matchtype = GETPOST('matchtype', 'alpha');
$matchid = GETPOSTINT('matchid');
if ($matchtype && $matchid > 0) {
if ($object->status != BankImportTransaction::STATUS_NEW) {
setEventMessages($langs->trans("TransactionAlreadyProcessed"), null, 'warnings');
} else {
$result = $object->confirmPayment($user, $matchtype, $matchid, $bankAccountId);
if ($result > 0) {
setEventMessages($langs->trans("PaymentCreatedSuccessfully", dol_escape_htmltag($object->name), price(abs($object->amount))), null, 'mesgs');
unset($_SESSION['bankimport_matches_'.$object->id]);
} else {
setEventMessages($object->error, $object->errors, 'errors');
}
}
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
}
}
// Confirm multiple invoices payment
if ($action == 'confirmmulti' && $object->id > 0 && !empty($bankAccountId)) {
$invoiceIds = GETPOST('invoices', 'array');
if (!empty($invoiceIds)) {
if ($object->status != BankImportTransaction::STATUS_NEW) {
setEventMessages($langs->trans("TransactionAlreadyProcessed"), null, 'warnings');
} else {
// Build invoices array with amounts
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
$invoices = array();
foreach ($invoiceIds as $invId) {
$invId = (int) $invId;
if ($invId > 0) {
$invoice = new FactureFournisseur($db);
if ($invoice->fetch($invId) > 0) {
$alreadyPaid = $invoice->getSommePaiement();
$creditnotes = $invoice->getSumCreditNotesUsed();
$deposits = $invoice->getSumDepositsUsed();
$remainToPay = price2num($invoice->total_ttc - $alreadyPaid - $creditnotes - $deposits, 'MT');
if ($remainToPay > 0) {
$invoices[] = array(
'type' => 'facture_fourn',
'id' => $invId,
'ref' => $invoice->ref,
'amount' => $remainToPay
);
}
}
}
}
if (!empty($invoices)) {
$result = $object->confirmMultiplePayment($user, $invoices, $bankAccountId);
if ($result > 0) {
setEventMessages($langs->trans("PaymentCreatedSuccessfully", dol_escape_htmltag($object->name), price(abs($object->amount))).' ('.count($invoices).' '.$langs->trans("Invoices").')', null, 'mesgs');
unset($_SESSION['bankimport_matches_'.$object->id]);
} else {
setEventMessages($object->error, $object->errors, 'errors');
}
} else {
setEventMessages($langs->trans("NoInvoicesSelected"), null, 'errors');
}
}
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
}
}
// Link existing payment to bank entry (for already paid invoices)
if ($action == 'linkpayment' && $object->id > 0 && !empty($bankAccountId)) {
$invoiceType = GETPOST('invoicetype', 'alpha');
$invoiceId = GETPOSTINT('invoiceid');
if ($invoiceType && $invoiceId > 0) {
if ($object->status != BankImportTransaction::STATUS_NEW) {
setEventMessages($langs->trans("TransactionAlreadyProcessed"), null, 'warnings');
} else {
$result = $object->linkExistingPayment($user, $invoiceType, $invoiceId, $bankAccountId);
if ($result > 0) {
setEventMessages($langs->trans("PaymentLinkedSuccessfully"), null, 'mesgs');
} else {
setEventMessages($object->error, $object->errors, 'errors');
}
}
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
}
}
// Link multiple existing payments to bank entry (for already paid invoices)
if ($action == 'linkpaymentmulti' && $object->id > 0 && !empty($bankAccountId)) {
$paidInvoiceIds = GETPOST('paid_invoice', 'array');
$paidInvoiceType = GETPOST('paid_invoice_type', 'alpha');
if (!empty($paidInvoiceIds) && !empty($paidInvoiceType)) {
if ($object->status != BankImportTransaction::STATUS_NEW) {
setEventMessages($langs->trans("TransactionAlreadyProcessed"), null, 'warnings');
} else {
// Build invoices array
$invoices = array();
foreach ($paidInvoiceIds as $invId) {
$invoices[] = array(
'type' => $paidInvoiceType,
'id' => (int) $invId
);
}
$result = $object->linkMultipleExistingPayments($user, $invoices, $bankAccountId);
if ($result > 0) {
setEventMessages($langs->trans("PaymentLinkedSuccessfully"), null, 'mesgs');
} else {
setEventMessages($object->error, $object->errors, 'errors');
}
}
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
} else {
setEventMessages($langs->trans("NoInvoicesSelected"), null, 'warnings');
}
}
// Search for invoice manually
if ($action == 'searchinvoice' && $object->id > 0) {
$searchInvoiceRef = GETPOST('search_invoice_ref', 'alpha');
$searchInvoiceType = GETPOST('search_invoice_type', 'alpha'); // 'customer' or 'supplier'
if (!empty($searchInvoiceRef)) {
$manualMatches = array();
if ($searchInvoiceType == 'customer' || empty($searchInvoiceType)) {
// Search customer invoices
require_once DOL_DOCUMENT_ROOT.'/compta/facture/class/facture.class.php';
$sql = "SELECT f.rowid, f.ref, f.ref_client, f.total_ttc, f.date_lim_reglement, s.nom as socname, s.rowid as socid";
$sql .= " FROM ".MAIN_DB_PREFIX."facture as f";
$sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe as s ON f.fk_soc = s.rowid";
$sql .= " WHERE f.entity = ".$conf->entity;
$sql .= " AND f.fk_statut = 1"; // Unpaid
$sql .= " AND (f.ref LIKE '%".$db->escape($searchInvoiceRef)."%' OR f.ref_client LIKE '%".$db->escape($searchInvoiceRef)."%')";
$resql = $db->query($sql);
if ($resql) {
while ($obj = $db->fetch_object($resql)) {
$manualMatches[] = array(
'type' => 'facture',
'id' => $obj->rowid,
'ref' => $obj->ref,
'ref_client' => $obj->ref_client,
'amount' => $obj->total_ttc,
'socname' => $obj->socname,
'socid' => $obj->socid,
'match_score' => 0,
'match_reasons' => array('manual'),
'date_due' => $obj->date_lim_reglement
);
}
}
}
if ($searchInvoiceType == 'supplier' || empty($searchInvoiceType)) {
// Search supplier invoices
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
$sql = "SELECT f.rowid, f.ref, f.ref_supplier, f.total_ttc, f.date_lim_reglement, s.nom as socname, s.rowid as socid";
$sql .= " FROM ".MAIN_DB_PREFIX."facture_fourn as f";
$sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe as s ON f.fk_soc = s.rowid";
$sql .= " WHERE f.entity = ".$conf->entity;
$sql .= " AND f.fk_statut = 1"; // Unpaid
$sql .= " AND (f.ref LIKE '%".$db->escape($searchInvoiceRef)."%' OR f.ref_supplier LIKE '%".$db->escape($searchInvoiceRef)."%')";
$resql = $db->query($sql);
if ($resql) {
while ($obj = $db->fetch_object($resql)) {
$manualMatches[] = array(
'type' => 'facture_fourn',
'id' => $obj->rowid,
'ref' => $obj->ref,
'ref_supplier' => $obj->ref_supplier,
'amount' => $obj->total_ttc,
'socname' => $obj->socname,
'socid' => $obj->socid,
'match_score' => 0,
'match_reasons' => array('manual'),
'date_due' => $obj->date_lim_reglement
);
}
}
}
if (!empty($manualMatches)) {
$_SESSION['bankimport_matches_'.$object->id] = $manualMatches;
setEventMessages($langs->trans("MatchesFound", count($manualMatches)), null, 'mesgs');
} else {
setEventMessages($langs->trans("NoMatchesFound"), null, 'warnings');
}
}
}
// Create various payment (for transactions without invoices)
if ($action == 'confirmcreatevarious' && $object->id > 0 && $object->status == BankImportTransaction::STATUS_NEW) {
$accountancyCode = GETPOST('accountancy_code', 'alpha');
$subledgerAccount = GETPOST('subledger_account', 'alpha');
$sens = GETPOSTINT('sens');
$variousLabel = GETPOST('various_label', 'alphanohtml');
if (empty($accountancyCode)) {
setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentities("AccountingAccount")), null, 'errors');
} else {
$bankAccountId = getDolGlobalInt('BANKIMPORT_BANK_ACCOUNT_ID');
if (empty($bankAccountId)) {
setEventMessages($langs->trans("ErrorNoBankAccountConfigured"), null, 'errors');
} else {
$result = $object->createVariousPayment($user, $bankAccountId, $accountancyCode, $sens, $variousLabel, $subledgerAccount);
if ($result > 0) {
setEventMessages($langs->trans("VariousPaymentCreated"), null, 'mesgs');
header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
exit;
} else {
setEventMessages($object->error, null, 'errors');
}
}
}
}
/*
* View
*/
$form = new Form($db);
$title = $langs->trans("Transaction").' - '.$object->ref;
llxHeader('', $title, '', '', 0, 0, '', '', '', 'mod-bankimport page-card');
if ($object->id > 0) {
print '<div class="fichecenter">';
// Title
print load_fiche_titre($langs->trans("Transaction"), '', 'bank');
// Card header
print '<div class="underbanner clearboth"></div>';
print '<table class="border centpercent tableforfield">';
// Reference
print '<tr>';
print '<td class="titlefield">'.$langs->trans("Ref").'</td>';
print '<td>'.dol_escape_htmltag($object->ref).'</td>';
print '</tr>';
// IBAN
print '<tr>';
print '<td>'.$langs->trans("AccountIBAN").'</td>';
print '<td>'.dol_escape_htmltag($object->iban).'</td>';
print '</tr>';
// Date
print '<tr>';
print '<td>'.$langs->trans("Date").'</td>';
print '<td>'.dol_print_date($object->date_trans, 'day').'</td>';
print '</tr>';
// Value date
if ($object->date_value) {
print '<tr>';
print '<td>'.$langs->trans("DateValue").'</td>';
print '<td>'.dol_print_date($object->date_value, 'day').'</td>';
print '</tr>';
}
// Name
print '<tr>';
print '<td>'.$langs->trans("Counterparty").'</td>';
print '<td><strong>'.dol_escape_htmltag($object->name).'</strong></td>';
print '</tr>';
// Counterparty IBAN
if ($object->counterparty_iban) {
print '<tr>';
print '<td>'.$langs->trans("CounterpartyIBAN").'</td>';
print '<td>'.dol_escape_htmltag($object->counterparty_iban).'</td>';
print '</tr>';
}
// Amount
print '<tr>';
print '<td>'.$langs->trans("Amount").'</td>';
print '<td>';
if ($object->amount >= 0) {
print '<span style="color: green; font-size: 1.3em; font-weight: bold;">+'.price($object->amount, 0, $langs, 1, -1, 2, $object->currency).'</span>';
} else {
print '<span style="color: red; font-size: 1.3em; font-weight: bold;">'.price($object->amount, 0, $langs, 1, -1, 2, $object->currency).'</span>';
}
print '</td>';
print '</tr>';
// Label
if ($object->label) {
print '<tr>';
print '<td>'.$langs->trans("Label").'</td>';
print '<td>'.dol_escape_htmltag($object->label).'</td>';
print '</tr>';
}
// Description
print '<tr>';
print '<td>'.$langs->trans("Description").'</td>';
print '<td>'.nl2br(dol_escape_htmltag($object->description)).'</td>';
print '</tr>';
// End-to-End ID
if ($object->end_to_end_id) {
print '<tr>';
print '<td>'.$langs->trans("EndToEndId").'</td>';
print '<td>'.dol_escape_htmltag($object->end_to_end_id).'</td>';
print '</tr>';
}
// Mandate ID
if ($object->mandate_id) {
print '<tr>';
print '<td>'.$langs->trans("MandateId").'</td>';
print '<td>'.dol_escape_htmltag($object->mandate_id).'</td>';
print '</tr>';
}
// Status
print '<tr>';
print '<td>'.$langs->trans("Status").'</td>';
print '<td>'.$object->getLibStatut(4).'</td>';
print '</tr>';
// Linked payment and invoices (for matched transactions)
if ($object->status == BankImportTransaction::STATUS_MATCHED || $object->status == BankImportTransaction::STATUS_RECONCILED) {
// Customer payment
if ($object->fk_paiement > 0) {
require_once DOL_DOCUMENT_ROOT.'/compta/paiement/class/paiement.class.php';
require_once DOL_DOCUMENT_ROOT.'/compta/facture/class/facture.class.php';
$payment = new Paiement($db);
$payment->fetch($object->fk_paiement);
print '<tr>';
print '<td>'.$langs->trans("Payment").'</td>';
print '<td>'.$payment->getNomUrl(1).' <span class="opacitymedium">('.dol_print_date($payment->datepaye, 'day').')</span></td>';
print '</tr>';
// Find all invoices linked to this payment
$sql = "SELECT pf.fk_facture, pf.amount, f.ref, f.total_ttc";
$sql .= " FROM ".MAIN_DB_PREFIX."paiement_facture as pf";
$sql .= " LEFT JOIN ".MAIN_DB_PREFIX."facture as f ON pf.fk_facture = f.rowid";
$sql .= " WHERE pf.fk_paiement = ".((int) $object->fk_paiement);
$resql = $db->query($sql);
if ($resql && $db->num_rows($resql) > 0) {
print '<tr>';
print '<td>'.$langs->trans("Invoices").'</td>';
print '<td>';
while ($obj = $db->fetch_object($resql)) {
$inv = new Facture($db);
$inv->fetch($obj->fk_facture);
print $inv->getNomUrl(1);
print ' <span class="opacitymedium">('.price($obj->amount, 0, $langs, 1, -1, 2, 'EUR').')</span>';
print '<br>';
}
print '</td>';
print '</tr>';
}
}
// Supplier payment
if ($object->fk_paiementfourn > 0) {
require_once DOL_DOCUMENT_ROOT.'/fourn/class/paiementfourn.class.php';
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
$payment = new PaiementFourn($db);
$payment->fetch($object->fk_paiementfourn);
print '<tr>';
print '<td>'.$langs->trans("Payment").'</td>';
print '<td>'.$payment->getNomUrl(1).' <span class="opacitymedium">('.dol_print_date($payment->datepaye, 'day').')</span></td>';
print '</tr>';
// Find all invoices linked to this payment
$sql = "SELECT pf.fk_facturefourn, pf.amount, f.ref, f.ref_supplier, f.total_ttc";
$sql .= " FROM ".MAIN_DB_PREFIX."paiementfourn_facturefourn as pf";
$sql .= " LEFT JOIN ".MAIN_DB_PREFIX."facture_fourn as f ON pf.fk_facturefourn = f.rowid";
$sql .= " WHERE pf.fk_paiementfourn = ".((int) $object->fk_paiementfourn);
$resql = $db->query($sql);
$invoicesFound = false;
if ($resql && $db->num_rows($resql) > 0) {
$invoicesFound = true;
print '<tr>';
print '<td>'.$langs->trans("SupplierInvoices").'</td>';
print '<td>';
while ($obj = $db->fetch_object($resql)) {
$inv = new FactureFournisseur($db);
$inv->fetch($obj->fk_facturefourn);
print $inv->getNomUrl(1);
if (!empty($obj->ref_supplier)) {
print ' <span class="opacitymedium small">('.$obj->ref_supplier.')</span>';
}
print ' <span class="opacitymedium">('.price($obj->amount, 0, $langs, 1, -1, 2, 'EUR').')</span>';
print '<br>';
}
print '</td>';
print '</tr>';
}
// Also check note_private for multi-invoice links (format: "Multi-invoice link: SI2602-0146, SI2602-0147")
if (!$invoicesFound && !empty($object->note_private) && strpos($object->note_private, 'Multi-invoice link:') !== false) {
if (preg_match('/Multi-invoice link:\s*(.+)$/m', $object->note_private, $matches)) {
$invoiceRefs = array_map('trim', explode(',', $matches[1]));
print '<tr>';
print '<td>'.$langs->trans("SupplierInvoices").'</td>';
print '<td>';
foreach ($invoiceRefs as $invRef) {
// Try to find invoice by ref
$sql2 = "SELECT rowid FROM ".MAIN_DB_PREFIX."facture_fourn WHERE ref = '".$db->escape($invRef)."' AND entity = ".$conf->entity;
$resql2 = $db->query($sql2);
if ($resql2 && $db->num_rows($resql2) > 0) {
$obj2 = $db->fetch_object($resql2);
$inv = new FactureFournisseur($db);
$inv->fetch($obj2->rowid);
print $inv->getNomUrl(1);
if (!empty($inv->ref_supplier)) {
print ' <span class="opacitymedium small">('.$inv->ref_supplier.')</span>';
}
} else {
print dol_escape_htmltag($invRef);
}
print '<br>';
}
print '</td>';
print '</tr>';
$invoicesFound = true;
}
}
// Fallback: Show linked invoice from transaction
if (!$invoicesFound && $object->fk_facture_fourn > 0) {
$inv = new FactureFournisseur($db);
$inv->fetch($object->fk_facture_fourn);
print '<tr>';
print '<td>'.$langs->trans("SupplierInvoice").'</td>';
print '<td>'.$inv->getNomUrl(1);
if (!empty($inv->ref_supplier)) {
print ' <span class="opacitymedium small">('.$inv->ref_supplier.')</span>';
}
print '</td>';
print '</tr>';
}
}
// Various payment (no invoice, linked via bank_url)
if (empty($object->fk_paiement) && empty($object->fk_paiementfourn) && $object->fk_bank > 0) {
$sql = "SELECT url_id FROM ".MAIN_DB_PREFIX."bank_url WHERE fk_bank = ".((int) $object->fk_bank)." AND type = 'payment_various'";
$resql = $db->query($sql);
if ($resql && $db->num_rows($resql) > 0) {
$obj = $db->fetch_object($resql);
require_once DOL_DOCUMENT_ROOT.'/compta/bank/class/paymentvarious.class.php';
$various = new PaymentVarious($db);
$various->fetch($obj->url_id);
print '<tr>';
print '<td>'.$langs->trans("Payment").'</td>';
print '<td>';
print '<a href="'.DOL_URL_ROOT.'/compta/bank/various_payment/card.php?id='.$various->id.'">';
print img_picto('', 'payment', 'class="pictofixedwidth"');
print $langs->trans("VariousPayment").' #'.$various->id;
print '</a>';
print ' <span class="opacitymedium">('.dol_print_date($various->datep, 'day').' - '.price($various->amount, 0, $langs, 1, -1, 2, 'EUR').')</span>';
print '</td>';
print '</tr>';
if (!empty($various->accountancy_code)) {
print '<tr>';
print '<td>'.$langs->trans("AccountingAccount").'</td>';
print '<td>'.dol_escape_htmltag($various->accountancy_code).'</td>';
print '</tr>';
}
}
}
// If no payment link but invoice link exists (edge case)
if (empty($object->fk_paiement) && empty($object->fk_paiementfourn)) {
if ($object->fk_facture_fourn > 0) {
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
$inv = new FactureFournisseur($db);
$inv->fetch($object->fk_facture_fourn);
print '<tr>';
print '<td>'.$langs->trans("SupplierInvoice").'</td>';
print '<td>'.$inv->getNomUrl(1);
if (!empty($inv->ref_supplier)) {
print ' <span class="opacitymedium small">('.$inv->ref_supplier.')</span>';
}
print '</td>';
print '</tr>';
}
if ($object->fk_facture > 0) {
require_once DOL_DOCUMENT_ROOT.'/compta/facture/class/facture.class.php';
$inv = new Facture($db);
$inv->fetch($object->fk_facture);
print '<tr>';
print '<td>'.$langs->trans("Invoice").'</td>';
print '<td>'.$inv->getNomUrl(1).'</td>';
print '</tr>';
}
}
// Bank entry
if ($object->fk_bank > 0) {
require_once DOL_DOCUMENT_ROOT.'/compta/bank/class/account.class.php';
$bankline = new AccountLine($db);
$bankline->fetch($object->fk_bank);
print '<tr>';
print '<td>'.$langs->trans("BankEntry").'</td>';
print '<td>';
print '<a href="'.DOL_URL_ROOT.'/compta/bank/line.php?rowid='.$object->fk_bank.'">';
print img_picto('', 'bank_account', 'class="pictofixedwidth"');
print $bankline->ref ?: '#'.$object->fk_bank;
print '</a>';
print ' <span class="opacitymedium">('.dol_print_date($bankline->dateo, 'day').' - '.price($bankline->amount, 0, $langs, 1, -1, 2, 'EUR').')</span>';
print '</td>';
print '</tr>';
}
} else {
// For non-matched transactions, show simple links if they exist
// Linked invoice
if ($object->fk_facture > 0) {
require_once DOL_DOCUMENT_ROOT.'/compta/facture/class/facture.class.php';
$invoice = new Facture($db);
$invoice->fetch($object->fk_facture);
print '<tr>';
print '<td>'.$langs->trans("Invoice").'</td>';
print '<td>'.$invoice->getNomUrl(1).'</td>';
print '</tr>';
}
// Linked supplier invoice
if ($object->fk_facture_fourn > 0) {
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
$invoice = new FactureFournisseur($db);
$invoice->fetch($object->fk_facture_fourn);
print '<tr>';
print '<td>'.$langs->trans("SupplierInvoice").'</td>';
print '<td>'.$invoice->getNomUrl(1).'</td>';
print '</tr>';
}
}
// Linked PDF statement
if (!empty($object->fk_statement)) {
dol_include_once('/bankimport/class/bankstatement.class.php');
$stmt = new BankImportStatement($db);
$stmt->fetch($object->fk_statement);
print '<tr>';
print '<td>'.$langs->trans("PDFStatement").'</td>';
print '<td>';
print '<a href="'.dol_buildpath('/bankimport/pdfstatements.php', 1).'?action=view&id='.$stmt->id.'&token='.newToken().'" target="_blank">';
print img_picto($langs->trans("ViewPDFStatement"), 'pdf').' ';
print $langs->trans("StatementNumber").' '.$stmt->statement_number.'/'.$stmt->statement_year;
print '</a>';
if ($stmt->date_from && $stmt->date_to) {
print ' <span class="opacitymedium">('.dol_print_date($stmt->date_from, 'day').' - '.dol_print_date($stmt->date_to, 'day').')</span>';
}
print '</td>';
print '</tr>';
}
// Linked third party
if ($object->fk_societe > 0) {
require_once DOL_DOCUMENT_ROOT.'/societe/class/societe.class.php';
$soc = new Societe($db);
$soc->fetch($object->fk_societe);
print '<tr>';
print '<td>'.$langs->trans("ThirdParty").'</td>';
print '<td>'.$soc->getNomUrl(1).'</td>';
print '</tr>';
}
// Import date
print '<tr>';
print '<td>'.$langs->trans("DateCreation").'</td>';
print '<td>'.dol_print_date($object->datec, 'dayhour').'</td>';
print '</tr>';
print '</table>';
print '</div>';
// Actions buttons
print '<div class="tabsAction">';
if ($object->status == BankImportTransaction::STATUS_NEW) {
// Find matches button
print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=findmatches&token='.newToken().'">'.$langs->trans("FindMatches").'</a>';
// Create various payment button
print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=createvarious&token='.newToken().'">'.$langs->trans("CreateVariousPayment").'</a>';
// Set as ignored
print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=setstatus&status='.BankImportTransaction::STATUS_IGNORED.'&token='.newToken().'">'.$langs->trans("SetAsIgnored").'</a>';
}
if ($object->status == BankImportTransaction::STATUS_MATCHED) {
// Edit/Unlink - allows correcting wrong matches
print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=unlink&token='.newToken().'">'.$langs->trans("UnlinkPayment").'</a>';
}
if ($object->status == BankImportTransaction::STATUS_IGNORED) {
// Reopen
print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=setstatus&status='.BankImportTransaction::STATUS_NEW.'&token='.newToken().'">'.$langs->trans("Reopen").'</a>';
}
print '</div>';
// Various payment inline form
if ($action == 'createvarious' && $object->status == BankImportTransaction::STATUS_NEW) {
// Auto-detect sens from amount sign: positive = credit (1), negative = debit (0)
$defaultSens = ($object->amount >= 0) ? 1 : 0;
$defaultLabel = $object->name.' - '.dol_trunc($object->description, 80);
require_once DOL_DOCUMENT_ROOT.'/core/class/html.formaccounting.class.php';
$formaccounting = new FormAccounting($db);
print '<br>';
print load_fiche_titre($langs->trans("CreateVariousPayment"), '', 'object_payment');
print '<form method="POST" action="'.$_SERVER["PHP_SELF"].'">';
print '<input type="hidden" name="token" value="'.newToken().'">';
print '<input type="hidden" name="id" value="'.$object->id.'">';
print '<input type="hidden" name="action" value="confirmcreatevarious">';
print '<table class="border centpercent">';
// Amount (read-only)
print '<tr>';
print '<td class="titlefield fieldrequired">'.$langs->trans("Amount").'</td>';
print '<td>';
$amountColor = ($object->amount >= 0) ? 'green' : 'red';
print '<span style="font-weight: bold; color: '.$amountColor.';">'.price($object->amount, 0, $langs, 1, -1, 2, 'EUR').'</span>';
print '</td>';
print '</tr>';
// Date (read-only)
print '<tr>';
print '<td>'.$langs->trans("Date").'</td>';
print '<td>'.dol_print_date($object->date_trans, 'day').'</td>';
print '</tr>';
// Counterparty (read-only)
print '<tr>';
print '<td>'.$langs->trans("Counterparty").'</td>';
print '<td>'.dol_escape_htmltag($object->name).'</td>';
print '</tr>';
// Accounting account (Select2 search)
print '<tr>';
print '<td class="fieldrequired">'.$langs->trans("AccountingAccount").'</td>';
print '<td>';
print $formaccounting->select_account(GETPOST('accountancy_code', 'alpha'), 'accountancy_code', 1, array(), 0, 0, 'minwidth200 maxwidth500', '', 1);
print '</td>';
print '</tr>';
// Subledger account (Select2 search)
print '<tr>';
print '<td>'.$langs->trans("SubledgerAccount").'</td>';
print '<td>';
print $formaccounting->select_auxaccount(GETPOST('subledger_account', 'alpha'), 'subledger_account', 1, 'minwidth200 maxwidth500');
print '</td>';
print '</tr>';
// Debit/Credit (auto-selected)
print '<tr>';
print '<td class="fieldrequired">'.$langs->trans("DebitCredit").'</td>';
print '<td>';
$sensValue = GETPOSTISSET('sens') ? GETPOSTINT('sens') : $defaultSens;
print '<select name="sens" class="flat minwidth100">';
print '<option value="0"'.($sensValue == 0 ? ' selected' : '').'>'.$langs->trans("Debit").' ('.$langs->trans("Expense").')</option>';
print '<option value="1"'.($sensValue == 1 ? ' selected' : '').'>'.$langs->trans("Credit").' ('.$langs->trans("Income").')</option>';
print '</select>';
print '</td>';
print '</tr>';
// Label (editable, pre-filled)
print '<tr>';
print '<td class="fieldrequired">'.$langs->trans("Label").'</td>';
print '<td>';
$labelValue = GETPOSTISSET('various_label') ? GETPOST('various_label', 'alphanohtml') : $defaultLabel;
print '<input type="text" name="various_label" class="flat minwidth300 maxwidth500" value="'.dol_escape_htmltag($labelValue).'">';
print '</td>';
print '</tr>';
print '</table>';
print '<div class="center" style="margin-top: 10px;">';
print '<input type="submit" class="button button-save" value="'.$langs->trans("Save").'">';
print ' &nbsp; ';
print '<a class="button button-cancel" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'">'.$langs->trans("Cancel").'</a>';
print '</div>';
print '</form>';
}
// Manual invoice selection (only for new transactions)
if ($object->status == BankImportTransaction::STATUS_NEW) {
// Determine if this is likely a supplier payment (negative amount) or customer (positive)
$defaultType = ($object->amount < 0) ? 'supplier' : 'customer';
$selectedType = GETPOST('invoice_type', 'alpha') ?: $defaultType;
$searchFilter = GETPOST('search_filter', 'alpha');
$showPaid = GETPOSTINT('show_paid');
print '<br>';
print load_fiche_titre($langs->trans("SelectInvoicesManually"), '', 'object_invoice');
// Type selector and search filter
print '<form method="GET" action="'.$_SERVER["PHP_SELF"].'" style="margin-bottom: 15px;">';
print '<input type="hidden" name="id" value="'.$object->id.'">';
print '<div class="inline-block" style="margin-right: 15px;">';
print '<label>'.$langs->trans("Type").': </label>';
print '<select name="invoice_type" class="flat" onchange="this.form.submit()">';
print '<option value="supplier"'.($selectedType == 'supplier' ? ' selected' : '').'>'.$langs->trans("SupplierInvoices").'</option>';
print '<option value="customer"'.($selectedType == 'customer' ? ' selected' : '').'>'.$langs->trans("CustomerInvoices").'</option>';
print '</select>';
print '</div>';
print '<div class="inline-block" style="margin-right: 15px;">';
print '<label><input type="checkbox" name="show_paid" value="1"'.($showPaid ? ' checked' : '').' onchange="this.form.submit()"> '.$langs->trans("ShowPaidInvoices").'</label>';
print '</div>';
print '<div class="inline-block">';
print '<label>'.$langs->trans("Search").': </label>';
print '<input type="text" name="search_filter" class="flat minwidth200" value="'.dol_escape_htmltag($searchFilter).'" placeholder="'.$langs->trans("SearchInvoiceByRef").'">';
print ' <input type="submit" class="button small" value="'.$langs->trans("Filter").'">';
if (!empty($searchFilter)) {
print ' <a href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&invoice_type='.$selectedType.'&show_paid='.$showPaid.'" class="button small">'.$langs->trans("RemoveFilter").'</a>';
}
print '</div>';
print '</form>';
// Fetch invoices
$invoiceList = array();
$absAmount = abs($object->amount);
if ($selectedType == 'supplier') {
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
$sql = "SELECT f.rowid, f.ref, f.ref_supplier, f.total_ttc, f.datef, f.date_lim_reglement, f.fk_statut,";
$sql .= " s.nom as socname, s.rowid as socid";
$sql .= " FROM ".MAIN_DB_PREFIX."facture_fourn as f";
$sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe as s ON f.fk_soc = s.rowid";
$sql .= " WHERE f.entity = ".$conf->entity;
if ($showPaid) {
$sql .= " AND f.fk_statut IN (1, 2)"; // 1=Unpaid, 2=Paid
} else {
$sql .= " AND f.fk_statut = 1"; // Unpaid only
}
if (!empty($searchFilter)) {
$sql .= " AND (f.ref LIKE '%".$db->escape($searchFilter)."%'";
$sql .= " OR f.ref_supplier LIKE '%".$db->escape($searchFilter)."%'";
$sql .= " OR s.nom LIKE '%".$db->escape($searchFilter)."%')";
}
$sql .= " ORDER BY f.datef DESC";
$sql .= " LIMIT 100";
$resql = $db->query($sql);
if ($resql) {
while ($obj = $db->fetch_object($resql)) {
$inv = new FactureFournisseur($db);
$inv->fetch($obj->rowid);
$alreadyPaid = $inv->getSommePaiement();
$creditnotes = $inv->getSumCreditNotesUsed();
$deposits = $inv->getSumDepositsUsed();
$remainToPay = price2num($inv->total_ttc - $alreadyPaid - $creditnotes - $deposits, 'MT');
$isPaid = ($obj->fk_statut == 2);
// For paid invoices, check if already linked via BankImport transaction
$isLinkedViaBankImport = false;
if ($isPaid) {
$sqlBi = "SELECT rowid FROM ".MAIN_DB_PREFIX."bankimport_transaction";
$sqlBi .= " WHERE fk_facture_fourn = ".((int) $obj->rowid);
$sqlBi .= " AND status > 0";
$resqlBi = $db->query($sqlBi);
if ($resqlBi && $db->num_rows($resqlBi) > 0) {
$isLinkedViaBankImport = true;
}
}
// Show if unpaid OR if paid but not yet linked via BankImport
if ($remainToPay > 0 || ($isPaid && !$isLinkedViaBankImport)) {
$invoiceList[] = array(
'id' => $obj->rowid,
'ref' => $obj->ref,
'ref_supplier' => $obj->ref_supplier,
'amount' => $isPaid ? $obj->total_ttc : $remainToPay,
'total_ttc' => $obj->total_ttc,
'socname' => $obj->socname,
'socid' => $obj->socid,
'datef' => $obj->datef,
'date_due' => $obj->date_lim_reglement,
'type' => 'facture_fourn',
'object' => $inv,
'is_paid' => $isPaid
);
}
}
}
} else {
require_once DOL_DOCUMENT_ROOT.'/compta/facture/class/facture.class.php';
$sql = "SELECT f.rowid, f.ref, f.ref_client, f.total_ttc, f.datef, f.date_lim_reglement, f.fk_statut,";
$sql .= " s.nom as socname, s.rowid as socid";
$sql .= " FROM ".MAIN_DB_PREFIX."facture as f";
$sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe as s ON f.fk_soc = s.rowid";
$sql .= " WHERE f.entity = ".$conf->entity;
if ($showPaid) {
$sql .= " AND f.fk_statut IN (1, 2)"; // 1=Unpaid, 2=Paid
} else {
$sql .= " AND f.fk_statut = 1"; // Unpaid only
}
if (!empty($searchFilter)) {
$sql .= " AND (f.ref LIKE '%".$db->escape($searchFilter)."%'";
$sql .= " OR f.ref_client LIKE '%".$db->escape($searchFilter)."%'";
$sql .= " OR s.nom LIKE '%".$db->escape($searchFilter)."%')";
}
$sql .= " ORDER BY f.datef DESC";
$sql .= " LIMIT 100";
$resql = $db->query($sql);
if ($resql) {
while ($obj = $db->fetch_object($resql)) {
$inv = new Facture($db);
$inv->fetch($obj->rowid);
$alreadyPaid = $inv->getSommePaiement();
$creditnotes = $inv->getSumCreditNotesUsed();
$deposits = $inv->getSumDepositsUsed();
$remainToPay = price2num($inv->total_ttc - $alreadyPaid - $creditnotes - $deposits, 'MT');
$isPaid = ($obj->fk_statut == 2);
// For paid invoices, check if already linked via BankImport transaction
$isLinkedViaBankImport = false;
if ($isPaid) {
$sqlBi = "SELECT rowid FROM ".MAIN_DB_PREFIX."bankimport_transaction";
$sqlBi .= " WHERE fk_facture = ".((int) $obj->rowid);
$sqlBi .= " AND status > 0";
$resqlBi = $db->query($sqlBi);
if ($resqlBi && $db->num_rows($resqlBi) > 0) {
$isLinkedViaBankImport = true;
}
}
// Show if unpaid OR if paid but not yet linked via BankImport
if ($remainToPay > 0 || ($isPaid && !$isLinkedViaBankImport)) {
$invoiceList[] = array(
'id' => $obj->rowid,
'ref' => $obj->ref,
'ref_client' => $obj->ref_client,
'amount' => $isPaid ? $obj->total_ttc : $remainToPay,
'total_ttc' => $obj->total_ttc,
'socname' => $obj->socname,
'socid' => $obj->socid,
'datef' => $obj->datef,
'date_due' => $obj->date_lim_reglement,
'type' => 'facture',
'object' => $inv,
'is_paid' => $isPaid
);
}
}
}
}
// Separate paid and unpaid invoices
$unpaidInvoices = array_filter($invoiceList, function($inv) { return empty($inv['is_paid']); });
$paidInvoices = array_filter($invoiceList, function($inv) { return !empty($inv['is_paid']); });
// Display unpaid invoices table with checkboxes
if (!empty($unpaidInvoices)) {
print '<form method="POST" action="'.$_SERVER["PHP_SELF"].'">';
print '<input type="hidden" name="token" value="'.newToken().'">';
print '<input type="hidden" name="action" value="confirmmulti">';
print '<input type="hidden" name="id" value="'.$object->id.'">';
// Table without div-table-responsive for compact layout
print '<table class="noborder centpercent" id="invoice_table" style="margin-bottom: 0;">';
// Info header row
print '<tr class="liste_titre">';
print '<td colspan="7" style="background: #f8f8f8; padding: 8px;">';
print $langs->trans("TransactionAmount").': <strong>'.price(abs($object->amount), 0, $langs, 1, -1, 2, 'EUR').'</strong>';
print ' &nbsp;|&nbsp; <span id="selected_sum">'.$langs->trans("Selected").': 0,00 €</span>';
print ' &nbsp;|&nbsp; <span id="difference">'.$langs->trans("Difference").': '.price(abs($object->amount), 0, $langs, 1, -1, 2, 'EUR').'</span>';
print '</td>';
print '</tr>';
print '<tr class="liste_titre">';
print '<th class="center" style="width: 30px;"><input type="checkbox" id="checkall" onclick="toggleAll(this)"></th>';
print '<th>'.$langs->trans("Ref").'</th>';
print '<th>'.($selectedType == 'supplier' ? $langs->trans("SupplierRef") : $langs->trans("CustomerRef")).'</th>';
print '<th>'.$langs->trans("ThirdParty").'</th>';
print '<th>'.$langs->trans("Date").'</th>';
print '<th>'.$langs->trans("DateDue").'</th>';
print '<th class="right">'.$langs->trans("AmountRemaining").'</th>';
print '</tr>';
foreach ($unpaidInvoices as $inv) {
$amountMatch = (abs($inv['amount'] - $absAmount) < 1.00);
$rowClass = $amountMatch ? 'oddeven highlight' : 'oddeven';
print '<tr class="'.$rowClass.'" data-amount="'.$inv['amount'].'">';
print '<td class="center">';
print '<input type="checkbox" name="invoices[]" value="'.$inv['id'].'" class="invoice_checkbox" onchange="updateSum()">';
print '</td>';
print '<td>'.$inv['object']->getNomUrl(1).'</td>';
print '<td>'.dol_escape_htmltag($inv['ref_supplier'] ?? $inv['ref_client'] ?? '').'</td>';
print '<td>'.dol_escape_htmltag($inv['socname']).'</td>';
print '<td>'.dol_print_date($inv['datef'], 'day').'</td>';
print '<td>'.($inv['date_due'] ? dol_print_date($inv['date_due'], 'day') : '-').'</td>';
print '<td class="right">'.price($inv['amount'], 0, $langs, 1, -1, 2, 'EUR').'</td>';
print '</tr>';
}
// Footer row with button
print '<tr class="liste_total">';
print '<td colspan="6"></td>';
print '<td class="right" style="padding: 8px;">';
if (!empty($bankAccountId)) {
print '<button type="submit" class="butAction" style="margin: 0; padding: 5px 12px;">'.$langs->trans("ConfirmPayment").'</button>';
} else {
print '<span class="butActionRefused" title="'.$langs->trans("NoBankAccountConfigured").'">'.$langs->trans("ConfirmPayment").'</span>';
}
print '</td>';
print '</tr>';
print '</table>';
print '</form>';
}
// Display paid invoices table (link to existing payment)
if (!empty($paidInvoices)) {
print '<div style="margin-top: 15px;"></div>';
print load_fiche_titre($langs->trans("PaidInvoices"), '', 'object_invoice');
print '<div class="opacitymedium" style="margin-bottom: 5px;">'.$langs->trans("PaidInvoicesInfo").'</div>';
// Determine invoice type for all paid invoices (should be same type)
$paidInvoiceType = $paidInvoices[0]['type'] ?? 'facture';
print '<form method="POST" action="'.$_SERVER["PHP_SELF"].'" id="paid_invoices_form">';
print '<input type="hidden" name="token" value="'.newToken().'">';
print '<input type="hidden" name="action" value="linkpaymentmulti">';
print '<input type="hidden" name="id" value="'.$object->id.'">';
print '<input type="hidden" name="paid_invoice_type" value="'.$paidInvoiceType.'">';
// Table without div-table-responsive wrapper for compact layout
print '<table class="noborder centpercent" id="paid_invoices_table" style="margin-bottom: 0;">';
print '<tr class="liste_titre">';
print '<th class="center" style="width: 30px;"><input type="checkbox" id="paid_select_all" onclick="toggleAllPaid(this)"></th>';
print '<th>'.$langs->trans("Ref").'</th>';
print '<th>'.($selectedType == 'supplier' ? $langs->trans("SupplierRef") : $langs->trans("CustomerRef")).'</th>';
print '<th>'.$langs->trans("ThirdParty").'</th>';
print '<th>'.$langs->trans("Date").'</th>';
print '<th class="right">'.$langs->trans("Amount").'</th>';
print '<th>'.$langs->trans("Status").'</th>';
print '<th class="center">'.$langs->trans("Action").'</th>';
print '</tr>';
foreach ($paidInvoices as $inv) {
$amountMatch = (abs($inv['amount'] - $absAmount) < 1.00);
$rowClass = $amountMatch ? 'oddeven highlight' : 'oddeven';
print '<tr class="'.$rowClass.'" data-amount="'.$inv['amount'].'">';
print '<td class="center">';
print '<input type="checkbox" name="paid_invoice[]" value="'.$inv['id'].'" class="paid_invoice_checkbox" onchange="updatePaidSum()">';
print '</td>';
print '<td>'.$inv['object']->getNomUrl(1).'</td>';
print '<td>'.dol_escape_htmltag($inv['ref_supplier'] ?? $inv['ref_client'] ?? '').'</td>';
print '<td>'.dol_escape_htmltag($inv['socname']).'</td>';
print '<td>'.dol_print_date($inv['datef'], 'day').'</td>';
print '<td class="right">'.price($inv['amount'], 0, $langs, 1, -1, 2, 'EUR').'</td>';
print '<td><span class="badge badge-status4">'.$langs->trans("Paid").'</span></td>';
print '<td class="center nowraponall">';
if (!empty($bankAccountId)) {
print '<a class="butActionSmall" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=linkpayment&invoicetype='.$inv['type'].'&invoiceid='.$inv['id'].'&token='.newToken().'">'.$langs->trans("LinkExistingPayment").'</a>';
}
print '</td>';
print '</tr>';
}
// Footer row with sum and button - always inside table
print '<tr class="liste_total">';
print '<td colspan="5" style="padding: 8px;">';
print '<span id="paid_selected_sum" style="font-weight: bold;">'.$langs->trans("Selected").': 0,00 €</span>';
print ' &nbsp;|&nbsp; ';
print '<span id="paid_difference">'.$langs->trans("Difference").': '.price(abs($object->amount), 0, $langs, 1, -1, 2, 'EUR').'</span>';
print '</td>';
print '<td colspan="3" class="right" style="padding: 8px;">';
if (!empty($bankAccountId)) {
print '<button type="submit" class="butAction" id="linkpayment_multi_btn" disabled style="margin: 0; padding: 5px 12px;">'.$langs->trans("LinkExistingPayment").'</button>';
} else {
print '<span class="butActionRefused" title="'.$langs->trans("NoBankAccountConfigured").'">'.$langs->trans("LinkExistingPayment").'</span>';
}
print '</td>';
print '</tr>';
print '</table>';
print '</form>';
}
if (empty($unpaidInvoices) && empty($paidInvoices)) {
print '<div class="opacitymedium">'.$langs->trans("NoUnpaidInvoices").'</div>';
}
// JavaScript for sum calculation (only if we have unpaid invoices)
if (!empty($unpaidInvoices)) {
print '<script>
var transactionAmount = '.abs($object->amount).';
function toggleAll(source) {
var checkboxes = document.getElementsByClassName("invoice_checkbox");
for (var i = 0; i < checkboxes.length; i++) {
checkboxes[i].checked = source.checked;
}
updateSum();
}
function updateSum() {
var checkboxes = document.getElementsByClassName("invoice_checkbox");
var sum = 0;
for (var i = 0; i < checkboxes.length; i++) {
if (checkboxes[i].checked) {
var row = checkboxes[i].closest("tr");
sum += parseFloat(row.getAttribute("data-amount"));
}
}
var diff = transactionAmount - sum;
document.getElementById("selected_sum").innerHTML = "'.$langs->trans("Selected").': " + sum.toFixed(2).replace(".", ",") + " €";
var diffEl = document.getElementById("difference");
if (Math.abs(diff) < 0.01) {
diffEl.innerHTML = "'.$langs->trans("Difference").': <span style=\"color: green; font-weight: bold;\">0,00 € ✓</span>";
} else if (diff > 0) {
diffEl.innerHTML = "'.$langs->trans("Difference").': <span style=\"color: orange;\">+" + diff.toFixed(2).replace(".", ",") + " €</span>";
} else {
diffEl.innerHTML = "'.$langs->trans("Difference").': <span style=\"color: red;\">" + diff.toFixed(2).replace(".", ",") + " €</span>";
}
}
</script>';
}
// JavaScript for paid invoices multi-select (only if we have paid invoices)
if (!empty($paidInvoices)) {
print '<script>
var transactionAmountPaid = '.abs($object->amount).';
function toggleAllPaid(source) {
var checkboxes = document.getElementsByClassName("paid_invoice_checkbox");
for (var i = 0; i < checkboxes.length; i++) {
checkboxes[i].checked = source.checked;
}
updatePaidSum();
}
function updatePaidSum() {
var checkboxes = document.getElementsByClassName("paid_invoice_checkbox");
var sum = 0;
var count = 0;
for (var i = 0; i < checkboxes.length; i++) {
if (checkboxes[i].checked) {
var row = checkboxes[i].closest("tr");
sum += parseFloat(row.getAttribute("data-amount"));
count++;
}
}
var diff = transactionAmountPaid - sum;
document.getElementById("paid_selected_sum").innerHTML = "'.$langs->trans("Selected").': " + sum.toFixed(2).replace(".", ",") + " €";
var diffEl = document.getElementById("paid_difference");
if (Math.abs(diff) < 0.01) {
diffEl.innerHTML = "'.$langs->trans("Difference").': <span style=\"color: green; font-weight: bold;\">0,00 € ✓</span>";
} else if (diff > 0) {
diffEl.innerHTML = "'.$langs->trans("Difference").': <span style=\"color: orange;\">+" + diff.toFixed(2).replace(".", ",") + " €</span>";
} else {
diffEl.innerHTML = "'.$langs->trans("Difference").': <span style=\"color: red;\">" + diff.toFixed(2).replace(".", ",") + " €</span>";
}
// Enable/disable submit button
var btn = document.getElementById("linkpayment_multi_btn");
if (btn) {
btn.disabled = (count == 0);
}
}
</script>';
}
// CSS for highlighting
print '<style>
tr.highlight { background-color: #ffffcc !important; }
</style>';
}
// Show matches if found
$matches = $_SESSION['bankimport_matches_'.$object->id] ?? array();
if (!empty($matches) && $object->status == BankImportTransaction::STATUS_NEW) {
print '<br>';
print load_fiche_titre($langs->trans("MatchesFound", count($matches)), '', 'object_invoice');
// Translate match reasons
$reasonLabels = array(
'ref' => $langs->trans("MatchByRef"),
'ref_client' => $langs->trans("MatchByClientRef"),
'ref_supplier' => $langs->trans("MatchBySupplierRef"),
'amount' => $langs->trans("MatchByAmount"),
'amount_close' => $langs->trans("MatchByAmountClose"),
'name_exact' => $langs->trans("MatchByNameExact"),
'name_similar' => $langs->trans("MatchByNameSimilar"),
'iban' => $langs->trans("MatchByIBAN"),
'multi_invoice' => $langs->trans("MatchByMultiInvoice"),
'manual' => $langs->trans("ManualSearch")
);
// Check for multi-invoice matches
$hasMultiMatch = false;
foreach ($matches as $match) {
if ($match['type'] == 'multi_facture_fourn') {
$hasMultiMatch = true;
break;
}
}
// If we have multi-invoice matches, show them first with selection form
if ($hasMultiMatch) {
foreach ($matches as $match) {
if ($match['type'] == 'multi_facture_fourn' && !empty($match['invoices'])) {
print '<div class="info" style="margin-bottom: 15px;">';
print '<strong>'.$langs->trans("MatchByMultiInvoice").'</strong>: ';
print count($match['invoices']).' '.$langs->trans("Invoices").' = '.price($match['total'] ?? $match['amount'], 0, $langs, 1, -1, 2, 'EUR');
if (isset($match['difference']) && abs($match['difference']) > 0.01) {
print ' <span class="opacitymedium">('.$langs->trans("Difference").': '.price($match['difference'], 0, $langs, 1, -1, 2, 'EUR').')</span>';
}
print '</div>';
print '<form method="POST" action="'.$_SERVER["PHP_SELF"].'">';
print '<input type="hidden" name="token" value="'.newToken().'">';
print '<input type="hidden" name="action" value="confirmmulti">';
print '<input type="hidden" name="id" value="'.$object->id.'">';
print '<div class="div-table-responsive">';
print '<table class="noborder centpercent">';
print '<tr class="liste_titre">';
print '<th class="center" style="width: 30px;"><input type="checkbox" id="checkall_multi" checked onclick="toggleAllCheckboxes(this)"></th>';
print '<th>'.$langs->trans("Ref").'</th>';
print '<th>'.$langs->trans("SupplierRef").'</th>';
print '<th>'.$langs->trans("ThirdParty").'</th>';
print '<th class="right">'.$langs->trans("Amount").'</th>';
print '<th>'.$langs->trans("DateDue").'</th>';
print '</tr>';
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
foreach ($match['invoices'] as $invData) {
$inv = new FactureFournisseur($db);
$inv->fetch($invData['id']);
print '<tr class="oddeven">';
print '<td class="center"><input type="checkbox" name="invoices[]" value="'.$invData['id'].'" checked></td>';
print '<td>'.$inv->getNomUrl(1).'</td>';
print '<td>'.dol_escape_htmltag($invData['ref_supplier']).'</td>';
print '<td>'.dol_escape_htmltag($invData['socname']).'</td>';
print '<td class="right">'.price($invData['amount'], 0, $langs, 1, -1, 2, 'EUR').'</td>';
print '<td>'.($invData['date_due'] ? dol_print_date($invData['date_due'], 'day') : '-').'</td>';
print '</tr>';
}
print '</table>';
print '</div>';
print '<div style="margin-top: 10px;">';
print '<button type="submit" class="butAction">'.$langs->trans("ConfirmPayment").'</button>';
print '</div>';
print '</form>';
print '<br>';
// JavaScript for toggle all
print '<script>
function toggleAllCheckboxes(source) {
var checkboxes = document.getElementsByName("invoices[]");
for (var i = 0; i < checkboxes.length; i++) {
checkboxes[i].checked = source.checked;
}
}
</script>';
}
}
}
// Show single invoice matches
$singleMatches = array_filter($matches, function($m) { return $m['type'] != 'multi_facture_fourn'; });
if (!empty($singleMatches)) {
if ($hasMultiMatch) {
print '<br>';
print load_fiche_titre($langs->trans("Alternatives"), '', '');
}
print '<div class="div-table-responsive">';
print '<table class="noborder centpercent">';
print '<tr class="liste_titre">';
print '<th>'.$langs->trans("Type").'</th>';
print '<th>'.$langs->trans("Ref").'</th>';
print '<th>'.$langs->trans("ThirdParty").'</th>';
print '<th class="right">'.$langs->trans("Amount").'</th>';
print '<th>'.$langs->trans("DateDue").'</th>';
print '<th>'.$langs->trans("Score").'</th>';
print '<th>'.$langs->trans("MatchReason").'</th>';
print '<th></th>';
print '</tr>';
foreach ($singleMatches as $match) {
print '<tr class="oddeven">';
print '<td>'.($match['type'] == 'facture' ? $langs->trans("CustomerInvoice") : $langs->trans("SupplierInvoice")).'</td>';
// Get invoice link
if ($match['type'] == 'facture') {
require_once DOL_DOCUMENT_ROOT.'/compta/facture/class/facture.class.php';
$inv = new Facture($db);
$inv->fetch($match['id']);
print '<td>'.$inv->getNomUrl(1).'</td>';
} else {
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
$inv = new FactureFournisseur($db);
$inv->fetch($match['id']);
print '<td>'.$inv->getNomUrl(1).'</td>';
}
// Third party with link
if ($match['socid'] > 0) {
require_once DOL_DOCUMENT_ROOT.'/societe/class/societe.class.php';
$soc = new Societe($db);
$soc->fetch($match['socid']);
print '<td>'.$soc->getNomUrl(1).'</td>';
} else {
print '<td>'.dol_escape_htmltag($match['socname']).'</td>';
}
print '<td class="right">'.price($match['amount'], 0, $langs, 1, -1, 2, 'EUR').'</td>';
print '<td>'.($match['date_due'] ? dol_print_date($match['date_due'], 'day') : '-').'</td>';
// Score with color
$scoreColor = $match['match_score'] >= 80 ? 'green' : ($match['match_score'] >= 60 ? 'orange' : 'gray');
print '<td><span style="color: '.$scoreColor.'; font-weight: bold;">'.$match['match_score'].'%</span></td>';
// Match reasons as badges
print '<td>';
if (!empty($match['match_reasons'])) {
foreach ($match['match_reasons'] as $reason) {
$label = $reasonLabels[$reason] ?? $reason;
print '<span class="badge badge-secondary">'.$label.'</span> ';
}
}
print '</td>';
// Action buttons
print '<td class="center nowraponall">';
if (!empty($bankAccountId)) {
// Confirm payment button (creates payment in Dolibarr)
print '<a class="butActionSmall" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=confirmpayment&matchtype='.urlencode($match['type']).'&matchid='.$match['id'].'&token='.newToken().'">'.$langs->trans("ConfirmPayment").'</a>';
} else {
// Just link (no bank account configured)
print '<a class="butActionSmall" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=linkto&linktype='.$match['type'].'&linkid='.$match['id'].'&token='.newToken().'">'.$langs->trans("Link").'</a>';
}
print '</td>';
print '</tr>';
}
print '</table>';
print '</div>';
}
}
} else {
print '<div class="error">'.$langs->trans("RecordNotFound").'</div>';
}
// Back link
print '<div class="tabsAction">';
print '<a class="butAction" href="'.dol_buildpath('/bankimport/list.php', 1).'">'.$langs->trans("BackToList").'</a>';
print '</div>';
llxFooter();
$db->close();