Fusion branche numerotation

FossilOrigin-Name: 85ee85c57467fd978fa9eb6f367472c8ff05f8770ec12d904f8edfce12673580
This commit is contained in:
engel 2023-01-24 13:35:39 +00:00
commit 9714062c65
13 changed files with 466 additions and 226 deletions

View File

@ -27,4 +27,6 @@ Vous pouvez télécharger [l'archive .tar.gz](https://ncloud6.zaclys.com/index.p
- signature (image) - signature (image)
- autres - autres
- ville (précède la date sur le formulaire) - ville (précède la date sur le formulaire)
- champs pour le nom et prénom (le libellé doit contenir le terme 'nom', casse indifférente) - paramétrage numéro de reçu (préfixe quelconque, année fiscale, numéro de memmbre ou séquentiel)
- possibilité imprimer adresse de courriel
- choix et ordre des champs pour le nom et prénom (le libellé doit contenir le terme 'nom', casse indifférente)

View File

@ -2,15 +2,15 @@
"articlesCGI" : [ "articlesCGI" : [
{ {
"titre" : "200", "titre" : "200",
"valeur" : 0 "valeur" : false
}, },
{ {
"titre" : "238 bis", "titre" : "238 bis",
"valeur" : 0 "valeur" : false
}, },
{ {
"titre" : "978", "titre" : "978",
"valeur" : 0 "valeur" : false
} }
], ],
"reduction" : [ "reduction" : [
@ -18,13 +18,21 @@
"taux" : "normal", "taux" : "normal",
"ligne" : "UF", "ligne" : "UF",
"remarque" : "", "remarque" : "",
"valeur" : 0 "valeur" : false
}, },
{ {
"taux" : "majoré", "taux" : "majoré",
"ligne" : "UD", "ligne" : "UD",
"remarque" : "aide aux personnes en difficulté", "remarque" : "aide aux personnes en difficulté",
"valeur" : 0 "valeur" : false
} }
] ],
"numerotation" : {
"prefixe" : "",
"annee" : false,
"membre" : false,
"sequentiel" : false,
"valeur_init": 1
},
"imprimerCourriel" : false
} }

View File

@ -1,8 +1,8 @@
nom="Reçus fiscaux" nom="Reçus fiscaux"
description="Génération de reçus fiscaux pour les dons des membres" description="Génération de reçus fiscaux pour les dons des membres"
auteur="jce" auteur="jce"
url="https://git.roflcopter.fr/lesanges/recus-fiscaux-garradin" url="https://ncloud6.zaclys.com/index.php/s/RZQK2So8HemkH3w"
version="0.7" version="0.8"
menu=1 menu=1
config=1 config=1
min_version="1.1.23" min_version="1.1.23"

View File

@ -8,6 +8,8 @@ namespace Garradin\Plugin\RecusFiscaux;
class Personne class Personne
{ {
public $id; public $id;
public $numero;
public $courriel;
public $rang; // par ordre alpha de nomPrenom ; sert aux tris public $rang; // par ordre alpha de nomPrenom ; sert aux tris
public $nomPrenom; public $nomPrenom;
public $adresse; public $adresse;
@ -17,6 +19,8 @@ class Personne
public function __construct( public function __construct(
$id, $id,
$numero,
$courriel,
$rang, $rang,
$nomPrenom, $nomPrenom,
$adresse, $adresse,
@ -25,6 +29,8 @@ class Personne
) )
{ {
$this->id = $id; $this->id = $id;
$this->numero = $numero;
$this->courriel = $courriel;
$this->rang = $rang; $this->rang = $rang;
$this->nomPrenom = $nomPrenom; $this->nomPrenom = $nomPrenom;
$this->adresse = $adresse; $this->adresse = $adresse;
@ -40,6 +46,8 @@ class Personne
{ {
return new Personne( return new Personne(
$this->id, $this->id,
$this->numero,
$this->courriel,
$this->rang, $this->rang,
$this->nomPrenom, $this->nomPrenom,
$this->adresse, $this->adresse,

View File

@ -297,6 +297,8 @@ class Utils
$sql = sprintf( $sql = sprintf(
'SELECT 'SELECT
membres.id as idUser, membres.id as idUser,
membres.numero,
membres.email,
row_number() over(order by %s) as rang, row_number() over(order by %s) as rang,
%s as nom, %s as nom,
membres.adresse as adresse, membres.adresse as adresse,
@ -327,6 +329,8 @@ class Utils
foreach (DB::getInstance()->iterate($sql) as $personne) foreach (DB::getInstance()->iterate($sql) as $personne)
{ {
$donateurs[$personne->idUser] = new Personne($personne->idUser, $donateurs[$personne->idUser] = new Personne($personne->idUser,
$personne->numero,
$personne->email,
$personne->rang, $personne->rang,
$personne->nom, $personne->nom,
$personne->adresse, $personne->adresse,

View File

@ -15,7 +15,7 @@
<fieldset> <fieldset>
<dl class="config"> <dl class="config">
<dt><label>Objet (but) de l'association</label> <b title="Champ obligatoire">(obligatoire)</b></dt> <dt><label>Objet (but) de l'association</label> <b title="Champ obligatoire">(obligatoire)</b></dt>
{input type="textarea" name="objet_asso" source=$plugin.config label="" required="required" cols="50" rows="4" maxlength=300} {input type="textarea" name="objet_asso" source=$plugin.config label="" required="required" cols="100" rows="3" maxlength=300}
</dl> </dl>
<dl class="config"> <dl class="config">
@ -23,47 +23,59 @@
<b title="Champ obligatoire">(obligatoire ; sélectionnez tous les articles qui s'appliquent à <b title="Champ obligatoire">(obligatoire ; sélectionnez tous les articles qui s'appliquent à
l'association)</b> l'association)</b>
</dt> </dt>
{foreach from=$plugin_config->articlesCGI key="key" item="article"} <div id="articles_cgi">
{* {foreach from=$plugin_config->articlesCGI key="key" item="article"}
{*
À VÉRIFIER : {input : checked ne fonctionne pas si l'attribut name est un tableau... À VÉRIFIER : {input : checked ne fonctionne pas si l'attribut name est un tableau...
{input type="checkbox" name="articlesCGI[]" value=$key label=$article.titre} {input type="checkbox" name="articlesCGI[]" value=$key label=$article.titre}
*} *}
<div> <div class="article">
<input type="checkbox" name="articlesCGI[]" id="article_{$key}" value="{$key}" class="choix" <input type="checkbox" name="articlesCGI[]" id="article_{$key}" value="{$key}" class="choix"
{if $article.valeur == 1}checked{/if} /> {if $article.valeur == 1}checked{/if} />
<label for="article_{$key}">Article {$article.titre}</label> <label for="article_{$key}">Article {$article.titre}</label>
</div> </div>
{/foreach} {/foreach}
</div>
</dl> </dl>
<dl class="config"> <dl class="config">
<dt><label>Taux de réduction applicables : </label> <dt><label>Taux de réduction applicables : </label>
<b title="Champ obligatoire">(obligatoire ; sélectionnez tous les taux qui s'appliquent à <b title="Champ obligatoire">(obligatoire ; sélectionnez tous les taux qui s'appliquent à
l'association)</b> l'association)</b>
</dt> </dt>
{foreach from=$plugin_config->reduction key="key" item="taux"} <div id="taux_reduction">
<div> {foreach from=$plugin_config->reduction key="key" item="taux"}
<input type="checkbox" name="tauxReduction[]" id="taux_{$key}" value="{$key}" class="choix" <input type="checkbox" name="tauxReduction[]" id="taux_{$key}" value="{$key}" class="choix"
{if $taux.valeur == 1}checked{/if} /> {if $taux.valeur == 1}checked{/if} />
<label for="taux_{$key}">Taux {$taux.taux}, ligne {$taux.ligne} de la déclaration <label for="taux_{$key}">Taux {$taux.taux}, ligne {$taux.ligne} de la déclaration
{if $taux.remarque !== ""}({$taux.remarque})</label>{/if} {if $taux.remarque !== ""}({$taux.remarque})</label>{/if}
</div> {/foreach}
{/foreach} </div>
</dl> </dl>
</fieldset> </fieldset>
<fieldset> <fieldset>
<legend>Nom, fonction et signature du responsable</legend> <legend>Nom, fonction et signature du responsable</legend>
<dl class="config"> <div id="config_nom_fonction">
<dt><label>Nom</label></dt>
{input type="text" name="nom_responsable" source=$plugin.config label="" maxlength=50}
</dl>
<dl class="config"> {* Nom du responsable *}
<dt><label>Fonction</label></dt> <dl class="config">
{input type="text" name="fonction_responsable" source=$plugin.config label="" maxlength=50} {input type="text" name="nom_responsable" source=$plugin.config label="Nom" help="du responsable" required=true maxlength=50}
</dl> </dl>
{* Fonction du responsable *}
<dl class="config">
{input type="text" name="fonction_responsable" source=$plugin.config label="Fonction" help="du responsable" maxlength=50}
</dl>
{* Ville avant signature *}
<dl class="config">
{input type="text" name="ville_asso" source=$plugin.config label="Ville" help="précède la date sur le formulaire" maxlength=50}
</dl>
</div>
{* Signature *}
<dl class="config"> <dl class="config">
<dt><label>Signature</label></dt> <dt><label>Signature</label></dt>
<p>L'image de la signature doit être d'une taille « raisonnable » et avoir un fond transparent</p> <p>L'image de la signature doit être d'une taille « raisonnable » et avoir un fond transparent</p>
@ -76,13 +88,68 @@
</dl> </dl>
</fieldset> </fieldset>
{* Numérotation des reçus *}
<fieldset>
<legend>Numérotation des reçus</legend>
<details>
<summary class="help block">
Sélectionner les éléments qui doivent faire partie du numéro de reçu
</summary>
<div class="help block">
<ul>
<li>Préfixe : texte qui sera imprimé tel quel au début du numéro (ex : sigle de l'association) ;
facultatif</li>
<li>Année fiscale : numéro de l'année fiscale (ex : 2022) ; facultatif</li>
<ul>
Sélectionner au moins un des deux numéros suivants
<li>Numéro de membre</li>
<li>Numéro séquentiel (1, 2, ...) : numéro d'ordre du reçu</li>
</ul>
<li>Valeur initiale : si vous avez choisi un numéro séquentiel, indiquez le numéro du premier reçu
</li>
</ul>
</div>
</details>
<div id="numero_recus">
{* Préfixe *}
<dl class="config">
{input type="text" name="prefixe" source=$numerotation label="Préfixe" maxlength=20}
</dl>
{* Autres éléments de la numérotation *}
<dl class="config">
{input type="checkbox" name="annee" source=$numerotation label="Année fiscale" value=1}
</dl>
<dl class="config">
{input type="checkbox" name="membre" source=$numerotation label="N° de membre" value=1}
</dl>
<dl class="config">
{input type="checkbox" name="sequentiel" source=$numerotation label="N° séquentiel" value=1}
</dl>
<dl class="config">
{input type="number" name="valeur_init" source=$numerotation label="Valeur initiale"}
</dl>
</div>
</fieldset>
<fieldset> <fieldset>
<legend>Autres informations</legend> <legend>Autres informations</legend>
<dl class="config">
<dt><label>Ville</label></dt> {* Adresse de courriel *}
<p>Précède la date sur le formulaire</p> <div id="courriel">
{input type="text" name="ville_asso" source=$plugin.config label="" maxlength=50} <dl class="config">
</dl> <dt><label>Adresse de courriel</label></dt>
{if $plugin.config.imprimerCourriel}
{input type="checkbox" name="imprimerCourriel" value="1" checked="checked" label="Imprimer" help="Cocher pour imprimer l'adresse de courriel des membres sur les reçus"}
{else}
{input type="checkbox" name="imprimerCourriel" value="1" label="Imprimer" help="Cocher pour imprimer l'adresse de courriel des membres sur les reçus"}
{/if}
</dl>
</div>
{* les champs de nom *} {* les champs de nom *}
<?php $nbChamps = count($champsNom); ?> <?php $nbChamps = count($champsNom); ?>
@ -92,13 +159,14 @@
<div> <div>
{foreach from=$champsNom key="nom" item="champ"} {foreach from=$champsNom key="nom" item="champ"}
<div class="champnom"> <div class="champnom">
<div class="actions"></div> <div class="actions"></div>
<div class="infos"> <div class="infos">
<input type="checkbox" name="champsNom[]" id="champ_{$nom}" value={$nom} class="choix" {if $nbChamps == 1 || $champ.position != 0}checked{/if} /> <input type="checkbox" name="champsNom[]" id="champ_{$nom}" value={$nom} class="choix"
{if $nbChamps == 1 || $champ.position != 0}checked{/if} />
<label for="champ_{$nom}">{$champ.titre}</label> <label for="champ_{$nom}">{$champ.titre}</label>
</div>
</div> </div>
</div>
{/foreach} {/foreach}
</div> </div>
</dl> </dl>
@ -108,28 +176,30 @@
<p class="submit"> <p class="submit">
{csrf_field key="recusfiscaux_config"} {csrf_field key="recusfiscaux_config"}
{button type="submit" name="save" label="Enregistrer" shape="right" class="main"} {button type="submit" name="save" label="Enregistrer" shape="right" class="main" onclick="return verifierConfig(articles_cgi, taux_reduction)"}
</p> </p>
</form> </form>
{literal} {literal}
<script type="text/javascript"> <script type="text/javascript">
(function() { (function() {
var lesDivs = document.querySelectorAll('.actions'); var lesDivs = document.querySelectorAll('.actions');
for (i = 0; i < lesDivs.length; ++i) { for (i = 0; i < lesDivs.length; ++i) {
var up = document.createElement('a'); var up = document.createElement('a');
up.className = 'icn up'; up.className = 'icn up';
up.innerHTML = '&uarr;'; up.innerHTML = '&uarr;';
up.title = 'Déplacer vers le haut'; up.title = 'Déplacer vers le haut';
up.onclick = function(e) { up.onclick = function(e) {
var field = this.parentNode.parentNode; var field = this.parentNode.parentNode;
var p = field.previousSibling; var p = field.previousSibling;
while (p != null && p.nodeType == 3) { p = p.previousSibling; } while (p != null && p.nodeType == 3) { p = p.previousSibling; }
field.parentNode.insertBefore(field, p); field.parentNode.insertBefore(field, p);
return false; return false;
}; };
lesDivs[i].appendChild(up); lesDivs[i].appendChild(up);
} }
}()); }());
</script> </script>
{/literal} {/literal}
{* scripts divers *}
<script src="script.js"></script>

View File

@ -78,7 +78,7 @@
<p class="submit"> <p class="submit">
{csrf_field key="generer_tous_recus"} {csrf_field key="generer_tous_recus"}
{button type="submit" name="generer_tous" label="Poursuivre" shape="right" class="main" onclick="return verifierRadio('menu_versements');" } {button type="submit" name="generer_tous" label="Poursuivre" shape="right" class="main" onclick="return verifierTaux(menu_versements);" }
</p> </p>
</div> </div>
@ -183,7 +183,7 @@
<p class="submit"> <p class="submit">
{csrf_field key="generer_recus_activites"} {csrf_field key="generer_recus_activites"}
{button type="submit" name="generer_activites" label="Poursuivre" shape="right" class="main" onclick="return verifierCases('menu_activites_tarifs');" } {button type="submit" name="generer_activites" label="Poursuivre" shape="right" class="main" onclick="return verifierActivitésTaux(menu_activites_tarifs);" }
</p> </p>
</div> </div>
</form> </form>

View File

@ -90,7 +90,7 @@
<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>
<p id="articles">Articles 200, 238 bis et 978 du code général des impôts</p> <p id="articles">Articles 200, 238 bis et 978 du code général des impôts</p>
<div id="numRecu"> <div id="numRecu">
<p class="important">Reçu numéro {{$annee_recu}}/{{$numero}}</p> <p class="important">Reçu {{$numero}}</p>
</div> </div>
</div> </div>
@ -103,9 +103,14 @@
<div class="cartouche" id="donateur"> <div class="cartouche" id="donateur">
<h3 class="rubrique">Donateur</h3> <h3 class="rubrique">Donateur</h3>
<p>{{$nom}}<br /> <p>
{{$adresse}}<br /> {{$nom}}<br />
{{$code_postal}} {{$ville}}</p> {{$adresse}}<br />
{{$code_postal}} {{$ville}}
{{if $courriel != ""}}
<br />courriel : <a href="mailto:{{$courriel}}">{{$courriel}}</a>
{{/if}}
</p>
</div> </div>
<div class="cartouche" id="versements"> <div class="cartouche" id="versements">

View File

@ -7,28 +7,29 @@ use Garradin\Entities\Files\File;
use Garradin\Plugin\RecusFiscaux\Utils; use Garradin\Plugin\RecusFiscaux\Utils;
$session->requireAccess($session::SECTION_CONFIG, $session::ACCESS_ADMIN); $session->requireAccess($session::SECTION_CONFIG, $session::ACCESS_ADMIN);
$art_sel = f('articlesCGI') ? : []; $art_sel = f('articlesCGI') ?: [];
$taux_sel = f('tauxReduction') ? : []; $taux_sel = f('tauxReduction') ?: [];
$noms_sel = f('champsNom') ? : []; $noms_sel = f('champsNom') ?: [];
// récupérer les champs des noms // récupérer les champs des noms
$champsNom = Utils::getChampsNom($config, $plugin); $champsNom = Utils::getChampsNom($config, $plugin);
if (f('save') && $form->check('recusfiscaux_config')) if (f('save') && $form->check('recusfiscaux_config')) {
{
try { try {
// objet de l'association // objet de l'association
$plugin->setConfig('objet_asso', trim(f('objet_asso'))); if ($plugin->getConfig('objet_asso') != trim(f('objet_asso'))) {
$plugin->setConfig('objet_asso', trim(f('objet_asso')));
}
// articles du CGI // articles du CGI
$confArticles = $plugin->getConfig('articlesCGI'); $confArticles = $plugin->getConfig('articlesCGI');
// effacer l'ancienne configuration // effacer l'ancienne configuration
for ($i = 0; $i < count($confArticles); ++$i) { for ($i = 0; $i < count($confArticles); ++$i) {
$confArticles[$i]->valeur = 0; $confArticles[$i]->valeur = false; // 0
} }
// et copier la nouvelle // et copier la nouvelle
foreach ($art_sel as $article) { foreach ($art_sel as $article) {
$confArticles[$article]->valeur = 1; $confArticles[$article]->valeur = true; // 1
} }
$plugin->setConfig("articlesCGI", $confArticles); $plugin->setConfig("articlesCGI", $confArticles);
@ -36,22 +37,35 @@ if (f('save') && $form->check('recusfiscaux_config'))
$confTaux = $plugin->getConfig('reduction'); $confTaux = $plugin->getConfig('reduction');
// effacer l'ancienne configuration // effacer l'ancienne configuration
for ($i = 0; $i < count($confTaux); ++$i) { for ($i = 0; $i < count($confTaux); ++$i) {
$confTaux[$i]->valeur = 0; $confTaux[$i]->valeur = false; // 0
} }
// et copier la nouvelle // et copier la nouvelle
foreach ($taux_sel as $taux) { foreach ($taux_sel as $taux) {
$confTaux[$taux]->valeur = 1; $confTaux[$taux]->valeur = true; // 1
} }
$plugin->setConfig("reduction", $confTaux); $plugin->setConfig("reduction", $confTaux);
// nom, fonction et signature du responsable // Informations au sujet du responsable
$plugin->setConfig('nom_responsable', trim(f('nom_responsable'))); if ($plugin->getConfig('nom_responsable') != trim(f('nom_responsable'))) {
$plugin->setConfig('fonction_responsable', trim(f('fonction_responsable'))); $plugin->setConfig('nom_responsable', trim(f('nom_responsable')));
if (isset($_SESSION['sig_file']) && count($_SESSION['sig_file']) > 0) }
{ if ($plugin->getConfig('fonction_responsable') != trim(f('fonction_responsable'))) {
$plugin->setConfig('fonction_responsable', trim(f('fonction_responsable')));
}
// ville
if ($plugin->getConfig('ville_asso') != trim(f('ville_asso'))) {
$plugin->setConfig('ville_asso', trim(f('ville_asso')));
}
// 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) {
// supprimer la signature précédente, si besoin // supprimer la signature précédente, si besoin
if (null !== $plugin->getConfig('signature')) if (
{ null !== $plugin->getConfig('signature') &&
$plugin->getConfig('signature') != $_SESSION['sig_file'][0]->path
) {
$sig_file = \Garradin\Files\Files::get($plugin->getConfig('signature')); $sig_file = \Garradin\Files\Files::get($plugin->getConfig('signature'));
if (null !== $sig_file) { if (null !== $sig_file) {
$sig_file->delete(); $sig_file->delete();
@ -62,40 +76,48 @@ if (f('save') && $form->check('recusfiscaux_config'))
} }
// autres informations // autres informations
// ville // numérotation des reçus
$plugin->setConfig('ville_asso', trim(f('ville_asso'))); $configNum = $plugin->getConfig('numerotation');
error_log("configNum=" . print_r($configNum, true));
$formNum = clone $configNum;
if ($configNum->prefixe != trim(f('prefixe'))) {
$formNum->prefixe = trim(f('prefixe'));
}
$formNum->annee = f('annee');
$formNum->membre = f('membre');
$formNum->sequentiel = f('sequentiel');
$formNum->valeur_init = f('valeur_init');
$plugin->setConfig('numerotation', $formNum);
// impression des adresses de courriel
$plugin->setConfig('imprimerCourriel', f('imprimerCourriel'));
// champs pour le nom et prénom // champs pour le nom et prénom
foreach ($champsNom as $nom => $champ) foreach ($champsNom as $nom => $champ) {
{
$champ->position = 0; $champ->position = 0;
} }
$i = -count($noms_sel); $i = -count($noms_sel);
foreach ($noms_sel as $nom) foreach ($noms_sel as $nom) {
{
$champsNom[$nom]->position = $i++; $champsNom[$nom]->position = $i++;
} }
$plugin->setConfig('champsNom', $champsNom); $plugin->setConfig('champsNom', $champsNom);
\Garradin\Utils::redirect(PLUGIN_URL . 'config.php?ok'); \Garradin\Utils::redirect(PLUGIN_URL . 'config.php?ok');
} } catch (UserException $e) {
catch (UserException $e)
{
$form->addError($e->getMessage()); $form->addError($e->getMessage());
} }
} }
// trier les champs de nom pour l'affichage // trier les champs de nom pour l'affichage
uasort($champsNom, function ($a, $b) uasort($champsNom, function ($a, $b) {
{
return $a->position - $b->position; return $a->position - $b->position;
}); });
$tpl->assign('ok', qg('ok') !== null); $tpl->assign('ok', qg('ok') !== null);
$path = qg('path') ?: File::CONTEXT_CONFIG; $path = qg('path') ?: File::CONTEXT_CONFIG;
$tpl->assign('path', $path);
$tpl->assign('default_signature', \Garradin\WWW_URL . "plugin/recusfiscaux/default_signature.png"); $tpl->assign('default_signature', \Garradin\WWW_URL . "plugin/recusfiscaux/default_signature.png");
$tpl->assign('plugin_config', $plugin->getConfig()); $tpl->assign('plugin_config', $plugin->getConfig());
$tpl->assign('champsNom', $champsNom);
$tpl->assign('plugin_css', ['style.css']); $tpl->assign('plugin_css', ['style.css']);
$tpl->assign('numerotation', $plugin->getConfig('numerotation'));
$tpl->assign(compact('path', 'champsNom'));
$tpl->display(PLUGIN_ROOT . '/templates/config.tpl'); $tpl->display(PLUGIN_ROOT . '/templates/config.tpl');

View File

@ -52,6 +52,24 @@ 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'));
$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');
$versementsSelectionnes = array(); $versementsSelectionnes = array();
@ -82,14 +100,33 @@ foreach ($totalPersonnes as $idPersonne => $personne)
$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('ville_asso', $plugin->getConfig('ville_asso'));
$tpl->assign('annee_recu', $_SESSION['annee_recu']);
$tpl->assign('numero', $personne->id);
$tpl->assign('nom', $personne->nomPrenom); $tpl->assign('nom', $personne->nomPrenom);
$tpl->assign('adresse', $personne->adresse); $tpl->assign('adresse', $personne->adresse);
$tpl->assign('code_postal', $personne->codePostal); $tpl->assign('code_postal', $personne->codePostal);
$tpl->assign('ville', $personne->ville); $tpl->assign('ville', $personne->ville);
$tpl->assign('date', date("j/m/Y")); $tpl->assign('date', date("j/m/Y"));
// numéro de reçu
$chaineNum = $prefixeNum;
if (isset($configNum->membre) && $configNum->membre) {
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 // les versements
$tpl->registerSection('versements', $tpl->registerSection('versements',
function () use($personne, $libelles_taux, $fmt) function () use($personne, $libelles_taux, $fmt)
@ -156,10 +193,10 @@ $fichierZip = Utils::makeArchive(
); );
//supprimer les fichiers pdf (utile ?) //supprimer les fichiers pdf (utile ?)
// foreach ($listeFichiersPDF as $f) foreach ($listeFichiersPDF as $f)
// { {
// unlink($f); unlink($f);
// } }
/** /**
* Cumuler les versements de chaque personne * Cumuler les versements de chaque personne

View File

@ -14,7 +14,7 @@ if (! isset($_SESSION['annee_recu']) || $_SESSION['annee_recu'] == "")
$nbTaux = 0; $nbTaux = 0;
foreach ($plugin->getConfig('reduction') as $taux) foreach ($plugin->getConfig('reduction') as $taux)
{ {
if ($taux->valeur == 1) { ++$nbTaux; } if ($taux->valeur) { ++$nbTaux; }
} }
// idem avec les champs nom/prénom // idem avec les champs nom/prénom

View File

@ -1,5 +1,9 @@
"use strict"; "use strict";
// ------------------------------------------------------------------------
// actions sur la liste des versements
// ------------------------------------------------------------------------
/** /**
* Fonction appelée quand on ()coche la case globale * Fonction appelée quand on ()coche la case globale
* ()sélectionner toutes les cases de toutes les activités * ()sélectionner toutes les cases de toutes les activités
@ -128,104 +132,6 @@ function changerMessage(message, idCase)
} }
} }
/**
* fonction appelée lors de la validation du formulaire
* @return vrai si au moins un choix a été fait
* @param {HTMLFormElement} formulaire
*/
function verifierChoix(formulaire)
{
let listeCheck = formulaire.getElementsByTagName("input");
let ok = false;
for (let i = 1; i < listeCheck.length; ++i)
{
if (listeCheck[i].checked)
{
ok = true;
break;
}
}
if (! ok)
{
alert("Erreur : il faut sélectionner au moins un versement");
}
return ok;
}
/**
* positionner l'action déclenchée par l'envoi du formulaire
* afficher et masquer des portions de formulaire selon l'action
* @param {HTMLFormElement} formulaire
* @param {string} action après envoi du formulaire
* @param {any} idElem id de l'élément à afficher
* @param {any} nomClasse classe des éléments à masquer (sauf idElem)
*/
function choixMethodeGeneration(formulaire, action, idElem, nomClasse)
{
formulaire.setAttribute('action', 'action.php?action=' + action);
for (let elem of formulaire.querySelectorAll(nomClasse))
{
if (elem.id == idElem)
{
elem.classList.remove('hidden');
}
else
{
elem.classList.add('hidden');
}
}
}
/**
* vérifier
* - qu'au moins une activité/tarif est sélectionnée
* - qu'un radio de chaque activité/tarif sélectionné a été sélectionné :)
* @param {string} idElem id du conteneur des cases à vérifier
*/
function verifierCases(idElem)
{
let div = document.getElementById(idElem);
let nbChoix = 0;
// parcourir les cases à cocher
for (let idCase of div.querySelectorAll("input[type=checkbox]"))
{
if (idCase.checked) {
++nbChoix;
// vérifier qu'un radio de la même ligne est sélectionné
let ligneCorrecte = false;
// trouver la ligne englobante
let ligne = idCase.closest("li");
for (let idRadio of ligne.querySelectorAll('input[type=radio]'))
{
if (idRadio.checked) { ligneCorrecte = true; break; }
}
if (! ligneCorrecte) {
alert("Erreur : il faut sélectionner un taux de réduction dans chaque ligne cochée");
return false;
}
}
}
if (nbChoix == 0) {
alert("Erreur : il faut sélectionner au moins une ligne");
}
return nbChoix != 0;
}
/**
* vérifier qu'un radio a été sélectionné dans la div paramètre
* @param {string} idElem id du conteneur des radios à vérifier
*/
function verifierRadio(idElem)
{
let div = document.getElementById(idElem);
for (let idRadio of div.querySelectorAll('input[type=radio]'))
{
if (idRadio.checked) { return true; }
}
alert("Erreur : il faut sélectionner un taux de réduction");
return false;
}
/** /**
* afficher/masquer les détails * afficher/masquer les détails
* @param {string} idElem bouton de masquage/affichage * @param {string} idElem bouton de masquage/affichage
@ -258,3 +164,124 @@ function montrerMasquerDetails(idElem, classe, texte)
} }
} }
} }
/**
* fonction appelée lors de la demande de génération des reçus
* vérifier qu'au moins un versement a été sélectionné
* @return vrai si au moins un choix a été fait
* @param {HTMLFormElement} formulaire
*/
function verifierChoix(formulaire)
{
return verifierCases(formulaire, 'checkbox', "au moins un versement");
}
// ------------------------------------------------------------------------
// actions sur la page d'accueil
// ------------------------------------------------------------------------
/**
* positionner l'action déclenchée par l'envoi du formulaire
* afficher et masquer des portions de formulaire selon l'action
* @param {HTMLFormElement} formulaire
* @param {string} action après envoi du formulaire
* @param {any} idElem id de l'élément à afficher
* @param {any} nomClasse classe des éléments à masquer (sauf idElem)
*/
function choixMethodeGeneration(formulaire, action, idElem, nomClasse)
{
formulaire.setAttribute('action', 'action.php?action=' + action);
for (let elem of formulaire.querySelectorAll(nomClasse))
{
if (elem.id == idElem)
{
elem.classList.remove('hidden');
}
else
{
elem.classList.add('hidden');
}
}
}
/**
* vérifier
* - qu'au moins une activité/tarif est sélectionnée
* - qu'un radio de chaque activité/tarif sélectionné a été sélectionné :)
* @param conteneur des cases à vérifier
*/
function verifierActivitésTaux(conteneur)
{
let nbChoix = 0;
// parcourir les cases à cocher
for (let idCase of conteneur.querySelectorAll("input[type=checkbox]"))
{
if (idCase.checked) {
++nbChoix;
// vérifier qu'un radio de la même ligne est sélectionné
let ligneCorrecte = false;
// trouver la ligne englobante
let ligne = idCase.closest("li");
for (let idRadio of ligne.querySelectorAll('input[type=radio]'))
{
if (idRadio.checked) { ligneCorrecte = true; break; }
}
if (! ligneCorrecte) {
alert("Erreur : il faut sélectionner un taux de réduction dans chaque ligne cochée");
return false;
}
}
}
if (nbChoix == 0) {
alert("Erreur : il faut sélectionner au moins une ligne");
}
return nbChoix != 0;
}
/**
* vérifier qu'un taux a été sélectionné dans le conteneur paramètre
*/
function verifierTaux(conteneur)
{
return verifierCases(conteneur, 'radio', "un taux de réduction");
}
// ------------------------------------------------------------------------
// actions sur la config
// ------------------------------------------------------------------------
/**
* vérifier les données saisies dans le formulaire de configuration
*/
function verifierConfig(divArticles, divTauxReduc)
{
// articles
if (! verifierCases(divArticles, "checkbox", "au moins un article")) { return false; }
// taux de réduction
if (! verifierCases(divTauxReduc, "checkbox", "au moins un taux de réduction")) { return false; }
// Nom, fonction, signature
// alert("Erreur : il faut sélectionner au moins un versement");
return true;
}
/**
* Vérifier qu'au moins une case est cochée dans le conteneur
* @param conteneur
* @param type de case à vérifier (radio, checkbox)
* @param message à afficher si erreur
*/
function verifierCases(conteneur, type, message)
{
let selecteur = "input[type=" + type + "]";
let listeCheck = conteneur.querySelectorAll(selecteur);
for (let elem of listeCheck)
{
if (elem.checked) { return true; }
}
alert("Erreur : il faut sélectionner " + message);
return false;
}

View File

@ -1,7 +1,11 @@
/* liste des versements */ /*
* liste des versements
*/
div.pair { div.pair {
background-color: rgba(var(--gSecondColor), 0.15); background-color: rgba(var(--gSecondColor), 0.15);
} }
fieldset.versements fieldset.versements
{ {
margin-bottom : 0; margin-bottom : 0;
@ -9,84 +13,128 @@ fieldset.versements
-webkit-border-radius:8px; -webkit-border-radius:8px;
border-radius:8px; border-radius:8px;
} }
div span { div span {
padding-left : 0.5em; padding-left : 0.5em;
padding-right : 0.5em; padding-right : 0.5em;
} }
td.montant { td.montant {
text-align : right; text-align : right;
} }
span.montant { span.montant {
width : 5em; width : 5em;
text-align : right; text-align : right;
} }
span.total span.total
{ {
font-weight : bold; font-weight : bold;
} }
summary.activite summary.activite
{ {
margin-bottom : 0.5em; margin-bottom : 0.5em;
} }
summary.personne summary.personne
{ {
margin-bottom : 0.5em; margin-bottom : 0.5em;
padding-top : 0; padding-top : 0;
padding-bottom : 0; padding-bottom : 0;
} }
div.activite div.activite
{ {
background-color: rgba(var(--gSecondColor), 0.3); background-color: rgba(var(--gSecondColor), 0.3);
} }
div.personne div.personne
{ {
font-weight : normal; font-weight : normal;
background-color: rgba(var(--gSecondColor), 0.25); background-color: rgba(var(--gSecondColor), 0.25);
} }
h3.activite h3.activite
{ {
display : inline; display : inline;
} }
p.activite p.activite
{ {
margin-left : 2.5em; margin-left : 2.5em;
} }
#signature
{
padding : 1em 0.5em 0 0.5em;
max-width: 300px;
max-height: 150px;
}
div.explications ul
{
list-style : initial;
}
div.actions
{
display : inline;
}
input.check_global input.check_global
{ {
margin : 0.2em 0.5em; margin : 0.2em 0.5em;
} }
dl#menu
{
min-width : 40em;
width : 50%;
}
div.versements div.versements
{ {
margin-left : 4em; margin-left : 4em;
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
} }
/* config */
/*
* page d'accueil
*/
div.explications ul
{
list-style : initial;
}
dl#menu
{
min-width : 40em;
width : 50%;
}
/*
* configuration
*/
#signature
{
padding : 1em 0.5em 0 0.5em;
max-width: 300px;
max-height: 120px;
}
div.actions
{
display : inline;
}
dl.config dl.config
{ {
padding : 1ex 0; padding-bottom : 1ex;
padding-right : 1em;
} }
div#articles_cgi, div#config_nom_fonction, div#numero_recus
{
display: flex;
}
div.article
{
margin-right : 3em;
}
/*
div#config_nom_fonction
{
display: flex;
}
div#numero_recus
{
display:flex;
}
*/
div.champnom div.champnom
{ {
display : flex; display : flex;
@ -110,3 +158,12 @@ ul.reduction span.radio-btn
margin-left : 2em; margin-left : 2em;
border-spacing : 0.1em; border-spacing : 0.1em;
} }
input#f_prefixe
{
width : 8em;
min-width : 8em;
}
input#f_valeur_init
{
max-width: 4em;
}