Merge branch 'evolution'

This commit is contained in:
Jean-Christophe Engel 2023-11-20 14:03:44 +01:00
commit 23743cda0f
12 changed files with 468 additions and 197 deletions

132
_calcul_dispo.html Normal file
View File

@ -0,0 +1,132 @@
{{*
Calcul des entrées/sorties de matériels à une date donnée
paramètres :
- liste de catégories
- date
*}}
{{* liste des catégories *}}
{{if $categories === null}}
{{#select
key
FROM module_data_equipment as cat
WHERE json_extract(cat.document, '$.type') = 'category'}}
{{:assign var="categories." value=$key}}
{{/select}}
{{/if}}
{{* date *}}
{{if $date === null}}
{{:assign date=$now|date:"Y-m-d"}}
{{/if}}
{{* Extraire et compiler les infos de la base *}}
{{#select
json_extract(mvt.document, '$.date') as 'date',
eqpmt.key as 'eqpmt_key',
json_extract(eqpmt.document, '$.designation') as 'designation',
cat.key as 'cat_key',
json_extract(cat.document, '$.name') as 'category',
mvt.key as 'mvt_key',
json_extract(mvt.document, '$.direction') as 'direction',
CASE json_extract(mvt.document, '$.direction')
WHEN 'entrée' THEN json_extract(mvt.document, '$.inputNature')
WHEN 'sortie' THEN json_extract(mvt.document, '$.outputNature')
END
AS 'operation',
json_extract(mvt.document, '$.amount') AS 'nombre'
FROM module_data_equipment as eqpmt
INNER JOIN module_data_equipment as cat
ON json_extract(eqpmt.document, '$.category') = cat.key
INNER JOIN module_data_equipment as mvt
ON json_extract(mvt.document, '$.equipment') = eqpmt.key
WHERE json_extract(eqpmt.document, '$.type') = "equipment"
AND !categories
AND json_extract(mvt.document, '$.date') <= :date
ORDER BY
json_extract(cat.document, '$.name'),
json_extract(eqpmt.document, '$.designation'),
json_extract(mvt.document, '$.date');
!categories='cat_key'|sql_where:'IN':$categories
:date=$date
}}
{{:assign
var="equipments.%s.name"|args:$cat_key
value=$category}}
{{:assign
var="equipments.%s.eqpmt.%s.designation"|args:$cat_key:$eqpmt_key
value=$designation}}
{{* matériel propriété de l'asso en stock *}}
{{:assign
var="stock"
from="equipments.%s.eqpmt.%s.stock"|args:$cat_key:$eqpmt_key}}
{{if $stock === null}}
{{:assign stock=0}}
{{/if}}
{{* matériel propriété de l'asso à l'extérieur *}}
{{:assign
var="exterieur"
from="equipments.%s.eqpmt.%s.exterieur"|args:$cat_key:$eqpmt_key}}
{{if $exterieur === null}}
{{:assign exterieur=0}}
{{/if}}
{{* matériel non propriété de l'asso *}}
{{:assign
var="nonproprio"
from="equipments.%s.eqpmt.%s.nonproprio"|args:$cat_key:$eqpmt_key}}
{{if $nonproprio === null}}
{{:assign nonproprio=0}}
{{/if}}
{{if $direction === 'entrée'}}
{{* chercher le type d'entrée parmi les types de la config *}}
{{#foreach from=$module.config.inputNature item="elem"}}
{{if $operation == $elem.label}}
{{if $elem.type == 'définitif'}}
{{:assign stock="%d+%d"|math:$stock:$nombre}}
{{:assign
var="equipments.%s.eqpmt.%s.stock"|args:$cat_key:$eqpmt_key
from=stock}}
{{elseif $elem.type == 'retour'}}
{{:assign exterieur="%d-%d"|math:$exterieur:$nombre}}
{{:assign
var="equipments.%s.eqpmt.%s.exterieur"|args:$cat_key:$eqpmt_key
from=exterieur}}
{{elseif $elem.type == 'temporaire'}}
{{:assign nonproprio="%d+%d"|math:$nonproprio:$nombre}}
{{:assign
var="equipments.%s.eqpmt.%s.nonproprio"|args:$cat_key:$eqpmt_key
from=nonproprio}}
{{/if}}
{{:break}}
{{/if}}
{{/foreach}}
{{elseif $direction === 'sortie'}}
{{* chercher le type de sortie parmi les types de la config *}}
{{#foreach from=$module.config.outputNature item="elem"}}
{{if $operation == $elem.label}}
{{if $elem.type == 'définitif'}}
{{:assign stock="%d-%d"|math:$stock:$nombre}}
{{:assign
var="equipments.%s.eqpmt.%s.stock"|args:$cat_key:$eqpmt_key
from=stock}}
{{elseif $elem.type == 'temporaire'}}
{{:assign exterieur="%d+%d"|math:$exterieur:$nombre}}
{{:assign
var="equipments.%s.eqpmt.%s.exterieur"|args:$cat_key:$eqpmt_key
from=exterieur}}
{{elseif $elem.type == 'retour'}}
{{:assign nonproprio="%d-%d"|math:$nonproprio:$nombre}}
{{:assign
var="equipments.%s.eqpmt.%s.nonproprio"|args:$cat_key:$eqpmt_key
from=nonproprio}}
{{/if}}
{{:break}}
{{/if}}
{{/foreach}}
{{/if}}
{{/select}}

View File

@ -15,44 +15,7 @@
</header>
{{/if}}
{{* afficher les matériels de la catégorie sélectionnée avec le stock, les entrées et les sorties *}}
<section class="categories">
{{#list
select="$$.designation AS 'Désignation'; '' AS 'Stock' ; '' AS 'Sortie' ; '' AS 'Disponible'"
type="equipment"
category=$_GET.key
order=1}}
{{:assign equipment_key=$key}}
{{:assign var="stock" value=0}}
{{:assign var="sortie" value=0}}
{{* récupérer les mouvements du matériel courant *}}
{{#load
type="movement"
equipment=$equipment_key
assign="movement"}}
{{if $movement.direction === 'entrée'}}
{{:assign stock="%d+%d"|math:$stock:$movement.amount}}
{{else}}
{{:assign sortie="%d+%d"|math:$sortie:$movement.amount}}
{{/if}}
{{/load}}
{{:assign dispo="%d-%d"|math:$stock:$sortie}}
<tr>
<td>{{$designation}}</td>
<td>{{$stock}}</td>
<td>{{$sortie}}</td>
<td>{{$dispo}}</td>
<td></td>
</tr>
{{else}}
<p class="block alert">Il n'y a aucun matériel dans cette catégorie.</p>
{{/list}}
</section>
{{:assign var="categories." value=$_GET.key}}
{{:include file="/%s/inventaire.html"|args:$module.name}}
{{:admin_footer}}

View File

@ -11,34 +11,29 @@
{{* Traiter l'envoi du formulaire *}}
{{#form on="save"}}
{{#foreach from=$_POST.input_fields|array_transpose item="field"}}
{{:assign var="input_fields." label=$field.label|trim owner=$field.owner|boolval}}
{{/foreach}}
{{#foreach from=$_POST.output_fields|array_transpose item="field"}}
{{:assign var="output_fields." label=$field.label|trim final=$field.final|boolval}}
{{/foreach}}
{{:save key="config"
validate_schema="./config.schema.json"
inputNature=$input_fields
outputNature=$output_fields
inputNature=$_POST.input_fields|array_transpose
outputNature=$_POST.output_fields|array_transpose
}}
{{:redirect to="./config.html?ok=1"}}
{{/form}}
{{* types d'entrées par défaut *}}
{{:assign var='types_entrees.Achat' value=1}}
{{:assign var='types_entrees.Don' value=1}}
{{:assign var='types_entrees.Récupération' value=1}}
{{:assign var='types_entrees.Location' value=0}}
{{:assign var='types_entrees.Emprunt' value=0}}
{{* types d'entrées *}}
{{:assign var='types_entrees' définitif="définitif" temporaire="temporaire" retour="retour"}}
{{* types de sorties par défaut *}}
{{:assign var='types_sorties.Vente' value=1}}
{{:assign var='types_sorties.Don' value=1}}
{{:assign var='types_sorties.Location' value=0}}
{{:assign var='types_sorties.Prêt' value=0}}
{{* entrées par défaut *}}
{{:assign var='entrees_defaut.Achat' value='définitif'}}
{{:assign var='entrees_defaut.Location' value='temporaire'}}
{{:assign var='entrees_defaut.Retour de Location/Prêt' value='retour'}}
{{* types de sorties *}}
{{:assign var='types_sorties' définitif="définitif" temporaire="temporaire" retour="retour"}}
{{* sorties par défaut *}}
{{:assign var='sorties_defaut.Vente' value='définitif'}}
{{:assign var='sorties_defaut.Prêt' value='temporaire'}}
{{:assign var='sorties_defaut.Retour de Location/Prêt' value='retour'}}
<form method="post" action="">
<fieldset>
@ -46,8 +41,8 @@
<table class="list input_fields">
<thead>
<tr>
<th>Type d'entrée</th>
<th>Propriétaire ? <br/><span class="help" style="font-weight:normal">Cocher si l'association devient propriétaire du matériel</span></th>
<th>Nature de l'entrée</th>
<th>Type</th>
<th>Action</th>
</tr>
</thead>
@ -55,28 +50,16 @@
{{#foreach from=$module.config.inputNature item="input_field"}}
<tr>
<td>{{:input type="text" name="input_fields[label][]" default=$input_field.label}}</td>
<td>
<label>
{{:input type="hidden" name="input_fields[owner][]" value=0}}
{{:input type="checkbox" name="input_fields[owner][]" value=1 default=$input_field.owner}}
Proprio ?
</label>
</td>
<td>{{:input type="select" name="input_fields[type][]" options=$types_entrees default=$input_field.type required=true default_empty="— Choisir un type —"}}</td>
<td class="action">
{{:button label="Retirer" shape="minus" onclick="this.parentNode.parentNode.remove();"}}
</td>
</tr>
{{else}}
{{#foreach from=$types_entrees key="label" item="value"}}
{{#foreach from=$entrees_defaut key="label" item="value"}}
<tr>
<td>{{:input type="text" name="input_fields[label][]" default=$label}}</td>
<td>
<label>
{{:input type="hidden" name="input_fields[owner][]" value=0}}
{{:input type="checkbox" name="input_fields[owner][]" value=1 default=$value}}
Proprio ?
</label>
</td>
<td>{{:input type="select" name="input_fields[type][]" options=$types_entrees required=true default=$value default_empty="— Choisir un type —"}}</td>
<td class="action">
{{:button label="Retirer" shape="minus" onclick="this.parentNode.parentNode.remove();"}}
</td>
@ -90,6 +73,14 @@
label="Ajouter un champ"
onclick="addLine('#input_body')"}}
</p>
<div class="help">
<h3>Signification du type d'entrée</h3>
<ul>
<li><b>définitif</b> : l'asso devient propriétaire du matériel (ex : achat, don)</li>
<li><b>temporaire</b> : l'asso ne devient <strong>pas</strong> propriétaire du matériel (ex : location, emprunt)</li>
<li><b>retour</b> : matériel qui revient après une sortie temporaire (ex : retour de location ou de prêt)</li>
</ul>
</div>
</fieldset>
<fieldset>
@ -97,8 +88,8 @@
<table class="list output_fields">
<thead>
<tr>
<th>Type de sortie</th>
<th>Définitif ? <br/><span class="help" style="font-weight:normal">Cocher si la sortie est définitive</span></th>
<th>Nature de la sortie</th>
<th>Type</th>
<th>Action</th>
</tr>
</thead>
@ -106,28 +97,16 @@
{{#foreach from=$module.config.outputNature item="output_field"}}
<tr>
<td>{{:input type="text" name="output_fields[label][]" default=$output_field.label}}</td>
<td>
<label>
{{:input type="hidden" name="output_fields[final][]" value=0}}
{{:input type="checkbox" name="output_fields[final][]" value=1 default=$output_field.final}}
Définitif ?
</label>
</td>
<td>{{:input type="select" name="output_fields[type][]" options=$types_sorties default=$output_field.type required=true default_empty="— Choisir un type —"}}</td>
<td class="action">
{{:button label="Retirer" shape="minus" onclick="this.parentNode.parentNode.remove();"}}
</td>
</tr>
{{else}}
{{#foreach from=$types_sorties key="label" item="value"}}
{{#foreach from=$sorties_defaut key="label" item="value"}}
<tr>
<td>{{:input type="text" name="output_fields[label][]" default=$label}}</td>
<td>
<label>
{{:input type="hidden" name="output_fields[final][]" value=0}}
{{:input type="checkbox" name="output_fields[final][]" value=1 default=$value}}
Définitif ?
</label>
</td>
<td>{{:input type="select" name="output_fields[type][]" options=$types_sorties required=true default=$value default_empty="— Choisir un type —"}}</td>
<td class="action">
{{:button label="Retirer" shape="minus" onclick="this.parentNode.parentNode.remove();"}}
</td>
@ -141,6 +120,14 @@
label="Ajouter un champ"
onclick="addLine('#output_body')"}}
</p>
<div class="help">
<h3>Signification du type de sortie</h3>
<ul>
<li><b>définitif</b> : le matériel n'appartient plus à l'asso (ex : vente, casse, perte, vol, ...)</li>
<li><b>temporaire</b> : le matériel sort temporairement de l'asso qui en reste propriétaire (ex : location, prêt)</li>
<li><b>retour</b> : le matériel <strong>non propriété de l'asso</strong> est rendu à son propriétaire (ex : retour de location ou d'emprunt)</li>
</dl>
</div>
</fieldset>
<p class="submit">
@ -150,7 +137,6 @@
label="Enregistrer"
shape="right"
class="main"
onclick="return desactiverCache('#input_body', '#output_body');"
}}
</p>
</form>
@ -162,26 +148,12 @@
var nelle = ligne.cloneNode(true);
let text = nelle.querySelector('input[type="text"]');
text.value = null;
let cb = nelle.querySelector('input[type="checkbox"]')
if (cb != null) {
cb.checked = false;
let menu = nelle.querySelector('select')
if (menu != null) {
menu[0].selected = 'selected';
}
ligne.parentNode.appendChild(nelle);
text.focus();
}
// Désactiver les champs caché des checkbox cochés
function desactiverCache(...id_body) {
for (const id of id_body) {
for (let ligne of document.querySelector(id).children) {
let idCase = ligne.querySelector("input[type=checkbox]");
let cache = ligne.querySelector("input[type=hidden]"); // le champ caché
if (idCase.checked) {
cache.disabled = true;
}
}
}
}
</script>
{{:admin_footer}}

View File

@ -11,8 +11,9 @@
"label" : {
"type" : "string"
},
"owner" : {
"type" : "boolean"
"type" : {
"type" : "string",
"enum" : ["définitif", "temporaire", "retour"]
}
}
}
@ -26,8 +27,9 @@
"label" : {
"type" : "string"
},
"final" : {
"type" : "boolean"
"type" : {
"type" : "string",
"enum" : ["définitif", "temporaire", "retour"]
}
}
}

View File

@ -7,6 +7,7 @@
"enum": ["equipment"]
},
"category" : {
"description": "Clé de la catégorie",
"type": "string"
},
"designation": {

View File

@ -3,53 +3,6 @@
{{* barre de navigation *}}
{{:include file="/%s/_nav.html"|args:$module.name current="index"}}
<section class="inventaire">
{{* itérer sur les catégories *}}
{{#load type="category" assign="category" order="$$.name"}}
{{:assign key_cat=$key}}
<h3>{{$category.name}}</h3>
{{* itérer sur les matériels de la catégorie courante *}}
{{#list
select="$$.designation AS 'Désignation'; '' AS 'Stock' ; '' AS 'Sortie' ; '' AS 'Disponible'"
type="equipment"
category=$key_cat
order=1
}}
{{:assign equipment_key=$key}}
{{:assign var="stock" value=0}}
{{:assign var="sortie" value=0}}
{{* itérer sur les mouvements du matériel courant *}}
{{#load type="movement" equipment=$equipment_key assign="movement"}}
{{if $movement.direction === 'entrée'}}
{{:assign stock="%d+%d"|math:$stock:$movement.amount}}
{{else}}
{{:assign sortie="%d+%d"|math:$sortie:$movement.amount}}
{{/if}}
{{/load}}
{{:assign dispo="%d-%d"|math:$stock:$sortie}}
<tr>
<th>{{$designation}}</th>
<td>{{$stock}}</td>
<td>{{$sortie}}</td>
<td>{{$dispo}}</td>
<td class="actions">
{{:linkbutton label="Historique des mouvements" href="historique.html?key=%s"|args:$equipment_key shape="table" target="_dialog"}}
</td>
</tr>
{{else}}
<p class="block alert">Il n'y a aucun matériel dans cette catégorie.</p>
{{/list}}
{{else}}
<p class="block alert">Il n'y a aucune catégorie => il faut en créer au moins une.</p>
{{/load}}
</section>
{{:include file="./inventaire.html" print_cat_name=true}}
{{:admin_footer}}

135
inventaire.html Normal file
View File

@ -0,0 +1,135 @@
{{* inventaire des entrées/sorties des matériels des catégories de la variable $categories *}}
{{* Extraire et compiler les infos de la base *}}
{{:include file="./_calcul_dispo.html" keep="equipments"}}
{{* Afficher les résultats *}}
{{if $equipments === null}}
{{if $categories|count == 1}}
<p class="block alert">Il n'y a aucun matériel dans cette catégorie.</p>
{{else}}
<p class="block alert">Il n'y a aucun matériel dans ces catégories.</p>
{{/if}}
{{else}}
<section class="inventaire">
<h2 class="ruler">Matériels dont l'association est propriétaire</h2>
{{* itérer sur les catégories *}}
{{#foreach from=$equipments key="cat_key" item="category"}}
{{* vérifier s'il y a des matériels dans cette catégorie *}}
{{:assign present=false}}
{{#foreach from=$category.eqpmt key="eqpmt_key" item="eqpmt"}}
{{if $eqpmt.stock !== null && $eqpmt.stock !== 0}}
{{:assign present=true}}
{{:break}}
{{/if}}
{{/foreach}}
{{if $print_cat_name}}
<h3>{{$category.name}}</h3>
{{/if}}
{{if $present}}
<table class="list">
<thead>
<tr>
<th>Désignation</th>
<th>Stock</th>
<th>Sortie</th>
<th>Disponible</th>
<th class="actions"></th>
</tr>
</thead>
<tbody>
{{:assign base_path="/m/%s"|args:$module.name}}
{{* itérer sur les matériels de la catégorie *}}
{{#foreach from=$category.eqpmt key="eqpmt_key" item="eqpmt"}}
{{:assign keys=$eqpmt|keys}}
{{if "stock"|in:$keys && $eqpmt.stock !== 0}}
{{:assign dispo="%d-%d"|math:$eqpmt.stock:$eqpmt.exterieur}}
<tr>
<td>{{$eqpmt.designation}}</td>
<td>{{$eqpmt.stock}}</td>
<td>{{if $eqpmt.exterieur == null}}0{{else}}{{$eqpmt.exterieur}}{{/if}}</td>
<td>{{$dispo}}</td>
<td class="actions">
{{:linkbutton
label="Historique des mouvements"
href="%s/historique.html?key=%s"|args:$base_path:$eqpmt_key
shape="table"
target="_dialog"}}
</td>
</tr>
{{/if}}
{{else}}
<p class="block alert">Il n'y a aucun matériel dans cette catégorie.</p>
{{/foreach}}
</tbody>
</table>
{{else}}
<p class="block alert">Il n'y a aucun matériel dans cette catégorie.</p>
{{/if}}
{{/foreach}}
<h2 class="ruler">Matériels dont l'association n'est pas propriétaire</h2>
{{* itérer sur les catégories *}}
{{#foreach from=$equipments key="cat_key" item="category"}}
{{* vérifier s'il y a des matériels dans cette catégorie *}}
{{:assign present=false}}
{{#foreach from=$category.eqpmt key="eqpmt_key" item="eqpmt"}}
{{if $eqpmt.nonproprio !== null && $eqpmt.nonproprio !== 0}}
{{:assign present=true}}
{{:break}}
{{/if}}
{{/foreach}}
{{if $print_cat_name}}
<h3>{{$category.name}}</h3>
{{/if}}
{{if $present}}
<table class="list">
<thead>
<tr>
<th>Désignation</th>
<th>Quantité</th>
<th class="actions"></th>
</tr>
</thead>
<tbody>
{{* itérer sur les matériels de la catégorie *}}
{{#foreach from=$category.eqpmt key="eqpmt_key" item="eqpmt"}}
{{:assign keys=$eqpmt|keys}}
{{if "nonproprio"|in:$keys && $eqpmt.nonproprio !== 0}}
{{:assign dispo="%d-%d"|math:$eqpmt.stock:$eqpmt.exterieur}}
<tr>
<td>{{$eqpmt.designation}}</td>
<td>{{$eqpmt.nonproprio}}</td>
<td class="actions">
{{:linkbutton
label="Historique des mouvements"
href="%s/historique.html?key=%s"|args:$base_path:$eqpmt_key
shape="table"
target="_dialog"}}
</td>
</tr>
{{/if}}
{{else}}
<p class="block alert">Il n'y a aucun matériel dans cette catégorie.</p>
{{/foreach}}
</tbody>
</table>
{{else}}
<p class="block alert">Il n'y a aucun matériel dans cette catégorie.</p>
{{/if}}
{{/foreach}}
</section>
{{/if}}

View File

@ -25,6 +25,7 @@
"minimum": 1
},
"equipment": {
"description": "Clé du matériel",
"type": "string"
},
"date" : {

View File

@ -1,9 +0,0 @@
{{* cumuler les entrées et les sorties de ce matériel *}}
{{:assign var="dispo" value=0}}
{{#load type="movement" where="$$.equipment = :eqpmt_key" :eqpmt_key=$equipment.key assign="movement"}}
{{if $movement.direction === 'entrée'}}
{{:assign dispo="%d+%d"|math:$dispo:$movement.amount}}
{{else}}
{{:assign dispo="%d-%d"|math:$dispo:$movement.amount}}
{{/if}}
{{/load}}

View File

@ -0,0 +1,112 @@
{{#restrict section="config" level="admin" block=true}}{{/restrict}}
{{:admin_header title="Gestion des matériels" current="module_equipment"}}
{{if $_GET.dialog === null}}
{{* barre de navigation *}}
{{:include file="/%s/_nav.html"|args:$module.name current="sorties"}}
{{else}}
<header class="header">
<h1>Sortie de matériel</h1>
{{/if}}
{{* types de sorties *}}
{{#foreach from=$module.config.outputNature item="elem"}}
{{if $elem.type == 'retour'}}
{{:assign var='types_sorties.' value="%s"|args:$elem.label}}
{{/if}}
{{/foreach}}
{{* Traiter l'envoi du formulaire *}}
{{#form on="save"}}
{{* récupérer les infos du matériel *}}
{{#load type="equipment" key=$_POST.equipment assign="equipment"}}
{{:assign designation=$equipment.designation}}
{{:assign var="categories." value=$equipment.category}}
{{/load}}
{{:assign date=$_POST.date|parse_date}}
{{* Extraire et compiler les infos de la base *}}
{{:include file="/%s/_calcul_dispo.html"|args:$module.name keep="equipments"}}
{{* déterminer la quantité présente de ce matériel à la date donnée *}}
{{:assign var=present from="equipments.%s.eqpmt.%s.nonproprio"|args:$equipment.category:$_POST.equipment}}
{{if $_POST.quantite|intval > $present}}
{{:error message="Erreur : la quantité indiquée (%s) est supérieure à celle présente (%d) à la date du %s"|args:$_POST.quantite:$present:$_POST.date}}
{{/if}}
{{* Enregistrer le mouvement *}}
{{:assign mvt_key=""|uuid}}
{{:assign var="operation" from="types_sorties.%d|args:$_POST.type_operation}}
{{:save
key=$mvt_key
validate_schema="/%s/mouvements/movement.schema.json"|args:$module.name
type="movement"
direction="sortie"
outputNature=$operation
amount=$_POST.quantite|intval
equipment=$equipment.key
date=$_POST.date|parse_date
comment=$_POST.remarques|trim
}}
{{:redirect force="index.html?ok=1&msg=Sortie de « %s » (%d) enregistrée"|args:$designation:$_POST.quantite}}
{{else}}
{{:form_errors}}
{{/form}}
{{* Extraire et compiler les infos de la base *}}
{{:include file="/%s/_calcul_dispo.html"|args:$module.name keep="equipments"}}
{{* calculer les quantité de matériels dont l'asso n'est pas propriétaire *}}
{{#foreach from=$equipments key="cat_key" item="category"}}
{{#foreach from=$category.eqpmt key="eqpmt_key" item="eqpmt"}}
{{:assign quantite="%d-%d"|math:$eqpmt.nonproprio:$eqpmt.retour}}
{{if $quantite != 0}}
{{:assign
var="temporaire.%s.%s"|args:$category.name:$eqpmt_key
value="%s (quantité : %d)"|args:$eqpmt.designation:$quantite
}}
{{/if}}
{{/foreach}}
{{/foreach}}
{{if $temporaire === null}}
{{:error message="Le matériel choisi (%s) n'est pas présent à la date du %s"|args:$designation:$_POST.date}}
{{/if}}
{{* formulaire de sortie de matériel *}}
<form method="post" action="">
<fieldset class="sortie">
<legend>Ajouter une sortie d'un matériel disponible en stock</legend>
{{:input type="select" name="type_operation" label="Type" required=true options=$types_sorties}}
{{:input type="date" name="date" label="Date de sortie" required=true default=$now|date_short}}
{{:input type="number" name="quantite" label="Quantité" required=true default=1 min=1}}
<fieldset>
<legend>Matériel</legend>
<p><span class="alert">La quantité disponible est celle à la date du jour</span></p>
{{:input type="select_groups" name="equipment" label="Matériel" required=true options=$temporaire onchange="fixerValeurMax('f_equipment', 'f_quantite')"}}
{{:input type="textarea" name="remarques" label="Remarques" cols="40" rows="3" required=false}}
</fieldset>
</fieldset>
<p class="submit">
{{:button type="submit" name="save" label="Enregistrer" shape="right" class="main"}}
</p>
</form>
<script type="text/javascript">
// fixer la valeur maximale du champ de saisie de la quantité
function fixerValeurMax(idSelect, idNumber) {
let option = document.getElementById(idSelect).selectedOptions[0];
let max = parseInt(option.label.match(/\(quantité : ([0-9]+)\)/)[1]);
let val = parseInt(document.getElementById(idNumber).value);
document.getElementById(idNumber).setAttribute("max", max);
if (val > max) {
document.getElementById(idNumber).value = max;
}
}
</script>
{{:admin_footer}}

View File

@ -12,7 +12,7 @@
<fieldset class="sortie">
<legend>Ajouter une sortie pour du </legend>
{{:linkbutton label="Matériel en stock disponible" shape="plus" href="stock_disponible.html?dialog" target="_dialog"}}
{{:linkbutton label="Matériel emprunté" shape="plus" href="emprunte.html}}
{{:linkbutton label="Matériel emprunté" shape="plus" href="emprunte.html?dialog" target="_dialog"}}
</fieldset>
</form>

View File

@ -11,27 +11,36 @@
{{* types de sorties *}}
{{#foreach from=$module.config.outputNature item="elem"}}
{{if $elem.type != 'retour'}}
{{:assign var='types_sorties.' value="%s"|args:$elem.label}}
{{/if}}
{{/foreach}}
{{* Traiter l'envoi du formulaire *}}
{{#form on="save"}}
{{* récupérer les infos du matériel *}}
{{#load type="equipment" key=$_POST.equipment assign="equipment"}}
{{:assign designation=$equipment.designation}}
{{:assign var="categories." value=$equipment.category}}
{{/load}}
{{:assign date=$_POST.date|parse_date}}
{{* TODO vérifier si qté sortie <= dispo *}}
{{:include file="./_calcul_dispo.html" keep="dispo" }}
{{* Extraire et compiler les infos de la base *}}
{{:include file="/%s/_calcul_dispo.html"|args:$module.name keep="equipments"}}
{{* déterminer la quantité disponible de ce matériel à la date donnée *}}
{{:assign var=stock from="equipments.%s.eqpmt.%s.stock"|args:$equipment.category:$_POST.equipment}}
{{:assign var=exterieur from="equipments.%s.eqpmt.%s.exterieur"|args:$equipment.category:$_POST.equipment}}
{{:assign dispo="%d-%d"|math:$stock:$exterieur}}
{{* cumuler les entrées et les sorties de ce matériel *}}
{{if $_POST.quantite|intval > $dispo}}
{{:error message="Erreur : la quantité indiquée (%s) est supérieure à celle disponible (%d)"|args:$_POST.quantite:$dispo}}
{{:error message="Erreur : la quantité indiquée (%s) est supérieure à celle disponible (%d) à la date du %s"|args:$_POST.quantite:$dispo:$_POST.date}}
{{/if}}
{{* Enregistrer le mouvement *}}
{{:assign mvt_key=""|uuid}}
{{:assign var="operation" from="types_sorties.%d|args:$_POST.type_operation}}
{{:save
key=$mvt_key
validate_schema="/%s/mouvements/movement.schema.json"|args:$module.name
@ -43,29 +52,29 @@
date=$_POST.date|parse_date
comment=$_POST.remarques|trim
}}
{{:redirect force="index.html?ok=1&msg=Sortie de « %s » enregistrée"|args:$equipment.designation}}
{{:redirect force="index.html?ok=1&msg=Sortie de « %s » (%d) enregistrée"|args:$designation:$_POST.quantite}}
{{else}}
{{:form_errors}}
{{/form}}
{{* lister les catégories et matériels associés *}}
{{#load type="category" assign="category" order="$$.name"}}
{{:assign nom_cat=$category.name}}
{{* Extraire et compiler les infos de la base *}}
{{:include file="/%s/_calcul_dispo.html"|args:$module.name keep="equipments"}}
{{#load type="equipment" where="$$.category = :key" :key=$category.key assign="equipment" order="$$.designation"}}
{{* cumuler les entrées et les sorties de ce matériel *}}
{{:include file="./_calcul_dispo.html" keep="dispo" }}
{{if $dispo > 0}}
{{:assign var="equipments.%s.%s"|args:$nom_cat:$equipment.key value="%s (dispo : %d)"|args:$equipment.designation:$dispo}}
{{/if}}
{{/load}}
{{/load}}
{{* calculer les disponibilités *}}
{{#foreach from=$equipments key="cat_key" item="category"}}
{{#foreach from=$category.eqpmt key="eqpmt_key" item="eqpmt"}}
{{:assign dispo="%d-%d"|math:$eqpmt.stock:$eqpmt.exterieur}}
{{:assign
var="disponibilites.%s.%s"|args:$category.name:$eqpmt_key
value="%s (dispo : %d)"|args:$eqpmt.designation:$dispo
}}
{{/foreach}}
{{/foreach}}
{{if $disponibilites === null}}
{{:error message="Le matériel choisi (%s) n'est pas présent à la date du %s"|args:$designation:$_POST.date}}
{{/if}}
{{* formulaire de sortie de matériel *}}
<form method="post" action="">
<fieldset class="sortie">
@ -73,10 +82,10 @@
{{:input type="select" name="type_operation" label="Type" required=true options=$types_sorties}}
{{:input type="date" name="date" label="Date de sortie" required=true default=$now|date_short}}
{{:input type="number" name="quantite" label="Quantité" required=true default=1 min=1}}
<fieldset>
<legend>Matériel</legend>
{{:input type="select_groups" name="equipment" label="Matériel" required=true options=$equipments onchange="fixerValeurMax('f_equipment', 'f_quantite')"}}
<p><span class="alert">La quantité disponible est celle à la date du jour</span></p>
{{:input type="select_groups" name="equipment" label="Matériel" required=true options=$disponibilites onchange="fixerValeurMax('f_equipment', 'f_quantite')"}}
{{:input type="textarea" name="remarques" label="Remarques" cols="40" rows="3" required=false}}
</fieldset>
</fieldset>