[AngularJs] placer un watcher (surveillance) sur un objet complet JSON

angularjs-300x300
Introduction

Si on veut surveiller un objet JSON en temps réel, et pas seulement un ensemble clef valeur, la syntaxe est différente .
RAPPEL : Un objet Json peut contenir tout un formulaire et des tas de données diverses, en TEMPS REEL, avec angularJs et le 2 ways binding. Un watcher peut servir à sauver un form en local Storage, au cas ou la personne pars de l’app pendant quelques temps .

Le code :

Sur un objet de ce type :

 $scope.form = {
        name: 'my name',
        surname: 'surname'
    }

Le watcher -> Ajouter true:
Call $watch with true as the third argument:

$scope.$watch('form', function(newVal, oldVal){
    console.log('changed');
}, true);

Lien :
https://stackoverflow.com/questions/19455501/watch-an-object

Publicités

[AngularJs] Filtre qui calcule l’age d’une personne en fonction de la date de naissance .

angularjs-300x300
Introduction

un filtre qui calcule l’age d’une personne en fonction de sa DDN

Le code :

app.filter('ageFilter', function() {
     function calculateAge(birthday) { // birthday is a date
         var ageDifMs = Date.now() - birthday.getTime();
         var ageDate = new Date(ageDifMs); // miliseconds from epoch
         return Math.abs(ageDate.getUTCFullYear() - 1970);
     }

     return function(birthdate) { 
           return calculateAge(birthdate);
     }; 
});

Utiliser dans une vue:

{{ footballeur.date_naissance | ageFilter }}

[Json] Encoder toutes les photos d’une application en base64

angularjs-300x300
Introduction

Ouiiin , J’ai Pas envie de stocker les photos des gens sur mon server au format JPEG ou BMP ou TIFF etc … ?

Il faut alors Encoder en Base 64 toutes les photos, puis les placer en base de donnée.

Code Compatible big data ou relationnel avec un bouton qui permet lupload, et le convertisseur base 64 …
Pour encoder une image en ligne : https://www.base64-image.de/

Le Code :
Dans la vue Html, un img src pour afficher et un boutton pour uploader :

// Limage 
<img src="{{footballeur.photo ||'defaut.png'}}" />

// Le bouton 
<div id="upload_button">
	<label>
	<input name="inputFileToLoad" id="inputFileToLoad" ng-model="logo"  type="file"  onchange=""  /> </input>
	<span class="btn btn-primary">Modifier la photo</span>
	</label>
</div>

Le code Front
Le code AngularJs (version $scope)qui permet de récupérer une photo, et le convertit en base 64 , ensuite, il ne reste plus qu’à le mettre en bdd big data ou relationnelle.

// GESTION UPLOAD PHOTO 
 $scope.encodeImageFileAsURL = function() {
		
		var filesSelected = document.getElementById("inputFileToLoad").files;
		if (filesSelected.length > 0) {
		  var fileToLoad = filesSelected[0];

		  var fileReader = new FileReader();

		  fileReader.onload = function(fileLoadedEvent) {
			
                        var srcData = fileLoadedEvent.target.result; // <--- data: base64
			
			$scope.footballeur.photo = srcData ; // Affiche la photo en temps réel dans la vue
			$scope.$apply();
			

		  }
		  fileReader.readAsDataURL(fileToLoad);
		}
	  }
	  $(document).on('change', 'input[type="file"]' , function(){
		$scope.encodeImageFileAsURL();
	
		// R?cup?ration des donn?es de la photo apr?s clic sur UPLOAD
		var file = this.files[0];
		var name = file.name;
		var size = file.size;
		var type = file.type;
	
	}); 

Explication : $(document).on fait office de watcher quand on uploade une photo avec le button . $scope.footballeur.photo contient ensuite la photo coenvertie au format base 64 du footballer Lionel Messi, par exemple, ça ressemble à ça :

 

[Php + AngularJs] En finir avec les messages TRY CATCH une bonne fois pour toute .

angularjs-300x300

Introduction :

Quand une requête SQL est en erreur dans le back end PHP PDO, on peut vouloir voir le message d’erreur SQL s’afficher en direct dans l’application  FRONT END angularJs.

Ce problème est insupportable si l’on gère différement les retours des requêtes $http à chaque fois, il faut donc  etre  géré toujours de la meme manière .
En fait , il s’agit d’un design pattern.

Comment faire :

Etape 1 :Activer l’affichage des erreurs de la connexion PDO : exemple :


$this->bdd = new PDO('mysql:host=' . self::DB_HOST . ';port='. self::PORT . ' ;dbname=' . self::DB_NAME , self::LOGIN , self::PWD);

$this->bdd->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);

Note : c’est la 2 ème ligne qui importe .

Etape 2 : Créer une fonction avec un try catch dans une classe PHP, exemple :


  protected function Update($table, $tab_value, $id) {

        try {
			$sql = 'UPDATE ' . $table . ' SET ';

            //  LES NOMS DE CHAMPS ET LEUR VALEUR
            if($tab_value) {
                foreach($tab_value as $i=>$v){
                    $sql .= $i.'= :'.$i.', ';
                }
            } else {
                $tab_value = array();
            }

            //  ENLEVER LA DERNIERE VIRGULE
            $sql = substr($sql, 0, -2);
            $sql .= ' WHERE id_'.$table.' = '.$id;

            $req = $this->bdd->prepare($sql);

            $req->execute($tab_value);

        } catch(PDOException  $e) {
			die('Erreur : '.$e->getMessage());
		}
		return true;
	}

Note : ici , on remarque qu’elle retourne TRUE, seulement à la fin, mais que le CATCH annulera le TRUE si la requête SQL est en erreur, et que le message d’erreur PDO parviendra bien au FRONT END .

Etape 3 :

Appeler la fonction dans le WEB SERVICE en PHP comme ceci (Exemple):

echo json_encode($user ->Update($table, $tab_value, $id));

Etape 4 : Dans le front end , Gérer l’erreur dans AngularJS :

$http.post(webServicesUrl+'?action=UpdateUser',this.utilisateur)
    .then(function (response, status, headers, config) { // return true or the PDO error
	
	if (response.data == "true") {
			 Notification.error({message: 'ok cest bien TRUE', delay: 5000});
			
	} else { 
		 
		 Notification.error({message: response.data, delay: 5000});
		

	}


	 
}.bind(this));

Note : On remarque que TRUE est un string, et que toute autre réponse que TRUE provenant du back end provoque l’affichage de l’erreur dans le FRONT END .

Conclusion :

Il faut désormais toujours utiliser ce pattern pour les requêtes. AngularJs attends soit un TRUE, soit autre chose, et rien d’autres (pas des 1 et des 0 , par exemple (booléens). En fait, il faut choisir, soit en prends les booléens, mais alors on ne récupère pas le message d’erreur PDO ; soit on fait comme plus haut.

Apparté :
En cas de problème :
En fait, le code n’est pas exactement celui ça décrit dans ce cas, car il y a

  • Le web Service
  • La classe Users
  • La classe Query qui comprends des méthodes génériques private pour updater automatiquement n’importe quelle table sql .

De ce fait, le code exact est :
Dans le web service :
echo json_encode($user ->UpdateUser($data));
( ou $data est un objet)
Dans la classe Users :
return $this->update(self::TABLE,$tab_value,$id);
ou les paramêtres sont l’objet $data scindé en plusieurs variables, dont un tableau, la classe query le nécessite.
Dans la classe query, c’est identique à expliqué au début.

[AngularJs] Un component de timer pour limiter des démos d’apps dans le temps

angularjs-300x300

Introduction


Un premier component de timer pour limiter des démos dans le temps (Version sans web worker)
AngularJS 1.7 avec les components est juste un framework… fantastique.
La seule chose un peu génante avec la syntaxe this, c’est le bind(this) qu’il faut rajouter parfois pour le 2 ways binding.
La syntaxe THIS est vraiment trop belle, et Ouah ce que AngularJS est puissant, flexible et logique ! Ca prends du temps d’essayer d’écrire du code angularjs proprement, mais ensuite… C’est top.
Les components réutilisables dialoguent entre eux, et sont réutilisables dans plusieurs APPS .
A venir : un component de gestion des droits utilisateurs, et un component de paiement (Paypal Checkout), et surement bien d’autres … Ca dépends si je passe sur vue.js ..

Le code :


Le component : 

monApp.component('timer', {
  templateUrl: 'vues/timer.html',

   controller: function($log,$scope,$location,footballersFactoryLocalStorage,pouchDB,stockeUtilisateur,Notification,$timeout) {

		// INIT
		this.demo = true;
		this.counter = 0;
        const dureeDemo  =  5000 // Durée de la démo

		// LE COMPTEUR POUR LA DEMO
		this.updateCounter = function() {

			if(this.counter > dureeDemo){
				Notification.error({message: 'La période de démonstration est finie, merci de vous identifier pour continuer', delay: 10000});
				$location.path('/login');

			}else{
				this.counter++;
				$timeout(this.updateCounter, 1000).then(function(success) {})
			}
		}.bind(this);

		// TEST DU LOGIN UTILISATEUR
		var utilisateur = stockeUtilisateur.getUtilisateur();

		if(angular.equals(utilisateur, {})){
			this.demo = true;
			this.updateCounter();
		}else{
			this.demo = false;
			Notification({message: 'Vous êtes bien identifié, bonjour '+utilisateur.username, title: 'Vous êtes bien identifié'});
		}

   }

});

Explication :

Le component teste si l’utilisateur est loggé, si il est loggé, il n’execute pas le timer de démonstration. Si il n’est pas loggé, il redirige l’utilisateur vers la page de login au bout de 5000 secondes (paramétrables).
Toutes les vairables préfixées par this agissent en temps réel dans la vue (Comme l’ancien $scope).

 

Le template timer.html , et oui, il est minus:

<span class="badge badge-warning"> Démo {{$ctrl.counter}}</span>

Dans le html, on l’appelle ensuite dans n’importe quelle app angularJS :

<timer counter="$resolve.counter"></timer>

Le service qui stocke un objet JSON ‘utilisateur ‘ (version sans local storage) :

monApp.service('stockeUtilisateur', function() {
   var utilisateur ={};

    return {
        getUtilisateur: function() {
            return utilisateur;
        },
        setUtilisateur: function(objet) {
            utilisateur = objet;
        }
    };
})

Résultat, dans n’importe quelle vue AngularJs :


Un timer compte jusque 5000, puis redirige vers la page de login /et/ou de paiement …

timer.jpg

Annexe :
Pour router vers un component ‘login’, voici la syntaxe du router de base angularJs :

$routeProvider
	.when('/login', {
		template: '<login user="$resolve.user"></login>',
		resolve: {
		  // user: function($http) { return $http.get('...'); }
		}
	  })
	.when('/gerer', {
		templateUrl: 'vues/gerer.html',
		controller: 'gerer',

	})
	.when('/matches', {
		templateUrl: 'vues/matches.html',
		controller: 'matches'
	})
	.when('/modifierFootballer/:id', {
		templateUrl: 'vues/modifierFootballer.html',
		controller: 'modifierFootballer'
	 })
	 .when('/achat', {
		templateUrl: 'vues/achat.html',
		controller: 'achat'
	 })

	.otherwise({redirectTo: '/gerer',templateUrl: 'vues/vignettes.html',controller: 'liste'});
})

[AngularJs] Calculer la moyenne des valeurs d’une colonne de table HTML, avec une seule fonction générique

angularjs-300x300.png

Introduction

Une fonction qui prends en paramêtre le modèle de données de la table, et le nom de la colonne (‘champs’) dont on veut calculer la moyenne .
( EDIT : Des bugs wordpress avaient pourri le code, et enlevé le ng-repeat)

Code

La fonction :

this.calculer_moyenne = function(modele, champs) {

    if (modele) {
        var sum = 0;
        for (var i = 0; i < modele.length; i++) {

            if (modele[i][champs]) {
                sum += parseInt(modele[i][champs]);
            }
        }

        var avg = sum / modele.length;

        return (avg).toFixed(2);
    }

}

Le HTML :

<table class="table table-striped table-bordered table-hover">
<tr>
<thead>
<th>Période</th>
<th title="Nombre de jours travaillés dans le mois">Nbre de JT</th>
<th title="Nombre d'appels par jours">Nb Appels</th>
<th title="Nombre d'appels par jours">Nbre d'appels/jour</th>
<th title="Nombre de fiches traitées">Nbre fichestraitées</th>
<th>Heures Tr</th>
<th>Appels/heure</th>
<th>Tx de rotation</th>
<th>Nbre contacts PDG</th>
<th>Nbre contact / JT</th>
<th>Tx contact</th>
<th>Nbre de devis créés</th>
<th>Tx d'accroche devis</th>
<th>Tx d'efficacité de vente</th>
<th>Tx efficacité entrant</th>
<th>Tx efficacité sortant</th>
<th>Nb de ventes</th>
<th>Nb Ventes entrantes</th>
<th>Nb Ventes sortantes</th>
<!--
<th>Ventes opé</th>
--></thead>
<thead>
<th><i class="fa fa-calendar  bg-secondary p-2 font-xl mr-3  "></th>
<th><i class="fa fa-industry  bg-primary p-2 font-xl mr-3 pull-center "> </i></th>
<th title="Nombre d'appels par jours"><i class="fa fa-phone  bg-primary p-2 font-xl mr-3 pull-center "> </i></th>
<th title="Nombre d'appels par jours"><i class="fa fa-phone  bg-secondary p-2 font-xl mr-3 pull-center "></i></th>
<th title="Nombre de fiches traitées"><i class="fa fa-map-o  bg-primary p-2 font-xl mr-3 pull-center "></i></th>
<th><i class="fa fa-industry  bg-primary p-2 font-xl mr-3 pull-center "></i></th>
<th><i class="fa fa-repeat  bg-warning p-2 font-xl mr-3 pull-center "></i></th>
<th><i class="fa fa-eur  bg-secondary p-2 font-xl mr-3  "></th>
<th><i class="fa fa-user-plus  bg-danger p-2 font-xl mr-3 "></i></th>
<th><i class="fa fa-user-plus  bg-secondary p-2 font-xl mr-3 "></i></th>
<th><i class="fa fa-user-plus  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-file-o  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-file-o  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-thumbs-o-up  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-thumbs-o-up  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-thumbs-o-up  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-eur  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-eur  bg-secondary p-2 font-xl mr-3  "></i></th>
<th><i class="fa fa-eur  bg-secondary p-2 font-xl mr-3  "></i></th>
<!--
<th>Ventes opé</th>
--></thead>
</tr>
<tr ng-repeat ="s in ctrl.sc">
<td> {{::s.nom}}</td>
<td> {{::s.nb_j_travailles}}</td>
<td> {{::s.nb_appels}}</td>
<td> {{(s.nb_appels / s.nb_j_travailles) | number: 2}}</td>
<td> {{::s.nbfiches_creees}}</td>
<td> {{::s.heures_mensuelle}}</td>
<td> {{(s.nb_appels / s[$index].travail_mois) | number: 2 }}</td>
<td> {{}}</td>
<td> {{::s.nb_contacts}}</td>
<td> {{}}</td>
<td> {{}}</td>
<td> {{::s.nbdevis_crees}}</td>
<td> {{(s.nbdevis_crees/s.nb_argus) | number: 2}}</td>
<td> {{::s.tx_efficacitevente}}</td>
<td> {{::s.tx_efficaciteventeentrant}}</td>
<td> {{::s.tx_efficaciteventesortant}}</td>
<td> {{::s.nb_vente}}</td>
<td> {{::s.nb_venteentrant}}</td>
<td> {{::s.nb_ventesortant}}</td>
<!--
<td> {{}}</td>
--></tr>
<tr>
<td>Moyenne</td>
<td>{{ctrl.calculer_moyenne(ctrl.sc,'nb_j_travailles')}}</td>
<td>{{ctrl.calculer_moyenne(ctrl.sc,'nb_appels')}}</td>
<td></td>
<td>{{ctrl.calculer_moyenne(ctrl.sc,'nbfiches_creees')}}</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</table>

[AngularJs 1.6.2] Communiquer des variables à des components

Introduction

On a 20 graphiques à réaliser, tous avec la même emprunte pattern de modèle de données , on va forcément éviter de créer 20 controllers pour gérer tout ça .
On tente donc de créer un component angularJS qui permet de simplifier tout cela , car on peut lui passer des variables.

Note : La v 1.7 de AngularJs gère le transfert de datas aux components un peu différement.

Lien :

https://stackoverflow.com/questions/53170065/angularjs-template-component-not-rendering-variables