Sarobidy22 1 month ago
parent
commit
25d8f834f5
  1. 81
      app/Controllers/AvanceController.php
  2. 211
      app/Controllers/OrderController.php
  3. 31
      app/Models/OrderItems.php
  4. 25
      app/Models/Orders.php
  5. 80
      app/Views/orders/createbyid.php
  6. 468
      app/Views/orders/edit.php
  7. 4
      app/Views/orders/index.php

81
app/Controllers/AvanceController.php

@ -1635,8 +1635,10 @@ private function generatePrintableInvoiceHTML($avance, $productName, $productDet
<title>Facture Avance - KELY SCOOTERS</title>
<style>
@media print {
body { margin: 0; padding: 10px; }
body { margin: 0; padding: 0; }
.no-print { display: none !important; }
.page { page-break-after: always; }
.page:last-child { page-break-after: auto; }
}
* { margin: 0; padding: 0; box-sizing: border-box; }
@ -1646,15 +1648,22 @@ private function generatePrintableInvoiceHTML($avance, $productName, $productDet
font-size: 12px;
line-height: 1.4;
background: #fff;
padding: 20px;
}
.invoice-container {
max-width: 800px;
.page {
width: 210mm;
height: 297mm;
padding: 15mm;
margin: 0 auto;
background: white;
}
.invoice-container {
border: 2px solid #000;
padding: 20px;
height: 100%;
display: flex;
flex-direction: column;
}
.header {
@ -1730,6 +1739,24 @@ private function generatePrintableInvoiceHTML($avance, $productName, $productDet
font-weight: bold;
}
.signature-section {
display: flex;
justify-content: space-between;
margin-top: auto;
padding: 20px 0;
}
.signature-box {
text-align: center;
width: 45%;
}
.signature-box p {
font-weight: bold;
margin-bottom: 50px;
}
/* Style spécifique pour le verso */
.contract-section {
margin-top: 20px;
padding: 15px;
@ -1752,25 +1779,16 @@ private function generatePrintableInvoiceHTML($avance, $productName, $productDet
text-decoration: underline;
}
.signature-section {
.verso-content {
flex: 1;
display: flex;
justify-content: space-between;
margin-top: 30px;
padding: 20px 0;
}
.signature-box {
text-align: center;
width: 45%;
}
.signature-box p {
font-weight: bold;
margin-bottom: 50px;
flex-direction: column;
}
</style>
</head>
<body>
<!-- RECTO -->
<div class="page">
<div class="invoice-container">
<!-- Header -->
<div class="header">
@ -1826,6 +1844,22 @@ private function generatePrintableInvoiceHTML($avance, $productName, $productDet
</tbody>
</table>
</div>
</div>
<!-- VERSO -->
<div class="page">
<div class="invoice-container">
<div class="invoice-title">
<div>
<h2>CONDITIONS GÉNÉRALES</h2>
<div>Date: {$avanceDate}</div>
<div>N°: {$avanceNumber}CI 2025</div>
<div class="verso-content">
<!-- Contract Section -->
<div class="contract-section">
<h3>FIFANEKENA ARA-BAROTRA (Réservations)</h3>
@ -1858,7 +1892,13 @@ private function generatePrintableInvoiceHTML($avance, $productName, $productDet
</div>
</div>
<!-- Signatures -->
<!-- Additional space for notes -->
<div style="margin-top: 20px; padding: 10px; border: 1px solid #000; flex: 1;">
<strong>OBSERVATIONS / NOTES:</strong>
<div style="height: 100px; margin-top: 10px;"></div>
</div>
<!-- Signatures for verso -->
<div class="signature-section">
<div class="signature-box">
<p>NY MPAMANDRIKA</p>
@ -1873,11 +1913,12 @@ private function generatePrintableInvoiceHTML($avance, $productName, $productDet
</div>
</div>
</div>
</div>
</div>
</body>
</html>
HTML;
}
/**
* ✅ NOUVELLE MÉTHODE : Traiter manuellement les avances expirées
* URL: /avances/processExpiredAvances

211
app/Controllers/OrderController.php

@ -341,6 +341,7 @@ class OrderController extends AdminController
$posts = $this->request->getPost('product[]');
$rates = $this->request->getPost('rate_value[]');
$amounts = $this->request->getPost('amount_value[]');
$puissances = $this->request->getPost('puissance[]'); // ✅ AJOUTER CETTE LIGNE
$discount = (float)$this->request->getPost('discount') ?? 0;
$gross_amount = $this->calculGross($amounts);
@ -400,13 +401,14 @@ class OrderController extends AdminController
'service_charge_rate' => 0,
'vat_charge_rate' => 0,
'vat_charge' => 0,
'net_amount' => $net_amount, // ✅ Net amount = tranches OU montant_a_payer
'net_amount' => $net_amount,
'discount' => $discount,
'paid_status' => 2, // Toujours En Attente au départ
'paid_status' => 2,
'user_id' => $user_id,
'amount_value' => $amounts,
'gross_amount' => $gross_amount,
'rate_value' => $rates,
'puissance' => $puissances, // ✅ AJOUTER CETTE LIGNE
'store_id' => $users['store_id'],
'tranche_1' => $tranche_1,
'tranche_2' => $tranche_2,
@ -586,8 +588,8 @@ public function markAsDelivered()
}
public function getProductValueById()
{
public function getProductValueById()
{
$product_id = $this->request->getPost('product_id');
if ($product_id) {
$Products = new \App\Models\Products();
@ -596,39 +598,77 @@ public function markAsDelivered()
$FourchettePrix = new \App\Models\FourchettePrix();
$fourchette = $FourchettePrix->where('product_id', $product_id)->first();
// ✅ Extraire la puissance - Plusieurs méthodes
$puissance_extracted = '';
// Méthode 1: Chercher dans le champ puissance de la BD
if (!empty($product_data['puissance'])) {
$puissance_extracted = $product_data['puissance'];
}
// Méthode 2: Chercher à la fin du nom (format: |150)
if (empty($puissance_extracted) && !empty($product_data['name'])) {
if (preg_match('/\|(\d+)(?:cc)?$/i', $product_data['name'], $matches)) {
$puissance_extracted = $matches[1];
}
}
// Méthode 3: Chercher n'importe où dans le nom (format: 150cc ou 150 CC)
if (empty($puissance_extracted) && !empty($product_data['name'])) {
if (preg_match('/(\d+)\s*cc/i', $product_data['name'], $matches)) {
$puissance_extracted = $matches[1];
}
}
// Log pour debug (à retirer en production)
log_message('info', 'Product ID: ' . $product_id . ' - Puissance: ' . $puissance_extracted);
$response = [
'id' => $product_data['id'],
'sku' => $product_data['sku'],
'name' => $product_data['name'],
'prix_vente' => $product_data['prix_vente'],
'prix_minimal' => $fourchette ? $fourchette['prix_minimal'] : 0,
'puissance' => $puissance_extracted,
'numero_de_moteur' => $product_data['numero_de_moteur'] ?? '',
];
return $this->response->setJSON($response);
}
}
public function getTableProductRow()
{
return $this->response->setJSON(['error' => 'Product ID missing']);
}
public function getTableProductRow()
{
$Products = new Products();
$session = session();
$users = $session->get('user');
$store_id = $users['store_id'];
$product_data = $Products->getProductData2($store_id);
die(var_dump($product_data));
return $this->response->setJSON($product_data);
// ✅ Nettoyer les données pour ne pas afficher la puissance dans le select
foreach ($product_data as &$product) {
// Extraire la puissance si elle est dans le nom
if (!empty($product['name']) && preg_match('/\|(\d+)(?:cc)?$/i', $product['name'], $matches)) {
// Stocker la puissance séparément
if (empty($product['puissance'])) {
$product['puissance'] = $matches[1];
}
}
}
return $this->response->setJSON($product_data);
}
public function update(int $id)
{
{
$this->verifyRole('updateOrder');
$data['page_title'] = $this->pageTitle;
$validation = \Config\Services::validation();
// ✅ NOUVELLE VÉRIFICATION : Bloquer UNIQUEMENT si statut = Validé (1) ou Validé et Livré (3)
$Orders = new Orders();
$current_order = $Orders->getOrdersData($id);
@ -637,14 +677,11 @@ public function markAsDelivered()
return redirect()->to('orders/');
}
// ✅ Bloquer UNIQUEMENT les statuts 1 (Validé) et 3 (Validé et Livré)
// Le statut 0 (Refusé) et 2 (En Attente) restent modifiables
if (in_array($current_order['paid_status'], [1, 3])) {
session()->setFlashData('errors', 'Cette commande ne peut plus être modifiée car elle est déjà validée ou livrée.');
return redirect()->to('orders/');
}
// Règles de validation
$validation->setRules([
'product' => 'required'
]);
@ -663,7 +700,6 @@ public function markAsDelivered()
if ($this->request->getMethod() === 'post' && $validation->run($validationData)) {
// ✅ DOUBLE VÉRIFICATION avant l'update
$current_order_check = $Orders->getOrdersData($id);
if (in_array($current_order_check['paid_status'], [1, 3])) {
session()->setFlashData('errors', 'Cette commande ne peut plus être modifiée car elle est déjà validée ou livrée.');
@ -672,14 +708,12 @@ public function markAsDelivered()
$old_paid_status = $current_order['paid_status'];
// ✅ Statut payé pour COMMERCIALE reste toujours "En attente"
if ($role === 'COMMERCIALE') {
$paid_status = 2;
} else {
$paid_status = $this->request->getPost('paid_status');
}
// Gestion remise
$discount = $this->request->getPost('discount');
$original_discount = $this->request->getPost('original_discount');
if ($discount === '' || $discount === null) {
@ -703,6 +737,7 @@ public function markAsDelivered()
'product_sold' => true,
'rate_value' => $this->request->getPost('rate_value'),
'amount_value' => $this->request->getPost('amount_value'),
'puissance' => $this->request->getPost('puissance'), // ✅ AJOUT PUISSANCE
'tranche_1' => $role !== 'COMMERCIALE' ? $this->request->getPost('tranche_1') : null,
'tranche_2' => $role !== 'COMMERCIALE' ? $this->request->getPost('tranche_2') : null,
'order_payment_mode' => $role !== 'COMMERCIALE' ? $this->request->getPost('order_payment_mode_1') : null,
@ -715,7 +750,6 @@ public function markAsDelivered()
$Notification = new NotificationController();
// Notification pour la sécurité si commande validée
if ($old_paid_status == 2 && $paid_status == 1) {
$customer_name = $this->request->getPost('customer_name');
$bill_no = $current_order['bill_no'];
@ -728,7 +762,6 @@ public function markAsDelivered()
);
}
// Gestion demande de remise
if ((float)$discount > 0) {
$productData = new Products();
$product_data_results = [];
@ -775,7 +808,6 @@ public function markAsDelivered()
}
}
// Si GET ou validation échoue
$company = $Company->getCompanyData(1);
$data['company_data'] = $company;
$data['is_vat_enabled'] = ($company['vat_charge_value'] > 0);
@ -783,10 +815,8 @@ public function markAsDelivered()
$orders_data = $Orders->getOrdersData($id);
// ✅ Ajouter un flag pour désactiver le formulaire UNIQUEMENT pour les statuts 1 et 3
$data['is_editable'] = !in_array($orders_data['paid_status'], [1, 3]);
// Montant tranches
$orders_data['montant_tranches'] = (!empty($orders_data['discount']) && $orders_data['discount'] > 0)
? $orders_data['discount']
: $orders_data['gross_amount'];
@ -803,8 +833,7 @@ public function markAsDelivered()
$data['validation'] = $validation;
return $this->render_template('orders/edit', $data);
}
}
public function lookOrder(int $id)
{
@ -1027,10 +1056,14 @@ public function markAsDelivered()
foreach ($orders_items as $item) {
$product_data = $Products->getProductData($item['product_id']);
// ✅ PRIORITÉ À LA PUISSANCE DE LA COMMANDE
$puissance_affichee = !empty($item['puissance']) ? $item['puissance'] : ($product_data['puissance'] ?? '');
$html .= '<tr>
<td>' . esc($product_data['sku']) . '</td>
<td>' . esc($product_data['numero_de_moteur']) . '</td>
<td><input type="text" name="customField" placeholder="Remplir ici" style="border: none; outline: none; width: 100%;" value="' . esc($product_data['puissance']) . '"></td>
<td><input type="text" value="' . esc($puissance_affichee) . '"></td>
<td style="text-align:right;">' . number_format((float) $item['amount'], 2, '.', ' ') . '</td>
</tr>';
}
@ -1104,7 +1137,7 @@ public function markAsDelivered()
}
public function print(int $id)
{
{
$this->verifyRole('viewOrder');
if ($id) {
@ -1124,9 +1157,12 @@ public function markAsDelivered()
foreach ($orders_items as $index => $item) {
$product_data = $Products->getProductData($item['product_id']);
// ✅ PRIORITÉ À LA PUISSANCE DE LA COMMANDE (si modifiée), sinon celle du produit
$puissance_affichee = !empty($item['puissance']) ? $item['puissance'] : ($product_data['puissance'] ?? '');
echo '<!DOCTYPE html>
<html lang="fr">
<head>
<html lang="fr">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="' . base_url('assets/bower_components/bootstrap/dist/css/bootstrap.min.css') . '">
@ -1184,8 +1220,8 @@ public function markAsDelivered()
}
}
</style>
</head>
<body onload="window.print()">
</head>
<body onload="window.print()">
<div class="invoice-container">
<div class="invoice-header">' . esc($company_info['company_name']) . '</div>
@ -1224,7 +1260,7 @@ public function markAsDelivered()
<tr>
<td>' . esc($product_data['sku']) . '</td>
<td>' . esc($product_data['numero_de_moteur']) . '</td>
<td><input type="text" value="' . esc($product_data['puissance']) . '" placeholder="Remplir ici"></td>
<td><input type="text" value="' . esc($puissance_affichee) . '" placeholder="Remplir ici"></td>
<td style="text-align:right;">' . number_format((float)$item['amount'], 2, '.', ' ') . '</td>
</tr>
</tbody>
@ -1276,12 +1312,11 @@ public function markAsDelivered()
<strong style="color:white;">Original</strong>
</div>
</div>
</body>
</html>';
</body>
</html>';
}
}
}
}
public function remove()
{
@ -1347,30 +1382,26 @@ public function markAsDelivered()
}
public function print3(int $id)
{
// Vérification du rôle
public function print3(int $id)
{
$this->verifyRole('viewOrder');
if (!$id) {
return $this->response->setStatusCode(400, 'ID manquant');
}
// Instanciation des modèles
$Orders = new Orders();
$Company = new Company();
$OrderItems = new OrderItems();
$Products = new Products();
// Récupération des données
$order_data = $Orders->getOrdersData($id);
$items = $OrderItems->getOrdersItemData($id);
$company_info = $Company->getCompanyData(1);
// Statut de paiement
$paid_status = $order_data['paid_status'] == 1
? "<span style='color: green; font-weight: bold;'>Payé</span>"
: "<span style='color: red; font-weight: bold;'>Non payé</span>";
? "<span style='color: green; font-weight: bold;'>Validé</span>"
: "<span style='color: red; font-weight: bold;'>Refusé</span>";
// STYLE COMMUN
$style = '
@ -1386,24 +1417,27 @@ public function print3(int $id)
table { width:100%; border-collapse:collapse; margin-top:15px; }
th, td { border:1px solid #ddd; padding:8px; text-align:left; }
th { background:#e9ecef; }
.summary {
width: 80%;
margin: 20px 0;
margin-right: 0;
}
input { border:none; outline:none; width:100%; font-size:14px; background:transparent; }
.summary { width: 80%; margin: 20px 0; margin-right: 0; }
.summary th, .summary td { border:none; padding:4px; text-align:right; }
.summary th { text-align:left; }
@media print {
body { font-size: 14px; font-family: Arial, sans-serif, margin: 1cm; }
body { font-size: 14px; font-family: Arial, sans-serif; margin: 1cm; }
@page { margin: 0; }
* { -webkit-print-color-adjust:exact; print-color-adjust:exact; } }
* { -webkit-print-color-adjust:exact; print-color-adjust:exact; }
}
</style>
';
// --- FACTURES : Une par produit ---
foreach ($items as $item) {
$product_data = $Products->getProductData($item['product_id']);
// ✅ PRIORITÉ À LA PUISSANCE DE LA COMMANDE
$puissance_affichee = !empty($item['puissance'])
? $item['puissance']
: (!empty($product_data['puissance']) ? $product_data['puissance'] : '');
$unitPrice = (float)$item['amount'];
$quantity = isset($item['qty']) ? (int)$item['qty'] : 1;
$subtotal = $unitPrice * $quantity;
@ -1438,34 +1472,34 @@ public function print3(int $id)
</div>
<table>
<thead><tr><th>Marque</th><th>Moteur</th><th>Puissance</th><th>Prix unitaire</th></tr></thead>
<thead><tr><th>Marque</th><th>Moteur</th><th>Puissance (CC)</th><th>Prix unitaire</th></tr></thead>
<tbody>
<tr>
<td>' . esc($product_data['sku']) . '</td>
<td>' . esc($product_data['numero_de_moteur']) . '</td>
<td>' . esc($product_data['puissance']) . '</td>
<td><input type="text" value="' . esc($puissance_affichee) . '" readonly></td>
<td>' . number_format($unitPrice, 2, '.', ' ') . '</td>
</tr>
</tbody>
</table>
<table class="table summary">
<tr><th>Total:</th><td></td></tr>
<tr><th>TVA (20%):</th><td></td></tr>
<tr><th>Réduction:</th><td></td></tr>
<tr><th>Total à payer:</th><td><strong></strong></td></tr>
<tr><th>Statut:</th><td></td></tr>';
<tr><th>Total HT:</th><td>' . number_format($subtotal, 2, '.', ' ') . ' Ar</td></tr>
<tr><th>TVA (20%):</th><td>' . number_format($vatAmount, 2, '.', ' ') . ' Ar</td></tr>
<tr><th>Réduction:</th><td>' . number_format($discount, 2, '.', ' ') . ' Ar</td></tr>
<tr><th>Total à payer:</th><td><strong>' . number_format($totalNet, 2, '.', ' ') . ' Ar</strong></td></tr>
<tr><th>Statut:</th><td>' . $paid_status . '</td></tr>';
if (!empty($order_data['order_payment_mode'])) {
echo '<tr><th>Mode de paiement:</th><td></td></tr>';
echo '<tr><th>Mode de paiement:</th><td>' . esc($order_data['order_payment_mode']) . '</td></tr>';
}
if (!empty($order_data['tranche_1'])) {
echo '<tr><th>Tranche 1:</th><td></td></tr>';
echo '<tr><th>Tranche 1:</th><td>' . number_format((float)$order_data['tranche_1'], 2, '.', ' ') . ' Ar</td></tr>';
}
if (!empty($order_data['tranche_2'])) {
echo '<tr><th>Tranche 2:</th><td></td></tr>';
echo '<tr><th>Tranche 2:</th><td>' . number_format((float)$order_data['tranche_2'], 2, '.', ' ') . ' Ar</td></tr>';
}
echo '</table>
@ -1513,7 +1547,7 @@ public function print3(int $id)
<tr>
<th>Marque</th>
<th>Moteur</th>
<th>Puissance</th>
<th>Puissance (CC)</th>
<th>Prix</th>
</tr>
</thead>
@ -1522,11 +1556,18 @@ public function print3(int $id)
$total_ht = 0;
foreach ($items as $item) {
$product_data = $Products->getProductData($item['product_id']);
// ✅ PRIORITÉ À LA PUISSANCE DE LA COMMANDE
$puissance_affichee = !empty($item['puissance'])
? $item['puissance']
: (!empty($product_data['puissance']) ? $product_data['puissance'] : '');
$total_ht += (float)$item['amount'];
echo '<tr>
<td>' . esc($product_data['sku']) . '</td>
<td>' . esc($product_data['numero_de_moteur']) . '</td>
<td><input type="text" value="' . esc($product_data['puissance']) . '" style="border:none;width:100%;outline:none;"></td>
<td><input type="text" value="' . esc($puissance_affichee) . '" style="border:none;width:100%;outline:none;"></td>
<td style="text-align:right;">' . number_format((float)$item['amount'], 2, '.', ' ') . '</td>
</tr>';
}
@ -1534,30 +1575,29 @@ public function print3(int $id)
$tva = $total_ht * 0.2;
$total_ttc = $total_ht + $tva - $order_data['discount'];
echo '</tbody>';
echo ' </table>
echo '</tbody>
</table>
<table class="table summary">
<tr><th>Total:</th><td>' . number_format($total_ht, 2, '.', ' ') . '</td></tr>
<tr><th>TVA:</th><td>' . number_format($tva, 2, '.', ' ') . '</td></tr>
<tr><th>Réduction:</th><td>' . number_format($order_data['discount'], 2, '.', ' ') . '</td></tr>
<tr><th>Total à payer:</th><td><strong>' . number_format($total_ttc, 2, '.', ' ') . '</strong></td></tr>
</table>';
<tr><th>Total HT:</th><td>' . number_format($total_ht, 2, '.', ' ') . ' Ar</td></tr>
<tr><th>TVA (20%):</th><td>' . number_format($tva, 2, '.', ' ') . ' Ar</td></tr>
<tr><th>Réduction:</th><td>' . number_format($order_data['discount'], 2, '.', ' ') . ' Ar</td></tr>
<tr><th>Total à payer:</th><td><strong>' . number_format($total_ttc, 2, '.', ' ') . ' Ar</strong></td></tr>
</table>
echo '<div style="display: flex; justify-content: space-around; margin-top: 5%;">
<div style="display: flex; justify-content: space-around; margin-top: 5%;">
<div><p style="font-weight:bold;">L\'acheteur</p></div>
<div><p style="font-weight:bold;">Le vendeur</p></div>
</div>';
</div>
echo'<div class="invoice-footer">
<div class="invoice-footer">
Merci pour votre commande !<br>
<strong style="color:white;">Original</strong>
</div>';
echo '</div>';
echo '</body>';
echo'</html>';
}
</div>
</div>
</body>
</html>';
}
// PRINT5 - Facture détaillée avec conditions générales
// ====================================
public function print5(int $id)
@ -1664,9 +1704,6 @@ public function print5(int $id)
$prixAffiche = ($discount > 0) ? $discount : $details['prix'];
$html .= '
<tr>
<td>'.esc($details['product_name']);
// Afficher le commentaire s'il existe
if (!empty($details['commentaire'])) {
@ -1703,13 +1740,14 @@ public function print5(int $id)
$prixAffiche = ($discount > 0) ? $discount : $details['prix'];
$html .= '
<tr>
<td>'.esc($details['marque']).'</td>
<td>'.esc($details['product_name']).'</td>
<td>'.esc($details['numero_moteur']).'</td>
<td>'.esc($details['numero_chassis']).'</td>
<td>'.esc($details['puissance']).'</td>
<td>'.esc($details['puissance']).'</td> <!-- ✅ ICI -->
<td class="right">'.number_format($prixAffiche, 0, '', ' ').'</td>
</tr>';
}
@ -1824,7 +1862,7 @@ private function numberToWords(int $num): string
// millions et plus
$m = intdiv($n, 1000000);
$rest = $n % 1000000;
$millionText = $m > 1 ? $convert($m) . ' million' : 'million';
$millionText = $m > 1 ? $convert($m) . ' million' : 'un million';
// pas de 's' à million en francais
return $millionText . ($rest ? ' ' . $convert($rest) : '');
};
@ -1833,7 +1871,6 @@ private function numberToWords(int $num): string
$words = $convert($num);
return trim($words) . ' ariary';
}
/**
* ✅ NOUVELLE MÉTHODE : Vérifier si une commande provient d'une avance "sur mer"
*/
@ -1908,7 +1945,7 @@ private function getOrderItemDetails($item)
'marque' => $brandName,
'numero_moteur' => $product['numero_de_moteur'] ?? '',
'numero_chassis' => $product['chasis'] ?? $product['sku'] ?? '',
'puissance' => $product['puissance'] ?? '',
'puissance' => !empty($item['puissance']) ? $item['puissance'] : ($product['puissance'] ?? ''), // ✅ Priorité à la commande
'commentaire' => '',
'prix' => (float)$item['amount']
];

31
app/Models/OrderItems.php

@ -5,35 +5,23 @@ namespace App\Models;
use CodeIgniter\Model;
use DateTime;
/**
* table pivot
*/
class OrderItems extends Model
{
/**
* table name
* @var string
*/
protected $table = 'orders_item';
protected $allowedFields = ['order_id', 'product_id', 'qty', 'rate' , 'amount' ];
protected $allowedFields = ['order_id', 'product_id', 'puissance', 'rate', 'amount'];
public function insertOrderItem($data)
{
return $this->insert($data);
}
/**
* get the orders item data
* @param mixed $order_id
* @return array|bool
*/
public function getOrdersItemData($order_id = null)
{
if (!$order_id) {
return false;
}
return $this->where('order_id', $order_id)->findAll(); // Get items of a specific order
return $this->where('order_id', $order_id)->findAll();
}
public function getAllSoldProductToday() {
@ -57,15 +45,14 @@ class OrderItems extends Model
}
public function updateOrderItem(int $id, array $data)
{
{
return $this->where('order_id', $id)
->set($data)
->update();
}
}
public function getProductIds(array $orderIds): array
{
public function getProductIds(array $orderIds): array
{
if (empty($orderIds)) {
return [];
}
@ -74,10 +61,6 @@ public function getProductIds(array $orderIds): array
->whereIn('order_id', $orderIds)
->findAll();
// Extrait la colonne product_id
return array_column($items, 'product_id');
}
}
}

25
app/Models/Orders.php

@ -196,47 +196,43 @@ class Orders extends Model
$orderItemModel = new OrderItems();
$productModel = new Products();
// ✅ RÉCUPÉRER LE TABLEAU DES PUISSANCES
$puissances = $data['puissance'] ?? [];
// Loop through products and insert order items
$count_product = count($post);
for ($x = 0; $x < $count_product; $x++) {
// ✅ AJOUT DE LA PUISSANCE
$items = [
'order_id' => $order_id,
'product_id' => $post[$x],
'rate' => $data['rate_value'][$x],
'qty' =>1,
'qty' => 1,
'amount' => $data['amount_value'][$x],
'puissance' => $puissances[$x] ?? 1, // ✅ CORRECTION ICI
];
$orderItemModel->insert($items);
// Insert order item
// // Decrease stock for the product
// Decrease stock for the product
$product_data = $productModel->find($post[$x]);
if((int)$data['paid_status'] == 1){
if ($product_data) {
$product_sold = true;
// Update product stock
$productModel->update($post[$x], ['product_sold' => $product_sold]);
}
else{
} else {
$product_sold = false;
$productModel->update($post[$x], ['product_sold' => $product_sold]);
}
}
}
return $order_id; // Return order ID if everything is successful
return $order_id;
} catch (\Exception $e) {
// Log the exception for debugging (optional)
log_message('error', 'Error creating order: ' . $e->getMessage());
return false; // Return false if an error occurs
return false;
}
}
/**
* count order item
* @param int $order_id
@ -284,6 +280,7 @@ class Orders extends Model
'order_id' => $id,
'product_id' => $data['product'][$x],
'rate' => $data['rate_value'][$x],
'puissance' => $data['puissance'][$x],
'amount' => $data['amount_value'][$x],
];

80
app/Views/orders/createbyid.php

@ -114,12 +114,16 @@
</div>
</div>
<br /> <br />
<!-- ✅ TABLEAU AVEC COLONNE PUISSANCE -->
<table class="table table-bordered" id="product_info_table">
<thead>
<tr>
<th style="width:50%">Produit</th>
<th style="width:10%">Prix unitaire</th>
<th style="width:40%">Produit</th>
<th style="width:15%">Puissance (CC)</th>
<th style="width:15%">Prix unitaire</th>
<th style="width:20%">Montant</th>
<th style="width:10%"></th>
</tr>
</thead>
<tbody>
@ -129,31 +133,46 @@
id="product_1" name="product[]" style="width:100%;"
onchange="getProductData(1)" required>
<option value="<?= $products['id'] ?>">
<?= $products['sku'] . ' | ' . $products['name'] . ' | ' . $products['numero_de_moteur'] . ' | ' . $products['puissance'] ?>
<?= $products['sku'] . ' | ' . $products['name'] . ' | ' . $products['numero_de_moteur'] ?>
</option>
</select>
</td>
<!-- ✅ COLONNE PUISSANCE -->
<td>
<input type="number" name="puissance[]" id="puissance_1"
class="form-control" placeholder="CC"
value="<?= esc($products['puissance'] ?? '1') ?>"
min="1" autocomplete="off">
</td>
<td>
<input type="text" name="rate[]" id="rate_1" class="form-control" disabled
autocomplete="off" value="<?= $pu ?>" min="0">
<input type="hidden" name="rate_value[]" value="<?= $pu ?>"
id="rate_value_1" class="form-control" autocomplete="off">
autocomplete="off" value="<?= esc($pu) ?>" min="0">
<input type="hidden" name="rate_value[]" id="rate_value_1"
value="<?= esc($pu) ?>" class="form-control" autocomplete="off">
<input type="hidden" id="min_price_1" name="min_price[]" value="">
</td>
<td>
<input type="text" name="amount[]" id="amount_1" class="form-control"
disabled autocomplete="off" min="0">
<input type="hidden" name="amount_value[]" id="amount_value_1"
class="form-control" autocomplete="off">
</td>
<td><button type="button" class="btn btn-default" onclick="removeRow('1')"><i
class="fa fa-close"></i></button></td>
<td>
<button type="button" class="btn btn-default" onclick="removeRow('1')">
<i class="fa fa-close"></i>
</button>
</td>
</tr>
</tbody>
</table>
<input type="hidden" name="qty[]" value="1" id="qty_1" min="1"
class="form-control valueqty" required onkeyup="getTotal(1)">
<!-- ✅ Champ qty caché (toujours = 1) -->
<input type="hidden" name="qty[]" value="1" id="qty_1" class="form-control">
<br /> <br />
<div class="col-md-6 col-xs-12 pull pull-right">
<div class="form-group">
@ -186,6 +205,7 @@
</div>
</div>
</div>
<div class="box-footer">
<input type="hidden" name="service_charge_rate"
value="<?php echo $company_data['service_charge_value'] ?>" autocomplete="off">
@ -210,7 +230,6 @@
function validatePositiveNumber(input) {
let value = parseFloat(input.value);
// Si la valeur est négative ou NaN, on la remet à 0
if (isNaN(value) || value < 0) {
input.value = '';
return false;
@ -236,7 +255,6 @@
let prixDemande = parseFloat(input.value);
let prixAffiche = parseFloat($('#gross_amount').val()) || 0;
// Vérifier si négatif
if (isNaN(prixDemande) || prixDemande < 0) {
input.value = '';
showDiscountError('Le prix demandé ne peut pas être négatif.');
@ -244,7 +262,6 @@
return false;
}
// Vérifier si supérieur au prix affiché
if (prixDemande > prixAffiche) {
showDiscountError('Le prix demandé (' + prixDemande.toFixed(2) + ') ne peut pas être supérieur au prix affiché (' + prixAffiche.toFixed(2) + ').');
input.value = prixAffiche;
@ -253,14 +270,12 @@
return false;
}
// Si tout est valide, masquer l'erreur
hideDiscountError();
return true;
}
// ✅ Fonction pour empêcher la saisie de caractères négatifs
function preventNegativeInput(e) {
// Empêche la saisie du signe moins (-)
if (e.key === '-' || e.key === 'e' || e.key === 'E' || e.key === '+') {
e.preventDefault();
return false;
@ -268,7 +283,12 @@
}
$(document).ready(function () {
// ✅ INITIALISATION : Calculer le total au chargement
getTotal(1);
// ✅ INITIALISATION : S'assurer que rate_value_1 est bien rempli
$("#rate_value_1").val($("#rate_1").val());
$(".select_group").select2();
$("#mainOrdersNav").addClass('active');
@ -295,7 +315,6 @@
// Bloquer la soumission du formulaire
$('form').on('submit', function(e) {
// Vérifier tous les champs numériques avant soumission
let hasNegative = false;
$('.numeric-input, input[type="number"]').each(function() {
if (parseFloat($(this).val()) < 0) {
@ -332,11 +351,14 @@
'<select class="form-control select_group product" data-row-id="' + row_id + '" id="product_' + row_id + '" name="product[]" style="width:100%;" onchange="getProductData(' + row_id + ')">' +
'<option value=""></option>';
$.each(response, function (index, value) {
html += '<option value="' + value.id + '">' + value.sku + ' | ' + value.name + ' | ' + value.numero_de_moteur + ' | ' + value.puissance + '</option>';
// ✅ NE PLUS AFFICHER LA PUISSANCE DANS LE SELECT
html += '<option value="' + value.id + '">' + value.sku + ' | ' + value.name + ' | ' + value.numero_de_moteur + '</option>';
});
html += '</select>' +
'</td>' +
// ✅ COLONNE PUISSANCE
'<td><input type="number" name="puissance[]" id="puissance_' + row_id + '" class="form-control" placeholder="CC" min="1" value="1" autocomplete="off"></td>' +
'<td><input type="text" name="rate[]" id="rate_' + row_id + '" class="form-control numeric-input" disabled min="0"><input type="hidden" name="rate_value[]" id="rate_value_' + row_id + '" class="form-control"><input type="hidden" id="min_price_' + row_id + '" name="min_price[]" value=""></td>' +
'<td><input type="text" name="amount[]" id="amount_' + row_id + '" class="form-control numeric-input" disabled min="0"><input type="hidden" name="amount_value[]" id="amount_value_' + row_id + '" class="form-control"></td>' +
'<td><button type="button" class="btn btn-default" onclick="removeRow(\'' + row_id + '\')"><i class="fa fa-close"></i></button></td>' +
@ -362,7 +384,6 @@
var rate = Number($("#rate_value_" + row).val());
var qty = Number($("#qty_" + row).val());
// ✅ Vérifier que les valeurs sont positives
if (rate < 0) rate = 0;
if (qty < 0) qty = 0;
@ -378,14 +399,14 @@
}
}
// get the product information from the server
// ✅ GET PRODUCT DATA - Récupère prix + puissance
function getProductData(row_id) {
var product_id = $("#product_" + row_id).val();
if (product_id == "") {
$("#rate_" + row_id).val("");
$("#rate_value_" + row_id).val("");
$("#min_price_" + row_id).val("");
$("#qty_" + row_id).val("");
$("#puissance_" + row_id).val("1");
$("#amount_" + row_id).val("");
$("#amount_value_" + row_id).val("");
@ -398,7 +419,8 @@
},
dataType: 'json',
success: function (response) {
// ✅ S'assurer que les prix ne sont pas négatifs
console.log('✅ Response:', response); // Debug
var prixVente = parseFloat(response.prix_vente) || 0;
var prixMinimal = parseFloat(response.prix_minimal) || 0;
@ -409,8 +431,12 @@
$("#rate_value_" + row_id).val(prixVente);
$("#min_price_" + row_id).val(prixMinimal);
// ✅ REMPLIR LA PUISSANCE
var puissanceValue = response.puissance || '1';
console.log('✅ Puissance extraite:', puissanceValue); // Debug
$("#puissance_" + row_id).val(puissanceValue);
$("#qty_" + row_id).val(1);
$("#qty_value_" + row_id).val(1);
var total = prixVente * 1;
total = total.toFixed(2);
@ -436,7 +462,6 @@
count = count.substring(4);
var amount = Number($("#amount_" + count).val());
// ✅ Vérifier que le montant n'est pas négatif
if (amount < 0) amount = 0;
totalSubAmount = Number(totalSubAmount) + amount;
}
@ -464,7 +489,6 @@
totalAmount = totalAmount.toFixed(2);
var discount = Number($("#discount").val()) || 0;
// ✅ S'assurer que le rabais n'est pas négatif
if (discount < 0) {
discount = 0;
$("#discount").val('');
@ -472,7 +496,6 @@
if (discount) {
var grandTotal = Number(totalAmount) - Number(discount);
// ✅ S'assurer que le total n'est pas négatif
if (grandTotal < 0) grandTotal = 0;
grandTotal = grandTotal.toFixed(2);
$("#net_amount").val(grandTotal);
@ -487,7 +510,6 @@
function checkMinimalPrice() {
var discount = Number($("#discount").val()) || 0;
// ✅ Vérifier que le rabais n'est pas négatif
if (discount < 0) {
alert("Le prix demandé ne peut pas être négatif.");
$("#discount").val('');
@ -495,7 +517,6 @@
return false;
}
// Si pas de rabais, pas de vérification nécessaire
if (discount === 0) return true;
var tableProductLength = $("#product_info_table tbody tr").length;
@ -507,7 +528,6 @@
var rowId = $(tr).attr('id').replace('row_', '');
var minPrice = Number($("#min_price_" + rowId).val()) || 0;
// ✅ Le rabais devient le prix de vente
if (minPrice > 0 && discount < minPrice) {
error = true;
var productText = $("#product_" + rowId + " option:selected").text();
@ -529,7 +549,7 @@
subAmount();
}
// new area of javascript in the order
// ✅ INITIALISATION AU CHARGEMENT
const quantity = document.getElementById('qty_1')
const rates = document.getElementById('rate_1')
const amount = document.getElementById('amount_1')
@ -549,13 +569,11 @@
net_amount_value.value = (gross_amount_value.value - remise.value);
remise.addEventListener('input', function () {
// ✅ Validation du prix demandé vs prix affiché
if (validatePrixDemande(this)) {
var discountValue = parseFloat(this.value) || 0;
var grossValue = parseFloat(gross_amount_value.value) || 0;
var netValue = grossValue - discountValue;
// ✅ S'assurer que le net amount n'est pas négatif
if (netValue < 0) netValue = 0;
net_amount.value = netValue.toFixed(2);

468
app/Views/orders/edit.php

@ -1,6 +1,4 @@
<!-- Content Wrapper. Contains page content -->
<div class="content-wrapper">
<!-- Content Header (Page header) -->
<section class="content-header">
<h1>
Gérer les
@ -12,9 +10,7 @@
</ol>
</section>
<!-- Main content -->
<section class="content">
<!-- Small boxes (Stat box) -->
<div class="row">
<div class="col-md-12 col-xs-12">
@ -53,37 +49,12 @@
</div>
<?php endif; ?>
<?php
// ✅ VÉRIFIER SI LA COMMANDE EST MODIFIABLE
// Seuls les statuts 1 (Validé) et 3 (Validé et Livré) sont NON modifiables
$is_editable = isset($is_editable) ? $is_editable : true;
$paid_status = $order_data['order']['paid_status'] ?? 2;
?>
<?php if (!$is_editable): ?>
<!-- ✅ ALERTE SI NON MODIFIABLE -->
<div class="alert alert-warning alert-dismissible" role="alert">
<button type="button" class="close" data-dismiss="alert" aria-label="Close">
<span aria-hidden="true">&times;</span>
</button>
<strong><i class="fa fa-lock"></i> Cette commande ne peut plus être modifiée</strong><br>
Elle a été <?php echo ($paid_status == 1) ? 'validée' : 'validée et livrée'; ?>.
Vous pouvez uniquement consulter les informations.
</div>
<?php endif; ?>
<div class="box">
<div class="box-header">
<h3 class="box-title">
<?php echo $is_editable ? 'Mise à jour de commande' : 'Consultation de commande'; ?>
</h3>
</div>
<!-- /.box-header -->
<form role="form"
action="<?php base_url('orders/create') ?>"
method="post"
class="form-horizontal"
<?php echo !$is_editable ? 'onsubmit="return false;"' : ''; ?>>
<h3 class="box-title">Mise à jour de commande</h3>
</div>
<form role="form" action="<?php base_url('orders/create') ?>" method="post" class="form-horizontal">
<div class="box-body">
<div class="form-group">
@ -98,7 +69,7 @@
<div class="form-group">
<label for="types" class="col-sm-5 control-label" style="text-align:left;">Types</label>
<div class="col-sm-7">
<select name="" id="typesCommande" class="form-control" <?php echo !$is_editable ? 'disabled' : ''; ?>>
<select name="" id="typesCommande" class="form-control">
<option value="1">Facture</option>
<option value="2">Bon de Livraison & Facture</option>
<option value="3">Bon de Livraison</option>
@ -109,56 +80,28 @@
<div class="form-group">
<label for="customer_name" class="col-sm-5 control-label" style="text-align:left;">Nom du client</label>
<div class="col-sm-7">
<input type="text"
class="form-control"
id="customer_name"
name="customer_name"
placeholder="Enter Customer Name"
value="<?php echo $order_data['order']['customer_name'] ?>"
autocomplete="off"
<?php echo !$is_editable ? 'disabled readonly' : ''; ?> />
<input type="text" class="form-control" id="customer_name" name="customer_name" placeholder="Enter Customer Name" value="<?php echo $order_data['order']['customer_name'] ?>" autocomplete="off" />
</div>
</div>
<div class="form-group">
<label for="gross_amount" class="col-sm-5 control-label" style="text-align:left;">Adresse du client</label>
<div class="col-sm-7">
<input type="text"
class="form-control"
id="customer_address"
name="customer_address"
placeholder="Enter Customer Address"
value="<?php echo $order_data['order']['customer_address'] ?>"
autocomplete="off"
<?php echo !$is_editable ? 'disabled readonly' : ''; ?>>
<input type="text" class="form-control" id="customer_address" name="customer_address" placeholder="Enter Customer Address" value="<?php echo $order_data['order']['customer_address'] ?>" autocomplete="off">
</div>
</div>
<div class="form-group">
<label for="gross_amount" class="col-sm-5 control-label" style="text-align:left;">Téléphone du client</label>
<div class="col-sm-7">
<input type="text"
class="form-control"
id="customer_phone"
name="customer_phone"
placeholder="Enter Customer Phone"
value="<?php echo $order_data['order']['customer_phone'] ?>"
autocomplete="off"
<?php echo !$is_editable ? 'disabled readonly' : ''; ?>>
<input type="text" class="form-control" id="customer_phone" name="customer_phone" placeholder="Enter Customer Phone" value="<?php echo $order_data['order']['customer_phone'] ?>" autocomplete="off">
</div>
</div>
<div class="form-group">
<label for="gross_amount" class="col-sm-5 control-label" style="text-align:left;">CIN du client</label>
<div class="col-sm-7">
<input type="text"
class="form-control"
id="customer_cin"
name="customer_cin"
placeholder="Enter Customer CIN"
value="<?php echo $order_data['order']['customer_cin'] ?>"
autocomplete="off"
<?php echo !$is_editable ? 'disabled readonly' : ''; ?>>
<input type="text" class="form-control" id="customer_cin" name="customer_cin" placeholder="Enter Customer CIN" value="<?php echo $order_data['order']['customer_cin'] ?>" autocomplete="off">
</div>
</div>
</div>
@ -168,14 +111,10 @@
<thead>
<tr>
<th style="width:50%">Produit</th>
<th style="width:10%">Quantité</th>
<th style="width:15%">Puissance (CC)</th>
<th style="width:10%">Prix unitaire</th>
<th style="width:20%">Montant</th>
<th style="width:10%">
<?php if ($is_editable): ?>
<button type="button" id="add_row" class="btn btn-default"><i class="fa fa-plus"></i></button>
<?php endif; ?>
</th>
<th style="width:10%"><button type="button" id="add_row" class="btn btn-default"><i class="fa fa-plus"></i></button></th>
</tr>
</thead>
@ -185,71 +124,59 @@
<?php foreach ($order_data['order_item'] as $key => $val): ?>
<tr id="row_<?php echo $x; ?>">
<td>
<select class="form-control select_group product"
data-row-id="row_<?php echo $x; ?>"
id="product_<?php echo $x; ?>"
name="product[]"
style="width:100%;"
onchange="getProductData(<?php echo $x; ?>)"
<?php echo !$is_editable ? 'disabled' : 'required'; ?>>
<select class="form-control select_group product" data-row-id="row_<?php echo $x; ?>"
id="product_<?php echo $x; ?>" name="product[]" style="width:100%;"
onchange="getProductData(<?php echo $x; ?>)" required>
<option value=""></option>
<?php foreach ($products as $k => $v): ?>
<option value="<?php echo $v['id'] ?>" <?php if ($val['product_id'] == $v['id']) { echo "selected='selected'"; } ?>><?php echo $v['name'] ?></option>
<option value="<?php echo $v['id'] ?>"
<?php if ($val['product_id'] == $v['id']) { echo "selected='selected'"; } ?>>
<?php
echo $v['sku'] . ' | ' . $v['name'];
if (!empty($v['numero_de_moteur'])) {
echo ' | ' . $v['numero_de_moteur'];
}
?>
</option>
<?php endforeach ?>
</select>
</td>
<!-- ✅ COLONNE PUISSANCE VISIBLE -->
<td>
<input type="number"
name="qty[]"
id="qty_<?php echo $x; ?>"
class="form-control"
onkeyup="getTotal(<?php echo $x; ?>)"
value="<?php echo $val['qty'] ?>"
autocomplete="off"
<?php echo !$is_editable ? 'disabled readonly' : 'required'; ?>>
<input type="number" name="puissance[]" id="puissance_<?php echo $x; ?>"
class="form-control" placeholder="Puissance"
value="<?php echo esc($val['puissance'] ?? '1') ?>"
autocomplete="off" min="1">
</td>
<td>
<input type="text"
name="rate[]"
id="rate_<?php echo $x; ?>"
class="form-control"
disabled
value="<?php echo $val['rate'] ?>"
autocomplete="off">
<input type="hidden"
name="rate_value[]"
id="rate_value_<?php echo $x; ?>"
class="form-control"
value="<?php echo $val['rate'] ?>"
autocomplete="off">
<input type="text" name="rate[]" id="rate_<?php echo $x; ?>"
class="form-control" disabled
value="<?php echo esc($val['rate']) ?>" autocomplete="off">
<input type="hidden" name="rate_value[]" id="rate_value_<?php echo $x; ?>"
class="form-control" value="<?php echo esc($val['rate']) ?>" autocomplete="off">
</td>
<td>
<input type="text"
name="amount[]"
id="amount_<?php echo $x; ?>"
class="form-control"
disabled
value="<?php echo $val['amount'] ?>"
autocomplete="off">
<input type="hidden"
name="amount_value[]"
id="amount_value_<?php echo $x; ?>"
class="form-control"
value="<?php echo $val['amount'] ?>"
autocomplete="off">
<input type="text" name="amount[]" id="amount_<?php echo $x; ?>"
class="form-control" disabled
value="<?php echo esc($val['amount']) ?>" autocomplete="off">
<input type="hidden" name="amount_value[]" id="amount_value_<?php echo $x; ?>"
class="form-control" value="<?php echo esc($val['amount']) ?>" autocomplete="off">
</td>
<td>
<?php if ($is_editable): ?>
<button type="button" class="btn btn-default" onclick="removeRow('<?php echo $x; ?>')">
<button type="button" class="btn btn-default"
onclick="removeRow('<?php echo $x; ?>')">
<i class="fa fa-close"></i>
</button>
<?php endif; ?>
</td>
</tr>
<?php $x++; ?>
<?php endforeach; ?>
<?php endif; ?>
</tbody>
</tbody>
</table>
<br /> <br />
@ -259,29 +186,8 @@
<div class="form-group">
<label for="gross_amount" class="col-sm-5 control-label">Prix affiché</label>
<div class="col-sm-7">
<input type="text"
class="form-control"
id="gross_amount"
name="gross_amount"
disabled
value="<?php echo $order_data['order']['gross_amount'] ?>"
autocomplete="off">
<input type="hidden"
class="form-control"
id="gross_amount_value"
name="gross_amount_value"
value="<?php echo $order_data['order']['gross_amount'] ?>"
autocomplete="off">
</div>
</div>
<!-- ✅ ALERTE VISUELLE INTÉGRÉE -->
<div id="price_alert" style="display: none; margin-bottom: 15px;">
<div class="col-sm-offset-5 col-sm-7">
<div style="background-color: #f8d7da; border: 1px solid #f5c6cb; color: #721c24; padding: 10px 15px; border-radius: 4px; animation: slideDown 0.3s ease-out;">
<i class="fa fa-exclamation-triangle" style="margin-right: 8px;"></i>
<strong>Attention !</strong> <span id="price_alert_message"></span>
</div>
<input type="text" class="form-control" id="gross_amount" name="gross_amount" disabled value="<?php echo $order_data['order']['gross_amount'] ?>" autocomplete="off">
<input type="hidden" class="form-control" id="gross_amount_value" name="gross_amount_value" value="<?php echo $order_data['order']['gross_amount'] ?>" autocomplete="off">
</div>
</div>
@ -290,44 +196,19 @@
<div class="col-sm-7">
<?php
$users = session()->get('user');
if($users && $users['group_name'] == 'COMMERCIALE' && $is_editable):
if($users && $users['group_name'] == 'COMMERCIALE'):
?>
<input type="text"
class="form-control"
id="discount"
name="discount"
placeholder="Discount"
onkeyup="subAmount()"
value="<?php echo $order_data['order']['discount'] ?>"
autocomplete="off">
<input type="text" class="form-control" id="discount" name="discount" placeholder="Discount" onkeyup="subAmount()" value="<?php echo $order_data['order']['discount'] ?>" autocomplete="off">
<?php else: ?>
<input type="text"
class="form-control"
id="discount"
name="discount"
readonly
value="<?php echo $order_data['order']['discount'] ?>"
autocomplete="off">
<input type="text" class="form-control" id="discount" name="discount" readonly value="<?php echo $order_data['order']['discount'] ?>" autocomplete="off">
<?php endif; ?>
</div>
</div>
<div class="form-group">
<label for="net_amount" class="col-sm-5 control-label">Remise</label>
<div class="col-sm-7">
<input type="text"
class="form-control"
id="net_amount"
name="net_amount"
disabled
value="<?php echo $order_data['order']['net_amount'] ?>"
autocomplete="off">
<input type="hidden"
class="form-control"
id="net_amount_value"
name="net_amount_value"
value="<?php echo $order_data['order']['net_amount'] ?>"
autocomplete="off">
<input type="text" class="form-control" id="net_amount" name="net_amount" disabled value="<?php echo $order_data['order']['net_amount'] ?>" autocomplete="off">
<input type="hidden" class="form-control" id="net_amount_value" name="net_amount_value" value="<?php echo $order_data['order']['net_amount'] ?>" autocomplete="off">
</div>
</div>
@ -339,10 +220,7 @@
<div class="form-group">
<label for="paid_status" class="col-sm-5 control-label">Tranche de paiement</label>
<div class="col-sm-7">
<select class="form-control"
id="payment_mode"
name="payment_mode"
<?php echo !$is_editable ? 'disabled' : ''; ?>>
<select class="form-control" id="payment_mode" name="payment_mode">
<option value="1" selected>une tranche</option>
<option value="2">deux tranches</option>
</select>
@ -362,57 +240,44 @@
<div class="form-group" id="paid_status_1" style="display: none">
<label for="paid_status_1" class="col-sm-5 control-label">Tranche 1</label>
<div class="col-sm-3">
<select class="form-control"
id="payment_mode_1"
name="order_payment_mode_1"
<?php echo !$is_editable ? 'disabled' : ''; ?>>
<select class="form-control" id="payment_mode_1" name="order_payment_mode_1">
<option value="MVOLA">MVOLA</option>
<option value="Virement Bancaire">Virement Bancaire</option>
<option value="En espèce">En espèce</option>
</select>
</div>
<div class="col-sm-4">
<input type="number"
class="form-control"
id="payment_amount_1"
name="tranche_1"
placeholder="Montant"
onkeyup="calculerTranche2()"
<?php echo !$is_editable ? 'disabled readonly' : ''; ?>>
<input type="number" class="form-control" id="payment_amount_1"
name="tranche_1" placeholder="Montant" onkeyup="calculerTranche2()">
</div>
</div>
<div class="form-group" id="paid_status_2" style="display: none">
<label for="paid_status_2" class="col-sm-5 control-label">Tranche 2 (Reste)</label>
<div class="col-sm-3">
<select class="form-control"
id="payment_mode_2"
name="order_payment_mode_2"
<?php echo !$is_editable ? 'disabled' : ''; ?>>
<select class="form-control" id="payment_mode_2" name="order_payment_mode_2">
<option value="MVOLA">MVOLA</option>
<option value="Virement Bancaire">Virement Bancaire</option>
<option value="En espèce">En espèce</option>
</select>
</div>
<div class="col-sm-4">
<input type="number"
class="form-control"
id="payment_amount_2"
name="tranche_2"
placeholder="Montant"
readonly>
<input type="number" class="form-control" id="payment_amount_2"
name="tranche_2" placeholder="Montant" readonly>
</div>
</div>
</div>
<?php endif; ?>
<?php
$users = session()->get('user');
if ($users && $users['group_name'] !== 'COMMERCIALE'):
?>
<div class="form-group">
<label for="paid_status" class="col-sm-5 control-label">Statut payant</label>
<div class="col-sm-7">
<select type="text"
class="form-control"
id="paid_status"
name="paid_status"
<?php echo !$is_editable ? 'disabled' : ''; ?>>
<select type="text" class="form-control" id="paid_status" name="paid_status">
<option value="1">Validé</option>
<option value="2">Refusé</option>
</select>
@ -422,64 +287,24 @@
</div>
</div>
<!-- /.box-body -->
<div class="box-footer">
<input type="hidden" name="service_charge_rate" value="<?php echo $company_data['service_charge_value'] ?>" autocomplete="off">
<input type="hidden" name="vat_charge_rate" value="<?php echo $company_data['vat_charge_value'] ?>" autocomplete="off">
<a target="__blank" id="Imprimente" href="<?php echo base_url() . 'orders/printDiv/' . $order_data['order']['id'] ?>" class="btn btn-default">Imprimer</a>
<?php if ($is_editable): ?>
<button type="submit" class="btn btn-primary">Enregistrer</button>
<?php endif; ?>
<a href="<?php echo base_url('orders/') ?>" class="btn btn-warning">Retour</a>
</div>
</form>
<!-- /.box-body -->
</div>
<!-- /.box -->
</div>
<!-- col-md-12 -->
</div>
<!-- /.row -->
</section>
<!-- /.content -->
</div>
<!-- /.content-wrapper -->
<style>
/* ✅ Animation pour l'alerte */
@keyframes slideDown {
from {
opacity: 0;
transform: translateY(-10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
/* ✅ Style pour l'input en erreur */
.input-error {
border: 2px solid #dc3545 !important;
box-shadow: 0 0 8px rgba(220, 53, 69, 0.5) !important;
animation: pulse 0.5s ease-in-out;
}
@keyframes pulse {
0%, 100% { box-shadow: 0 0 8px rgba(220, 53, 69, 0.5); }
50% { box-shadow: 0 0 15px rgba(220, 53, 69, 0.8); }
}
</style>
<script type="text/javascript">
var base_url = "<?php echo base_url(); ?>";
var idData = "<?php echo $order_data['order']['id']; ?>";
var is_editable = <?php echo $is_editable ? 'true' : 'false'; ?>;
let Imprimente = document.getElementById('Imprimente');
let typesCommande = document.getElementById('typesCommande');
@ -496,77 +321,12 @@
}
});
// ============================================
// 🔹 FONCTION D'ALERTE VISUELLE
// ============================================
function showPriceAlert(message) {
const alertBox = $('#price_alert');
const discountInput = $('#discount');
$('#price_alert_message').text(message);
alertBox.slideDown(300);
discountInput.addClass('input-error');
setTimeout(function() {
alertBox.slideUp(300);
discountInput.removeClass('input-error');
}, 3500);
}
function hidePriceAlert() {
$('#price_alert').slideUp(300);
$('#discount').removeClass('input-error');
}
// ============================================
// 🔹 VALIDATION DU PRIX DEMANDÉ
// ============================================
function validateDiscount() {
if (!is_editable) return true; // ✅ Pas de validation si non éditable
var discount = parseFloat($('#discount').val());
var grossAmount = parseFloat($('#gross_amount_value').val()) || 0;
if (discount < 0) {
showPriceAlert("Le prix demandé ne peut pas être négatif !");
$('#discount').val('');
return false;
}
if (discount > grossAmount) {
showPriceAlert("Le prix demandé ne peut pas dépasser le prix affiché (" + grossAmount.toFixed(2) + ") !");
$('#discount').val(grossAmount.toFixed(2));
subAmount();
return false;
}
hidePriceAlert();
return true;
}
$(document).ready(function() {
$(".select_group").select2();
$("#mainOrdersNav").addClass('active');
$("#manageOrdersNav").addClass('active');
// ✅ Désactiver toutes les interactions si non éditable
if (!is_editable) {
$('input, select, textarea').prop('disabled', true);
$('#add_row').hide();
$('.btn-default[onclick^="removeRow"]').hide();
}
// ✅ Validation en temps réel du prix demandé
if (is_editable) {
$('#discount').on('keyup change', function() {
validateDiscount();
});
}
// ============================================
// 🔹 GESTION DES TRANCHES DE PAIEMENT
// ============================================
var paymentTranche = 1;
var netAmount = parseFloat($('#net_amount_value').val()) || 0;
$('#payment_amount_1').val(netAmount);
@ -587,26 +347,18 @@
}
$("#payment_mode").on("change", function() {
if (is_editable) {
addPaymentTranche($(this).val());
updateMontantTranches();
}
});
$('#payment_amount_1').on("input", function() {
if (is_editable) {
var amount1 = parseFloat($(this).val()) || 0;
var amount2 = netAmount - amount1;
$('#payment_amount_2').val(amount2);
}
});
addPaymentTranche(paymentTranche);
// ============================================
// 🔹 TABLEAU DE PRODUITS
// ============================================
if (is_editable) {
$("#add_row").unbind('click').bind('click', function() {
var table = $("#product_info_table");
var count_table_tbody_tr = $("#product_info_table tbody tr").length;
@ -621,15 +373,22 @@
'<td>' +
'<select class="form-control select_group product" data-row-id="' + row_id + '" id="product_' + row_id + '" name="product[]" style="width:100%;" onchange="getProductData(' + row_id + ')">' +
'<option value=""></option>';
$.each(response, function(index, value) {
html += '<option value="' + value.id + '">' + value.name + '</option>';
var displayText = value.sku + ' | ' + value.name;
if (value.numero_de_moteur) {
displayText += ' | ' + value.numero_de_moteur;
}
// ✅ Ne plus afficher la puissance
html += '<option value="' + value.id + '">' + displayText + '</option>';
});
html += '</select>' +
'</td>' +
'<td><input type="number" name="qty[]" id="qty_' + row_id + '" class="form-control" onkeyup="getTotal(' + row_id + ')"></td>' +
'<td><input type="text" name="rate[]" id="rate_' + row_id + '" class="form-control" disabled><input type="hidden" name="rate_value[]" id="rate_value_' + row_id + '" class="form-control"></td>' +
'<td><input type="text" name="amount[]" id="amount_' + row_id + '" class="form-control" disabled><input type="hidden" name="amount_value[]" id="amount_value_' + row_id + '" class="form-control"></td>' +
// ✅ Colonne puissance visible et modifiable
'<td><input type="text" name="puissance[]" id="puissance_' + row_id + '" class="form-control" placeholder="Puissance" autocomplete="off" value="1"></td>' +
'<td><input type="text" name="rate[]" id="rate_' + row_id + '" class="form-control numeric-input" disabled min="0"><input type="hidden" name="rate_value[]" id="rate_value_' + row_id + '" class="form-control"><input type="hidden" id="min_price_' + row_id + '" name="min_price[]" value=""></td>' +
'<td><input type="text" name="amount[]" id="amount_' + row_id + '" class="form-control numeric-input" disabled min="0"><input type="hidden" name="amount_value[]" id="amount_value_' + row_id + '" class="form-control"></td>' +
'<td><button type="button" class="btn btn-default" onclick="removeRow(\'' + row_id + '\')"><i class="fa fa-close"></i></button></td>' +
'</tr>';
@ -644,18 +403,11 @@
});
return false;
});
}
}); // /document.ready
});
// ============================================
// 🔹 CALCUL DU TOTAL
// ============================================
function getTotal(row = null) {
if (!is_editable) return; // ✅ Bloquer si non éditable
if (row) {
var total = Number($("#rate_value_" + row).val()) * Number($("#qty_" + row).val());
var total = Number($("#rate_value_" + row).val());
total = total.toFixed(2);
$("#amount_" + row).val(total);
$("#amount_value_" + row).val(total);
@ -665,15 +417,14 @@
}
}
// 🔹 OBTENIR LES DONNÉES PRODUIT
function getProductData(row_id) {
if (!is_editable) return; // ✅ Bloquer si non éditable
var product_id = $("#product_" + row_id).val();
if (product_id == "") {
$("#rate_" + row_id).val("");
$("#rate_value_" + row_id).val("");
$("#qty_" + row_id).val("");
$("#min_price_" + row_id).val("");
$("#puissance_" + row_id).val("1"); // ✅ Réinitialiser à 1
$("#amount_" + row_id).val("");
$("#amount_value_" + row_id).val("");
} else {
@ -683,26 +434,39 @@
data: { product_id: product_id },
dataType: 'json',
success: function(response) {
$("#rate_" + row_id).val(response.price);
$("#rate_value_" + row_id).val(response.price);
$("#qty_" + row_id).val(1);
$("#qty_value_" + row_id).val(1);
var total = Number(response.price) * 1;
console.log('✅ Response:', response); // Debug
var prixVente = parseFloat(response.prix_vente) || 0;
var prixMinimal = parseFloat(response.prix_minimal) || 0;
if (prixVente < 0) prixVente = 0;
if (prixMinimal < 0) prixMinimal = 0;
$("#rate_" + row_id).val(prixVente);
$("#rate_value_" + row_id).val(prixVente);
$("#min_price_" + row_id).val(prixMinimal);
// ✅ CORRECTION : Remplir la puissance
var puissanceValue = response.puissance || '1';
console.log('✅ Puissance extraite:', puissanceValue); // Debug
$("#puissance_" + row_id).val(puissanceValue);
// ✅ Calculer le montant (prix * puissance)
var total = prixVente * 1; // Pour l'instant on garde qty=1
total = total.toFixed(2);
$("#amount_" + row_id).val(total);
$("#amount_value_" + row_id).val(total);
subAmount();
},
error: function(xhr, status, error) {
console.error('❌ Erreur AJAX:', error);
}
});
}
}
}
// ============================================
// 🔹 CALCUL DU MONTANT TOTAL (AVEC TVA, REMISE, ETC.)
// ============================================
function subAmount() {
if (!is_editable) return; // ✅ Bloquer si non éditable
var service_charge = <?php echo ($company_data['service_charge_value'] > 0) ? $company_data['service_charge_value'] : 0; ?>;
var vat_charge = <?php echo ($company_data['vat_charge_value'] > 0) ? $company_data['vat_charge_value'] : 0; ?>;
@ -734,12 +498,10 @@
var discount = $("#discount").val();
if (discount) {
if (validateDiscount()) {
var grandTotal = Number(totalAmount) - Number(discount);
grandTotal = grandTotal.toFixed(2);
$("#net_amount").val(grandTotal);
$("#net_amount_value").val(grandTotal);
}
} else {
$("#net_amount").val(totalAmount);
$("#net_amount_value").val(totalAmount);
@ -756,12 +518,7 @@
updateMontantTranches();
}
// ============================================
// 🔹 AUTRES FONCTIONS
// ============================================
function paidAmount() {
if (!is_editable) return;
var grandTotal = $("#net_amount_value").val();
if (grandTotal) {
var dueAmount = Number($("#net_amount_value").val()) - Number($("#paid_amount").val());
@ -772,15 +529,10 @@
}
function removeRow(tr_id) {
if (!is_editable) return; // ✅ Bloquer si non éditable
$("#product_info_table tbody tr#row_" + tr_id).remove();
subAmount();
}
// ============================================
// 🔹 GESTION MONTANT DE TRANCHES
// ============================================
function getMontantPourTranches() {
var discount = parseFloat($("#discount").val()) || 0;
var grossAmount = parseFloat($("#gross_amount_value").val()) || 0;
@ -788,8 +540,6 @@
}
function updateMontantTranches() {
if (!is_editable) return;
var montant = getMontantPourTranches();
var discount = parseFloat($("#discount").val()) || 0;
@ -808,8 +558,6 @@
}
function calculerTranche2() {
if (!is_editable) return;
var montantTotal = getMontantPourTranches();
var tranche1 = parseFloat($("#payment_amount_1").val()) || 0;
var tranche2 = montantTotal - tranche1;
@ -817,16 +565,12 @@
$("#payment_amount_2").val(tranche2.toFixed(2));
}
if (is_editable) {
$("#discount").on('keyup', function() {
updateMontantTranches();
});
}
const net_amount_value = document.getElementById('net_amount_value');
const net_amount = document.getElementById('net_amount');
const payment_amount_1 = document.getElementById('payment_amount_1');
if (payment_amount_1 && net_amount) {
payment_amount_1.value = net_amount.value;
}
</script>

4
app/Views/orders/index.php

@ -56,7 +56,7 @@
<th>Nom du client</th>
<th>Téléphone du client</th>
<th>Date et Heure</th>
<th>Remise</th>
<th>Prix demandé</th>
<th>Prix de vente</th>
<th>Validation</th>
@ -77,7 +77,7 @@
<th>Nom du produit</th>
<th>Commerciale</th>
<th>Date et Heure</th>
<th>Remise</th>
<th>Prix demandé</th>
<th>Prix de vente</th>
<th>Validation</th>
<?php if (

Loading…
Cancel
Save