IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Symfony PHP Discussion :

Impossible d'obtenir un tableau de string pour Chart.js


Sujet :

Symfony PHP

  1. #1
    Membre éprouvé
    Homme Profil pro
    Inscrit en
    Janvier 2011
    Messages
    1 126
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Janvier 2011
    Messages : 1 126
    Par défaut Impossible d'obtenir un tableau de string pour Chart.js
    Bonjour à tous,

    Je m'arrache les cheveux depuis une semaine :

    Impossible d'obtenir un tableau de string dans mon contrôleur ni dans TWIG, rendant l'affichage et récupération du data dans chart.js impossible...

    J'ai essayé toutes les méthodes pour remplir le tableau $colDepenses, à chaque fois il me ressort les indexes au lieu d'un tableau de valeurs uniquement... Deplus, il me sort une montagne de décimales après la virgule même si je fais un round ...

    J'ai pourtant pris soin de requêter dans le repository avec un getScalarResult() puis de faire un Json_encode() dans le controller puis ajouter un pipe raw dans twig, rien n'y fait, il renvoie les indexes dont je n'ai pas besoin :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    "[{"1":0,"2":74652.2,"3":24120.18,"4":63289.89,"5":0,"6":0,"7":39838.939999999995,"8":0,"9":73335.99,"10":11832.35,"11":46736.37,"12":22343.440000000002}]"
    qui n'est donc pas lisible pour chart.js dans son champ data ...

    Comment faire ?

    Merci d'avance

    StatistiquesController.php :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    <?php
     
    namespace App\Controller;
     
    use App\Repository\DepensesRepository;
    use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
    use Symfony\Component\HttpFoundation\Response;
    use Symfony\Component\Routing\Annotation\Route;
     
    #[Route('admin/')]
    class StatistiquesController extends AbstractController
    {
     
        #[Route('statistiques2', 'statistiques2')]
        public function statistiques(DepensesRepository $depensesRepository): Response
        {
            $depenses = $depensesRepository->getColonnesDepensesParAnnee(2022);
     
            $colDepenses = [];
     
            foreach ($depenses as $key => $value) {
                $colDepenses[$key] = $value;
            }
     
            return $this->render('statistiques/statistiques2.html.twig', [
                'colDepenses' => json_encode($colDepenses)
            ]);
        }
    }
    TWIG :
    Code twig : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    {% extends 'base.html.twig' %}
    {% block title %}
    	Statistiques
    {% endblock %}
    {% block stylesheets %}
    	<link rel="stylesheet" href="">
    {% endblock %}
    {% block body %}
    	{{dump(colDepenses)}}
    	<div class="wrapper-table">
    		<h1>Statistiques</h1>
    		<canvas id="graphDepenses" width="400" height="400"></canvas>
    	</div>
    {% endblock %}
    {% block javascripts %}
    	<script src="https://cdn.jsdelivr.net/npm/chart.js@4.4.0/dist/chart.umd.min.js"></script>
    	<script type="text/javascript">
                    const graphDepenses = document.querySelector("#graphDepenses")
    console.log(graphDepenses)
    let graph = new Chart(graphDepenses, {
    type: "bar",
    data: {
    labels: [
    "Janv",
    "Févr",
    "Mars",
    "Avr",
    "Mai",
    "Juin",
    "Juil",
    "Août",
    "Sept",
    "Oct",
    "Nov",
    "Déc"
    ],
    datasets: [
    {
    label: "Dépenses pour l'année",
    data: {{colDepenses|raw}},
    backgroundColor: [
    "red",
    "blue",
    "green",
    "red",
    "blue",
    "green",
    "red",
    "blue",
    "green",
    "red",
    "blue",
    "green"
    ],
    borderWidth: 1
     
    }
    ]
    }
    })
            </script>
    {% endblock %}

    repository :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    //Obtenir les colonnes de dépenses par mois et année pour graphiques (année spécifique)
        public function getColonnesDepensesParAnnee($selectionAnnee)
        {
            return $this->createQueryBuilder('d')
                ->select("
    SUM(CASE WHEN MONTH(d.date_depense)='1' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='2' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='3' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='4' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='5' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='6' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='7' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='8' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='9' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='10' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='11' THEN d.prix_unitaire * d.quantite ELSE 0 END),
    SUM(CASE WHEN MONTH(d.date_depense)='12' THEN d.prix_unitaire * d.quantite ELSE 0 END)")
                ->setParameter(':annee', $selectionAnnee)
                ->andWhere('YEAR(d.date_depense)=:annee')
                ->getQuery()->getScalarResult();
        }

  2. #2
    Membre émérite
    Homme Profil pro
    Autre
    Inscrit en
    Juillet 2021
    Messages
    427
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Loire Atlantique (Pays de la Loire)

    Informations professionnelles :
    Activité : Autre

    Informations forums :
    Inscription : Juillet 2021
    Messages : 427
    Par défaut
    Bonjour,

    Ajoute des print_r/var_dump (ou utilise xdebug) lorsque tu débogues ton code pour vérifier le contenu de tes variables.

    D'après ton json, la requête retourne un tableau contenant un seul résultat (un print_r($depenses); permettrait de vérifier)
    Le résultat retourné est sous la forme d'un objet et pas d'un tableau. Cet objet n'a donc pas d'indexes mais des propriétés nommés "1", "2", "3", etc. (d'où la confusion avec des indexes).
    Tu peux éventuellement essayer de convertir l'objet en tableau pour générer le json attendu (solution pas très propre, il serait mieux de récupérer un tableau pour le résultat) :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
            return $this->render('statistiques/statistiques2.html.twig', [
                'colDepenses' => json_encode(array_values((array) $colDepenses[0]));
            ]);

    Il serait surement mieux de récupérer un tableau au lieu d'un objet pour le résultat de ta requête.
    Quel est le but de l'utilisation de getScalarResult() ? Pas sûr que tu utilise cette méthode pour la bonne raison. Pour récupérer un tableau de résultat, tu peux utiliser la méthode getResult(Query::HYDRATE_ARRAY) ou getArrayResult().

    Il n'y a pas d'utilisation de round dans les extraits de code publiés.

  3. #3
    Membre éprouvé
    Homme Profil pro
    Inscrit en
    Janvier 2011
    Messages
    1 126
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Janvier 2011
    Messages : 1 126
    Par défaut
    Merci grandement Pytet pour ta réponse

    Ta solution fonctionne cependant par curiosité j'aimerai savoir comment arrondir les valeurs contenues dans le tableau final de valeurs... Faut il faire simplement une boucle foreach() ?

    Concernant getScalarResult c'est justement le remplaçant des fonctions que tu m'as conseillé d'utiliser, cependant aucune des 3 retourne un tableau de valeurs mais bien le genre de tableau dont je t'ai fait la capture dans mon premier message... Raison pour laquelle je m'arrachais les cheveux car contrairement à ce qui est dit dans la doc de symfony je n'arrivais pas a obtenir un simple tableau...

  4. #4
    Membre émérite
    Homme Profil pro
    Autre
    Inscrit en
    Juillet 2021
    Messages
    427
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Loire Atlantique (Pays de la Loire)

    Informations professionnelles :
    Activité : Autre

    Informations forums :
    Inscription : Juillet 2021
    Messages : 427
    Par défaut
    Je suppose que tu dois pouvoir arrondir le résultat directement dans les clauses THEN de ta requête : THEN ROUND(d.prix_unitaire * d.quantite, 2) par exemple.

    Pas de capture dans ton premier message, tu as posté le résultat json mais il serait plus simple de voir le résultat de la requête (via un dd($depenses) dans le contrôleur).
    Je n'ai pas accès à un Symfony actuellement pour tester, il faudrait faire un print_r/var_dump/dd sur le résultat retournés par les fonctions getScalarResult et getArrayResult pour comparer.

  5. #5
    Membre éprouvé
    Homme Profil pro
    Inscrit en
    Janvier 2011
    Messages
    1 126
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Janvier 2011
    Messages : 1 126
    Par défaut
    Bonjour Pytet

    Alors voilà où j'en suis :

    Ta solution me donne bien le tableau requis mais cependant maintenant je vois le graphique et sa grille mais pas les colonnes de valeurs dans mon chart.js, le tableau est ramené correctement à chaque changement de requête dans la console, il doit y avoir à reformater quelque chose pour que ça marche ? Si je mets des données manuellement ça marche donc cela me laisse supposer que la variable colDepenses ne serait pas encore dans le bon format...
    A signaler que je n'ai plus aucune erreur en console (au début j'avais des "undefined" mais plus maintenant...).
    Exemple de résultat retourné :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    0:"[0,66536.07,21216.160000000003,52111.55,0,89264.48000000001,0,0,29522.000000000004,0,41984.43,0]"
    que vient faire le "0:" devant le tableau json() ? cela serait il le problème ?

    Code twig : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138
    139
    140
    141
    142
    143
    144
    145
    146
    147
    148
    149
    150
    151
    152
    153
    154
    155
    156
    157
    158
    159
    160
    161
    162
    163
    164
    165
    166
    167
    168
    169
    170
    171
    172
    173
    {% extends 'base.html.twig' %}
    {% block title %}Statistiques
    {% endblock %}
    {% block body %}
    	{% block stylesheets %}
    		<link rel="stylesheet" href="https://unpkg.com/charts.css/dist/charts.min.css">
    		<link rel="stylesheet" href="{{asset('assets/css/graphiques.css')}}">
    	{% endblock %}
     
    	<div class="wrapper-table">
    		<h1>Statistiques</h1>
     
    		{# Graphique des dépenses #}
    		<div class="container-graphiques">
    			<select name="selectAnneesDepenses" id="selectAnneesDepenses" class="btn select selectAnneesDepenses" data-select="graphique-mois-depenses">
    				{% for a in selectAnneesDepenses %}
    					<option value="{{a.annee}}">{{a.annee}}</option>
    				{% endfor %}
    			</select>
    			<div id="graphique-depenses">
    				<canvas id="graphDepenses" width="600" height="400"></canvas>
    			</div>
     
    			<select name="selectAnneesRecettes" id="selectAnneesRecettes" class="btn select selectAnneesRecettes" data-select="graphique-mois-recettes">
    				{% for a in selectAnneesRecettes %}
    					<option value="{{a.annee}}">{{a.annee}}</option>
    				{% endfor %}
    			</select>
    			<div id="graphique-recettes"></div>
    		</div>
    	</div>
    {% endblock %}
     
    {% block javascripts %}
    	<script src="https://cdn.jsdelivr.net/npm/chart.js@4.4.0/dist/chart.umd.min.js"></script>
    	<script type="text/javascript">
     
                    const selectAnneeDepenses = document.querySelector(".selectAnneeDepenses");
    const selectAnneeRecettes = document.querySelector(".selectAnneeRecettes");
    const graphDepenses = document.querySelector("#graphDepenses")
    const graphRecettes = document.querySelector("#graphique-recettes")
     
    selectAnneesDepenses.addEventListener("change", (e) => {
    let selectionAnnee = e.target.value;
    filtreAnneeDepenses(selectionAnnee);
    });
     
    const filtreAnneeDepenses = (selectionAnnee) => {
    let formData = new FormData();
    formData.append("selectionAnnee", selectionAnnee);
     
    for (let item of formData) {
    console.log(item[0], item[1]);
    }
     
    let colDepenses
     
    async function statsDepenses(colDepenses) {
    let response = await fetch("/admin/statistiques", {
    body: formData,
    method: "POST",
    cache: "no-cache",
    headers: {
    "X-Requested-With": "XMLHttpRequest"
    }
    })
    colDepenses = await response.json();
    return colDepenses
    };
    statsDepenses()
    }
     
    // Obtenir dernier mois contenu dans le select et afficher les dépenses au démarrage de la page
    if (selectAnneesDepenses.value) {
    let selectionAnnee = selectAnneesDepenses[0].value;
    selectAnneesDepenses.value = selectionAnnee;
    filtreAnneeDepenses(selectionAnnee);
     
    }
     
    selectAnneesDepenses.addEventListener("change", (e) => {
    let selectionAnnee = e.target.value;
    filtreAnneeDepenses(selectionAnnee);
    });
     
    // Graphique dépenses
    const afficherGraphique = (colDepenses) => {
     
    let graph = new Chart(graphDepenses, {
    type: "bar",
    data: {
    labels: [
    "Janv",
    "Févr",
    "Mars",
    "Avr",
    "Mai",
    "Juin",
    "Juil",
    "Août",
    "Sept",
    "Oct",
    "Nov",
    "Déc",
    ],
    datasets: [
    {
    label: "Dépenses pour l'année",
    data: colDepenses,
    backgroundColor: [
    "red",
    "blue",
    "green",
    "red",
    "blue",
    "green",
    "red",
    "blue",
    "green",
    "red",
    "blue",
    "green",
    ],
    borderWidth: 1,
    borderColor: ["darkcyan"]
    },
    ]
    }
    })
    };
    afficherGraphique()
            </script>
    {% endblock %}
     
     
    {# graphique recettes #}
    {# selectAnneesRecettes.addEventListener("change", (e) => {
    	let selectionAnnee = e.target.value;
    	filtreAnneeRecettes(selectionAnnee);
    	});
    	const filtreAnneeRecettes = async (selectionAnnee) => {
    	let formData = new FormData();
    	formData.append("selectionAnnee", selectionAnnee);
     
    	for (let item of formData) {
    	console.log(item[0], item[1]);
    	}
     
    	const reponse = await fetch("/admin/statistiques", {
    	body: formData,
    	method: "POST",
    	cache: "no-cache",
    	headers: {
    	"X-Requested-With": "XMLHttpRequest"
    	}
    	}).then((response) => response.json()).then(data => console.log(data)).catch((error) => alert("Erreur : " + error));
    	return reponse;
    	};
    	}
     
    	// //Obtenir dernier mois contenu dans le select et afficher les recettes au démarrage de la page
    	{
    	# if (selectAnneesRecettes.value) {
    	let selectionAnnee = selectAnneesRecettes[0].value;
    	selectAnneesRecettes.value = selectionAnnee;
    	filtreAnneeRecettes(selectionAnnee);
    	}
    	#
    	} {
    	# selectAnneesRecettes.addEventListener("change", (e) => {
    	let selectionAnnee = e.target.value;
    	console.log('selection année : ', selectionAnnee)
    	});#}

  6. #6
    Membre éprouvé
    Homme Profil pro
    Inscrit en
    Janvier 2011
    Messages
    1 126
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Janvier 2011
    Messages : 1 126
    Par défaut
    Bon j'ai trouvé la solution, il fallait faire un JSON.parse sur le tableau, tout marche maintenant !

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. Réponses: 2
    Dernier message: 30/10/2019, 09h44
  2. Réponses: 5
    Dernier message: 01/04/2014, 10h01
  3. Parser un hexa en String pour obtenir un byte
    Par Djobird dans le forum Langage
    Réponses: 4
    Dernier message: 30/04/2009, 13h55
  4. Réponses: 2
    Dernier message: 26/09/2008, 22h32
  5. Réponses: 6
    Dernier message: 10/01/2007, 09h30

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo