Compare commits

..

33 Commits

Author SHA1 Message Date
Sana EL HIRI
adfd1db2ff correction jeremy 2022-03-22 14:54:07 +01:00
HarmandI
d09b147af2
Merge pull request #8 from HarmandI/feature/supprimer
Feature/supprimer
2022-03-22 12:55:35 +01:00
HarmandI
4aced57191 der des der? 2022-03-22 12:54:31 +01:00
Vincent Ramiere
a2ddf52c88 lien Page administrateur dans navbar 2022-03-22 12:52:32 +01:00
Vincent Ramiere
200221dc5c lien admin dans navbar 2022-03-22 12:49:35 +01:00
Sana
4f5e6cd180
Merge pull request #7 from HarmandI/css
css
2022-03-22 12:38:08 +01:00
Sana EL HIRI
33c431921d css 2022-03-22 12:36:25 +01:00
Vincent Ramiere
bd4468a9d6 Merge branch 'main' of https://github.com/HarmandI/Admin-La-Belle-Plante
pull
2022-03-22 12:14:04 +01:00
Vincent Ramiere
983533e4e1 Merge branch 'main' into feature/supprimer 2022-03-22 12:11:34 +01:00
HarmandI
a2cf4f55ed
Merge pull request #6 from HarmandI/feature/ajouter
Feature/ajouter
2022-03-22 12:09:42 +01:00
HarmandI
49edb1782d derniere? 2022-03-22 12:08:27 +01:00
HarmandI
be9e510a94 erreur 2022-03-22 11:47:05 +01:00
HarmandI
a99dbc57ab
Merge pull request #5 from HarmandI/feature/modifier-plante
Feature/modifier plante
2022-03-22 11:45:31 +01:00
HarmandI
426c359dbf
Merge branch 'main' into feature/modifier-plante 2022-03-22 11:45:21 +01:00
Sana EL HIRI
7d8e72e4ac modifier ok 2022-03-22 11:43:33 +01:00
Sana EL HIRI
1b5b7f0e3e modifs 2022-03-22 10:55:06 +01:00
Vincent Ramiere
1b18f648a0 formulaire, ajouter, modifier html 2022-03-22 10:01:26 +01:00
HarmandI
e8c45f87e9
Merge pull request #4 from HarmandI/feature/ajouter
Feature/ajouter
2022-03-22 09:52:58 +01:00
HarmandI
5e8d6c3532
Merge branch 'main' into feature/ajouter 2022-03-22 09:52:51 +01:00
HarmandI
26c96850ca
Merge pull request #3 from HarmandI/feature/supprimer
modif inStock
2022-03-22 09:47:47 +01:00
HarmandI
2773b54531
Merge branch 'main' into feature/supprimer 2022-03-22 09:47:39 +01:00
HarmandI
2ba13fa864
Merge pull request #2 from HarmandI/feature/modifier-plante
Feature/modifier plante
2022-03-22 09:45:33 +01:00
Vincent Ramiere
19d48cc995 modif inStock 2022-03-22 09:44:39 +01:00
Sana EL HIRI
03a825ec76 inStock 2022-03-22 09:42:03 +01:00
HarmandI
f2279bc977 select category et divers 2022-03-22 08:54:41 +01:00
Sana EL HIRI
d9800d3ae5 cette branche vient remplacer feature/modifier qui est corrompue 2022-03-21 20:51:17 +01:00
HarmandI
4d93763aa5 Merge branch 'main' into feature/ajouter 2022-03-21 18:59:04 +01:00
Vincent Ramiere
2d808feb25 merge page supprimer 2022-03-21 18:49:20 +01:00
Vincent Ramiere
04f3b22fd3 conflit de merge 2022-03-21 18:36:26 +01:00
Vincent Ramiere
1f747c6933 supprimer 2022-03-21 18:32:20 +01:00
HarmandI
8db3a6cde1 conflit merge 2022-03-21 18:30:32 +01:00
HarmandI
1406011efc ajouter 1 2022-03-21 18:25:59 +01:00
HarmandI
1afa24289d
Merge pull request #1 from HarmandI/ajout-subject
subject fini
2022-03-21 18:21:03 +01:00
17 changed files with 6511 additions and 6300 deletions

2295
db.json

File diff suppressed because it is too large Load Diff

View File

@ -21,6 +21,7 @@
<a class="nav-link disabled" *ngIf="likeCounter == 0"> Pas de plante likée</a> <a class="nav-link disabled" *ngIf="likeCounter == 0"> Pas de plante likée</a>
<a class="nav-link disabled" *ngIf="likeCounter == 1">Plante likée : {{ likeCounter }}</a> <a class="nav-link disabled" *ngIf="likeCounter == 1">Plante likée : {{ likeCounter }}</a>
<a class="nav-link disabled" *ngIf="likeCounter > 1">Plantes likées : {{ likeCounter }}</a> <a class="nav-link disabled" *ngIf="likeCounter > 1">Plantes likées : {{ likeCounter }}</a>
<a routerLink="admin" routerLinkActive="active-custom" class="nav-link">Page administrateur</a>
</div> </div>
</div> </div>
</div> </div>

View File

@ -6,7 +6,7 @@ import { PageTableauComponent } from './pages/page-tableau/page-tableau.componen
const routes: Routes = [ const routes: Routes = [
{ path: '', redirectTo: 'tableau', pathMatch: 'full'}, { path: '', redirectTo: 'tableau', pathMatch: 'full'},
{ path: 'modifier', component: PageModifierComponent }, { path: 'modifier/:id', component: PageModifierComponent },
{ path: 'ajouter', component: PageAjouterComponent }, { path: 'ajouter', component: PageAjouterComponent },
{ path: 'tableau', component: PageTableauComponent }, { path: 'tableau', component: PageTableauComponent },
]; ];

View File

@ -1,4 +1,5 @@
<form (ngSubmit)="onSubmit()" [formGroup]="plantForm"> <div class="form-plant">
<form [formGroup]="plantForm" (ngSubmit)="onSubmit()">
<div class="form-floating"> <div class="form-floating">
<input <input
type="text" type="text"
@ -8,12 +9,8 @@
name="name" name="name"
formControlName="nameFc" formControlName="nameFc"
[ngClass]="{ [ngClass]="{
'is-valid': 'is-valid': plantForm.controls['nameFc'].valid,
plantForm.controls['nameFc'].touched && 'is-invalid': !plantForm.controls['nameFc'].valid
plantForm.controls['nameFc'].valid,
'is-invalid':
plantForm.controls['nameFc'].touched &&
!plantForm.controls['nameFc'].valid
}" }"
/> />
<label for="floatingInputName">Nom</label> <label for="floatingInputName">Nom</label>
@ -27,12 +24,8 @@
name="price" name="price"
formControlName="priceFc" formControlName="priceFc"
[ngClass]="{ [ngClass]="{
'is-valid': 'is-valid': plantForm.controls['priceFc'].valid,
plantForm.controls['priceFc'].touched && 'is-invalid': !plantForm.controls['priceFc'].valid
plantForm.controls['priceFc'].valid,
'is-invalid':
plantForm.controls['priceFc'].touched &&
!plantForm.controls['priceFc'].valid
}" }"
/> />
<label for="floatingInputPrice">Prix</label> <label for="floatingInputPrice">Prix</label>
@ -46,34 +39,28 @@
name="quantity" name="quantity"
formControlName="quantityFc" formControlName="quantityFc"
[ngClass]="{ [ngClass]="{
'is-valid': 'is-valid': plantForm.controls['quantityFc'].valid,
plantForm.controls['quantityFc'].touched && 'is-invalid': !plantForm.controls['quantityFc'].valid
plantForm.controls['quantityFc'].valid,
'is-invalid':
plantForm.controls['quantityFc'].touched &&
!plantForm.controls['quantityFc'].valid
}" }"
/> />
<label for="floatingInputQuantity">Quantité</label> <label for="floatingInputQuantity">Quantité</label>
</div> </div>
<div class="form-floating"> <div class="form-floating select">
<input <select
type="text"
class="form-control" class="form-control"
id="floatingCategory" id="floatingInputCategory"
placeholder="" placeholder=""
name="category" name="category"
formControlName="categoryFc" formControlName="categoryFc"
[ngClass]="{ >
'is-valid': <option value="plantes fleuries">Plantes fleuries</option>
plantForm.controls['categoryFc'].touched && <option value="orchides">Orchidées</option>
plantForm.controls['categoryFc'].valid, <option value="cactus et plantes grasses">Cactus et plantes grasses</option>
'is-invalid': <option value="bonsas">Bonsas</option>
plantForm.controls['categoryFc'].touched && <option value="plantes vertes">Plantes vertes</option>
!plantForm.controls['categoryFc'].valid <option value="palmier dintrieur">Palmier d'intérieur</option>
}" </select>
/> <label value="category">Sélectionnez catégorie</label>
<label for="floatingInputCategory">Catégorie</label>
</div> </div>
<div class="form-floating"> <div class="form-floating">
<input <input
@ -84,12 +71,8 @@
name="rating" name="rating"
formControlName="ratingFc" formControlName="ratingFc"
[ngClass]="{ [ngClass]="{
'is-valid': 'is-valid': plantForm.controls['ratingFc'].valid,
plantForm.controls['ratingFc'].touched && 'is-invalid': !plantForm.controls['ratingFc'].valid
plantForm.controls['ratingFc'].valid,
'is-invalid':
plantForm.controls['ratingFc'].touched &&
!plantForm.controls['ratingFc'].valid
}" }"
/> />
<label for="floatingRating">Note</label> <label for="floatingRating">Note</label>
@ -114,13 +97,8 @@
type="submit" type="submit"
[disabled]="plantForm.invalid" [disabled]="plantForm.invalid"
> >
Créer une plante {{ buttonLabel }}
</button> </button>
<button </form>
class="w-100 btn btn-lg btn-outline-success" </div>
type="submit"
[disabled]="plantForm.invalid"
>
Modifier une plante
</button>
</form>

View File

@ -0,0 +1,75 @@
.login-form {
height: 100vh;
padding-top: 40px;
background-color: #f5f5f5;
}
.form-plant {
width: 100%;
max-width: 330px;
padding: 15px;
margin-top: 50px;
margin-left: auto;
margin-right: auto;
border: solid 1px;
border-radius: 10px;
background-color: #306340;
border-color: #306340;
}
.form-plant .checkbox {
font-weight: 400;
}
.form-plant .form-floating:focus-within {
z-index: 2;
}
.form-plant input[type="email"] {
margin-bottom: 10px;
border-bottom-right-radius: 0;
border-bottom-left-radius: 0;
}
.form-plant input[type="color"] {
border: none;
margin-bottom: 10px;
margin-top: -70px;
margin-left: 100px;
border-radius: 50%;
width: 100px;
height: 100px;
}
.form-plant input[type="color"]::-webkit-color-swatch {
border: none;
margin-top: -15px;
border-radius: 80%;
width: 80px;
height: 80px;
}
.form-plant input[type="text"] {
margin-bottom: 10px;
border-bottom-right-radius: 0;
border-bottom-left-radius: 0;
}
.form-plant input[type="date"] {
margin-bottom: 10px;
border-bottom-right-radius: 0;
border-bottom-left-radius: 0;
}
.form-plant input[type="password"] {
margin-bottom: 10px;
border-top-left-radius: 0;
border-top-right-radius: 0;
}
.btn-outline-success {
margin-top: 10px;
background-color: #ffff;
color: #306340 !important;
border-color: #306340 !important;
}

View File

@ -1,5 +1,6 @@
import { Component, OnInit } from '@angular/core'; import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';
import { FormGroup } from '@angular/forms'; import { FormBuilder, FormControl, FormGroup, Validators } from '@angular/forms';
import { Plant } from '../../models/plant';
@Component({ @Component({
selector: 'app-formulaire', selector: 'app-formulaire',
@ -8,10 +9,28 @@ import { FormGroup } from '@angular/forms';
}) })
export class FormulaireComponent implements OnInit { export class FormulaireComponent implements OnInit {
plantForm!: FormGroup; plantForm!: FormGroup;
constructor() {} @Input()buttonLabel!:String;
@Input() plantInfos!: any;
@Output() submitted = new EventEmitter();
ngOnInit(): void {} constructor(private fb : FormBuilder) {
}
ngOnInit(): void {
console.log(this.plantInfos);
this.plantForm = this.fb.group({
nameFc: new FormControl(this.plantInfos.product_name, [Validators.required]),
priceFc: new FormControl(this.plantInfos.product_price, [Validators.required]),
quantityFc: new FormControl(this.plantInfos.product_qty, [Validators.required]),
inStockFc: new FormControl(this.plantInfos.product_instock, [Validators.required]),
categoryFc: new FormControl(this.plantInfos.product_breadcrumb_label, [Validators.required]),
ratingFc: new FormControl(this.plantInfos.product_rating, [Validators.required]),
});
}
onSubmit(){
this.submitted.emit(this.plantForm.value)
}
onSubmit(){}
} }

View File

@ -1,14 +1,12 @@
import { Category } from "./category";
export class Plant { export class Plant {
constructor( constructor(
public name: string='', public name: string='',
public price: number =1, public price: number =1,
public quantity: number= 0, public quantity: number= 0,
public instock: boolean= true, public inStock: string[]=['disponible','partiellement disponible', 'non disponible'],
public category:Category, public category:string[]=['plantes fleuries','orchides','cactus et plantes grasses','bonsas','plantes vertes','palmier dintrieur'],
public urlPicture: string = "https//picsum.photos/id/18/200/300", public urlPicture: string = "https//picsum.photos/id/18/200/300",
public rating: number = 0, public rating: number = 0,
public id?: number public id: string = ''
){} ){}
} }

View File

@ -1 +1,2 @@
<p>page-ajouter works!</p> <h1>Ajouter une plante</h1>
<app-formulaire [plantInfos]="newplant" (submitted)="addPlant($event)" [buttonLabel]="'Ajouter une plante'"></app-formulaire>

View File

@ -0,0 +1,7 @@
h1{
display: flex;
justify-content: center;
color: rgba(0, 0, 0, 0.658);
font-size: 28px;
margin-top: 20px;
}

View File

@ -1,15 +1,67 @@
import { Component, OnInit } from '@angular/core'; import { Component, OnInit } from '@angular/core';
import {
FormBuilder,
FormControl,
FormGroup,
Validators,
} from '@angular/forms';
import { Router } from '@angular/router';
import { FormulaireComponent } from '../../components/formulaire/formulaire.component';
import { Plant } from '../../models/plant';
import { AdminService } from '../../services/admin.service';
@Component({ @Component({
selector: 'app-page-ajouter', selector: 'app-page-ajouter',
templateUrl: './page-ajouter.component.html', templateUrl: './page-ajouter.component.html',
styleUrls: ['./page-ajouter.component.scss'] styleUrls: ['./page-ajouter.component.scss'],
}) })
export class PageAjouterComponent implements OnInit { export class PageAjouterComponent implements OnInit {
public newplant = new Plant();
constructor() { }
ngOnInit(): void { constructor(private fb: FormBuilder, private router: Router, private adminService: AdminService) {
} }
ngOnInit(): void {
}
public addPlant(plant: any): void {
const nameValue = plant.nameFc;
const priceValue = plant.priceFc;
const ratingValue = plant.ratingFc;
const quantityValue = plant.quantityFc;
const categoryValue = plant.categoryFc;
const inStockValue = plant.inStockFc;
const plante: any = {
product_name: nameValue,
product_price: priceValue,
product_qty: quantityValue,
product_rating: ratingValue,
product_breadcrumb_label: categoryValue,
product_instock: [inStockValue],
product_url_picture : "https//picsum.photos/id/18/200/300",
product_discount_code : "",
product_color: "",
product_unitprice_ati: "",
product_unitprice_tf: "",
product_discount_tf: "",
product_discount_ati: "",
product_url_page: "",
product_shipping_method: null,
product_image_source: "",
product_seller: "market place",
product_web_only: "non"
};
this.adminService.addPlant(plante)?.subscribe((resp)=>{
this.router.navigate(['admin']);
})
}
} }

View File

@ -1 +1,3 @@
<p>page-modifier works!</p> <h1>Modifier la plante</h1>
<app-formulaire *ngIf="editPlant" [plantInfos]="editPlant" (submitted)="update($event)" [buttonLabel]="'Modifier la plante'"></app-formulaire>

View File

@ -0,0 +1,7 @@
h1{
display: flex;
justify-content: center;
color: rgba(0, 0, 0, 0.658);
font-size: 28px;
margin-top: 20px;
}

View File

@ -1,4 +1,8 @@
import { Component, OnInit } from '@angular/core'; import { Component, OnInit } from '@angular/core';
import { FormBuilder,FormGroup} from '@angular/forms';
import { ActivatedRoute, ParamMap, Router } from '@angular/router';
import { Plant } from '../../models/plant';
import { AdminService } from '../../services/admin.service';
@Component({ @Component({
selector: 'app-page-modifier', selector: 'app-page-modifier',
@ -6,10 +10,73 @@ import { Component, OnInit } from '@angular/core';
styleUrls: ['./page-modifier.component.scss'] styleUrls: ['./page-modifier.component.scss']
}) })
export class PageModifierComponent implements OnInit { export class PageModifierComponent implements OnInit {
editPlant!: Plant;
plantId!: string;
constructor() { }
constructor(private adminService: AdminService,
private router: Router,
private fb: FormBuilder,
private route: ActivatedRoute) {
}
ngOnInit(): void { ngOnInit(): void {
/** Pour récuperer l'id de la plante à modifier et appel Api**/
this.route.paramMap.subscribe((params : ParamMap) => {
const id = params.get('id')
if( id != null){
this.plantId = id;
//console.log(this.plantId);
this.adminService
.getPlantById(this.plantId)
.subscribe((plantData: any) => {
this.editPlant = plantData;
//console.log(this.editPlant);
//console.log(this.editPlant.name);
});
}
});
}
/** Méthode qui envoie les champs modifiés pour mise à jour **/
public update(plant: any): void {
//console.log(plant);
const nameValue = plant.nameFc;
const priceValue = plant.priceFc;
const ratingValue = plant.ratingFc;
const quantityValue = plant.quantityFc;
const categoryValue = plant.categoryFc;
const inStockValue = plant.inStockFc;
const plante: any = {
id: this.plantId,
product_name: nameValue,
product_price: priceValue,
product_qty: quantityValue,
product_rating: ratingValue,
product_breadcrumb_label: categoryValue,
product_instock: [inStockValue],
product_url_picture : "https//picsum.photos/id/18/200/300",
product_discount_code : "",
product_color: "",
product_unitprice_ati: "",
product_unitprice_tf: "",
product_discount_tf: "",
product_discount_ati: "",
product_url_page: "",
product_shipping_method: null,
product_image_source: "",
product_seller: "market place",
product_web_only: "non"
};
this.adminService.updatePlant(plante)?.subscribe((resp) => {
this.router.navigate(['admin']);
});
} }
} }

View File

@ -1,5 +1,9 @@
<table class="table" *ngIf="subCollection$ | async as collection"> <div id="container">
<thead class="thead-dark"> <div id="button">
<button class="w-100 btn btn-lg btn-outline-success" routerLink="../ajouter">Ajouter une plante </button>
</div>
<table class="table" *ngIf="subCollection$ | async as collection">
<thead class="thead-dark" style="background-color: #306340 ; color:#f3f7f4 ;">
<tr> <tr>
<th scope="col">Id</th> <th scope="col">Id</th>
<th scope="col">Nom</th> <th scope="col">Nom</th>
@ -15,15 +19,17 @@
</thead> </thead>
<tbody> <tbody>
<tr *ngFor="let products of collection"> <tr *ngFor="let products of collection">
<th scope="row">{{products.id}}</th> <td>{{products.id}}</td>
<td>{{products.name}}</td> <td>{{products.name}}</td>
<td>{{products.price}}</td> <td>{{products.price}}</td>
<td>{{products.quantity}}</td> <td>{{products.quantity}}</td>
<td>{{products.instock}}</td> <td>{{products.inStock}}</td>
<!-- <td>{{products.Catégorie.}}</td> --> <td>{{products.category}}</td>
<td>{{products.rating}}</td> <td>{{products.rating}}</td>
<td><a class="bi-pencil-square" routerLink="../modifier"></a></td> <td><a class="bi-pencil-square" style="color: #5472b1;" routerLink="../modifier/{{products.id}}"></a></td>
<td class="bi-trash-fill" style="color: red; cursor: pointer;" (click)="onClickDelete(5)"></td> <td class="bi-trash-fill" style="color: rgb(231, 73, 73); cursor: pointer;" (click)="onClickDelete(products.id)"></td>
</tr> </tr>
</tbody> </tbody>
</table> </table>
</div>

View File

@ -0,0 +1,19 @@
#container{
display: flex;
justify-content: center;
margin-top: 5rem;
}
table{
width: 75%;
margin-left: 2rem;
}
button{
background-color: #306340;
color: #f3f7f4 !important;
border-color: #306340 !important;
}
#button{
width: 10%;
}

View File

@ -28,8 +28,10 @@ export class PageTableauComponent implements OnInit {
// }) // })
} }
onClickDelete(id: number){ onClickDelete(id: string){
console.log(id); this.adminService.onClickDelete(id).subscribe((resp) => {
console.log("Suppression successful : ", resp);
});
} }

View File

@ -1,11 +1,11 @@
import { HttpClient } from '@angular/common/http'; import { HttpClient } from '@angular/common/http';
import { Injectable } from '@angular/core'; import { Injectable } from '@angular/core';
import { map, Observable, Subject } from 'rxjs'; import { map, Observable, Subject, tap } from 'rxjs';
import { environment } from 'src/environments/environment'; import { environment } from 'src/environments/environment';
import { Plant } from '../models/plant'; import { Plant } from '../models/plant';
@Injectable({ @Injectable({
providedIn: 'root' providedIn: 'root',
}) })
export class AdminService { export class AdminService {
public collection$!: Observable<Plant[]>; public collection$!: Observable<Plant[]>;
@ -21,7 +21,7 @@ export class AdminService {
//console.log("avant mapping: ", tabObj); //console.log("avant mapping: ", tabObj);
// Ici grâce à la méthode .map() on transforme tout les objet json du tableau en instance de notre classe Plant() // Ici grâce à la méthode .map() on transforme tout les objet json du tableau en instance de notre classe Plant()
return tabObj.map((obj: any) => { return tabObj.map((obj: any) => {
return new Plant(obj.product_name, obj.product_price,obj.product_quantity,obj.product_instock,obj.product_breadcrumb_label,obj.product_image_source,obj.product_rating,obj.id); return new Plant(obj.product_name, obj.product_price,obj.product_qty,obj.product_instock,obj.product_breadcrumb_label,obj.product_image_source,obj.product_rating,obj.id);
}) })
})); }));
//this.collection$ = this.httpClient.get<Plant[]>(`${this.apiUrl}/list_products`); //this.collection$ = this.httpClient.get<Plant[]>(`${this.apiUrl}/list_products`);
@ -31,13 +31,29 @@ export class AdminService {
public refreshCollection(): void { public refreshCollection(): void {
// On se sert de notre flux de donnée type observable froid // On se sert de notre flux de donnée type observable froid
this.collection$.subscribe((listPlant: Plant[]) => { this.collection$.subscribe((listPlant: Plant[]) => {
this.plantCollection = [...listPlant];
// Utiliser un observable chaud (subject) pour nexter nos données recues de notre observable froid // Utiliser un observable chaud (subject) pour nexter nos données recues de notre observable froid
this.subCollection$.next(listPlant); this.subCollection$.next(listPlant);
}) })
} }
addPlant(plant: Plant): Observable<any> | void {
return this.httpClient.post(`${this.apiUrl}/list_products`, plant);
}
onClickDelete(id: string): Observable<any> {
return this.httpClient.delete<any>(`${this.apiUrl}/list_products/${id}`).pipe(
tap(() => this.refreshCollection())
);
}
// getData(): Observable<any[]> { // getData(): Observable<any[]> {
// return this.httpClient.get<any[]>(`${this.apiUrl}/list_products`); // return this.httpClient.get<any[]>(`${this.apiUrl}/list_products`);
// } // }
/** Récupérer une plante par son Id **/
getPlantById(id: any): Observable<Plant> {
return this.httpClient.get<Plant>(`${this.apiUrl}/list_products/${id}`);
}
/** Modifier Plante **/
updatePlant(plant: Plant): Observable<any> {
return this.httpClient.put(`${this.apiUrl}/list_products/${plant.id}`, plant);
}
} }