Hoy comparto este simple pero efectivo y elegante soporte para mando de PS5.
Algunas imágenes/diseños de concepto:
Se sostiene por sí solo sin tornillos ni pegamento. Se cuelga directamente del lateral de la PS5 y es expandible por diseño (probablemente quepan hasta 4 mandos en cada lado).
Estaba tentado de intercambiar la carcasa de mi Raspberry Pi 4 a la Pi 5, pero mírala…
16×2 pantalla LCD de un Arduino Starter Kit
Programa en C++ con la librería WiringPi
Carcasa casera
Estaba realmente orgulloso de ella en su momento y fue un buen aprendizaje allá por 2020, pero seguro que lo podía hacer mejor. Seguro que me podía currar un pedazo panel de estado!!
Contenido
Componentes
Raspberry Pi (probablemente sirva en cualquiera de ellas)
Si tienes problemas con tu versión de python para encontrar Circuitpython, incluye --break-system-packages al final. (No romperá nada hoy, pero no te acostumbres a ello...)
Usa un sitio como pinout.xyz para encontrar una configuración adecuada para tu cableado.
Estás listo para hacer algunos tests antes de mover tu diseño a la PCB.
Configuración de script
Puedes probar el script demo de Adafruit. Solo asegúrate de escoger la pantalla adecuada y actualiza los cambios en la asignación de pines (usa los números/nombres IO en lugar de los números físicos del pin):
# Configuration for CS and DC pins (adjust to your wiring):
cs_pin = digitalio.DigitalInOut(board.CE0)
dc_pin = digitalio.DigitalInOut(board.D24)
reset_pin = digitalio.DigitalInOut(board.D25)
# Setup SPI bus using hardware SPI:
spi = board.SPI()
disp = ssd1351.SSD1351(spi, rotation=270, # 1.5" SSD1351
cs=cs_pin,
dc=dc_pin,
rst=reset_pin,
baudrate=BAUDRATE
)
Ensamblaje
Aquí están los archivos STL en 3D para este diseño de carcasa:
La verdad es que imprimir con resina es otro nivel. A pesar de las complicaciones de la limpieza, el resultado es realmente increíble, y hoy te enseño mi primera impresión con la Anycubic Photon Mono X2: un Batman para colgar de tu salón con su pistola gancho.
Contenido
Modelos
La impresión que ves consta de dos modelos gratuitos. Por ello, mi enorme agradecimiento a los autores y mi referencia a su trabajo:
Lo único que he hecho yo es añadirle un esqueleto al modelo y darle la postura deseada, uniéndole luego la pistola a la mano. Asique aquí tienes mi modelo para que lo puedas imprimir tal y como se muestra en las imágenes.
Para completar el modelo, deberás crearle una capa y un gancho o pinza con el cual colgarlo.
Capa
Para la capa yo he recortado un trozo de tela de una prenda de deporte antigua. Solemos tener prendas de este estilo en color negro y suelen tener también una buena textura y brillo que dan el pego.
Empieza cortando un rectángulo y luego dale forma.
En la parte superior, enrolla un alambre o un clip que te permita ajustar la capa alrededor del cuello de la figura.
Gancho
Simplemente busca un enganche que te venga bien para colgarlo. Yo he usado una pinza a la cual he atado un fino hilo negro, lo que me permite enrollarlo sobre sí para ajustar la altura. De esta forma lo puedo colgar de algún libro de una estantería.
Y así tienes tu genial Batman colgando de tu salón. Espero te guste! Otras ideas o comentarios? Al twitter!
PAL es una versión simplificada de mi asistente del hogar en python, que se ejecuta en una placa UNIHIKER de DFRobot, y que lanzo en código abierto gratuito.
Esta es solo una pequeña demostración de la simplicidad de la ejecución de comandos con reconocimiento de voz en python, y con suerte servirá de guía para tu propio asistente.
Version actual: v0.2.0(actualizado septiembre 2024)
Contenido
Características
La versión actual incluye lo siguiente:
Reconocimiento de voz: usando la librería de código abierto SpeechRecognition, se obtiene una lista con las cadenas de texto reconocidas.
Pronóstico del tiempo: usando la API de datos de World Meteorological Organization, devuelve el tiempo de hoy y el pronóstico de los próximos 3 días. Incluye los iconos de la WMO.
Temperatura local: reads local BMP-280 temperature sensor to provide a room temperature indicator.
Comandos HTTP para IoT: flujo de trabajo básico para controlar dispositivos inteligentes de IoT mediante comandos HTTP. Actualmente ENCIENDE y APAGA un interruptor inteligente Shelly2.5.
Modo ahorro de energía: controla el brillo para reducir el consumo de energía.
Gestor de conexión: periódicamente comprueba el wifi y llama a internet para restaurar la conexión cuando se pierda.
Muestra de voces de PAL: voz clonada de PAL de "Los Mitchells contra las Máquinas" usando el modelo de IA de voz CoquiAI-TTS v2.
Botones UNIHIKER: botón A despliega un simple menú (está pensado para desarrollar un menú más complejo en el futuro).
Controles táctiles: restaura el brillo (centro), cambia de programa (izquierda) y cierra el programa (derecha) pulsando en distintas zonas de la pantalla.
En este momento, PAL v0.2.0 solo incluye compatibilidad con Shelly2.5 para fines demostrativos.
Usa las variables lampBrand, lampChannel y lampIP para ajustar tu configuración de Shelly2.5.
Esto es solo un ejemplo para mostrar cómo se pueden configurar distintos dispositivos. Estas variables se deberían usar para cambiar las peculiaridades de los comandos HTTP que se envían a cada dispositivo IoT.
Se añadirán más dispositivos en futuras actualizaciones, como Shelly1, ShellyDimmer, Sonoff D1, etc.
Modo ahorro de energía
El ahorro de energía disminuye el brillo del dispositivo para reducir el consumo de energía de la UNIHIKER. Esto se hace usando el comando del sistema "brightness".
Cambia la variable "ps_mode" para habilitar ("1") o deshabilitar ("0") el modo ahorro de energía.
Temperatura de habitación
Cambia la variable "room_temp" para habilitar ("1") o deshabilitar ("0") el módulo de lectura de sensor de temperatura. Esto requiere un sensor BMP-280 instalado mediante el conector I2C.
Comprueba este otro post para los detalles de la instalación del sensor:
Se habilita la personalización de la imagen de fondo que representa los ojos de PAL.
Usa las variables "eyesA" y "eyesB" para especificar uno de los siguientes valores y cambia la expresión del PAL:
"happy"
"angry"
"surprised"
"sad"
"happy""angry""surprised""sad"
"eyesA" se usa como fondo por defecto y "eyesB" se usa como transición cuando el reconocimiento de voz se activa y PAL habla.
El valor por defecto de "eyesA" es "surprised" y cambiará a "happy" cuando se reconoce un comando.
Comandos personalizados
Añadir tus propios comandos a PAL es simple usando la función "comandos".
Cada audio reconocido por SpeechRecognition se envía como una cadena a la función "comandos", que filtra y lanza uno u otro comando coincidente con la orden.
Solo define todas las posibles cadenas que pueden ser reconozidas para lanzar tu comando (a veces SpeechRecognition devuelve transcripciones erróneas o imprecisas).
Finalmente, define el comando que se lanza cuando la cadena coincide.
def comandos(msg):
# LAMP ON
if any(keyword in msg for keyword in ["turn on the lamp", "turn the lights on","turn the light on", "turn on the light", "turn on the lights"]):
turnLAMP("on")
os.system("aplay '/root/upload/PAL/mp3/Turn_ON_lights.wav'")
Palabra clave para la activación
Puedes personalizar las palabras clave o cadenas que activan las funciones de comandos. Si cualquiera de las palabras clave en la lista es reconocida, toda la frase se envía a la función "comandos" para buscar un comando coincidente y ejecutarlo.
En caso de PAL v0.2, estas son las palabras clave que lo activan (el 90% de las veces es Paypal...):
Puedes cambiar esto a cualquier otra frase o nombre para activar a PAL cuando la llames de esa manera.
Voz de PAL
Usa la muestra de audio "PAL_full" abajo (también en el repositorio de github en /mp3) como audio de referencia para la clonación de voz de CoquiAI-TTS v2 y genera tus propias voces:
A continuación hay algunos ejemplos de consultas y respuestas de PAL:
"Hey PAL, turn on the lights!""Hey PAL, turn the lights off""Hey PAL, what's the weather like?"
Futuros lanzamientos (To-Do list)
Iré desarrollando estas características en mi asistente personal, e iré actualizando la versión de código abierto de vez en cuando. Ponte en contacto via github si tienes especial interés en alguna de ellas:
Menú avanzado: añadir configuración y comandos manuales.
Dispositivos IoT: incluir todos los comandos HTTP de Shelly y Sonoff.
Consultar la hora: requiere clonar la voz de todas las combinaciones de números...
Consulta de Wikipedia/internet: requiere generar voz en tiempo real...
Mejoradas animaciones / temas.
Comentarios, problemas o mejoras, me encantará recibirlos en github o Twitter!
Sigo experimentando con la placa UNIHIKER de DFRobot y es increíblemente rápido hacer que las cosas funcionen en ella. Hoy traigo cómo configurar en 2 minutos la visualización en tiempo real de un sensor de temperatura, usando un módulo BMP-280 y absolutamente nada de programación.
Contenido
Prerrequisitos
Aquí está el truco, y es que esperaba que tuvieras un par de cosas configuradas antes de activar el cronómetro:
Descarga e instala Mind+, el IDE para UNIHIKER de DFRobot. En Linux se trata de fichero .deb que se toma su tiempo para instalar: https://mindplus.cc/download-en.html
Suelda un modulo de presión y temperatura BMP-280 y conéctalo a un cable I2C que vienen con la placa. Tendrás que doblar ligeramente algunos pines para que encajen en el conector que parece un nano JST de 1mm.
En Mind+, ve al Blocks editor y abre el menú Extensions.
En la pestaña pinpong selecciona el propio módulo pinpong (que habilita la interacción con los pines de la UNIHIKER) y el módulo de extensión BMP-280, que añade la interacción con estos módulos.
Vuelve al Blocks editor y comienza a construir tu bloque de código. Simplemente navega por el menú lateral izquierdo y arrastra todo lo que necesitas debajo del bloque Python program start:
pinpong - initialize board.
bmp280 - initialize I2C address 0x76 para inicializar el módulo.
control - bloque forever (introduce un bucle while True).
unihiker - añade objetos a la pantalla. Primero añado un objeto filled rectangle para limpiar texto anterior, luego añado el objeto text. Especifica coordenadas X,Y donde se mostrará cada objeto, al igual que su color.
bmp280 - objeto read temperature. Arrástralo dentro del campo de texto del objeto text.
python - (opcional) añade un objeto print para mostrar los datos en terminal. Yo he incluido todas las otras propiedades que devuelve el sensor.
control - añade un objeto wait para detener el programa durante 1 segundo antes del siguiente bucle.
Debería quedar todo así (haz click para ampliar)
Ejecución
Y ya está todo el programa montado, sin programar! Pulsa RUN en la parte superior de Mind+ y observa cómo se carga y se muestra por la pantalla de la UNIHIKER. Toca el sensor ligeramente con el dedo para ver cómo varían los valores con el incremento de temperatura.
El pequeño y eficiente factor de forma de la UNIHIKER hace que sea muy fácil diseñar una carcasa para ella.
Para my asistente inteligente buscaba un estilo tipo androide, y el logo de DFRobot es perfecto para la UNIHIKER, haciendo un guiño a sus desarrolladores.
Contenido
Repositorio Github
He publicado un repositorio donde abriré las fuentes (open-source) de todos los archivos de los modelos, pudiendo cualquiera contribuir con los suyos, así que anímate a crear una solicitud de cambios (pull request) y comparte tus diseños!
Hace unos meses traía el repositorio TorToiSe-TTS que facilitaba muchísmo la generación de voz a partir de texto, aunque solo se disponía de modelos en inglés.
Pero el mundo de la IA avanza tan rápido, que hoy os traigo una evolución que desbanca por completo al anterior post, con generación de voces complejas en cuestión de segundos y multilenguaje: Coqui-AI TTS.
Configura el resto de opciones (marca las casillas: Cleanup Reference Voice, Do not use language auto-detect, Agree)
Solicita tu clonación al servidor (Send)
Instalación
En cuanto a la instalación local:
Necesitarás python > 3.9, < 3.12.
RAM: no necesita tanta como para generación de imágenes. 4GB deberían bastar.
Crea una carpeta para el proyecto, por ejemplo "text-2-speech". Usando un terminal de Linux: mkdir text-2-speech
Es conveniente crear un entorno de python específico para evitar incompatibilidad de paquetes, para lo que debes instalar el paquete python3-venv. Yo crearé el entorno TTSenv: cd text-2-speech python3 -m venv TTSenv
Activa el entorno en el terminal: source TTSenv/bin/activate
Si solo necesitas generación de voz (sin clonación ni entrenamiento), instala directamente TTS en python: pip install TTS
Sin embargo, yo instalaré el repositorio completo de Coqui-AI TTS: git clone https://github.com/coqui-ai/TTS cd TTS pip install -e .[all]
Comprobación de modelos de lenguaje y voces
Lo primero que puedes hacer es comprobar los modelos disponibles para transformar tu texto a voz en distintos idiomas.
En el terminal escribe lo siguiente:
tts --list_models
No API token found for 🐸Coqui Studio voices - https://coqui.ai
Visit 🔗https://app.coqui.ai/account to get one.
Set it as an environment variable `export COQUI_STUDIO_TOKEN=`
Asegúrate de que el directorio de salida existe, y de que eliges el nombre dle modelo correto. La primera vez se descargarán los archivos necesarios y deberás aceptar la licencia de Coqui-AI. Después la generación se realiza en cuestión de segundos:
Clonación de voz
Por último, lo realmente impresionante de este modelo es la clonación de voz a partir de unos pocos segundos de grabación de audio.
Igual que en el post anterior, yo he tomado unos 30 segundos de la voz de Ultrón de escenas de la película Los Vengadores: la era de Ultrón.
Muestra en español:
Muestra en inglés:
Ahora, preparamos un script de python donde parametrizaremos todo lo necesario, que a grandes rasgos hará lo siguiente:
Importar torch y TTS import torch from TTS.api import TTS
Definir dispositivo de memoria (cuda o cpu). Con usar cpu sobra. Cuda puede petar. device="cpu"
Definir el texto a generar. txt="Voz generada a partir de texto"
Definir la muestra de audio de referencia (archivos .wav de unos 30 segundos) sample="/voice-folder/voice.wav"
Llamada al modelo TTS tts1=TTS("model_name").to(device)
Creación del fichero tts1.tts_to_file(txt, speaker_wav=sample, language="es", file_path="output-folder/output-file.wav")
Yo lo he llamado TRW-clone.py y queda así:
import torch
from TTS.api import TTS
# Get device ('cuda' or 'cpu')
device="cpu"
#Define text
txt="Bienvenido a este nuevo artículo del blog. Disfruta de tu visita."
#txt="Welcome to this new block post... Enjoy your visit!"
#Define audio sample
sample="../my-voices/ultron-es/mix.wav"
#sample="../my-voices/ultron-en/mix.wav"
#Run cloning
tts1 = TTS("tts_models/multilingual/multi-dataset/xtts_v2").to(device)
tts1.tts_to_file(txt, speaker_wav=sample, language="es", file_path="../output/ultron-es.wav")
Ejecútalo desde la carpeta TTS donde se encuentra el repositorio completo:
cd TTS python3 TRW-clone.py
Resultados
Aquí te dejo los resultados obtenidos en mis primeras pruebas.
Español:
Inglés:
Con un par de iteraciones podrás obtener resultados realmente espectaculares.
Cualquier duda o comentario aún la puedes dejar en Twitter/X
Hoy os traigo mi segunda oportunidad que le voy a dar a Shelly. Y es que mi primer Shelly Dimmer petó por exceso de temperatura, al tenerlo dentro de una caja de conexiones, pero el Shelly 2.5 que controla dos luces está aguantando también insertado en la pared.
Quizás la diferencia son los 5ºC más que soportan, así que hoy voy a montar un enchufe de pared con interruptor/programador wifi añadiéndole un diminuto Shelly 1, y esperemos que aguante.
Además de su pequeño tamaño, los Shelly son muy fáciles de configurar, así que también veremos cómo controlarlos localmente mediante su API HTTP.
Contenido
Requisitos
Mi objetivo es habilitar un enchufe de pared que pueda controlar y programar mediante WIFI, en mi caso, para controlar el calentador de agua eléctrico. Veamos qué voy a usar:
Shelly 1.
Cable eléctrico de dos hilos (fase y neutro).
Clavija de enchufe macho.
Clavija de enchufe hembra.
Material para ensamblar una carcasa (impresora 3D, madera, etc).
Shelly 1Cable eléctricoClavija de enchufe machoClavija de enchufe hembra (anverso)Clavija de enchufe hembra (reverso)Soporte para ensamblaje impreso en 3D
Conexión eléctrica
Veamos el manual de Shelly a ver cómo nos indica que realicemos las conexiones:
La idea de este esquema estándar es que conectes el Shelly 1 a una luz y a su interruptor, significando cada símbolo lo siguiente:
L: línea o fase
N: neutro
SW: switch (interruptor)
I: input o entrada
O: output o salida
Como quiero habilitar una clavija de enchufe, el esquema variará ligeramente, ya que no usaré el interruptor y puedo conectar la línea directamente a la entrada. Por otro lado, y por motivos que desconozco, no tengo cableado en esta caja de conexiones, así que traigo la línea eléctrica mediante un cable que va a otro enchufe... En fin, queda todo así:
TIP! Juraría que me he rayado confundiendo los colores, pero en este caso el circuito propio es cerrado así que no importa demasiado si no eres maniático, ya que el circuito está conectado correctamente.
Ensamblaje
Como verás, me he hecho un pequeño soporte en 3D para guiar las conexiones y también una tapa para cubrir el hueco del enchufe. Además, simular las piezas en 3D, con medidas reales , te ayudará a distribuir bien el espacio y asegurarte de que tu solución se acopla correctamente:
Te dejo los modelos 3D en .stl para mandar a imprimir en software slicer.
Al final, queda todo así montado. No es el ajuste perfecto, pero me vale para cumplir su función.
Conexión a internet
Ahora veamos como traer a la vida al Shelly 1 y manejarlo localmente.
Al contrario que los Sonoff, Shelly nos lo pone mucho más fácil y solo hay que seguir su manual.
Enchufa el Shelly 1 a la corriente mediante el cable con clavija macho.
Se activará un AP (Access Point) o red Wi-Fi con una SSID del estilo "shelly1-01A3B4". Conéctate a esta red Wi-Fi mediante un smartphone o PC.
Cuando te conectes, usa un navegador web para acceder a la IP 192.168.33.1 y accederás a la interfaz web de configuración del dispositivo.
Una vez dentro, deberás configurar el dispositivo (en el menú Internet & Security) para que se conecte automáticamente a tu red Wi-Fi, así como también es recomendable que bloquees el acceso al dispositivo con usuario y contraseña, por lo que pudiera ocurrir...
Con esto ya podemos comunicarnos localmente con nuestro Shelly 1.
Uso de Shelly HTTP API
Para usar los comandos de la API HTTP deberás conocer la IP del dispositivo en la red local.
Averiguar IP mediante router
Puedes acceder al mapa de red de tu router, normalmente mediante la dirección http://192.168.1.1
Tanto la dirección como la contraseña de acceso deberían aparecer en una pegatina en tu router. Luego verás tu dispositivo con algún nombre tipo shelly1-XXXXXXXXXXXX:
Averiguar IP mediante nmap
Mediante un terminal, puedes usar el programa nmap para escanear tu red local.
Descárgalo si no lo tienes: sudo apt-get update sudo apt-get install nmap
Escanea tu red (con sudo aportará también la dirección MAC, útil por que la IP podría llegar a cambiar al reiniciar tu router) sudo nmap -v -sn 192.168.1.0/24
Enviar solicitudes HTTP al dispositivo
La API HTTP de Shelly viene documentada en su web dedicada:
Para poder comunicarte con el dispositivo necesitas enviar solicitudes HTTP mediante algún programa tipo Postman o mediante curl o wget en un terminal.
La solicitud se hará a la IP del dispositivo con:
$ curl -X GET http://192.168.1.XX/command
Si has definido usuario y contraseña, deberás ingresarlo también en la URL de esta manera o devolverá una respuesta "401 unauthorized":
El valor de 0 en la URL corresponde con el número del relay o interruptor interno del Shelly. En este caso solo hay uno, pero en caso del Shelly 2.5 tienes dos relays, por lo que puedes manejarlos individualmente cambiando este número.
En este caso, en la URL se definen los siguientes parámetros de las reglas:
HHMM: hora y minuto que activan la regla
0123456: los días de la semana en los que se produce la regla
on/off: estado que se produce al activarse la regla
De esta forma, para programar el encendido y apagado del dispositivo (excepto fines de semana), tendrías una solicitud de este estilo:
curl -X GET http://192.168.1.XX/settings/relay/0?schedule_rules=2300-01234-on,0700-01234-off
Obviamente, también podrás configurar las reglas de programación desde la interfaz web, o comprobar que los comandos han funcionado:
Y con eso estaría todo. Lánzate a llenar tu casa de discretos Shellys totalmente personalizables. Cualquier duda o comentario, al Twitter 🐦, gracias! (aunque visto lo de X, no sé yo lo que me durará...)
Todo empezó como un ejercicio de curiosidad, pensando si el ataque al puente de Kerch en Crimea se habría visto desde el espacio.
Para mi sorpresa, este será uno de los menos visibles de los ataques ocurridos durante la guerra de Ucrania.
Con cada noticia, comprobaba las imágenes satélite (accesibles públicamente) de la Agencia Espacial Europea y encontré evidencias de muchas de las informaciones. Las imágenes satélite son puramente objetivas. No hay manipulación o un discurso detrás.
Esto es totalmente opuesto a tomar un bando. Se trata de mostrar la realidad y la escala de la guerra. La auténtica y única verdad es que la violencia y la guerra tienen que ser condenadas en todas sus formas y prevenidas con cada mínimo esfuerzo que uno pueda hacer.
#WarTraces empieza en Ucrania, porque está cerca de los europeos, 24/7 en TV, y afectando a muchas y poderosas economías. Pero hay muchos otros conflictos activos en el mundo. Muchos otros que parecen olvidados. Muchos otros que no parecen tener importancia. Muchos otros que deben cesar igual que este.
Sígueme en 🐦 Twitter para saber cuando se actualiza este post!
Los primeros objetivos rusos fueron aeródromos que neutralizarían el combate aéreo. El Aeropuerto Internacional Antonov supuso una dura batalla desde el día 1 El poblado de Verkhnotoretske fue un frente duro durante el primer mes, a escasos 20km de Donetsk Mariupol fue fuertemente bombardeada al inicio del conflicto. Observa la diferencia de la señal infrarroja habitual y la causada por el incendio de las bombas.
El edificio de gobierno de Mikolaiv fue parcialmente destruido y 38 personas murieron el 29 de marzo.
Al sur de Zaporizhzhia se encuentra la central nuclear más grande de Europa. Fue rápidamente atacada y ocupada en marzo. Ataques continuados, como este de agosto, podrían causar una catástrofe nuclear.
La reclamada Crimea está conectada a la Rusia continental por el puente de Kerch. Un camión bomba causó su destrucción parcial en octubre de 2022.
El ataque al puente de Kerch llevó a un cambio de estrategia y la infraestructura crítica para la electricidad y el agua está siendo los nuevos objetivos desde entonces, a la vista de la inminente llegada del invierno.
Suele estar nublado en Ucrania, pero estas nubes no son muy normales. Las estelas que dejan las aeronaves también se ven desde el aire y nos dan información sobre sus rutas y su altitud.
Las calles residenciales de Bakhmut, ciudad clave para la ofensiva en el Donbás, están siendo objeto de duros combates. La mina de sal más grande de Europa, en Soledar, consta de 200km de túneles de 30 metros de alto, útiles para resguardar tropas y equipamiento bélico durante el invierno, a escasos 15km de Bakhmut.
Un mes desde que observé la battal por Bakhmut en un barrio residencial al este.
Infrared activity in Bakhmut doesn't cease being the hardest front of the war so far Desde el 10 de octubre de 2023, los brutales intentos de capturar Adviidka siguen causando muertos.
El 6 de noviembre de 2022, ataques sobre campos de refugiados acabaron en 9 muertes (3 de ellas niños). La gente ha sido desplazada de sus hogares después de destruir los principales asentamientos y ciudades. Puedes ver que no existían estos campos en 2015.
El final de 2017 resultó en intensos bombardeos de ciudades presuntamente ocupadas en el norte y este de Siria.
Hace un año que surgió endleZZ en una sala de espera sin cobertura, como una forma de desarrollar un juego offline que no requiriese nada especial: solo un navegador y un fichero .html en tu móvil.
La idea es genial, me ha servido muchas veces para matar el tiempo, pero se merecía una mínima actualización visual y funcional.
Así que aquí está la versión 0.2: actualización de aniversario! 🎉
Contenido
Versión web
Accede a la versión web oficial que hospedo en este servidor: