14. Mostrar datos en el cliente

Versión para imprimir.

A. Introducción

B. Diagrama de despliegue

Diagrama de despliegue

C. Hazlo funcionar (con videos)

  1. Prueba el ejemplo en http://srvmuestra.rf.gd/.

  2. Descarga el archivo /src/srvmuestra.zip y descompáctalo.

  3. Crea una cuenta de email pqra ti, por ejemplo, pepito@google.com. Si ya tienes un email, omite este paso.

  4. Crea una cuenta de GitHub usando el email anterior y selecciona el nombre de usuario unsando la parte inicial del correo electrónico, por ejemplo pepito. Si ya tienes una cuenta, omite este paso.

  5. Crea un repositorio nuevo. En el nombre del repositorio debes poner el nombre de tu sitio; por ejemplo devuelvejson

  6. Importa el proyecto de GitHub a Visual Studio Code

  7. Edita los archivos que desees.

  8. Prueba tu sitio localmente.

  9. Necesitas un hosting. En este ejemplo se muestra como usar el hosting. https://infinityfree.com/ Si no lo has usado, lo primero que tienes que hacer es entrar a registrar tu email con el botón Registrar. Si ya tienes tu email registrado, omite este paso.

  10. Crea una cuenta. Si ya tienes cuenta, entra a ella y crea un nuevo domino. En este ejemplo no se crean los archivos directamente en el hosting.

  11. Sube tus archivos al hosting usando ftp.

  12. Sube tus archivos a GitHub. En este ejemplo no hay archivo sw.js ni necesitas esperar 11 o más minutos.

D. Hazlo funcionar (texto)

  1. Prueba el ejemplo en http://srvmuestra.rf.gd/.

  2. Descarga el archivo /src/srvmuestra.zip y descompáctalo.

  3. Crea tu proyecto en GitHub:

    1. Crea una cuenta de email para tí, por ejemplo, pepito@google.com. Si ya tienes un email, omite este paso.

    2. Crea una cuenta de GitHub usando el email anterior y selecciona el nombre de usuario unsando la parte inicial del correo electrónico, por ejemplo pepito. Si ya tienes una cuenta, omite este paso.

    3. Crea un repositorio nuevo. En la página principal de GitHub cliquea 📘 New.

    4. En la página Create a new repository introduce los siguientes datos:

      • Proporciona el nombre de tu repositorio debajo de donde dice Repository name *.

      • Mantén la selección Public para que otros usuarios puedan ver tu proyecto.

      • Verifica la casilla Add a README file. En este archivo se muestra información sobre tu proyecto.

      • Cliquea License: None. y selecciona la licencia que consideres más adecuada para tu proyecto.

      • Cliquea Create repository.

  4. Importa el proyecto en GitHub:

    1. En la página principal de tu proyecto en GitHub, en la pestaña < > Code, cliquea < > Code y en la sección Branches y copia la dirección que está en HTTPS, debajo de Clone.

    2. En Visual Studio Code, usa el botón de la izquierda para Source Control.

      Imagen de Source Control
    3. Cliquea el botón Clone Repository.

    4. Pega la url que copiaste anteriormente hasta arriba, donde dice algo como Provide repository URL y presiona la teclea Intro.

    5. Selecciona la carpeta donde se guardará la carpeta del proyecto.

    6. Abre la carpeta del proyecto importado.

    7. Añade el contenido de la carpeta descompactada que contiene el código del ejemplo.

  5. Edita los archivos que desees.

  6. Haz clic derecho en index.html, selecciona PHP Server: serve project y se abre el navegador para que puedas probar localmente el ejemplo.

  7. Para depurar paso a paso haz lo siguiente:

    1. En el navegador, haz clic derecho en la página que deseas depurar y selecciona inspeccionar.

    2. Recarga la página, de preferencia haciendo clic derecho en el ícono de volver a cargar la página Ïmagen del ícono de recarga y seleccionando vaciar caché y volver a cargar de manera forzada (o algo parecido). Si no aparece un menú emergente, simplemente cliquea volver a cargar la página Ïmagen del ícono de recarga. Revisa que no aparezca ningún error ni en la pestañas Consola, ni en Red.

    3. Selecciona la pestaña Fuentes (o Sources si tu navegador está en Inglés).

    4. Selecciona el archivo donde vas a empezar a depurar.

    5. Haz clic en el número de la línea donde vas a empezar a depurar.

    6. En Visual Studio Code, abre el archivo de PHP donde vas a empezar a depurar.

    7. Haz clic en Run and Debug .

    8. Si no está configurada la depuración, haz clic en create a launch json file.

    9. Haz clic en la flechita RUN AND DEBUG, al lado de la cual debe decir Listen for Xdebug .

    10. Aparece un cuadro con los controles de depuración

    11. Selecciona otra vez el archivo de PHP y haz clic en el número de la línea donde vas a empezar a depurar.

    12. Regresa al navegador, recarga la página y empieza a usarla.

    13. Si se ejecuta alguna de las líneas de código seleccionadas, aparece resaltada en la pestaña de fuentes. Usa los controles de depuración para avanzar, como se muestra en este video.

  8. Sube el proyecto al hosting que elijas.

    1. Crea una nueva carpeta para crear un nuevo proyecto que estará conectado directamente al servidor web por ftp.

    2. Abre la nueva carpeta con Visual Studio Code.

    3. Tecle al mismo Mayúsculas+Control+P y selecciona SFTP: Config. Aparece un archivo de configuración de FTP. Llena los datos con la configuración de FTP de tu servidor, excepto la contraseña.

    4. Cliquea el botón de SFTP y luego haz clic en la URL de tu servidos. En la barra superior te pide la contraseña y ENTER.

    5. Pásate a la parte de archivos y coloca tus archivos.

    6. Cliquea con el botón derecho en la sección de archivos y selecciona Sync: Local -> Remote.

  9. Abre un navegador y prueba el proyecto en tu hosting.

  10. En el hosting InfinityFree, la primera vez que corres la página, puede marcar un mensaje de error, pero al recargar funciona correctamente. Puedes evitar este problema usando un dominio propio.

  11. Para subir el código a GitHub, en la sección de SOURCE CONTROL, en Message introduce un mensaje sobre los cambios que hiciste, por ejemplo index.html corregido, selecciona v y luego Commit & Push.

    Imagen de Commit & Push

E. Archivos

Haz clic en los triángulos para expandir las carpetas

F. index.html

1
<!DOCTYPE html>
2
<html lang="es">
3
4
<head>
5
 <meta charset="UTF-8">
6
 <meta name="viewport" content="width=device-width">
7
8
 <title>Muestra datos</title>
9
10
 <script type="module" src="js/lib/manejaErrores.js"></script>
11
12
</head>
13
14
<body>
15
16
 <h1>Muestra datos</h1>
17
18
 <!-- Al mostrar un dato, se usan los atributos "id", "name" o "data-name"
19
  para buscar la propiedad a desplegar en la página. -->
20
21
 <p>
22
  <label>
23
   Nombre
24
   <!-- Muestra la propiedad nombre del objeto descargado, usando la propiedad
25
    value del output. -->
26
   <output name="nombre"></output>
27
  </label>
28
 </p>
29
30
 <p>
31
  <label>
32
   Apellido
33
   <!-- Muestra la propiedad apellido del objeto descargado, usando la propiedad
34
    value del input. -->
35
   <input name="apellido" type="text">
36
  </label>
37
 </p>
38
39
 <p>
40
  <label>
41
   Género
42
   <!-- Muestra la propiedad genero del objeto descargado, usando la propiedad
43
    value del select. -->
44
   <select name="genero">
45
    <option value="">Sin selección</option>
46
    <option value="pop">Pop</option>
47
    <option value="reg">Reguetón</option>
48
   </select>
49
  </label>
50
 </p>
51
52
 <p>
53
  <label>
54
   Géneración
55
   <!-- Muestra la propiedad generacion del objeto descargado, usando la
56
    propiedad value del select. -->
57
   <select name="generacion">
58
    <option value="boom">Baby Boom</option>
59
    <option value="X">X</option>
60
    <option value="">Sin selección</option>
61
    <option value="Y">Millenoals</option>
62
    <option value="Z">Z</option>
63
    <option value="alfa">Alfa</option>
64
   </select>
65
  </label>
66
 </p>
67
68
 <p>
69
  <label>
70
   Edad
71
   <!-- Muestra la propiedad edad del objeto descargado, usando la propiedad
72
    valueAsNumber del input. -->
73
   <input data-name="edad" type="number">
74
  </label>
75
 </p>
76
77
 <p>
78
  <label>
79
   Número de la suerte
80
   <!-- Muestra la propiedad numero del objeto descargado, usando la propiedad
81
    value del output. -->
82
   <output id="numero"></output>
83
  </label>
84
 </p>
85
86
 <p>
87
  <label>
88
   Avance
89
   <!-- Muestra la propiedad avance del objeto descargado, usando la propiedad
90
    value del progress. -->
91
   <progress id="avance" max="100"></progress>
92
  </label>
93
 </p>
94
95
 <p>
96
  <label>
97
   Capacidad
98
   <!-- Muestra la propiedad capacidad del objeto descargado, usando la
99
    propiedad value del meter. -->
100
   <meter id="capacidad" min="50" max="80"></meter>
101
  </label>
102
 </p>
103
104
 <p>
105
  <label>
106
   Temperatura
107
   <!-- Muestra la propiedad temperatura del objeto descargado, usando la
108
    propiedad valueAsNumber del meter. -->
109
   <input id="temperatura" type="range" min="0" max="50">
110
  </label>
111
 </p>
112
113
 <p>
114
  <label>
115
   <!-- Muestra la propiedad aprobado del objeto descargado, usando la propiedad
116
    checked del input, porque el tipo del dato es boolean y el input tiene
117
    type="checkbox". -->
118
   <input id="aprobado" type="checkbox">
119
   Aprobado
120
  </label>
121
 </p>
122
123
 <p>
124
  <label>
125
   Gracioso
126
   <!-- Muestra la propiedad gracioso del objeto descargado, usando la propiedad
127
    value del output. -->
128
   <output id="gracioso"></output>
129
  </label>
130
 </p>
131
132
 <p>
133
  <label>
134
   Emplacado
135
   <!-- Muestra la propiedad emplacado del objeto descargado, usando la
136
    propiedad value del select. -->
137
   <select name="emplacado">
138
    <option value="">Sin selección</option>
139
    <option value="true">Si</option>
140
    <option value="false">No</option>
141
   </select>
142
  </label>
143
 </p>
144
145
 <label>Dirección</label>
146
 <!-- Muestra la propiedad direccion del objeto descargado, usando la propiedad
147
  textContent del pre. -->
148
 <pre id="direccion"></pre>
149
150
 <p>
151
  <label>
152
   Encabezado
153
   <!-- Muestra la propiedad encabezado del objeto descargado, usando la
154
    propiedad innerHTML del span. -->
155
   <span id="encabezado"></span>
156
  </label>
157
 </p>
158
159
 <p>
160
  <label>
161
   Nacimiento
162
   <!-- Muestra la propiedad nacimiento del objeto descargado, usando la
163
    propiedad value del input. Revisa la especificación de input para
164
    los distintos formatos de fecha que se pueden usar. -->
165
   <input id="nacimiento" type="date">
166
  </label>
167
 </p>
168
169
 <figure>
170
  <!-- Muestra la propiedad imagen1 del objeto descargado, usando la propiedad
171
   src del img. -->
172
  <img id="imagen1" alt="Imagen 2">
173
  <figcaption>Imagen 1</figcaption>
174
 </figure>
175
176
 <figure>
177
  <!-- Muestra la propiedad imagen2 del objeto descargado, usando la propiedad
178
   src del img. Como el valor es "", se oculta el img usando hidden = true. -->
179
  <img id="imagen2" alt="Imagen 2">
180
  <figcaption>Imagen 2</figcaption>
181
 </figure>
182
183
 <!-- Muestra la propiedad pasatiempos[] del objeto descargado; como es un
184
   array, usa los elementos con name="pasatiempos[]" y les pone la propiedad
185
   checked en true si su value está en el array; de lo contrario se las pone en
186
   false. Los corchetes([]), le indican a PHP que la propiedad es un array
187
   que puede llegar a tener 0, 1 o más elementos. -->
188
 <fieldset>
189
  <legend>Pasatiempos</legend>
190
  <label>
191
   <input type="checkbox" name="pasatiempos[]" value="fut">Futbol
192
  </label>
193
  <label>
194
   <input type="checkbox" name="pasatiempos[]" value="chess">Ajedrez
195
  </label>
196
  <label>
197
   <input type="checkbox" data-name="pasatiempos[]" value="basket">Basketbol
198
  </label>
199
 </fieldset>
200
201
 <!-- Muestra la propiedad madrugador del objeto descargado; como es un
202
   array, usa los elementos con name="madrugador" y les pone la propiedad
203
   checked en true si su value está en el array; de lo contrario se las pone en
204
   false. Como el name no tiene [], el array solo tiene 0 o un elemento. -->
205
 <fieldset>
206
  <legend>Madrugador</legend>
207
  <label><input type="radio" name="madrugador" value="si">Si</label>
208
  <label><input type="radio" name="madrugador" value="no">No</label>
209
 </fieldset>
210
211
 <p>
212
  <label>
213
   Patos
214
   <!-- Muestra la propiedad patos del objeto descargado; como es un
215
     array, usa los las opciones del select con name="patos[]" y les pone la
216
     propiedad selected en true si su value está en el array; de lo contrario se
217
     las pone en false.  Los corchetes([]), le indican a PHP que la propiedad es
218
     un array que puede llegar a tener 0, 1 o más elementos. -->
219
   <select name="patos[]" multiple size="3">
220
    <option value="hugo">Hugo</option>
221
    <option value="paco">Paco</option>
222
    <option value="luis">Luis</option>
223
   </select>
224
  </label>
225
 </p>
226
</body>
227
228
<script type="module">
229
230
 import { consume } from "./js/lib/consume.js"
231
 import { recibeJson } from "./js/lib/recibeJson.js"
232
 import { muestraObjeto } from "./js/lib/muestraObjeto.js"
233
234
 descargaDatos()
235
236
 async function descargaDatos() {
237
  const respuesta = await consume(recibeJson("php/datos.php"))
238
  const json = await respuesta.json()
239
  muestraObjeto(document, json)
240
 }
241
242
</script>
243
244
</html>

G. Carpeta « php »

Versión para imprimir.

A. php / datos.php

1
<?php
2
3
require_once __DIR__ . "/lib/manejaErrores.php";
4
require_once __DIR__ . "/lib/devuelveJson.php";
5
6
devuelveJson([
7
 "nombre" => ["value" => "pp"],
8
 "apellido" => ["value" => "tkt"],
9
 "genero" => ["value" => "pop"],
10
 "generacion" => ["value" => ""],
11
 "edad" => ["valueAsNumber" => 18],
12
 "numero" => ["value" => 5],
13
 "avance" => ["value" => 70],
14
 "capacidad" => ["value" => 60],
15
 "temperatura" => ["valueAsNumber" => 40],
16
 "aprobado" => ["checked" => true],
17
 "gracioso" => ["value" => false],
18
 "emplacado" => ["value" => false],
19
 "direccion" => ["textContent" => "Girasoles 23\ncolonia Rosales"],
20
 "encabezado" => ["innerHTML" => "<em>Hola, soy <strong>pp</strong>"],
21
 "nacimiento" => ["value" => "2000-07-04"],
22
 "imagen1" => [
23
  "src" => "https://gilpgawoas.github.io/img/icono/maskable_icon_x48.png"
24
 ],
25
 "imagen2" => ["src" => "", "hidden" => true],
26
 "pasatiempos[]" => ["fut", "basket"],
27
 "madrugador" => ["no"],
28
 "patos[]" => ["paco", "luis"],
29
]);
30

B. Carpeta « php / lib »

1. php / lib / devuelveJson.php

1
<?php
2
3
require_once __DIR__ . "/devuelveResultadoNoJson.php";
4
5
function devuelveJson($resultado)
6
{
7
 $json = json_encode($resultado);
8
 if ($json === false) {
9
  devuelveResultadoNoJson();
10
 } else {
11
  header("Content-Type: application/json; charset=utf-8");
12
  echo $json;
13
 }
14
 exit();
15
}
16

2. php / lib / devuelveResultadoNoJson.php

1
<?php
2
3
require_once __DIR__ . "/INTERNAL_SERVER_ERROR.php";
4
5
function devuelveResultadoNoJson()
6
{
7
 http_response_code(INTERNAL_SERVER_ERROR);
8
 header("Content-Type: application/problem+json; charset=utf-8");
9
10
 echo '{' .
11
  "status: " . INTERNAL_SERVER_ERROR .
12
  '"title": "El resultado no puede representarse como JSON."' .
13
  '"type": "/errors/resultadonojson.html"' .
14
  '}';
15
}
16

3. php / lib / INTERNAL_SERVER_ERROR.php

1
<?php
2
3
const INTERNAL_SERVER_ERROR = 500;

4. php / lib / manejaErrores.php

1
<?php
2
3
require_once __DIR__ . "/INTERNAL_SERVER_ERROR.php";
4
require_once __DIR__ . "/ProblemDetailsException.php";
5
6
7
8
// Hace que se lance una excepción automáticamente cuando se genere un error.
9
set_error_handler(function ($severity, $message, $file, $line) {
10
 throw new ErrorException($message, 0, $severity, $file, $line);
11
});
12
13
// Código cuando una excepción no es atrapada.
14
set_exception_handler(function (Throwable $excepcion) {
15
 if ($excepcion instanceof ProblemDetailsException) {
16
  devuelveProblemDetails($excepcion->problemDetails);
17
 } else {
18
  devuelveProblemDetails([
19
   "status" => INTERNAL_SERVER_ERROR,
20
   "title" => "Error interno del servidor",
21
   "detail" => $excepcion->getMessage(),
22
   "type" => "/errors/errorinterno.html",
23
  ]);
24
 }
25
 exit();
26
});
27
28
function devuelveProblemDetails(array $array)
29
{
30
 $json = json_encode($array);
31
 if ($json === false) {
32
  devuelveResultadoNoJson();
33
 } else {
34
  http_response_code(isset($array["status"]) ? $array["status"] : 500);
35
  header("Content-Type: application/problem+json; charset=utf-8");
36
  echo $json;
37
 }
38
}
39

5. php / lib / ProblemDetailsException.php

1
<?php
2
3
require_once __DIR__ . "/INTERNAL_SERVER_ERROR.php";
4
5
/**
6
 * Detalle de los errores devueltos por un servicio.
7
 */
8
class ProblemDetailsException extends Exception
9
{
10
11
 public array $problemDetails;
12
13
 public function __construct(
14
  array $problemDetails,
15
 ) {
16
  
17
  parent::__construct(
18
   isset($problemDetails["detail"])
19
    ? $problemDetails["detail"]
20
    : (isset($problemDetails["title"])
21
     ? $problemDetails["title"]
22
     : "Error"),
23
   $problemDetails["status"]
24
    ? $problemDetails["status"]
25
    : INTERNAL_SERVER_ERROR
26
  );
27
28
  $this->problemDetails = $problemDetails;
29
 }
30
}
31

H. Carpeta « js »

Versión para imprimir.

A. Carpeta « js / lib »

1. js / lib / consume.js

1
import { ProblemDetailsError } from "./ProblemDetailsError.js"
2
3
/**
4
 * Espera a que la promesa de un fetch termine. Si
5
 * hay error, lanza una excepción.
6
 * 
7
 * @param {Promise<Response> } servicio
8
 */
9
export async function consume(servicio) {
10
 const respuesta = await servicio
11
 if (respuesta.ok) {
12
  return respuesta
13
 } else {
14
  const contentType = respuesta.headers.get("Content-Type")
15
  if (
16
   contentType !== null && contentType.startsWith("application/problem+json")
17
  )
18
   throw new ProblemDetailsError(await respuesta.json())
19
  else
20
   throw new Error(respuesta.statusText)
21
 }
22
}

2. js / lib / manejaErrores.js

1
import { muestraError } from "./muestraError.js"
2
3
/**
4
 * Intercepta Response.prototype.json para capturar errores de parseo
5
 * y asegurar que se reporten correctamente en navegadores Chromium.
6
 */
7
{
8
 const originalJson = Response.prototype.json
9
10
 Response.prototype.json = function () {
11
  // Llamamos al método original usando el contexto (this) de la respuesta
12
  return originalJson.apply(this, arguments)
13
   .catch((/** @type {any} */ error) => {
14
    // Corrige un error de Chrome que evita el manejo correcto de errores.
15
    throw new Error(error)
16
   })
17
 }
18
}
19
20
window.onerror = function (
21
  /** @type {string} */ _message,
22
  /** @type {string} */ _url,
23
  /** @type {number} */ _line,
24
  /** @type {number} */ _column,
25
  /** @type {Error} */ errorObject
26
) {
27
 muestraError(errorObject)
28
 return true
29
}
30
31
window.addEventListener('unhandledrejection', event => {
32
 muestraError(event.reason)
33
 event.preventDefault()
34
})
35

3. js / lib / muestraError.js

1
import { ProblemDetailsError } from "./ProblemDetailsError.js"
2
3
/**
4
 * Muestra los datos de una Error en la consola y en un cuadro de alerta.
5
 * @param { ProblemDetailsError | Error | null } error descripción del error.
6
 */
7
export function muestraError(error) {
8
9
 if (error === null) {
10
11
  console.error("Error")
12
  alert("Error")
13
14
 } else if (error instanceof ProblemDetailsError) {
15
16
  const problemDetails = error.problemDetails
17
18
  let mensaje =
19
   typeof problemDetails["title"] === "string" ? problemDetails["title"] : ""
20
  if (typeof problemDetails["detail"] === "string") {
21
   if (mensaje !== "") {
22
    mensaje += "\n\n"
23
   }
24
   mensaje += problemDetails["detail"]
25
  }
26
  if (mensaje === "") {
27
   mensaje = "Error"
28
  }
29
  console.error(error, problemDetails)
30
  alert(mensaje)
31
32
 } else {
33
34
  console.error(error)
35
  alert(error.message)
36
37
 }
38
39
}

4. js / lib / muestraObjeto.js

1
/**
2
 * @param {Document | HTMLElement | ShadowRoot} raizHtml
3
 * @param { any } objeto
4
 */
5
export function muestraObjeto(raizHtml, objeto) {
6
 for (const [nombre, definiciones] of Object.entries(objeto)) {
7
  if (Array.isArray(definiciones)) {
8
   muestraArray(raizHtml, nombre, definiciones)
9
  } else if (definiciones !== undefined && definiciones !== null) {
10
   muestraElemento(raizHtml, nombre, definiciones)
11
  }
12
 }
13
}
14
15
/**
16
 * @param { string } nombre
17
 */
18
export function selectorDeNombre(nombre) {
19
 return `[id="${nombre}"],[name="${nombre}"],[data-name="${nombre}"]`
20
}
21
22
/**
23
 * @param { Document | HTMLElement | ShadowRoot } raizHtml
24
 * @param { string } propiedad
25
 * @param {any[]} valores
26
 */
27
function muestraArray(raizHtml, propiedad, valores) {
28
 const conjunto = new Set(valores)
29
 const elementos = raizHtml.querySelectorAll(selectorDeNombre(propiedad))
30
 if (elementos.length === 1 && elementos[0] instanceof HTMLSelectElement) {
31
  muestraOptions(elementos[0], conjunto)
32
 } else {
33
  muestraInputs(elementos, conjunto)
34
 }
35
36
}
37
38
/**
39
 * @param {HTMLSelectElement} select
40
 * @param {Set<any>} conjunto
41
 */
42
function muestraOptions(select, conjunto) {
43
 for (let i = 0, options = select.options, len = options.length; i < len; i++) {
44
  const option = options[i]
45
  option.selected = conjunto.has(option.value)
46
 }
47
}
48
49
/**
50
 * @param {NodeListOf<Element>} elementos
51
 * @param {Set<any>} conjunto
52
 */
53
function muestraInputs(elementos, conjunto) {
54
 for (let i = 0, len = elementos.length; i < len; i++) {
55
  const elemento = elementos[i]
56
  if (elemento instanceof HTMLInputElement) {
57
   elemento.checked = conjunto.has(elemento.value)
58
  }
59
 }
60
}
61
62
const data_ = "data-"
63
const data_Length = data_.length
64
65
/**
66
 * @param {Document | HTMLElement | ShadowRoot} raizHtml
67
 * @param {string} nombre
68
 * @param {{ [s: string]: any; } } definiciones
69
 */
70
function muestraElemento(raizHtml, nombre, definiciones) {
71
 const elemento = raizHtml.querySelector(selectorDeNombre(nombre))
72
 if (elemento !== null) {
73
  for (const [propiedad, valor] of Object.entries(definiciones)) {
74
   if (propiedad in elemento) {
75
    elemento[propiedad] = valor
76
   } else if (
77
    propiedad.length > data_Length
78
    && propiedad.startsWith(data_)
79
    && elemento instanceof HTMLElement
80
   ) {
81
    elemento.dataset[propiedad.substring(data_Length)] = valor
82
   }
83
  }
84
 }
85
}

5. js / lib / ProblemDetailsError.js

1
export class ProblemDetailsError extends Error {
2
3
 /**
4
  * Detalle de los errores devueltos por un servicio.
5
  * Crea una instancia de ProblemDetailsError.
6
  * @param {object} problemDetails Objeto con la descripcipon del error.
7
  */
8
 constructor(problemDetails) {
9
10
  super(typeof problemDetails["detail"] === "string"
11
   ? problemDetails["detail"]
12
   : (typeof problemDetails["title"] === "string"
13
    ? problemDetails["title"]
14
    : "Error"))
15
16
  this.problemDetails = problemDetails
17
18
 }
19
20
}

6. js / lib / recibeJson.js

1
2
/**
3
 * @param {string} url
4
 * @param { "GET" | "POST"| "PUT" | "PATCH" | "DELETE" | "TRACE" | "OPTIONS"
5
 *  | "CONNECT" | "HEAD" } metodoHttp
6
 */
7
export async function recibeJson(url, metodoHttp = "GET") {
8
 return fetch(
9
  url,
10
  {
11
   method: metodoHttp,
12
   headers: { "Accept": "application/json, application/problem+json" }
13
  }
14
 )
15
}

I. Carpeta « errors »

Versión para imprimir.

A. errors / errorinterno.html

1
<!DOCTYPE html>
2
<html lang="es">
3
4
<head>
5
6
 <meta charset="UTF-8">
7
 <meta name="viewport" content="width=device-width">
8
9
 <title>Error interno del servidor</title>
10
11
</head>
12
13
<body>
14
15
 <h1>Error interno del servidor</h1>
16
17
 <p>Se presentó de forma inesperada un error interno del servidor.</p>
18
19
</body>
20
21
</html>

B. errors / resultadonojson.html

1
<!DOCTYPE html>
2
<html lang="es">
3
4
<head>
5
6
 <meta charset="UTF-8">
7
 <meta name="viewport" content="width=device-width">
8
9
 <title>El resultado no puede representarse como JSON</title>
10
11
</head>
12
13
<body>
14
15
 <h1>El resultado no puede representarse como JSON</h1>
16
17
 <p>
18
  Debido a un error interno del servidor, el resultado generado, no se puede
19
  recuperar.
20
 </p>
21
22
</body>
23
24
</html>

J. jsconfig.json

1
{
2
 "compilerOptions": {
3
  "checkJs": true,
4
  "strictNullChecks": true,
5
  "target": "ES6",
6
  "module": "Node16",
7
  "moduleResolution": "Node16",
8
  "lib": [
9
   "ES2017",
10
   "WebWorker",
11
   "DOM"
12
  ]
18
}

K. Resumen