feat: Sistema de gestión de impresoras persistente
- Crear modelo Printer con campos: id, name, host, deviceId, timeout, isDefault - Almacenamiento persistente en data/printers.json - APIs CRUD: GET/POST /api/printers, GET/PUT/DELETE /api/printers/:id - API para seleccionar impresora activa: POST /api/printers/select - Endpoint de impresión ahora usa la impresora seleccionada o la especificada por printerId - Composable usePrinters() para el cliente - UI: Selector de impresora en sidebar, tab Impresoras en mobile - Componentes: PrintersList, PrintersCard, PrintersForm, PrintersSelector
This commit is contained in:
@@ -1,18 +1,20 @@
|
||||
// Endpoint genérico de impresión que acepta una lista de operaciones
|
||||
import { buildFromOperations, type Operation } from '../../utils/eposBuilder'
|
||||
import { buildSoapEnvelope, sendToPrinter, parsePrinterResponse } from '../../utils/printer'
|
||||
import { getSelectedPrinter, getPrinterById } from '../../utils/printers'
|
||||
|
||||
export default defineEventHandler(async (event) => {
|
||||
try {
|
||||
const config = useRuntimeConfig()
|
||||
const body = await readBody(event)
|
||||
|
||||
const {
|
||||
operations = [],
|
||||
dryRun = false
|
||||
dryRun = false,
|
||||
printerId
|
||||
} = body as {
|
||||
operations?: Operation[]
|
||||
dryRun?: boolean
|
||||
printerId?: string
|
||||
}
|
||||
|
||||
// Construir el XML interior con las operaciones
|
||||
@@ -28,12 +30,24 @@ export default defineEventHandler(async (event) => {
|
||||
}
|
||||
}
|
||||
|
||||
// Obtener la impresora a usar
|
||||
let printer = printerId
|
||||
? await getPrinterById(printerId)
|
||||
: await getSelectedPrinter()
|
||||
|
||||
if (!printer) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'No hay impresora configurada. Por favor, agrega una impresora primero.'
|
||||
}
|
||||
}
|
||||
|
||||
// Enviar a la impresora
|
||||
const result = await sendToPrinter(
|
||||
soap,
|
||||
config.printerHost,
|
||||
config.printerDeviceId,
|
||||
parseInt(config.printerTimeoutMs)
|
||||
printer.host,
|
||||
printer.deviceId,
|
||||
printer.timeout
|
||||
)
|
||||
|
||||
// Parsear la respuesta
|
||||
@@ -43,7 +57,11 @@ export default defineEventHandler(async (event) => {
|
||||
ok: success,
|
||||
httpStatus: result.status,
|
||||
code,
|
||||
raw: result.data
|
||||
raw: result.data,
|
||||
printerUsed: {
|
||||
id: printer.id,
|
||||
name: printer.name
|
||||
}
|
||||
}
|
||||
} catch (err: any) {
|
||||
return {
|
||||
|
||||
34
server/api/printers/[id].delete.ts
Normal file
34
server/api/printers/[id].delete.ts
Normal file
@@ -0,0 +1,34 @@
|
||||
// DELETE /api/printers/:id - Eliminar una impresora
|
||||
import { deletePrinter } from '../../utils/printers'
|
||||
|
||||
export default defineEventHandler(async (event) => {
|
||||
try {
|
||||
const id = getRouterParam(event, 'id')
|
||||
|
||||
if (!id) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'ID requerido'
|
||||
}
|
||||
}
|
||||
|
||||
const deleted = await deletePrinter(id)
|
||||
|
||||
if (!deleted) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'Impresora no encontrada'
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
ok: true,
|
||||
message: 'Impresora eliminada'
|
||||
}
|
||||
} catch (err: any) {
|
||||
return {
|
||||
ok: false,
|
||||
error: err.message
|
||||
}
|
||||
}
|
||||
})
|
||||
34
server/api/printers/[id].get.ts
Normal file
34
server/api/printers/[id].get.ts
Normal file
@@ -0,0 +1,34 @@
|
||||
// GET /api/printers/:id - Obtener una impresora por ID
|
||||
import { getPrinterById } from '../../utils/printers'
|
||||
|
||||
export default defineEventHandler(async (event) => {
|
||||
try {
|
||||
const id = getRouterParam(event, 'id')
|
||||
|
||||
if (!id) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'ID requerido'
|
||||
}
|
||||
}
|
||||
|
||||
const printer = await getPrinterById(id)
|
||||
|
||||
if (!printer) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'Impresora no encontrada'
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
ok: true,
|
||||
printer
|
||||
}
|
||||
} catch (err: any) {
|
||||
return {
|
||||
ok: false,
|
||||
error: err.message
|
||||
}
|
||||
}
|
||||
})
|
||||
50
server/api/printers/[id].put.ts
Normal file
50
server/api/printers/[id].put.ts
Normal file
@@ -0,0 +1,50 @@
|
||||
// PUT /api/printers/:id - Actualizar una impresora
|
||||
import { updatePrinter } from '../../utils/printers'
|
||||
|
||||
export default defineEventHandler(async (event) => {
|
||||
try {
|
||||
const id = getRouterParam(event, 'id')
|
||||
|
||||
if (!id) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'ID requerido'
|
||||
}
|
||||
}
|
||||
|
||||
const body = await readBody(event)
|
||||
|
||||
const { name, host, deviceId, timeout, isDefault } = body as {
|
||||
name?: string
|
||||
host?: string
|
||||
deviceId?: string
|
||||
timeout?: number
|
||||
isDefault?: boolean
|
||||
}
|
||||
|
||||
const printer = await updatePrinter(id, {
|
||||
name,
|
||||
host,
|
||||
deviceId,
|
||||
timeout,
|
||||
isDefault
|
||||
})
|
||||
|
||||
if (!printer) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'Impresora no encontrada'
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
ok: true,
|
||||
printer
|
||||
}
|
||||
} catch (err: any) {
|
||||
return {
|
||||
ok: false,
|
||||
error: err.message
|
||||
}
|
||||
}
|
||||
})
|
||||
20
server/api/printers/index.get.ts
Normal file
20
server/api/printers/index.get.ts
Normal file
@@ -0,0 +1,20 @@
|
||||
// GET /api/printers - Listar todas las impresoras
|
||||
import { getAllPrinters, getSelectedPrinter } from '../../utils/printers'
|
||||
|
||||
export default defineEventHandler(async () => {
|
||||
try {
|
||||
const printers = await getAllPrinters()
|
||||
const selected = await getSelectedPrinter()
|
||||
|
||||
return {
|
||||
ok: true,
|
||||
printers,
|
||||
selectedPrinterId: selected?.id || null
|
||||
}
|
||||
} catch (err: any) {
|
||||
return {
|
||||
ok: false,
|
||||
error: err.message
|
||||
}
|
||||
}
|
||||
})
|
||||
41
server/api/printers/index.post.ts
Normal file
41
server/api/printers/index.post.ts
Normal file
@@ -0,0 +1,41 @@
|
||||
// POST /api/printers - Crear nueva impresora
|
||||
import { createPrinter } from '../../utils/printers'
|
||||
|
||||
export default defineEventHandler(async (event) => {
|
||||
try {
|
||||
const body = await readBody(event)
|
||||
|
||||
const { name, host, deviceId, timeout, isDefault } = body as {
|
||||
name: string
|
||||
host: string
|
||||
deviceId: string
|
||||
timeout?: number
|
||||
isDefault?: boolean
|
||||
}
|
||||
|
||||
if (!name || !host || !deviceId) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'name, host y deviceId son requeridos'
|
||||
}
|
||||
}
|
||||
|
||||
const printer = await createPrinter({
|
||||
name,
|
||||
host,
|
||||
deviceId,
|
||||
timeout,
|
||||
isDefault
|
||||
})
|
||||
|
||||
return {
|
||||
ok: true,
|
||||
printer
|
||||
}
|
||||
} catch (err: any) {
|
||||
return {
|
||||
ok: false,
|
||||
error: err.message
|
||||
}
|
||||
}
|
||||
})
|
||||
31
server/api/printers/select.post.ts
Normal file
31
server/api/printers/select.post.ts
Normal file
@@ -0,0 +1,31 @@
|
||||
// POST /api/printers/select - Seleccionar la impresora activa
|
||||
import { setSelectedPrinter, getPrinterById } from '../../utils/printers'
|
||||
|
||||
export default defineEventHandler(async (event) => {
|
||||
try {
|
||||
const body = await readBody(event)
|
||||
const { printerId } = body as { printerId: string | null }
|
||||
|
||||
if (printerId) {
|
||||
const printer = await getPrinterById(printerId)
|
||||
if (!printer) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'Impresora no encontrada'
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
await setSelectedPrinter(printerId)
|
||||
|
||||
return {
|
||||
ok: true,
|
||||
selectedPrinterId: printerId
|
||||
}
|
||||
} catch (err: any) {
|
||||
return {
|
||||
ok: false,
|
||||
error: err.message
|
||||
}
|
||||
}
|
||||
})
|
||||
Reference in New Issue
Block a user