2023-11-14 21:39:15 +00:00
|
|
|
import { For, createMemo, createSignal } from "solid-js";
|
2023-11-13 19:29:14 +00:00
|
|
|
import { FilledButton } from "../components/FilledButton";
|
|
|
|
import { FilledCard } from "../components/FilledCard";
|
2023-11-14 21:39:15 +00:00
|
|
|
import { LoadingIcon } from "../icons/LoadingIcon";
|
2023-11-13 21:59:37 +00:00
|
|
|
import { MagnifyingGlassIcon } from "../icons/MagnifyingGlassIcon";
|
2023-11-14 21:39:15 +00:00
|
|
|
import { LoadingStatus, useLoading } from "../utils/functions";
|
2023-11-16 15:35:30 +00:00
|
|
|
import { PDFIcon } from "../icons/PDFIcon";
|
2023-11-14 21:39:15 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Represents the data about the scans that were detected.
|
|
|
|
* Serialization & Deserialization is done by the backend.
|
|
|
|
*/
|
|
|
|
interface ScanData {
|
|
|
|
Ok: Array<[string, ScanResult]>,
|
|
|
|
}
|
|
|
|
|
|
|
|
type ScanResult =
|
|
|
|
| {Full: [string, number, string]}
|
|
|
|
| {Partial: [string, string]}
|
|
|
|
| {Error: string};
|
|
|
|
|
|
|
|
function dataFromScanResult(result: ScanResult) {
|
|
|
|
if ("Full" in result) {
|
|
|
|
return `DNI: ${result.Full[0]}, iid: ${result.Full[1]}`;
|
|
|
|
} else if ("Partial" in result) {
|
|
|
|
return `DNI: ${result.Partial}`;
|
|
|
|
} else if ("Error" in result) {
|
|
|
|
return `Error: ${result.Error}`;
|
|
|
|
}
|
|
|
|
}
|
2023-11-13 19:29:14 +00:00
|
|
|
|
|
|
|
export function Scans() {
|
2023-11-14 21:39:15 +00:00
|
|
|
const {status, setStatus, error, setError} = useLoading();
|
|
|
|
const [scanData, setScanData] = createSignal<ScanData | null>(null);
|
|
|
|
|
|
|
|
const loading = createMemo(() => status() === LoadingStatus.Loading);
|
|
|
|
|
2023-11-13 19:29:14 +00:00
|
|
|
const detectScans = () => {
|
2023-11-14 21:39:15 +00:00
|
|
|
setStatus(LoadingStatus.Loading);
|
2023-11-16 15:35:30 +00:00
|
|
|
setScanData(null);
|
2023-11-14 21:39:15 +00:00
|
|
|
|
2023-11-13 19:29:14 +00:00
|
|
|
fetch(`${import.meta.env.VITE_BACKEND_URL}/api/scans/detect`)
|
|
|
|
.then((res) => res.json())
|
2023-11-14 21:39:15 +00:00
|
|
|
.then((data) => {
|
|
|
|
setStatus(LoadingStatus.Ok);
|
|
|
|
setScanData(data);
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
setStatus(LoadingStatus.Error);
|
|
|
|
console.error(err);
|
|
|
|
});
|
2023-11-13 19:29:14 +00:00
|
|
|
};
|
|
|
|
|
2023-11-16 15:35:30 +00:00
|
|
|
const convertScans = () => {
|
|
|
|
setStatus(LoadingStatus.Loading);
|
|
|
|
|
|
|
|
if (scanData() === null) {
|
|
|
|
setError("No se detectaron escaneos");
|
|
|
|
setStatus(LoadingStatus.Error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fetch(
|
|
|
|
`${import.meta.env.VITE_BACKEND_URL}/api/scans/convert`,
|
|
|
|
{
|
|
|
|
method: "POST",
|
|
|
|
headers: {
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
},
|
|
|
|
body: JSON.stringify(scanData()),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.then((res) => res.json())
|
|
|
|
.then((res) => {
|
|
|
|
setStatus(LoadingStatus.Ok);
|
|
|
|
console.log(res);
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
setStatus(LoadingStatus.Error);
|
|
|
|
console.error(err);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2023-11-13 19:29:14 +00:00
|
|
|
return (
|
|
|
|
<div class="grid grid-cols-[25rem_25rem_auto]">
|
|
|
|
<div>
|
|
|
|
<FilledCard>
|
|
|
|
<h1 class="py-2 px-4 font-medium text-xl mb-2">
|
|
|
|
Parámetros de detección
|
|
|
|
</h1>
|
|
|
|
<div class="px-4 pb-4">
|
|
|
|
Para que el sistema detecte y clasifique automáticamente los escaneos
|
|
|
|
se debe cumplir lo siguiente:
|
|
|
|
<ol class="py-2 px-2 list-decimal list-inside">
|
|
|
|
<li>
|
|
|
|
Escanear el certificado en formato
|
|
|
|
<code class="border border-c-outline rounded p-1 bg-c-background">
|
|
|
|
jpg
|
|
|
|
</code>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
El nombre del archivo debe iniciar con
|
|
|
|
<code class="border border-c-outline rounded p-1 bg-c-background">
|
|
|
|
eeg
|
|
|
|
</code>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
El nombre del archivo debe terminar en
|
|
|
|
<code class="border border-c-outline rounded p-1 bg-c-background">
|
|
|
|
.jpg
|
|
|
|
</code>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
El nombre del archivo puede tener otras letras en medio,
|
|
|
|
siempre que se cumplan las condiciones anteriores.
|
|
|
|
</li>
|
|
|
|
</ol>
|
|
|
|
Por ejemplo, los siguientes nombres de archivo son válidos:
|
|
|
|
<ul class="py-2 px-2 list-disc list-inside">
|
|
|
|
<li>
|
|
|
|
<code class="border border-c-outline rounded p-1 bg-c-background">
|
|
|
|
eeg.jpg
|
|
|
|
</code>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<code class="border border-c-outline rounded p-1 bg-c-background">
|
|
|
|
eeg0001.jpg
|
|
|
|
</code>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<code class="border border-c-outline rounded p-1 bg-c-background">
|
|
|
|
eeg archivo con letras en medio.jpg
|
|
|
|
</code>
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
</FilledCard>
|
2023-11-14 21:39:15 +00:00
|
|
|
|
2023-11-13 19:29:14 +00:00
|
|
|
<FilledCard>
|
|
|
|
<h1 class="py-2 px-4 font-medium text-xl mb-2">
|
|
|
|
Detectar escaneos
|
|
|
|
</h1>
|
|
|
|
<div class="px-4 pb-4">
|
2023-11-13 21:59:37 +00:00
|
|
|
<p>
|
|
|
|
El siguiente botón detectará los archivos
|
|
|
|
que cumplen con los parámetros de detección
|
|
|
|
en la carpeta <code>ESCANEOS</code>, y los mostrará en una lista.
|
|
|
|
</p>
|
|
|
|
<p class="my-2">
|
|
|
|
La detección demora aprox. 1 segundo por cada archivo que
|
|
|
|
cumple con los parámetros.
|
|
|
|
</p>
|
|
|
|
<p class="my-2">
|
|
|
|
Si la detección demora más de 90 segundos, se cancelará.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="relative">
|
|
|
|
<FilledButton
|
|
|
|
onClick={detectScans}
|
2023-11-16 15:35:30 +00:00
|
|
|
disabled={loading()}
|
2023-11-13 21:59:37 +00:00
|
|
|
>
|
|
|
|
<span
|
2023-11-14 21:39:15 +00:00
|
|
|
class="absolute top-[1.35rem] left-2"
|
|
|
|
style={{display: loading() ? "none" : "inline-block"}}
|
2023-11-13 21:59:37 +00:00
|
|
|
>
|
|
|
|
<MagnifyingGlassIcon
|
|
|
|
fill="var(--c-on-primary)"
|
|
|
|
/>
|
|
|
|
</span>
|
2023-11-14 21:39:15 +00:00
|
|
|
<span
|
|
|
|
class="absolute top-[1.35rem] left-2"
|
|
|
|
style={{display: loading() ? "inline-block" : "none"}}
|
|
|
|
>
|
|
|
|
<LoadingIcon
|
|
|
|
class="animate-spin"
|
|
|
|
fill="var(--c-primary-container)"
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
|
2023-11-13 21:59:37 +00:00
|
|
|
<span class="ml-5">
|
|
|
|
Detectar escaneos
|
|
|
|
</span>
|
|
|
|
</FilledButton>
|
|
|
|
</div>
|
2023-11-13 19:29:14 +00:00
|
|
|
</div>
|
|
|
|
</FilledCard>
|
2023-11-14 21:39:15 +00:00
|
|
|
</div>
|
|
|
|
<div>
|
2023-11-13 19:29:14 +00:00
|
|
|
<FilledCard class="border border-c-outline overflow-hidden">
|
|
|
|
<h1 class="p-3 font-medium text-xl">
|
|
|
|
Escaneos detectados
|
|
|
|
</h1>
|
|
|
|
|
2023-11-14 21:39:15 +00:00
|
|
|
<div class="bg-c-surface py-2">
|
|
|
|
<For each={scanData()?.Ok ?? []}>
|
|
|
|
{([path, result]) => (
|
|
|
|
<p class="py-1 px-4 hover:bg-c-surface-variant transition-colors">
|
|
|
|
<span class="font-mono">
|
|
|
|
{path.substring(path.lastIndexOf("/") + 1)}
|
|
|
|
</span>
|
|
|
|
<br />
|
|
|
|
{dataFromScanResult(result)}
|
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
</For>
|
2023-11-13 19:29:14 +00:00
|
|
|
</div>
|
2023-11-14 21:39:15 +00:00
|
|
|
|
|
|
|
<p class="p-4">
|
2023-11-16 15:35:30 +00:00
|
|
|
- Al convertir los escaneos a PDF se eliminan los archivos JPG.
|
2023-11-14 21:39:15 +00:00
|
|
|
<br />
|
2023-11-16 15:35:30 +00:00
|
|
|
- Solo se convertiran los escaneos mostrados en esta lista.
|
|
|
|
<br />
|
|
|
|
- Si ejecutas la detección de escaneos nuevamente los nombres de
|
|
|
|
los archivos cambiarán. Esto es normal. El contenido de los JPG
|
|
|
|
es el mismo.
|
2023-11-14 21:39:15 +00:00
|
|
|
</p>
|
2023-11-16 15:35:30 +00:00
|
|
|
|
|
|
|
<div class="relative mx-4 mb-4">
|
|
|
|
<FilledButton
|
|
|
|
onClick={convertScans}
|
|
|
|
disabled={loading()}
|
|
|
|
>
|
|
|
|
<span
|
|
|
|
class="absolute top-[1.35rem] left-2"
|
|
|
|
style={{display: loading() ? "none" : "inline-block"}}
|
|
|
|
>
|
|
|
|
<PDFIcon
|
|
|
|
fill="var(--c-on-primary)"
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
<span
|
|
|
|
class="absolute top-[1.35rem] left-2"
|
|
|
|
style={{display: loading() ? "inline-block" : "none"}}
|
|
|
|
>
|
|
|
|
<LoadingIcon
|
|
|
|
class="animate-spin"
|
|
|
|
fill="var(--c-primary-container)"
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
|
|
|
|
<span class="ml-5">
|
|
|
|
Convertir escaneos
|
|
|
|
</span>
|
|
|
|
</FilledButton>
|
|
|
|
</div>
|
2023-11-13 19:29:14 +00:00
|
|
|
</FilledCard>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|