Juan Felipe Zapata Moreno 2c7d2f2001 feat: agregar componentes de gestión de almacenes y movimientos
- Implementar vistas CRUD para administración de almacenes (Index, Create, Edit, Delete).
- Añadir  para realizar traspasos de productos entre almacenes.
- Configurar lógica de rutas y API (Module.js) para almacenes y movimientos.
2026-02-06 00:01:45 -06:00

305 lines
14 KiB
Vue

<script setup>
import { onMounted, ref, computed } from 'vue';
import { useSearcher, apiURL } from '@Services/Api';
import { can } from './Module.js';
import { formatDate } from '@/utils/formatters';
import SearcherHead from '@Holos/Searcher.vue';
import Table from '@Holos/Table.vue';
import GoogleIcon from '@Shared/GoogleIcon.vue';
import EntryModal from './EntryModal.vue';
import ExitModal from './ExitModal.vue';
import TransferModal from './TransferModal.vue';
import DetailModal from './DetailModal.vue';
/** Estado */
const movements = ref({});
const selectedType = ref('');
const selectedWarehouse = ref('');
const fromDate = ref('');
const toDate = ref('');
const warehouses = ref([]);
const showEntryModal = ref(false);
const showExitModal = ref(false);
const showTransferModal = ref(false);
const showDetailModal = ref(false);
const selectedMovementId = ref(null);
/** Filtros computados */
const filters = computed(() => {
const f = {};
if (selectedType.value) f.movement_type = selectedType.value;
if (selectedWarehouse.value) f.warehouse_id = selectedWarehouse.value;
if (fromDate.value) f.from_date = fromDate.value;
if (toDate.value) f.to_date = toDate.value;
return f;
});
/** Tipos de movimiento */
const movementTypes = [
{ value: '', label: 'Todos', icon: 'list', active: 'bg-gray-600 text-white shadow-sm', inactive: 'bg-gray-100 text-gray-700 dark:bg-gray-900/30 dark:text-gray-300 hover:bg-gray-200 dark:hover:bg-gray-900/50' },
{ value: 'entry', label: 'Entradas', icon: 'add_circle', active: 'bg-green-600 text-white shadow-sm', inactive: 'bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-300 hover:bg-green-200 dark:hover:bg-green-900/50' },
{ value: 'exit', label: 'Salidas', icon: 'remove_circle', active: 'bg-red-600 text-white shadow-sm', inactive: 'bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-300 hover:bg-red-200 dark:hover:bg-red-900/50' },
{ value: 'transfer', label: 'Traspasos', icon: 'swap_horiz', active: 'bg-blue-600 text-white shadow-sm', inactive: 'bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-300 hover:bg-blue-200 dark:hover:bg-blue-900/50' },
/* { value: 'sale', label: 'Ventas', icon: 'point_of_sale', active: 'bg-purple-600 text-white shadow-sm', inactive: 'bg-purple-100 text-purple-700 dark:bg-purple-900/30 dark:text-purple-300 hover:bg-purple-200 dark:hover:bg-purple-900/50' },
{ value: 'return', label: 'Devoluciones', icon: 'undo', active: 'bg-amber-600 text-white shadow-sm', inactive: 'bg-amber-100 text-amber-700 dark:bg-amber-900/30 dark:text-amber-300 hover:bg-amber-200 dark:hover:bg-amber-900/50' },
*/];
const getTypeBadge = (type) => {
const badges = {
entry: { label: 'Entrada', class: 'bg-green-100 text-green-800 dark:bg-green-900/30 dark:text-green-300' },
exit: { label: 'Salida', class: 'bg-red-100 text-red-800 dark:bg-red-900/30 dark:text-red-300' },
transfer: { label: 'Traspaso', class: 'bg-blue-100 text-blue-800 dark:bg-blue-900/30 dark:text-blue-300' },
/* sale: { label: 'Venta', class: 'bg-purple-100 text-purple-800 dark:bg-purple-900/30 dark:text-purple-300' },
return: { label: 'Devolución', class: 'bg-amber-100 text-amber-800 dark:bg-amber-900/30 dark:text-amber-300' }, */
};
return badges[type] || { label: type, class: 'bg-gray-100 text-gray-800' };
};
/** Searcher */
const searcher = useSearcher({
url: apiURL('movimientos'),
onSuccess: (r) => {
movements.value = r.movements || r;
},
onError: () => movements.value = {}
});
/** Métodos */
const loadWarehouses = async () => {
try {
const response = await fetch(apiURL('almacenes'), {
headers: {
'Authorization': `Bearer ${sessionStorage.token}`,
'Accept': 'application/json'
}
});
const result = await response.json();
if (result.data) {
warehouses.value = result.data.warehouses?.data || result.data.data || [];
}
} catch (error) {
console.error('Error loading warehouses:', error);
}
};
const applyFilters = () => {
searcher.search('', filters.value);
};
const selectType = (type) => {
selectedType.value = type;
applyFilters();
};
const openDetail = (movement) => {
selectedMovementId.value = movement.id;
showDetailModal.value = true;
};
const closeDetailModal = () => {
showDetailModal.value = false;
selectedMovementId.value = null;
};
const onMovementCreated = () => {
applyFilters();
};
/** Ciclo de vida */
onMounted(() => {
searcher.search();
loadWarehouses();
});
</script>
<template>
<div>
<SearcherHead
:title="$t('movements.title')"
placeholder="Buscar movimientos..."
@search="(x) => searcher.search(x, filters)"
>
<div class="flex items-center gap-2">
<button
v-if="can('create')"
class="flex items-center gap-1.5 px-3 py-2 bg-green-600 hover:bg-green-700 text-white text-sm font-semibold rounded-lg transition-colors shadow-sm"
@click="showEntryModal = true"
>
<GoogleIcon name="add_circle" class="text-lg" />
Entrada
</button>
<button
v-if="can('create')"
class="flex items-center gap-1.5 px-3 py-2 bg-red-600 hover:bg-red-700 text-white text-sm font-semibold rounded-lg transition-colors shadow-sm"
@click="showExitModal = true"
>
<GoogleIcon name="remove_circle" class="text-lg" />
Salida
</button>
<button
v-if="can('create')"
class="flex items-center gap-1.5 px-3 py-2 bg-blue-600 hover:bg-blue-700 text-white text-sm font-semibold rounded-lg transition-colors shadow-sm"
@click="showTransferModal = true"
>
<GoogleIcon name="swap_horiz" class="text-lg" />
Traspaso
</button>
</div>
</SearcherHead>
<!-- Filtros -->
<div class="mb-4 space-y-3">
<!-- Chips de tipo -->
<div class="flex flex-wrap gap-2">
<button
v-for="type in movementTypes"
:key="type.value"
@click="selectType(type.value)"
:class="[
'inline-flex items-center gap-1.5 px-3 py-1.5 rounded-full text-xs font-semibold transition-all',
selectedType === type.value ? type.active : type.inactive
]"
>
<GoogleIcon :name="type.icon" class="text-sm" />
{{ type.label }}
</button>
</div>
<!-- Filtros adicionales -->
<div class="flex flex-wrap items-end gap-3">
<div>
<label class="block text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase mb-1">Almacén</label>
<select
v-model="selectedWarehouse"
@change="applyFilters"
class="px-3 py-1.5 text-sm border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-800 text-gray-900 dark:text-gray-100 focus:ring-2 focus:ring-indigo-500"
>
<option value="">Todos</option>
<option v-for="wh in warehouses" :key="wh.id" :value="wh.id">
{{ wh.name }}
</option>
</select>
</div>
<div>
<label class="block text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase mb-1">Desde</label>
<input
v-model="fromDate"
@change="applyFilters"
type="date"
class="px-3 py-1.5 text-sm border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-800 text-gray-900 dark:text-gray-100 focus:ring-2 focus:ring-indigo-500"
/>
</div>
<div>
<label class="block text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase mb-1">Hasta</label>
<input
v-model="toDate"
@change="applyFilters"
type="date"
class="px-3 py-1.5 text-sm border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-800 text-gray-900 dark:text-gray-100 focus:ring-2 focus:ring-indigo-500"
/>
</div>
<button
v-if="selectedWarehouse || fromDate || toDate"
@click="selectedWarehouse = ''; fromDate = ''; toDate = ''; selectedType = ''; applyFilters();"
class="px-3 py-1.5 text-sm text-gray-600 dark:text-gray-400 hover:text-gray-900 dark:hover:text-gray-100 transition-colors"
>
Limpiar filtros
</button>
</div>
</div>
<!-- Tabla -->
<div class="pt-2 w-full">
<Table
:items="movements"
@send-pagination="(page) => searcher.pagination(page, filters)"
>
<template #head>
<th class="px-6 py-3 text-left text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider">PRODUCTO</th>
<th class="px-6 py-3 text-center text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider">TIPO</th>
<th class="px-6 py-3 text-center text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider">CANTIDAD</th>
<th class="px-6 py-3 text-center text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider">ORIGEN</th>
<th class="px-6 py-3 text-center text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider">DESTINO</th>
<th class="px-6 py-3 text-center text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider">USUARIO</th>
<th class="px-6 py-3 text-center text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider">FECHA</th>
</template>
<template #body="{items}">
<tr
v-for="movement in items"
:key="movement.id"
class="hover:bg-gray-50 dark:hover:bg-gray-800 transition-colors cursor-pointer"
@click="openDetail(movement)"
>
<td class="px-6 py-4 text-left">
<p class="text-sm font-semibold text-gray-900 dark:text-gray-100">{{ movement.inventory?.name || 'N/A' }}</p>
<p class="text-xs text-gray-500 dark:text-gray-400 font-mono">{{ movement.inventory?.sku || '' }}</p>
</td>
<td class="px-6 py-4 text-center">
<span :class="['inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium', getTypeBadge(movement.movement_type).class]">
{{ getTypeBadge(movement.movement_type).label }}
</span>
</td>
<td class="px-6 py-4 text-center">
<p class="text-sm font-bold text-gray-900 dark:text-gray-100">{{ movement.quantity }}</p>
</td>
<td class="px-6 py-4 text-center">
<p v-if="movement.warehouse_from" class="text-sm text-gray-700 dark:text-gray-300">{{ movement.warehouse_from.name }}</p>
<span v-else class="text-sm text-gray-400 dark:text-gray-500">—</span>
</td>
<td class="px-6 py-4 text-center">
<p v-if="movement.warehouse_to" class="text-sm text-gray-700 dark:text-gray-300">{{ movement.warehouse_to.name }}</p>
<span v-else class="text-sm text-gray-400 dark:text-gray-500">—</span>
</td>
<td class="px-6 py-4 text-center">
<p class="text-sm text-gray-700 dark:text-gray-300">{{ movement.user?.name || 'N/A' }}</p>
</td>
<td class="px-6 py-4 text-center">
<p class="text-sm text-gray-700 dark:text-gray-300">{{ formatDate(movement.created_at) }}</p>
</td>
</tr>
</template>
<template #empty>
<td colspan="7" class="table-cell text-center">
<div class="flex flex-col items-center justify-center py-8 text-gray-500">
<GoogleIcon
name="swap_horiz"
class="text-6xl mb-2 opacity-50"
/>
<p class="font-semibold">
{{ $t('registers.empty') }}
</p>
</div>
</td>
</template>
</Table>
</div>
<!-- Modales -->
<EntryModal
v-if="can('create')"
:show="showEntryModal"
@close="showEntryModal = false"
@created="onMovementCreated"
/>
<ExitModal
v-if="can('create')"
:show="showExitModal"
@close="showExitModal = false"
@created="onMovementCreated"
/>
<TransferModal
v-if="can('create')"
:show="showTransferModal"
@close="showTransferModal = false"
@created="onMovementCreated"
/>
<DetailModal
:show="showDetailModal"
:movement-id="selectedMovementId"
@close="closeDetailModal"
/>
</div>
</template>