Ir al contenido principal

Creando aplicaciones GUI con Rust (y Druid framework)

Druid es un framework para crear aplicaciones gráficas simples. Se podía decir que es un Java Swing para el lenguaje Rust.

La documentación oficial dice:

"El framework está orientado a datos. Comparte muchas ideas (y se inspira directamente) en frameworks de interfaz de usuario declarativos contemporáneos como Flutter, Jetpack Compose y SwiftUI, a la vez que intenta ser conceptualmente simple y, en gran medida, no mágico".

Druid permite crear aplicaciones gráficas interactivas sencillas que se pueden implementar en Windows, macOS, Linux, OpenBSD, FreeBSD y la web.

Druid se basa en druid-shell, que implementa todo el código de nivel inferior específico de la plataforma, proporcionando una abstracción común para funciones como eventos de teclado y ratón, la creación de ventanas y el lanzamiento de aplicaciones. Por debajo de druid-shell se encuentra piet, una biblioteca de gráficos 2D multiplataforma que proporciona una API de dibujo sencilla y familiar que se puede implementar en diversas plataformas.

Druid es un framework declarativo basado en datos. Describes el modelo de tu aplicación en términos del rasgo "Datos" y luego construyes un árbol de widgets que pueden mostrar y modificar tus datos.

Tus widgets gestionan eventos, como el movimiento del ratón, y pueden modificar los datos; estos cambios se envían a los widgets correspondientes, que pueden actualizar su estado y redibujarlos.

A medida que su aplicación crece, puede usar Lenses para exponer solo ciertos subconjuntos de su modelo de datos a ciertos subconjuntos de su árbol de widgets.

Requisitos:

  • Tener instalado Rust. 
  • Tener instalado Cargo.

Creando una aplicación con Cargo

Crear proyecto y ubicarse en el directorio creado:

$ cargo new hola_mundo_gui
$ cd hola_mundo_gui

Editar el archivo Cargo.toml para agregar dependencia:

[package]
name = "hola_mundo_gui"
version = "0.1.0"
edition = "2021"

[dependencies]
druid = "0.8.3"

También podemos usar esta forma para agregar la dependencia:

$ cargo add druid

Editar el programa main.rs

use druid::widget::{Align, Flex, Label, TextBox};
use druid::{AppLauncher, Data, Env, Lens, LocalizedString, Widget, WindowDesc, WidgetExt};

const VERTICAL_WIDGET_SPACING: f64 = 20.0;
const TEXT_BOX_WIDTH: f64 = 200.0;
const WINDOW_TITLE: LocalizedString<HelloState> = LocalizedString::new("GUI con Druid y Rust");

#[derive(Clone, Data, Lens)]
struct HelloState {
    name: String,
}

fn main() {
    // describe the main window
    let main_window = WindowDesc::new(build_root_widget())
        .title(WINDOW_TITLE)
        .window_size((400.0, 400.0));

    // create the initial app state
    let initial_state = HelloState {
        name: "Mundo".into(),
    };

    // start the application
    AppLauncher::with_window(main_window)
        .launch(initial_state)
        .expect("Failed to launch application");
}

fn build_root_widget() -> impl Widget<HelloState> {
    // a label that will determine its text based on the current app data.
    let label = Label::new(|data: &HelloState, _env: &Env| format!("Hola {}!", data.name));
    // a textbox that modifies `name`.
    let textbox = TextBox::new()
        .with_placeholder("Cual es tu nombre?")
        .fix_width(TEXT_BOX_WIDTH)
        .lens(HelloState::name);

    // arrange the two widgets vertically, with some padding
    let layout = Flex::column()
        .with_child(label)
        .with_spacer(VERTICAL_WIDGET_SPACING)
        .with_child(textbox);

    // center the two widgets in the available space
    Align::centered(layout)
}

Construir y ejecutar:

$ cargo build
$ cargo run

Se mostrará una ventana con la leyenda "Hola Mundo!" junto con un textbox con el texto de "Cual es tu nombre?". Podrás introducir tu nombre o cualquier texto y la leyenda cambiará a "Hola TEXTO_INTRODUCIDO". 

Quienes hayan usado Java Swing algunas cosas les parecerán familiares. La GUI es muy sencilla, solo contiene un layout y una caja de texto o textbox que cambiará la leyenda con el texto introducido. El método main se encargará de ejecutar la ventana.

Modificaremos el programa para crear un incrementador o decrementador. Tendrá los botones para incrementar o decrementar un contador y se mostrará en el layout cuando el usuario de clic a alguno de los dos botones.

main.rs

use druid::widget::{Button, Flex, Label};
use druid::{AppLauncher, Data, Lens, LocalizedString, Widget, WidgetExt, WindowDesc};

#[derive(Clone, Data, Lens)]
struct AppState {
    count: i32,
}

fn build_ui() -> impl Widget<AppState> {
    // Etiqueta que muestra el contador
    let label = Label::new(|data: &AppState, _env: &_| format!("Contador: {}", data.count))
        .padding(5.0)
        .center();

    // Botones para incrementar y decrementar
    let increment = Button::new("Incrementar")
        .on_click(|_ctx, data: &mut AppState, _env| data.count += 1)
        .padding(5.0);

    let decrement = Button::new("Decrementar")
        .on_click(|_ctx, data: &mut AppState, _env| data.count -= 1)
        .padding(5.0);

    // Organiza los widgets en una columna
    Flex::column()
        .with_child(label)
        .with_child(Flex::row().with_child(increment).with_child(decrement))
        .padding(10.0)
}

fn main() {
    // Configura la ventana principal
    let main_window = WindowDesc::new(build_ui())
        .title(LocalizedString::new("Contador Simple"))
        .window_size((400.0, 200.0));

    // Estado inicial
    let initial_state = AppState { count: 0 };

    // Lanza la aplicación
    AppLauncher::with_window(main_window)
        .launch(initial_state)
        .expect("Fallo al lanzar la aplicación");
}

Construimos y ejecutamos:

$ cargo build
$ cargo run

Se ejecutará el programa y mostrará una ventana con dos botones, uno para incrementar y el otro para decrementar un contador.

¡Hemos creado nuestras primeras aplicaciones GUI con Druid Framework!

Continuaremos con esta serie sobre Rust en próximas entregas.

Enlaces:

https://areweguiyet.com/
https://github.com/linebender/druid/tree/master/druid/examples
https://docs.rs/druid/latest/druid/
https://linebender.org/druid/

Comentarios

Entradas populares de este blog

Programación Windows Batch (CMD) parte 3

Crear ciclos para efectuar operaciones tediosas nos ahorrará tiempo para realizar otras tareas. En está ocasión veremos ciclos con FOR . ¿Cuál es la síntaxis de bucle FOR en Windows Batch? Si está dentro de un archivo *.bat : FOR %%variable IN (seq) DO operaciones Si lo ejecutamos en una terminal: FOR %variable IN (seq) DO operaciones Ejemplo 1 . Recorrer una secuencia de números del 0 al 5: recorrer.bat @ echo off FOR %%i in ( 0 1 2 3 4 5 ) DO echo Hola no. %%i pause Nos imprimirá en pantalla: Hola no. 0 Hola no. 1 Hola no. 2 Hola no. 3 Hola no. 4 Hola no. 5 ¿Puedo usar contadores? Si, se pueden usar. Ejemplo 2 . Uso de contadores: contador.bat @ echo off set numeros = 1 2 3 4 5 6 7 8 9 10 set cont = 0 for %%a in ( %numeros% ) do ( echo Hola no. %%a :: Contador set /a cont+ = 1 ) echo Total: %cont% Este código nos imprimirá, además de los mensajes Hola no. 0 ..., el total de valores conta...

Programación Windows Batch (CMD) parte 4

Siguiendo con la serie de post sobre programación ( 1 , 2 , y 3 ) batch ahora veremos algunas cosas como operaciones aritméticas, operadores lógicos  y uso de ficheros. Cuando somos administradores de servidores o desarrolladores muchas veces tenemos que realizar tareas que nos quitan, relativamente, tiempo valioso que podríamos ocupar para otras cosas (como ver nuestro Facebook, jeje, broma).  Aprender a escribir scripts que nos faciliten algunas tareas siempre es útil. Por ejemplo, conocer todas las características técnicas de nuestro equipo de cómputo nos servirá cuando se realiza peritajes informáticos y soporte al equipo. Realizar respaldos automáticos a nuestras carpetas , archivos y directorios será más sencillo gracias a un script. Pero antes debemos aprender lo básico de la programación en batch. Ejemplo 1. Operaciones aritméticas básicas. aritmetica.bat @ echo off ::Nombre del archivo, imprimirá: aritmetica.bat echo %0 :: Set nos servirá para a...

TIOBE index 2024

El índice TIOBE (o índice de la comunidad de programación TIOBE) mide la popularidad de los lenguajes de programación. Su índice no se basa en qué lenguaje de programación es mejor, si tiene mejor perfomance, si posee escalabilidad, si es más sencillo de aprender, de usar, de implementar, etc. Sólo se basa en la popularidad. En el número de referencias ya sea en blogs, foros de consulta, etc. No necesariamente si las empresas lo están usando en alguno de sus desarrollos. Este índice es útil para darse una idea qué lenguaje está cobrando más popularidad y prestigio. Enseguida una tabla con los primeros 5 lenguajes más populares. Índice de Noviembre 2024   Nov 2024  Nov 2023  Change  Programming Language  Ratings  Change  1  1    Python  22.85%  +8.69%  2  3  +  C++  10.64%  +0.29%  3  4  +  Java  9.60%  +1.26%  4 ...