Merge pull request 'funcion/actualizar-canasta-desde-compras' (#39) from funcion/actualizar-canasta-desde-compras into master
Reviewed-on: #39
This commit is contained in:
commit
d4bffd1df8
|
@ -12,6 +12,7 @@ npm-debug.log
|
|||
yarn-error.log
|
||||
.idea
|
||||
/resources/csv/exports/*.csv
|
||||
/resources/csv/canastas/*.csv
|
||||
/public/css/
|
||||
/public/js/
|
||||
/public/mix-manifest.json
|
||||
|
|
|
@ -0,0 +1,11 @@
|
|||
<?php
|
||||
|
||||
namespace App;
|
||||
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
|
||||
class CanastaLog extends Model
|
||||
{
|
||||
protected $fillable = ["path", "descripcion"];
|
||||
protected $table = "carga_de_canastas";
|
||||
}
|
|
@ -0,0 +1,151 @@
|
|||
<?php
|
||||
|
||||
namespace App\Helpers;
|
||||
|
||||
use App\Producto;
|
||||
use App\Proveedor;
|
||||
use App\CanastaLog;
|
||||
use DatabaseSeeder;
|
||||
use Illuminate\Support\Arr;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\Support\Str;
|
||||
use League\Csv\Reader;
|
||||
|
||||
class CanastaHelper
|
||||
{
|
||||
const FILA_HEADER = "Tipo";
|
||||
const ULTIMA_FILA = "TOTAL";
|
||||
const ARCHIVO_SUBIDO = 'Archivo subido';
|
||||
const CANASTA_CARGADA = 'Canasta cargada';
|
||||
|
||||
public static function guardarCanasta($data, $path): string {
|
||||
$nombre = $data->getClientOriginalName();
|
||||
|
||||
$data->move(resource_path($path), $nombre);
|
||||
|
||||
self::log($path . $nombre, self::ARCHIVO_SUBIDO);
|
||||
|
||||
return $nombre;
|
||||
}
|
||||
|
||||
public static function cargarCanasta($archivo) {
|
||||
self::limpiarTablas();
|
||||
|
||||
$csv = Reader::createFromPath(resource_path($archivo), 'r');
|
||||
$csv->setDelimiter("|");
|
||||
$iHeader = self::obtenerIndiceDeHeader($csv);
|
||||
$csv->setHeaderOffset($iHeader);
|
||||
$registros = $csv->getRecords();
|
||||
|
||||
$toInsert = [];
|
||||
$categoria = '';
|
||||
foreach($registros as $i => $registro){
|
||||
//filas que están arriba del header
|
||||
if ($i <= $iHeader){
|
||||
continue;
|
||||
}
|
||||
|
||||
//finalizar
|
||||
if ($registro[self::FILA_HEADER] == self::ULTIMA_FILA) {
|
||||
break;
|
||||
}
|
||||
|
||||
//filas que no tienen tipo
|
||||
if (!Arr::has($registro,self::FILA_HEADER)|| trim($registro[self::FILA_HEADER]) == ''){
|
||||
var_dump("no hay tipo en la fila " . $i);
|
||||
continue;
|
||||
}
|
||||
|
||||
//saltear bono de transporte
|
||||
if ($registro[self::FILA_HEADER] == "T"){
|
||||
continue;
|
||||
}
|
||||
|
||||
//obtener categoria
|
||||
if ($registro['Producto'] == '') {
|
||||
//es la pregunta de la copa?
|
||||
if (Str::contains($registro[self::FILA_HEADER],"¿")) { continue; }
|
||||
$categoria = $registro[self::FILA_HEADER];
|
||||
continue;
|
||||
}
|
||||
|
||||
//completar producto
|
||||
$toInsert[] = [
|
||||
'fila' => $i,
|
||||
'categoria' => $categoria,
|
||||
'nombre' => trim(str_replace('*', ' ',$registro['Producto'])),
|
||||
'precio' => $registro['Precio'],
|
||||
'proveedor_id' => self::obtenerProveedor($registro['Producto']),
|
||||
'bono' => $registro[self::FILA_HEADER] == "B",
|
||||
'requiere_notas'=> $registro[self::FILA_HEADER] =="PTC",
|
||||
];
|
||||
}
|
||||
|
||||
foreach (array_chunk($toInsert,DatabaseSeeder::CHUNK_SIZE) as $chunk) {
|
||||
DB::table('productos')->insert($chunk);
|
||||
}
|
||||
|
||||
self::agregarBonoBarrial();
|
||||
|
||||
self::log($archivo, self::CANASTA_CARGADA);
|
||||
}
|
||||
|
||||
private static function obtenerIndiceDeHeader($csv){
|
||||
$registros = $csv->getRecords();
|
||||
$iheader = 0;
|
||||
foreach ($registros as $i => $registro){
|
||||
if (strtolower($registro[0]) == strtolower(self::FILA_HEADER)) {
|
||||
$iheader = $i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return $iheader;
|
||||
}
|
||||
|
||||
private static function obtenerProveedor($nombre) {
|
||||
$result = null;
|
||||
if (Str::contains($nombre,"*")){
|
||||
$result = Proveedor::firstOrCreate([
|
||||
'nombre' => 'Proveedor de economía solidaria',
|
||||
'economia_solidaria' => 1,
|
||||
'nacional' => 1
|
||||
])->id;
|
||||
}
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $path
|
||||
* @param $descripcion
|
||||
* @return void
|
||||
*/
|
||||
private static function log($path, $descripcion): void
|
||||
{
|
||||
$log = new CanastaLog([
|
||||
'path' => $path,
|
||||
'descripcion' => $descripcion,
|
||||
]);
|
||||
$log->save();
|
||||
}
|
||||
|
||||
private static function limpiarTablas()
|
||||
{
|
||||
DB::delete('delete from producto_subpedido');
|
||||
DB::delete('delete from productos');
|
||||
DB::delete('delete from subpedidos');
|
||||
}
|
||||
|
||||
private static function agregarBonoBarrial()
|
||||
{
|
||||
$categoria = Producto::all()->pluck('categoria')->unique()->flatten()->first(function ($c) { return Str::contains($c, 'BONO'); });
|
||||
DB::table('productos')->insert([
|
||||
'fila' => 420,
|
||||
'nombre' => "Bono barrial",
|
||||
'precio' => 20,
|
||||
'categoria' => $categoria,
|
||||
'bono' => 1,
|
||||
'proveedor_id' => null,
|
||||
'requiere_notas'=> false,
|
||||
]);
|
||||
}
|
||||
}
|
|
@ -3,10 +3,14 @@
|
|||
namespace App\Http\Controllers;
|
||||
|
||||
use App\GrupoDeCompra;
|
||||
use App\Helpers\CanastaHelper;
|
||||
use App\Producto;
|
||||
use Illuminate\Http\Request;
|
||||
|
||||
class ComprasController
|
||||
{
|
||||
const CANASTAS_PATH = 'csv/canastas/';
|
||||
|
||||
public function indexPedidos() {
|
||||
return view('compras_pedidos');
|
||||
}
|
||||
|
@ -33,4 +37,23 @@ class ComprasController
|
|||
{
|
||||
return view('auth/compras_login');
|
||||
}
|
||||
|
||||
public function cargarCanasta(Request $request)
|
||||
{
|
||||
$request->validate([
|
||||
'data' => 'required|file|mimes:csv,txt|max:2048',
|
||||
]);
|
||||
|
||||
$nombre = CanastaHelper::guardarCanasta($request->file('data'), self::CANASTAS_PATH);
|
||||
CanastaHelper::cargarCanasta(self::CANASTAS_PATH . $nombre);
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Canasta cargada exitosamente',
|
||||
], 200);
|
||||
}
|
||||
|
||||
public function descargarCanastaEjemplo() {
|
||||
$file = resource_path('csv/productos.csv');
|
||||
return response()->download($file);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,33 @@
|
|||
<?php
|
||||
|
||||
use Illuminate\Database\Migrations\Migration;
|
||||
use Illuminate\Database\Schema\Blueprint;
|
||||
use Illuminate\Support\Facades\Schema;
|
||||
|
||||
class CrearCargaDeCanastas extends Migration
|
||||
{
|
||||
/**
|
||||
* Run the migrations.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function up()
|
||||
{
|
||||
Schema::create('carga_de_canastas', function (Blueprint $table) {
|
||||
$table->id();
|
||||
$table->string('path');
|
||||
$table->string('descripcion');
|
||||
$table->timestamps();
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Reverse the migrations.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function down()
|
||||
{
|
||||
Schema::dropIfExists('carga_de_canastas');
|
||||
}
|
||||
}
|
|
@ -1,13 +1,11 @@
|
|||
<?php
|
||||
|
||||
use App\Helpers\CanastaHelper;
|
||||
use Illuminate\Database\Seeder;
|
||||
use League\Csv\Reader;
|
||||
use App\Proveedor;
|
||||
|
||||
class CanastaSeeder extends Seeder
|
||||
{
|
||||
const FILA_HEADER = "Tipo";
|
||||
const ULTIMA_FILA = "TOTAL";
|
||||
const ARCHIVO_DEFAULT = 'csv/productos.csv';
|
||||
|
||||
/**
|
||||
* Run the database seeds.
|
||||
|
@ -16,83 +14,6 @@ class CanastaSeeder extends Seeder
|
|||
*/
|
||||
public function run()
|
||||
{
|
||||
$csv = Reader::createFromPath(resource_path('csv/productos.csv'), 'r');
|
||||
$csv->setDelimiter("|");
|
||||
$iHeader = $this->obtenerIndiceDeHeader($csv);
|
||||
$csv->setHeaderOffset($iHeader);
|
||||
$registros = $csv->getRecords();
|
||||
|
||||
$toInsert = [];
|
||||
$categoria = '';
|
||||
foreach($registros as $i => $registro){
|
||||
//filas que están arriba del header
|
||||
if ($i <= $iHeader){
|
||||
continue;
|
||||
}
|
||||
|
||||
//finalizar
|
||||
if ($registro[$this::FILA_HEADER] == $this::ULTIMA_FILA){
|
||||
break;
|
||||
}
|
||||
|
||||
//filas que no tienen tipo
|
||||
if (!Arr::has($registro,$this::FILA_HEADER)|| trim($registro[$this::FILA_HEADER]) == ''){
|
||||
var_dump("no hay tipo en la fila " . $i);
|
||||
continue;
|
||||
}
|
||||
|
||||
//saltear bono de transporte
|
||||
if ($registro[$this::FILA_HEADER] == "T"){
|
||||
continue;
|
||||
}
|
||||
|
||||
//obtener categoria
|
||||
if ($registro['Producto'] == '') {
|
||||
//es la pregunta de la copa?
|
||||
if (Str::contains($registro[$this::FILA_HEADER],"¿")) { continue; }
|
||||
$categoria = $registro[$this::FILA_HEADER];
|
||||
continue;
|
||||
}
|
||||
|
||||
//completar producto
|
||||
$toInsert[] = [
|
||||
'fila' => $i,
|
||||
'categoria' => $categoria,
|
||||
'nombre' => trim(str_replace('*', ' ',$registro['Producto'])),
|
||||
'precio' => $registro['Precio'],
|
||||
'proveedor_id' => $this->obtenerProveedor($registro['Producto']),
|
||||
'bono' => $registro[$this::FILA_HEADER] == "B",
|
||||
'requiere_notas'=> $registro[$this::FILA_HEADER] =="PTC",
|
||||
];
|
||||
}
|
||||
|
||||
foreach (array_chunk($toInsert,DatabaseSeeder::CHUNK_SIZE) as $chunk)
|
||||
{
|
||||
DB::table('productos')->insert($chunk);
|
||||
}
|
||||
}
|
||||
|
||||
private function obtenerIndiceDeHeader($csv){
|
||||
$registros = $csv->getRecords();
|
||||
$iheader = 0;
|
||||
foreach ($registros as $i => $registro){
|
||||
if (strtolower($registro[0]) == strtolower($this::FILA_HEADER)) {
|
||||
$iheader = $i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return $iheader;
|
||||
}
|
||||
|
||||
private function obtenerProveedor($nombre) {
|
||||
$result = null;
|
||||
if (Str::contains($nombre,"*")){
|
||||
$result = Proveedor::firstOrCreate([
|
||||
'nombre' => 'Proveedor de economía solidaria',
|
||||
'economia_solidaria' => 1,
|
||||
'nacional' => 1
|
||||
])->id;
|
||||
}
|
||||
return $result;
|
||||
CanastaHelper::cargarCanasta(self::ARCHIVO_DEFAULT);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -32,7 +32,7 @@ Vue.prototype.$rootMiga = {
|
|||
Vue.prototype.$settearProducto = function(cantidad, id) {
|
||||
Event.$emit("sync-subpedido", this.cant, this.producto.id)
|
||||
}
|
||||
Vue.prototype.$toast = function(mensaje, duration = 1000) {
|
||||
Vue.prototype.$toast = function(mensaje, duration = 2000) {
|
||||
return window.bulmaToast.toast({
|
||||
message: mensaje,
|
||||
duration: duration,
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
<template>
|
||||
<div class="block ml-3 mr-3 is-max-widescreen is-max-desktop">
|
||||
<admin-tabs-secciones></admin-tabs-secciones>
|
||||
<comunes-tabs-secciones :tabs="tabs" :tabInicial="tabActiva"></comunes-tabs-secciones>
|
||||
<div class="block" id="pedidos-seccion"
|
||||
:class="seccionActiva === 'pedidos-seccion' ? 'is-active' : 'is-hidden'">
|
||||
<div class="block pb-6" id="pedidos-tabla-y-dropdown" v-show="hayPedidos">
|
||||
|
@ -34,7 +34,7 @@
|
|||
|
||||
<script>
|
||||
import CaracteristicasOpcionales from "./CaracteristicasOpcionales.vue";
|
||||
import TabsSecciones from "./TabsSecciones.vue";
|
||||
import TabsSecciones from "../comunes/TabsSecciones.vue";
|
||||
import DropdownDescargar from "./DropdownDescargar.vue";
|
||||
import TablaPedidos from "./TablaPedidos.vue";
|
||||
import TablaBonos from "./TablaBonos.vue";
|
||||
|
@ -52,6 +52,9 @@ export default {
|
|||
pedidos: [],
|
||||
bonosDeTransporte: 0,
|
||||
totalBonosBarriales: 0,
|
||||
tabs: [{ id: "pedidos", nombre: "Pedidos" },
|
||||
{ id: "bonos", nombre: "Bonos" },
|
||||
{ id: "caracteristicas", nombre: "Caracteristicas opcionales" }],
|
||||
tabActiva: "pedidos",
|
||||
seccionActiva: "pedidos-seccion",
|
||||
}
|
||||
|
|
|
@ -6,7 +6,9 @@ export default {
|
|||
caracteristica: Object
|
||||
},
|
||||
data() {
|
||||
gdc: undefined
|
||||
return {
|
||||
gdc: undefined
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
'$root.gdc' : {
|
||||
|
|
|
@ -1,42 +1,75 @@
|
|||
<template>
|
||||
<div class="container is-fluid has-text-centered">
|
||||
<div class="block">
|
||||
<div class="field">
|
||||
<p class="control">
|
||||
<a href="/compras/pedidos/descargar" class="button">
|
||||
<span class="icon is-small">
|
||||
<i class="fas fa-download"></i>
|
||||
</span>
|
||||
<span>Descargar planilla de totales</span>
|
||||
</a>
|
||||
</p>
|
||||
<div class="block ml-3 mr-3 is-max-widescreen is-max-desktop">
|
||||
<comunes-tabs-secciones :tabs="tabs" :tabInicial="tabActiva"></comunes-tabs-secciones>
|
||||
<div class="block pb-6" id="pedidos-compras-seccion"
|
||||
:class="seccionActiva === 'pedidos-compras-seccion' ? 'is-active' : 'is-hidden'">
|
||||
<div class="block" id="pedidos-compras-tabla-y-dropdown">
|
||||
<compras-dropdown-descargar>
|
||||
</compras-dropdown-descargar>
|
||||
</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 class="field">
|
||||
<p class="control">
|
||||
<a href="/compras/pedidos/transporte" class="button">
|
||||
<span class="icon is-small">
|
||||
<i class="fa fa-truck"></i>
|
||||
</span>
|
||||
<span>Descargar planilla de transporte</span>
|
||||
</a>
|
||||
</p>
|
||||
</div>
|
||||
<div class="block pb-6" id="canasta-compras-seccion"
|
||||
:class="seccionActiva === 'canasta-compras-seccion' ? 'is-active' : 'is-hidden'">
|
||||
<div class="block" id="canasta-compras-seccion">
|
||||
<article class="message is-warning">
|
||||
<div class="message-header">
|
||||
<p>Formato de la canasta</p>
|
||||
</div>
|
||||
<div class="message-body">
|
||||
<div class="content">
|
||||
La planilla de la canasta tiene que tener el siguiente formato para que la aplicación la lea correctamente:
|
||||
<ul>
|
||||
<li> Los precios deben usar punto y no coma decimal </li>
|
||||
<li> El nombre de la columna de precios debe ser "Precio" </li>
|
||||
<li> Las celdas deben separarse con '|' </li>
|
||||
<li> No puede haber "enters" en ninguna celda </li>
|
||||
<li> Todos los bonos deben tener tipo 'B' para evitar que paguen transporte </li>
|
||||
<li> El bono de transporte debe tener tipo 'T' </li>
|
||||
</ul>
|
||||
<a class="has-text-info" href="/compras/canasta/ejemplo">Planilla de ejemplo.</a>
|
||||
<article class="message is-danger mt-2">
|
||||
<div class="message-body">
|
||||
<div class="content">
|
||||
Cuidado! Cargar una nueva canasta elimina todos los pedidos de la aplicación.
|
||||
</div>
|
||||
</div>
|
||||
</article>
|
||||
</div>
|
||||
</div>
|
||||
</article>
|
||||
<div class="buttons is-right">
|
||||
<compras-canasta-input></compras-canasta-input>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
import TabsSecciones from "../comunes/TabsSecciones.vue";
|
||||
import DropdownDescargar from "./DropdownDescargar.vue";
|
||||
import CanastaInput from "./CanastaInput.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
TabsSecciones,
|
||||
DropdownDescargar,
|
||||
CanastaInput,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
tabs: [{ id: "pedidos-compras", nombre: "Pedidos" },
|
||||
{ id: "canasta-compras", nombre: "Canasta" }],
|
||||
tabActiva: "pedidos-compras",
|
||||
seccionActiva: "pedidos-compras-seccion",
|
||||
archivo: undefined,
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
setSeccionActiva(tabId) {
|
||||
this.tabActiva = tabId;
|
||||
this.seccionActiva = tabId + "-seccion";
|
||||
},
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
|
|
@ -0,0 +1,69 @@
|
|||
<template>
|
||||
<div class="block">
|
||||
<div class="file has-name is-right">
|
||||
<label class="file-label">
|
||||
<input
|
||||
class="file-input"
|
||||
type="file"
|
||||
name="canasta"
|
||||
@change="archivoSubido"
|
||||
/>
|
||||
<span class="file-cta">
|
||||
<span class="file-icon">
|
||||
<i class="fas fa-cloud-upload-alt"></i>
|
||||
</span>
|
||||
<span class="file-label">Subir canasta</span>
|
||||
</span>
|
||||
<span class="file-name" v-if="archivo">
|
||||
{{ 'Cargando ' + archivo.nombre }}
|
||||
</span>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import axios from "axios";
|
||||
|
||||
export default {
|
||||
name: "CanastaInput",
|
||||
data() {
|
||||
return {
|
||||
archivo: null,
|
||||
cargando: false,
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
async archivoSubido(event) {
|
||||
const archivo = event.target.files[0];
|
||||
if (archivo && archivo.type === "text/csv") {
|
||||
this.archivo = {data: archivo, nombre: archivo.name};
|
||||
const formData = new FormData();
|
||||
formData.append("data", this.archivo.data);
|
||||
|
||||
try {
|
||||
this.cargando = true;
|
||||
const response = await axios.post("/compras/canasta", formData, {
|
||||
headers: {
|
||||
"Content-Type": "multipart/form-data",
|
||||
},
|
||||
});
|
||||
this.$root.$toast(response.data.message || "Canasta cargada exitosamente");
|
||||
} catch (error) {
|
||||
this.$root.$toast(error.response?.data?.message || "Hubo errores.");
|
||||
} finally {
|
||||
this.cargando = false;
|
||||
this.archivo = null;
|
||||
}
|
||||
} else {
|
||||
this.$root.$toast("La canasta debe ser .CSV")
|
||||
this.archivo = null;
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
|
@ -0,0 +1,44 @@
|
|||
<template>
|
||||
<div class="buttons is-right">
|
||||
<div class="dropdown" :class="{'is-active': dropdownActivo}" @mouseleave="dropdownActivo = false">
|
||||
<div class="dropdown-trigger">
|
||||
<button class="button" aria-haspopup="true" aria-controls="dropdown-menu" @click="dropdownActivo = !dropdownActivo">
|
||||
<span class="icon is-small">
|
||||
<i class="fas fa-download"></i>
|
||||
</span>
|
||||
<span>Descargar planillas</span>
|
||||
<span class="icon is-small">
|
||||
<i class="fas fa-angle-down" aria-hidden="true"></i>
|
||||
</span>
|
||||
</button>
|
||||
</div>
|
||||
<div class="dropdown-menu" id="dropdown-menu" role="menu">
|
||||
<div class="dropdown-content">
|
||||
<a href="/compras/pedidos/descargar" class="dropdown-item">
|
||||
Pedidos por barrio
|
||||
</a>
|
||||
<a href="/compras/pedidos/notas" class="dropdown-item">
|
||||
Notas por barrio
|
||||
</a>
|
||||
<a href="/compras/pedidos/transporte" class="dropdown-item">
|
||||
Transporte por barrio
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
data() {
|
||||
return {
|
||||
dropdownActivo: false
|
||||
}
|
||||
},
|
||||
}
|
||||
</script>
|
||||
|
||||
<style>
|
||||
|
||||
</style>
|
|
@ -19,23 +19,13 @@
|
|||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
tabs: Array,
|
||||
tabInicial: String,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
tabActiva: "pedidos",
|
||||
tabs: [
|
||||
{
|
||||
id: "pedidos",
|
||||
nombre: "Pedidos"
|
||||
},
|
||||
{
|
||||
id: "bonos",
|
||||
nombre: "Bonos"
|
||||
},
|
||||
{
|
||||
id: "caracteristicas",
|
||||
nombre: "Caracteristicas opcionales"
|
||||
}
|
||||
]
|
||||
tabActiva: this.tabInicial,
|
||||
}
|
||||
},
|
||||
methods: {
|
|
@ -83,4 +83,6 @@ Route::middleware(['compras'])->group( function() {
|
|||
Route::get('/compras/pedidos/descargar', 'ComprasController@descargarPedidos')->name('compras.pedidos.descargar');
|
||||
Route::get('/compras/pedidos/notas', 'ComprasController@descargarNotas')->name('compras.pedidos.descargar');
|
||||
Route::get('/compras/pedidos/transporte', 'ComprasController@descargarTransporte')->name('compras.pedidos.descargar');
|
||||
Route::post('/compras/canasta', 'ComprasController@cargarCanasta')->name('compras.canasta');
|
||||
Route::get('/compras/canasta/ejemplo', 'ComprasController@descargarCanastaEjemplo')->name('compras.canasta.ejemplo');
|
||||
});
|
||||
|
|
Loading…
Reference in New Issue