Compare commits

..

16 commits

Author SHA1 Message Date
ba22988026 Merge pull request 'funcion: notas de producto' (#36) from funcion/notas-producto into master
Reviewed-on: #36
2024-11-12 22:08:36 -03:00
ale
9c9b1dc6cb Merge branch 'refs/heads/master' into funcion/notas-producto 2024-11-12 21:48:48 -03:00
Rodrigo
f9e55b38a5 Merge branch 'master' into funcion/notas-producto 2024-10-18 19:10:20 -03:00
Rodrigo
3ad9500f23 Alerta de notas sale sólo después de intentar de confirmar 2024-10-16 22:32:16 -03:00
Rodrigo
538cc84e10 Desactivar agregar producto si requiere notas y están vacías 2024-10-16 21:56:05 -03:00
Rodrigo
be945b0eee Agregué las notas al PDF para el armado del barrio 2024-10-15 20:56:37 -03:00
Rodrigo
b5f4443836 Boton para descargar notas en pag de compras 2024-10-15 20:44:28 -03:00
Rodrigo
1e443ea2ca Merge branch 'master' into funcion/notas-producto 2024-10-08 20:36:16 -03:00
Rodrigo
c1af6909c4 No habilitar sync producto si cantidad es 0 2024-10-08 20:33:20 -03:00
Rodrigo
e31c375867 Ajuste visual de notas 2024-10-08 20:28:58 -03:00
Rodrigo
6d10fbc0bf Sincronizar notas de producto 2024-10-08 20:16:25 -03:00
Rodrigo
9fc47513c2 Arreglo para usuarios por defecto 2024-09-19 22:31:58 -03:00
Rodrigo
ff332fadc1 Merge branch 'master' into funcion/notas-producto 2024-09-19 21:47:09 -03:00
Rodrigo
177ba193de Agregado "requiere notas" a productos con 'PTC' 2024-09-19 20:58:44 -03:00
Rodrigo
82f5862063 Mostrar notas en pedido 2024-09-17 21:28:08 -03:00
Rodrigo
7eeeae6a1e Agregar notas a la base de datos 2024-09-17 21:27:58 -03:00
18 changed files with 229 additions and 57 deletions

View file

@ -233,10 +233,36 @@ class GrupoDeCompra extends Model
// Guardar en un archivo .csv // Guardar en un archivo .csv
try { try {
$writer = Writer::createFromPath(resource_path('csv/exports/total-pedidos.csv'), 'w'); $writer = Writer::createFromPath(resource_path('csv/exports/total-pedidos.csv'), 'w');
$writer->insertAll($planilla); $writer->insertAll($planilla);
} catch (CannotInsertRecord $e) { } catch (CannotInsertRecord $e) {
var_export($e->getRecords()); var_export($e->getRecords());
}
}
public static function exportarProductosConNotasEnCSV() {
$gdcs = GrupoDeCompra::all();
foreach ($gdcs as $i => $gdc) {
$productos_en_pedido = DB::table('pedidos_aprobados')->where('grupo_de_compra_id', $gdc->id)->get()->keyBy('producto_id');
$pedidos = $gdc->pedidosAprobados();
foreach ($productos_en_pedido as $id => $producto_pedido) {
foreach ($pedidos as $pedido) {
$producto = $pedido->productos()->find($id);
if ($producto != null && $producto->requiere_notas) {
$planilla[$i+1][0] = $gdc->nombre;
$planilla[$i+1][1] = $producto->nombre;
$planilla[$i+1][2] = $producto->pivot->cantidad;
$planilla[$i+1][3] = $producto->pivot->notas;
}
}
}
}
// Guardar en un archivo .csv
try {
$writer = Writer::createFromPath(resource_path('csv/exports/pedidos-notas.csv'), 'w');
$writer->insertAll($planilla);
} catch (CannotInsertRecord $e) {
var_export($e->getRecords());
} }
} }
} }

View file

@ -79,6 +79,7 @@ class SubpedidoController extends Controller
$valid = request()->validate([ $valid = request()->validate([
'cantidad' => 'required|min:0', 'cantidad' => 'required|min:0',
'notas' => 'nullable',
'producto_id' => [ 'producto_id' => [
'required', 'required',
Rule::in(Producto::all()->pluck('id')), Rule::in(Producto::all()->pluck('id')),
@ -86,7 +87,11 @@ class SubpedidoController extends Controller
]); ]);
$producto = Producto::find($valid['producto_id']); $producto = Producto::find($valid['producto_id']);
$subpedido->syncProducto($producto, $valid['cantidad']); $notas = $valid['notas'];
if ($notas == null) {
$notas = "";
}
$subpedido->syncProducto($producto, $valid['cantidad'], $notas);
return new SubpedidoResource($subpedido); return new SubpedidoResource($subpedido);
} }

View file

@ -16,6 +16,12 @@ class ComprasController
return response()->download($file); return response()->download($file);
} }
public function descargarNotas() {
GrupoDeCompra::exportarProductosConNotasEnCSV();
$file = resource_path('csv/exports/pedidos-notas.csv');
return response()->download($file);
}
public function show() public function show()
{ {
return view('auth/compras_login'); return view('auth/compras_login');

View file

@ -25,7 +25,8 @@ class ProductoResource extends JsonResource
'imagen' => optional($this->poster)->url(), 'imagen' => optional($this->poster)->url(),
'descripcion' => $this->descripcion, 'descripcion' => $this->descripcion,
'apto_veganxs' => $this->apto_veganxs, 'apto_veganxs' => $this->apto_veganxs,
'apto_celiacxs' => $this->apto_celiacxs 'apto_celiacxs' => $this->apto_celiacxs,
'requiere_notas' => $this->requiere_notas,
]; ];
} }
} }

View file

@ -15,7 +15,7 @@ class Producto extends Model
public function subpedidos() public function subpedidos()
{ {
return $this->belongsToMany('App\Subpedido','productos_subpedidos')->withPivot(["cantidad"]); return $this->belongsToMany('App\Subpedido','productos_subpedidos')->withPivot(["cantidad", "notas"]);
} }
public function proveedor() public function proveedor()

View file

@ -15,7 +15,7 @@ class Subpedido extends Model
public function productos() public function productos()
{ {
return $this->belongsToMany('App\Producto')->withPivot(["cantidad","total"]); return $this->belongsToMany('App\Producto')->withPivot(["cantidad","total", "notas"]);
} }
//Bonos del MPS, Sororo, etc. NO devuelve bonos de transporte //Bonos del MPS, Sororo, etc. NO devuelve bonos de transporte
@ -84,13 +84,14 @@ class Subpedido extends Model
} }
//Actualiza el pedido, agregando o quitando del subpedido según sea necesario. Debe ser llamado desde el controlador de subpedidos, luego de validar que los parámetros $producto y $cantidad son correctos. También calcula el subtotal por producto. //Actualiza el pedido, agregando o quitando del subpedido según sea necesario. Debe ser llamado desde el controlador de subpedidos, luego de validar que los parámetros $producto y $cantidad son correctos. También calcula el subtotal por producto.
public function syncProducto(Producto $producto, Int $cantidad) { public function syncProducto(Producto $producto, Int $cantidad, string $notas) {
if ($cantidad){ if ($cantidad){
//si la cantidad es 1 o más se agrega el producto o actualiza la cantidad //si la cantidad es 1 o más se agrega el producto o actualiza la cantidad
$this->productos()->syncWithoutDetaching([ $this->productos()->syncWithoutDetaching([
$producto->id => [ $producto->id => [
'cantidad' => $cantidad, 'cantidad' => $cantidad,
'total' => $cantidad * $producto->precio 'total' => $cantidad * $producto->precio,
'notas' => $notas,
] ]
]); ]);
} else { } else {

View file

@ -0,0 +1,32 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
class NotasProducto extends Migration
{
/**
* Run the migrations.
*
* @return void
*/
public function up()
{
Schema::table('producto_subpedido', function (Blueprint $table) {
$table->string('notas')->nullable();
});
}
/**
* Reverse the migrations.
*
* @return void
*/
public function down()
{
Schema::table('producto_subpedido', function (Blueprint $table) {
$table->dropColumn('notas');
});
}
}

View file

@ -0,0 +1,32 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
class ProductoRequiereNotas extends Migration
{
/**
* Run the migrations.
*
* @return void
*/
public function up()
{
Schema::table('productos', function (Blueprint $table) {
$table->boolean('requiere_notas')->default(false);
});
}
/**
* Reverse the migrations.
*
* @return void
*/
public function down()
{
Schema::table('productos', function (Blueprint $table) {
$table->dropColumn('requiere_notas');
});
}
}

View file

@ -61,7 +61,8 @@ class CanastaSeeder extends Seeder
'nombre' => trim(str_replace('*', ' ',$registro['Producto'])), 'nombre' => trim(str_replace('*', ' ',$registro['Producto'])),
'precio' => $registro['Precio'], 'precio' => $registro['Precio'],
'proveedor_id' => $this->obtenerProveedor($registro['Producto']), 'proveedor_id' => $this->obtenerProveedor($registro['Producto']),
'bono' => $registro[$this::FILA_HEADER] == "B" 'bono' => $registro[$this::FILA_HEADER] == "B",
'requiere_notas'=> $registro[$this::FILA_HEADER] =="PTC",
]; ];
} }

View file

@ -13,5 +13,6 @@ class DatabaseSeeder extends Seeder
public function run() public function run()
{ {
$this->call(CanastaSeeder::class); $this->call(CanastaSeeder::class);
$this->call(GrupoDeCompraSeeder::class);
} }
} }

View file

@ -31,14 +31,14 @@ class GrupoDeCompraSeeder extends Seeder
$usersToInsert[] = [ $usersToInsert[] = [
'name' => $registro['barrio'], 'name' => $registro['barrio'],
'password' => Hash::make($registro['barrio']), 'password' => Hash::make("asd"),
"is_admin" => 0, "is_admin" => 0,
'grupo_de_compra_id' => $key 'grupo_de_compra_id' => $key
]; ];
$usersToInsert[] = [ $usersToInsert[] = [
'name' => $registro['barrio'] . "_admin", 'name' => $registro['barrio'] . "_admin",
'password' => Hash::make($registro['barrio'] . "admin"), 'password' => Hash::make("asd"),
"is_admin" => 1, "is_admin" => 1,
'grupo_de_compra_id' => $key 'grupo_de_compra_id' => $key
]; ];

9
resources/js/app.js vendored
View file

@ -70,6 +70,10 @@ const app = new Vue({
cantidad(producto) { cantidad(producto) {
let pedido = this.productos.some(p => p.id == producto.id) let pedido = this.productos.some(p => p.id == producto.id)
return pedido ? this.productos.find(p => p.id == producto.id).pivot.cantidad : 0 return pedido ? this.productos.find(p => p.id == producto.id).pivot.cantidad : 0
},
notas(producto) {
let pedido = this.productos.some(p => p.id == producto.id);
return pedido ? this.productos.find(p => p.id == producto.id).pivot.notas : "";
}, },
settearDevoluciones() { settearDevoluciones() {
axios.get(`/api/grupos-de-compra/${this.gdc}/devoluciones`) axios.get(`/api/grupos-de-compra/${this.gdc}/devoluciones`)
@ -99,14 +103,15 @@ const app = new Vue({
} }
}) })
}) })
Event.$on('sync-subpedido', (cantidad, id) => { Event.$on('sync-subpedido', (cantidad, id, notas) => {
if (this.pedido.aprobado) { if (this.pedido.aprobado) {
this.$toast('No se puede modificar un pedido ya aprobado', 2000); this.$toast('No se puede modificar un pedido ya aprobado', 2000);
return; return;
} }
axios.post("/api/subpedidos/" + this.pedido.id + "/sync", { axios.post("/api/subpedidos/" + this.pedido.id + "/sync", {
cantidad: cantidad, cantidad: cantidad,
producto_id: id producto_id: id,
notas: notas,
}).then((response) => { }).then((response) => {
this.pedido = response.data.data this.pedido = response.data.data
this.$toast('Pedido actualizado exitosamente') this.$toast('Pedido actualizado exitosamente')

View file

@ -11,6 +11,16 @@
</a> </a>
</p> </p>
</div> </div>
<div class="field">
<p class="control">
<a href="/compras/pedidos/notas" class="button">
<span class="icon is-small">
<i class="fas fa-sticky-note"></i>
</span>
<span>Descargar planilla de notas</span>
</a>
</p>
</div>
</div> </div>
</div> </div>
</template> </template>

View file

@ -1,28 +1,44 @@
<template> <template>
<div class="field has-addons contador"> <div>
<div class="control"> <div class="field has-addons contador">
<button class="button is-small" @click.capture="decrementar();"> <div class="control">
<i class="fa fa-solid fa-minus"></i> <button class="button is-small" @click.capture="decrementar();">
<i class="fa fa-solid fa-minus"></i>
</button>
</div>
<div class="control">
<input id="cantidad" v-model="cantidad" class="input is-small" type="number" style="text-align: center">
</div>
<div class="control" @click="incrementar();">
<button class="button is-small">
<i class="fa fa-solid fa-plus"></i>
</button>
</div>
<button :disabled="disableConfirm()" class="button is-small is-success ml-1" @click="confirmar()">
<span class="icon">
<i class="fas fa-check"></i>
</span>
</button>
<button :disabled="!puedeBorrar()" class="button is-small is-danger ml-1" @click="borrar()">
<span class="icon">
<i class="fas fa-trash-alt"></i>
</span>
</button> </button>
</div> </div>
<div class="control"> <div v-if="producto.requiere_notas" v-bind:class="{'has-icons-right': notas_warning_visible}" class="control is-full-width has-icons-left">
<input id="cantidad" v-model="cantidad" class="input is-small" type="number" style="text-align: center"> <span class="icon is-small is-left">
</div> <i class="fas fa-sticky-note"></i>
<div class="control" @click="incrementar();">
<button class="button is-small">
<i class="fa fa-solid fa-plus"></i>
</button>
</div>
<button :disabled="!hayCambios()" class="button is-small is-success ml-1" @click="confirmar()">
<span class="icon">
<i class="fas fa-check"></i>
</span> </span>
</button> <input v-model="notas" v-bind:class="{'is-danger': notas_warning_visible}" id="notas" class="input" type="text" placeholder="Talle o color" />
<button :disabled="!puedeBorrar()" class="button is-small is-danger ml-1" @click="borrar()"> <span v-if="notas_warning_visible" class="icon is-small is-right">
<span class="icon"> <i class="fas fa-exclamation-triangle"></i>
<i class="fas fa-trash-alt"></i>
</span> </span>
</button> <article v-if="notas_warning_visible" class="message is-danger is-small">
<div class="message-body">
No se puede dejar este campo vac&iacute;o
</div>
</article>
</div>
</div> </div>
</template> </template>
@ -33,21 +49,24 @@
}, },
data() { data() {
return { return {
cantidad: this.producto.cantidad, cantidad: this.cantidadEnChismosa(),
enChismosa: this.producto.cantidad, notas: this.notasEnChismosa(),
notas_warning_visible: false,
} }
}, },
mounted() { mounted() {
if (this.producto.pivot !== undefined) { Event.$on('sync-subpedido', (cantidad, productoId, notas) => {
this.cantidad = this.producto.pivot.cantidad;
this.enChismosa = this.cantidad;
}
Event.$on('sync-subpedido', (cantidad,productoId) => {
if (this.producto.id === productoId) if (this.producto.id === productoId)
this.sincronizar(cantidad); this.sincronizar(cantidad, notas);
}); });
}, },
methods: { methods: {
notasEnChismosa() {
return this.producto.pivot !== undefined ? this.producto.pivot.notas : "";
},
cantidadEnChismosa() {
return this.producto.pivot !== undefined ? this.producto.pivot.cantidad : 0;
},
decrementar() { decrementar() {
this.cantidad -= 1; this.cantidad -= 1;
}, },
@ -55,26 +74,39 @@
this.cantidad += 1; this.cantidad += 1;
}, },
confirmar() { confirmar() {
Event.$emit('sync-subpedido', this.cantidad, this.producto.id); if (this.warningNotas()) {
this.notas_warning_visible = true;
return;
}
console.log("Emit sync " + this.cantidad + " " + this.notas);
Event.$emit('sync-subpedido', this.cantidad, this.producto.id, this.notas);
}, },
borrar() { borrar() {
this.cantidad = 0; this.cantidad = 0;
this.confirmar(); this.confirmar();
}, },
sincronizar(cantidad) { sincronizar(cantidad, notas) {
this.notas_warning_visible = false;
this.notas = notas;
this.cantidad = cantidad; this.cantidad = cantidad;
if (this.producto.pivot != null) { if (this.producto.pivot !== undefined) {
this.producto.pivot.cantidad = cantidad; this.producto.pivot.cantidad = cantidad;
} else { this.producto.pivot.notas = notas;
this.producto.cantidad = cantidad;
} }
this.enChismosa = cantidad;
}, },
hayCambios() { hayCambios() {
return this.cantidad != this.enChismosa; if (this.cantidad != this.cantidadEnChismosa()) return true;
return this.cantidad > 0 && this.notas != this.notasEnChismosa();
}, },
puedeBorrar() { puedeBorrar() {
return this.enChismosa > 0; return this.cantidadEnChismosa() > 0;
},
warningNotas() {
return this.producto.requiere_notas && this.cantidad > 0 && !this.notas;
},
disableConfirm() {
return !this.hayCambios();
}, },
} }
} }
@ -97,4 +129,12 @@
.contador { .contador {
min-width: 178px; min-width: 178px;
} }
.is-danger {
background-color: #fca697;
}
.is-danger::placeholder {
color: #fff;
opacity: 1; /* Firefox */
}
</style> </style>

View file

@ -8,12 +8,13 @@ export default {
return { return {
cantidad: this.producto.cantidad, cantidad: this.producto.cantidad,
enChismosa: this.producto.cantidad, enChismosa: this.producto.cantidad,
notas: this.producto.notas,
} }
}, },
mounted() { mounted() {
Event.$on('sync-subpedido', (cantidad,productoId) => { Event.$on('sync-subpedido', (cantidad, productoId, notas) => {
if (this.producto.id === productoId) if (this.producto.id === productoId)
this.sincronizar(cantidad); this.sincronizar(cantidad, notas);
}); });
}, },
methods: { methods: {
@ -24,19 +25,21 @@ export default {
this.cantidad += 1; this.cantidad += 1;
}, },
confirmar() { confirmar() {
Event.$emit('sync-subpedido', this.cantidad, this.producto.id); Event.$emit('sync-subpedido', this.cantidad, this.producto.id, this.notas);
}, },
borrar() { borrar() {
this.cantidad = 0; this.cantidad = 0;
this.confirmar(); this.confirmar();
}, },
sincronizar(cantidad) { sincronizar(cantidad, notas) {
this.cantidad = cantidad; this.cantidad = cantidad;
this.producto.cantidad = cantidad; this.producto.cantidad = cantidad;
this.enChismosa = cantidad; this.enChismosa = cantidad;
this.notas = notas;
this.producto.notas = notas;
}, },
hayCambios() { hayCambios() {
return this.cantidad != this.enChismosa; return this.cantidad != this.enChismosa || this.notas != this.producto.notas;
}, },
puedeBorrar() { puedeBorrar() {
return this.enChismosa > 0; return this.enChismosa > 0;

View file

@ -34,7 +34,11 @@ export default {
params: this.params(filtro,valor) params: this.params(filtro,valor)
}).then(response => { }).then(response => {
this.productos = response.data.data; this.productos = response.data.data;
this.productos.forEach(p => p.cantidad = this.$root.cantidad(p)) this.productos.forEach(p => {
p.pivot = {};
p.pivot.cantidad = this.$root.cantidad(p);
p.pivot.notas = this.$root.notas(p);
});
}); });
this.visible = true; this.visible = true;
Event.$emit("migas-agregar",this.miga); Event.$emit("migas-agregar",this.miga);

View file

@ -27,9 +27,13 @@
@foreach($subpedido->productos as $producto) @foreach($subpedido->productos as $producto)
@if(!$producto->bono) @if(!$producto->bono)
<tr> <tr>
<td> <td>
{{ $producto->nombre }} {{ $producto->nombre }}
@if($producto->pivot->notas)
<br /><b>Talle/Color:</b> {{ $producto->pivot->notas }}
@endif
</td> </td>
<td style="text-align: center"> <td style="text-align: center">
{{ $producto->pivot->cantidad }} {{ $producto->pivot->cantidad }}

View file

@ -81,4 +81,5 @@ Route::get('/compras', 'ComprasController@show')->name('compras_login.show');
Route::middleware(['compras'])->group( function() { Route::middleware(['compras'])->group( function() {
Route::get('/compras/pedidos', 'ComprasController@indexPedidos')->name('compras.pedidos'); Route::get('/compras/pedidos', 'ComprasController@indexPedidos')->name('compras.pedidos');
Route::get('/compras/pedidos/descargar', 'ComprasController@descargarPedidos')->name('compras.pedidos.descargar'); Route::get('/compras/pedidos/descargar', 'ComprasController@descargarPedidos')->name('compras.pedidos.descargar');
Route::get('/compras/pedidos/notas', 'ComprasController@descargarNotas')->name('compras.pedidos.descargar');
}); });