implémentation impression reçus

FossilOrigin-Name: 81f54c69ca30c343f75b4190728e55d7679b6580724471ec69a6cca70f8f94ea
This commit is contained in:
engel 2023-01-31 18:35:27 +00:00
parent f5130172f0
commit 0e3c356d1a
7 changed files with 609 additions and 267 deletions

View File

@ -8,15 +8,12 @@
size: A4 portrait; size: A4 portrait;
margin: 1cm; margin: 1cm;
} }
body body.print
{ {
width : 19cm;
font-family: Serif; font-family: Serif;
font-size: 11pt; font-size: 11pt;
background-color: white; background-color: white;
width : 19cm;
}
#entete
{
} }
#logo #logo
{ {
@ -24,14 +21,13 @@
} }
#titre #titre
{ {
margin : 0 2.5cm 0 2.5cm; margin : 0 2cm 0 2cm;
text-align : center; text-align : center;
font-size : 14pt; font-size : 14pt;
font-weight: bold; font-weight: bold;
} }
#articles #articles
{ {
margin : 0 2.5cm 0.5cm 2.5cm;
text-align : center; text-align : center;
} }
#numRecu #numRecu
@ -39,20 +35,6 @@
text-align : right; text-align : right;
margin-right: 1em; margin-right: 1em;
} }
#beneficiaire, #donateur
{
}
#beneficiaire, #donateur, #versements, #final
{
}
#beneficiaire > h3
{
}
#donateur > h3
{
}
#versements #versements
{ {
@ -84,22 +66,10 @@
margin: 0 auto; margin: 0 auto;
padding-bottom : 2mm; padding-bottom : 2mm;
} }
#final
{
page-break-after: auto;
}
/*
@media print
{
div#imprimer {
display:none;
}
}
*/
</style> </style>
</head> </head>
<body> <body class="print">
<div class="cartouche" id="entete"> <div class="cartouche" id="entete">
<img id="logo" src="{{$logo_asso}}" /> <img id="logo" src="{{$logo_asso}}" />
<p id="titre">Reçu au titre des dons à certains organismes d'intérêt général</p> <p id="titre">Reçu au titre des dons à certains organismes d'intérêt général</p>

128
templates/recu_html.tpl Normal file
View File

@ -0,0 +1,128 @@
<!-- nav bar -->
{include file="%s/templates/_nav.tpl"|args:$plugin_root current_nav="activite"}
<?php
$fmt = new \NumberFormatter('fr_FR', \NumberFormatter::SPELLOUT);
if ($numero_sequentiel) { $numero_courant = $numero_sequentiel; }
?>
<div id="global" onload="changerStyle(this.document)">
{* Itération sur les personnes *}
{foreach from=$totalPersonnes key="idPersonne" item="personne"}
<div class="previs_recu">
<div class="cartouche" id="entete">
<img id="logo" src="{$logo_asso}" />
<p id="titre">Reçu au titre des dons à certains organismes d'intérêt général</p>
<p id="articles">Articles 200, 238 bis et 978 du code général des impôts</p>
<div id="numRecu">
{if $numero_sequentiel}
{afficher_numero_recu prefixe=$prefixeNum membre=$membre numero_personne=$personne->numero numero_sequentiel=$numero_courant}
<?php
++$numero_courant;
?>
{else}
{afficher_numero_recu prefixe=$prefixeNum membre=$membre numero_personne=$personne->numero numero_sequentiel=$numero_sequentiel}
{/if}
</div>
</div>
<div class="cartouche" id="beneficiaire">
<h3 class="rubrique">Bénéficiaire des versements</h3>
<p class="important">Association « {$nom_asso} »<br />
{$adresse_asso}<br />
<span class="titre">Objet&nbsp;:&nbsp;</span><span class="libelle">{$objet_asso}</span>
</p>
</div>
<div class="cartouche" id="donateur">
<h3 class="rubrique">Donateur</h3>
<p>
{$personne.nomPrenom}<br />
{$personne.adresse}<br />
{$personne.code} {$personne.ville}
{if $courriel && $personne.courriel != ""}
<br />courriel : <a href="mailto:{$personne.courriel}">{$personne.courriel}</a>
{/if}
</p>
</div>
<div class="cartouche" id="versements">
<p>Le bénéficiaire reconnaît avoir reçu au titre des dons et versements ouvrant droit à réduction d'impôt :</p>
<ul>
{foreach from=$personne.versements key="taux" item="versement"}
<li>
la somme de <b>***{$versement.montant|raw|money}*** euros</b>
<?php
$euros = $fmt->format((int)($versement->montant / 100));
if ($versement->montant % 100 != 0) {
$cents = $fmt->format($versement->montant % 100);
} else {
$cents = "";
}
?>
{if $cents != ""}
(<b>{$euros} euros et {$cents} cents</b>)
{else}
(<b>{$euros} euros</b>)
{/if}
<?php
$libelle = $libelles_taux[$taux];
?>
{if $libelle != ""}
({$libelle})
{/if}
<br /><span id="date_versements">date des versements&nbsp;:
{if $versement.dateMin == $versement.dateMax}
le {$versement.dateMin|date_format:"%d/%m/%Y"}
{else}
du {$versement.dateMin|date_format:"%d/%m/%Y"} au {$versement.dateMax|date_format:"%d/%m/%Y"}
{/if}
</span>
</li>
{/foreach}
</ul>
{foreach from=$complements item="elem"}
<p class="complements"><span class="titre">{$elem.titre}</span>&nbsp;<span class="libelle">{$elem.libelle}</span></p>
{/foreach}
<p class="complements">Le bénéficiaire certifie sur lhonneur que les dons et versements quil reçoit ouvrent droit à la réduction d'impôt prévue {$texteArticles} du code général des impôts.</p>
</div>
<div class="cartouche" id="final">
<p id="ville">{$ville_asso} le {$date}
<img id="signature" src="{$signature}" />
</p>
<div>
<span id="nom">{$nom_responsable}</span><br />
<span id="fonction">{$fonction_responsable}</span>
</div>
</div>
</div>
{/foreach} {* Itération sur les personnes *}
<div>
{* scripts divers *}
<script src="script.js"></script>
{*
* remplacer la feuille de style d'impression de paheko par la mienne
* puis déclencher l'impression
*}
{literal}
<script type="text/javascript">
document.addEventListener('DOMContentLoaded',
function() {
changerStyle(document);
setTimeout(function() {
window.print()
}, 750);
});
</script>
{/literal}
<!-- footer -->
{include file="admin/_foot.tpl"}

View File

@ -57,8 +57,6 @@ if (f('save') && $form->check('recusfiscaux_config')) {
$plugin->setConfig('ville_asso', trim(f('ville_asso'))); $plugin->setConfig('ville_asso', trim(f('ville_asso')));
} }
// signature // signature
// error_log("SESSION['sig_file'] = " . print_r($_SESSION['sig_file'], true));
// error_log("plugin->getConfig('signature') = " . $plugin->getConfig('signature'));
if (isset($_SESSION['sig_file']) && count($_SESSION['sig_file']) > 0) { if (isset($_SESSION['sig_file']) && count($_SESSION['sig_file']) > 0) {
// supprimer la signature précédente, si besoin // supprimer la signature précédente, si besoin
@ -78,7 +76,6 @@ if (f('save') && $form->check('recusfiscaux_config')) {
// autres informations // autres informations
// numérotation des reçus // numérotation des reçus
$configNum = $plugin->getConfig('numerotation'); $configNum = $plugin->getConfig('numerotation');
error_log("configNum=" . print_r($configNum, true));
$formNum = clone $configNum; $formNum = clone $configNum;
if ($configNum->prefixe != trim(f('prefixe'))) { if ($configNum->prefixe != trim(f('prefixe'))) {
$formNum->prefixe = trim(f('prefixe')); $formNum->prefixe = trim(f('prefixe'));

View File

@ -11,39 +11,34 @@ use Garradin\Plugin\RecusFiscaux\Personne;
// signature // signature
$signature = $signature =
(null !== $plugin->getConfig('signature')) ? (null !== $plugin->getConfig('signature')) ?
Files::get($plugin->getConfig('signature'))->fullpath() : \KD2\HTTP::getScheme() . '://' . \KD2\HTTP::getHost() . WWW_URI . "/" . $plugin->getConfig('signature') :
""; "";
// logo // logo
$logo_file = Files::get(File::CONTEXT_CONFIG . '/logo.png'); $config = Config::getInstance();
$logo_asso = $logo_asso =
(null !== $logo_file) ? (null !== $config->fileURL('logo')) ?
Files::get($logo_file->path)->fullpath() : $config->fileURL('logo') :
""; "";
// articles du CGI // articles du CGI
$articlesCGI = array(); $articlesCGI = array();
foreach ($plugin->getConfig('articlesCGI') as $article) foreach ($plugin->getConfig('articlesCGI') as $article) {
{
if ($article->valeur == 1) { if ($article->valeur == 1) {
$articlesCGI[] = $article->titre; $articlesCGI[] = $article->titre;
} }
} }
$nbArticles = count($articlesCGI); $nbArticles = count($articlesCGI);
if ($nbArticles == 1) if ($nbArticles == 1) {
{
$texteArticles = 'à larticle ' . $articlesCGI[0]; $texteArticles = 'à larticle ' . $articlesCGI[0];
} } elseif ($nbArticles > 1) {
elseif ($nbArticles > 1)
{
$texteArticles = 'aux articles '; $texteArticles = 'aux articles ';
for ($i = 0; $i < $nbArticles; ++$i) { for ($i = 0; $i < $nbArticles; ++$i) {
$texteArticles .= $articlesCGI[$i]; $texteArticles .= $articlesCGI[$i];
if ($i < $nbArticles - 2) { if ($i < $nbArticles - 2) {
$texteArticles .= ", "; $texteArticles .= ", ";
} } else if ($i == $nbArticles - 2) {
else if ($i == $nbArticles - 2) {
$texteArticles .= " et "; $texteArticles .= " et ";
} }
} }
@ -52,23 +47,8 @@ elseif ($nbArticles > 1)
// libellés pour les taux de réduction // libellés pour les taux de réduction
$libelles_taux = Utils::getLignesReduction($plugin->getConfig('reduction')); $libelles_taux = Utils::getLignesReduction($plugin->getConfig('reduction'));
// numérotation des reçus
$configNum = $plugin->getConfig('numerotation'); $configNum = $plugin->getConfig('numerotation');
error_log("config num = " . print_r($configNum, true));
$prefixeNum = "";
if (isset($configNum->prefixe) && $configNum->prefixe != "") {
$prefixeNum = $configNum->prefixe;
}
if (isset($configNum->annee) && $configNum->annee) {
if ($prefixeNum != "") { $prefixeNum .= "-"; }
$prefixeNum .= $_SESSION['annee_recu'];
}
if (isset($configNum->sequentiel) && $configNum->sequentiel) {
if (isset($configNum->valeur_init) && $configNum->valeur_init !="") {
$numero_sequentiel = $configNum->valeur_init;
} else {
$numero_sequentiel = 1;
}
}
// filtrer les versements sélectionnés // filtrer les versements sélectionnés
$lesLignes = f('selected'); $lesLignes = f('selected');
@ -78,125 +58,178 @@ foreach ($lesLignes as $ligne) {
} }
// cumuler les versements // cumuler les versements
if ($_GET['type'] == 'personne') if ($_GET['type'] == 'personne') {
{
$totalPersonnes = cumulerVersementsPersonne($versementsSelectionnes); $totalPersonnes = cumulerVersementsPersonne($versementsSelectionnes);
} } elseif ($_GET['type'] == 'activite') {
elseif ($_GET['type'] == 'activite')
{
$totalPersonnes = cumulerVersementsTarif($versementsSelectionnes); $totalPersonnes = cumulerVersementsTarif($versementsSelectionnes);
} }
// générer les reçus // générer les reçus
$listeFichiersPDF = array(); if ($_GET['format'] == 'pdf') {
$fmt = new \NumberFormatter('fr_FR', \NumberFormatter::SPELLOUT); genererRecusPDF($totalPersonnes,
foreach ($totalPersonnes as $idPersonne => $personne) $signature,
{ $logo_asso,
$tpl = new UserTemplate(); $texteArticles,
$tpl->setSource(PLUGIN_ROOT . '/templates/recu.skel'); $plugin,
$configNum,
$libelles_taux
);
} else if ($_GET['format'] == 'print') {
generererRecusHTML($tpl,
$totalPersonnes,
$signature,
$logo_asso,
$texteArticles,
$plugin,
$configNum,
$libelles_taux
);
} else {
// Erreur : format inconnu ; ne devrait pas se produire
}
$tpl->assignArray(compact('signature', 'logo_asso', 'texteArticles')); function genererRecusPDF($totalPersonnes,
$signature,
$logo_asso,
$texteArticles,
$plugin,
$configNum,
$libelles_taux
)
{
$listeFichiersPDF = array();
$fmt = new \NumberFormatter('fr_FR', \NumberFormatter::SPELLOUT);
$prefixeNum = getNumPrefixe($configNum);
$numero_sequentiel = getNumSequentiel($configNum);
foreach ($totalPersonnes as $idPersonne => $personne) {
$tpl = new UserTemplate();
$tpl->setSource(PLUGIN_ROOT . '/templates/recu.skel');
$tpl->assignArray(compact('signature', 'logo_asso', 'texteArticles'));
$tpl->assign('objet_asso', $plugin->getConfig('objet_asso'));
$tpl->assign('nom_responsable', $plugin->getConfig('nom_responsable'));
$tpl->assign('fonction_responsable', $plugin->getConfig('fonction_responsable'));
$tpl->assign('ville_asso', $plugin->getConfig('ville_asso'));
$tpl->assign('nom', $personne->nomPrenom);
$tpl->assign('adresse', $personne->adresse);
$tpl->assign('code_postal', $personne->codePostal);
$tpl->assign('ville', $personne->ville);
$tpl->assign('date', date("j/m/Y"));
// numéro de reçu
$tpl->assign('numero',
faireNumeroRecu($prefixeNum,
$configNum->membre,
$personne->numero,
$numero_sequentiel));
// adresse de courriel
if ($plugin->getConfig('imprimerCourriel')) {
$courriel = $personne->courriel;
} else {
$courriel = "";
}
$tpl->assign('courriel', $courriel);
// les versements
$tpl->registerSection(
'versements',
function () use ($personne, $libelles_taux, $fmt) {
foreach ($personne->versements as $taux => $versement) {
$ligne['montant'] = $versement->montant;
$ligne['euros'] = $fmt->format((int)($versement->montant / 100));
if ($versement->montant % 100 != 0) {
$ligne['cents'] = $fmt->format($versement->montant % 100);
} else {
$ligne['cents'] = "";
}
$ligne['libelle'] = $libelles_taux[$taux];
$ligne['dateMin'] = date("d/m/Y", $versement->dateMin);
$ligne['dateMax'] = date("d/m/Y", $versement->dateMax);
yield $ligne;
}
}
);
// mentions complémentaires
$complements = mentionsComplémentaires();
$tpl->registerSection(
'informations',
function () use ($complements) {
foreach ($complements as $elem) {
yield (array) $elem;
}
}
);
// fabriquer le fichier PDF
genererPDF($tpl->fetch(), $personne->nomPrenom, $listeFichiersPDF);
}
// faire une archive zip
$fichierZip = Utils::makeArchive(
$listeFichiersPDF,
$_SESSION['annee_recu'],
PLUGIN_ROOT . "/zip"
);
//supprimer les fichiers pdf (utile ?)
foreach ($listeFichiersPDF as $f) {
// Utils::safe_unlink($f);
}
} // genererRecusPDF
function generererRecusHTML($tpl,
$totalPersonnes,
$signature,
$logo_asso,
$texteArticles,
$plugin,
$configNum,
$libelles_taux
)
{
$tpl->register_function('afficher_numero_recu', function($params)
{
$prefixeNum = $params['prefixe'];
$membre = $params['membre'];
$numero_personne = $params['numero_personne'];
$numero_sequentiel = $params['numero_sequentiel'];
$numero = faireNumeroRecu($prefixeNum,
$membre,
$numero_personne,
$numero_sequentiel);
$out = sprintf('
<p class="important">Reçu %s</p>',
$numero
);
return $out;
});
$tpl->assign(compact(
'totalPersonnes',
'logo_asso',
'signature',
'libelles_taux',
'texteArticles'
));
$tpl->assign('prefixeNum', getNumPrefixe($configNum));
$tpl->assign('membre', $configNum->membre);
$tpl->assign('numero_sequentiel', getNumSequentiel($configNum));
$tpl->assign('nom_asso', Config::getInstance()->get('nom_asso'));
$tpl->assign('adresse_asso', Config::getInstance()->get('adresse_asso'));
$tpl->assign('objet_asso', $plugin->getConfig('objet_asso')); $tpl->assign('objet_asso', $plugin->getConfig('objet_asso'));
$tpl->assign('courriel', $plugin->getConfig('imprimerCourriel'));
$tpl->assign('complements', mentionsComplémentaires());
$tpl->assign('ville_asso', $plugin->getConfig('ville_asso'));
$tpl->assign('date', date("j/m/Y"));
$tpl->assign('nom_responsable', $plugin->getConfig('nom_responsable')); $tpl->assign('nom_responsable', $plugin->getConfig('nom_responsable'));
$tpl->assign('fonction_responsable', $plugin->getConfig('fonction_responsable')); $tpl->assign('fonction_responsable', $plugin->getConfig('fonction_responsable'));
$tpl->assign('ville_asso', $plugin->getConfig('ville_asso'));
$tpl->assign('nom', $personne->nomPrenom);
$tpl->assign('adresse', $personne->adresse);
$tpl->assign('code_postal', $personne->codePostal);
$tpl->assign('ville', $personne->ville);
$tpl->assign('date', date("j/m/Y"));
// numéro de reçu $tpl->assign('plugin_css', ['previs_recu.css', 'imprimer_recu.css']);
$chaineNum = $prefixeNum; $tpl->display(PLUGIN_ROOT . '/templates/recu_html.tpl');
if (isset($configNum->membre) && $configNum->membre) { } // generererRecusHTML
if ($chaineNum != "") { $chaineNum .= "-"; }
$chaineNum .= $personne->numero;
}
if (isset($configNum->sequentiel) && $configNum->sequentiel) {
if ($chaineNum != "") { $chaineNum .= "-"; }
$chaineNum .= $numero_sequentiel;
++$numero_sequentiel;
}
$tpl->assign('numero', $chaineNum);
// adresse de courriel
if ($plugin->getConfig('imprimerCourriel')) {
$courriel = $personne->courriel;
}
else {
$courriel = "";
}
$tpl->assign('courriel', $courriel);
// les versements
$tpl->registerSection('versements',
function () use($personne, $libelles_taux, $fmt)
{
foreach ($personne->versements as $taux => $versement)
{
$ligne['montant'] = $versement->montant;
$ligne['euros'] = $fmt->format((int)($versement->montant/100));
if ($versement->montant % 100 != 0) {
$ligne['cents'] = $fmt->format($versement->montant % 100);
} else {
$ligne['cents'] = "";
}
$ligne['libelle'] = $libelles_taux[$taux];
$ligne['dateMin'] = date("d/m/Y", $versement->dateMin);
$ligne['dateMax'] = date("d/m/Y", $versement->dateMax);
yield $ligne;
}
});
// mentions complémentaires
$donnees = array(
'Nature du don : ' => "Numéraire",
'Mode de versement : ' => "chèque et/ou virement"
);
$infos = array();
foreach ($donnees as $titre => $libelle)
{
$elem = new \stdClass();
$elem->titre = $titre;
$elem->libelle = $libelle;
$infos[] = $elem;
}
$tpl->registerSection('informations',
function () use ($infos)
{
foreach ($infos as $elem)
{
yield (array) $elem;
}
});
// fabriquer le fichier PDF
$result = $tpl->fetch();
$nomPDF = \Garradin\Utils::filePDF($result);
// changer le nom du fichier
$nom = str_replace(' ', '_', $personne->nomPrenom);
$nom = str_replace("'", "", $nom);
$nomFichier = sprintf('%s/recu_%s_%s.pdf',
dirname($nomPDF),
$_SESSION['annee_recu'],
$nom);
rename($nomPDF, $nomFichier);
// ajouter le nom du fichier à la liste pour mettre dans une archive
$listeFichiersPDF[] = $nomFichier;
}
// faire une archive zip
$fichierZip = Utils::makeArchive(
$listeFichiersPDF,
$_SESSION['annee_recu'],
PLUGIN_ROOT . "/zip"
);
//supprimer les fichiers pdf (utile ?)
foreach ($listeFichiersPDF as $f)
{
unlink($f);
}
/** /**
* Cumuler les versements de chaque personne * Cumuler les versements de chaque personne
@ -210,13 +243,10 @@ function cumulerVersementsPersonne($versements)
$dateMin = PHP_INT_MAX; $dateMin = PHP_INT_MAX;
$dateMax = -1; $dateMax = -1;
$totalVersements = 0; $totalVersements = 0;
foreach ($versements as $ligne) foreach ($versements as $ligne) {
{ if ($ligne->idUser != $idPersonneCourant) {
if ($ligne->idUser != $idPersonneCourant)
{
// changement de personne // changement de personne
if ($idPersonneCourant != -1) if ($idPersonneCourant != -1) {
{
$totalPersonnes[$idPersonneCourant]->ajouterVersement( $totalPersonnes[$idPersonneCourant]->ajouterVersement(
$_SESSION['taux_reduction'], $_SESSION['taux_reduction'],
$totalVersements, $totalVersements,
@ -229,15 +259,18 @@ function cumulerVersementsPersonne($versements)
$idPersonneCourant = $ligne->idUser; $idPersonneCourant = $ligne->idUser;
$totalVersements = $ligne->versement; $totalVersements = $ligne->versement;
// créer les infos de la personne, sauf si elle est déjà présente // créer les infos de la personne, sauf si elle est déjà présente
if (!array_key_exists($idPersonneCourant, $totalPersonnes)) if (!array_key_exists($idPersonneCourant, $totalPersonnes)) {
{ $totalPersonnes[$idPersonneCourant] = $_SESSION['membresDonateurs'][$ligne->idUser]->clone();
$totalPersonnes["$idPersonneCourant"] = $_SESSION['membresDonateurs'][$ligne->idUser]->clone();
} }
} else { } else {
// même personne : cumuler versements et mettre à jour les dates // même personne : cumuler versements et mettre à jour les dates
$totalVersements += $ligne->versement; $totalVersements += $ligne->versement;
if (strtotime($ligne->date) < $dateMin) { $dateMin = strtotime($ligne->date); } if (strtotime($ligne->date) < $dateMin) {
if (strtotime($ligne->date) > $dateMax) { $dateMax = strtotime($ligne->date); } $dateMin = strtotime($ligne->date);
}
if (strtotime($ligne->date) > $dateMax) {
$dateMax = strtotime($ligne->date);
}
} }
} }
// et le dernier // et le dernier
@ -248,7 +281,7 @@ function cumulerVersementsPersonne($versements)
$dateMax $dateMax
); );
return $totalPersonnes; return $totalPersonnes;
} } // cumulerVersementsPersonne
/** /**
* Cumuler les versements de chaque personne par tarif * Cumuler les versements de chaque personne par tarif
@ -264,20 +297,17 @@ function cumulerVersementsTarif($versements)
$dateMin = PHP_INT_MAX; $dateMin = PHP_INT_MAX;
$dateMax = -1; $dateMax = -1;
$totalVersements = 0; $totalVersements = 0;
foreach ($versements as $ligne) foreach ($versements as $ligne) {
{
if ( if (
$ligne->idTarif != $idTarifCourant || $ligne->idTarif != $idTarifCourant ||
$ligne->idUser != $idPersonneCourant || $ligne->idUser != $idPersonneCourant ||
$ligne->idCompte != $idCompteCourant $ligne->idCompte != $idCompteCourant
) ) {
{ if ($idTarifCourant != -1) {
if ($idTarifCourant != -1)
{
// changement de tarif, de personne ou de compte // changement de tarif, de personne ou de compte
$tarifCompte = ($idTarifCourant == 0) ? $tarifCompte = ($idTarifCourant == 0) ?
$idCompteCourant : $idCompteCourant :
$idTarifCourant . "_" . $idCompteCourant; $idTarifCourant . "_" . $idCompteCourant;
$totalPersonnes[$idPersonneCourant]->ajouterVersement( $totalPersonnes[$idPersonneCourant]->ajouterVersement(
$_SESSION['tauxSelectionnes'][$tarifCompte], $_SESSION['tauxSelectionnes'][$tarifCompte],
$totalVersements, $totalVersements,
@ -292,21 +322,24 @@ function cumulerVersementsTarif($versements)
$idCompteCourant = $ligne->idCompte; $idCompteCourant = $ligne->idCompte;
$totalVersements = $ligne->versement; $totalVersements = $ligne->versement;
// créer les infos de la personne, sauf si elle est déjà présente // créer les infos de la personne, sauf si elle est déjà présente
if (!array_key_exists($idPersonneCourant, $totalPersonnes)) if (!array_key_exists($idPersonneCourant, $totalPersonnes)) {
{ $totalPersonnes[$idPersonneCourant] = $_SESSION['membresDonateurs'][$ligne->idUser]->clone();
$totalPersonnes["$idPersonneCourant"] = $_SESSION['membresDonateurs'][$ligne->idUser]->clone();
} }
} else { } else {
// même personne : cumuler versements et mettre à jour les dates // même personne : cumuler versements et mettre à jour les dates
$totalVersements += $ligne->versement; $totalVersements += $ligne->versement;
if (strtotime($ligne->date) < $dateMin) { $dateMin = strtotime($ligne->date); } if (strtotime($ligne->date) < $dateMin) {
if (strtotime($ligne->date) > $dateMax) { $dateMax = strtotime($ligne->date); } $dateMin = strtotime($ligne->date);
}
if (strtotime($ligne->date) > $dateMax) {
$dateMax = strtotime($ligne->date);
}
} }
} }
// et le dernier // et le dernier
$tarifCompte = ($idTarifCourant == 0) ? $tarifCompte = ($idTarifCourant == 0) ?
$idCompteCourant : $idCompteCourant :
$idTarifCourant . "_" . $idCompteCourant; $idTarifCourant . "_" . $idCompteCourant;
$totalPersonnes[$idPersonneCourant]->ajouterVersement( $totalPersonnes[$idPersonneCourant]->ajouterVersement(
$_SESSION['tauxSelectionnes'][$tarifCompte], $_SESSION['tauxSelectionnes'][$tarifCompte],
$totalVersements, $totalVersements,
@ -314,4 +347,94 @@ function cumulerVersementsTarif($versements)
$dateMax $dateMax
); );
return $totalPersonnes; return $totalPersonnes;
} // cumulerVersementsTarif
/**
* génère un fichier PDF à partir d'un document html
* ajoute son nom à la liste de fichiers
*/
function genererPDF($docHTML, $nomPersonne, &$listeFichiersPDF)
{
// fabriquer le fichier PDF
$nomPDF = \Garradin\Utils::filePDF($docHTML);
// changer le nom du fichier
$nom = str_replace(' ', '_', $nomPersonne);
$nom = str_replace("'", "", $nom);
$nomFichier = sprintf(
'%s/recu_%s_%s.pdf',
dirname($nomPDF),
$_SESSION['annee_recu'],
$nom
);
rename($nomPDF, $nomFichier);
// ajouter le nom du fichier à la liste pour mettre dans une archive
$listeFichiersPDF[] = $nomFichier;
} // genererPDF
function faireNumeroRecu($prefixeNum, $membre, $numero, &$numero_sequentiel)
{
if (isset($membre) && $membre) {
if ($prefixeNum != "") {
$prefixeNum .= "-";
}
$prefixeNum .= $numero;
}
if ($numero_sequentiel) {
if ($prefixeNum != "") {
$prefixeNum .= "-";
}
$prefixeNum .= $numero_sequentiel;
++$numero_sequentiel;
}
return $prefixeNum;
}
/**
* renvoyer le préfixe du numéro de reçu
*/
function getNumPrefixe($configNum)
{
$prefixeNum = "";
if (isset($configNum->prefixe) && $configNum->prefixe != "") {
$prefixeNum = $configNum->prefixe;
}
if (isset($configNum->annee) && $configNum->annee) {
if ($prefixeNum != "") {
$prefixeNum .= "-";
}
$prefixeNum .= $_SESSION['annee_recu'];
}
return $prefixeNum;
}
/**
* renvoyer le premier numéro de la numérotation séquentielle
* renvoie false si pas de numérotation séquentielle
*/
function getNumSequentiel($configNum)
{
if (isset($configNum->sequentiel) && $configNum->sequentiel) {
if (isset($configNum->valeur_init) && $configNum->valeur_init != "") {
$numero_sequentiel = $configNum->valeur_init;
} else {
$numero_sequentiel = 1;
}
}
return isset($numero_sequentiel) ? $numero_sequentiel : false;
}
function mentionsComplémentaires()
{
$donnees = array(
'Nature du don : ' => "Numéraire",
'Mode de versement : ' => "chèque et/ou virement"
);
$complements = array();
foreach ($donnees as $titre => $libelle) {
$elem = new \stdClass();
$elem->titre = $titre;
$elem->libelle = $libelle;
$complements[] = $elem;
}
return $complements;
} }

View File

@ -0,0 +1,38 @@
/*
* impression
*/
@page
{
size: A4 portrait;
}
header.header {
display: none;
}
body {
background: #fff;
padding: 0;
margin: 0;
}
.noprint {
display: none;
}
nav.tabs
{
display: none;
}
main {
margin: 0;
}
div.previs_recu
{
font-family: Serif;
font-size: 11pt;
background-color: white;
page-break-after: always;
}

97
www/admin/previs_recu.css Normal file
View File

@ -0,0 +1,97 @@
/*
* prévisualisation reçu au format HTML
*/
div.previs_recu
{
width : 18.5cm;
}
#logo
{
max-height : 4cm;
}
#titre
{
margin : 0 2cm 0 2cm;
text-align : center;
font-size : 14pt;
font-weight: bold;
}
#articles
{
margin-bottom: 0.5cm;
text-align : center;
}
#numRecu
{
text-align : right;
margin-right: 1em;
}
#versements
{
border-top: 1px solid rgb(0, 0, 128);
border-bottom: 1px solid rgb(0, 0, 128);
padding-top: 1em;
margin-bottom: 1em;
}
.cartouche
{
padding-bottom: 1em;
}
.rubrique
{
background-color : rgb(200, 200, 250);
padding : 0 0 0 1mm;
margin-bottom: 1em;
}
.titre, .important
{
font-weight:bold;
}
.libelle
{
font-weight: normal;
}
#ville
{
margin-bottom: 0;
}
#signature
{
display: block;
max-width : 7cm;
max-height : 4cm;
margin: 0 auto;
padding-bottom : 2mm;
}
#versements > ul
{
list-style: inside;
margin-left: 2em;
}
#date_versements
{
margin-left: 1.5em;
}
p.complements
{
margin-top : 1em;
}
span.titre, span.libelle
{
display : inline;
}

View File

@ -9,12 +9,10 @@
* ()sélectionner toutes les cases de toutes les activités * ()sélectionner toutes les cases de toutes les activités
* @param {HTMLInputElement} idCaseGlobale id de la case globale * @param {HTMLInputElement} idCaseGlobale id de la case globale
*/ */
function cocherDecocherTout(idCaseGlobale) function cocherDecocherTout(idCaseGlobale) {
{
// itérer sur la liste des éléments détails : 1 par couple <activité, tarif> // itérer sur la liste des éléments détails : 1 par couple <activité, tarif>
let lesDetails = document.querySelectorAll("details.activite"); let lesDetails = document.querySelectorAll("details.activite");
for (let i = 0; i < lesDetails.length; ++i) for (let i = 0; i < lesDetails.length; ++i) {
{
let idCase = lesDetails[i].querySelector("input[type=checkbox]"); let idCase = lesDetails[i].querySelector("input[type=checkbox]");
idCase.checked = idCaseGlobale.checked; idCase.checked = idCaseGlobale.checked;
cocherDecocherTarif(idCase); cocherDecocherTarif(idCase);
@ -28,8 +26,7 @@ function cocherDecocherTout(idCaseGlobale)
* ()sélectionner toutes les cases de cette activité * ()sélectionner toutes les cases de cette activité
* @param {HTMLInputElement} idCaseGlobale id de la case d'activité * @param {HTMLInputElement} idCaseGlobale id de la case d'activité
*/ */
function cocherDecocherTarif(idCaseGlobale) function cocherDecocherTarif(idCaseGlobale) {
{
let lesPersonnes = idCaseGlobale.closest("details").querySelectorAll("div.personne"); let lesPersonnes = idCaseGlobale.closest("details").querySelectorAll("div.personne");
cocherDecocherLesPersonnes(idCaseGlobale, lesPersonnes); cocherDecocherLesPersonnes(idCaseGlobale, lesPersonnes);
} }
@ -38,8 +35,7 @@ function cocherDecocherTarif(idCaseGlobale)
* idem dans le cas des versements des personnes * idem dans le cas des versements des personnes
* @param {HTMLInputElement} idCaseGlobale id case à cocher d'une personne * @param {HTMLInputElement} idCaseGlobale id case à cocher d'une personne
*/ */
function cocherDecocherToutesLesPersonnes(idCaseGlobale) function cocherDecocherToutesLesPersonnes(idCaseGlobale) {
{
let lesPersonnes = document.querySelectorAll("div.personne"); let lesPersonnes = document.querySelectorAll("div.personne");
cocherDecocherLesPersonnes(idCaseGlobale, lesPersonnes); cocherDecocherLesPersonnes(idCaseGlobale, lesPersonnes);
changerMessage(idCaseGlobale.nextElementSibling, idCaseGlobale); changerMessage(idCaseGlobale.nextElementSibling, idCaseGlobale);
@ -49,10 +45,8 @@ function cocherDecocherToutesLesPersonnes(idCaseGlobale)
* @param {HTMLInputElement} idCaseGlobale * @param {HTMLInputElement} idCaseGlobale
* @param {NodeListOf<Element>} lesPersonnes * @param {NodeListOf<Element>} lesPersonnes
*/ */
function cocherDecocherLesPersonnes(idCaseGlobale, lesPersonnes) function cocherDecocherLesPersonnes(idCaseGlobale, lesPersonnes) {
{ for (let j = 0; j < lesPersonnes.length; ++j) {
for (let j = 0; j < lesPersonnes.length; ++j)
{
// trouver l'élément total de la personne // trouver l'élément total de la personne
let idTotal = lesPersonnes[j].querySelector("span"); let idTotal = lesPersonnes[j].querySelector("span");
// puis la case à cocher // puis la case à cocher
@ -70,13 +64,11 @@ function cocherDecocherLesPersonnes(idCaseGlobale, lesPersonnes)
* @param {HTMLInputElement} idCase id de la case qui a été cochée * @param {HTMLInputElement} idCase id de la case qui a été cochée
* @param {HTMLSpanElement} idTotal id de l'élément afficher le total * @param {HTMLSpanElement} idTotal id de l'élément afficher le total
*/ */
function cocherDecocherPersonne(idCase, idTotal) function cocherDecocherPersonne(idCase, idTotal) {
{
// chercher le fieldset des versements // chercher le fieldset des versements
let fieldset = idCase.closest("details").querySelector("div.versements"); let fieldset = idCase.closest("details").querySelector("div.versements");
let listeCases = fieldset.querySelectorAll("input[type=checkbox]"); let listeCases = fieldset.querySelectorAll("input[type=checkbox]");
for (let i = 0; i < listeCases.length; ++i) for (let i = 0; i < listeCases.length; ++i) {
{
listeCases[i].checked = idCase.checked; listeCases[i].checked = idCase.checked;
cocherDecocherVersement(listeCases[i], idTotal); cocherDecocherVersement(listeCases[i], idTotal);
} }
@ -89,8 +81,7 @@ function cocherDecocherPersonne(idCase, idTotal)
* @param {HTMLInputElement} idCase id de la case qui a été cochée * @param {HTMLInputElement} idCase id de la case qui a été cochée
* @param {HTMLSpanElement} idTotal id de l'élément afficher le total * @param {HTMLSpanElement} idTotal id de l'élément afficher le total
*/ */
function cocherDecocherVersement(idCase, idTotal) function cocherDecocherVersement(idCase, idTotal) {
{
let fieldset = idCase.closest("div.versements"); let fieldset = idCase.closest("div.versements");
let listeCases = fieldset.querySelectorAll("input[type=checkbox]"); let listeCases = fieldset.querySelectorAll("input[type=checkbox]");
let listeMontants = fieldset.querySelectorAll("span.montant"); let listeMontants = fieldset.querySelectorAll("span.montant");
@ -103,19 +94,19 @@ function cocherDecocherVersement(idCase, idTotal)
* @param {NodeListOf<Element>} listeMontants liste des montants associés * @param {NodeListOf<Element>} listeMontants liste des montants associés
* @param {HTMLSpanElement} idTotal id de l'élément afficher le total * @param {HTMLSpanElement} idTotal id de l'élément afficher le total
*/ */
function calculerTotal(listeCases, listeMontants, idTotal) function calculerTotal(listeCases, listeMontants, idTotal) {
{
let total = 0; let total = 0;
for (let i = 0; i < listeCases.length; ++i) for (let i = 0; i < listeCases.length; ++i) {
{
if (listeCases[i].checked) { if (listeCases[i].checked) {
total += parseFloat(listeMontants[i].textContent.replace(/\s/g, "").replace(",", ".")); total += parseFloat(listeMontants[i].textContent.replace(/\s/g, "").replace(",", "."));
} }
} }
// afficher le total // afficher le total
idTotal.innerHTML = idTotal.innerHTML =
total.toLocaleString('fr-FR', {style: 'currency', currency: 'EUR', total.toLocaleString('fr-FR', {
minimumFractionDigits: 2}); style: 'currency', currency: 'EUR',
minimumFractionDigits: 2
});
} }
/** /**
@ -123,8 +114,7 @@ function calculerTotal(listeCases, listeMontants, idTotal)
* @param {Element} message * @param {Element} message
* @param {HTMLInputElement} idCase * @param {HTMLInputElement} idCase
*/ */
function changerMessage(message, idCase) function changerMessage(message, idCase) {
{
if (idCase.checked) { if (idCase.checked) {
message.innerHTML = "Cliquer pour dé-cocher toutes les lignes"; message.innerHTML = "Cliquer pour dé-cocher toutes les lignes";
} else { } else {
@ -138,14 +128,11 @@ function changerMessage(message, idCase)
* @param {string} classe des détails à afficher/masquer * @param {string} classe des détails à afficher/masquer
* @param {string} texte du bouton * @param {string} texte du bouton
*/ */
function montrerMasquerDetails(idElem, classe, texte) function montrerMasquerDetails(idElem, classe, texte) {
{
let lesDetails = document.querySelectorAll(classe); let lesDetails = document.querySelectorAll(classe);
if (lesDetails.length > 0) if (lesDetails.length > 0) {
{
let leBouton = document.getElementById(idElem); let leBouton = document.getElementById(idElem);
if (leBouton.textContent.includes('Replier')) if (leBouton.textContent.includes('Replier')) {
{
// masquer // masquer
lesDetails.forEach((e) => { lesDetails.forEach((e) => {
e.removeAttribute('open'); e.removeAttribute('open');
@ -153,8 +140,7 @@ function montrerMasquerDetails(idElem, classe, texte)
leBouton.textContent = "Déplier " + texte; leBouton.textContent = "Déplier " + texte;
leBouton.setAttribute('data-icon', '↓'); leBouton.setAttribute('data-icon', '↓');
} }
else else {
{
// montrer // montrer
lesDetails.forEach((e) => { lesDetails.forEach((e) => {
e.setAttribute('open', 'open'); e.setAttribute('open', 'open');
@ -171,8 +157,7 @@ function montrerMasquerDetails(idElem, classe, texte)
* @return vrai si au moins un choix a été fait * @return vrai si au moins un choix a été fait
* @param {HTMLFormElement} formulaire * @param {HTMLFormElement} formulaire
*/ */
function verifierChoix(formulaire) function verifierChoix(formulaire) {
{
return verifierCases(formulaire, 'checkbox', "au moins un versement"); return verifierCases(formulaire, 'checkbox', "au moins un versement");
} }
@ -188,17 +173,13 @@ function verifierChoix(formulaire)
* @param {any} idElem id de l'élément à afficher * @param {any} idElem id de l'élément à afficher
* @param {any} nomClasse classe des éléments à masquer (sauf idElem) * @param {any} nomClasse classe des éléments à masquer (sauf idElem)
*/ */
function choixMethodeGeneration(formulaire, action, idElem, nomClasse) function choixMethodeGeneration(formulaire, action, idElem, nomClasse) {
{
formulaire.setAttribute('action', 'action.php?action=' + action); formulaire.setAttribute('action', 'action.php?action=' + action);
for (let elem of formulaire.querySelectorAll(nomClasse)) for (let elem of formulaire.querySelectorAll(nomClasse)) {
{ if (elem.id == idElem) {
if (elem.id == idElem)
{
elem.classList.remove('hidden'); elem.classList.remove('hidden');
} }
else else {
{
elem.classList.add('hidden'); elem.classList.add('hidden');
} }
} }
@ -210,23 +191,20 @@ function choixMethodeGeneration(formulaire, action, idElem, nomClasse)
* - qu'un radio de chaque activité/tarif sélectionné a été sélectionné :) * - qu'un radio de chaque activité/tarif sélectionné a été sélectionné :)
* @param conteneur des cases à vérifier * @param conteneur des cases à vérifier
*/ */
function verifierActivitésTaux(conteneur) function verifierActivitésTaux(conteneur) {
{
let nbChoix = 0; let nbChoix = 0;
// parcourir les cases à cocher // parcourir les cases à cocher
for (let idCase of conteneur.querySelectorAll("input[type=checkbox]")) for (let idCase of conteneur.querySelectorAll("input[type=checkbox]")) {
{
if (idCase.checked) { if (idCase.checked) {
++nbChoix; ++nbChoix;
// vérifier qu'un radio de la même ligne est sélectionné // vérifier qu'un radio de la même ligne est sélectionné
let ligneCorrecte = false; let ligneCorrecte = false;
// trouver la ligne englobante // trouver la ligne englobante
let ligne = idCase.closest("li"); let ligne = idCase.closest("li");
for (let idRadio of ligne.querySelectorAll('input[type=radio]')) for (let idRadio of ligne.querySelectorAll('input[type=radio]')) {
{
if (idRadio.checked) { ligneCorrecte = true; break; } if (idRadio.checked) { ligneCorrecte = true; break; }
} }
if (! ligneCorrecte) { if (!ligneCorrecte) {
alert("Erreur : il faut sélectionner un taux de réduction dans chaque ligne cochée"); alert("Erreur : il faut sélectionner un taux de réduction dans chaque ligne cochée");
return false; return false;
} }
@ -241,8 +219,7 @@ function verifierActivitésTaux(conteneur)
/** /**
* vérifier qu'un taux a été sélectionné dans le conteneur paramètre * vérifier qu'un taux a été sélectionné dans le conteneur paramètre
*/ */
function verifierTaux(conteneur) function verifierTaux(conteneur) {
{
return verifierCases(conteneur, 'radio', "un taux de réduction"); return verifierCases(conteneur, 'radio', "un taux de réduction");
} }
@ -253,17 +230,16 @@ function verifierTaux(conteneur)
/** /**
* vérifier les données saisies dans le formulaire de configuration * vérifier les données saisies dans le formulaire de configuration
*/ */
function verifierConfig(divArticles, divTauxReduc) function verifierConfig(divArticles, divTauxReduc) {
{
// articles // articles
if (! verifierCases(divArticles, "checkbox", "au moins un article")) { return false; } if (!verifierCases(divArticles, "checkbox", "au moins un article")) { return false; }
// taux de réduction // taux de réduction
if (! verifierCases(divTauxReduc, "checkbox", "au moins un taux de réduction")) { return false; } if (!verifierCases(divTauxReduc, "checkbox", "au moins un taux de réduction")) { return false; }
// Nom, fonction, signature // Nom, fonction, signature
// alert("Erreur : il faut sélectionner au moins un versement"); // alert("Erreur : il faut sélectionner au moins un versement");
return true; return true;
} }
@ -274,14 +250,27 @@ function verifierConfig(divArticles, divTauxReduc)
* @param type de case à vérifier (radio, checkbox) * @param type de case à vérifier (radio, checkbox)
* @param message à afficher si erreur * @param message à afficher si erreur
*/ */
function verifierCases(conteneur, type, message) function verifierCases(conteneur, type, message) {
{
let selecteur = "input[type=" + type + "]"; let selecteur = "input[type=" + type + "]";
let listeCheck = conteneur.querySelectorAll(selecteur); let listeCheck = conteneur.querySelectorAll(selecteur);
for (let elem of listeCheck) for (let elem of listeCheck) {
{
if (elem.checked) { return true; } if (elem.checked) { return true; }
} }
alert("Erreur : il faut sélectionner " + message); alert("Erreur : il faut sélectionner " + message);
return false; return false;
} }
/**
* petite bidouille pour utiliser ma feuille de style pour imprimer les reçus
* à la place de la feuille de style de paheko
* @param {*} document
*/
function changerStyle(document) {
let styles = document.querySelectorAll('link[rel="stylesheet"]');
// console.log(styles);
for (let sheet of styles) {
if (sheet.href.includes('print.css')) { sheet.media = "tv"; sheet.remove; }
if (sheet.href.includes('imprimer_recu.css')) { sheet.media = 'print'; }
}
// console.log(styles);
}