Compra de productes: diferència entre les revisions
| Línia 26: | Línia 26: | ||
$total_compra = 0; // Definim el total de la compra | $total_compra = 0; // Definim el total de la compra | ||
$detalls_compra = []; // Definim una array on especificarem els detalls de cada compra (quantitat, preu, etc) | $detalls_compra = []; // Definim una array on especificarem els detalls de cada compra (quantitat, preu, etc) | ||
</pre> | </pre>A partir d'aquest moment, per cada producte dins del cistell realitzarem una sèrie de verificacions i càlculs | ||
=== Verificació de la quantitat de productes === | === Verificació de la quantitat de productes === | ||
Per cada producte dins del cistell haurem de verificar que la quantitat sigui correcte<pre> | |||
foreach ($_SESSION['cistell'] as $id_producte => $producte) { // Per cada producte | |||
$quantitat = (int)$producte['quantitat']; // Agafem la quantitat | |||
if ($quantitat <= 0){ // En cas que la quantitat sigui 0 o menys | |||
throw new Exception("Quantitat no vàlida"); // Llençem una excepció per invocar el rollback | |||
} | |||
</pre> | |||
=== Recuperació de les dades del producte === | === Recuperació de les dades del producte === | ||
A partir de la ID del producte del cistell, recuperarem les seves dades i verificarem que el producte realment sigui correcte<pre> | |||
// Recuperem el preu i l'stock del producte del cistell | |||
$stmt = $pdo->prepare("SELECT preu, estoc FROM productes WHERE id = :id_producte"); | |||
$stmt->bindParam(":id_producte",$id_producte); | |||
$stmt->execute(); | |||
$db_producte = $stmt->fetch(); | |||
// Si la consulta no retorna res, voldrà dir que l'ID no existeix | |||
if (!$db_producte){ | |||
throw new Exception("El producte no existeix"); | |||
} | |||
</pre> | |||
=== Verificació de quantitat de productes amb stock === | === Verificació de quantitat de productes amb stock === | ||
Si, tot i havent-hi verificacions d’stock real i visible, es dóna el cas que al cistell hi ha més productes que stock real, llençarem una excepció<pre> | |||
// Si per qualsevol motiu, la quantitat del cistell és superior a l'stock, llençarem una excepció | |||
if ($quantitat > $db_producte['estoc']){ | |||
throw new Exception("No hi ha prou estoc del producte ID $id_producte"); | |||
} | |||
</pre> | |||
=== Càlcul de subtotals i el total === | |||
De la mateixa manera que es fa amb el cistell, calculem el subtotal de cada producte amb la funció [[Funcions de la pràctica 5.2#Funció calcularPreus|calcularPreus]] i anem sumant el subtotal al total de compra per cada producte.<pre> | |||
// Calculem el preu dels productes aplicant un possible descompte | |||
$total_producte = calcularPreus($db_producte['preu'], $quantitat); // És una array amb: 1. Preu original sense descompte 2. Preu amb descompte 3. Descompte aplicat | |||
$total_compra = $total_compra + $total_producte['total_final']; // Per cada producte, sumem el seu import al total de la compra | |||
</pre> | |||
=== Desar els detalls del producte dins d'una array === | === Desar els detalls del producte dins d'una array === | ||
Per cada producte comprat, desarem els seus detalls dins d'una array. D'aquesta manera, posteriorment podrem inserir aquestes dades dins la taula compres_detall. | |||
Si es compren 2 productes, hi haurà 2 vectors dins d'aquesta array, <pre> | |||
// Desem dels detalls de la compra | |||
$detalls_compra[] = [ //[] a una array indica afegir al final un objecte (en aquest cas una altra array). En aquesta array hi haura tants objectes com productes comprats (no quantitat, sinó "quin producte") | |||
'id_producte' => $id_producte, | |||
'quantitat' => $quantitat, | |||
'preu_unitari' => $db_producte['preu'], | |||
'total_final' => $total_producte['total_final'] | |||
]; | |||
</pre>Tot aquest procés es realitza per cada producte dins del cistell | |||
=== Inserció de dades a la BD === | === Inserció de dades a la BD === | ||
Un cop recuperats tots els productes, inserirem a la taula compres el total de la compra i la data de compra. És important recuperar la ID d’aquesta compra per després inserir-la als registres de compres_detall (ja que recordem que és una clau forània)<pre> | |||
// Registrar la compra | |||
$stmt = $pdo->prepare("INSERT INTO compres (data_compra, total) VALUES (NOW(), :totalCompra)"); | |||
$stmt->bindParam(":totalCompra",$total_compra); | |||
$stmt->execute(); | |||
$compra_id = $pdo->lastInsertId(); // Desem la ID per relacionar els detalls de la compra a aquesta compra | |||
</pre>A continuació, a partir de l’array de detalls de compres, per cada producte comprat, inserirem els seus detalls (quantitat comprada, preus, etc) a la taula compres_detall, i la ID de compra la recuperem de l’anterior registre a la taula compres. Si a l'array hi ha 2 vectors (2 objectes comprats) s'inseriràn 2 registres a la taula<pre> | |||
// Registra els detalls de la compra i actualitzar l'stock dels productes | |||
foreach ($detalls_compra as $detall) { // (Per cada producte comprat) Per cada detall de la compra, el qual és un tipus de producte | |||
$stmt = $pdo->prepare( // Insertarem els detalls a una base de dades | |||
"INSERT INTO compres_detall | |||
(compra_id, producte_id, quantitat, preu_unitari, total_producte) | |||
VALUES (:compra_id, :producte_id, :quantitat, :preu_unitari, :total_producte)" | |||
); | |||
$stmt->bindParam(":compra_id", $compra_id); // Relacionem els detalls de la compra a la compra | |||
$stmt->bindParam(":producte_id", $detall['id_producte']); // Definim quin producte és | |||
$stmt->bindParam(":quantitat", $detall['quantitat']); // La quantitat | |||
$stmt->bindParam(":preu_unitari", $detall['preu_unitari']); // El preu unitari | |||
$stmt->bindParam(":total_producte", $detall['total_final']); // El total d'aquell conjunt d'unitats (incloent-hi descompte) | |||
$stmt->execute(); | |||
</pre> | |||
=== Actualització de l'stock dels productes === | === Actualització de l'stock dels productes === | ||
Revisió del 17:21, 31 des 2025
Compra de productes
A l'aplicació s'implementa la funcionalitat de comprar els productes dins del cistell. La funció d'aquesta funcionalitat és desar la compra i els seus detalls a la BD.
Inicialització de la sessió i connexió a la BD
Sempre que es treballa amb sessions, el primer que hem de fer és iniciar-la abans d'escriure el codi HTML
session_start();
En aquest cas, si el cistell no està definit (o està buit) no el crearem, sinó que redirigirem a l'usuari al catàleg
if (!isset($_SESSION['cistell']) || empty($_SESSION['cistell'])) { // En cas que s'accedeixi a la pàgina amb el cistell buit
header("Location: cataleg.php"); exit;}
A més de requerir la connexió d'escriptura a la BD i el fitxer funcions.php on es troba la funcio calcularPreus
require_once "./connexioBD/connexioRW.php"; // Requerim un fitxer on definim la funció per calcular els preus i descomptes require_once "funcions.php";
Inici de la transacció
En aquest cas realitzarem tot el procés mitjançant una transacció. Una transacció ens permet rectificar tot el que s'ha fet abans de que es produis un error.
En aquest codi estarem insertant i actualitzant molts registres a la BD, i no volem que un registre es mantingui quan realment ha sigut una compra incorrecte i no s'ha pogut processar correctament
Gràcies a una transacció, en cas de que certes operacions es realitzin correctament, en cas d'haver-hi algun error en alguna futura, totes les que s'han realitzat correctament seran desfetes.
$pdo->beginTransaction();
A més, iniciarem una variable per calcular el total de compra i una array on desarem els detalls dels productes
$total_compra = 0; // Definim el total de la compra $detalls_compra = []; // Definim una array on especificarem els detalls de cada compra (quantitat, preu, etc)
A partir d'aquest moment, per cada producte dins del cistell realitzarem una sèrie de verificacions i càlculs
Verificació de la quantitat de productes
Per cada producte dins del cistell haurem de verificar que la quantitat sigui correcte
foreach ($_SESSION['cistell'] as $id_producte => $producte) { // Per cada producte
$quantitat = (int)$producte['quantitat']; // Agafem la quantitat
if ($quantitat <= 0){ // En cas que la quantitat sigui 0 o menys
throw new Exception("Quantitat no vàlida"); // Llençem una excepció per invocar el rollback
}
Recuperació de les dades del producte
A partir de la ID del producte del cistell, recuperarem les seves dades i verificarem que el producte realment sigui correcte
// Recuperem el preu i l'stock del producte del cistell
$stmt = $pdo->prepare("SELECT preu, estoc FROM productes WHERE id = :id_producte");
$stmt->bindParam(":id_producte",$id_producte);
$stmt->execute();
$db_producte = $stmt->fetch();
// Si la consulta no retorna res, voldrà dir que l'ID no existeix
if (!$db_producte){
throw new Exception("El producte no existeix");
}
Verificació de quantitat de productes amb stock
Si, tot i havent-hi verificacions d’stock real i visible, es dóna el cas que al cistell hi ha més productes que stock real, llençarem una excepció
// Si per qualsevol motiu, la quantitat del cistell és superior a l'stock, llençarem una excepció
if ($quantitat > $db_producte['estoc']){
throw new Exception("No hi ha prou estoc del producte ID $id_producte");
}
Càlcul de subtotals i el total
De la mateixa manera que es fa amb el cistell, calculem el subtotal de cada producte amb la funció calcularPreus i anem sumant el subtotal al total de compra per cada producte.
// Calculem el preu dels productes aplicant un possible descompte $total_producte = calcularPreus($db_producte['preu'], $quantitat); // És una array amb: 1. Preu original sense descompte 2. Preu amb descompte 3. Descompte aplicat $total_compra = $total_compra + $total_producte['total_final']; // Per cada producte, sumem el seu import al total de la compra
Desar els detalls del producte dins d'una array
Per cada producte comprat, desarem els seus detalls dins d'una array. D'aquesta manera, posteriorment podrem inserir aquestes dades dins la taula compres_detall.
Si es compren 2 productes, hi haurà 2 vectors dins d'aquesta array,
// Desem dels detalls de la compra
$detalls_compra[] = [ //[] a una array indica afegir al final un objecte (en aquest cas una altra array). En aquesta array hi haura tants objectes com productes comprats (no quantitat, sinó "quin producte")
'id_producte' => $id_producte,
'quantitat' => $quantitat,
'preu_unitari' => $db_producte['preu'],
'total_final' => $total_producte['total_final']
];
Tot aquest procés es realitza per cada producte dins del cistell
Inserció de dades a la BD
Un cop recuperats tots els productes, inserirem a la taula compres el total de la compra i la data de compra. És important recuperar la ID d’aquesta compra per després inserir-la als registres de compres_detall (ja que recordem que és una clau forània)
// Registrar la compra
$stmt = $pdo->prepare("INSERT INTO compres (data_compra, total) VALUES (NOW(), :totalCompra)");
$stmt->bindParam(":totalCompra",$total_compra);
$stmt->execute();
$compra_id = $pdo->lastInsertId(); // Desem la ID per relacionar els detalls de la compra a aquesta compra
A continuació, a partir de l’array de detalls de compres, per cada producte comprat, inserirem els seus detalls (quantitat comprada, preus, etc) a la taula compres_detall, i la ID de compra la recuperem de l’anterior registre a la taula compres. Si a l'array hi ha 2 vectors (2 objectes comprats) s'inseriràn 2 registres a la taula
// Registra els detalls de la compra i actualitzar l'stock dels productes
foreach ($detalls_compra as $detall) { // (Per cada producte comprat) Per cada detall de la compra, el qual és un tipus de producte $stmt = $pdo->prepare( // Insertarem els detalls a una base de dades "INSERT INTO compres_detall (compra_id, producte_id, quantitat, preu_unitari, total_producte) VALUES (:compra_id, :producte_id, :quantitat, :preu_unitari, :total_producte)" ); $stmt->bindParam(":compra_id", $compra_id); // Relacionem els detalls de la compra a la compra $stmt->bindParam(":producte_id", $detall['id_producte']); // Definim quin producte és $stmt->bindParam(":quantitat", $detall['quantitat']); // La quantitat $stmt->bindParam(":preu_unitari", $detall['preu_unitari']); // El preu unitari $stmt->bindParam(":total_producte", $detall['total_final']); // El total d'aquell conjunt d'unitats (incloent-hi descompte)$stmt->execute();