From 359ab967795b47e7e447356d1b97dd3266510468 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Fri, 21 Feb 2025 10:28:41 -0500 Subject: [PATCH 01/34] Prototype for an l10n message bundle database --- Cargo.lock | 587 +++++++--------------------------------- Cargo.toml | 2 +- l10n-db/Cargo.toml | 19 ++ l10n-db/src/bin/main.rs | 37 +++ l10n-db/src/bundle.rs | 24 ++ l10n-db/src/lib.rs | 20 ++ l10n-db/src/types.rs | 46 ++++ 7 files changed, 248 insertions(+), 487 deletions(-) create mode 100644 l10n-db/Cargo.toml create mode 100644 l10n-db/src/bin/main.rs create mode 100644 l10n-db/src/bundle.rs create mode 100644 l10n-db/src/lib.rs create mode 100644 l10n-db/src/types.rs diff --git a/Cargo.lock b/Cargo.lock index 2d0019f..8f28b30 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -29,18 +29,6 @@ version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "aae1277d39aeec15cb388266ecc24b11c80469deae6067e17a1a7aa9e5c1f234" -[[package]] -name = "ahash" -version = "0.8.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e89da841a80418a9b391ebaea17f5c112ffaaa96f621d2c285b5174da76b9011" -dependencies = [ - "cfg-if", - "once_cell", - "version_check 0.9.5", - "zerocopy", -] - [[package]] name = "aho-corasick" version = "1.1.3" @@ -137,16 +125,6 @@ version = "1.0.95" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04" -[[package]] -name = "assert-json-diff" -version = "2.0.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "47e4f2b81832e72834d7518d8487a0396a28cc408186a2e8854c0f98011faf12" -dependencies = [ - "serde 1.0.217", - "serde_json", -] - [[package]] name = "async-channel" version = "1.9.0" @@ -293,7 +271,7 @@ dependencies = [ "base64ct", "clap", "cool_asserts", - "serde 1.0.217", + "serde 1.0.218", "sha2", "sqlx", "thiserror 1.0.69", @@ -301,12 +279,6 @@ dependencies = [ "uuid 0.4.0", ] -[[package]] -name = "auto-future" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3c1e7e457ea78e524f48639f551fd79703ac3f2237f5ecccdf4708f8a75ad373" - [[package]] name = "autocfg" version = "0.1.8" @@ -322,103 +294,6 @@ version = "1.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26" -[[package]] -name = "axum" -version = "0.7.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "edca88bc138befd0323b20752846e6587272d3b03b0343c8ea28a6f819e6e71f" -dependencies = [ - "async-trait", - "axum-core", - "axum-macros", - "bytes", - "futures-util", - "http 1.2.0", - "http-body 1.0.1", - "http-body-util", - "hyper 1.5.2", - "hyper-util", - "itoa", - "matchit", - "memchr", - "mime 0.3.17", - "percent-encoding 2.3.1", - "pin-project-lite", - "rustversion", - "serde 1.0.217", - "serde_json", - "serde_path_to_error", - "serde_urlencoded", - "sync_wrapper 1.0.2", - "tokio", - "tower", - "tower-layer", - "tower-service", - "tracing", -] - -[[package]] -name = "axum-core" -version = "0.4.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "09f2bd6146b97ae3359fa0cc6d6b376d9539582c7b4220f041a33ec24c226199" -dependencies = [ - "async-trait", - "bytes", - "futures-util", - "http 1.2.0", - "http-body 1.0.1", - "http-body-util", - "mime 0.3.17", - "pin-project-lite", - "rustversion", - "sync_wrapper 1.0.2", - "tower-layer", - "tower-service", - "tracing", -] - -[[package]] -name = "axum-macros" -version = "0.4.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "57d123550fa8d071b7255cb0cc04dc302baa6c8c4a79f55701552684d8399bce" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.96", -] - -[[package]] -name = "axum-test" -version = "16.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "63e3a443d2608936a02a222da7b746eb412fede7225b3030b64fe9be99eab8dc" -dependencies = [ - "anyhow", - "assert-json-diff", - "auto-future", - "axum", - "bytes", - "bytesize", - "cookie 0.18.1", - "http 1.2.0", - "http-body-util", - "hyper 1.5.2", - "hyper-util", - "mime 0.3.17", - "pretty_assertions", - "reserve-port", - "rust-multipart-rfc7578_2", - "serde 1.0.217", - "serde_json", - "serde_urlencoded", - "smallvec", - "tokio", - "tower", - "url 2.5.4", -] - [[package]] name = "az" version = "1.2.1" @@ -522,7 +397,7 @@ version = "2.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8f68f53c83ab957f72c32642f3868eec03eb974d1fb82e453128456482613d36" dependencies = [ - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -577,12 +452,6 @@ version = "1.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "325918d6fe32f23b19878fe4b34794ae41fc19ddbe53b10571a4874d44ffd39b" -[[package]] -name = "bytesize" -version = "1.3.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a3e368af43e418a04d52505cf3dbc23dda4e3407ae2fa99fd0e4f308ce546acc" - [[package]] name = "cairo-rs" version = "0.18.5" @@ -659,7 +528,7 @@ dependencies = [ "iana-time-zone", "js-sys", "num-traits", - "serde 1.0.217", + "serde 1.0.218", "wasm-bindgen", "windows-targets 0.52.6", ] @@ -673,7 +542,7 @@ dependencies = [ "chrono", "chrono-tz-build", "phf 0.11.3", - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -700,9 +569,9 @@ dependencies = [ [[package]] name = "clap" -version = "4.5.26" +version = "4.5.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a8eb5e908ef3a6efbe1ed62520fb7287959888c88485abe072543190ecc66783" +checksum = "92b7b18d71fad5313a1e320fa9897994228ce274b60faa4d694fe0ea89cd9e6d" dependencies = [ "clap_builder", "clap_derive", @@ -710,9 +579,9 @@ dependencies = [ [[package]] name = "clap_builder" -version = "4.5.26" +version = "4.5.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "96b01801b5fc6a0a232407abc821660c9c6d25a1cafc0d4f85f29fb8d9afc121" +checksum = "a35db2071778a7344791a4fb4f95308b5673d219dee3ae348b86642574ecc90c" dependencies = [ "anstream", "anstyle", @@ -722,9 +591,9 @@ dependencies = [ [[package]] name = "clap_derive" -version = "4.5.24" +version = "4.5.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "54b755194d6389280185988721fffba69495eed5ee9feeee9a599b53db80318c" +checksum = "bf4ced95c6f4a675af3da73304b9ac4ed991640c36374e4b46795c49e17cf1ed" dependencies = [ "heck 0.5.0", "proc-macro2", @@ -774,7 +643,7 @@ version = "0.1.0" dependencies = [ "config-derive", "cool_asserts", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "thiserror 1.0.69", ] @@ -812,16 +681,6 @@ dependencies = [ "version_check 0.9.5", ] -[[package]] -name = "cookie" -version = "0.18.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4ddef33a339a91ea89fb53151bd0a4689cfce27055c291dfa69945475d22c747" -dependencies = [ - "time 0.3.37", - "version_check 0.9.5", -] - [[package]] name = "cookie-factory" version = "0.3.3" @@ -958,7 +817,7 @@ dependencies = [ "gio", "glib", "gtk4", - "serde 1.0.217", + "serde 1.0.218", "serde_yml", ] @@ -1004,7 +863,7 @@ dependencies = [ "libadwaita", "memorycache", "reqwest", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "tokio", "unic-langid", @@ -1046,12 +905,6 @@ dependencies = [ "powerfmt", ] -[[package]] -name = "diff" -version = "0.1.13" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "56254986775e3233ffa9c4d7d3faaf6d36a2c09d30b20687e9f88bc8bafc16c8" - [[package]] name = "digest" version = "0.10.7" @@ -1072,7 +925,7 @@ checksum = "2517b0555262aeeda0d107a40ecfbbcf185921180ffb4acf316ebe0887467e26" dependencies = [ "generic-array 0.11.2", "num-traits", - "serde 1.0.217", + "serde 1.0.218", "typenum", ] @@ -1084,7 +937,7 @@ checksum = "a0b0a86c5d31c93238ff4b694fa31f3acdf67440770dc314c57d90e433914397" dependencies = [ "generic-array 0.14.7", "num-traits", - "serde 1.0.217", + "serde 1.0.218", "typenum", ] @@ -1111,7 +964,7 @@ version = "1.13.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0" dependencies = [ - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -1121,7 +974,7 @@ dependencies = [ "chrono", "chrono-tz", "dimensioned 0.7.0", - "serde 1.0.217", + "serde 1.0.218", "serde_derive", "serde_json", "tempfile", @@ -1220,18 +1073,6 @@ dependencies = [ "zune-inflate", ] -[[package]] -name = "fallible-iterator" -version = "0.3.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2acce4a10f12dc2fb14a218589d4f1f62ef011b2d0cc4b3cb1bba8e94da14649" - -[[package]] -name = "fallible-streaming-iterator" -version = "0.1.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7360491ce676a36bf9bb3c56c1aa791658183a54d2744120f27285738d90465a" - [[package]] name = "fastrand" version = "2.3.0" @@ -1267,7 +1108,7 @@ dependencies = [ "bytes", "chrono", "clap", - "cookie 0.17.0", + "cookie", "cool_asserts", "futures-util", "hex-string", @@ -1278,7 +1119,7 @@ dependencies = [ "mime 0.3.17", "mime_guess 2.0.5", "pretty_env_logger", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "sha2", "tempdir", @@ -1440,7 +1281,7 @@ dependencies = [ "chrono-tz", "dimensioned 0.8.0", "emseries", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "tempfile", ] @@ -1762,7 +1603,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0bb0228f477c0900c880fd78c8759b95c7636dbd7842707f49e132378aa2acdc" dependencies = [ "heck 0.4.1", - "proc-macro-crate 2.0.2", + "proc-macro-crate 2.0.0", "proc-macro-error", "proc-macro2", "quote", @@ -1810,7 +1651,7 @@ dependencies = [ "glib-build-tools 0.16.3", "gtk4", "libadwaita", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "tokio", ] @@ -1820,7 +1661,7 @@ name = "gm-dash" version = "0.1.0" dependencies = [ "pipewire", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "tokio", "warp", @@ -1983,15 +1824,6 @@ dependencies = [ "crunchy", ] -[[package]] -name = "hashbrown" -version = "0.14.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" -dependencies = [ - "ahash", -] - [[package]] name = "hashbrown" version = "0.15.2" @@ -2003,22 +1835,13 @@ dependencies = [ "foldhash", ] -[[package]] -name = "hashlink" -version = "0.9.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6ba4ff7128dee98c7dc9794b6a411377e1404dba1c97deb8d1a55297bd25d8af" -dependencies = [ - "hashbrown 0.14.5", -] - [[package]] name = "hashlink" version = "0.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7382cf6263419f2d8df38c55d7da83da5c18aef87fc7a7fc1fb1e344edfe14c1" dependencies = [ - "hashbrown 0.15.2", + "hashbrown", ] [[package]] @@ -2154,29 +1977,6 @@ dependencies = [ "pin-project-lite", ] -[[package]] -name = "http-body" -version = "1.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" -dependencies = [ - "bytes", - "http 1.2.0", -] - -[[package]] -name = "http-body-util" -version = "0.1.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" -dependencies = [ - "bytes", - "futures-util", - "http 1.2.0", - "http-body 1.0.1", - "pin-project-lite", -] - [[package]] name = "httparse" version = "1.9.5" @@ -2226,7 +2026,7 @@ dependencies = [ "futures-util", "h2", "http 0.2.12", - "http-body 0.4.6", + "http-body", "httparse", "httpdate", "itoa", @@ -2238,26 +2038,6 @@ dependencies = [ "want", ] -[[package]] -name = "hyper" -version = "1.5.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "256fb8d4bd6413123cc9d91832d78325c48ff41677595be797d90f42969beae0" -dependencies = [ - "bytes", - "futures-channel", - "futures-util", - "http 1.2.0", - "http-body 1.0.1", - "httparse", - "httpdate", - "itoa", - "pin-project-lite", - "smallvec", - "tokio", - "want", -] - [[package]] name = "hyper-tls" version = "0.5.0" @@ -2271,25 +2051,6 @@ dependencies = [ "tokio-native-tls", ] -[[package]] -name = "hyper-util" -version = "0.1.10" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "df2dcfbe0677734ab2f3ffa7fa7bfd4706bfdc1ef393f2ee30184aed67e631b4" -dependencies = [ - "bytes", - "futures-channel", - "futures-util", - "http 1.2.0", - "http-body 1.0.1", - "hyper 1.5.2", - "pin-project-lite", - "socket2", - "tokio", - "tower-service", - "tracing", -] - [[package]] name = "iana-time-zone" version = "0.1.61" @@ -2343,6 +2104,7 @@ checksum = "13acbb8371917fc971be86fc8057c41a64b521c184808a698c02acc242dbf637" dependencies = [ "displaydoc", "litemap", + "serde 1.0.218", "tinystr", "writeable", "zerovec", @@ -2510,25 +2272,6 @@ dependencies = [ "tiff 0.9.1", ] -[[package]] -name = "include_dir" -version = "0.7.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "923d117408f1e49d914f1a379a309cffe4f18c05cf4e3d12e613a15fc81bd0dd" -dependencies = [ - "include_dir_macros", -] - -[[package]] -name = "include_dir_macros" -version = "0.7.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7cab85a7ed0bd5f0e76d93846e0147172bed2e2d3f859bcc33a8d9699cad1a75" -dependencies = [ - "proc-macro2", - "quote", -] - [[package]] name = "indent_write" version = "2.2.0" @@ -2542,7 +2285,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "62f822373a4fe84d4bb149bf54e584a7f4abec90e072ed49cda0edea5b95471f" dependencies = [ "equivalent", - "hashbrown 0.15.2", + "hashbrown", ] [[package]] @@ -2655,6 +2398,16 @@ dependencies = [ "log 0.4.25", ] +[[package]] +name = "l10n-db" +version = "0.1.0" +dependencies = [ + "clap", + "icu_locid", + "serde 1.0.218", + "toml", +] + [[package]] name = "language-tags" version = "0.2.2" @@ -2850,12 +2603,6 @@ version = "0.1.10" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2532096657941c2fea9c289d370a250971c689d4f143798ff67113ec042024a5" -[[package]] -name = "matchit" -version = "0.7.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0e7465ac9959cc2b1404e8e2367b43684a6d13790fe23056cc8c6c5a6b7bcb94" - [[package]] name = "md-5" version = "0.10.6" @@ -2887,7 +2634,7 @@ version = "0.1.0" dependencies = [ "chrono", "futures", - "serde 1.0.217", + "serde 1.0.218", "serde_derive", "tokio", ] @@ -3205,7 +2952,7 @@ dependencies = [ "cool_asserts", "grid", "nary_tree", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "sgf", "thiserror 1.0.69", @@ -3555,16 +3302,6 @@ dependencies = [ "zerocopy", ] -[[package]] -name = "pretty_assertions" -version = "1.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3ae130e2f271fbc2ac3a40fb1d07180839cdbbe443c7a27e1e3c13c5cac0116d" -dependencies = [ - "diff", - "yansi", -] - [[package]] name = "pretty_env_logger" version = "0.5.0" @@ -3587,11 +3324,10 @@ dependencies = [ [[package]] name = "proc-macro-crate" -version = "2.0.2" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b00f26d3400549137f92511a46ac1cd8ce37cb5598a96d382381458b992a5d24" +checksum = "7e8366a6159044a37876a2b9817124296703c586a5c92e2c53751fa06d8d43e8" dependencies = [ - "toml_datetime", "toml_edit 0.20.2", ] @@ -3929,7 +3665,7 @@ dependencies = [ "futures-util", "h2", "http 0.2.12", - "http-body 0.4.6", + "http-body", "hyper 0.14.32", "hyper-tls", "ipnet", @@ -3941,10 +3677,10 @@ dependencies = [ "percent-encoding 2.3.1", "pin-project-lite", "rustls-pemfile", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "serde_urlencoded", - "sync_wrapper 0.1.2", + "sync_wrapper", "system-configuration", "tokio", "tokio-native-tls", @@ -3956,16 +3692,6 @@ dependencies = [ "winreg", ] -[[package]] -name = "reserve-port" -version = "2.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9838134a2bfaa8e1f40738fcc972ac799de6e0e06b5157acb95fc2b05a0ea283" -dependencies = [ - "lazy_static", - "thiserror 1.0.69", -] - [[package]] name = "result-extended" version = "0.1.0" @@ -3993,47 +3719,6 @@ dependencies = [ "zeroize", ] -[[package]] -name = "rusqlite" -version = "0.32.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7753b721174eb8ff87a9a0e799e2d7bc3749323e773db92e0984debb00019d6e" -dependencies = [ - "bitflags 2.8.0", - "fallible-iterator", - "fallible-streaming-iterator", - "hashlink 0.9.1", - "libsqlite3-sys", - "smallvec", -] - -[[package]] -name = "rusqlite_migration" -version = "1.3.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "923b42e802f7dc20a0a6b5e097ba7c83fe4289da07e49156fecf6af08aa9cd1c" -dependencies = [ - "include_dir", - "log 0.4.25", - "rusqlite", -] - -[[package]] -name = "rust-multipart-rfc7578_2" -version = "0.6.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "03b748410c0afdef2ebbe3685a6a862e2ee937127cdaae623336a459451c8d57" -dependencies = [ - "bytes", - "futures-core", - "futures-util", - "http 0.2.12", - "mime 0.3.17", - "mime_guess 2.0.5", - "rand 0.8.5", - "thiserror 1.0.69", -] - [[package]] name = "rustc-demangle" version = "0.1.24" @@ -4195,18 +3880,18 @@ checksum = "34b623917345a631dc9608d5194cc206b3fe6c3554cd1c75b937e55e285254af" [[package]] name = "serde" -version = "1.0.217" +version = "1.0.218" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "02fc4265df13d6fa1d00ecff087228cc0a2b5f3c0e87e258d8b94a156e984c70" +checksum = "e8dfc9d19bdbf6d17e22319da49161d5d0108e4188e8b680aef6299eed22df60" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.217" +version = "1.0.218" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5a9bf7cf98d04a2b28aead066b7496853d4779c9cc183c440dbac457641e19a0" +checksum = "f09503e191f4e797cb8aac08e9a4a4695c5edf6a2e70e376d961ddd5c969f82b" dependencies = [ "proc-macro2", "quote", @@ -4222,17 +3907,7 @@ dependencies = [ "itoa", "memchr", "ryu", - "serde 1.0.217", -] - -[[package]] -name = "serde_path_to_error" -version = "0.1.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "af99884400da37c88f5e9146b7f1fd0fbcae8f6eec4e9da38b67d05486f814a6" -dependencies = [ - "itoa", - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -4241,7 +3916,7 @@ version = "0.6.8" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1" dependencies = [ - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -4253,7 +3928,7 @@ dependencies = [ "form_urlencoded", "itoa", "ryu", - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -4267,10 +3942,14 @@ dependencies = [ "libyml", "memchr", "ryu", - "serde 1.0.217", + "serde 1.0.218", "version_check 0.9.5", ] +[[package]] +name = "server" +version = "0.1.0" + [[package]] name = "sgf" version = "0.1.0" @@ -4279,7 +3958,7 @@ dependencies = [ "cool_asserts", "nary_tree", "nom", - "serde 1.0.217", + "serde 1.0.218", "thiserror 1.0.69", "typeshare", "uuid 0.8.2", @@ -4380,7 +4059,7 @@ version = "1.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" dependencies = [ - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -4446,14 +4125,14 @@ dependencies = [ "futures-intrusive", "futures-io", "futures-util", - "hashbrown 0.15.2", - "hashlink 0.10.0", + "hashbrown", + "hashlink", "indexmap", "log 0.4.25", "memchr", "once_cell", "percent-encoding 2.3.1", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "sha2", "smallvec", @@ -4490,7 +4169,7 @@ dependencies = [ "once_cell", "proc-macro2", "quote", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "sha2", "sqlx-core", @@ -4534,7 +4213,7 @@ dependencies = [ "percent-encoding 2.3.1", "rand 0.8.5", "rsa", - "serde 1.0.217", + "serde 1.0.218", "sha1", "sha2", "smallvec", @@ -4571,7 +4250,7 @@ dependencies = [ "memchr", "once_cell", "rand 0.8.5", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "sha2", "smallvec", @@ -4598,7 +4277,7 @@ dependencies = [ "libsqlite3-sys", "log 0.4.25", "percent-encoding 2.3.1", - "serde 1.0.217", + "serde 1.0.218", "serde_urlencoded", "sqlx-core", "tracing", @@ -4662,12 +4341,6 @@ version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" -[[package]] -name = "sync_wrapper" -version = "1.0.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0bf256ce5efdfa370213c1dabab5935a12e49f2c58d15e9eac2870d3b4f27263" - [[package]] name = "synstructure" version = "0.13.1" @@ -4835,7 +4508,7 @@ dependencies = [ "itoa", "num-conv", "powerfmt", - "serde 1.0.217", + "serde 1.0.218", "time-core", "time-macros", ] @@ -4871,6 +4544,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9117f5d4db391c1cf6927e7bea3db74b9a1c1add8f7eda9ffd5364f40f57b82f" dependencies = [ "displaydoc", + "serde 1.0.218", "zerovec", ] @@ -4966,23 +4640,23 @@ dependencies = [ [[package]] name = "toml" -version = "0.8.2" +version = "0.8.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "185d8ab0dfbb35cf1399a6344d8484209c088f75f8f68230da55d48d95d43e3d" +checksum = "cd87a5cdd6ffab733b2f74bc4fd7ee5fff6634124999ac278c35fc78c6120148" dependencies = [ - "serde 1.0.217", + "serde 1.0.218", "serde_spanned", "toml_datetime", - "toml_edit 0.20.2", + "toml_edit 0.22.24", ] [[package]] name = "toml_datetime" -version = "0.6.3" +version = "0.6.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7cda73e2f1397b1262d6dfdcef8aafae14d1de7748d66822d3bfeeb6d03e5e4b" +checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41" dependencies = [ - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -4993,7 +4667,7 @@ checksum = "1b5bb770da30e5cbfde35a2d7b9b8a2c4b8ef89548a7a6aeab5c9a576e3e7421" dependencies = [ "indexmap", "toml_datetime", - "winnow", + "winnow 0.5.40", ] [[package]] @@ -5003,48 +4677,23 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "396e4d48bbb2b7554c944bde63101b5ae446cff6ec4a24227428f15eb72ef338" dependencies = [ "indexmap", - "serde 1.0.217", + "toml_datetime", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.22.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17b4795ff5edd201c7cd6dca065ae59972ce77d1b80fa0a84d94950ece7d1474" +dependencies = [ + "indexmap", + "serde 1.0.218", "serde_spanned", "toml_datetime", - "winnow", + "winnow 0.7.3", ] -[[package]] -name = "tower" -version = "0.5.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d039ad9159c98b70ecfd540b2573b97f7f52c3e8d9f8ad57a24b916a536975f9" -dependencies = [ - "futures-core", - "futures-util", - "pin-project-lite", - "sync_wrapper 1.0.2", - "tokio", - "tower-layer", - "tower-service", - "tracing", -] - -[[package]] -name = "tower-http" -version = "0.6.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "403fa3b783d4b626a8ad51d766ab03cb6d2dbfc46b1c5d4448395e6628dc9697" -dependencies = [ - "bitflags 2.8.0", - "bytes", - "http 1.2.0", - "pin-project-lite", - "tower-layer", - "tower-service", -] - -[[package]] -name = "tower-layer" -version = "0.3.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "121c2a6cda46980bb0fcd1647ffaf6cd3fc79a013de288782836f6df9c48780e" - [[package]] name = "tower-service" version = "0.3.3" @@ -5155,7 +4804,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "19be0f411120091e76e13e5a0186d8e2bcc3e7e244afdb70152197f1a8486ceb" dependencies = [ "chrono", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "typeshare-annotation", ] @@ -5279,12 +4928,6 @@ dependencies = [ "percent-encoding 2.3.1", ] -[[package]] -name = "urlencoding" -version = "2.1.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "daf8dba3b7eb870caf1ddeed7bc9d2a049f3cfdfae7cb521b087cc33ae4c49da" - [[package]] name = "utf-8" version = "0.7.6" @@ -5326,7 +4969,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bc5cf98d8186244414c848017f0e2676b3fcb46807f6668a97dfe67359a3c4b7" dependencies = [ "getrandom", - "serde 1.0.217", + "serde 1.0.218", ] [[package]] @@ -5368,37 +5011,6 @@ version = "0.9.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a" -[[package]] -name = "visions" -version = "0.1.0" -dependencies = [ - "async-std", - "async-trait", - "authdb", - "axum", - "axum-test", - "chrono", - "cool_asserts", - "futures", - "include_dir", - "lazy_static", - "mime 0.3.17", - "mime_guess 2.0.5", - "pretty_env_logger", - "result-extended", - "rusqlite", - "rusqlite_migration", - "serde 1.0.217", - "serde_json", - "thiserror 2.0.11", - "tokio", - "tokio-stream", - "tower-http", - "typeshare", - "urlencoding", - "uuid 1.12.0", -] - [[package]] name = "wait-timeout" version = "0.2.0" @@ -5436,7 +5048,7 @@ dependencies = [ "percent-encoding 2.3.1", "pin-project", "scoped-tls", - "serde 1.0.217", + "serde 1.0.218", "serde_json", "serde_urlencoded", "tokio", @@ -5758,6 +5370,15 @@ dependencies = [ "memchr", ] +[[package]] +name = "winnow" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e7f4ea97f6f78012141bcdb6a216b2609f0979ada50b20ca5b52dde2eac2bb1" +dependencies = [ + "memchr", +] + [[package]] name = "winreg" version = "0.50.0" @@ -5780,12 +5401,6 @@ version = "0.5.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1e9df38ee2d2c3c5948ea468a8406ff0db0b29ae1ffde1bcf20ef305bcc95c51" -[[package]] -name = "yansi" -version = "1.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cfe53a6657fd280eaa890a3bc59152892ffa3e30101319d168b781ed6529b049" - [[package]] name = "yansi-term" version = "0.1.2" @@ -5801,7 +5416,7 @@ version = "0.7.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "120e6aef9aa629e3d4f52dc8cc43a015c7724194c97dfaf45180d2daf2b77f40" dependencies = [ - "serde 1.0.217", + "serde 1.0.218", "stable_deref_trait", "yoke-derive", "zerofrom", diff --git a/Cargo.toml b/Cargo.toml index cc29ea7..d982233 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -33,4 +33,4 @@ members = [ "tree", "visions/server", "gm-dash/server" -] +, "l10n-db"] diff --git a/l10n-db/Cargo.toml b/l10n-db/Cargo.toml new file mode 100644 index 0000000..d50c181 --- /dev/null +++ b/l10n-db/Cargo.toml @@ -0,0 +1,19 @@ +[package] +name = "l10n-db" +version = "0.1.0" +edition = "2021" + +[dependencies] +clap = { version = "4.5.30", features = ["derive"] } +icu_locid = { version = "1.5.0", features = ["serde"] } +serde = { version = "1.0.218", features = ["derive"] } +toml = "0.8.20" + +# [lib] +# name = "l10n_db" +# path = "src/lib.rs" +# +# [[bin]] +# name = "l10n-db" +# path = "src/main.rs" + diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs new file mode 100644 index 0000000..0ff2a5c --- /dev/null +++ b/l10n-db/src/bin/main.rs @@ -0,0 +1,37 @@ +use clap::{Parser, Subcommand}; + +use l10n_db; + +#[derive(Parser)] +#[command(version, about, long_about = None)] +struct Cli { + #[command(subcommand)] + command: Option<Commands>, +} + +#[derive(Subcommand)] +enum Commands { + // Edit, potentially creating, a key + // EditKey { }, + // List al keys in the database + ListKeys, + // Search the database + // Search { }, + /// Export the database + Export { + #[arg(short, long)] + format: String, + #[arg(short, long)] + locale: String + }, +} + +fn main() { + let cli = Cli::parse(); + + match &cli.command { + Some(Commands::ListKeys) => todo!(), + Some(Commands::Export{..}) => todo!(), + None => {}, + } +} diff --git a/l10n-db/src/bundle.rs b/l10n-db/src/bundle.rs new file mode 100644 index 0000000..2f7572c --- /dev/null +++ b/l10n-db/src/bundle.rs @@ -0,0 +1,24 @@ +use std::{collections::HashMap, path::Path}; + +use crate::Message; + +pub struct Bundle { + messages: HashMap<String, Message> +} + +impl Bundle { + pub fn new() -> Self { + Self{ messages: HashMap::new() } + } + + pub fn load_from_disk(path: &Path) -> Self { + Self { + messages: HashMap::new() + } + } +} + + +#[cfg(test)] +mod test { +} diff --git a/l10n-db/src/lib.rs b/l10n-db/src/lib.rs new file mode 100644 index 0000000..1dd4cc3 --- /dev/null +++ b/l10n-db/src/lib.rs @@ -0,0 +1,20 @@ +mod bundle; +pub use bundle::Bundle; + +mod types; +pub use types::{Message, Variant}; + +/* +#[cfg(test)] +mod test { + #[test] + fn it_can_represent_an_untranslated_message() { + todo!() + } + + #[test] + fn it_can_represent_a_partially_translated_message() { + todo!() + } +} +*/ diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs new file mode 100644 index 0000000..0e7de32 --- /dev/null +++ b/l10n-db/src/types.rs @@ -0,0 +1,46 @@ +use std::{collections::HashMap, time::SystemTime}; + +use icu_locid::LanguageIdentifier; +use serde::{Deserialize, Serialize}; + +#[derive(Deserialize, Serialize)] +pub struct Message { + key: String, + description: String, + variants: HashMap<LanguageIdentifier, Variant>, +} + +impl Message { + pub fn new(key: String) -> Self { + Self { + key, + description: "".to_owned(), + variants: HashMap::new(), + } + } + + pub fn set_description(&mut self, desc: String) { + self.description = desc; + } + + pub fn description(&self) -> &str { + &self.description + } + + pub fn variant_mut(&mut self, locale: LanguageIdentifier) -> &mut Variant { + self.variants.entry(locale.clone()).or_insert(Variant { + locale, + content: "".to_owned(), + modified: SystemTime::now(), + }) + } +} + +#[derive(Deserialize, Serialize)] +pub struct Variant { + locale: LanguageIdentifier, + content: String, + modified: SystemTime, +} + + From e0392a41506ba64c3c65312df33d227c2167cfed Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 22 Feb 2025 10:23:46 -0500 Subject: [PATCH 02/34] Write a single phrase to disk --- l10n-db/i18n/OpenSandbox.toml | 7 +++++++ l10n-db/src/bin/main.rs | 29 ++++++++++++++++++++++++---- l10n-db/src/bundle.rs | 36 +++++++++++++++++++++++++---------- l10n-db/src/types.rs | 19 ++++++++++++++---- 4 files changed, 73 insertions(+), 18 deletions(-) create mode 100644 l10n-db/i18n/OpenSandbox.toml diff --git a/l10n-db/i18n/OpenSandbox.toml b/l10n-db/i18n/OpenSandbox.toml new file mode 100644 index 0000000..7df243d --- /dev/null +++ b/l10n-db/i18n/OpenSandbox.toml @@ -0,0 +1,7 @@ +key = "OpenSandbox" +description = "Open a sandbox" + +[variants.en-US] +locale = "en-US" +content = "Open a sandbox vault" +modified = 1740198524 diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index 0ff2a5c..39602cb 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -1,6 +1,9 @@ +use std::path::PathBuf; + use clap::{Parser, Subcommand}; -use l10n_db; +use icu_locid::{langid, LanguageIdentifier}; +use l10n_db::{self, Bundle}; #[derive(Parser)] #[command(version, about, long_about = None)] @@ -11,9 +14,9 @@ struct Cli { #[derive(Subcommand)] enum Commands { - // Edit, potentially creating, a key - // EditKey { }, - // List al keys in the database + /// Edit, potentially creating, a key + EditKey { name: String, locale: String }, + /// List al keys in the database ListKeys, // Search the database // Search { }, @@ -26,10 +29,28 @@ enum Commands { }, } +fn edit_key(bundle: &mut Bundle, key: String, editor: &str) { + let message = bundle.message(key); + + println!("message: {:?}", message); + + message.set_description("Open a sandbox".to_owned()); + let variant = message.variant_mut(langid!("en-US")); + variant.set_content("Open a sandbox vault".to_owned()); + + bundle.save(); +} + fn main() { + let editor = std::env::var("EDITOR").unwrap(); + let db_path = std::env::var("DB_PATH").unwrap(); + let cli = Cli::parse(); + let mut bundle = Bundle::load_from_disk(PathBuf::from(&db_path)); + match &cli.command { + Some(Commands::EditKey{ name, locale } ) => edit_key(&mut bundle, name.to_owned(), &editor), Some(Commands::ListKeys) => todo!(), Some(Commands::Export{..}) => todo!(), None => {}, diff --git a/l10n-db/src/bundle.rs b/l10n-db/src/bundle.rs index 2f7572c..aa0f8ee 100644 --- a/l10n-db/src/bundle.rs +++ b/l10n-db/src/bundle.rs @@ -1,24 +1,40 @@ -use std::{collections::HashMap, path::Path}; +use std::{collections::HashMap, fs::File, io::Write, path::PathBuf}; use crate::Message; pub struct Bundle { - messages: HashMap<String, Message> + path: PathBuf, + messages: HashMap<String, Message>, } impl Bundle { - pub fn new() -> Self { - Self{ messages: HashMap::new() } - } - - pub fn load_from_disk(path: &Path) -> Self { + pub fn load_from_disk(path: PathBuf) -> Self { Self { - messages: HashMap::new() + path, + messages: HashMap::new(), } } + + pub fn message(&mut self, name: String) -> &mut Message { + self.messages + .entry(name.to_owned()) + .or_insert(Message::new(name.to_owned())) + } + + pub fn save(&self) { + self.messages.iter().for_each(|(key, value)| { + let mut path = self.path.clone(); + path.push(key); + path.set_extension("toml"); + save_file(&path, toml::to_string(value).unwrap().as_bytes()); + }); + } } +fn save_file(path: &PathBuf, s: &[u8]) { + let mut f = File::create(path).unwrap(); + f.write(s).unwrap(); +} #[cfg(test)] -mod test { -} +mod test {} diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs index 0e7de32..c6ca0d1 100644 --- a/l10n-db/src/types.rs +++ b/l10n-db/src/types.rs @@ -1,9 +1,9 @@ -use std::{collections::HashMap, time::SystemTime}; +use std::{collections::HashMap, time::{SystemTime, UNIX_EPOCH}}; use icu_locid::LanguageIdentifier; -use serde::{Deserialize, Serialize}; +use serde::{Deserialize, Serialize, Serializer}; -#[derive(Deserialize, Serialize)] +#[derive(Deserialize, Serialize, Debug)] pub struct Message { key: String, description: String, @@ -36,11 +36,22 @@ impl Message { } } -#[derive(Deserialize, Serialize)] +#[derive(Deserialize, Serialize, Debug)] pub struct Variant { locale: LanguageIdentifier, content: String, + #[serde(serialize_with = "time_to_number")] modified: SystemTime, } +impl Variant { + pub fn set_content(&mut self, content: String) { + self.content = content; + self.modified = SystemTime::now(); + } +} +fn time_to_number<S>(time: &SystemTime, s: S) -> Result<S::Ok, S::Error> +where S: Serializer { + s.serialize_u64(time.duration_since(UNIX_EPOCH).unwrap().as_secs()) +} From e16fef2b14daa27c979c07c75b863fc3b645545c Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 22 Feb 2025 11:22:14 -0500 Subject: [PATCH 03/34] Write a rudimentary editor --- Cargo.lock | 43 +++++++++++++++++++++++++---- l10n-db/Cargo.toml | 1 + l10n-db/i18n/OpenSandbox.toml | 6 ++-- l10n-db/src/bin/main.rs | 29 ++++++++++++++++--- l10n-db/src/editor.rs | 52 +++++++++++++++++++++++++++++++++++ l10n-db/src/lib.rs | 3 ++ l10n-db/src/types.rs | 4 +++ 7 files changed, 125 insertions(+), 13 deletions(-) create mode 100644 l10n-db/src/editor.rs diff --git a/Cargo.lock b/Cargo.lock index 8f28b30..372fc49 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1497,6 +1497,18 @@ dependencies = [ "wasi 0.11.0+wasi-snapshot-preview1", ] +[[package]] +name = "getrandom" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8" +dependencies = [ + "cfg-if", + "libc", + "wasi 0.13.3+wasi-0.2.2", + "windows-targets 0.52.6", +] + [[package]] name = "gif" version = "0.11.4" @@ -2405,6 +2417,7 @@ dependencies = [ "clap", "icu_locid", "serde 1.0.218", + "tempfile", "toml", ] @@ -3502,7 +3515,7 @@ version = "0.6.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" dependencies = [ - "getrandom", + "getrandom 0.2.15", ] [[package]] @@ -4404,13 +4417,13 @@ dependencies = [ [[package]] name = "tempfile" -version = "3.15.0" +version = "3.17.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9a8a559c81686f576e8cd0290cd2a24a2a9ad80c98b3478856500fcbd7acd704" +checksum = "22e5a0acb1f3f55f65cc4a866c361b2fb2a0ff6366785ae6fbb5f85df07ba230" dependencies = [ "cfg-if", "fastrand", - "getrandom", + "getrandom 0.3.1", "once_cell", "rustix", "windows-sys 0.59.0", @@ -4968,7 +4981,7 @@ version = "0.8.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bc5cf98d8186244414c848017f0e2676b3fcb46807f6668a97dfe67359a3c4b7" dependencies = [ - "getrandom", + "getrandom 0.2.15", "serde 1.0.218", ] @@ -4978,7 +4991,7 @@ version = "1.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "744018581f9a3454a9e15beb8a33b017183f1e7c0cd170232a2d1453b23a51c4" dependencies = [ - "getrandom", + "getrandom 0.2.15", ] [[package]] @@ -5070,6 +5083,15 @@ version = "0.11.0+wasi-snapshot-preview1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" +[[package]] +name = "wasi" +version = "0.13.3+wasi-0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26816d2e1a4a36a2940b96c5296ce403917633dff8f3440e9b236ed6f6bacad2" +dependencies = [ + "wit-bindgen-rt", +] + [[package]] name = "wasite" version = "0.1.0" @@ -5389,6 +5411,15 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "wit-bindgen-rt" +version = "0.33.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3268f3d866458b787f390cf61f4bbb563b922d091359f9608842999eaee3943c" +dependencies = [ + "bitflags 2.8.0", +] + [[package]] name = "write16" version = "1.0.0" diff --git a/l10n-db/Cargo.toml b/l10n-db/Cargo.toml index d50c181..8dda9ff 100644 --- a/l10n-db/Cargo.toml +++ b/l10n-db/Cargo.toml @@ -7,6 +7,7 @@ edition = "2021" clap = { version = "4.5.30", features = ["derive"] } icu_locid = { version = "1.5.0", features = ["serde"] } serde = { version = "1.0.218", features = ["derive"] } +tempfile = "3.17.1" toml = "0.8.20" # [lib] diff --git a/l10n-db/i18n/OpenSandbox.toml b/l10n-db/i18n/OpenSandbox.toml index 7df243d..1f24302 100644 --- a/l10n-db/i18n/OpenSandbox.toml +++ b/l10n-db/i18n/OpenSandbox.toml @@ -1,7 +1,7 @@ key = "OpenSandbox" -description = "Open a sandbox" +description = "a basic description without any content" [variants.en-US] locale = "en-US" -content = "Open a sandbox vault" -modified = 1740198524 +content = "" +modified = 1740241312 diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index 39602cb..a51d432 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -1,9 +1,9 @@ -use std::path::PathBuf; +use std::{io::{BufReader, Read, Write}, path::PathBuf, process::Command}; use clap::{Parser, Subcommand}; use icu_locid::{langid, LanguageIdentifier}; -use l10n_db::{self, Bundle}; +use l10n_db::{self, Bundle, Editor}; #[derive(Parser)] #[command(version, about, long_about = None)] @@ -29,14 +29,35 @@ enum Commands { }, } -fn edit_key(bundle: &mut Bundle, key: String, editor: &str) { +fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor: &str) { let message = bundle.message(key); + Editor::edit(message, locale, editor); + + println!("final version"); + println!("{}", toml::to_string(&message).unwrap()); + + /* + let mut editor_file = tempfile::NamedTempFile::new().unwrap(); + let _ = editor_file.write(toml::to_string(message).unwrap().as_bytes()); + let _ = editor_file.flush(); + let mut cmd = Command::new(editor).args([editor_file.path()]).spawn().unwrap(); + cmd.wait().unwrap(); + let _ = editor_file.flush(); + let mut reader = BufReader::new(editor_file); + let mut content = Vec::new(); + let _ = reader.read_to_end(&mut content); + + println!("content"); + println!("{}", String::from_utf8(content).unwrap()); + + println!("message: {:?}", message); message.set_description("Open a sandbox".to_owned()); let variant = message.variant_mut(langid!("en-US")); variant.set_content("Open a sandbox vault".to_owned()); + */ bundle.save(); } @@ -50,7 +71,7 @@ fn main() { let mut bundle = Bundle::load_from_disk(PathBuf::from(&db_path)); match &cli.command { - Some(Commands::EditKey{ name, locale } ) => edit_key(&mut bundle, name.to_owned(), &editor), + Some(Commands::EditKey{ name, locale } ) => edit_key(&mut bundle, name.to_owned(), langid!("en-US"), &editor), Some(Commands::ListKeys) => todo!(), Some(Commands::Export{..}) => todo!(), None => {}, diff --git a/l10n-db/src/editor.rs b/l10n-db/src/editor.rs new file mode 100644 index 0000000..a650b39 --- /dev/null +++ b/l10n-db/src/editor.rs @@ -0,0 +1,52 @@ +use std::{io::{BufReader, Read, Write}, path::Path, process::Command}; + +use icu_locid::LanguageIdentifier; +use serde::{Deserialize, Serialize}; + +use crate::{Message, Variant}; + + +#[derive(Serialize, Deserialize, Debug, Clone)] +struct EditorMessage { + description: String, + content: String, +} + +impl EditorMessage { + fn from_variant(description: String, variant: &Variant) -> Self { + Self { + description, + content: variant.content().to_string() + } + } +} + +pub struct Editor { +} + +impl Editor { + pub fn edit(msg: &mut Message, locale: LanguageIdentifier, editor: &str) { + let description = msg.description().to_owned(); + let variant = msg.variant_mut(locale); + let editable_content = EditorMessage::from_variant(description, &variant); + + let mut file = tempfile::NamedTempFile::new().unwrap(); + let _ = file.write(toml::to_string(&editable_content).unwrap().as_bytes()); + let _ = file.flush(); + let mut cmd = Command::new(editor).args([file.path()]).spawn().unwrap(); + cmd.wait().unwrap(); + let file = file.reopen().unwrap(); + let mut reader = BufReader::new(file); + let mut content = Vec::new(); + let _ = reader.read_to_end(&mut content); + + println!("content"); + println!("{}", String::from_utf8(content.clone()).unwrap()); + + let new_variant: EditorMessage = toml::from_str(&String::from_utf8(content).unwrap()).unwrap(); + + variant.set_content(new_variant.content); + msg.set_description(new_variant.description); + } +} + diff --git a/l10n-db/src/lib.rs b/l10n-db/src/lib.rs index 1dd4cc3..9dd0718 100644 --- a/l10n-db/src/lib.rs +++ b/l10n-db/src/lib.rs @@ -1,6 +1,9 @@ mod bundle; pub use bundle::Bundle; +mod editor; +pub use editor::Editor; + mod types; pub use types::{Message, Variant}; diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs index c6ca0d1..c6ee3e5 100644 --- a/l10n-db/src/types.rs +++ b/l10n-db/src/types.rs @@ -45,6 +45,10 @@ pub struct Variant { } impl Variant { + pub fn content(&self) -> &str { + &self.content + } + pub fn set_content(&mut self, content: String) { self.content = content; self.modified = SystemTime::now(); From 1c3d0711e1c028a3ebfb6ffca3cbef79172085cf Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 22 Feb 2025 16:40:23 -0500 Subject: [PATCH 04/34] Start reading the bundle --- Cargo.lock | 1 + l10n-db/Cargo.toml | 1 + l10n-db/i18n/OpenSandbox.toml | 6 +++--- l10n-db/src/bin/main.rs | 33 +++++---------------------------- l10n-db/src/bundle.rs | 18 +++++++++++++++--- l10n-db/src/types.rs | 34 ++++++++++++++++++++++++++-------- 6 files changed, 51 insertions(+), 42 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 372fc49..d48d3bd 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2414,6 +2414,7 @@ dependencies = [ name = "l10n-db" version = "0.1.0" dependencies = [ + "chrono", "clap", "icu_locid", "serde 1.0.218", diff --git a/l10n-db/Cargo.toml b/l10n-db/Cargo.toml index 8dda9ff..977cd68 100644 --- a/l10n-db/Cargo.toml +++ b/l10n-db/Cargo.toml @@ -4,6 +4,7 @@ version = "0.1.0" edition = "2021" [dependencies] +chrono = { version = "0.4.39", features = ["serde"] } clap = { version = "4.5.30", features = ["derive"] } icu_locid = { version = "1.5.0", features = ["serde"] } serde = { version = "1.0.218", features = ["derive"] } diff --git a/l10n-db/i18n/OpenSandbox.toml b/l10n-db/i18n/OpenSandbox.toml index 1f24302..4c963f6 100644 --- a/l10n-db/i18n/OpenSandbox.toml +++ b/l10n-db/i18n/OpenSandbox.toml @@ -1,7 +1,7 @@ key = "OpenSandbox" -description = "a basic description without any content" +description = "A sandbox vault refers to a stock vault that contains test data and allows the user to make edits and run experiments on test data." [variants.en-US] locale = "en-US" -content = "" -modified = 1740241312 +content = "Open Sandbox vault" +modified = "2025-02-22T21:35:02.032300406Z" diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index a51d432..ecd079f 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -31,34 +31,7 @@ enum Commands { fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor: &str) { let message = bundle.message(key); - Editor::edit(message, locale, editor); - - println!("final version"); - println!("{}", toml::to_string(&message).unwrap()); - - /* - let mut editor_file = tempfile::NamedTempFile::new().unwrap(); - let _ = editor_file.write(toml::to_string(message).unwrap().as_bytes()); - let _ = editor_file.flush(); - let mut cmd = Command::new(editor).args([editor_file.path()]).spawn().unwrap(); - cmd.wait().unwrap(); - let _ = editor_file.flush(); - let mut reader = BufReader::new(editor_file); - let mut content = Vec::new(); - let _ = reader.read_to_end(&mut content); - - println!("content"); - println!("{}", String::from_utf8(content).unwrap()); - - - println!("message: {:?}", message); - - message.set_description("Open a sandbox".to_owned()); - let variant = message.variant_mut(langid!("en-US")); - variant.set_content("Open a sandbox vault".to_owned()); - */ - bundle.save(); } @@ -72,7 +45,11 @@ fn main() { match &cli.command { Some(Commands::EditKey{ name, locale } ) => edit_key(&mut bundle, name.to_owned(), langid!("en-US"), &editor), - Some(Commands::ListKeys) => todo!(), + Some(Commands::ListKeys) => { + for (key, _) in bundle.message_iter() { + println!("{}", key); + } + }, Some(Commands::Export{..}) => todo!(), None => {}, } diff --git a/l10n-db/src/bundle.rs b/l10n-db/src/bundle.rs index aa0f8ee..3144ea9 100644 --- a/l10n-db/src/bundle.rs +++ b/l10n-db/src/bundle.rs @@ -9,10 +9,22 @@ pub struct Bundle { impl Bundle { pub fn load_from_disk(path: PathBuf) -> Self { - Self { - path, - messages: HashMap::new(), + let mut messages = HashMap::new(); + if path.is_dir() { + for entry in std::fs::read_dir(&path).unwrap() { + let entry = entry.unwrap(); + let path = entry.path().clone(); + let key = path.file_stem().unwrap(); + + let message = Message::from_file(&entry.path()); + messages.insert(key.to_str().unwrap().to_owned(), message); + } } + Self { path, messages } + } + + pub fn message_iter(&self) -> impl Iterator<Item = (&String, &Message)> { + self.messages.iter() } pub fn message(&mut self, name: String) -> &mut Message { diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs index c6ee3e5..32c4268 100644 --- a/l10n-db/src/types.rs +++ b/l10n-db/src/types.rs @@ -1,7 +1,8 @@ -use std::{collections::HashMap, time::{SystemTime, UNIX_EPOCH}}; +use std::{collections::HashMap, io::{BufReader, Read}, path::Path}; +use chrono::{DateTime, Utc}; use icu_locid::LanguageIdentifier; -use serde::{Deserialize, Serialize, Serializer}; +use serde::{de, Deserialize, Deserializer, Serialize, Serializer}; #[derive(Deserialize, Serialize, Debug)] pub struct Message { @@ -19,6 +20,14 @@ impl Message { } } + pub fn from_file(path: &Path) -> Message { + let file = std::fs::File::open(path).unwrap(); + let mut content = Vec::new(); + let mut reader = BufReader::new(file); + let _ = reader.read_to_end(&mut content); + toml::from_str(&String::from_utf8(content).unwrap()).unwrap() + } + pub fn set_description(&mut self, desc: String) { self.description = desc; } @@ -31,7 +40,7 @@ impl Message { self.variants.entry(locale.clone()).or_insert(Variant { locale, content: "".to_owned(), - modified: SystemTime::now(), + modified: Utc::now(), }) } } @@ -40,8 +49,7 @@ impl Message { pub struct Variant { locale: LanguageIdentifier, content: String, - #[serde(serialize_with = "time_to_number")] - modified: SystemTime, + modified: DateTime<Utc>, } impl Variant { @@ -51,11 +59,21 @@ impl Variant { pub fn set_content(&mut self, content: String) { self.content = content; - self.modified = SystemTime::now(); + self.modified = Utc::now(); } } -fn time_to_number<S>(time: &SystemTime, s: S) -> Result<S::Ok, S::Error> +/* +fn time_to_number<S>(time: &DateTime, s: S) -> Result<S::Ok, S::Error> where S: Serializer { - s.serialize_u64(time.duration_since(UNIX_EPOCH).unwrap().as_secs()) + let seconds: u64 = time.as_secs(); + s.serialize_u64(seconds) } + +fn number_to_time<'de, D>(d: D) -> Result<DateTime, D::Error> +where D: Deserializer<'de> { + let buf = String::deserialize(d)?; + let num = buf.parse::<u64>().unwrap(); + Ok(DateTime::try_from(num).unwrap()) +} +*/ From 200c13a14e2ad8b5777a3602ce6c1bb0559357ad Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 22 Feb 2025 18:39:32 -0500 Subject: [PATCH 05/34] Add a configuration file --- l10n-db/Cargo.toml | 1 + l10n-db/config.toml | 8 ++++++ l10n-db/i18n/OpenSandbox.toml | 9 +++++-- l10n-db/src/bin/main.rs | 46 ++++++++++++++++++++++++++--------- l10n-db/src/editor.rs | 22 ++++++++++------- l10n-db/src/lib.rs | 3 +++ l10n-db/src/types.rs | 2 +- l10n-db/src/utils.rs | 31 +++++++++++++++++++++++ 8 files changed, 99 insertions(+), 23 deletions(-) create mode 100644 l10n-db/config.toml create mode 100644 l10n-db/src/utils.rs diff --git a/l10n-db/Cargo.toml b/l10n-db/Cargo.toml index 977cd68..e36077b 100644 --- a/l10n-db/Cargo.toml +++ b/l10n-db/Cargo.toml @@ -9,6 +9,7 @@ clap = { version = "4.5.30", features = ["derive"] } icu_locid = { version = "1.5.0", features = ["serde"] } serde = { version = "1.0.218", features = ["derive"] } tempfile = "3.17.1" +thiserror = "2.0.11" toml = "0.8.20" # [lib] diff --git a/l10n-db/config.toml b/l10n-db/config.toml new file mode 100644 index 0000000..913bdbb --- /dev/null +++ b/l10n-db/config.toml @@ -0,0 +1,8 @@ +db_path = "./i18n" +base_locale = "en" +locales = [ + "en", + "eo", + "de", + "es", +] diff --git a/l10n-db/i18n/OpenSandbox.toml b/l10n-db/i18n/OpenSandbox.toml index 4c963f6..81b7d4f 100644 --- a/l10n-db/i18n/OpenSandbox.toml +++ b/l10n-db/i18n/OpenSandbox.toml @@ -1,7 +1,12 @@ key = "OpenSandbox" description = "A sandbox vault refers to a stock vault that contains test data and allows the user to make edits and run experiments on test data." -[variants.en-US] -locale = "en-US" +[variants.eo] +locale = "eo" +content = "Malfermi sekurkofro" +modified = "2025-02-22T23:35:29.692299091Z" + +[variants.en] +locale = "en" content = "Open Sandbox vault" modified = "2025-02-22T21:35:02.032300406Z" diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index ecd079f..1860136 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -1,9 +1,14 @@ -use std::{io::{BufReader, Read, Write}, path::PathBuf, process::Command}; +use std::{ + io::{BufReader, Read, Write}, + path::PathBuf, + process::Command, +}; use clap::{Parser, Subcommand}; use icu_locid::{langid, LanguageIdentifier}; -use l10n_db::{self, Bundle, Editor}; +use l10n_db::{self, read_file, Bundle, Editor, ReadError}; +use serde::Deserialize; #[derive(Parser)] #[command(version, about, long_about = None)] @@ -15,7 +20,12 @@ struct Cli { #[derive(Subcommand)] enum Commands { /// Edit, potentially creating, a key - EditKey { name: String, locale: String }, + EditKey { + #[arg(short, long)] + name: String, + #[arg(short, long)] + locale: String, + }, /// List al keys in the database ListKeys, // Search the database @@ -25,10 +35,17 @@ enum Commands { #[arg(short, long)] format: String, #[arg(short, long)] - locale: String + locale: String, }, } +#[derive(Debug, Deserialize)] +struct Config { + db_path: PathBuf, + base_locale: LanguageIdentifier, + locales: Vec<LanguageIdentifier>, +} + fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor: &str) { let message = bundle.message(key); Editor::edit(message, locale, editor); @@ -36,21 +53,28 @@ fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor } fn main() { - let editor = std::env::var("EDITOR").unwrap(); - let db_path = std::env::var("DB_PATH").unwrap(); + let editor = std::env::var("EDITOR").expect("Set EDITOR to the path to your favorite editor"); + + let config: Config = read_file(&PathBuf::from("./config.toml")) + .and_then(|bytes| String::from_utf8(bytes).map_err(|_| ReadError::InvalidFormat)) + .and_then(|content| toml::from_str(&content).map_err(|_| ReadError::InvalidFormat)) + .unwrap(); let cli = Cli::parse(); - let mut bundle = Bundle::load_from_disk(PathBuf::from(&db_path)); + let mut bundle = Bundle::load_from_disk(PathBuf::from(&config.db_path)); match &cli.command { - Some(Commands::EditKey{ name, locale } ) => edit_key(&mut bundle, name.to_owned(), langid!("en-US"), &editor), + Some(Commands::EditKey { name, locale }) => { + let identifier = locale.parse::<LanguageIdentifier>().unwrap(); + edit_key(&mut bundle, name.to_owned(), identifier, &editor) + } Some(Commands::ListKeys) => { for (key, _) in bundle.message_iter() { println!("{}", key); } - }, - Some(Commands::Export{..}) => todo!(), - None => {}, + } + Some(Commands::Export { .. }) => todo!(), + None => {} } } diff --git a/l10n-db/src/editor.rs b/l10n-db/src/editor.rs index a650b39..c51d580 100644 --- a/l10n-db/src/editor.rs +++ b/l10n-db/src/editor.rs @@ -1,17 +1,19 @@ use std::{io::{BufReader, Read, Write}, path::Path, process::Command}; -use icu_locid::LanguageIdentifier; +use icu_locid::{langid, LanguageIdentifier}; use serde::{Deserialize, Serialize}; -use crate::{Message, Variant}; +use crate::{read_fh, Message, Variant}; #[derive(Serialize, Deserialize, Debug, Clone)] struct EditorMessage { description: String, + source: String, content: String, } +/* impl EditorMessage { fn from_variant(description: String, variant: &Variant) -> Self { Self { @@ -20,6 +22,7 @@ impl EditorMessage { } } } +*/ pub struct Editor { } @@ -27,8 +30,14 @@ pub struct Editor { impl Editor { pub fn edit(msg: &mut Message, locale: LanguageIdentifier, editor: &str) { let description = msg.description().to_owned(); + let source_string = msg.variant_mut(langid!("en")).content().to_owned(); let variant = msg.variant_mut(locale); - let editable_content = EditorMessage::from_variant(description, &variant); + // let editable_content = EditorMessage::from_variant(description, &variant); + let editable_content = EditorMessage { + description, + source: source_string, + content: variant.content().to_owned(), + }; let mut file = tempfile::NamedTempFile::new().unwrap(); let _ = file.write(toml::to_string(&editable_content).unwrap().as_bytes()); @@ -36,12 +45,7 @@ impl Editor { let mut cmd = Command::new(editor).args([file.path()]).spawn().unwrap(); cmd.wait().unwrap(); let file = file.reopen().unwrap(); - let mut reader = BufReader::new(file); - let mut content = Vec::new(); - let _ = reader.read_to_end(&mut content); - - println!("content"); - println!("{}", String::from_utf8(content.clone()).unwrap()); + let content = read_fh(&file).unwrap(); let new_variant: EditorMessage = toml::from_str(&String::from_utf8(content).unwrap()).unwrap(); diff --git a/l10n-db/src/lib.rs b/l10n-db/src/lib.rs index 9dd0718..ab1eec2 100644 --- a/l10n-db/src/lib.rs +++ b/l10n-db/src/lib.rs @@ -7,6 +7,9 @@ pub use editor::Editor; mod types; pub use types::{Message, Variant}; +mod utils; +pub use utils::*; + /* #[cfg(test)] mod test { diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs index 32c4268..13b447d 100644 --- a/l10n-db/src/types.rs +++ b/l10n-db/src/types.rs @@ -45,7 +45,7 @@ impl Message { } } -#[derive(Deserialize, Serialize, Debug)] +#[derive(Deserialize, Serialize, Debug, Clone)] pub struct Variant { locale: LanguageIdentifier, content: String, diff --git a/l10n-db/src/utils.rs b/l10n-db/src/utils.rs new file mode 100644 index 0000000..b499b9d --- /dev/null +++ b/l10n-db/src/utils.rs @@ -0,0 +1,31 @@ +use std::{fs::File, io::{BufReader, ErrorKind, Read}, path::Path}; +use thiserror::Error; + +#[derive(Debug, Error)] +pub enum ReadError { + #[error("file not found")] + FileNotFound, + + #[error("invalid file format")] + InvalidFormat, + + #[error("unhandled read error")] + Unhandled(ErrorKind), +} + +pub fn read_file(path: &Path) -> Result<Vec<u8>, ReadError> { + let file = File::open(path).map_err(|err| { + match err.kind() { + ErrorKind::NotFound => ReadError::FileNotFound, + _ => ReadError::Unhandled( err.kind()), + } + })?; + read_fh(&file) +} + +pub fn read_fh(file: &File) -> Result<Vec<u8>, ReadError> { + let mut content = Vec::new(); + let mut reader = BufReader::new(file); + reader.read_to_end(&mut content).map_err(|err| ReadError::Unhandled(err.kind()))?; + Ok(content) +} From 52a0d6e3f2cc8626736664ac98f756c6968cd15e Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 22 Feb 2025 18:45:36 -0500 Subject: [PATCH 06/34] Put in more meaningful working text --- Cargo.lock | 1 + l10n-db/i18n/OpenSandbox.toml | 12 ------------ l10n-db/i18n/SaveSettings.toml | 7 +++++++ l10n-db/i18n/Welcome.toml | 7 +++++++ 4 files changed, 15 insertions(+), 12 deletions(-) delete mode 100644 l10n-db/i18n/OpenSandbox.toml create mode 100644 l10n-db/i18n/SaveSettings.toml create mode 100644 l10n-db/i18n/Welcome.toml diff --git a/Cargo.lock b/Cargo.lock index d48d3bd..2f50fae 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2419,6 +2419,7 @@ dependencies = [ "icu_locid", "serde 1.0.218", "tempfile", + "thiserror 2.0.11", "toml", ] diff --git a/l10n-db/i18n/OpenSandbox.toml b/l10n-db/i18n/OpenSandbox.toml deleted file mode 100644 index 81b7d4f..0000000 --- a/l10n-db/i18n/OpenSandbox.toml +++ /dev/null @@ -1,12 +0,0 @@ -key = "OpenSandbox" -description = "A sandbox vault refers to a stock vault that contains test data and allows the user to make edits and run experiments on test data." - -[variants.eo] -locale = "eo" -content = "Malfermi sekurkofro" -modified = "2025-02-22T23:35:29.692299091Z" - -[variants.en] -locale = "en" -content = "Open Sandbox vault" -modified = "2025-02-22T21:35:02.032300406Z" diff --git a/l10n-db/i18n/SaveSettings.toml b/l10n-db/i18n/SaveSettings.toml new file mode 100644 index 0000000..c2a4d1c --- /dev/null +++ b/l10n-db/i18n/SaveSettings.toml @@ -0,0 +1,7 @@ +key = "SaveSettings" +description = "This is a label on a button which will save the settings when clicked" + +[variants.en] +locale = "en" +content = "Save Settings" +modified = "2025-02-22T23:44:18.874218939Z" diff --git a/l10n-db/i18n/Welcome.toml b/l10n-db/i18n/Welcome.toml new file mode 100644 index 0000000..556c993 --- /dev/null +++ b/l10n-db/i18n/Welcome.toml @@ -0,0 +1,7 @@ +key = "Welcome" +description = "This is a welcome content that will be shown on first app opening, before configuration." + +[variants.en] +locale = "en" +content = "Welcome to FitnessTrax" +modified = "2025-02-22T23:43:24.786544124Z" From 44ee6ec8a5ceba050d21cfdbb8e02299e4d92be0 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 22 Feb 2025 19:11:29 -0500 Subject: [PATCH 07/34] Export to json --- Cargo.lock | 5 +++-- l10n-db/Cargo.toml | 1 + l10n-db/output.json | 1 + l10n-db/src/bin/main.rs | 9 ++++++--- l10n-db/src/bundle.rs | 2 +- l10n-db/src/js_format.rs | 22 ++++++++++++++++++++++ l10n-db/src/lib.rs | 3 +++ l10n-db/src/types.rs | 4 ++++ l10n-db/src/utils.rs | 6 ++++++ 9 files changed, 47 insertions(+), 6 deletions(-) create mode 100644 l10n-db/output.json create mode 100644 l10n-db/src/js_format.rs diff --git a/Cargo.lock b/Cargo.lock index 2f50fae..4982051 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2418,6 +2418,7 @@ dependencies = [ "clap", "icu_locid", "serde 1.0.218", + "serde_json", "tempfile", "thiserror 2.0.11", "toml", @@ -3915,9 +3916,9 @@ dependencies = [ [[package]] name = "serde_json" -version = "1.0.136" +version = "1.0.139" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "336a0c23cf42a38d9eaa7cd22c7040d04e1228a19a933890805ffd00a16437d2" +checksum = "44f86c3acccc9c65b153fe1b85a3be07fe5515274ec9f0653b4a0875731c72a6" dependencies = [ "itoa", "memchr", diff --git a/l10n-db/Cargo.toml b/l10n-db/Cargo.toml index e36077b..8016440 100644 --- a/l10n-db/Cargo.toml +++ b/l10n-db/Cargo.toml @@ -8,6 +8,7 @@ chrono = { version = "0.4.39", features = ["serde"] } clap = { version = "4.5.30", features = ["derive"] } icu_locid = { version = "1.5.0", features = ["serde"] } serde = { version = "1.0.218", features = ["derive"] } +serde_json = "1.0.139" tempfile = "3.17.1" thiserror = "2.0.11" toml = "0.8.20" diff --git a/l10n-db/output.json b/l10n-db/output.json new file mode 100644 index 0000000..fe23c55 --- /dev/null +++ b/l10n-db/output.json @@ -0,0 +1 @@ +{"SaveSettings":"Save Settings","Welcome":"Welcome to FitnessTrax"} \ No newline at end of file diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index 1860136..a7880f8 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -7,7 +7,7 @@ use std::{ use clap::{Parser, Subcommand}; use icu_locid::{langid, LanguageIdentifier}; -use l10n_db::{self, read_file, Bundle, Editor, ReadError}; +use l10n_db::{self, export_file, read_file, Bundle, Editor, ReadError}; use serde::Deserialize; #[derive(Parser)] @@ -35,7 +35,7 @@ enum Commands { #[arg(short, long)] format: String, #[arg(short, long)] - locale: String, + locale: Option<String>, }, } @@ -74,7 +74,10 @@ fn main() { println!("{}", key); } } - Some(Commands::Export { .. }) => todo!(), + Some(Commands::Export { format, locale }) => { + let locale = locale.as_ref().map(|l| l.clone().parse::<LanguageIdentifier>().unwrap()).unwrap_or(langid!("en")); + export_file(&bundle, locale, &PathBuf::from("output.json")).unwrap(); + }, None => {} } } diff --git a/l10n-db/src/bundle.rs b/l10n-db/src/bundle.rs index 3144ea9..df1e6b1 100644 --- a/l10n-db/src/bundle.rs +++ b/l10n-db/src/bundle.rs @@ -1,6 +1,6 @@ use std::{collections::HashMap, fs::File, io::Write, path::PathBuf}; -use crate::Message; +use crate::{Message, WriteError}; pub struct Bundle { path: PathBuf, diff --git a/l10n-db/src/js_format.rs b/l10n-db/src/js_format.rs new file mode 100644 index 0000000..c8c5844 --- /dev/null +++ b/l10n-db/src/js_format.rs @@ -0,0 +1,22 @@ +use std::{collections::BTreeMap, fs::File, io::Write, path::Path}; + +use icu_locid::LanguageIdentifier; + +use crate::{Bundle, WriteError}; + +pub fn export_file(bundle: &Bundle, locale: LanguageIdentifier, path: &Path) -> Result<(), WriteError> { + let mut file = File::create(path).unwrap(); + export_fh(bundle, locale, &mut file) +} + +pub fn export_fh(bundle: &Bundle, locale: LanguageIdentifier, fh: &mut File) -> Result<(), WriteError> { + let messages = bundle.message_iter().map(|(key, message)| { + let content = message.variant(&locale).unwrap().content().to_owned(); + (key.to_owned(), content) + }).collect::<Vec<(String, String)>>(); + + let messages: BTreeMap<String, String> = messages.into_iter().collect(); + fh.write(serde_json::to_string(&messages).unwrap().as_bytes()).unwrap(); + + Ok(()) +} diff --git a/l10n-db/src/lib.rs b/l10n-db/src/lib.rs index ab1eec2..a5286f6 100644 --- a/l10n-db/src/lib.rs +++ b/l10n-db/src/lib.rs @@ -4,6 +4,9 @@ pub use bundle::Bundle; mod editor; pub use editor::Editor; +mod js_format; +pub use js_format::{export_file, export_fh}; + mod types; pub use types::{Message, Variant}; diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs index 13b447d..ad5caa0 100644 --- a/l10n-db/src/types.rs +++ b/l10n-db/src/types.rs @@ -36,6 +36,10 @@ impl Message { &self.description } + pub fn variant(&self, locale: &LanguageIdentifier) -> Option<&Variant> { + self.variants.get(locale) + } + pub fn variant_mut(&mut self, locale: LanguageIdentifier) -> &mut Variant { self.variants.entry(locale.clone()).or_insert(Variant { locale, diff --git a/l10n-db/src/utils.rs b/l10n-db/src/utils.rs index b499b9d..82435a0 100644 --- a/l10n-db/src/utils.rs +++ b/l10n-db/src/utils.rs @@ -29,3 +29,9 @@ pub fn read_fh(file: &File) -> Result<Vec<u8>, ReadError> { reader.read_to_end(&mut content).map_err(|err| ReadError::Unhandled(err.kind()))?; Ok(content) } + +#[derive(Debug, Error)] +pub enum WriteError { + #[error("unhandled write error")] + Unhandled(ErrorKind), +} From 0df0ff941967abb89604583f85b678ce97e2c70f Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sun, 23 Feb 2025 21:48:13 -0500 Subject: [PATCH 08/34] Start on an xliff output --- Cargo.lock | 7 +++ l10n-db/Cargo.toml | 1 + l10n-db/output.json | 1 - l10n-db/src/bin/main.rs | 9 +++- l10n-db/src/{js_format.rs => formats/js.rs} | 0 l10n-db/src/formats/mod.rs | 4 ++ l10n-db/src/formats/xliff.rs | 48 +++++++++++++++++++++ l10n-db/src/lib.rs | 18 +------- 8 files changed, 69 insertions(+), 19 deletions(-) delete mode 100644 l10n-db/output.json rename l10n-db/src/{js_format.rs => formats/js.rs} (100%) create mode 100644 l10n-db/src/formats/mod.rs create mode 100644 l10n-db/src/formats/xliff.rs diff --git a/Cargo.lock b/Cargo.lock index 4982051..b4bc05f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2422,6 +2422,7 @@ dependencies = [ "tempfile", "thiserror 2.0.11", "toml", + "xml-rs", ] [[package]] @@ -5435,6 +5436,12 @@ version = "0.5.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1e9df38ee2d2c3c5948ea468a8406ff0db0b29ae1ffde1bcf20ef305bcc95c51" +[[package]] +name = "xml-rs" +version = "0.8.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c5b940ebc25896e71dd073bad2dbaa2abfe97b0a391415e22ad1326d9c54e3c4" + [[package]] name = "yansi-term" version = "0.1.2" diff --git a/l10n-db/Cargo.toml b/l10n-db/Cargo.toml index 8016440..5a1f64a 100644 --- a/l10n-db/Cargo.toml +++ b/l10n-db/Cargo.toml @@ -12,6 +12,7 @@ serde_json = "1.0.139" tempfile = "3.17.1" thiserror = "2.0.11" toml = "0.8.20" +xml-rs = "0.8.25" # [lib] # name = "l10n_db" diff --git a/l10n-db/output.json b/l10n-db/output.json deleted file mode 100644 index fe23c55..0000000 --- a/l10n-db/output.json +++ /dev/null @@ -1 +0,0 @@ -{"SaveSettings":"Save Settings","Welcome":"Welcome to FitnessTrax"} \ No newline at end of file diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index a7880f8..d1b304e 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -7,7 +7,7 @@ use std::{ use clap::{Parser, Subcommand}; use icu_locid::{langid, LanguageIdentifier}; -use l10n_db::{self, export_file, read_file, Bundle, Editor, ReadError}; +use l10n_db::{self, js, read_file, xliff, Bundle, Editor, ReadError}; use serde::Deserialize; #[derive(Parser)] @@ -76,7 +76,12 @@ fn main() { } Some(Commands::Export { format, locale }) => { let locale = locale.as_ref().map(|l| l.clone().parse::<LanguageIdentifier>().unwrap()).unwrap_or(langid!("en")); - export_file(&bundle, locale, &PathBuf::from("output.json")).unwrap(); + + match format.as_ref() { + "js" => js::export_file(&bundle, locale, &PathBuf::from("output.json")).unwrap(), + "xliff" => xliff::export_file(&bundle, locale, &PathBuf::from("output.xliff")).unwrap(), + _ => todo!(), + } }, None => {} } diff --git a/l10n-db/src/js_format.rs b/l10n-db/src/formats/js.rs similarity index 100% rename from l10n-db/src/js_format.rs rename to l10n-db/src/formats/js.rs diff --git a/l10n-db/src/formats/mod.rs b/l10n-db/src/formats/mod.rs new file mode 100644 index 0000000..6fbd8ce --- /dev/null +++ b/l10n-db/src/formats/mod.rs @@ -0,0 +1,4 @@ +pub mod js; +pub mod xliff; + + diff --git a/l10n-db/src/formats/xliff.rs b/l10n-db/src/formats/xliff.rs new file mode 100644 index 0000000..9563e52 --- /dev/null +++ b/l10n-db/src/formats/xliff.rs @@ -0,0 +1,48 @@ +use std::{fs::File, io, path::Path}; + +use icu_locid::LanguageIdentifier; +use xml::{writer::XmlEvent, EmitterConfig, EventWriter}; + +use crate::{Bundle, Message, WriteError}; + +pub fn export_file( + bundle: &Bundle, + locale: LanguageIdentifier, + path: &Path, +) -> Result<(), WriteError> { + let mut file = File::create(path).unwrap(); + export_fh(bundle, locale, &mut file) +} + +pub fn export_fh( + bundle: &Bundle, + locale: LanguageIdentifier, + fh: &mut File, +) -> Result<(), WriteError> { + let output = io::stdout(); + let mut writer = EmitterConfig::new() + .perform_indent(true) + .create_writer(output); + + writer.write(XmlEvent::start_element("xliff")).unwrap(); + writer.write(XmlEvent::start_element("file")).unwrap(); + + for (key, message) in bundle.message_iter() { + write_message(&mut writer, message, &locale); + } + + writer.write(XmlEvent::end_element()).unwrap(); + writer.write(XmlEvent::end_element()).unwrap(); + Ok(()) +} + +fn write_message<T>(writer: &mut EventWriter<T>, message: &Message, locale: &LanguageIdentifier) +where + T: std::io::Write, +{ + writer.write(XmlEvent::start_element("unit")).unwrap(); + writer.write(XmlEvent::start_element("segment")).unwrap(); + writer.write(XmlEvent::characters(message.variant(locale).unwrap().content())).unwrap(); + writer.write(XmlEvent::end_element()).unwrap(); + writer.write(XmlEvent::end_element()).unwrap(); +} diff --git a/l10n-db/src/lib.rs b/l10n-db/src/lib.rs index a5286f6..d826d47 100644 --- a/l10n-db/src/lib.rs +++ b/l10n-db/src/lib.rs @@ -4,8 +4,8 @@ pub use bundle::Bundle; mod editor; pub use editor::Editor; -mod js_format; -pub use js_format::{export_file, export_fh}; +mod formats; +pub use formats::{js, xliff}; mod types; pub use types::{Message, Variant}; @@ -13,17 +13,3 @@ pub use types::{Message, Variant}; mod utils; pub use utils::*; -/* -#[cfg(test)] -mod test { - #[test] - fn it_can_represent_an_untranslated_message() { - todo!() - } - - #[test] - fn it_can_represent_a_partially_translated_message() { - todo!() - } -} -*/ From a8a61cf03f8ae9a623ab2129b056bac625b049a6 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 09:44:44 -0500 Subject: [PATCH 09/34] Fill out the xliff exporter --- l10n-db/i18n/TimeDistance.toml | 7 +++++ l10n-db/src/formats/xliff.rs | 47 +++++++++++++++++++++++++--------- 2 files changed, 42 insertions(+), 12 deletions(-) create mode 100644 l10n-db/i18n/TimeDistance.toml diff --git a/l10n-db/i18n/TimeDistance.toml b/l10n-db/i18n/TimeDistance.toml new file mode 100644 index 0000000..f13ee13 --- /dev/null +++ b/l10n-db/i18n/TimeDistance.toml @@ -0,0 +1,7 @@ +key = "TimeDistance" +description = "A summary of a workout or many workouts that involve a time and a distance" + +[variants.en] +locale = "en" +content = "{distance} of {activity} in {hours, plural, =1 {{hours} hour}, other {{hours} hours} and {minutes, plural, =1 {{minutes} minute} other {{minutes} minutes}}" +modified = "2025-02-24T14:09:17.361641899Z" diff --git a/l10n-db/src/formats/xliff.rs b/l10n-db/src/formats/xliff.rs index 9563e52..075971b 100644 --- a/l10n-db/src/formats/xliff.rs +++ b/l10n-db/src/formats/xliff.rs @@ -19,16 +19,24 @@ pub fn export_fh( locale: LanguageIdentifier, fh: &mut File, ) -> Result<(), WriteError> { - let output = io::stdout(); let mut writer = EmitterConfig::new() .perform_indent(true) - .create_writer(output); + .create_writer(fh); - writer.write(XmlEvent::start_element("xliff")).unwrap(); - writer.write(XmlEvent::start_element("file")).unwrap(); + writer + .write( + XmlEvent::start_element("xliff") + .attr("xmlns", "urn:oasis:names:tc:xliff:document:2.0") + .attr("version", "2.0") + .attr("srcLang", &format!("{}", locale)), + ) + .unwrap(); + writer + .write(XmlEvent::start_element("file").attr("id", "main")) + .unwrap(); for (key, message) in bundle.message_iter() { - write_message(&mut writer, message, &locale); + write_message(&mut writer, key, message, &locale); } writer.write(XmlEvent::end_element()).unwrap(); @@ -36,13 +44,28 @@ pub fn export_fh( Ok(()) } -fn write_message<T>(writer: &mut EventWriter<T>, message: &Message, locale: &LanguageIdentifier) -where +fn write_message<T>( + writer: &mut EventWriter<T>, + key: &str, + message: &Message, + locale: &LanguageIdentifier, +) where T: std::io::Write, { - writer.write(XmlEvent::start_element("unit")).unwrap(); - writer.write(XmlEvent::start_element("segment")).unwrap(); - writer.write(XmlEvent::characters(message.variant(locale).unwrap().content())).unwrap(); - writer.write(XmlEvent::end_element()).unwrap(); - writer.write(XmlEvent::end_element()).unwrap(); + [ + XmlEvent::start_element("unit").attr("id", key).into(), + XmlEvent::start_element("notes").into(), + XmlEvent::start_element("note").into(), + XmlEvent::characters(message.description()).into(), + XmlEvent::end_element().into(), + XmlEvent::end_element().into(), + XmlEvent::start_element("segment").into(), + XmlEvent::start_element("source").into(), + XmlEvent::characters(message.variant(locale).unwrap().content()).into(), + XmlEvent::end_element().into(), + XmlEvent::end_element().into(), + XmlEvent::end_element().into(), + ] + .into_iter() + .for_each(|elem: XmlEvent| writer.write(elem).unwrap()); } From cd5837a4376990f09fcf02a8398cf4fc89171a0c Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 12:35:54 -0500 Subject: [PATCH 10/34] Fix the ICU message in TimeDistance --- l10n-db/i18n/TimeDistance.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/l10n-db/i18n/TimeDistance.toml b/l10n-db/i18n/TimeDistance.toml index f13ee13..2e541f1 100644 --- a/l10n-db/i18n/TimeDistance.toml +++ b/l10n-db/i18n/TimeDistance.toml @@ -3,5 +3,5 @@ description = "A summary of a workout or many workouts that involve a time and a [variants.en] locale = "en" -content = "{distance} of {activity} in {hours, plural, =1 {{hours} hour}, other {{hours} hours} and {minutes, plural, =1 {{minutes} minute} other {{minutes} minutes}}" +content = "{distance} of {activity} in {hours, plural, =1 {{hours} hour} other {{hours} hours}} and {minutes, plural, =1 {{minutes} minute} other {{minutes} minutes}}" modified = "2025-02-24T14:09:17.361641899Z" From 704009b76c77cda0c00ee3fa62fe9ab7032fed4c Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 21:11:12 -0500 Subject: [PATCH 11/34] Import translated xliff data --- l10n-db/i18n/SaveSettings.toml | 15 ++++ l10n-db/i18n/TimeDistance.toml | 15 ++++ l10n-db/i18n/Welcome.toml | 15 ++++ l10n-db/src/bin/main.rs | 10 ++- l10n-db/src/formats/js.rs | 2 +- l10n-db/src/formats/xliff.rs | 144 ++++++++++++++++++++++++++------- 6 files changed, 170 insertions(+), 31 deletions(-) diff --git a/l10n-db/i18n/SaveSettings.toml b/l10n-db/i18n/SaveSettings.toml index c2a4d1c..3f8d37d 100644 --- a/l10n-db/i18n/SaveSettings.toml +++ b/l10n-db/i18n/SaveSettings.toml @@ -5,3 +5,18 @@ description = "This is a label on a button which will save the settings when cli locale = "en" content = "Save Settings" modified = "2025-02-22T23:44:18.874218939Z" + +[variants.eo] +locale = "eo" +content = "Konservi Agordojn" +modified = "2025-02-24T19:32:11.246639077Z" + +[variants.de] +locale = "de" +content = "Einstellungen Speichern" +modified = "2025-02-24T19:33:19.516005843Z" + +[variants.es] +locale = "es" +content = "Guardar Configuraciones" +modified = "2025-02-24T19:33:23.861329923Z" diff --git a/l10n-db/i18n/TimeDistance.toml b/l10n-db/i18n/TimeDistance.toml index 2e541f1..3525af3 100644 --- a/l10n-db/i18n/TimeDistance.toml +++ b/l10n-db/i18n/TimeDistance.toml @@ -5,3 +5,18 @@ description = "A summary of a workout or many workouts that involve a time and a locale = "en" content = "{distance} of {activity} in {hours, plural, =1 {{hours} hour} other {{hours} hours}} and {minutes, plural, =1 {{minutes} minute} other {{minutes} minutes}}" modified = "2025-02-24T14:09:17.361641899Z" + +[variants.eo] +locale = "eo" +content = "{distance} de {activity} en {hours, plural, =1 {{hours} horo} other {{hours} horoj}} {minutes, plural, =1 {{minutes} minuto} other {{minutes} minutoj}}" +modified = "2025-02-24T19:32:11.246943602Z" + +[variants.de] +locale = "de" +content = "{distance} von {activity} in {hours, plural, one {}=1 {{hours} Stunde} other {{hours} Stunden}} und {minutes, plural, one {}=1 {{minutes} Minute} other {{minutes} Minuten}}" +modified = "2025-02-24T19:33:19.516210807Z" + +[variants.es] +locale = "es" +content = "{distance} de {activity} en {hours, plural, one {}=1 {{hours} hora} other {{hours} horas}} y {minutes, plural, one {}=1 {{minutes} minuto} other {{minutes} minutos}}" +modified = "2025-02-24T19:33:23.861604738Z" diff --git a/l10n-db/i18n/Welcome.toml b/l10n-db/i18n/Welcome.toml index 556c993..47cc700 100644 --- a/l10n-db/i18n/Welcome.toml +++ b/l10n-db/i18n/Welcome.toml @@ -1,7 +1,22 @@ key = "Welcome" description = "This is a welcome content that will be shown on first app opening, before configuration." +[variants.eo] +locale = "eo" +content = "Bonvenon al FitnessTrax" +modified = "2025-02-24T19:32:11.246407627Z" + [variants.en] locale = "en" content = "Welcome to FitnessTrax" modified = "2025-02-22T23:43:24.786544124Z" + +[variants.es] +locale = "es" +content = "Bienvenido a FitnessTrax" +modified = "2025-02-24T19:33:23.861143003Z" + +[variants.de] +locale = "de" +content = "Willkommen bei FitnessTrax" +modified = "2025-02-24T19:33:19.515861453Z" diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index d1b304e..f7cbd54 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -7,7 +7,7 @@ use std::{ use clap::{Parser, Subcommand}; use icu_locid::{langid, LanguageIdentifier}; -use l10n_db::{self, js, read_file, xliff, Bundle, Editor, ReadError}; +use l10n_db::{self, js, read_file, xliff::{self, import_file}, Bundle, Editor, ReadError}; use serde::Deserialize; #[derive(Parser)] @@ -30,6 +30,10 @@ enum Commands { ListKeys, // Search the database // Search { }, + Import { + #[arg(short, long)] + file: String, + }, /// Export the database Export { #[arg(short, long)] @@ -74,6 +78,10 @@ fn main() { println!("{}", key); } } + Some(Commands::Import { file }) => { + import_file(&mut bundle, &PathBuf::from(file)).unwrap(); + bundle.save(); + } Some(Commands::Export { format, locale }) => { let locale = locale.as_ref().map(|l| l.clone().parse::<LanguageIdentifier>().unwrap()).unwrap_or(langid!("en")); diff --git a/l10n-db/src/formats/js.rs b/l10n-db/src/formats/js.rs index c8c5844..79d68cb 100644 --- a/l10n-db/src/formats/js.rs +++ b/l10n-db/src/formats/js.rs @@ -16,7 +16,7 @@ pub fn export_fh(bundle: &Bundle, locale: LanguageIdentifier, fh: &mut File) -> }).collect::<Vec<(String, String)>>(); let messages: BTreeMap<String, String> = messages.into_iter().collect(); - fh.write(serde_json::to_string(&messages).unwrap().as_bytes()).unwrap(); + fh.write(serde_json::to_string_pretty(&messages).unwrap().as_bytes()).unwrap(); Ok(()) } diff --git a/l10n-db/src/formats/xliff.rs b/l10n-db/src/formats/xliff.rs index 075971b..cbde7a3 100644 --- a/l10n-db/src/formats/xliff.rs +++ b/l10n-db/src/formats/xliff.rs @@ -1,9 +1,24 @@ -use std::{fs::File, io, path::Path}; +use std::{ + collections::HashMap, + fs::File, + io::{self, BufReader, Read, Write}, + path::Path, +}; -use icu_locid::LanguageIdentifier; -use xml::{writer::XmlEvent, EmitterConfig, EventWriter}; +use chrono::{DateTime, Utc}; +use icu_locid::{langid, LanguageIdentifier}; +use xml::{attribute::OwnedAttribute, reader, writer, EmitterConfig, EventReader, EventWriter}; -use crate::{Bundle, Message, WriteError}; +use crate::{Bundle, Message, ReadError, WriteError}; + +struct PartialMessage { + variants: HashMap<LanguageIdentifier, PartialVariant>, +} + +struct PartialVariant { + content: Option<String>, + modified: Option<DateTime<Utc>>, +} pub fn export_file( bundle: &Bundle, @@ -14,33 +29,93 @@ pub fn export_file( export_fh(bundle, locale, &mut file) } -pub fn export_fh( - bundle: &Bundle, - locale: LanguageIdentifier, - fh: &mut File, -) -> Result<(), WriteError> { - let mut writer = EmitterConfig::new() - .perform_indent(true) - .create_writer(fh); +pub fn export_fh<W>(bundle: &Bundle, locale: LanguageIdentifier, fh: W) -> Result<(), WriteError> +where + W: Write, +{ + let mut writer = EmitterConfig::new().perform_indent(true).create_writer(fh); writer .write( - XmlEvent::start_element("xliff") + writer::XmlEvent::start_element("xliff") .attr("xmlns", "urn:oasis:names:tc:xliff:document:2.0") .attr("version", "2.0") .attr("srcLang", &format!("{}", locale)), ) .unwrap(); writer - .write(XmlEvent::start_element("file").attr("id", "main")) + .write(writer::XmlEvent::start_element("file").attr("id", "main")) .unwrap(); for (key, message) in bundle.message_iter() { write_message(&mut writer, key, message, &locale); } - writer.write(XmlEvent::end_element()).unwrap(); - writer.write(XmlEvent::end_element()).unwrap(); + writer.write(writer::XmlEvent::end_element()).unwrap(); + writer.write(writer::XmlEvent::end_element()).unwrap(); + Ok(()) +} + +pub fn import_file(bundle: &mut Bundle, path: &Path) -> Result<(), ReadError> { + let file = File::open(path).unwrap(); + let file = BufReader::new(file); + + import_reader(bundle, file) +} + +pub fn import_reader<R>(bundle: &mut Bundle, fh: R) -> Result<(), ReadError> +where + R: Read, +{ + let parser = EventReader::new(fh); + + let mut locale: LanguageIdentifier = langid!("en"); + let mut current_key = None; + let mut current_text: Option<String> = None; + let mut in_target = false; + + for event in parser { + match event { + Ok(reader::XmlEvent::StartElement { + name, attributes, .. + }) => match name.local_name.as_ref() { + "xliff" => { + locale = find_attribute(&attributes, "trgLang") + .unwrap() + .parse::<LanguageIdentifier>() + .unwrap(); + } + "unit" => current_key = find_attribute(&attributes, "id"), + "target" => in_target = true, + _ => println!("name: {}", name), + }, + Ok(reader::XmlEvent::EndElement { name }) => match name.local_name.as_ref() { + "unit" => { + if let Some(key) = current_key { + let message = bundle.message(key); + let variant = message.variant_mut(locale.clone()); + if let Some(ref text) = current_text { + variant.set_content(text.clone()); + } + } + current_key = None; + } + "target" => in_target = false, + _ => {} + }, + Ok(reader::XmlEvent::Characters(data)) => { + if in_target { + current_text = Some(data) + } + } + Err(e) => { + eprintln!("error: {e}"); + break; + } + _ => {} + } + } + Ok(()) } @@ -53,19 +128,30 @@ fn write_message<T>( T: std::io::Write, { [ - XmlEvent::start_element("unit").attr("id", key).into(), - XmlEvent::start_element("notes").into(), - XmlEvent::start_element("note").into(), - XmlEvent::characters(message.description()).into(), - XmlEvent::end_element().into(), - XmlEvent::end_element().into(), - XmlEvent::start_element("segment").into(), - XmlEvent::start_element("source").into(), - XmlEvent::characters(message.variant(locale).unwrap().content()).into(), - XmlEvent::end_element().into(), - XmlEvent::end_element().into(), - XmlEvent::end_element().into(), + writer::XmlEvent::start_element("unit") + .attr("id", key) + .into(), + writer::XmlEvent::start_element("notes").into(), + writer::XmlEvent::start_element("note").into(), + writer::XmlEvent::characters(message.description()).into(), + writer::XmlEvent::end_element().into(), + writer::XmlEvent::end_element().into(), + writer::XmlEvent::start_element("segment").into(), + writer::XmlEvent::start_element("source").into(), + writer::XmlEvent::characters(message.variant(locale).unwrap().content()).into(), + writer::XmlEvent::end_element().into(), + writer::XmlEvent::end_element().into(), + writer::XmlEvent::end_element().into(), ] .into_iter() - .for_each(|elem: XmlEvent| writer.write(elem).unwrap()); + .for_each(|elem: writer::XmlEvent| writer.write(elem).unwrap()); +} + +fn find_attribute(attrs: &Vec<OwnedAttribute>, name: &str) -> Option<String> { + for f in attrs { + if name == f.name.local_name { + return Some(f.value.clone()); + } + } + None } From e5b3c7e4e163c421cd4113a5349cdce7c08ca8a8 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 22:18:42 -0500 Subject: [PATCH 12/34] Write a rudimentary report --- l10n-db/i18n/SaveSettings.toml | 20 ++++++------ l10n-db/i18n/TimeDistance.toml | 18 +++++----- l10n-db/i18n/Welcome.toml | 10 +++--- l10n-db/src/bin/main.rs | 60 +++++++++++++++++++++++++++++++--- l10n-db/src/types.rs | 4 +++ 5 files changed, 84 insertions(+), 28 deletions(-) diff --git a/l10n-db/i18n/SaveSettings.toml b/l10n-db/i18n/SaveSettings.toml index 3f8d37d..384705e 100644 --- a/l10n-db/i18n/SaveSettings.toml +++ b/l10n-db/i18n/SaveSettings.toml @@ -1,22 +1,22 @@ key = "SaveSettings" description = "This is a label on a button which will save the settings when clicked" -[variants.en] -locale = "en" -content = "Save Settings" -modified = "2025-02-22T23:44:18.874218939Z" - [variants.eo] locale = "eo" content = "Konservi Agordojn" modified = "2025-02-24T19:32:11.246639077Z" -[variants.de] -locale = "de" -content = "Einstellungen Speichern" -modified = "2025-02-24T19:33:19.516005843Z" - [variants.es] locale = "es" content = "Guardar Configuraciones" modified = "2025-02-24T19:33:23.861329923Z" + +[variants.en] +locale = "en" +content = "Save Settings" +modified = "2025-02-22T23:44:18.874218939Z" + +[variants.de] +locale = "de" +content = "Einstellungen Speichern" +modified = "2025-02-24T19:33:19.516005843Z" diff --git a/l10n-db/i18n/TimeDistance.toml b/l10n-db/i18n/TimeDistance.toml index 3525af3..aa6ffb2 100644 --- a/l10n-db/i18n/TimeDistance.toml +++ b/l10n-db/i18n/TimeDistance.toml @@ -1,22 +1,22 @@ key = "TimeDistance" description = "A summary of a workout or many workouts that involve a time and a distance" -[variants.en] -locale = "en" -content = "{distance} of {activity} in {hours, plural, =1 {{hours} hour} other {{hours} hours}} and {minutes, plural, =1 {{minutes} minute} other {{minutes} minutes}}" -modified = "2025-02-24T14:09:17.361641899Z" +[variants.es] +locale = "es" +content = "{distance} de {activity} en {hours, plural, one {}=1 {{hours} hora} other {{hours} horas}} y {minutes, plural, one {}=1 {{minutes} minuto} other {{minutes} minutos}}" +modified = "2025-02-24T19:33:23.861604738Z" [variants.eo] locale = "eo" content = "{distance} de {activity} en {hours, plural, =1 {{hours} horo} other {{hours} horoj}} {minutes, plural, =1 {{minutes} minuto} other {{minutes} minutoj}}" modified = "2025-02-24T19:32:11.246943602Z" +[variants.en] +locale = "en" +content = "{distance} of {activity} in {hours, plural, =1 {{hours} hour} other {{hours} hours}} and {minutes, plural, =1 {{minutes} minute} other {{minutes} minutes}}" +modified = "2025-02-24T14:09:17.361641899Z" + [variants.de] locale = "de" content = "{distance} von {activity} in {hours, plural, one {}=1 {{hours} Stunde} other {{hours} Stunden}} und {minutes, plural, one {}=1 {{minutes} Minute} other {{minutes} Minuten}}" modified = "2025-02-24T19:33:19.516210807Z" - -[variants.es] -locale = "es" -content = "{distance} de {activity} en {hours, plural, one {}=1 {{hours} hora} other {{hours} horas}} y {minutes, plural, one {}=1 {{minutes} minuto} other {{minutes} minutos}}" -modified = "2025-02-24T19:33:23.861604738Z" diff --git a/l10n-db/i18n/Welcome.toml b/l10n-db/i18n/Welcome.toml index 47cc700..db36ae3 100644 --- a/l10n-db/i18n/Welcome.toml +++ b/l10n-db/i18n/Welcome.toml @@ -1,16 +1,16 @@ key = "Welcome" description = "This is a welcome content that will be shown on first app opening, before configuration." +[variants.en] +locale = "en" +content = "Welcome to FitnessTrax, the privacy-centered fitness tracker" +modified = "2025-02-25T02:12:25.757240004Z" + [variants.eo] locale = "eo" content = "Bonvenon al FitnessTrax" modified = "2025-02-24T19:32:11.246407627Z" -[variants.en] -locale = "en" -content = "Welcome to FitnessTrax" -modified = "2025-02-22T23:43:24.786544124Z" - [variants.es] locale = "es" content = "Bienvenido a FitnessTrax" diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index f7cbd54..56a8f0a 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -7,7 +7,11 @@ use std::{ use clap::{Parser, Subcommand}; use icu_locid::{langid, LanguageIdentifier}; -use l10n_db::{self, js, read_file, xliff::{self, import_file}, Bundle, Editor, ReadError}; +use l10n_db::{ + self, js, read_file, + xliff::{self, import_file}, + Bundle, Editor, ReadError, +}; use serde::Deserialize; #[derive(Parser)] @@ -41,6 +45,7 @@ enum Commands { #[arg(short, long)] locale: Option<String>, }, + Report, } #[derive(Debug, Deserialize)] @@ -56,6 +61,44 @@ fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor bundle.save(); } +#[derive(Clone, Debug, Default)] +struct Report { + keys: Vec<String>, + source_deleted: Vec<String>, + out_of_date: Vec<String>, +} + +fn generate_report( + bundle: &Bundle, + base_locale: &LanguageIdentifier, + locales: Vec<LanguageIdentifier>, +) -> Report { + let mut report: Report = Default::default(); + let locales: Vec<LanguageIdentifier> = + locales.into_iter().filter(|a| a != base_locale).collect(); + for (key, message) in bundle.message_iter() { + match message.variant(base_locale) { + Some(ref base_variant) => { + for locale in locales.iter() { + match message.variant(locale) { + Some(v) if v.modified() < base_variant.modified() => { + report.out_of_date.push(key.to_owned()) + } + Some(_) => {} + None => report.out_of_date.push(key.to_owned()), + } + } + } + None => { + report.source_deleted.push(key.clone()); + } + } + let base_variant = message.variant(base_locale).clone(); + } + + report +} + fn main() { let editor = std::env::var("EDITOR").expect("Set EDITOR to the path to your favorite editor"); @@ -83,14 +126,23 @@ fn main() { bundle.save(); } Some(Commands::Export { format, locale }) => { - let locale = locale.as_ref().map(|l| l.clone().parse::<LanguageIdentifier>().unwrap()).unwrap_or(langid!("en")); + let locale = locale + .as_ref() + .map(|l| l.clone().parse::<LanguageIdentifier>().unwrap()) + .unwrap_or(langid!("en")); match format.as_ref() { "js" => js::export_file(&bundle, locale, &PathBuf::from("output.json")).unwrap(), - "xliff" => xliff::export_file(&bundle, locale, &PathBuf::from("output.xliff")).unwrap(), + "xliff" => { + xliff::export_file(&bundle, locale, &PathBuf::from("output.xliff")).unwrap() + } _ => todo!(), } - }, + } + Some(Commands::Report) => { + let report = generate_report(&bundle, &config.base_locale, config.locales); + println!("{:?}", report); + } None => {} } } diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs index ad5caa0..5b8c9ee 100644 --- a/l10n-db/src/types.rs +++ b/l10n-db/src/types.rs @@ -65,6 +65,10 @@ impl Variant { self.content = content; self.modified = Utc::now(); } + + pub fn modified(&self) -> DateTime<Utc> { + self.modified + } } /* From 76de75210f027e9f17a7abf86936f0d7c34faff1 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 22:50:06 -0500 Subject: [PATCH 13/34] Improve the report format --- l10n-db/src/bin/main.rs | 36 ++++++++++++++---------------------- l10n-db/src/types.rs | 27 ++++++++++++++++++++++++++- 2 files changed, 40 insertions(+), 23 deletions(-) diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index 56a8f0a..9df9dde 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -1,7 +1,5 @@ use std::{ - io::{BufReader, Read, Write}, - path::PathBuf, - process::Command, + fmt, io::{BufReader, Read, Write}, path::PathBuf, process::Command }; use clap::{Parser, Subcommand}; @@ -68,32 +66,26 @@ struct Report { out_of_date: Vec<String>, } +impl fmt::Display for Report { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + write!(f, "Out of date messages\n")?; + for key in self.out_of_date.iter() { + write!(f, "\t{}\n", key)?; + } + Ok(()) + } +} + fn generate_report( bundle: &Bundle, base_locale: &LanguageIdentifier, locales: Vec<LanguageIdentifier>, ) -> Report { let mut report: Report = Default::default(); - let locales: Vec<LanguageIdentifier> = - locales.into_iter().filter(|a| a != base_locale).collect(); for (key, message) in bundle.message_iter() { - match message.variant(base_locale) { - Some(ref base_variant) => { - for locale in locales.iter() { - match message.variant(locale) { - Some(v) if v.modified() < base_variant.modified() => { - report.out_of_date.push(key.to_owned()) - } - Some(_) => {} - None => report.out_of_date.push(key.to_owned()), - } - } - } - None => { - report.source_deleted.push(key.clone()); - } + if message.variants_out_of_date(base_locale).len() > 0 { + report.out_of_date.push(key.to_owned()) } - let base_variant = message.variant(base_locale).clone(); } report @@ -141,7 +133,7 @@ fn main() { } Some(Commands::Report) => { let report = generate_report(&bundle, &config.base_locale, config.locales); - println!("{:?}", report); + println!("{}", report); } None => {} } diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs index 5b8c9ee..7b878d8 100644 --- a/l10n-db/src/types.rs +++ b/l10n-db/src/types.rs @@ -1,4 +1,8 @@ -use std::{collections::HashMap, io::{BufReader, Read}, path::Path}; +use std::{ + collections::HashMap, + io::{BufReader, Read}, + path::Path, +}; use chrono::{DateTime, Utc}; use icu_locid::LanguageIdentifier; @@ -47,6 +51,27 @@ impl Message { modified: Utc::now(), }) } + + pub fn variants_out_of_date( + &self, + base_locale: &LanguageIdentifier, + ) -> Vec<LanguageIdentifier> { + match self + .variants + .get(base_locale) + .map(|variant| variant.modified()) + { + Some(base_date) => self + .variants + .iter() + .filter(|(_, value)| base_date > value.modified()) + .map(|(locale, _)| locale.clone()) + .collect(), + None => vec![], + } + } + + // pub fn missing_variants(&self, locals: Vec<LanguageIdentifier>) -> Vec<LanguageIdentifier> {} } #[derive(Deserialize, Serialize, Debug, Clone)] From a07ecae04a20e13474ee2c4fbd0b050703910751 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 22:54:39 -0500 Subject: [PATCH 14/34] Mvoe teh message and variant types into the bundle --- l10n-db/src/bundle.rs | 95 ++++++++++++++++++++++++++++++++++- l10n-db/src/editor.rs | 2 +- l10n-db/src/lib.rs | 6 +-- l10n-db/src/types.rs | 112 ------------------------------------------ 4 files changed, 95 insertions(+), 120 deletions(-) delete mode 100644 l10n-db/src/types.rs diff --git a/l10n-db/src/bundle.rs b/l10n-db/src/bundle.rs index df1e6b1..72af3e3 100644 --- a/l10n-db/src/bundle.rs +++ b/l10n-db/src/bundle.rs @@ -1,6 +1,8 @@ -use std::{collections::HashMap, fs::File, io::Write, path::PathBuf}; +use std::{collections::HashMap, fs::File, io::{BufReader, Read, Write}, path::{Path, PathBuf}}; -use crate::{Message, WriteError}; +use chrono::{DateTime, Utc}; +use icu_locid::LanguageIdentifier; +use serde::{Deserialize, Serialize}; pub struct Bundle { path: PathBuf, @@ -48,5 +50,94 @@ fn save_file(path: &PathBuf, s: &[u8]) { f.write(s).unwrap(); } +#[derive(Deserialize, Serialize, Debug)] +pub struct Message { + key: String, + description: String, + variants: HashMap<LanguageIdentifier, Variant>, +} + +impl Message { + pub fn new(key: String) -> Self { + Self { + key, + description: "".to_owned(), + variants: HashMap::new(), + } + } + + pub fn from_file(path: &Path) -> Message { + let file = std::fs::File::open(path).unwrap(); + let mut content = Vec::new(); + let mut reader = BufReader::new(file); + let _ = reader.read_to_end(&mut content); + toml::from_str(&String::from_utf8(content).unwrap()).unwrap() + } + + pub fn set_description(&mut self, desc: String) { + self.description = desc; + } + + pub fn description(&self) -> &str { + &self.description + } + + pub fn variant(&self, locale: &LanguageIdentifier) -> Option<&Variant> { + self.variants.get(locale) + } + + pub fn variant_mut(&mut self, locale: LanguageIdentifier) -> &mut Variant { + self.variants.entry(locale.clone()).or_insert(Variant { + locale, + content: "".to_owned(), + modified: Utc::now(), + }) + } + + pub fn variants_out_of_date( + &self, + base_locale: &LanguageIdentifier, + ) -> Vec<LanguageIdentifier> { + match self + .variants + .get(base_locale) + .map(|variant| variant.modified()) + { + Some(base_date) => self + .variants + .iter() + .filter(|(_, value)| base_date > value.modified()) + .map(|(locale, _)| locale.clone()) + .collect(), + None => vec![], + } + } + + // pub fn missing_variants(&self, locals: Vec<LanguageIdentifier>) -> Vec<LanguageIdentifier> {} +} + +#[derive(Deserialize, Serialize, Debug, Clone)] +pub struct Variant { + locale: LanguageIdentifier, + content: String, + modified: DateTime<Utc>, +} + +impl Variant { + pub fn content(&self) -> &str { + &self.content + } + + pub fn set_content(&mut self, content: String) { + self.content = content; + self.modified = Utc::now(); + } + + pub fn modified(&self) -> DateTime<Utc> { + self.modified + } +} + + #[cfg(test)] mod test {} diff --git a/l10n-db/src/editor.rs b/l10n-db/src/editor.rs index c51d580..039ea9e 100644 --- a/l10n-db/src/editor.rs +++ b/l10n-db/src/editor.rs @@ -3,7 +3,7 @@ use std::{io::{BufReader, Read, Write}, path::Path, process::Command}; use icu_locid::{langid, LanguageIdentifier}; use serde::{Deserialize, Serialize}; -use crate::{read_fh, Message, Variant}; +use crate::{read_fh, Message}; #[derive(Serialize, Deserialize, Debug, Clone)] diff --git a/l10n-db/src/lib.rs b/l10n-db/src/lib.rs index d826d47..2268a86 100644 --- a/l10n-db/src/lib.rs +++ b/l10n-db/src/lib.rs @@ -1,5 +1,5 @@ mod bundle; -pub use bundle::Bundle; +pub use bundle::{Bundle, Message, Variant}; mod editor; pub use editor::Editor; @@ -7,9 +7,5 @@ pub use editor::Editor; mod formats; pub use formats::{js, xliff}; -mod types; -pub use types::{Message, Variant}; - mod utils; pub use utils::*; - diff --git a/l10n-db/src/types.rs b/l10n-db/src/types.rs deleted file mode 100644 index 7b878d8..0000000 --- a/l10n-db/src/types.rs +++ /dev/null @@ -1,112 +0,0 @@ -use std::{ - collections::HashMap, - io::{BufReader, Read}, - path::Path, -}; - -use chrono::{DateTime, Utc}; -use icu_locid::LanguageIdentifier; -use serde::{de, Deserialize, Deserializer, Serialize, Serializer}; - -#[derive(Deserialize, Serialize, Debug)] -pub struct Message { - key: String, - description: String, - variants: HashMap<LanguageIdentifier, Variant>, -} - -impl Message { - pub fn new(key: String) -> Self { - Self { - key, - description: "".to_owned(), - variants: HashMap::new(), - } - } - - pub fn from_file(path: &Path) -> Message { - let file = std::fs::File::open(path).unwrap(); - let mut content = Vec::new(); - let mut reader = BufReader::new(file); - let _ = reader.read_to_end(&mut content); - toml::from_str(&String::from_utf8(content).unwrap()).unwrap() - } - - pub fn set_description(&mut self, desc: String) { - self.description = desc; - } - - pub fn description(&self) -> &str { - &self.description - } - - pub fn variant(&self, locale: &LanguageIdentifier) -> Option<&Variant> { - self.variants.get(locale) - } - - pub fn variant_mut(&mut self, locale: LanguageIdentifier) -> &mut Variant { - self.variants.entry(locale.clone()).or_insert(Variant { - locale, - content: "".to_owned(), - modified: Utc::now(), - }) - } - - pub fn variants_out_of_date( - &self, - base_locale: &LanguageIdentifier, - ) -> Vec<LanguageIdentifier> { - match self - .variants - .get(base_locale) - .map(|variant| variant.modified()) - { - Some(base_date) => self - .variants - .iter() - .filter(|(_, value)| base_date > value.modified()) - .map(|(locale, _)| locale.clone()) - .collect(), - None => vec![], - } - } - - // pub fn missing_variants(&self, locals: Vec<LanguageIdentifier>) -> Vec<LanguageIdentifier> {} -} - -#[derive(Deserialize, Serialize, Debug, Clone)] -pub struct Variant { - locale: LanguageIdentifier, - content: String, - modified: DateTime<Utc>, -} - -impl Variant { - pub fn content(&self) -> &str { - &self.content - } - - pub fn set_content(&mut self, content: String) { - self.content = content; - self.modified = Utc::now(); - } - - pub fn modified(&self) -> DateTime<Utc> { - self.modified - } -} - -/* -fn time_to_number<S>(time: &DateTime, s: S) -> Result<S::Ok, S::Error> -where S: Serializer { - let seconds: u64 = time.as_secs(); - s.serialize_u64(seconds) -} - -fn number_to_time<'de, D>(d: D) -> Result<DateTime, D::Error> -where D: Deserializer<'de> { - let buf = String::deserialize(d)?; - let num = buf.parse::<u64>().unwrap(); - Ok(DateTime::try_from(num).unwrap()) -} -*/ From 254a2aefd78df897638379b9d77dc713f5423346 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 23:05:18 -0500 Subject: [PATCH 15/34] Remove remaining warnings --- l10n-db/src/bin/main.rs | 47 +++++++++++++++++++----------------- l10n-db/src/bundle.rs | 2 +- l10n-db/src/editor.rs | 21 +++------------- l10n-db/src/formats/js.rs | 2 +- l10n-db/src/formats/xliff.rs | 17 +++---------- 5 files changed, 34 insertions(+), 55 deletions(-) diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/main.rs index 9df9dde..d02b30e 100644 --- a/l10n-db/src/bin/main.rs +++ b/l10n-db/src/bin/main.rs @@ -1,6 +1,4 @@ -use std::{ - fmt, io::{BufReader, Read, Write}, path::PathBuf, process::Command -}; +use std::{fmt, path::PathBuf}; use clap::{Parser, Subcommand}; @@ -24,9 +22,9 @@ enum Commands { /// Edit, potentially creating, a key EditKey { #[arg(short, long)] - name: String, + key: String, #[arg(short, long)] - locale: String, + locale: Option<String>, }, /// List al keys in the database ListKeys, @@ -41,6 +39,8 @@ enum Commands { #[arg(short, long)] format: String, #[arg(short, long)] + file: String, + #[arg(short, long)] locale: Option<String>, }, Report, @@ -50,7 +50,6 @@ enum Commands { struct Config { db_path: PathBuf, base_locale: LanguageIdentifier, - locales: Vec<LanguageIdentifier>, } fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor: &str) { @@ -62,28 +61,26 @@ fn edit_key(bundle: &mut Bundle, key: String, locale: LanguageIdentifier, editor #[derive(Clone, Debug, Default)] struct Report { keys: Vec<String>, - source_deleted: Vec<String>, + // source_deleted: Vec<String>, out_of_date: Vec<String>, } impl fmt::Display for Report { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - write!(f, "Out of date messages\n")?; + writeln!(f, "{} messages in bundle", self.keys.len())?; + writeln!(f, "Out of date messages")?; for key in self.out_of_date.iter() { - write!(f, "\t{}\n", key)?; + writeln!(f, "\t{}", key)?; } Ok(()) } } -fn generate_report( - bundle: &Bundle, - base_locale: &LanguageIdentifier, - locales: Vec<LanguageIdentifier>, -) -> Report { +fn generate_report(bundle: &Bundle, base_locale: &LanguageIdentifier) -> Report { let mut report: Report = Default::default(); for (key, message) in bundle.message_iter() { - if message.variants_out_of_date(base_locale).len() > 0 { + report.keys.push(key.to_owned()); + if !message.variants_out_of_date(base_locale).is_empty() { report.out_of_date.push(key.to_owned()) } } @@ -104,9 +101,11 @@ fn main() { let mut bundle = Bundle::load_from_disk(PathBuf::from(&config.db_path)); match &cli.command { - Some(Commands::EditKey { name, locale }) => { - let identifier = locale.parse::<LanguageIdentifier>().unwrap(); - edit_key(&mut bundle, name.to_owned(), identifier, &editor) + Some(Commands::EditKey { key, locale }) => { + let identifier = locale.as_ref() + .map(|l| l.parse::<LanguageIdentifier>().unwrap()) + .unwrap_or(config.base_locale); + edit_key(&mut bundle, key.to_owned(), identifier, &editor) } Some(Commands::ListKeys) => { for (key, _) in bundle.message_iter() { @@ -117,22 +116,26 @@ fn main() { import_file(&mut bundle, &PathBuf::from(file)).unwrap(); bundle.save(); } - Some(Commands::Export { format, locale }) => { + Some(Commands::Export { + format, + file, + locale, + }) => { let locale = locale .as_ref() .map(|l| l.clone().parse::<LanguageIdentifier>().unwrap()) .unwrap_or(langid!("en")); match format.as_ref() { - "js" => js::export_file(&bundle, locale, &PathBuf::from("output.json")).unwrap(), + "js" => js::export_file(&bundle, locale, &PathBuf::from(file)).unwrap(), "xliff" => { - xliff::export_file(&bundle, locale, &PathBuf::from("output.xliff")).unwrap() + xliff::export_file(&bundle, locale, &PathBuf::from(file)).unwrap() } _ => todo!(), } } Some(Commands::Report) => { - let report = generate_report(&bundle, &config.base_locale, config.locales); + let report = generate_report(&bundle, &config.base_locale); println!("{}", report); } None => {} diff --git a/l10n-db/src/bundle.rs b/l10n-db/src/bundle.rs index 72af3e3..1b55392 100644 --- a/l10n-db/src/bundle.rs +++ b/l10n-db/src/bundle.rs @@ -47,7 +47,7 @@ impl Bundle { fn save_file(path: &PathBuf, s: &[u8]) { let mut f = File::create(path).unwrap(); - f.write(s).unwrap(); + let _ = f.write(s).unwrap(); } #[derive(Deserialize, Serialize, Debug)] diff --git a/l10n-db/src/editor.rs b/l10n-db/src/editor.rs index 039ea9e..b6d9a9f 100644 --- a/l10n-db/src/editor.rs +++ b/l10n-db/src/editor.rs @@ -1,11 +1,10 @@ -use std::{io::{BufReader, Read, Write}, path::Path, process::Command}; +use std::{io::Write, process::Command}; use icu_locid::{langid, LanguageIdentifier}; use serde::{Deserialize, Serialize}; use crate::{read_fh, Message}; - #[derive(Serialize, Deserialize, Debug, Clone)] struct EditorMessage { description: String, @@ -13,19 +12,7 @@ struct EditorMessage { content: String, } -/* -impl EditorMessage { - fn from_variant(description: String, variant: &Variant) -> Self { - Self { - description, - content: variant.content().to_string() - } - } -} -*/ - -pub struct Editor { -} +pub struct Editor {} impl Editor { pub fn edit(msg: &mut Message, locale: LanguageIdentifier, editor: &str) { @@ -47,10 +34,10 @@ impl Editor { let file = file.reopen().unwrap(); let content = read_fh(&file).unwrap(); - let new_variant: EditorMessage = toml::from_str(&String::from_utf8(content).unwrap()).unwrap(); + let new_variant: EditorMessage = + toml::from_str(&String::from_utf8(content).unwrap()).unwrap(); variant.set_content(new_variant.content); msg.set_description(new_variant.description); } } - diff --git a/l10n-db/src/formats/js.rs b/l10n-db/src/formats/js.rs index 79d68cb..6aa94b2 100644 --- a/l10n-db/src/formats/js.rs +++ b/l10n-db/src/formats/js.rs @@ -16,7 +16,7 @@ pub fn export_fh(bundle: &Bundle, locale: LanguageIdentifier, fh: &mut File) -> }).collect::<Vec<(String, String)>>(); let messages: BTreeMap<String, String> = messages.into_iter().collect(); - fh.write(serde_json::to_string_pretty(&messages).unwrap().as_bytes()).unwrap(); + let _ = fh.write(serde_json::to_string_pretty(&messages).unwrap().as_bytes()).unwrap(); Ok(()) } diff --git a/l10n-db/src/formats/xliff.rs b/l10n-db/src/formats/xliff.rs index cbde7a3..d05ecaa 100644 --- a/l10n-db/src/formats/xliff.rs +++ b/l10n-db/src/formats/xliff.rs @@ -1,25 +1,14 @@ use std::{ - collections::HashMap, fs::File, - io::{self, BufReader, Read, Write}, + io::{BufReader, Read, Write}, path::Path, }; -use chrono::{DateTime, Utc}; use icu_locid::{langid, LanguageIdentifier}; use xml::{attribute::OwnedAttribute, reader, writer, EmitterConfig, EventReader, EventWriter}; use crate::{Bundle, Message, ReadError, WriteError}; -struct PartialMessage { - variants: HashMap<LanguageIdentifier, PartialVariant>, -} - -struct PartialVariant { - content: Option<String>, - modified: Option<DateTime<Utc>>, -} - pub fn export_file( bundle: &Bundle, locale: LanguageIdentifier, @@ -133,12 +122,12 @@ fn write_message<T>( .into(), writer::XmlEvent::start_element("notes").into(), writer::XmlEvent::start_element("note").into(), - writer::XmlEvent::characters(message.description()).into(), + writer::XmlEvent::characters(message.description()), writer::XmlEvent::end_element().into(), writer::XmlEvent::end_element().into(), writer::XmlEvent::start_element("segment").into(), writer::XmlEvent::start_element("source").into(), - writer::XmlEvent::characters(message.variant(locale).unwrap().content()).into(), + writer::XmlEvent::characters(message.variant(locale).unwrap().content()), writer::XmlEvent::end_element().into(), writer::XmlEvent::end_element().into(), writer::XmlEvent::end_element().into(), From afa846f7e0efc4e2ff20fe48d969f776425d48c8 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 24 Feb 2025 23:37:59 -0500 Subject: [PATCH 16/34] Add a crane build for l10n-db --- Cargo.nix | 2044 +++++------------------ crate-hashes.json | 59 +- flake.lock | 24 +- flake.nix | 13 +- l10n-db/src/bin/{main.rs => l10n-db.rs} | 0 5 files changed, 449 insertions(+), 1691 deletions(-) rename l10n-db/src/bin/{main.rs => l10n-db.rs} (100%) diff --git a/Cargo.nix b/Cargo.nix index 5e3a985..5af2594 100644 --- a/Cargo.nix +++ b/Cargo.nix @@ -225,6 +225,16 @@ rec { # File a bug if you depend on any for non-debug work! debug = internal.debugCrate { inherit packageId; }; }; + "l10n-db" = rec { + packageId = "l10n-db"; + build = internal.buildRustCrateWithFeatures { + packageId = "l10n-db"; + }; + + # Debug support which might change between releases. + # File a bug if you depend on any for non-debug work! + debug = internal.debugCrate { inherit packageId; }; + }; "lights-core" = rec { packageId = "lights-core"; build = internal.buildRustCrateWithFeatures { @@ -295,6 +305,16 @@ rec { # File a bug if you depend on any for non-debug work! debug = internal.debugCrate { inherit packageId; }; }; + "server" = rec { + packageId = "server"; + build = internal.buildRustCrateWithFeatures { + packageId = "server"; + }; + + # Debug support which might change between releases. + # File a bug if you depend on any for non-debug work! + debug = internal.debugCrate { inherit packageId; }; + }; "sgf" = rec { packageId = "sgf"; build = internal.buildRustCrateWithFeatures { @@ -331,16 +351,6 @@ rec { packageId = "tree"; }; - # Debug support which might change between releases. - # File a bug if you depend on any for non-debug work! - debug = internal.debugCrate { inherit packageId; }; - }; - "visions" = rec { - packageId = "visions"; - build = internal.buildRustCrateWithFeatures { - packageId = "visions"; - }; - # Debug support which might change between releases. # File a bug if you depend on any for non-debug work! debug = internal.debugCrate { inherit packageId; }; @@ -448,49 +458,6 @@ rec { }; resolvedDefaultFeatures = [ "default" "std" ]; }; - "ahash" = rec { - crateName = "ahash"; - version = "0.8.11"; - edition = "2018"; - sha256 = "04chdfkls5xmhp1d48gnjsmglbqibizs3bpbj6rsj604m10si7g8"; - authors = [ - "Tom Kaitchuck <Tom.Kaitchuck@gmail.com>" - ]; - dependencies = [ - { - name = "cfg-if"; - packageId = "cfg-if"; - } - { - name = "once_cell"; - packageId = "once_cell"; - usesDefaultFeatures = false; - target = { target, features }: (!(("arm" == target."arch") && ("none" == target."os"))); - features = [ "alloc" ]; - } - { - name = "zerocopy"; - packageId = "zerocopy"; - usesDefaultFeatures = false; - features = [ "simd" ]; - } - ]; - buildDependencies = [ - { - name = "version_check"; - packageId = "version_check 0.9.5"; - } - ]; - features = { - "atomic-polyfill" = [ "dep:atomic-polyfill" "once_cell/atomic-polyfill" ]; - "compile-time-rng" = [ "const-random" ]; - "const-random" = [ "dep:const-random" ]; - "default" = [ "std" "runtime-rng" ]; - "getrandom" = [ "dep:getrandom" ]; - "runtime-rng" = [ "getrandom" ]; - "serde" = [ "dep:serde" ]; - }; - }; "aho-corasick" = rec { crateName = "aho-corasick"; version = "1.1.3"; @@ -721,34 +688,6 @@ rec { }; resolvedDefaultFeatures = [ "default" "std" ]; }; - "assert-json-diff" = rec { - crateName = "assert-json-diff"; - version = "2.0.2"; - edition = "2018"; - sha256 = "04mg3w0rh3schpla51l18362hsirl23q93aisws2irrj32wg5r27"; - libName = "assert_json_diff"; - authors = [ - "David Pedersen <david.pdrsn@gmail.com>" - ]; - dependencies = [ - { - name = "serde"; - packageId = "serde 1.0.217"; - } - { - name = "serde_json"; - packageId = "serde_json"; - } - ]; - devDependencies = [ - { - name = "serde"; - packageId = "serde 1.0.217"; - features = [ "derive" ]; - } - ]; - - }; "async-channel 1.9.0" = rec { crateName = "async-channel"; version = "1.9.0"; @@ -1267,7 +1206,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -1301,17 +1240,6 @@ rec { } ]; - }; - "auto-future" = rec { - crateName = "auto-future"; - version = "1.0.0"; - edition = "2021"; - sha256 = "0wykbakzh227vz6frx9p48zsq0wpswgmb7v3917m53m7gr2pw7iw"; - libName = "auto_future"; - authors = [ - "Joseph Lenton <josephlenton@gmail.com>" - ]; - }; "autocfg 0.1.8" = rec { crateName = "autocfg"; @@ -1339,429 +1267,6 @@ rec { ]; }; - "axum" = rec { - crateName = "axum"; - version = "0.7.9"; - edition = "2021"; - sha256 = "07z7wqczi9i8xb4460rvn39p4wjqwr32hx907crd1vwb2fy8ijpd"; - dependencies = [ - { - name = "async-trait"; - packageId = "async-trait"; - } - { - name = "axum-core"; - packageId = "axum-core"; - } - { - name = "axum-macros"; - packageId = "axum-macros"; - optional = true; - } - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - features = [ "alloc" ]; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - { - name = "http-body"; - packageId = "http-body 1.0.1"; - } - { - name = "http-body-util"; - packageId = "http-body-util"; - } - { - name = "hyper"; - packageId = "hyper 1.5.2"; - optional = true; - } - { - name = "hyper-util"; - packageId = "hyper-util"; - optional = true; - features = [ "tokio" "server" "service" ]; - } - { - name = "itoa"; - packageId = "itoa"; - } - { - name = "matchit"; - packageId = "matchit"; - } - { - name = "memchr"; - packageId = "memchr"; - } - { - name = "mime"; - packageId = "mime 0.3.17"; - } - { - name = "percent-encoding"; - packageId = "percent-encoding 2.3.1"; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - } - { - name = "rustversion"; - packageId = "rustversion"; - } - { - name = "serde"; - packageId = "serde 1.0.217"; - } - { - name = "serde_json"; - packageId = "serde_json"; - optional = true; - features = [ "raw_value" ]; - } - { - name = "serde_path_to_error"; - packageId = "serde_path_to_error"; - optional = true; - } - { - name = "serde_urlencoded"; - packageId = "serde_urlencoded"; - optional = true; - } - { - name = "sync_wrapper"; - packageId = "sync_wrapper 1.0.2"; - } - { - name = "tokio"; - packageId = "tokio"; - rename = "tokio"; - optional = true; - features = [ "time" ]; - } - { - name = "tower"; - packageId = "tower"; - usesDefaultFeatures = false; - features = [ "util" ]; - } - { - name = "tower-layer"; - packageId = "tower-layer"; - } - { - name = "tower-service"; - packageId = "tower-service"; - } - { - name = "tracing"; - packageId = "tracing"; - optional = true; - usesDefaultFeatures = false; - } - ]; - devDependencies = [ - { - name = "axum-macros"; - packageId = "axum-macros"; - features = [ "__private" ]; - } - { - name = "serde"; - packageId = "serde 1.0.217"; - features = [ "derive" ]; - } - { - name = "serde_json"; - packageId = "serde_json"; - features = [ "raw_value" ]; - } - { - name = "tokio"; - packageId = "tokio"; - rename = "tokio"; - features = [ "macros" "rt" "rt-multi-thread" "net" "test-util" ]; - } - { - name = "tower"; - packageId = "tower"; - rename = "tower"; - features = [ "util" "timeout" "limit" "load-shed" "steer" "filter" ]; - } - { - name = "tracing"; - packageId = "tracing"; - } - ]; - features = { - "__private_docs" = [ "axum-core/__private_docs" "tower/full" "dep:tower-http" ]; - "default" = [ "form" "http1" "json" "matched-path" "original-uri" "query" "tokio" "tower-log" "tracing" ]; - "form" = [ "dep:serde_urlencoded" ]; - "http1" = [ "dep:hyper" "hyper?/http1" "hyper-util?/http1" ]; - "http2" = [ "dep:hyper" "hyper?/http2" "hyper-util?/http2" ]; - "json" = [ "dep:serde_json" "dep:serde_path_to_error" ]; - "macros" = [ "dep:axum-macros" ]; - "multipart" = [ "dep:multer" ]; - "query" = [ "dep:serde_urlencoded" ]; - "tokio" = [ "dep:hyper-util" "dep:tokio" "tokio/net" "tokio/rt" "tower/make" "tokio/macros" ]; - "tower-log" = [ "tower/log" ]; - "tracing" = [ "dep:tracing" "axum-core/tracing" ]; - "ws" = [ "dep:hyper" "tokio" "dep:tokio-tungstenite" "dep:sha1" "dep:base64" ]; - }; - resolvedDefaultFeatures = [ "default" "form" "http1" "json" "macros" "matched-path" "original-uri" "query" "tokio" "tower-log" "tracing" ]; - }; - "axum-core" = rec { - crateName = "axum-core"; - version = "0.4.5"; - edition = "2021"; - sha256 = "16b1496c4gm387q20hkv5ic3k5bd6xmnvk50kwsy6ymr8rhvvwh9"; - libName = "axum_core"; - dependencies = [ - { - name = "async-trait"; - packageId = "async-trait"; - } - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - features = [ "alloc" ]; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - { - name = "http-body"; - packageId = "http-body 1.0.1"; - } - { - name = "http-body-util"; - packageId = "http-body-util"; - } - { - name = "mime"; - packageId = "mime 0.3.17"; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - } - { - name = "rustversion"; - packageId = "rustversion"; - } - { - name = "sync_wrapper"; - packageId = "sync_wrapper 1.0.2"; - } - { - name = "tower-layer"; - packageId = "tower-layer"; - } - { - name = "tower-service"; - packageId = "tower-service"; - } - { - name = "tracing"; - packageId = "tracing"; - optional = true; - usesDefaultFeatures = false; - } - ]; - devDependencies = [ - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - features = [ "alloc" ]; - } - ]; - features = { - "__private_docs" = [ "dep:tower-http" ]; - "tracing" = [ "dep:tracing" ]; - }; - resolvedDefaultFeatures = [ "tracing" ]; - }; - "axum-macros" = rec { - crateName = "axum-macros"; - version = "0.4.2"; - edition = "2021"; - sha256 = "1klv77c889jm05bzayaaiinalarhvh2crc2w4nvp3l581xaj7lap"; - procMacro = true; - libName = "axum_macros"; - dependencies = [ - { - name = "proc-macro2"; - packageId = "proc-macro2"; - } - { - name = "quote"; - packageId = "quote"; - } - { - name = "syn"; - packageId = "syn 2.0.96"; - features = [ "full" "parsing" "extra-traits" ]; - } - ]; - devDependencies = [ - { - name = "syn"; - packageId = "syn 2.0.96"; - features = [ "full" "extra-traits" ]; - } - ]; - features = { - "__private" = [ "syn/visit-mut" ]; - }; - resolvedDefaultFeatures = [ "default" ]; - }; - "axum-test" = rec { - crateName = "axum-test"; - version = "16.4.1"; - edition = "2021"; - sha256 = "1p5qxacvxsagnqq30nr2wznjyhgb8svsfb925ah3d2b0s91s9qv3"; - libName = "axum_test"; - authors = [ - "Joseph Lenton <josephlenton@gmail.com>" - ]; - dependencies = [ - { - name = "anyhow"; - packageId = "anyhow"; - } - { - name = "assert-json-diff"; - packageId = "assert-json-diff"; - } - { - name = "auto-future"; - packageId = "auto-future"; - } - { - name = "axum"; - packageId = "axum"; - } - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "bytesize"; - packageId = "bytesize"; - } - { - name = "cookie"; - packageId = "cookie 0.18.1"; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - { - name = "http-body-util"; - packageId = "http-body-util"; - } - { - name = "hyper"; - packageId = "hyper 1.5.2"; - features = [ "http1" ]; - } - { - name = "hyper-util"; - packageId = "hyper-util"; - features = [ "client" "http1" "client-legacy" ]; - } - { - name = "mime"; - packageId = "mime 0.3.17"; - } - { - name = "pretty_assertions"; - packageId = "pretty_assertions"; - optional = true; - } - { - name = "reserve-port"; - packageId = "reserve-port"; - } - { - name = "rust-multipart-rfc7578_2"; - packageId = "rust-multipart-rfc7578_2"; - } - { - name = "serde"; - packageId = "serde 1.0.217"; - } - { - name = "serde_json"; - packageId = "serde_json"; - } - { - name = "serde_urlencoded"; - packageId = "serde_urlencoded"; - } - { - name = "smallvec"; - packageId = "smallvec"; - } - { - name = "tokio"; - packageId = "tokio"; - features = [ "rt" ]; - } - { - name = "tower"; - packageId = "tower"; - features = [ "util" "make" ]; - } - { - name = "url"; - packageId = "url 2.5.4"; - } - ]; - devDependencies = [ - { - name = "axum"; - packageId = "axum"; - features = [ "multipart" "tokio" "ws" ]; - } - { - name = "tokio"; - packageId = "tokio"; - features = [ "rt" "rt-multi-thread" "sync" "time" "macros" ]; - } - ]; - features = { - "all" = [ "pretty-assertions" "yaml" "msgpack" "reqwest" "shuttle" "typed-routing" "ws" ]; - "default" = [ "pretty-assertions" ]; - "msgpack" = [ "dep:rmp-serde" ]; - "pretty-assertions" = [ "dep:pretty_assertions" ]; - "reqwest" = [ "dep:reqwest" ]; - "shuttle" = [ "dep:shuttle-axum" ]; - "typed-routing" = [ "dep:axum-extra" ]; - "ws" = [ "axum/ws" "tokio/time" "dep:uuid" "dep:base64" "dep:tokio-tungstenite" "dep:futures-util" ]; - "yaml" = [ "dep:serde_yaml" ]; - }; - resolvedDefaultFeatures = [ "default" "pretty-assertions" ]; - }; "az" = rec { crateName = "az"; version = "1.2.1"; @@ -2055,7 +1560,7 @@ rec { dependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; usesDefaultFeatures = false; } @@ -2199,19 +1704,6 @@ rec { }; resolvedDefaultFeatures = [ "default" "std" ]; }; - "bytesize" = rec { - crateName = "bytesize"; - version = "1.3.0"; - edition = "2015"; - sha256 = "1k3aak70iwz4s2gsjbxf0ws4xnixqbdz6p2ha96s06748fpniqx3"; - authors = [ - "Hyunsik Choi <hyunsik.choi@gmail.com>" - ]; - features = { - "serde" = [ "dep:serde" ]; - }; - resolvedDefaultFeatures = [ "default" ]; - }; "cairo-rs" = rec { crateName = "cairo-rs"; version = "0.18.5"; @@ -2434,7 +1926,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; usesDefaultFeatures = false; } @@ -2494,7 +1986,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; usesDefaultFeatures = false; } @@ -2615,10 +2107,10 @@ rec { }; "clap" = rec { crateName = "clap"; - version = "4.5.26"; + version = "4.5.30"; edition = "2021"; crateBin = []; - sha256 = "10v7qvn90calfbhap1c4r249i5c7fbxj09fn3szfz9pkis85xsx8"; + sha256 = "0vcyrn4ymq2gd56sl3xnfki8q8llg64sj3rj3qx33mgsf66v3dwj"; dependencies = [ { name = "clap_builder"; @@ -2647,6 +2139,7 @@ rec { "unicode" = [ "clap_builder/unicode" ]; "unstable-doc" = [ "clap_builder/unstable-doc" "derive" ]; "unstable-ext" = [ "clap_builder/unstable-ext" ]; + "unstable-markdown" = [ "clap_derive/unstable-markdown" ]; "unstable-styles" = [ "clap_builder/unstable-styles" ]; "unstable-v5" = [ "clap_builder/unstable-v5" "clap_derive?/unstable-v5" "deprecated" ]; "usage" = [ "clap_builder/usage" ]; @@ -2656,9 +2149,9 @@ rec { }; "clap_builder" = rec { crateName = "clap_builder"; - version = "4.5.26"; + version = "4.5.30"; edition = "2021"; - sha256 = "08f1mzcvi7zjhm7hvz6al4jnv70ccqhwiaq74hihlspwnl0iic4n"; + sha256 = "0369xis2ar46icsaxqyy37976mlb62alzyx4j53k99vq2w3v4pd3"; dependencies = [ { name = "anstream"; @@ -2695,9 +2188,9 @@ rec { }; "clap_derive" = rec { crateName = "clap_derive"; - version = "4.5.24"; + version = "4.5.28"; edition = "2021"; - sha256 = "131ih3dm76srkbpfx7zfspp9b556zgzj31wqhl0ji2b39lcmbdsl"; + sha256 = "1vgigkhljp3r8r5lwdrn1ij93nafmjwh8cx77nppb9plqsaysk5z"; procMacro = true; dependencies = [ { @@ -2720,6 +2213,7 @@ rec { ]; features = { "raw-deprecated" = [ "deprecated" ]; + "unstable-markdown" = [ "dep:pulldown-cmark" "dep:anstyle" ]; "unstable-v5" = [ "deprecated" ]; }; resolvedDefaultFeatures = [ "default" ]; @@ -2811,7 +2305,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -2887,7 +2381,7 @@ rec { "random" = [ "rand" ]; }; }; - "cookie 0.17.0" = rec { + "cookie" = rec { crateName = "cookie"; version = "0.17.0"; edition = "2018"; @@ -2926,45 +2420,6 @@ rec { "subtle" = [ "dep:subtle" ]; }; }; - "cookie 0.18.1" = rec { - crateName = "cookie"; - version = "0.18.1"; - edition = "2018"; - sha256 = "0iy749flficrlvgr3hjmf3igr738lk81n5akzf4ym4cs6cxg7pjd"; - authors = [ - "Sergio Benitez <sb@sergio.bz>" - "Alex Crichton <alex@alexcrichton.com>" - ]; - dependencies = [ - { - name = "time"; - packageId = "time 0.3.37"; - usesDefaultFeatures = false; - features = [ "std" "parsing" "formatting" "macros" ]; - } - ]; - buildDependencies = [ - { - name = "version_check"; - packageId = "version_check 0.9.5"; - } - ]; - features = { - "aes-gcm" = [ "dep:aes-gcm" ]; - "base64" = [ "dep:base64" ]; - "hkdf" = [ "dep:hkdf" ]; - "hmac" = [ "dep:hmac" ]; - "key-expansion" = [ "sha2" "hkdf" ]; - "percent-encode" = [ "percent-encoding" ]; - "percent-encoding" = [ "dep:percent-encoding" ]; - "private" = [ "aes-gcm" "base64" "rand" "subtle" ]; - "rand" = [ "dep:rand" ]; - "secure" = [ "private" "signed" "key-expansion" ]; - "sha2" = [ "dep:sha2" ]; - "signed" = [ "hmac" "sha2" "base64" "rand" "subtle" ]; - "subtle" = [ "dep:subtle" ]; - }; - }; "cookie-factory" = rec { crateName = "cookie-factory"; version = "0.3.3"; @@ -3325,7 +2780,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -3495,7 +2950,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -3626,16 +3081,6 @@ rec { }; resolvedDefaultFeatures = [ "alloc" "powerfmt" "std" ]; }; - "diff" = rec { - crateName = "diff"; - version = "0.1.13"; - edition = "2015"; - sha256 = "1j0nzjxci2zqx63hdcihkp0a4dkdmzxd7my4m7zk6cjyfy34j9an"; - authors = [ - "Utkarsh Kukreti <utkarshkukreti@gmail.com>" - ]; - - }; "digest" = rec { crateName = "digest"; version = "0.10.7"; @@ -3702,7 +3147,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; } { @@ -3742,7 +3187,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; } { @@ -3821,7 +3266,7 @@ rec { dependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; features = [ "derive" ]; } @@ -3857,7 +3302,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } { name = "serde_derive"; @@ -4145,33 +3590,6 @@ rec { ]; }; - "fallible-iterator" = rec { - crateName = "fallible-iterator"; - version = "0.3.0"; - edition = "2018"; - sha256 = "0ja6l56yka5vn4y4pk6hn88z0bpny7a8k1919aqjzp0j1yhy9k1a"; - libName = "fallible_iterator"; - authors = [ - "Steven Fackler <sfackler@gmail.com>" - ]; - features = { - "default" = [ "alloc" ]; - "std" = [ "alloc" ]; - }; - resolvedDefaultFeatures = [ "alloc" "default" ]; - }; - "fallible-streaming-iterator" = rec { - crateName = "fallible-streaming-iterator"; - version = "0.1.9"; - edition = "2015"; - sha256 = "0nj6j26p71bjy8h42x6jahx1hn0ng6mc2miwpgwnp8vnwqf4jq3k"; - libName = "fallible_streaming_iterator"; - authors = [ - "Steven Fackler <sfackler@gmail.com>" - ]; - features = { - }; - }; "fastrand" = rec { crateName = "fastrand"; version = "2.3.0"; @@ -4277,7 +3695,7 @@ rec { } { name = "cookie"; - packageId = "cookie 0.17.0"; + packageId = "cookie"; } { name = "futures-util"; @@ -4317,7 +3735,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -4841,7 +4259,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -5481,7 +4899,7 @@ rec { libName = "geo_types"; }; - "getrandom" = rec { + "getrandom 0.2.15" = rec { crateName = "getrandom"; version = "0.2.15"; edition = "2018"; @@ -5517,6 +4935,84 @@ rec { }; resolvedDefaultFeatures = [ "std" ]; }; + "getrandom 0.3.1" = rec { + crateName = "getrandom"; + version = "0.3.1"; + edition = "2021"; + sha256 = "1y154yzby383p63ndw6zpfm0fz3vf6c0zdwc7df6vkl150wrr923"; + authors = [ + "The Rand Project Developers" + ]; + dependencies = [ + { + name = "cfg-if"; + packageId = "cfg-if"; + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: ((("linux" == target."os") || ("android" == target."os")) && (!(("custom" == target."getrandom_backend") || ("rdrand" == target."getrandom_backend") || ("rndr" == target."getrandom_backend")))); + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: (("dragonfly" == target."os") || ("freebsd" == target."os") || ("hurd" == target."os") || ("illumos" == target."os") || (("horizon" == target."os") && ("arm" == target."arch"))); + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: (("haiku" == target."os") || ("redox" == target."os") || ("nto" == target."os") || ("aix" == target."os")); + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: (("ios" == target."os") || ("visionos" == target."os") || ("watchos" == target."os") || ("tvos" == target."os")); + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: (("macos" == target."os") || ("openbsd" == target."os") || ("vita" == target."os") || ("emscripten" == target."os")); + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: ("netbsd" == target."os"); + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: ("solaris" == target."os"); + } + { + name = "libc"; + packageId = "libc"; + usesDefaultFeatures = false; + target = { target, features }: ("vxworks" == target."os"); + } + { + name = "wasi"; + packageId = "wasi 0.13.3+wasi-0.2.2"; + usesDefaultFeatures = false; + target = { target, features }: (("wasm32" == target."arch") && ("wasi" == target."os") && ("p2" == target."env")); + } + { + name = "windows-targets"; + packageId = "windows-targets 0.52.6"; + target = { target, features }: ((target."windows" or false) && (!("win7" == target."vendor"))); + } + ]; + features = { + "rustc-dep-of-std" = [ "dep:compiler_builtins" "dep:core" ]; + "wasm_js" = [ "dep:wasm-bindgen" "dep:js-sys" ]; + }; + }; "gif 0.11.4" = rec { crateName = "gif"; version = "0.11.4"; @@ -5868,7 +5364,7 @@ rec { } { name = "proc-macro-crate"; - packageId = "proc-macro-crate 2.0.2"; + packageId = "proc-macro-crate 2.0.0"; } { name = "proc-macro-error"; @@ -6027,7 +5523,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } { name = "serde_json"; @@ -6075,7 +5571,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "alloc" "derive" ]; } { @@ -6642,39 +6138,7 @@ rec { }; resolvedDefaultFeatures = [ "alloc" "default" "std" ]; }; - "hashbrown 0.14.5" = rec { - crateName = "hashbrown"; - version = "0.14.5"; - edition = "2021"; - sha256 = "1wa1vy1xs3mp11bn3z9dv0jricgr6a2j0zkf1g19yz3vw4il89z5"; - authors = [ - "Amanieu d'Antras <amanieu@gmail.com>" - ]; - dependencies = [ - { - name = "ahash"; - packageId = "ahash"; - optional = true; - usesDefaultFeatures = false; - } - ]; - features = { - "ahash" = [ "dep:ahash" ]; - "alloc" = [ "dep:alloc" ]; - "allocator-api2" = [ "dep:allocator-api2" ]; - "compiler_builtins" = [ "dep:compiler_builtins" ]; - "core" = [ "dep:core" ]; - "default" = [ "ahash" "inline-more" "allocator-api2" ]; - "equivalent" = [ "dep:equivalent" ]; - "nightly" = [ "allocator-api2?/nightly" "bumpalo/allocator_api" ]; - "rayon" = [ "dep:rayon" ]; - "rkyv" = [ "dep:rkyv" ]; - "rustc-dep-of-std" = [ "nightly" "core" "compiler_builtins" "alloc" "rustc-internal-api" ]; - "serde" = [ "dep:serde" ]; - }; - resolvedDefaultFeatures = [ "ahash" "inline-more" ]; - }; - "hashbrown 0.15.2" = rec { + "hashbrown" = rec { crateName = "hashbrown"; version = "0.15.2"; edition = "2021"; @@ -6718,7 +6182,7 @@ rec { }; resolvedDefaultFeatures = [ "allocator-api2" "default" "default-hasher" "equivalent" "inline-more" "raw-entry" ]; }; - "hashlink 0.10.0" = rec { + "hashlink" = rec { crateName = "hashlink"; version = "0.10.0"; edition = "2018"; @@ -6729,7 +6193,7 @@ rec { dependencies = [ { name = "hashbrown"; - packageId = "hashbrown 0.15.2"; + packageId = "hashbrown"; usesDefaultFeatures = false; features = [ "default-hasher" "inline-more" ]; } @@ -6739,27 +6203,6 @@ rec { "serde_impl" = [ "serde" ]; }; }; - "hashlink 0.9.1" = rec { - crateName = "hashlink"; - version = "0.9.1"; - edition = "2018"; - sha256 = "1byq4nyrflm5s6wdx5qwp96l1qbp2d0nljvrr5yqrsfy51qzz93b"; - authors = [ - "kyren <kerriganw@gmail.com>" - ]; - dependencies = [ - { - name = "hashbrown"; - packageId = "hashbrown 0.14.5"; - usesDefaultFeatures = false; - features = [ "ahash" "inline-more" ]; - } - ]; - features = { - "serde" = [ "dep:serde" ]; - "serde_impl" = [ "serde" ]; - }; - }; "headers" = rec { crateName = "headers"; version = "0.3.9"; @@ -7073,7 +6516,7 @@ rec { }; resolvedDefaultFeatures = [ "default" "std" ]; }; - "http-body 0.4.6" = rec { + "http-body" = rec { crateName = "http-body"; version = "0.4.6"; edition = "2018"; @@ -7099,65 +6542,6 @@ rec { } ]; - }; - "http-body 1.0.1" = rec { - crateName = "http-body"; - version = "1.0.1"; - edition = "2018"; - sha256 = "111ir5k2b9ihz5nr9cz7cwm7fnydca7dx4hc7vr16scfzghxrzhy"; - libName = "http_body"; - authors = [ - "Carl Lerche <me@carllerche.com>" - "Lucio Franco <luciofranco14@gmail.com>" - "Sean McArthur <sean@seanmonstar.com>" - ]; - dependencies = [ - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - ]; - - }; - "http-body-util" = rec { - crateName = "http-body-util"; - version = "0.1.2"; - edition = "2018"; - sha256 = "0kslwazg4400qnc2azkrgqqci0fppv12waicnsy5d8hncvbjjd3r"; - libName = "http_body_util"; - authors = [ - "Carl Lerche <me@carllerche.com>" - "Lucio Franco <luciofranco14@gmail.com>" - "Sean McArthur <sean@seanmonstar.com>" - ]; - dependencies = [ - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - { - name = "http-body"; - packageId = "http-body 1.0.1"; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - } - ]; - }; "httparse" = rec { crateName = "httparse"; @@ -7288,7 +6672,7 @@ rec { } { name = "http-body"; - packageId = "http-body 0.4.6"; + packageId = "http-body"; } { name = "httparse"; @@ -7357,108 +6741,6 @@ rec { }; resolvedDefaultFeatures = [ "client" "default" "h2" "http1" "http2" "runtime" "server" "socket2" "stream" "tcp" ]; }; - "hyper 1.5.2" = rec { - crateName = "hyper"; - version = "1.5.2"; - edition = "2021"; - sha256 = "1q7akfb443yrjzkmnnbp2vs8zi15hgbk466rr4y144v4ppabhvr5"; - authors = [ - "Sean McArthur <sean@seanmonstar.com>" - ]; - dependencies = [ - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "futures-channel"; - packageId = "futures-channel"; - optional = true; - } - { - name = "futures-util"; - packageId = "futures-util"; - optional = true; - usesDefaultFeatures = false; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - { - name = "http-body"; - packageId = "http-body 1.0.1"; - } - { - name = "httparse"; - packageId = "httparse"; - optional = true; - } - { - name = "httpdate"; - packageId = "httpdate"; - optional = true; - } - { - name = "itoa"; - packageId = "itoa"; - optional = true; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - optional = true; - } - { - name = "smallvec"; - packageId = "smallvec"; - optional = true; - features = [ "const_generics" "const_new" ]; - } - { - name = "tokio"; - packageId = "tokio"; - features = [ "sync" ]; - } - { - name = "want"; - packageId = "want"; - optional = true; - } - ]; - devDependencies = [ - { - name = "futures-channel"; - packageId = "futures-channel"; - features = [ "sink" ]; - } - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - features = [ "alloc" "sink" ]; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - } - { - name = "tokio"; - packageId = "tokio"; - features = [ "fs" "macros" "net" "io-std" "io-util" "rt" "rt-multi-thread" "sync" "time" "test-util" ]; - } - ]; - features = { - "client" = [ "dep:want" "dep:pin-project-lite" "dep:smallvec" ]; - "ffi" = [ "dep:http-body-util" "futures-util?/alloc" ]; - "full" = [ "client" "http1" "http2" "server" ]; - "http1" = [ "dep:futures-channel" "dep:futures-util" "dep:httparse" "dep:itoa" ]; - "http2" = [ "dep:futures-channel" "dep:futures-util" "dep:h2" ]; - "server" = [ "dep:httpdate" "dep:pin-project-lite" "dep:smallvec" ]; - "tracing" = [ "dep:tracing" ]; - }; - resolvedDefaultFeatures = [ "client" "default" "http1" "server" ]; - }; "hyper-tls" = rec { crateName = "hyper-tls"; version = "0.5.0"; @@ -7509,101 +6791,6 @@ rec { "vendored" = [ "native-tls/vendored" ]; }; }; - "hyper-util" = rec { - crateName = "hyper-util"; - version = "0.1.10"; - edition = "2021"; - sha256 = "1d1iwrkysjhq63pg54zk3vfby1j7zmxzm9zzyfr4lwvp0szcybfz"; - libName = "hyper_util"; - authors = [ - "Sean McArthur <sean@seanmonstar.com>" - ]; - dependencies = [ - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "futures-channel"; - packageId = "futures-channel"; - optional = true; - } - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - { - name = "http-body"; - packageId = "http-body 1.0.1"; - } - { - name = "hyper"; - packageId = "hyper 1.5.2"; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - } - { - name = "socket2"; - packageId = "socket2"; - optional = true; - features = [ "all" ]; - } - { - name = "tokio"; - packageId = "tokio"; - optional = true; - usesDefaultFeatures = false; - } - { - name = "tower-service"; - packageId = "tower-service"; - optional = true; - } - { - name = "tracing"; - packageId = "tracing"; - optional = true; - usesDefaultFeatures = false; - features = [ "std" ]; - } - ]; - devDependencies = [ - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "hyper"; - packageId = "hyper 1.5.2"; - features = [ "full" ]; - } - { - name = "tokio"; - packageId = "tokio"; - features = [ "macros" "test-util" "signal" ]; - } - ]; - features = { - "client" = [ "hyper/client" "dep:tracing" "dep:futures-channel" "dep:tower-service" ]; - "client-legacy" = [ "client" "dep:socket2" "tokio/sync" ]; - "full" = [ "client" "client-legacy" "server" "server-auto" "server-graceful" "service" "http1" "http2" "tokio" ]; - "http1" = [ "hyper/http1" ]; - "http2" = [ "hyper/http2" ]; - "server" = [ "hyper/server" ]; - "server-auto" = [ "server" "http1" "http2" ]; - "server-graceful" = [ "server" "tokio/sync" "futures-util/alloc" ]; - "service" = [ "dep:tower-service" ]; - "tokio" = [ "dep:tokio" "tokio/net" "tokio/rt" "tokio/time" ]; - }; - resolvedDefaultFeatures = [ "client" "client-legacy" "default" "http1" "server" "service" "tokio" ]; - }; "iana-time-zone" = rec { crateName = "iana-time-zone"; version = "0.1.61"; @@ -7766,6 +6953,13 @@ rec { usesDefaultFeatures = false; features = [ "alloc" ]; } + { + name = "serde"; + packageId = "serde 1.0.218"; + optional = true; + usesDefaultFeatures = false; + features = [ "alloc" "derive" ]; + } { name = "tinystr"; packageId = "tinystr"; @@ -7784,13 +6978,21 @@ rec { usesDefaultFeatures = false; } ]; + devDependencies = [ + { + name = "serde"; + packageId = "serde 1.0.218"; + usesDefaultFeatures = false; + features = [ "derive" ]; + } + ]; features = { "bench" = [ "serde" ]; "databake" = [ "dep:databake" ]; "serde" = [ "dep:serde" "tinystr/serde" ]; "zerovec" = [ "dep:zerovec" ]; }; - resolvedDefaultFeatures = [ "zerovec" ]; + resolvedDefaultFeatures = [ "serde" "zerovec" ]; }; "icu_locid_transform" = rec { crateName = "icu_locid_transform"; @@ -8369,49 +7571,6 @@ rec { }; resolvedDefaultFeatures = [ "bmp" "dds" "default" "dxt" "exr" "farbfeld" "gif" "hdr" "ico" "jpeg" "jpeg_rayon" "openexr" "png" "pnm" "qoi" "tga" "tiff" "webp" ]; }; - "include_dir" = rec { - crateName = "include_dir"; - version = "0.7.4"; - edition = "2021"; - sha256 = "1pfh3g45z88kwq93skng0n6g3r7zkhq9ldqs9y8rvr7i11s12gcj"; - authors = [ - "Michael Bryan <michaelfbryan@gmail.com>" - ]; - dependencies = [ - { - name = "include_dir_macros"; - packageId = "include_dir_macros"; - } - ]; - features = { - "glob" = [ "dep:glob" ]; - "metadata" = [ "include_dir_macros/metadata" ]; - "nightly" = [ "include_dir_macros/nightly" ]; - }; - resolvedDefaultFeatures = [ "default" ]; - }; - "include_dir_macros" = rec { - crateName = "include_dir_macros"; - version = "0.7.4"; - edition = "2021"; - sha256 = "0x8smnf6knd86g69p19z5lpfsaqp8w0nx14kdpkz1m8bxnkqbavw"; - procMacro = true; - authors = [ - "Michael Bryan <michaelfbryan@gmail.com>" - ]; - dependencies = [ - { - name = "proc-macro2"; - packageId = "proc-macro2"; - } - { - name = "quote"; - packageId = "quote"; - } - ]; - features = { - }; - }; "indent_write" = rec { crateName = "indent_write"; version = "2.2.0"; @@ -8437,7 +7596,7 @@ rec { } { name = "hashbrown"; - packageId = "hashbrown 0.15.2"; + packageId = "hashbrown"; usesDefaultFeatures = false; } ]; @@ -8732,6 +7891,67 @@ rec { } ]; + }; + "l10n-db" = rec { + crateName = "l10n-db"; + version = "0.1.0"; + edition = "2021"; + crateBin = [ + { + name = "main"; + path = "src/bin/main.rs"; + requiredFeatures = [ ]; + } + ]; + # We can't filter paths with references in Nix 2.4 + # See https://github.com/NixOS/nix/issues/5410 + src = if ((lib.versionOlder builtins.nixVersion "2.4pre20211007") || (lib.versionOlder "2.5" builtins.nixVersion )) + then lib.cleanSourceWith { filter = sourceFilter; src = ./l10n-db; } + else ./l10n-db; + libName = "l10n_db"; + dependencies = [ + { + name = "chrono"; + packageId = "chrono"; + features = [ "serde" ]; + } + { + name = "clap"; + packageId = "clap"; + features = [ "derive" ]; + } + { + name = "icu_locid"; + packageId = "icu_locid"; + features = [ "serde" ]; + } + { + name = "serde"; + packageId = "serde 1.0.218"; + features = [ "derive" ]; + } + { + name = "serde_json"; + packageId = "serde_json"; + } + { + name = "tempfile"; + packageId = "tempfile"; + } + { + name = "thiserror"; + packageId = "thiserror 2.0.11"; + } + { + name = "toml"; + packageId = "toml"; + } + { + name = "xml-rs"; + packageId = "xml-rs"; + } + ]; + }; "language-tags" = rec { crateName = "language-tags"; @@ -9090,7 +8310,7 @@ rec { "syn" = [ "dep:syn" ]; "vcpkg" = [ "dep:vcpkg" ]; }; - resolvedDefaultFeatures = [ "bundled" "bundled_bindings" "cc" "default" "min_sqlite_version_3_14_0" "pkg-config" "unlock_notify" "vcpkg" ]; + resolvedDefaultFeatures = [ "bundled" "bundled_bindings" "cc" "pkg-config" "unlock_notify" "vcpkg" ]; }; "libyml" = rec { crateName = "libyml"; @@ -9308,18 +8528,6 @@ rec { libPath = "lib.rs"; }; - "matchit" = rec { - crateName = "matchit"; - version = "0.7.3"; - edition = "2021"; - sha256 = "156bgdmmlv4crib31qhgg49nsjk88dxkdqp80ha2pk2rk6n6ax0f"; - authors = [ - "Ibraheem Ahmed <ibraheem@ibraheem.ca>" - ]; - features = { - }; - resolvedDefaultFeatures = [ "default" ]; - }; "md-5" = rec { crateName = "md-5"; version = "0.10.6"; @@ -9412,7 +8620,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } { name = "serde_derive"; @@ -10383,7 +9591,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -11349,31 +10557,6 @@ rec { }; resolvedDefaultFeatures = [ "simd" "std" ]; }; - "pretty_assertions" = rec { - crateName = "pretty_assertions"; - version = "1.4.1"; - edition = "2018"; - sha256 = "0v8iq35ca4rw3rza5is3wjxwsf88303ivys07anc5yviybi31q9s"; - authors = [ - "Colin Kiegel <kiegel@gmx.de>" - "Florent Fayolle <florent.fayolle69@gmail.com>" - "Tom Milligan <code@tommilligan.net>" - ]; - dependencies = [ - { - name = "diff"; - packageId = "diff"; - } - { - name = "yansi"; - packageId = "yansi"; - } - ]; - features = { - "default" = [ "std" ]; - }; - resolvedDefaultFeatures = [ "default" "std" ]; - }; "pretty_env_logger" = rec { crateName = "pretty_env_logger"; version = "0.5.0"; @@ -11415,20 +10598,16 @@ rec { ]; }; - "proc-macro-crate 2.0.2" = rec { + "proc-macro-crate 2.0.0" = rec { crateName = "proc-macro-crate"; - version = "2.0.2"; + version = "2.0.0"; edition = "2021"; - sha256 = "092x5acqnic14cw6vacqap5kgknq3jn4c6jij9zi6j85839jc3xh"; + sha256 = "1s23imns07vmacn2xjd5hv2h6rr94iqq3fd2frwa6i4h2nk6d0vy"; libName = "proc_macro_crate"; authors = [ "Bastian Köcher <git@kchr.de>" ]; dependencies = [ - { - name = "toml_datetime"; - packageId = "toml_datetime"; - } { name = "toml_edit"; packageId = "toml_edit 0.20.2"; @@ -11962,7 +11141,7 @@ rec { dependencies = [ { name = "getrandom"; - packageId = "getrandom"; + packageId = "getrandom 0.2.15"; optional = true; } ]; @@ -12447,7 +11626,7 @@ rec { } { name = "http-body"; - packageId = "http-body 0.4.6"; + packageId = "http-body"; target = { target, features }: (!("wasm32" == target."arch")); } { @@ -12513,7 +11692,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } { name = "serde_json"; @@ -12531,7 +11710,7 @@ rec { } { name = "sync_wrapper"; - packageId = "sync_wrapper 0.1.2"; + packageId = "sync_wrapper"; } { name = "system-configuration"; @@ -12598,7 +11777,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; target = {target, features}: (!("wasm32" == target."arch")); features = [ "derive" ]; } @@ -12664,27 +11843,6 @@ rec { }; resolvedDefaultFeatures = [ "__tls" "default" "default-tls" "hyper-tls" "json" "native-tls-crate" "serde_json" "tokio-native-tls" ]; }; - "reserve-port" = rec { - crateName = "reserve-port"; - version = "2.0.1"; - edition = "2021"; - sha256 = "10x21rdb1hjzp6n5flbbw3hfd7brmirckz1q0zsf3a7s5d516f4q"; - libName = "reserve_port"; - authors = [ - "Joseph Lenton <josephlenton@gmail.com>" - ]; - dependencies = [ - { - name = "lazy_static"; - packageId = "lazy_static"; - } - { - name = "thiserror"; - packageId = "thiserror 1.0.69"; - } - ]; - - }; "result-extended" = rec { crateName = "result-extended"; version = "0.1.0"; @@ -12803,158 +11961,6 @@ rec { }; resolvedDefaultFeatures = [ "default" "pem" "std" "u64_digit" ]; }; - "rusqlite" = rec { - crateName = "rusqlite"; - version = "0.32.1"; - edition = "2021"; - sha256 = "0vlx040bppl414pbjgbp7qr4jdxwszi9krx0m63zzf2f2whvflvp"; - authors = [ - "The rusqlite developers" - ]; - dependencies = [ - { - name = "bitflags"; - packageId = "bitflags 2.8.0"; - } - { - name = "fallible-iterator"; - packageId = "fallible-iterator"; - } - { - name = "fallible-streaming-iterator"; - packageId = "fallible-streaming-iterator"; - } - { - name = "hashlink"; - packageId = "hashlink 0.9.1"; - } - { - name = "libsqlite3-sys"; - packageId = "libsqlite3-sys"; - } - { - name = "smallvec"; - packageId = "smallvec"; - } - ]; - features = { - "array" = [ "vtab" ]; - "buildtime_bindgen" = [ "libsqlite3-sys/buildtime_bindgen" ]; - "bundled" = [ "libsqlite3-sys/bundled" "modern_sqlite" ]; - "bundled-full" = [ "modern-full" "bundled" ]; - "bundled-sqlcipher" = [ "libsqlite3-sys/bundled-sqlcipher" "bundled" ]; - "bundled-sqlcipher-vendored-openssl" = [ "libsqlite3-sys/bundled-sqlcipher-vendored-openssl" "bundled-sqlcipher" ]; - "bundled-windows" = [ "libsqlite3-sys/bundled-windows" ]; - "chrono" = [ "dep:chrono" ]; - "csv" = [ "dep:csv" ]; - "csvtab" = [ "csv" "vtab" ]; - "in_gecko" = [ "modern_sqlite" "libsqlite3-sys/in_gecko" ]; - "loadable_extension" = [ "libsqlite3-sys/loadable_extension" ]; - "modern-full" = [ "array" "backup" "blob" "modern_sqlite" "chrono" "collation" "column_decltype" "csvtab" "extra_check" "functions" "hooks" "i128_blob" "limits" "load_extension" "serde_json" "series" "time" "trace" "unlock_notify" "url" "uuid" "vtab" "window" ]; - "modern_sqlite" = [ "libsqlite3-sys/bundled_bindings" ]; - "preupdate_hook" = [ "libsqlite3-sys/preupdate_hook" "hooks" ]; - "rusqlite-macros" = [ "dep:rusqlite-macros" ]; - "serde_json" = [ "dep:serde_json" ]; - "serialize" = [ "modern_sqlite" ]; - "series" = [ "vtab" ]; - "session" = [ "libsqlite3-sys/session" "hooks" ]; - "sqlcipher" = [ "libsqlite3-sys/sqlcipher" ]; - "time" = [ "dep:time" ]; - "unlock_notify" = [ "libsqlite3-sys/unlock_notify" ]; - "url" = [ "dep:url" ]; - "uuid" = [ "dep:uuid" ]; - "wasm32-wasi-vfs" = [ "libsqlite3-sys/wasm32-wasi-vfs" ]; - "window" = [ "functions" ]; - "with-asan" = [ "libsqlite3-sys/with-asan" ]; - }; - }; - "rusqlite_migration" = rec { - crateName = "rusqlite_migration"; - version = "1.3.1"; - edition = "2021"; - sha256 = "076dm65g0sngzrb93r07va4l5zl3gjx9gq5mlsh21p7p0bl44fwj"; - authors = [ - "Clément Joly <foss@131719.xyz>" - ]; - dependencies = [ - { - name = "include_dir"; - packageId = "include_dir"; - optional = true; - } - { - name = "log"; - packageId = "log 0.4.25"; - } - { - name = "rusqlite"; - packageId = "rusqlite"; - usesDefaultFeatures = false; - } - ]; - features = { - "alpha-async-tokio-rusqlite" = [ "dep:tokio-rusqlite" "dep:tokio" ]; - "from-directory" = [ "dep:include_dir" ]; - }; - resolvedDefaultFeatures = [ "default" "from-directory" ]; - }; - "rust-multipart-rfc7578_2" = rec { - crateName = "rust-multipart-rfc7578_2"; - version = "0.6.1"; - edition = "2021"; - sha256 = "0mwd3i2mk91n6diaxnkw28vyjbifhrm5ls73pcpfzz8a1i0lidq3"; - libName = "rust_multipart_rfc7578_2"; - authors = [ - "Joseph Lenton <josephlenton@gmail.com>" - "Ferris Tseng <ferristseng@fastmail.fm>" - ]; - dependencies = [ - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "futures-core"; - packageId = "futures-core"; - } - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - features = [ "io" ]; - } - { - name = "http"; - packageId = "http 0.2.12"; - } - { - name = "mime"; - packageId = "mime 0.3.17"; - } - { - name = "mime_guess"; - packageId = "mime_guess 2.0.5"; - } - { - name = "rand"; - packageId = "rand 0.8.5"; - features = [ "small_rng" ]; - } - { - name = "thiserror"; - packageId = "thiserror 1.0.69"; - } - ]; - devDependencies = [ - { - name = "futures-util"; - packageId = "futures-util"; - usesDefaultFeatures = false; - features = [ "std" ]; - } - ]; - - }; "rustc-demangle" = rec { crateName = "rustc-demangle"; version = "0.1.24"; @@ -13422,11 +12428,11 @@ rec { }; resolvedDefaultFeatures = [ "default" "std" ]; }; - "serde 1.0.217" = rec { + "serde 1.0.218" = rec { crateName = "serde"; - version = "1.0.217"; + version = "1.0.218"; edition = "2018"; - sha256 = "0w2ck1p1ajmrv1cf51qf7igjn2nc51r0izzc00fzmmhkvxjl5z02"; + sha256 = "0q6z4bnrwagnms0bds4886711l6mc68s979i49zd3xnvkg8wkpz8"; authors = [ "Erick Tryzelaar <erick.tryzelaar@gmail.com>" "David Tolnay <dtolnay@gmail.com>" @@ -13458,9 +12464,9 @@ rec { }; "serde_derive" = rec { crateName = "serde_derive"; - version = "1.0.217"; + version = "1.0.218"; edition = "2015"; - sha256 = "180r3rj5gi5s1m23q66cr5wlfgc5jrs6n1mdmql2njnhk37zg6ss"; + sha256 = "0azqd74xbpb1v5vf6w1fdbgmwp39ljjfj25cib5rgrzlj7hh75gh"; procMacro = true; authors = [ "Erick Tryzelaar <erick.tryzelaar@gmail.com>" @@ -13492,9 +12498,9 @@ rec { }; "serde_json" = rec { crateName = "serde_json"; - version = "1.0.136"; + version = "1.0.139"; edition = "2021"; - sha256 = "1lipcjhh1zazh283i4wsl4l14knh81q2rlkwmag8v8s2rwihqsik"; + sha256 = "19kj3irpa22a7djz1jaf4wambzh7psiqa6zyafqnb76crhx6ry24"; authors = [ "Erick Tryzelaar <erick.tryzelaar@gmail.com>" "David Tolnay <dtolnay@gmail.com>" @@ -13515,14 +12521,14 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; usesDefaultFeatures = false; } ]; devDependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } ]; @@ -13535,26 +12541,6 @@ rec { }; resolvedDefaultFeatures = [ "default" "raw_value" "std" ]; }; - "serde_path_to_error" = rec { - crateName = "serde_path_to_error"; - version = "0.1.16"; - edition = "2021"; - sha256 = "19hlz2359l37ifirskpcds7sxg0gzpqvfilibs7whdys0128i6dg"; - authors = [ - "David Tolnay <dtolnay@gmail.com>" - ]; - dependencies = [ - { - name = "itoa"; - packageId = "itoa"; - } - { - name = "serde"; - packageId = "serde 1.0.217"; - } - ]; - - }; "serde_spanned" = rec { crateName = "serde_spanned"; version = "0.6.8"; @@ -13563,14 +12549,14 @@ rec { dependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; } ]; devDependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } ]; features = { @@ -13601,7 +12587,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } ]; @@ -13638,7 +12624,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } ]; buildDependencies = [ @@ -13650,7 +12636,7 @@ rec { devDependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } ]; @@ -13658,6 +12644,24 @@ rec { }; resolvedDefaultFeatures = [ "default" ]; }; + "server" = rec { + crateName = "server"; + version = "0.1.0"; + edition = "2021"; + crateBin = [ + { + name = "server"; + path = "src/main.rs"; + requiredFeatures = [ ]; + } + ]; + # We can't filter paths with references in Nix 2.4 + # See https://github.com/NixOS/nix/issues/5410 + src = if ((lib.versionOlder builtins.nixVersion "2.4pre20211007") || (lib.versionOlder "2.5" builtins.nixVersion )) + then lib.cleanSourceWith { filter = sourceFilter; src = ./visions/server; } + else ./visions/server; + + }; "sgf" = rec { crateName = "sgf"; version = "0.1.0"; @@ -13690,7 +12694,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -13992,7 +12996,7 @@ rec { dependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; usesDefaultFeatures = false; } @@ -14245,11 +13249,11 @@ rec { } { name = "hashbrown"; - packageId = "hashbrown 0.15.2"; + packageId = "hashbrown"; } { name = "hashlink"; - packageId = "hashlink 0.10.0"; + packageId = "hashlink"; } { name = "indexmap"; @@ -14275,7 +13279,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; features = [ "derive" "rc" ]; } @@ -14478,7 +13482,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -14697,7 +13701,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; } { @@ -14864,7 +13868,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -14986,7 +13990,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; features = [ "derive" ]; } @@ -15152,7 +14156,7 @@ rec { }; resolvedDefaultFeatures = [ "clone-impls" "default" "derive" "extra-traits" "fold" "full" "parsing" "printing" "proc-macro" "visit" "visit-mut" ]; }; - "sync_wrapper 0.1.2" = rec { + "sync_wrapper" = rec { crateName = "sync_wrapper"; version = "0.1.2"; edition = "2018"; @@ -15165,19 +14169,6 @@ rec { "futures-core" = [ "dep:futures-core" ]; }; }; - "sync_wrapper 1.0.2" = rec { - crateName = "sync_wrapper"; - version = "1.0.2"; - edition = "2021"; - sha256 = "0qvjyasd6w18mjg5xlaq5jgy84jsjfsvmnn12c13gypxbv75dwhb"; - authors = [ - "Actyx AG <developer@actyx.io>" - ]; - features = { - "futures" = [ "futures-core" ]; - "futures-core" = [ "dep:futures-core" ]; - }; - }; "synstructure" = rec { crateName = "synstructure"; version = "0.13.1"; @@ -15330,9 +14321,9 @@ rec { }; "tempfile" = rec { crateName = "tempfile"; - version = "3.15.0"; + version = "3.17.1"; edition = "2021"; - sha256 = "016pmkbwn3shas44gcwq1kc9lajalb90qafhiip5fvv8h6f5b2ls"; + sha256 = "0c52ggq5vy5mzgk5ly36cgzs1cig3cv6r1jarijmzxgkn6na1r92"; authors = [ "Steven Allen <steven@stebalien.com>" "The Rust Project Developers" @@ -15350,7 +14341,7 @@ rec { } { name = "getrandom"; - packageId = "getrandom"; + packageId = "getrandom 0.3.1"; optional = true; usesDefaultFeatures = false; target = { target, features }: ((target."unix" or false) || (target."windows" or false) || ("wasi" == target."os")); @@ -15613,7 +14604,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; usesDefaultFeatures = false; } @@ -15634,7 +14625,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; usesDefaultFeatures = false; features = [ "derive" ]; } @@ -15734,6 +14725,13 @@ rec { packageId = "displaydoc"; usesDefaultFeatures = false; } + { + name = "serde"; + packageId = "serde 1.0.218"; + optional = true; + usesDefaultFeatures = false; + features = [ "alloc" ]; + } { name = "zerovec"; packageId = "zerovec"; @@ -15747,7 +14745,7 @@ rec { "serde" = [ "dep:serde" ]; "zerovec" = [ "dep:zerovec" ]; }; - resolvedDefaultFeatures = [ "alloc" "default" "zerovec" ]; + resolvedDefaultFeatures = [ "alloc" "default" "serde" "zerovec" ]; }; "tinyvec" = rec { crateName = "tinyvec"; @@ -16109,16 +15107,16 @@ rec { }; "toml" = rec { crateName = "toml"; - version = "0.8.2"; + version = "0.8.20"; edition = "2021"; - sha256 = "0g9ysjaqvm2mv8q85xpqfn7hi710hj24sd56k49wyddvvyq8lp8q"; + sha256 = "0j012b37iz1mihksr6a928s6dzszxvblzg3l5wxp7azzsv6sb1yd"; authors = [ "Alex Crichton <alex@alexcrichton.com>" ]; dependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } { name = "serde_spanned"; @@ -16132,39 +15130,40 @@ rec { } { name = "toml_edit"; - packageId = "toml_edit 0.20.2"; + packageId = "toml_edit 0.22.24"; optional = true; + usesDefaultFeatures = false; features = [ "serde" ]; } ]; devDependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } ]; features = { "default" = [ "parse" "display" ]; - "display" = [ "dep:toml_edit" ]; + "display" = [ "dep:toml_edit" "toml_edit?/display" ]; "indexmap" = [ "dep:indexmap" ]; - "parse" = [ "dep:toml_edit" ]; + "parse" = [ "dep:toml_edit" "toml_edit?/parse" ]; "preserve_order" = [ "indexmap" ]; }; - resolvedDefaultFeatures = [ "parse" ]; + resolvedDefaultFeatures = [ "default" "display" "parse" ]; }; "toml_datetime" = rec { crateName = "toml_datetime"; - version = "0.6.3"; + version = "0.6.8"; edition = "2021"; - sha256 = "0jsy7v8bdvmzsci6imj8fzgd255fmy5fzp6zsri14yrry7i77nkw"; + sha256 = "0hgv7v9g35d7y9r2afic58jvlwnf73vgd1mz2k8gihlgrf73bmqd"; authors = [ "Alex Crichton <alex@alexcrichton.com>" ]; dependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; } ]; @@ -16194,7 +15193,7 @@ rec { } { name = "winnow"; - packageId = "winnow"; + packageId = "winnow 0.5.40"; } ]; features = { @@ -16212,6 +15211,36 @@ rec { "Andronik Ordian <write@reusable.software>" "Ed Page <eopage@gmail.com>" ]; + dependencies = [ + { + name = "indexmap"; + packageId = "indexmap"; + features = [ "std" ]; + } + { + name = "toml_datetime"; + packageId = "toml_datetime"; + } + { + name = "winnow"; + packageId = "winnow 0.5.40"; + } + ]; + features = { + "perf" = [ "dep:kstring" ]; + "serde" = [ "dep:serde" "toml_datetime/serde" "dep:serde_spanned" ]; + }; + resolvedDefaultFeatures = [ "default" ]; + }; + "toml_edit 0.22.24" = rec { + crateName = "toml_edit"; + version = "0.22.24"; + edition = "2021"; + sha256 = "0x0lgp70x5cl9nla03xqs5vwwwlrwmd0djkdrp3h3lpdymgpkd0p"; + authors = [ + "Andronik Ordian <write@reusable.software>" + "Ed Page <eopage@gmail.com>" + ]; dependencies = [ { name = "indexmap"; @@ -16220,7 +15249,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; } { @@ -16235,209 +15264,17 @@ rec { } { name = "winnow"; - packageId = "winnow"; + packageId = "winnow 0.7.3"; + optional = true; } ]; features = { + "default" = [ "parse" "display" ]; + "parse" = [ "dep:winnow" ]; "perf" = [ "dep:kstring" ]; "serde" = [ "dep:serde" "toml_datetime/serde" "dep:serde_spanned" ]; }; - resolvedDefaultFeatures = [ "default" "serde" ]; - }; - "tower" = rec { - crateName = "tower"; - version = "0.5.2"; - edition = "2018"; - sha256 = "1ybmd59nm4abl9bsvy6rx31m4zvzp5rja2slzpn712y9b68ssffh"; - authors = [ - "Tower Maintainers <team@tower-rs.com>" - ]; - dependencies = [ - { - name = "futures-core"; - packageId = "futures-core"; - optional = true; - } - { - name = "futures-util"; - packageId = "futures-util"; - optional = true; - usesDefaultFeatures = false; - features = [ "alloc" ]; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - optional = true; - } - { - name = "sync_wrapper"; - packageId = "sync_wrapper 1.0.2"; - optional = true; - } - { - name = "tokio"; - packageId = "tokio"; - optional = true; - features = [ "sync" ]; - } - { - name = "tower-layer"; - packageId = "tower-layer"; - } - { - name = "tower-service"; - packageId = "tower-service"; - } - { - name = "tracing"; - packageId = "tracing"; - optional = true; - usesDefaultFeatures = false; - features = [ "std" ]; - } - ]; - devDependencies = [ - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - } - { - name = "tokio"; - packageId = "tokio"; - features = [ "macros" "sync" "test-util" "rt-multi-thread" ]; - } - { - name = "tracing"; - packageId = "tracing"; - usesDefaultFeatures = false; - features = [ "std" ]; - } - ]; - features = { - "__common" = [ "futures-core" "pin-project-lite" ]; - "balance" = [ "discover" "load" "ready-cache" "make" "slab" "util" ]; - "buffer" = [ "__common" "tokio/sync" "tokio/rt" "tokio-util" "tracing" ]; - "discover" = [ "__common" ]; - "filter" = [ "__common" "futures-util" ]; - "full" = [ "balance" "buffer" "discover" "filter" "hedge" "limit" "load" "load-shed" "make" "ready-cache" "reconnect" "retry" "spawn-ready" "steer" "timeout" "util" ]; - "futures-core" = [ "dep:futures-core" ]; - "futures-util" = [ "dep:futures-util" ]; - "hdrhistogram" = [ "dep:hdrhistogram" ]; - "hedge" = [ "util" "filter" "futures-util" "hdrhistogram" "tokio/time" "tracing" ]; - "indexmap" = [ "dep:indexmap" ]; - "limit" = [ "__common" "tokio/time" "tokio/sync" "tokio-util" "tracing" ]; - "load" = [ "__common" "tokio/time" "tracing" ]; - "load-shed" = [ "__common" ]; - "log" = [ "tracing/log" ]; - "make" = [ "futures-util" "pin-project-lite" "tokio/io-std" ]; - "pin-project-lite" = [ "dep:pin-project-lite" ]; - "ready-cache" = [ "futures-core" "futures-util" "indexmap" "tokio/sync" "tracing" "pin-project-lite" ]; - "reconnect" = [ "make" "tokio/io-std" "tracing" ]; - "retry" = [ "__common" "tokio/time" "util" ]; - "slab" = [ "dep:slab" ]; - "spawn-ready" = [ "__common" "futures-util" "tokio/sync" "tokio/rt" "util" "tracing" ]; - "sync_wrapper" = [ "dep:sync_wrapper" ]; - "timeout" = [ "pin-project-lite" "tokio/time" ]; - "tokio" = [ "dep:tokio" ]; - "tokio-stream" = [ "dep:tokio-stream" ]; - "tokio-util" = [ "dep:tokio-util" ]; - "tracing" = [ "dep:tracing" ]; - "util" = [ "__common" "futures-util" "pin-project-lite" "sync_wrapper" ]; - }; - resolvedDefaultFeatures = [ "__common" "futures-core" "futures-util" "log" "make" "pin-project-lite" "sync_wrapper" "tokio" "tracing" "util" ]; - }; - "tower-http" = rec { - crateName = "tower-http"; - version = "0.6.2"; - edition = "2018"; - sha256 = "15wnvhl6cpir9125s73bqjzjsvfb0fmndmsimnl2ddnlhfvs6gs0"; - libName = "tower_http"; - authors = [ - "Tower Maintainers <team@tower-rs.com>" - ]; - dependencies = [ - { - name = "bitflags"; - packageId = "bitflags 2.8.0"; - } - { - name = "bytes"; - packageId = "bytes"; - } - { - name = "http"; - packageId = "http 1.2.0"; - } - { - name = "pin-project-lite"; - packageId = "pin-project-lite"; - } - { - name = "tower-layer"; - packageId = "tower-layer"; - } - { - name = "tower-service"; - packageId = "tower-service"; - } - ]; - devDependencies = [ - { - name = "bytes"; - packageId = "bytes"; - } - ]; - features = { - "async-compression" = [ "dep:async-compression" ]; - "auth" = [ "base64" "validate-request" ]; - "base64" = [ "dep:base64" ]; - "catch-panic" = [ "tracing" "futures-util/std" "dep:http-body" "dep:http-body-util" ]; - "compression-br" = [ "async-compression/brotli" "futures-core" "dep:http-body" "tokio-util" "tokio" ]; - "compression-deflate" = [ "async-compression/zlib" "futures-core" "dep:http-body" "tokio-util" "tokio" ]; - "compression-full" = [ "compression-br" "compression-deflate" "compression-gzip" "compression-zstd" ]; - "compression-gzip" = [ "async-compression/gzip" "futures-core" "dep:http-body" "tokio-util" "tokio" ]; - "compression-zstd" = [ "async-compression/zstd" "futures-core" "dep:http-body" "tokio-util" "tokio" ]; - "decompression-br" = [ "async-compression/brotli" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ]; - "decompression-deflate" = [ "async-compression/zlib" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ]; - "decompression-full" = [ "decompression-br" "decompression-deflate" "decompression-gzip" "decompression-zstd" ]; - "decompression-gzip" = [ "async-compression/gzip" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ]; - "decompression-zstd" = [ "async-compression/zstd" "futures-core" "dep:http-body" "dep:http-body-util" "tokio-util" "tokio" ]; - "follow-redirect" = [ "futures-util" "dep:http-body" "iri-string" "tower/util" ]; - "fs" = [ "futures-util" "dep:http-body" "dep:http-body-util" "tokio/fs" "tokio-util/io" "tokio/io-util" "dep:http-range-header" "mime_guess" "mime" "percent-encoding" "httpdate" "set-status" "futures-util/alloc" "tracing" ]; - "full" = [ "add-extension" "auth" "catch-panic" "compression-full" "cors" "decompression-full" "follow-redirect" "fs" "limit" "map-request-body" "map-response-body" "metrics" "normalize-path" "propagate-header" "redirect" "request-id" "sensitive-headers" "set-header" "set-status" "timeout" "trace" "util" "validate-request" ]; - "futures-core" = [ "dep:futures-core" ]; - "futures-util" = [ "dep:futures-util" ]; - "httpdate" = [ "dep:httpdate" ]; - "iri-string" = [ "dep:iri-string" ]; - "limit" = [ "dep:http-body" "dep:http-body-util" ]; - "metrics" = [ "dep:http-body" "tokio/time" ]; - "mime" = [ "dep:mime" ]; - "mime_guess" = [ "dep:mime_guess" ]; - "percent-encoding" = [ "dep:percent-encoding" ]; - "request-id" = [ "uuid" ]; - "timeout" = [ "dep:http-body" "tokio/time" ]; - "tokio" = [ "dep:tokio" ]; - "tokio-util" = [ "dep:tokio-util" ]; - "tower" = [ "dep:tower" ]; - "trace" = [ "dep:http-body" "tracing" ]; - "tracing" = [ "dep:tracing" ]; - "util" = [ "tower" ]; - "uuid" = [ "dep:uuid" ]; - "validate-request" = [ "mime" ]; - }; - resolvedDefaultFeatures = [ "cors" "default" ]; - }; - "tower-layer" = rec { - crateName = "tower-layer"; - version = "0.3.3"; - edition = "2018"; - sha256 = "03kq92fdzxin51w8iqix06dcfgydyvx7yr6izjq0p626v9n2l70j"; - libName = "tower_layer"; - authors = [ - "Tower Maintainers <team@tower-rs.com>" - ]; - + resolvedDefaultFeatures = [ "display" "parse" "serde" ]; }; "tower-service" = rec { crateName = "tower-service"; @@ -16743,7 +15580,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; features = [ "derive" ]; } { @@ -17047,17 +15884,6 @@ rec { }; resolvedDefaultFeatures = [ "default" "std" ]; }; - "urlencoding" = rec { - crateName = "urlencoding"; - version = "2.1.3"; - edition = "2021"; - sha256 = "1nj99jp37k47n0hvaz5fvz7z6jd0sb4ppvfy3nphr1zbnyixpy6s"; - authors = [ - "Kornel <kornel@geekhood.net>" - "Bertram Truong <b@bertramtruong.com>" - ]; - - }; "utf-8" = rec { crateName = "utf-8"; version = "0.7.6"; @@ -17146,12 +15972,12 @@ rec { dependencies = [ { name = "getrandom"; - packageId = "getrandom"; + packageId = "getrandom 0.2.15"; optional = true; } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; usesDefaultFeatures = false; } @@ -17186,7 +16012,7 @@ rec { dependencies = [ { name = "getrandom"; - packageId = "getrandom"; + packageId = "getrandom 0.2.15"; optional = true; } ]; @@ -17278,129 +16104,6 @@ rec { "Sergio Benitez <sb@sergio.bz>" ]; - }; - "visions" = rec { - crateName = "visions"; - version = "0.1.0"; - edition = "2021"; - crateBin = [ - { - name = "visions"; - path = "src/main.rs"; - requiredFeatures = [ ]; - } - ]; - # We can't filter paths with references in Nix 2.4 - # See https://github.com/NixOS/nix/issues/5410 - src = if ((lib.versionOlder builtins.nixVersion "2.4pre20211007") || (lib.versionOlder "2.5" builtins.nixVersion )) - then lib.cleanSourceWith { filter = sourceFilter; src = ./visions/server; } - else ./visions/server; - dependencies = [ - { - name = "async-std"; - packageId = "async-std"; - } - { - name = "async-trait"; - packageId = "async-trait"; - } - { - name = "authdb"; - packageId = "authdb"; - } - { - name = "axum"; - packageId = "axum"; - features = [ "macros" ]; - } - { - name = "futures"; - packageId = "futures"; - } - { - name = "include_dir"; - packageId = "include_dir"; - } - { - name = "lazy_static"; - packageId = "lazy_static"; - } - { - name = "mime"; - packageId = "mime 0.3.17"; - } - { - name = "mime_guess"; - packageId = "mime_guess 2.0.5"; - } - { - name = "pretty_env_logger"; - packageId = "pretty_env_logger"; - } - { - name = "result-extended"; - packageId = "result-extended"; - } - { - name = "rusqlite"; - packageId = "rusqlite"; - } - { - name = "rusqlite_migration"; - packageId = "rusqlite_migration"; - features = [ "from-directory" ]; - } - { - name = "serde"; - packageId = "serde 1.0.217"; - } - { - name = "serde_json"; - packageId = "serde_json"; - } - { - name = "thiserror"; - packageId = "thiserror 2.0.11"; - } - { - name = "tokio"; - packageId = "tokio"; - features = [ "full" ]; - } - { - name = "tokio-stream"; - packageId = "tokio-stream"; - } - { - name = "tower-http"; - packageId = "tower-http"; - features = [ "cors" ]; - } - { - name = "typeshare"; - packageId = "typeshare"; - } - { - name = "urlencoding"; - packageId = "urlencoding"; - } - { - name = "uuid"; - packageId = "uuid 1.12.0"; - features = [ "v4" ]; - } - ]; - devDependencies = [ - { - name = "axum-test"; - packageId = "axum-test"; - } - { - name = "cool_asserts"; - packageId = "cool_asserts"; - } - ]; - }; "wait-timeout" = rec { crateName = "wait-timeout"; @@ -17505,7 +16208,7 @@ rec { } { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; } { name = "serde_json"; @@ -17598,6 +16301,29 @@ rec { }; resolvedDefaultFeatures = [ "default" "std" ]; }; + "wasi 0.13.3+wasi-0.2.2" = rec { + crateName = "wasi"; + version = "0.13.3+wasi-0.2.2"; + edition = "2021"; + sha256 = "1lnapbvdcvi3kc749wzqvwrpd483win2kicn1faa4dja38p6v096"; + authors = [ + "The Cranelift Project Developers" + ]; + dependencies = [ + { + name = "wit-bindgen-rt"; + packageId = "wit-bindgen-rt"; + features = [ "bitflags" ]; + } + ]; + features = { + "compiler_builtins" = [ "dep:compiler_builtins" ]; + "core" = [ "dep:core" ]; + "default" = [ "std" ]; + "rustc-dep-of-std" = [ "compiler_builtins" "core" "rustc-std-workspace-alloc" ]; + "rustc-std-workspace-alloc" = [ "dep:rustc-std-workspace-alloc" ]; + }; + }; "wasite" = rec { crateName = "wasite"; version = "0.1.0"; @@ -19474,7 +18200,7 @@ rec { ]; }; - "winnow" = rec { + "winnow 0.5.40" = rec { crateName = "winnow"; version = "0.5.40"; edition = "2021"; @@ -19496,6 +18222,28 @@ rec { }; resolvedDefaultFeatures = [ "alloc" "default" "std" ]; }; + "winnow 0.7.3" = rec { + crateName = "winnow"; + version = "0.7.3"; + edition = "2021"; + sha256 = "1c9bmhpdwbdmll6b4l6skabz0296dchnmnxw84hh2y3ggyllwzqf"; + dependencies = [ + { + name = "memchr"; + packageId = "memchr"; + optional = true; + usesDefaultFeatures = false; + } + ]; + features = { + "debug" = [ "std" "dep:anstream" "dep:anstyle" "dep:is-terminal" "dep:terminal_size" ]; + "default" = [ "std" ]; + "simd" = [ "dep:memchr" ]; + "std" = [ "alloc" "memchr?/std" ]; + "unstable-doc" = [ "alloc" "std" "simd" "unstable-recover" ]; + }; + resolvedDefaultFeatures = [ "alloc" "default" "std" ]; + }; "winreg" = rec { crateName = "winreg"; version = "0.50.0"; @@ -19521,6 +18269,24 @@ rec { "serialization-serde" = [ "transactions" "serde" ]; }; }; + "wit-bindgen-rt" = rec { + crateName = "wit-bindgen-rt"; + version = "0.33.0"; + edition = "2021"; + sha256 = "0g4lwfp9x6a2i1hgjn8k14nr4fsnpd5izxhc75zpi2s5cvcg6s1j"; + libName = "wit_bindgen_rt"; + dependencies = [ + { + name = "bitflags"; + packageId = "bitflags 2.8.0"; + optional = true; + } + ]; + features = { + "bitflags" = [ "dep:bitflags" ]; + }; + resolvedDefaultFeatures = [ "bitflags" ]; + }; "write16" = rec { crateName = "write16"; version = "1.0.0"; @@ -19544,23 +18310,17 @@ rec { "either" = [ "dep:either" ]; }; }; - "yansi" = rec { - crateName = "yansi"; - version = "1.0.1"; + "xml-rs" = rec { + crateName = "xml-rs"; + version = "0.8.25"; edition = "2021"; - sha256 = "0jdh55jyv0dpd38ij4qh60zglbw9aa8wafqai6m0wa7xaxk3mrfg"; + crateBin = []; + sha256 = "1i73ajf6scni5bi1a51r19xykgrambdx5fkks0fyg5jqqbml1ff5"; + libName = "xml"; authors = [ - "Sergio Benitez <sb@sergio.bz>" + "Vladimir Matveev <vmatveev@citrine.cc>" ]; - features = { - "default" = [ "std" ]; - "detect-env" = [ "std" ]; - "detect-tty" = [ "is-terminal" "std" ]; - "hyperlink" = [ "std" ]; - "is-terminal" = [ "dep:is-terminal" ]; - "std" = [ "alloc" ]; - }; - resolvedDefaultFeatures = [ "alloc" "default" "std" ]; + }; "yansi-term" = rec { crateName = "yansi-term"; @@ -19598,7 +18358,7 @@ rec { dependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; optional = true; usesDefaultFeatures = false; } @@ -19623,7 +18383,7 @@ rec { devDependencies = [ { name = "serde"; - packageId = "serde 1.0.217"; + packageId = "serde 1.0.218"; usesDefaultFeatures = false; } ]; diff --git a/crate-hashes.json b/crate-hashes.json index 47b533e..480f3ca 100644 --- a/crate-hashes.json +++ b/crate-hashes.json @@ -3,7 +3,6 @@ "registry+https://github.com/rust-lang/crates.io-index#adler2@2.0.0": "09r6drylvgy8vv8k20lnbvwq8gp09h7smfn6h1rxsy15pgh629si", "registry+https://github.com/rust-lang/crates.io-index#adler32@1.2.0": "0d7jq7jsjyhsgbhnfq5fvrlh9j0i9g1fqrl2735ibv5f75yjgqda", "registry+https://github.com/rust-lang/crates.io-index#adler@1.0.2": "1zim79cvzd5yrkzl3nyfx0avijwgk9fqv3yrscdy1cc79ih02qpj", - "registry+https://github.com/rust-lang/crates.io-index#ahash@0.8.11": "04chdfkls5xmhp1d48gnjsmglbqibizs3bpbj6rsj604m10si7g8", "registry+https://github.com/rust-lang/crates.io-index#aho-corasick@1.1.3": "05mrpkvdgp5d20y2p989f187ry9diliijgwrs254fs9s1m1x6q4f", "registry+https://github.com/rust-lang/crates.io-index#allocator-api2@0.2.21": "08zrzs022xwndihvzdn78yqarv2b9696y67i6h78nla3ww87jgb8", "registry+https://github.com/rust-lang/crates.io-index#android-tzdata@0.1.1": "1w7ynjxrfs97xg3qlcdns4kgfpwcdv824g611fq32cag4cdr96g9", @@ -15,7 +14,6 @@ "registry+https://github.com/rust-lang/crates.io-index#anstyle-wincon@3.0.7": "0kmf0fq4c8yribdpdpylzz1zccpy84hizmcsac3wrac1f7kk8dfa", "registry+https://github.com/rust-lang/crates.io-index#anstyle@1.0.10": "1yai2vppmd7zlvlrp9grwll60knrmscalf8l2qpfz8b7y5lkpk2m", "registry+https://github.com/rust-lang/crates.io-index#anyhow@1.0.95": "010vd1ki8w84dzgx6c81sc8qm9n02fxic1gkpv52zp4nwrn0kb1l", - "registry+https://github.com/rust-lang/crates.io-index#assert-json-diff@2.0.2": "04mg3w0rh3schpla51l18362hsirl23q93aisws2irrj32wg5r27", "registry+https://github.com/rust-lang/crates.io-index#async-channel@1.9.0": "0dbdlkzlncbibd3ij6y6jmvjd0cmdn48ydcfdpfhw09njd93r5c1", "registry+https://github.com/rust-lang/crates.io-index#async-channel@2.3.1": "0skvwxj6ysfc6d7bhczz9a2550260g62bm5gl0nmjxxyn007id49", "registry+https://github.com/rust-lang/crates.io-index#async-executor@1.13.1": "1v6w1dbvsmw6cs4dk4lxj5dvrikc6xi479wikwaab2qy3h09mjih", @@ -27,13 +25,8 @@ "registry+https://github.com/rust-lang/crates.io-index#async-trait@0.1.85": "0mm0gwad44zs7mna4a0m1z4dhzpmydfj73w4wm23c8xpnhrli4rz", "registry+https://github.com/rust-lang/crates.io-index#atoi@2.0.0": "0a05h42fggmy7h0ajjv6m7z72l924i7igbx13hk9d8pyign9k3gj", "registry+https://github.com/rust-lang/crates.io-index#atomic-waker@1.1.2": "1h5av1lw56m0jf0fd3bchxq8a30xv0b4wv8s4zkp4s0i7mfvs18m", - "registry+https://github.com/rust-lang/crates.io-index#auto-future@1.0.0": "0wykbakzh227vz6frx9p48zsq0wpswgmb7v3917m53m7gr2pw7iw", "registry+https://github.com/rust-lang/crates.io-index#autocfg@0.1.8": "0y4vw4l4izdxq1v0rrhvmlbqvalrqrmk60v1z0dqlgnlbzkl7phd", "registry+https://github.com/rust-lang/crates.io-index#autocfg@1.4.0": "09lz3by90d2hphbq56znag9v87gfpd9gb8nr82hll8z6x2nhprdc", - "registry+https://github.com/rust-lang/crates.io-index#axum-core@0.4.5": "16b1496c4gm387q20hkv5ic3k5bd6xmnvk50kwsy6ymr8rhvvwh9", - "registry+https://github.com/rust-lang/crates.io-index#axum-macros@0.4.2": "1klv77c889jm05bzayaaiinalarhvh2crc2w4nvp3l581xaj7lap", - "registry+https://github.com/rust-lang/crates.io-index#axum-test@16.4.1": "1p5qxacvxsagnqq30nr2wznjyhgb8svsfb925ah3d2b0s91s9qv3", - "registry+https://github.com/rust-lang/crates.io-index#axum@0.7.9": "07z7wqczi9i8xb4460rvn39p4wjqwr32hx907crd1vwb2fy8ijpd", "registry+https://github.com/rust-lang/crates.io-index#az@1.2.1": "0ww9k1w3al7x5qmb7f13v3s9c2pg1pdxbs8xshqy6zyrchj4qzkv", "registry+https://github.com/rust-lang/crates.io-index#backtrace@0.3.74": "06pfif7nwx66qf2zaanc2fcq7m64i91ki9imw9xd3bnz5hrwp0ld", "registry+https://github.com/rust-lang/crates.io-index#base64@0.21.7": "0rw52yvsk75kar9wgqfwgb414kvil1gn7mqkrhn9zf1537mpsacx", @@ -53,7 +46,6 @@ "registry+https://github.com/rust-lang/crates.io-index#bytemuck@1.21.0": "18wj81x9xhqcd6985r8qxmbik6szjfjfj62q3xklw8h2p3x7srgg", "registry+https://github.com/rust-lang/crates.io-index#byteorder@1.5.0": "0jzncxyf404mwqdbspihyzpkndfgda450l0893pz5xj685cg5l0z", "registry+https://github.com/rust-lang/crates.io-index#bytes@1.9.0": "16ykzx24v1x4f42v2lxyvlczqhdfji3v7r4ghwckpwijzvb1hn9j", - "registry+https://github.com/rust-lang/crates.io-index#bytesize@1.3.0": "1k3aak70iwz4s2gsjbxf0ws4xnixqbdz6p2ha96s06748fpniqx3", "registry+https://github.com/rust-lang/crates.io-index#cairo-rs@0.18.5": "1qjfkcq3mrh3p01nnn71dy3kn99g21xx3j8xcdvzn8ll2pq6x8lc", "registry+https://github.com/rust-lang/crates.io-index#cairo-sys-rs@0.18.2": "0lfsxl7ylw3phbnwmz3k58j1gnqi6kc2hdc7g3bb7f4hwnl9yp38", "registry+https://github.com/rust-lang/crates.io-index#cc@1.2.10": "0aaj2ivamhfzhgb9maasnfkh03s2mzhzpzwrkghgzbkfnv5qy80k", @@ -64,9 +56,9 @@ "registry+https://github.com/rust-lang/crates.io-index#chrono-tz@0.8.6": "0vlksnmpb6rd4h55245agnfhphnpslwnq9al3aw3is43dd3f16nm", "registry+https://github.com/rust-lang/crates.io-index#chrono@0.4.39": "09g8nf409lb184kl9j4s85k0kn8wzgjkp5ls9zid50b886fwqdky", "registry+https://github.com/rust-lang/crates.io-index#clang-sys@1.8.1": "1x1r9yqss76z8xwpdanw313ss6fniwc1r7dzb5ycjn0ph53kj0hb", - "registry+https://github.com/rust-lang/crates.io-index#clap@4.5.26": "10v7qvn90calfbhap1c4r249i5c7fbxj09fn3szfz9pkis85xsx8", - "registry+https://github.com/rust-lang/crates.io-index#clap_builder@4.5.26": "08f1mzcvi7zjhm7hvz6al4jnv70ccqhwiaq74hihlspwnl0iic4n", - "registry+https://github.com/rust-lang/crates.io-index#clap_derive@4.5.24": "131ih3dm76srkbpfx7zfspp9b556zgzj31wqhl0ji2b39lcmbdsl", + "registry+https://github.com/rust-lang/crates.io-index#clap@4.5.30": "0vcyrn4ymq2gd56sl3xnfki8q8llg64sj3rj3qx33mgsf66v3dwj", + "registry+https://github.com/rust-lang/crates.io-index#clap_builder@4.5.30": "0369xis2ar46icsaxqyy37976mlb62alzyx4j53k99vq2w3v4pd3", + "registry+https://github.com/rust-lang/crates.io-index#clap_derive@4.5.28": "1vgigkhljp3r8r5lwdrn1ij93nafmjwh8cx77nppb9plqsaysk5z", "registry+https://github.com/rust-lang/crates.io-index#clap_lex@0.7.4": "19nwfls5db269js5n822vkc8dw0wjq2h1wf0hgr06ld2g52d2spl", "registry+https://github.com/rust-lang/crates.io-index#cloudabi@0.0.3": "0kxcg83jlihy0phnd2g8c2c303px3l2p3pkjz357ll6llnd5pz6x", "registry+https://github.com/rust-lang/crates.io-index#color_quant@1.1.0": "12q1n427h2bbmmm1mnglr57jaz2dj9apk0plcxw7nwqiai7qjyrx", @@ -76,7 +68,6 @@ "registry+https://github.com/rust-lang/crates.io-index#convert_case@0.6.0": "1jn1pq6fp3rri88zyw6jlhwwgf6qiyc08d6gjv0qypgkl862n67c", "registry+https://github.com/rust-lang/crates.io-index#cookie-factory@0.3.3": "18mka6fk3843qq3jw1fdfvzyv05kx7kcmirfbs2vg2kbw9qzm1cq", "registry+https://github.com/rust-lang/crates.io-index#cookie@0.17.0": "096c52jg9iq4lfcps2psncswv33fc30mmnaa2sbzzcfcw71kgyvy", - "registry+https://github.com/rust-lang/crates.io-index#cookie@0.18.1": "0iy749flficrlvgr3hjmf3igr738lk81n5akzf4ym4cs6cxg7pjd", "registry+https://github.com/rust-lang/crates.io-index#cool_asserts@2.0.3": "1v18dg7ifx41k2f82j3gsnpm1fg9wk5s4zv7sf42c7pnad72b7zf", "registry+https://github.com/rust-lang/crates.io-index#core-foundation-sys@0.8.7": "12w8j73lazxmr1z0h98hf3z623kl8ms7g07jch7n4p8f9nwlhdkp", "registry+https://github.com/rust-lang/crates.io-index#core-foundation@0.9.4": "13zvbbj07yk3b61b8fhwfzhy35535a583irf23vlcg59j7h9bqci", @@ -94,7 +85,6 @@ "registry+https://github.com/rust-lang/crates.io-index#deflate@0.8.6": "0x6iqlayg129w63999kz97m279m0jj4x4sm6gkqlvmp73y70yxvk", "registry+https://github.com/rust-lang/crates.io-index#der@0.7.9": "1h4vzjfa1lczxdf8avfj9qlwh1qianqlxdy1g5rn762qnvkzhnzm", "registry+https://github.com/rust-lang/crates.io-index#deranged@0.3.11": "1d1ibqqnr5qdrpw8rclwrf1myn3wf0dygl04idf4j2s49ah6yaxl", - "registry+https://github.com/rust-lang/crates.io-index#diff@0.1.13": "1j0nzjxci2zqx63hdcihkp0a4dkdmzxd7my4m7zk6cjyfy34j9an", "registry+https://github.com/rust-lang/crates.io-index#digest@0.10.7": "14p2n6ih29x81akj097lvz7wi9b6b9hvls0lwrv7b6xwyy0s5ncy", "registry+https://github.com/rust-lang/crates.io-index#dimensioned@0.7.0": "09ky8s3higkf677lmyqg30hmj66gpg7hx907s6hfvbk2a9av05r5", "registry+https://github.com/rust-lang/crates.io-index#dimensioned@0.8.0": "15s3j4ry943xqlac63bp81sgdk9s3yilysabzww35j9ibmnaic50", @@ -110,8 +100,6 @@ "registry+https://github.com/rust-lang/crates.io-index#event-listener@2.5.3": "1q4w3pndc518crld6zsqvvpy9lkzwahp2zgza9kbzmmqh9gif1h2", "registry+https://github.com/rust-lang/crates.io-index#event-listener@5.4.0": "1bii2gn3vaa33s0gr2zph7cagiq0ppcfxcxabs24ri9z9kgar4il", "registry+https://github.com/rust-lang/crates.io-index#exr@1.73.0": "1q47yq78q9k210r6jy1wwrilxwwxqavik9l3l426rd17k7srfcgq", - "registry+https://github.com/rust-lang/crates.io-index#fallible-iterator@0.3.0": "0ja6l56yka5vn4y4pk6hn88z0bpny7a8k1919aqjzp0j1yhy9k1a", - "registry+https://github.com/rust-lang/crates.io-index#fallible-streaming-iterator@0.1.9": "0nj6j26p71bjy8h42x6jahx1hn0ng6mc2miwpgwnp8vnwqf4jq3k", "registry+https://github.com/rust-lang/crates.io-index#fastrand@2.3.0": "1ghiahsw1jd68df895cy5h3gzwk30hndidn3b682zmshpgmrx41p", "registry+https://github.com/rust-lang/crates.io-index#fdeflate@0.3.7": "130ga18vyxbb5idbgi07njymdaavvk6j08yh1dfarm294ssm6s0y", "registry+https://github.com/rust-lang/crates.io-index#field-offset@0.3.6": "0zq5sssaa2ckmcmxxbly8qgz3sxpb8g1lwv90sdh1z74qif2gqiq", @@ -146,6 +134,7 @@ "registry+https://github.com/rust-lang/crates.io-index#generic-array@0.11.2": "0a7w8w0rg47nmcinnfzv443lcyb8mplwc251p1jyr5xj2yh6wzv6", "registry+https://github.com/rust-lang/crates.io-index#generic-array@0.14.7": "16lyyrzrljfq424c3n8kfwkqihlimmsg5nhshbbp48np3yjrqr45", "registry+https://github.com/rust-lang/crates.io-index#getrandom@0.2.15": "1mzlnrb3dgyd1fb84gvw10pyr8wdqdl4ry4sr64i1s8an66pqmn4", + "registry+https://github.com/rust-lang/crates.io-index#getrandom@0.3.1": "1y154yzby383p63ndw6zpfm0fz3vf6c0zdwc7df6vkl150wrr923", "registry+https://github.com/rust-lang/crates.io-index#gif@0.11.4": "01hbw3isapzpzff8l6aw55jnaqx2bcscrbwyf3rglkbbfp397p9y", "registry+https://github.com/rust-lang/crates.io-index#gif@0.13.1": "1whrkvdg26gp1r7f95c6800y6ijqw5y0z8rgj6xihpi136dxdciz", "registry+https://github.com/rust-lang/crates.io-index#gimli@0.31.1": "0gvqc0ramx8szv76jhfd4dms0zyamvlg4whhiz11j34hh3dqxqh7", @@ -170,10 +159,8 @@ "registry+https://github.com/rust-lang/crates.io-index#gtk4@0.7.3": "0hh8nzglmz94v1m1h6vy8z12m6fr7ia467ry0md5fa4p7sm53sss", "registry+https://github.com/rust-lang/crates.io-index#h2@0.3.26": "1s7msnfv7xprzs6xzfj5sg6p8bjcdpcqcmjjbkd345cyi1x55zl1", "registry+https://github.com/rust-lang/crates.io-index#half@2.4.1": "123q4zzw1x4309961i69igzd1wb7pj04aaii3kwasrz3599qrl3d", - "registry+https://github.com/rust-lang/crates.io-index#hashbrown@0.14.5": "1wa1vy1xs3mp11bn3z9dv0jricgr6a2j0zkf1g19yz3vw4il89z5", "registry+https://github.com/rust-lang/crates.io-index#hashbrown@0.15.2": "12dj0yfn59p3kh3679ac0w1fagvzf4z2zp87a13gbbqbzw0185dz", "registry+https://github.com/rust-lang/crates.io-index#hashlink@0.10.0": "1h8lzvnl9qxi3zyagivzz2p1hp6shgddfmccyf6jv7s1cdicz0kk", - "registry+https://github.com/rust-lang/crates.io-index#hashlink@0.9.1": "1byq4nyrflm5s6wdx5qwp96l1qbp2d0nljvrr5yqrsfy51qzz93b", "registry+https://github.com/rust-lang/crates.io-index#headers-core@0.2.0": "0ab469xfpd411mc3dhmjhmzrhqikzyj8a17jn5bkj9zfpy0n9xp7", "registry+https://github.com/rust-lang/crates.io-index#headers@0.3.9": "0w62gnwh2p1lml0zqdkrx9dp438881nhz32zrzdy61qa0a9kns06", "registry+https://github.com/rust-lang/crates.io-index#heck@0.4.1": "1a7mqsnycv5z4z5vnv1k34548jzmc0ajic7c1j8jsaspnhw5ql4m", @@ -185,19 +172,15 @@ "registry+https://github.com/rust-lang/crates.io-index#hkdf@0.12.4": "1xxxzcarz151p1b858yn5skmhyrvn8fs4ivx5km3i1kjmnr8wpvv", "registry+https://github.com/rust-lang/crates.io-index#hmac@0.12.1": "0pmbr069sfg76z7wsssfk5ddcqd9ncp79fyz6zcm6yn115yc6jbc", "registry+https://github.com/rust-lang/crates.io-index#home@0.5.11": "1kxb4k87a9sayr8jipr7nq9wpgmjk4hk4047hmf9kc24692k75aq", - "registry+https://github.com/rust-lang/crates.io-index#http-body-util@0.1.2": "0kslwazg4400qnc2azkrgqqci0fppv12waicnsy5d8hncvbjjd3r", "registry+https://github.com/rust-lang/crates.io-index#http-body@0.4.6": "1lmyjfk6bqk6k9gkn1dxq770sb78pqbqshga241hr5p995bb5skw", - "registry+https://github.com/rust-lang/crates.io-index#http-body@1.0.1": "111ir5k2b9ihz5nr9cz7cwm7fnydca7dx4hc7vr16scfzghxrzhy", "registry+https://github.com/rust-lang/crates.io-index#http@0.2.12": "1w81s4bcbmcj9bjp7mllm8jlz6b31wzvirz8bgpzbqkpwmbvn730", "registry+https://github.com/rust-lang/crates.io-index#http@1.2.0": "1skglzdf98j5nzxlii540n11is0w4l80mi5sm3xrj716asps4v7i", "registry+https://github.com/rust-lang/crates.io-index#httparse@1.9.5": "0ip9v8m9lvgvq1lznl31wvn0ch1v254na7lhid9p29yx9rbx6wbx", "registry+https://github.com/rust-lang/crates.io-index#httpdate@1.0.3": "1aa9rd2sac0zhjqh24c9xvir96g188zldkx0hr6dnnlx5904cfyz", "registry+https://github.com/rust-lang/crates.io-index#humantime@2.1.0": "1r55pfkkf5v0ji1x6izrjwdq9v6sc7bv99xj6srywcar37xmnfls", "registry+https://github.com/rust-lang/crates.io-index#hyper-tls@0.5.0": "01crgy13102iagakf6q4mb75dprzr7ps1gj0l5hxm1cvm7gks66n", - "registry+https://github.com/rust-lang/crates.io-index#hyper-util@0.1.10": "1d1iwrkysjhq63pg54zk3vfby1j7zmxzm9zzyfr4lwvp0szcybfz", "registry+https://github.com/rust-lang/crates.io-index#hyper@0.10.16": "0wwjh9p3mzvg3fss2lqz5r7ddcgl1fh9w6my2j69d6k0lbcm41ha", "registry+https://github.com/rust-lang/crates.io-index#hyper@0.14.32": "1rvcb0smz8q1i0y6p7rwxr02x5sclfg2hhxf3g0774zczn0cgps1", - "registry+https://github.com/rust-lang/crates.io-index#hyper@1.5.2": "1q7akfb443yrjzkmnnbp2vs8zi15hgbk466rr4y144v4ppabhvr5", "registry+https://github.com/rust-lang/crates.io-index#iana-time-zone-haiku@0.1.2": "17r6jmj31chn7xs9698r122mapq85mfnv98bb4pg6spm0si2f67k", "registry+https://github.com/rust-lang/crates.io-index#iana-time-zone@0.1.61": "085jjsls330yj1fnwykfzmb2f10zp6l7w4fhq81ng81574ghhpi3", "registry+https://github.com/rust-lang/crates.io-index#icu_collections@1.5.0": "09j5kskirl59mvqc8kabhy7005yyy7dp88jw9f6f3gkf419a8byv", @@ -215,8 +198,6 @@ "registry+https://github.com/rust-lang/crates.io-index#idna_adapter@1.2.0": "0wggnkiivaj5lw0g0384ql2d7zk4ppkn3b1ry4n0ncjpr7qivjns", "registry+https://github.com/rust-lang/crates.io-index#image@0.23.14": "18gn2f7xp30pf9aqka877knlq308khxqiwjvsccvzaa4f9zcpzr4", "registry+https://github.com/rust-lang/crates.io-index#image@0.24.9": "17gnr6ifnpzvhjf6dwbl9hki8x6bji5mwcqp0048x1jm5yfi742n", - "registry+https://github.com/rust-lang/crates.io-index#include_dir@0.7.4": "1pfh3g45z88kwq93skng0n6g3r7zkhq9ldqs9y8rvr7i11s12gcj", - "registry+https://github.com/rust-lang/crates.io-index#include_dir_macros@0.7.4": "0x8smnf6knd86g69p19z5lpfsaqp8w0nx14kdpkz1m8bxnkqbavw", "registry+https://github.com/rust-lang/crates.io-index#indent_write@2.2.0": "1hqjp80argdskrhd66g9sh542yxy8qi77j6rc69qd0l7l52rdzhc", "registry+https://github.com/rust-lang/crates.io-index#indexmap@2.7.0": "07s7jmdymvd0rm4yswp0j3napx57hkjm9gs9n55lvs2g78vj5y32", "registry+https://github.com/rust-lang/crates.io-index#intl-memoizer@0.5.2": "1nkvql7c7b76axv4g68di1p2m9bnxq1cbn6mlqcawf72zhhf08py", @@ -251,7 +232,6 @@ "registry+https://github.com/rust-lang/crates.io-index#log@0.4.25": "17ydv5zhfv1zzygy458bmg3f3jx1vfziv9d74817w76yhfqgbjq4", "registry+https://github.com/rust-lang/crates.io-index#logger@0.4.0": "14xlxvkspcfnspjil0xi63qj5cybxn1hjmr5gq8m4v1g9k5p54bc", "registry+https://github.com/rust-lang/crates.io-index#matches@0.1.10": "1994402fq4viys7pjhzisj4wcw894l53g798kkm2y74laxk0jci5", - "registry+https://github.com/rust-lang/crates.io-index#matchit@0.7.3": "156bgdmmlv4crib31qhgg49nsjk88dxkdqp80ha2pk2rk6n6ax0f", "registry+https://github.com/rust-lang/crates.io-index#md-5@0.10.6": "1kvq5rnpm4fzwmyv5nmnxygdhhb2369888a06gdc9pxyrzh7x7nq", "registry+https://github.com/rust-lang/crates.io-index#memchr@2.7.4": "18z32bhxrax0fnjikv475z7ii718hq457qwmaryixfxsl2qrmjkq", "registry+https://github.com/rust-lang/crates.io-index#memoffset@0.9.1": "12i17wh9a9plx869g7j4whf62xw68k5zd4k0k5nh6ys5mszid028", @@ -317,10 +297,9 @@ "registry+https://github.com/rust-lang/crates.io-index#polling@3.7.4": "0bs4nhwfwsvlzlhah2gbhj3aa9ynvchv2g350wapswh26a65c156", "registry+https://github.com/rust-lang/crates.io-index#powerfmt@0.2.0": "14ckj2xdpkhv3h6l5sdmb9f1d57z8hbfpdldjc2vl5givq2y77j3", "registry+https://github.com/rust-lang/crates.io-index#ppv-lite86@0.2.20": "017ax9ssdnpww7nrl1hvqh2lzncpv04nnsibmnw9nxjnaqlpp5bp", - "registry+https://github.com/rust-lang/crates.io-index#pretty_assertions@1.4.1": "0v8iq35ca4rw3rza5is3wjxwsf88303ivys07anc5yviybi31q9s", "registry+https://github.com/rust-lang/crates.io-index#pretty_env_logger@0.5.0": "076w9dnvcpx6d3mdbkqad8nwnsynb7c8haxmscyrz7g3vga28mw6", "registry+https://github.com/rust-lang/crates.io-index#proc-macro-crate@1.3.1": "069r1k56bvgk0f58dm5swlssfcp79im230affwk6d9ck20g04k3z", - "registry+https://github.com/rust-lang/crates.io-index#proc-macro-crate@2.0.2": "092x5acqnic14cw6vacqap5kgknq3jn4c6jij9zi6j85839jc3xh", + "registry+https://github.com/rust-lang/crates.io-index#proc-macro-crate@2.0.0": "1s23imns07vmacn2xjd5hv2h6rr94iqq3fd2frwa6i4h2nk6d0vy", "registry+https://github.com/rust-lang/crates.io-index#proc-macro-error-attr@1.0.4": "0sgq6m5jfmasmwwy8x4mjygx5l7kp8s4j60bv25ckv2j1qc41gm1", "registry+https://github.com/rust-lang/crates.io-index#proc-macro-error@1.0.4": "1373bhxaf0pagd8zkyd03kkx6bchzf6g0dkwrwzsnal9z47lj9fs", "registry+https://github.com/rust-lang/crates.io-index#proc-macro2@1.0.93": "169dw9wch753if1mgyi2nfl1il77gslvh6y2q46qplprwml6m530", @@ -353,11 +332,7 @@ "registry+https://github.com/rust-lang/crates.io-index#regex@1.11.1": "148i41mzbx8bmq32hsj1q4karkzzx5m60qza6gdw4pdc9qdyyi5m", "registry+https://github.com/rust-lang/crates.io-index#remove_dir_all@0.5.3": "1rzqbsgkmr053bxxl04vmvsd1njyz0nxvly97aip6aa2cmb15k9s", "registry+https://github.com/rust-lang/crates.io-index#reqwest@0.11.27": "0qjary4hpplpgdi62d2m0xvbn6lnzckwffm0rgkm2x51023m6ryx", - "registry+https://github.com/rust-lang/crates.io-index#reserve-port@2.0.1": "10x21rdb1hjzp6n5flbbw3hfd7brmirckz1q0zsf3a7s5d516f4q", "registry+https://github.com/rust-lang/crates.io-index#rsa@0.9.7": "06amqm85raq26v6zg00fbf93lbj3kx559n2lpxc3wrvbbiy5vis7", - "registry+https://github.com/rust-lang/crates.io-index#rusqlite@0.32.1": "0vlx040bppl414pbjgbp7qr4jdxwszi9krx0m63zzf2f2whvflvp", - "registry+https://github.com/rust-lang/crates.io-index#rusqlite_migration@1.3.1": "076dm65g0sngzrb93r07va4l5zl3gjx9gq5mlsh21p7p0bl44fwj", - "registry+https://github.com/rust-lang/crates.io-index#rust-multipart-rfc7578_2@0.6.1": "0mwd3i2mk91n6diaxnkw28vyjbifhrm5ls73pcpfzz8a1i0lidq3", "registry+https://github.com/rust-lang/crates.io-index#rustc-demangle@0.1.24": "07zysaafgrkzy2rjgwqdj2a8qdpsm6zv6f5pgpk9x0lm40z9b6vi", "registry+https://github.com/rust-lang/crates.io-index#rustc-hash@1.1.0": "1qkc5khrmv5pqi5l5ca9p5nl5hs742cagrndhbrlk3dhlrx3zm08", "registry+https://github.com/rust-lang/crates.io-index#rustc_version@0.4.1": "14lvdsmr5si5qbqzrajgb6vfn69k0sfygrvfvr2mps26xwi3mjyg", @@ -377,10 +352,9 @@ "registry+https://github.com/rust-lang/crates.io-index#self_cell@1.1.0": "1gmxk5bvnnimcif7v1jk8ai2azfvh9djki545nd86vsnphjgrzf2", "registry+https://github.com/rust-lang/crates.io-index#semver@1.0.24": "1fmvjjkd3f64y5fqr1nakkq371mnwzv09fbz5mbmdxril63ypdiw", "registry+https://github.com/rust-lang/crates.io-index#serde@0.9.15": "1bsla8l5xr9pp5sirkal6mngxcq6q961km88jvf339j5ff8j7dil", - "registry+https://github.com/rust-lang/crates.io-index#serde@1.0.217": "0w2ck1p1ajmrv1cf51qf7igjn2nc51r0izzc00fzmmhkvxjl5z02", - "registry+https://github.com/rust-lang/crates.io-index#serde_derive@1.0.217": "180r3rj5gi5s1m23q66cr5wlfgc5jrs6n1mdmql2njnhk37zg6ss", - "registry+https://github.com/rust-lang/crates.io-index#serde_json@1.0.136": "1lipcjhh1zazh283i4wsl4l14knh81q2rlkwmag8v8s2rwihqsik", - "registry+https://github.com/rust-lang/crates.io-index#serde_path_to_error@0.1.16": "19hlz2359l37ifirskpcds7sxg0gzpqvfilibs7whdys0128i6dg", + "registry+https://github.com/rust-lang/crates.io-index#serde@1.0.218": "0q6z4bnrwagnms0bds4886711l6mc68s979i49zd3xnvkg8wkpz8", + "registry+https://github.com/rust-lang/crates.io-index#serde_derive@1.0.218": "0azqd74xbpb1v5vf6w1fdbgmwp39ljjfj25cib5rgrzlj7hh75gh", + "registry+https://github.com/rust-lang/crates.io-index#serde_json@1.0.139": "19kj3irpa22a7djz1jaf4wambzh7psiqa6zyafqnb76crhx6ry24", "registry+https://github.com/rust-lang/crates.io-index#serde_spanned@0.6.8": "1q89g70azwi4ybilz5jb8prfpa575165lmrffd49vmcf76qpqq47", "registry+https://github.com/rust-lang/crates.io-index#serde_urlencoded@0.7.1": "1zgklbdaysj3230xivihs30qi5vkhigg323a9m62k8jwf4a1qjfk", "registry+https://github.com/rust-lang/crates.io-index#serde_yml@0.0.12": "1p8xwz4znd6fj962y22fdvvv16gb8c0hx4iv5hjplngiidcdvqjr", @@ -412,14 +386,13 @@ "registry+https://github.com/rust-lang/crates.io-index#syn@1.0.109": "0ds2if4600bd59wsv7jjgfkayfzy3hnazs394kz6zdkmna8l3dkj", "registry+https://github.com/rust-lang/crates.io-index#syn@2.0.96": "102wk3cgawimi3i0q3r3xw3i858zkyingg6y7gsxfy733amsvl6m", "registry+https://github.com/rust-lang/crates.io-index#sync_wrapper@0.1.2": "0q01lyj0gr9a93n10nxsn8lwbzq97jqd6b768x17c8f7v7gccir0", - "registry+https://github.com/rust-lang/crates.io-index#sync_wrapper@1.0.2": "0qvjyasd6w18mjg5xlaq5jgy84jsjfsvmnn12c13gypxbv75dwhb", "registry+https://github.com/rust-lang/crates.io-index#synstructure@0.13.1": "0wc9f002ia2zqcbj0q2id5x6n7g1zjqba7qkg2mr0qvvmdk7dby8", "registry+https://github.com/rust-lang/crates.io-index#system-configuration-sys@0.5.0": "1jckxvdr37bay3i9v52izgy52dg690x5xfg3hd394sv2xf4b2px7", "registry+https://github.com/rust-lang/crates.io-index#system-configuration@0.5.1": "1rz0r30xn7fiyqay2dvzfy56cvaa3km74hnbz2d72p97bkf3lfms", "registry+https://github.com/rust-lang/crates.io-index#system-deps@6.2.2": "0j93ryw031n3h8b0nfpj5xwh3ify636xmv8kxianvlyyipmkbrd3", "registry+https://github.com/rust-lang/crates.io-index#target-lexicon@0.12.16": "1cg3bnx1gdkdr5hac1hzxy64fhw4g7dqkd0n3dxy5lfngpr1mi31", "registry+https://github.com/rust-lang/crates.io-index#tempdir@0.3.7": "1n5n86zxpgd85y0mswrp5cfdisizq2rv3la906g6ipyc03xvbwhm", - "registry+https://github.com/rust-lang/crates.io-index#tempfile@3.15.0": "016pmkbwn3shas44gcwq1kc9lajalb90qafhiip5fvv8h6f5b2ls", + "registry+https://github.com/rust-lang/crates.io-index#tempfile@3.17.1": "0c52ggq5vy5mzgk5ly36cgzs1cig3cv6r1jarijmzxgkn6na1r92", "registry+https://github.com/rust-lang/crates.io-index#termcolor@1.4.1": "0mappjh3fj3p2nmrg4y7qv94rchwi9mzmgmfflr8p2awdj7lyy86", "registry+https://github.com/rust-lang/crates.io-index#thiserror-impl@1.0.69": "1h84fmn2nai41cxbhk6pqf46bxqq1b344v8yz089w1chzi76rvjg", "registry+https://github.com/rust-lang/crates.io-index#thiserror-impl@2.0.11": "1hkkn7p2y4cxbffcrprybkj0qy1rl1r6waxmxqvr764axaxc3br6", @@ -440,14 +413,12 @@ "registry+https://github.com/rust-lang/crates.io-index#tokio-tungstenite@0.21.0": "0f5wj0crsx74rlll97lhw0wk6y12nhdnqvmnjx002hjn08fmcfy8", "registry+https://github.com/rust-lang/crates.io-index#tokio-util@0.7.13": "0y0h10a52c7hrldmr3410bp7j3fadq0jn9nf7awddgd2an6smz6p", "registry+https://github.com/rust-lang/crates.io-index#tokio@1.43.0": "17pdm49ihlhfw3rpxix3kdh2ppl1yv7nwp1kxazi5r1xz97zlq9x", - "registry+https://github.com/rust-lang/crates.io-index#toml@0.8.2": "0g9ysjaqvm2mv8q85xpqfn7hi710hj24sd56k49wyddvvyq8lp8q", - "registry+https://github.com/rust-lang/crates.io-index#toml_datetime@0.6.3": "0jsy7v8bdvmzsci6imj8fzgd255fmy5fzp6zsri14yrry7i77nkw", + "registry+https://github.com/rust-lang/crates.io-index#toml@0.8.20": "0j012b37iz1mihksr6a928s6dzszxvblzg3l5wxp7azzsv6sb1yd", + "registry+https://github.com/rust-lang/crates.io-index#toml_datetime@0.6.8": "0hgv7v9g35d7y9r2afic58jvlwnf73vgd1mz2k8gihlgrf73bmqd", "registry+https://github.com/rust-lang/crates.io-index#toml_edit@0.19.15": "08bl7rp5g6jwmfpad9s8jpw8wjrciadpnbaswgywpr9hv9qbfnqv", "registry+https://github.com/rust-lang/crates.io-index#toml_edit@0.20.2": "0f7k5svmxw98fhi28jpcyv7ldr2s3c867pjbji65bdxjpd44svir", - "registry+https://github.com/rust-lang/crates.io-index#tower-http@0.6.2": "15wnvhl6cpir9125s73bqjzjsvfb0fmndmsimnl2ddnlhfvs6gs0", - "registry+https://github.com/rust-lang/crates.io-index#tower-layer@0.3.3": "03kq92fdzxin51w8iqix06dcfgydyvx7yr6izjq0p626v9n2l70j", + "registry+https://github.com/rust-lang/crates.io-index#toml_edit@0.22.24": "0x0lgp70x5cl9nla03xqs5vwwwlrwmd0djkdrp3h3lpdymgpkd0p", "registry+https://github.com/rust-lang/crates.io-index#tower-service@0.3.3": "1hzfkvkci33ra94xjx64vv3pp0sq346w06fpkcdwjcid7zhvdycd", - "registry+https://github.com/rust-lang/crates.io-index#tower@0.5.2": "1ybmd59nm4abl9bsvy6rx31m4zvzp5rja2slzpn712y9b68ssffh", "registry+https://github.com/rust-lang/crates.io-index#tracing-attributes@0.1.28": "0v92l9cxs42rdm4m5hsa8z7ln1xsiw1zc2iil8c6k7lzq0jf2nir", "registry+https://github.com/rust-lang/crates.io-index#tracing-core@0.1.33": "170gc7cxyjx824r9kr17zc9gvzx89ypqfdzq259pr56gg5bwjwp6", "registry+https://github.com/rust-lang/crates.io-index#tracing@0.1.41": "1l5xrzyjfyayrwhvhldfnwdyligi1mpqm8mzbi2m1d6y6p2hlkkq", @@ -474,7 +445,6 @@ "registry+https://github.com/rust-lang/crates.io-index#unsafe-any@0.4.2": "0zwwphsqkw5qaiqmjwngnfpv9ym85qcsyj7adip9qplzjzbn00zk", "registry+https://github.com/rust-lang/crates.io-index#url@1.7.2": "0nim1c90mxpi9wgdw2xh8dqd72vlklwlzam436akcrhjac6pqknx", "registry+https://github.com/rust-lang/crates.io-index#url@2.5.4": "0q6sgznyy2n4l5lm16zahkisvc9nip9aa5q1pps7656xra3bdy1j", - "registry+https://github.com/rust-lang/crates.io-index#urlencoding@2.1.3": "1nj99jp37k47n0hvaz5fvz7z6jd0sb4ppvfy3nphr1zbnyixpy6s", "registry+https://github.com/rust-lang/crates.io-index#utf-8@0.7.6": "1a9ns3fvgird0snjkd3wbdhwd3zdpc2h5gpyybrfr6ra5pkqxk09", "registry+https://github.com/rust-lang/crates.io-index#utf16_iter@1.0.5": "0ik2krdr73hfgsdzw0218fn35fa09dg2hvbi1xp3bmdfrp9js8y8", "registry+https://github.com/rust-lang/crates.io-index#utf8_iter@1.0.4": "1gmna9flnj8dbyd8ba17zigrp9c4c3zclngf5lnb5yvz1ri41hdn", @@ -492,6 +462,7 @@ "registry+https://github.com/rust-lang/crates.io-index#warp@0.3.7": "07137zd13lchy5hxpspd0hs6sl19b0fv2zc1chf02nwnzw1d4y23", "registry+https://github.com/rust-lang/crates.io-index#wasi@0.10.0+wasi-snapshot-preview1": "07y3l8mzfzzz4cj09c8y90yak4hpsi9g7pllyzpr6xvwrabka50s", "registry+https://github.com/rust-lang/crates.io-index#wasi@0.11.0+wasi-snapshot-preview1": "08z4hxwkpdpalxjps1ai9y7ihin26y9f476i53dv98v45gkqg3cw", + "registry+https://github.com/rust-lang/crates.io-index#wasi@0.13.3+wasi-0.2.2": "1lnapbvdcvi3kc749wzqvwrpd483win2kicn1faa4dja38p6v096", "registry+https://github.com/rust-lang/crates.io-index#wasite@0.1.0": "0nw5h9nmcl4fyf4j5d4mfdjfgvwi1cakpi349wc4zrr59wxxinmq", "registry+https://github.com/rust-lang/crates.io-index#wasm-bindgen-backend@0.2.100": "1ihbf1hq3y81c4md9lyh6lcwbx6a5j0fw4fygd423g62lm8hc2ig", "registry+https://github.com/rust-lang/crates.io-index#wasm-bindgen-futures@0.4.50": "0q8ymi6i9r3vxly551dhxcyai7nc491mspj0j1wbafxwq074fpam", @@ -528,11 +499,13 @@ "registry+https://github.com/rust-lang/crates.io-index#windows_x86_64_msvc@0.48.5": "0f4mdp895kkjh9zv8dxvn4pc10xr7839lf5pa9l0193i2pkgr57d", "registry+https://github.com/rust-lang/crates.io-index#windows_x86_64_msvc@0.52.6": "1v7rb5cibyzx8vak29pdrk8nx9hycsjs4w0jgms08qk49jl6v7sq", "registry+https://github.com/rust-lang/crates.io-index#winnow@0.5.40": "0xk8maai7gyxda673mmw3pj1hdizy5fpi7287vaywykkk19sk4zm", + "registry+https://github.com/rust-lang/crates.io-index#winnow@0.7.3": "1c9bmhpdwbdmll6b4l6skabz0296dchnmnxw84hh2y3ggyllwzqf", "registry+https://github.com/rust-lang/crates.io-index#winreg@0.50.0": "1cddmp929k882mdh6i9f2as848f13qqna6czwsqzkh1pqnr5fkjj", + "registry+https://github.com/rust-lang/crates.io-index#wit-bindgen-rt@0.33.0": "0g4lwfp9x6a2i1hgjn8k14nr4fsnpd5izxhc75zpi2s5cvcg6s1j", "registry+https://github.com/rust-lang/crates.io-index#write16@1.0.0": "0dnryvrrbrnl7vvf5vb1zkmwldhjkf2n5znliviam7bm4900z2fi", "registry+https://github.com/rust-lang/crates.io-index#writeable@0.5.5": "0lawr6y0bwqfyayf3z8zmqlhpnzhdx0ahs54isacbhyjwa7g778y", + "registry+https://github.com/rust-lang/crates.io-index#xml-rs@0.8.25": "1i73ajf6scni5bi1a51r19xykgrambdx5fkks0fyg5jqqbml1ff5", "registry+https://github.com/rust-lang/crates.io-index#yansi-term@0.1.2": "1w8vjlvxba6yvidqdvxddx3crl6z66h39qxj8xi6aqayw2nk0p7y", - "registry+https://github.com/rust-lang/crates.io-index#yansi@1.0.1": "0jdh55jyv0dpd38ij4qh60zglbw9aa8wafqai6m0wa7xaxk3mrfg", "registry+https://github.com/rust-lang/crates.io-index#yoke-derive@0.7.5": "0m4i4a7gy826bfvnqa9wy6sp90qf0as3wps3wb0smjaamn68g013", "registry+https://github.com/rust-lang/crates.io-index#yoke@0.7.5": "0h3znzrdmll0a7sglzf9ji0p5iqml11wrj1dypaf6ad6kbpnl3hj", "registry+https://github.com/rust-lang/crates.io-index#zerocopy-derive@0.7.35": "0gnf2ap2y92nwdalzz3x7142f2b83sni66l39vxp2ijd6j080kzs", diff --git a/flake.lock b/flake.lock index 7fc0bd8..faf7ce0 100644 --- a/flake.lock +++ b/flake.lock @@ -1,5 +1,20 @@ { "nodes": { + "crane": { + "locked": { + "lastModified": 1739936662, + "narHash": "sha256-x4syUjNUuRblR07nDPeLDP7DpphaBVbUaSoeZkFbGSk=", + "owner": "ipetkov", + "repo": "crane", + "rev": "19de14aaeb869287647d9461cbd389187d8ecdb7", + "type": "github" + }, + "original": { + "owner": "ipetkov", + "repo": "crane", + "type": "github" + } + }, "flake-utils": { "inputs": { "systems": "systems" @@ -20,16 +35,16 @@ }, "nixpkgs": { "locked": { - "lastModified": 1704732714, - "narHash": "sha256-ABqK/HggMYA/jMUXgYyqVAcQ8QjeMyr1jcXfTpSHmps=", + "lastModified": 1740339700, + "narHash": "sha256-cbrw7EgQhcdFnu6iS3vane53bEagZQy/xyIkDWpCgVE=", "owner": "NixOS", "repo": "nixpkgs", - "rev": "6723fa4e4f1a30d42a633bef5eb01caeb281adc3", + "rev": "04ef94c4c1582fd485bbfdb8c4a8ba250e359195", "type": "github" }, "original": { "id": "nixpkgs", - "ref": "nixos-23.11", + "ref": "nixos-24.11", "type": "indirect" } }, @@ -50,6 +65,7 @@ }, "root": { "inputs": { + "crane": "crane", "nixpkgs": "nixpkgs", "typeshare": "typeshare", "unstable": "unstable" diff --git a/flake.nix b/flake.nix index 90b1d89..970fa21 100644 --- a/flake.nix +++ b/flake.nix @@ -2,12 +2,13 @@ description = "Lumenescent Dreams Tools"; inputs = { - nixpkgs.url = "nixpkgs/nixos-23.11"; + nixpkgs.url = "nixpkgs/nixos-24.11"; unstable.url = "nixpkgs/nixos-unstable"; typeshare.url = "github:1Password/typeshare"; + crane.url = "github:ipetkov/crane"; }; - outputs = { self, nixpkgs, unstable, typeshare, ... }: + outputs = { self, nixpkgs, unstable, typeshare, crane, ... }: let version = builtins.string 0 8 self.lastModifiedDate; supportedSystems = [ "x86_64-linux" ]; @@ -56,6 +57,8 @@ packages."x86_64-linux" = let pkgs = import nixpkgs { system = "x86_64-linux"; }; + craneLib = crane.mkLib pkgs; + src = craneLib.cleanCargoSource ./.; gtkNativeInputs = [ pkgs.pkg-config @@ -88,6 +91,11 @@ dashboard = cargo_nix.workspaceMembers.dashboard.build; # file-service = cargo_nix.workspaceMembers.file-service.build; fitnesstrax = cargo_nix.workspaceMembers.fitnesstrax.build; + l10n-db = craneLib.buildPackage { + pname = "l10n-db"; + cargoExtraArgs = "-p l10n-db"; + src = ./.; + }; otg-gtk = cargo_nix.workspaceMembers.otg-gtk.build; all = pkgs.symlinkJoin { @@ -98,6 +106,7 @@ dashboard # file-service fitnesstrax + l10n-db otg-gtk ]; }; diff --git a/l10n-db/src/bin/main.rs b/l10n-db/src/bin/l10n-db.rs similarity index 100% rename from l10n-db/src/bin/main.rs rename to l10n-db/src/bin/l10n-db.rs From 47e90cc6f946859f79de7d7e87e6043bae3dfc2e Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sun, 10 Nov 2024 19:04:38 -0500 Subject: [PATCH 17/34] Set up an app for the adafruit TFT --- Cargo.lock | 1030 ++++++++++++++++++++++++-------- Cargo.toml | 10 +- pico-st7789/.cargo/config.toml | 11 + pico-st7789/Cargo.toml | 11 + pico-st7789/src/main.rs | 225 +++++++ 5 files changed, 1050 insertions(+), 237 deletions(-) create mode 100644 pico-st7789/.cargo/config.toml create mode 100644 pico-st7789/Cargo.toml create mode 100644 pico-st7789/src/main.rs diff --git a/Cargo.lock b/Cargo.lock index b4bc05f..bb828ca 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -121,9 +121,15 @@ dependencies = [ [[package]] name = "anyhow" -version = "1.0.95" +version = "1.0.96" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04" +checksum = "6b964d184e89d9b6b67dd2715bc8e74cf3107fb2b529990c90cf517326150bf4" + +[[package]] +name = "arrayvec" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50" [[package]] name = "async-channel" @@ -223,7 +229,7 @@ dependencies = [ "futures-lite", "gloo-timers", "kv-log-macro", - "log 0.4.25", + "log 0.4.26", "memchr", "once_cell", "pin-project-lite", @@ -240,13 +246,13 @@ checksum = "8b75356056920673b02621b35afd0f7dda9306d03c79a30f5c56c44cf256e3de" [[package]] name = "async-trait" -version = "0.1.85" +version = "0.1.86" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3f934833b4b7233644e5848f235df3f57ed8c80f1528a26c3dfa13d2147fa056" +checksum = "644dd749086bf3771a2fbc5f256fdb982d53f011c7d5d560304eafeecebce79d" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -309,12 +315,21 @@ dependencies = [ "addr2line", "cfg-if", "libc", - "miniz_oxide 0.8.3", + "miniz_oxide 0.8.5", "object", "rustc-demangle", "windows-targets 0.52.6", ] +[[package]] +name = "bare-metal" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5deb64efa5bd81e31fcd1938615a6d98c82eafcbcd787162b6f63b91d6bac5b3" +dependencies = [ + "rustc_version 0.2.3", +] + [[package]] name = "base64" version = "0.9.3" @@ -353,7 +368,7 @@ dependencies = [ "bitflags 2.8.0", "cexpr", "clang-sys", - "itertools", + "itertools 0.12.1", "lazy_static", "lazycell", "proc-macro2", @@ -361,7 +376,7 @@ dependencies = [ "regex", "rustc-hash", "shlex", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -385,6 +400,18 @@ version = "0.10.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "dc827186963e592360843fb5ba4b973e145841266c1357f7180c43526f2e5b61" +[[package]] +name = "bitfield" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "46afbd2983a5d5a7bd740ccb198caf5b82f45c40c09c0eed36052d91cb92e719" + +[[package]] +name = "bitfield" +version = "0.14.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2d7e60934ceec538daadb9d8432424ed043a904d8e0243f3c6446bce549a46ac" + [[package]] name = "bitflags" version = "1.3.2" @@ -424,15 +451,15 @@ dependencies = [ [[package]] name = "build_html" -version = "2.5.0" +version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "225eb82ce9e70dcc0cfa6e404d0f353326b6e163bf500ec4711cec317d11935c" +checksum = "01b01f54cbdd56298a506b086691594ded3b68dcbc9437adc87c616a35e7fc89" [[package]] name = "bumpalo" -version = "3.16.0" +version = "3.17.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c" +checksum = "1628fb46dfa0b37568d12e5edd512553eccf6a22a78e8bde00bb4aed84d5bdbf" [[package]] name = "bytemuck" @@ -448,9 +475,9 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.9.0" +version = "1.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "325918d6fe32f23b19878fe4b34794ae41fc19ddbe53b10571a4874d44ffd39b" +checksum = "f61dac84819c6588b558454b194026eb1f09c293b9036ae9b159e74e73ab6cf9" [[package]] name = "cairo-rs" @@ -460,7 +487,7 @@ checksum = "8ca26ef0159422fb77631dc9d17b102f253b876fe1586b03b803e63a309b4ee2" dependencies = [ "bitflags 2.8.0", "cairo-sys-rs", - "glib", + "glib 0.18.5", "libc", "once_cell", "thiserror 1.0.69", @@ -472,16 +499,16 @@ version = "0.18.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "685c9fa8e590b8b3d678873528d83411db17242a73fccaed827770ea0fedda51" dependencies = [ - "glib-sys", + "glib-sys 0.18.1", "libc", - "system-deps", + "system-deps 6.2.2", ] [[package]] name = "cc" -version = "1.2.10" +version = "1.2.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "13208fcbb66eaeffe09b99fffbe1af420f00a7b35aa99ad683dfc1aa76145229" +checksum = "c736e259eea577f443d5c86c304f9f4ae0295c43f3ba05c21f1d66b5f06001af" dependencies = [ "shlex", ] @@ -505,6 +532,16 @@ dependencies = [ "target-lexicon", ] +[[package]] +name = "cfg-expr" +version = "0.17.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8d4ba6e40bd1184518716a6e1a781bf9160e286d219ccdb8ab2612e74cfe4789" +dependencies = [ + "smallvec", + "target-lexicon", +] + [[package]] name = "cfg-if" version = "1.0.0" @@ -515,14 +552,14 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" name = "changeset" version = "0.1.0" dependencies = [ - "uuid 1.12.0", + "uuid 1.15.1", ] [[package]] name = "chrono" -version = "0.4.39" +version = "0.4.40" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7e36cc9d416881d2e24f9a963be5fb1cd90966419ac844274161d10488b3e825" +checksum = "1a7964611d71df112cb1730f2ee67324fcf4d0fc6606acbbe9bfe06df124637c" dependencies = [ "android-tzdata", "iana-time-zone", @@ -530,7 +567,7 @@ dependencies = [ "num-traits", "serde 1.0.218", "wasm-bindgen", - "windows-targets 0.52.6", + "windows-link", ] [[package]] @@ -569,9 +606,9 @@ dependencies = [ [[package]] name = "clap" -version = "4.5.30" +version = "4.5.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "92b7b18d71fad5313a1e320fa9897994228ce274b60faa4d694fe0ea89cd9e6d" +checksum = "027bb0d98429ae334a8698531da7077bdf906419543a35a55c2cb1b66437d767" dependencies = [ "clap_builder", "clap_derive", @@ -579,9 +616,9 @@ dependencies = [ [[package]] name = "clap_builder" -version = "4.5.30" +version = "4.5.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a35db2071778a7344791a4fb4f95308b5673d219dee3ae348b86642574ecc90c" +checksum = "5589e0cba072e0f3d23791efac0fd8627b49c829c196a492e88168e6a669d863" dependencies = [ "anstream", "anstyle", @@ -598,7 +635,7 @@ dependencies = [ "heck 0.5.0", "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -725,10 +762,42 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b" [[package]] -name = "cpufeatures" -version = "0.2.16" +name = "cortex-m" +version = "0.7.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "16b80225097f2e5ae4e7179dd2266824648f3e2f49d9134d584b76389d31c4c3" +checksum = "8ec610d8f49840a5b376c69663b6369e71f4b34484b9b2eb29fb918d92516cb9" +dependencies = [ + "bare-metal", + "bitfield 0.13.2", + "embedded-hal 0.2.7", + "volatile-register", +] + +[[package]] +name = "cortex-m-rt" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "801d4dec46b34c299ccf6b036717ae0fce602faa4f4fe816d9013b9a7c9f5ba6" +dependencies = [ + "cortex-m-rt-macros", +] + +[[package]] +name = "cortex-m-rt-macros" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e37549a379a9e0e6e576fd208ee60394ccb8be963889eebba3ffe0980364f472" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.98", +] + +[[package]] +name = "cpufeatures" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280" dependencies = [ "libc", ] @@ -742,6 +811,15 @@ dependencies = [ "crc-catalog", ] +[[package]] +name = "crc-any" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a62ec9ff5f7965e4d7280bd5482acd20aadb50d632cf6c1d74493856b011fa73" +dependencies = [ + "debug-helper", +] + [[package]] name = "crc-catalog" version = "2.4.0" @@ -757,6 +835,12 @@ dependencies = [ "cfg-if", ] +[[package]] +name = "critical-section" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "790eea4361631c5e7d22598ecd5723ff611904e3344ce8720784c93e3d83d40b" + [[package]] name = "crossbeam-deque" version = "0.8.6" @@ -793,9 +877,9 @@ checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28" [[package]] name = "crunchy" -version = "0.2.2" +version = "0.2.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" +checksum = "43da5946c66ffcc7745f48db692ffbb10a83bfe0afd96235c5c2a4fb23994929" [[package]] name = "crypto-common" @@ -814,8 +898,8 @@ dependencies = [ "async-std", "cairo-rs", "cyberpunk", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "gtk4", "serde 1.0.218", "serde_yml", @@ -826,8 +910,8 @@ name = "cyberpunk" version = "0.1.0" dependencies = [ "cairo-rs", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "gtk4", ] @@ -838,8 +922,8 @@ dependencies = [ "async-std", "cairo-rs", "cyberpunk", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "gtk4", ] @@ -855,8 +939,8 @@ dependencies = [ "futures", "gdk4", "geo-types", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "glib-build-tools 0.18.0", "gtk4", "lazy_static", @@ -871,9 +955,15 @@ dependencies = [ [[package]] name = "data-encoding" -version = "2.7.0" +version = "2.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0e60eed09d8c01d3cee5b7d30acb059b76614c918fa0f992e0dd6eeb10daad6f" +checksum = "575f75dfd25738df5b91b8e43e14d44bda14637a58fae779fd2b064f8bf3e010" + +[[package]] +name = "debug-helper" +version = "0.3.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f578e8e2c440e7297e008bb5486a3a8a194775224bbc23729b0dbdfaeebf162e" [[package]] name = "deflate" @@ -949,7 +1039,7 @@ checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -960,13 +1050,63 @@ checksum = "1aaf95b3e5c8f23aa320147307562d361db0ae0d51242340f558153b4eb2439b" [[package]] name = "either" -version = "1.13.0" +version = "1.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0" +checksum = "b7914353092ddf589ad78f25c5c1c21b7f80b0ff8621e7c814c3485b5306da9d" dependencies = [ "serde 1.0.218", ] +[[package]] +name = "embedded-dma" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "994f7e5b5cb23521c22304927195f236813053eb9c065dd2226a32ba64695446" +dependencies = [ + "stable_deref_trait", +] + +[[package]] +name = "embedded-hal" +version = "0.2.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "35949884794ad573cf46071e41c9b60efb0cb311e3ca01f7af807af1debc66ff" +dependencies = [ + "nb 0.1.3", + "void", +] + +[[package]] +name = "embedded-hal" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "361a90feb7004eca4019fb28352a9465666b24f840f5c3cddf0ff13920590b89" + +[[package]] +name = "embedded-hal-async" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c4c685bbef7fe13c3c6dd4da26841ed3980ef33e841cddfa15ce8a8fb3f1884" +dependencies = [ + "embedded-hal 1.0.0", +] + +[[package]] +name = "embedded-hal-nb" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fba4268c14288c828995299e59b12babdbe170f6c6d73731af1b4648142e8605" +dependencies = [ + "embedded-hal 1.0.0", + "nb 1.1.0", +] + +[[package]] +name = "embedded-io" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "edd0f118536f44f5ccd48bcb8b111bdc3de888b58c74639dfb034a357d0f206d" + [[package]] name = "emseries" version = "0.6.0" @@ -999,16 +1139,16 @@ checksum = "4cd405aab171cb85d6735e5c8d9db038c17d3ca007a4d2c25f337935c3d90580" dependencies = [ "humantime", "is-terminal", - "log 0.4.25", + "log 0.4.26", "regex", "termcolor", ] [[package]] name = "equivalent" -version = "1.0.1" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" +checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f" [[package]] name = "errno" @@ -1067,7 +1207,7 @@ dependencies = [ "bit_field", "half", "lebe", - "miniz_oxide 0.8.3", + "miniz_oxide 0.8.5", "rayon-core", "smallvec", "zune-inflate", @@ -1095,7 +1235,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "38e2275cc4e4fc009b0669731a1e5ab7ebf11f469eaede2bab9309a5b4d6057f" dependencies = [ "memoffset", - "rustc_version", + "rustc_version 0.4.1", ] [[package]] @@ -1114,7 +1254,7 @@ dependencies = [ "hex-string", "http 0.2.12", "image 0.23.14", - "log 0.4.25", + "log 0.4.26", "logger", "mime 0.3.17", "mime_guess 2.0.5", @@ -1141,8 +1281,8 @@ dependencies = [ "emseries", "ft-core", "gdk4", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "glib-build-tools 0.18.0", "gtk4", "libadwaita", @@ -1152,9 +1292,9 @@ dependencies = [ [[package]] name = "fixed" -version = "1.28.0" +version = "1.29.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "85c6e0b89bf864acd20590dbdbad56f69aeb898abfc9443008fd7bd48b2cc85a" +checksum = "707070ccf8c4173548210893a0186e29c266901b71ed20cd9e2ca0193dfe95c3" dependencies = [ "az", "bytemuck", @@ -1164,12 +1304,12 @@ dependencies = [ [[package]] name = "flate2" -version = "1.0.35" +version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c936bfdafb507ebbf50b8074c54fa31c5be9a1e7e5f467dd659697041407d07c" +checksum = "11faaf5a5236997af9848be0bef4db95824b1d534ebc64d0f0c6cf3e67bd38dc" dependencies = [ "crc32fast", - "miniz_oxide 0.8.3", + "miniz_oxide 0.8.5", ] [[package]] @@ -1273,6 +1413,45 @@ dependencies = [ "percent-encoding 2.3.1", ] +[[package]] +name = "frunk" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "874b6a17738fc273ec753618bac60ddaeac48cb1d7684c3e7bd472e57a28b817" +dependencies = [ + "frunk_core", + "frunk_derives", +] + +[[package]] +name = "frunk_core" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3529a07095650187788833d585c219761114005d5976185760cf794d265b6a5c" + +[[package]] +name = "frunk_derives" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e99b8b3c28ae0e84b604c75f721c21dc77afb3706076af5e8216d15fd1deaae3" +dependencies = [ + "frunk_proc_macro_helpers", + "quote", + "syn 2.0.98", +] + +[[package]] +name = "frunk_proc_macro_helpers" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05a956ef36c377977e512e227dcad20f68c2786ac7a54dacece3746046fea5ce" +dependencies = [ + "frunk_core", + "proc-macro2", + "quote", + "syn 2.0.98", +] + [[package]] name = "ft-core" version = "0.1.0" @@ -1292,6 +1471,15 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a06f77d526c1a601b7c4cdd98f54b5eaabffc14d5f2f0296febdc7f357c6d3ba" +[[package]] +name = "fugit" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17186ad64927d5ac8f02c1e77ccefa08ccd9eaa314d5a4772278aa204a22f7e7" +dependencies = [ + "gcd", +] + [[package]] name = "futures" version = "0.3.31" @@ -1372,7 +1560,7 @@ checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -1405,6 +1593,12 @@ dependencies = [ "slab", ] +[[package]] +name = "gcd" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d758ba1b47b00caf47f24925c0074ecb20d6dfcffe7f6d53395c0465674841a" + [[package]] name = "gdk-pixbuf" version = "0.18.5" @@ -1412,8 +1606,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "50e1f5f1b0bfb830d6ccc8066d18db35c487b1b2b1e8589b5dfe9f07e8defaec" dependencies = [ "gdk-pixbuf-sys", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "libc", "once_cell", ] @@ -1424,11 +1618,11 @@ version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3f9839ea644ed9c97a34d129ad56d38a25e6756f99f3a88e15cd39c20629caf7" dependencies = [ - "gio-sys", - "glib-sys", - "gobject-sys", + "gio-sys 0.18.1", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "libc", - "system-deps", + "system-deps 6.2.2", ] [[package]] @@ -1440,10 +1634,10 @@ dependencies = [ "cairo-rs", "gdk-pixbuf", "gdk4-sys", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "libc", - "pango", + "pango 0.18.3", ] [[package]] @@ -1454,13 +1648,13 @@ checksum = "dbab43f332a3cf1df9974da690b5bb0e26720ed09a228178ce52175372dcfef0" dependencies = [ "cairo-sys-rs", "gdk-pixbuf-sys", - "gio-sys", - "glib-sys", - "gobject-sys", + "gio-sys 0.18.1", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "libc", - "pango-sys", + "pango-sys 0.18.0", "pkg-config", - "system-deps", + "system-deps 6.2.2", ] [[package]] @@ -1545,8 +1739,8 @@ dependencies = [ "futures-core", "futures-io", "futures-util", - "gio-sys", - "glib", + "gio-sys 0.18.1", + "glib 0.18.5", "libc", "once_cell", "pin-project-lite", @@ -1554,19 +1748,49 @@ dependencies = [ "thiserror 1.0.69", ] +[[package]] +name = "gio" +version = "0.20.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4f00c70f8029d84ea7572dd0e1aaa79e5329667b4c17f329d79ffb1e6277487" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-util", + "gio-sys 0.20.9", + "glib 0.20.9", + "libc", + "pin-project-lite", + "smallvec", +] + [[package]] name = "gio-sys" version = "0.18.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "37566df850baf5e4cb0dfb78af2e4b9898d817ed9263d1090a2df958c64737d2" dependencies = [ - "glib-sys", - "gobject-sys", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "libc", - "system-deps", + "system-deps 6.2.2", "winapi", ] +[[package]] +name = "gio-sys" +version = "0.20.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "160eb5250a26998c3e1b54e6a3d4ea15c6c7762a6062a19a7b63eff6e2b33f9e" +dependencies = [ + "glib-sys 0.20.9", + "gobject-sys 0.20.9", + "libc", + "system-deps 7.0.3", + "windows-sys 0.59.0", +] + [[package]] name = "glib" version = "0.18.5" @@ -1579,10 +1803,10 @@ dependencies = [ "futures-executor", "futures-task", "futures-util", - "gio-sys", - "glib-macros", - "glib-sys", - "gobject-sys", + "gio-sys 0.18.1", + "glib-macros 0.18.5", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "libc", "memchr", "once_cell", @@ -1590,6 +1814,27 @@ dependencies = [ "thiserror 1.0.69", ] +[[package]] +name = "glib" +version = "0.20.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "707b819af8059ee5395a2de9f2317d87a53dbad8846a2f089f0bb44703f37686" +dependencies = [ + "bitflags 2.8.0", + "futures-channel", + "futures-core", + "futures-executor", + "futures-task", + "futures-util", + "gio-sys 0.20.9", + "glib-macros 0.20.7", + "glib-sys 0.20.9", + "gobject-sys 0.20.9", + "libc", + "memchr", + "smallvec", +] + [[package]] name = "glib-build-tools" version = "0.16.3" @@ -1619,7 +1864,20 @@ dependencies = [ "proc-macro-error", "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", +] + +[[package]] +name = "glib-macros" +version = "0.20.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "715601f8f02e71baef9c1f94a657a9a77c192aea6097cf9ae7e5e177cd8cde68" +dependencies = [ + "heck 0.5.0", + "proc-macro-crate 3.2.0", + "proc-macro2", + "quote", + "syn 2.0.98", ] [[package]] @@ -1629,7 +1887,17 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "063ce2eb6a8d0ea93d2bf8ba1957e78dbab6be1c2220dd3daca57d5a9d869898" dependencies = [ "libc", - "system-deps", + "system-deps 6.2.2", +] + +[[package]] +name = "glib-sys" +version = "0.20.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8928869a44cfdd1fccb17d6746e4ff82c8f82e41ce705aa026a52ca8dc3aefb" +dependencies = [ + "libc", + "system-deps 7.0.3", ] [[package]] @@ -1658,8 +1926,8 @@ dependencies = [ "config-derive", "futures", "gdk4", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "glib-build-tools 0.16.3", "gtk4", "libadwaita", @@ -1685,9 +1953,20 @@ version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0850127b514d1c4a4654ead6dedadb18198999985908e6ffe4436f53c785ce44" dependencies = [ - "glib-sys", + "glib-sys 0.18.1", "libc", - "system-deps", + "system-deps 6.2.2", +] + +[[package]] +name = "gobject-sys" +version = "0.20.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c773a3cb38a419ad9c26c81d177d96b4b08980e8bdbbf32dace883e96e96e7e3" +dependencies = [ + "glib-sys 0.20.9", + "libc", + "system-deps 7.0.3", ] [[package]] @@ -1696,7 +1975,7 @@ version = "0.18.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3b2228cda1505613a7a956cca69076892cfbda84fc2b7a62b94a41a272c0c401" dependencies = [ - "glib", + "glib 0.18.5", "graphene-sys", "libc", ] @@ -1707,10 +1986,10 @@ version = "0.18.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cc4144cee8fc8788f2a9b73dc5f1d4e1189d1f95305c4cb7bd9c1af1cfa31f59" dependencies = [ - "glib-sys", + "glib-sys 0.18.1", "libc", "pkg-config", - "system-deps", + "system-deps 6.2.2", ] [[package]] @@ -1730,11 +2009,11 @@ checksum = "0d958e351d2f210309b32d081c832d7de0aca0b077aa10d88336c6379bd01f7e" dependencies = [ "cairo-rs", "gdk4", - "glib", + "glib 0.18.5", "graphene-rs", "gsk4-sys", "libc", - "pango", + "pango 0.18.3", ] [[package]] @@ -1745,12 +2024,12 @@ checksum = "12bd9e3effea989f020e8f1ff3fa3b8c63ba93d43b899c11a118868853a56d55" dependencies = [ "cairo-sys-rs", "gdk4-sys", - "glib-sys", - "gobject-sys", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "graphene-sys", "libc", - "pango-sys", - "system-deps", + "pango-sys 0.18.0", + "system-deps 6.2.2", ] [[package]] @@ -1764,14 +2043,14 @@ dependencies = [ "futures-channel", "gdk-pixbuf", "gdk4", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "graphene-rs", "gsk4", "gtk4-macros", "gtk4-sys", "libc", - "pango", + "pango 0.18.3", ] [[package]] @@ -1797,14 +2076,14 @@ dependencies = [ "cairo-sys-rs", "gdk-pixbuf-sys", "gdk4-sys", - "gio-sys", - "glib-sys", - "gobject-sys", + "gio-sys 0.18.1", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "graphene-sys", "gsk4-sys", "libc", - "pango-sys", - "system-deps", + "pango-sys 0.18.0", + "system-deps 6.2.2", ] [[package]] @@ -1836,6 +2115,15 @@ dependencies = [ "crunchy", ] +[[package]] +name = "hash32" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47d60b12902ba28e2730cd37e95b8c9223af2808df9e902d4df49588d1470606" +dependencies = [ + "byteorder", +] + [[package]] name = "hashbrown" version = "0.15.2" @@ -1880,6 +2168,16 @@ dependencies = [ "http 0.2.12", ] +[[package]] +name = "heapless" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bfb9eb618601c89945a70e254898da93b13be0388091d42117462b265bb3fad" +dependencies = [ + "hash32", + "stable_deref_trait", +] + [[package]] name = "heck" version = "0.4.1" @@ -1916,8 +2214,8 @@ version = "0.1.0" dependencies = [ "cairo-rs", "coordinates", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "glib-build-tools 0.18.0", "gtk4", "image 0.24.9", @@ -1991,9 +2289,9 @@ dependencies = [ [[package]] name = "httparse" -version = "1.9.5" +version = "1.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7d71d3574edd2771538b901e6549113b4006ece66150fb69c0fb6d9a2adae946" +checksum = "f2d708df4e7140240a16cd6ab0ab65c972d7433ab77819ea693fde9c43811e2a" [[package]] name = "httpdate" @@ -2090,8 +2388,8 @@ dependencies = [ name = "icon-test" version = "0.1.0" dependencies = [ - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "gtk4", "libadwaita", ] @@ -2212,7 +2510,7 @@ checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -2292,9 +2590,9 @@ checksum = "0cfe9645a18782869361d9c8732246be7b410ad4e919d3609ebabdac00ba12c3" [[package]] name = "indexmap" -version = "2.7.0" +version = "2.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "62f822373a4fe84d4bb149bf54e584a7f4abec90e072ed49cda0edea5b95471f" +checksum = "8c9c992b02b5b4c94ea26e32fe5bccb7aa7d9f390ab5c1221ff895bc7ea8b652" dependencies = [ "equivalent", "hashbrown", @@ -2321,9 +2619,9 @@ dependencies = [ [[package]] name = "ipnet" -version = "2.10.1" +version = "2.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ddc24109865250148c2e0f3d25d4f0f479571723792d3802153c60922a4fb708" +checksum = "469fb0b9cefa57e3ef31275ee7cacb78f2fdca44e4765491884a2b119d4eb130" [[package]] name = "iron" @@ -2343,13 +2641,13 @@ dependencies = [ [[package]] name = "is-terminal" -version = "0.4.13" +version = "0.4.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "261f68e344040fbd0edea105bef17c66edf46f984ddb1115b775ce31be948f4b" +checksum = "e19b23d53f35ce9f56aebc7d1bb4e6ac1e9c0db7ac85c8d1760c04379edced37" dependencies = [ "hermit-abi 0.4.0", "libc", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -2358,6 +2656,15 @@ version = "1.70.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf" +[[package]] +name = "itertools" +version = "0.10.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b0fd2260e829bddf4cb6ea802289de2f86d6a7a690192fbe91b3f46e0f2c8473" +dependencies = [ + "either", +] + [[package]] name = "itertools" version = "0.12.1" @@ -2407,7 +2714,7 @@ version = "1.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0de8b303297635ad57c9f5059fd9cee7a47f8e8daa09df0fcd07dd39fb22977f" dependencies = [ - "log 0.4.25", + "log 0.4.26", ] [[package]] @@ -2460,12 +2767,12 @@ checksum = "2fe7e70c06507ed10a16cda707f358fbe60fe0dc237498f78c686ade92fd979c" dependencies = [ "gdk-pixbuf", "gdk4", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "gtk4", "libadwaita-sys", "libc", - "pango", + "pango 0.18.3", ] [[package]] @@ -2475,20 +2782,20 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5e10aaa38de1d53374f90deeb4535209adc40cc5dba37f9704724169bceec69a" dependencies = [ "gdk4-sys", - "gio-sys", - "glib-sys", - "gobject-sys", + "gio-sys 0.18.1", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "gtk4-sys", "libc", - "pango-sys", - "system-deps", + "pango-sys 0.18.0", + "system-deps 6.2.2", ] [[package]] name = "libc" -version = "0.2.169" +version = "0.2.170" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b5aba8db14291edd000dfcc4d620c7ebfb122c613afb886ca8803fa4e128a20a" +checksum = "875b3680cb2f8f71bdcf9a30f38d48282f5d3c95cbf9b3fa57269bb5d5c06828" [[package]] name = "libloading" @@ -2520,7 +2827,7 @@ dependencies = [ "libspa-sys", "nix", "nom", - "system-deps", + "system-deps 6.2.2", ] [[package]] @@ -2531,7 +2838,7 @@ checksum = "bf0d9716420364790e85cbb9d3ac2c950bde16a7dd36f3209b7dfdfc4a24d01f" dependencies = [ "bindgen", "cc", - "system-deps", + "system-deps 6.2.2", ] [[package]] @@ -2571,9 +2878,9 @@ checksum = "d26c52dbd32dccf2d10cac7725f8eae5296885fb5703b261f7d0a0739ec807ab" [[package]] name = "litemap" -version = "0.7.4" +version = "0.7.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4ee93343901ab17bd981295f2cf0026d4ad018c7c31ba84549a4ddbb47a45104" +checksum = "23fb14cb19457329c82206317a5663005a4d404783dc74f4252769b0d5f42856" [[package]] name = "lock_api" @@ -2591,14 +2898,14 @@ version = "0.3.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e19e8d5c34a3e0e2223db8e060f9e8264aeeb5c5fc64a4ee9965c062211c024b" dependencies = [ - "log 0.4.25", + "log 0.4.26", ] [[package]] name = "log" -version = "0.4.25" +version = "0.4.26" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04cbf5b083de1c7e0222a7a51dbfdba1cbe1c6ab0b15e29fff3f6c077fd9cd9f" +checksum = "30bde2b3dc3671ae49d8e2e9f044c7c005836e7a023ee57cffa25ab82764bb9e" dependencies = [ "value-bag", ] @@ -2720,9 +3027,9 @@ dependencies = [ [[package]] name = "miniz_oxide" -version = "0.8.3" +version = "0.8.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b8402cab7aefae129c6977bb0ff1b8fd9a04eb5b51efc50a70bea51cda0c7924" +checksum = "8e3e04debbb59698c15bacbb6d93584a8c0ca9cc3213cb423d31f760d8843ce5" dependencies = [ "adler2", "simd-adler32", @@ -2756,7 +3063,7 @@ dependencies = [ "futures-util", "http 0.2.12", "httparse", - "log 0.4.25", + "log 0.4.26", "memchr", "mime 0.3.17", "spin", @@ -2775,12 +3082,12 @@ dependencies = [ [[package]] name = "native-tls" -version = "0.2.12" +version = "0.2.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a8614eb2c83d59d1c8cc974dd3f920198647674a0a035e1af1fa58707e317466" +checksum = "87de3442987e9dbec73158d5c715e7ad9072fda936bb03d19d7fa10e00520f0e" dependencies = [ "libc", - "log 0.4.25", + "log 0.4.26", "openssl", "openssl-probe", "openssl-sys", @@ -2790,6 +3097,21 @@ dependencies = [ "tempfile", ] +[[package]] +name = "nb" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "801d31da0513b6ec5214e9bf433a77966320625a37860f910be265be6e18d06f" +dependencies = [ + "nb 1.1.0", +] + +[[package]] +name = "nb" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8d5439c4ad607c3c23abf66de8c8bf57ba8adcd1f129e699851a6e43935d339d" + [[package]] name = "nix" version = "0.27.1" @@ -2899,6 +3221,26 @@ dependencies = [ "libc", ] +[[package]] +name = "num_enum" +version = "0.5.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f646caf906c20226733ed5b1374287eb97e3c2a5c227ce668c1f2ce20ae57c9" +dependencies = [ + "num_enum_derive", +] + +[[package]] +name = "num_enum_derive" +version = "0.5.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dcbff9bc912032c62bf65ef1d5aea88983b420f4f839db1e9b0c281a25c9c799" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + [[package]] name = "object" version = "0.36.7" @@ -2910,15 +3252,15 @@ dependencies = [ [[package]] name = "once_cell" -version = "1.20.2" +version = "1.20.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775" +checksum = "945462a4b81e43c4e3ba96bd7b49d834c6f61198356aa858733bc4acf3cbe62e" [[package]] name = "openssl" -version = "0.10.68" +version = "0.10.71" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6174bc48f102d208783c2c84bf931bb75927a617866870de8a4ea85597f871f5" +checksum = "5e14130c6a98cd258fdcb0fb6d744152343ff729cbfcb28c656a9d12b999fbcd" dependencies = [ "bitflags 2.8.0", "cfg-if", @@ -2937,20 +3279,20 @@ checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] name = "openssl-probe" -version = "0.1.5" +version = "0.1.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" +checksum = "d05e27ee213611ffe7d6348b942e8f942b37114c00cc03cec254295a4a17852e" [[package]] name = "openssl-sys" -version = "0.9.104" +version = "0.9.106" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "45abf306cbf99debc8195b66b7346498d7b10c210de50418b5ccd7ceba08c741" +checksum = "8bb61ea9811cc39e3c2069f40b8b8e2e70d8569b361f879786cc7ed48b777cdd" dependencies = [ "cc", "libc", @@ -2983,14 +3325,14 @@ dependencies = [ "async-channel 2.3.1", "async-std", "cairo-rs", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "glib-build-tools 0.17.10", "gtk4", "image 0.24.9", "libadwaita", "otg-core", - "pango", + "pango 0.20.9", "sgf", "tokio", "uuid 0.8.2", @@ -3002,11 +3344,23 @@ version = "0.18.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7ca27ec1eb0457ab26f3036ea52229edbdb74dee1edd29063f5b9b010e7ebee4" dependencies = [ - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "libc", "once_cell", - "pango-sys", + "pango-sys 0.18.0", +] + +[[package]] +name = "pango" +version = "0.20.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6b1f5dc1b8cf9bc08bfc0843a04ee0fa2e78f1e1fa4b126844a383af4f25f0ec" +dependencies = [ + "gio 0.20.9", + "glib 0.20.9", + "libc", + "pango-sys 0.20.9", ] [[package]] @@ -3015,12 +3369,30 @@ version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "436737e391a843e5933d6d9aa102cb126d501e815b83601365a948a518555dc5" dependencies = [ - "glib-sys", - "gobject-sys", + "glib-sys 0.18.1", + "gobject-sys 0.18.0", "libc", - "system-deps", + "system-deps 6.2.2", ] +[[package]] +name = "pango-sys" +version = "0.20.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0dbb9b751673bd8fe49eb78620547973a1e719ed431372122b20abd12445bab5" +dependencies = [ + "glib-sys 0.20.9", + "gobject-sys 0.20.9", + "libc", + "system-deps 7.0.3", +] + +[[package]] +name = "panic-halt" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a513e167849a384b7f9b746e517604398518590a9142f4846a32e3c2a4de7b11" + [[package]] name = "parking" version = "2.2.1" @@ -3059,6 +3431,12 @@ dependencies = [ "regex", ] +[[package]] +name = "paste" +version = "1.0.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a" + [[package]] name = "pem-rfc7468" version = "0.7.0" @@ -3157,24 +3535,35 @@ dependencies = [ "siphasher 1.0.1", ] +[[package]] +name = "pico-st7789" +version = "0.1.0" +dependencies = [ + "cortex-m-rt", + "embedded-hal 1.0.0", + "fugit", + "panic-halt", + "rp-pico", +] + [[package]] name = "pin-project" -version = "1.1.8" +version = "1.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1e2ec53ad785f4d35dac0adea7f7dc6f1bb277ad84a680c7afefeae05d1f5916" +checksum = "dfe2e71e1471fe07709406bf725f710b02927c9c54b2b5b2ec0e8087d97c327d" dependencies = [ "pin-project-internal", ] [[package]] name = "pin-project-internal" -version = "1.1.8" +version = "1.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d56a66c0c55993aa927429d0f8a0abfd74f084e4d9c192cffed01e418d83eefb" +checksum = "f6e859e6e5bd50440ab63c47e3ebabc90f26251f7c73c3d3e837b74a1cc3fa67" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -3189,6 +3578,17 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" +[[package]] +name = "pio" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76e09694b50f89f302ed531c1f2a7569f0be5867aee4ab4f8f729bbeec0078e3" +dependencies = [ + "arrayvec", + "num_enum", + "paste", +] + [[package]] name = "piper" version = "0.2.4" @@ -3225,7 +3625,7 @@ checksum = "849e188f90b1dda88fe2bfe1ad31fe5f158af2c98f80fb5d13726c44f3f01112" dependencies = [ "bindgen", "libspa-sys", - "system-deps", + "system-deps 6.2.2", ] [[package]] @@ -3286,7 +3686,7 @@ dependencies = [ "crc32fast", "fdeflate", "flate2", - "miniz_oxide 0.8.3", + "miniz_oxide 0.8.5", ] [[package]] @@ -3304,6 +3704,12 @@ dependencies = [ "windows-sys 0.59.0", ] +[[package]] +name = "portable-atomic" +version = "1.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "350e9b48cbc6b0e028b0473b114454c6316e57336ee184ceab6e53f72c178b3e" + [[package]] name = "powerfmt" version = "0.2.0" @@ -3326,7 +3732,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "865724d4dbe39d9f3dd3b52b88d859d66bcb2d6a0acfd5ea68a65fb66d4bdc1c" dependencies = [ "env_logger", - "log 0.4.25", + "log 0.4.26", ] [[package]] @@ -3345,7 +3751,16 @@ version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7e8366a6159044a37876a2b9817124296703c586a5c92e2c53751fa06d8d43e8" dependencies = [ - "toml_edit 0.20.2", + "toml_edit 0.20.7", +] + +[[package]] +name = "proc-macro-crate" +version = "3.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ecf48c7ca261d60b74ab1a7b20da18bede46776b2e55535cb958eb595c5fa7b" +dependencies = [ + "toml_edit 0.22.24", ] [[package]] @@ -3624,9 +4039,9 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.5.8" +version = "0.5.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "03a862b389f93e68874fbf580b9de08dd02facb9a788ebadaf4a3fd33cf58834" +checksum = "82b568323e98e49e2a0899dcee453dd679fae22d69adf9b11dd508d1549b7e2f" dependencies = [ "bitflags 2.8.0", ] @@ -3687,7 +4102,7 @@ dependencies = [ "hyper-tls", "ipnet", "js-sys", - "log 0.4.25", + "log 0.4.26", "mime 0.3.17", "native-tls", "once_cell", @@ -3716,6 +4131,81 @@ dependencies = [ "thiserror 1.0.69", ] +[[package]] +name = "rp-pico" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b9342d3ac7011ac688300979e9b52a81f0add1d05feb02868cf94bfee0705b28" +dependencies = [ + "cortex-m-rt", + "fugit", + "rp2040-boot2", + "rp2040-hal", + "usb-device", +] + +[[package]] +name = "rp2040-boot2" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c92f344f63f950ee36cf4080050e4dce850839b9175da38f9d2ffb69b4dbb21" +dependencies = [ + "crc-any", +] + +[[package]] +name = "rp2040-hal" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d11e711940087f2cdff8aeae9f4b902e2014c06a00b39a1092686b81ec973d6f" +dependencies = [ + "bitfield 0.14.0", + "cortex-m", + "critical-section", + "embedded-dma", + "embedded-hal 0.2.7", + "embedded-hal 1.0.0", + "embedded-hal-async", + "embedded-hal-nb", + "embedded-io", + "frunk", + "fugit", + "itertools 0.10.5", + "nb 1.1.0", + "paste", + "pio", + "rand_core 0.6.4", + "rp2040-hal-macros", + "rp2040-pac", + "usb-device", + "vcell", + "void", +] + +[[package]] +name = "rp2040-hal-macros" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "86479063e497efe1ae81995ef9071f54fd1c7427e04d6c5b84cde545ff672a5e" +dependencies = [ + "cortex-m-rt", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "rp2040-pac" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "83cbcd3f7a0ca7bbe61dc4eb7e202842bee4e27b769a7bf3a4a72fa399d6e404" +dependencies = [ + "cortex-m", + "cortex-m-rt", + "critical-section", + "vcell", +] + [[package]] name = "rsa" version = "0.9.7" @@ -3748,20 +4238,29 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2" +[[package]] +name = "rustc_version" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "138e3e0acb6c9fb258b19b67cb8abd63c00679d2851805ea151465464fe9030a" +dependencies = [ + "semver 0.9.0", +] + [[package]] name = "rustc_version" version = "0.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cfcb3a22ef46e85b45de6ee7e79d063319ebb6594faafcf1c225ea92ab6e9b92" dependencies = [ - "semver", + "semver 1.0.25", ] [[package]] name = "rustix" -version = "0.38.43" +version = "0.38.44" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a78891ee6bf2340288408954ac787aa063d8e8817e9f53abb37c695c6d834ef6" +checksum = "fdb5bc1ae2baa591800df16c9ca78619bf65c0488b41b96ccec5d11220d8c154" dependencies = [ "bitflags 2.8.0", "errno", @@ -3799,9 +4298,9 @@ dependencies = [ [[package]] name = "ryu" -version = "1.0.18" +version = "1.0.19" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f" +checksum = "6ea1a2d0a644769cc99faa24c3ad26b379b786fe7c36fd3c546254801650e6dd" [[package]] name = "safemem" @@ -3841,7 +4340,7 @@ name = "screenplay" version = "0.1.0" dependencies = [ "async-std", - "glib", + "glib 0.18.5", "gtk4", ] @@ -3885,9 +4384,24 @@ checksum = "c2fdfc24bc566f839a2da4c4295b82db7d25a24253867d5c64355abb5799bdbe" [[package]] name = "semver" -version = "1.0.24" +version = "0.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3cb6eb87a131f756572d7fb904f6e7b68633f09cca868c5df1c4b8d1a694bbba" +checksum = "1d7eb9ef2c18661902cc47e535f9bc51b78acd254da71d375c2f6720d9a40403" +dependencies = [ + "semver-parser", +] + +[[package]] +name = "semver" +version = "1.0.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f79dfe2d285b0488816f30e700a7438c5a73d816b5b7d3ac72fbc48b0d185e03" + +[[package]] +name = "semver-parser" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3" [[package]] name = "serde" @@ -3912,7 +4426,7 @@ checksum = "f09503e191f4e797cb8aac08e9a4a4695c5edf6a2e70e376d961ddd5c969f82b" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4041,12 +4555,12 @@ dependencies = [ "async-std", "cairo-rs", "fixed", - "gio", - "glib", + "gio 0.18.4", + "glib 0.18.5", "gtk4", "libadwaita", "lights-core", - "pango", + "pango 0.20.9", ] [[package]] @@ -4072,9 +4586,9 @@ dependencies = [ [[package]] name = "smallvec" -version = "1.13.2" +version = "1.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" +checksum = "7fcf8323ef1faaee30a44a340193b1ac6814fd9b7b4e88e9d4519a3e4abe1cfd" dependencies = [ "serde 1.0.218", ] @@ -4145,7 +4659,7 @@ dependencies = [ "hashbrown", "hashlink", "indexmap", - "log 0.4.25", + "log 0.4.26", "memchr", "once_cell", "percent-encoding 2.3.1", @@ -4170,7 +4684,7 @@ dependencies = [ "quote", "sqlx-core", "sqlx-macros-core", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4193,7 +4707,7 @@ dependencies = [ "sqlx-mysql", "sqlx-postgres", "sqlx-sqlite", - "syn 2.0.96", + "syn 2.0.98", "tempfile", "tokio", "url 2.5.4", @@ -4223,7 +4737,7 @@ dependencies = [ "hkdf", "hmac", "itoa", - "log 0.4.25", + "log 0.4.26", "md-5", "memchr", "once_cell", @@ -4262,7 +4776,7 @@ dependencies = [ "hmac", "home", "itoa", - "log 0.4.25", + "log 0.4.26", "md-5", "memchr", "once_cell", @@ -4292,7 +4806,7 @@ dependencies = [ "futures-intrusive", "futures-util", "libsqlite3-sys", - "log 0.4.25", + "log 0.4.26", "percent-encoding 2.3.1", "serde 1.0.218", "serde_urlencoded", @@ -4343,9 +4857,9 @@ dependencies = [ [[package]] name = "syn" -version = "2.0.96" +version = "2.0.98" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d5d0adab1ae378d7f53bdebc67a39f1f151407ef230f0ce2883572f5d8985c80" +checksum = "36147f1a48ae0ec2b5b3bc5b537d267457555a10dc06f3dbc8cb11ba3006d3b1" dependencies = [ "proc-macro2", "quote", @@ -4366,7 +4880,7 @@ checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4396,7 +4910,20 @@ version = "6.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a3e535eb8dded36d55ec13eddacd30dec501792ff23a0b1682c38601b8cf2349" dependencies = [ - "cfg-expr", + "cfg-expr 0.15.8", + "heck 0.5.0", + "pkg-config", + "toml", + "version-compare", +] + +[[package]] +name = "system-deps" +version = "7.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "66d23aaf9f331227789a99e8de4c91bf46703add012bdfd45fdecdfb2975a005" +dependencies = [ + "cfg-expr 0.17.2", "heck 0.5.0", "pkg-config", "toml", @@ -4468,7 +4995,7 @@ checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4479,7 +5006,7 @@ checksum = "26afc1baea8a989337eeb52b6e72a039780ce45c3edfcc9c5b9d112feeb173c2" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4606,7 +5133,7 @@ checksum = "6e06d43f1345a3bcd39f6a56dbb7dcab2ba47e68e8ac134855e7e2bdbaf8cab8" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4637,7 +5164,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c83b561d025642014097b66e6c1bb422783339e0909e4429cde4749d1990bc38" dependencies = [ "futures-util", - "log 0.4.25", + "log 0.4.26", "tokio", "tungstenite", ] @@ -4689,9 +5216,9 @@ dependencies = [ [[package]] name = "toml_edit" -version = "0.20.2" +version = "0.20.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "396e4d48bbb2b7554c944bde63101b5ae446cff6ec4a24227428f15eb72ef338" +checksum = "70f427fce4d84c72b5b732388bf4a9f4531b53f74e2887e3ecb2481f68f66d81" dependencies = [ "indexmap", "toml_datetime", @@ -4723,7 +5250,7 @@ version = "0.1.41" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "784e0ac535deb450455cbfa28a6f0df145ea1bb7ae51b821cf5e7927fdcfbdd0" dependencies = [ - "log 0.4.25", + "log 0.4.26", "pin-project-lite", "tracing-attributes", "tracing-core", @@ -4737,7 +5264,7 @@ checksum = "395ae124c09f9e6918a2310af6038fba074bcf474ac352496d5910dd59a2226d" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4751,9 +5278,9 @@ dependencies = [ [[package]] name = "traitobject" -version = "0.1.0" +version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "efd1f82c56340fdf16f2a953d7bda4f8fdffba13d93b00844c25572110b26079" +checksum = "04a79e25382e2e852e8da874249358d382ebaf259d0d34e75d8db16a7efabbc7" [[package]] name = "tree" @@ -4776,7 +5303,7 @@ dependencies = [ "data-encoding", "http 1.2.0", "httparse", - "log 0.4.25", + "log 0.4.26", "rand 0.8.5", "sha1", "thiserror 1.0.69", @@ -4810,9 +5337,9 @@ dependencies = [ [[package]] name = "typenum" -version = "1.17.0" +version = "1.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" +checksum = "1dccffe3ce07af9386bfd29e80c0ab1a8205a2fc34e4bcd40364df902cfa8f3f" [[package]] name = "typeshare" @@ -4833,7 +5360,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a615d6c2764852a2e88a4f16e9ce1ea49bb776b5872956309e170d63a042a34f" dependencies = [ "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] @@ -4883,9 +5410,9 @@ checksum = "5c1cb5db39152898a79168971543b1cb5020dff7fe43c8dc468b0885f5e29df5" [[package]] name = "unicode-ident" -version = "1.0.14" +version = "1.0.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "adb9e6ca4f869e1180728b7950e35922a7fc6397f7b641499e8f3ef06e50dc83" +checksum = "00e2473a93778eb0bad35909dff6a10d28e63f792f16ed15e404fca9d5eeedbe" [[package]] name = "unicode-normalization" @@ -4945,6 +5472,16 @@ dependencies = [ "percent-encoding 2.3.1", ] +[[package]] +name = "usb-device" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "98816b1accafbb09085168b90f27e93d790b4bfa19d883466b5e53315b5f06a6" +dependencies = [ + "heapless", + "portable-atomic", +] + [[package]] name = "utf-8" version = "0.7.6" @@ -4991,11 +5528,11 @@ dependencies = [ [[package]] name = "uuid" -version = "1.12.0" +version = "1.15.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "744018581f9a3454a9e15beb8a33b017183f1e7c0cd170232a2d1453b23a51c4" +checksum = "e0f540e3240398cce6128b64ba83fdbdd86129c16a3aa1a3a252efd66eb3d587" dependencies = [ - "getrandom 0.2.15", + "getrandom 0.3.1", ] [[package]] @@ -5004,6 +5541,12 @@ version = "1.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3ef4c4aa54d5d05a279399bfa921ec387b7aba77caf7a682ae8d86785b8fdad2" +[[package]] +name = "vcell" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77439c1b53d2303b20d9459b1ade71a83c716e3f9c34f3228c00e6f185d6c002" + [[package]] name = "vcpkg" version = "0.2.15" @@ -5029,10 +5572,25 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a" [[package]] -name = "wait-timeout" -version = "0.2.0" +name = "void" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9f200f5b12eb75f8c1ed65abd4b2db8a6e1b138a20de009dacee265a2498f3f6" +checksum = "6a02e4885ed3bc0f2de90ea6dd45ebcbb66dacffe03547fadbb0eeae2770887d" + +[[package]] +name = "volatile-register" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "de437e2a6208b014ab52972a27e59b33fa2920d3e00fe05026167a1c509d19cc" +dependencies = [ + "vcell", +] + +[[package]] +name = "wait-timeout" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09ac3b126d3914f9849036f826e054cbabdc8519970b8998ddaf3b5bd3c65f11" dependencies = [ "libc", ] @@ -5058,7 +5616,7 @@ dependencies = [ "headers", "http 0.2.12", "hyper 0.14.32", - "log 0.4.25", + "log 0.4.26", "mime 0.3.17", "mime_guess 2.0.5", "multer", @@ -5121,10 +5679,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2f0a0651a5c2bc21487bde11ee802ccaf4c51935d0d3d42a6101f98161700bc6" dependencies = [ "bumpalo", - "log 0.4.25", + "log 0.4.26", "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", "wasm-bindgen-shared", ] @@ -5159,7 +5717,7 @@ checksum = "8ae87ea40c9f689fc23f209965b6fb8a99ad69aeeb0231408be24920604395de" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", "wasm-bindgen-backend", "wasm-bindgen-shared", ] @@ -5239,6 +5797,12 @@ dependencies = [ "windows-targets 0.52.6", ] +[[package]] +name = "windows-link" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6dccfd733ce2b1753b03b6d3c65edf020262ea35e20ccdf3e288043e6dd620e3" + [[package]] name = "windows-sys" version = "0.48.0" @@ -5471,7 +6035,7 @@ checksum = "2380878cad4ac9aac1e2435f3eb4020e8374b5f13c296cb75b4620ff8e229154" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", "synstructure", ] @@ -5493,27 +6057,27 @@ checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] name = "zerofrom" -version = "0.1.5" +version = "0.1.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cff3ee08c995dee1859d998dea82f7374f2826091dd9cd47def953cae446cd2e" +checksum = "50cc42e0333e05660c3587f3bf9d0478688e15d870fab3346451ce7f8c9fbea5" dependencies = [ "zerofrom-derive", ] [[package]] name = "zerofrom-derive" -version = "0.1.5" +version = "0.1.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "595eed982f7d355beb85837f651fa22e90b3c044842dc7f2c2842c086f295808" +checksum = "d71e5d6e06ab090c67b5e44993ec16b72dcbaabc526db883a360057678b48502" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", "synstructure", ] @@ -5542,7 +6106,7 @@ checksum = "6eafa6dfb17584ea3e2bd6e76e0cc15ad7af12b09abdd1ca55961bed9b1063c6" dependencies = [ "proc-macro2", "quote", - "syn 2.0.96", + "syn 2.0.98", ] [[package]] diff --git a/Cargo.toml b/Cargo.toml index d982233..cb38613 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -9,28 +9,30 @@ members = [ "config", "config-derive", "coordinates", - "cyberpunk", "cyber-slides", + "cyberpunk", "cyberpunk-splash", "dashboard", "emseries", "file-service", - "fitnesstrax/core", "fitnesstrax/app", + "fitnesstrax/core", "fluent-ergonomics", "geo-types", "gm-control-panel", + "gm-dash/server", "hex-grid", "icon-test", + "l10n-db", "memorycache", "nom-training", "otg/core", "otg/gtk", + "pico-st7789", "result-extended", "screenplay", "sgf", "timezone-testing", "tree", "visions/server", - "gm-dash/server" -, "l10n-db"] +] diff --git a/pico-st7789/.cargo/config.toml b/pico-st7789/.cargo/config.toml new file mode 100644 index 0000000..9585789 --- /dev/null +++ b/pico-st7789/.cargo/config.toml @@ -0,0 +1,11 @@ +[build] +target = "thumbv6m-none-eabi" + +[target.thumbv6m-none-eabi] +rustflags = [ + "-C", "link-arg=--nmagic", + "-C", "link-arg=-Tlink.x", + "-C", "no-vectorize-loops", +] + +runner = "elf2uf2-rs -d" diff --git a/pico-st7789/Cargo.toml b/pico-st7789/Cargo.toml new file mode 100644 index 0000000..6d8012e --- /dev/null +++ b/pico-st7789/Cargo.toml @@ -0,0 +1,11 @@ +[package] +name = "pico-st7789" +version = "0.1.0" +edition = "2021" + +[dependencies] +cortex-m-rt = "0.7.3" +embedded-hal = "1.0.0" +fugit = "0.3.7" +panic-halt = "1.0.0" +rp-pico = "0.9.0" diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs new file mode 100644 index 0000000..85068f2 --- /dev/null +++ b/pico-st7789/src/main.rs @@ -0,0 +1,225 @@ +#![no_main] +#![no_std] + +use embedded_hal::{delay::DelayNs, digital::OutputPin, spi::SpiBus}; +use fugit::RateExtU32; +use panic_halt as _; +use rp_pico::{ + entry, + hal::{ + clocks::init_clocks_and_plls, + gpio::{FunctionSio, Pin, PinId, PullDown, SioOutput}, + spi::{Enabled, Spi, SpiDevice, ValidSpiPinout}, + Clock, Sio, Timer, Watchdog, + }, + pac, Pins, +}; + +const XOSC_CRYSTAL_FREQ: u32 = 12_000_000; // MHz, https://forums.raspberrypi.com/viewtopic.php?t=356764 + +struct Step { + param_cnt: usize, + command: u8, + params: [u8; 4], + delay: Option<u32>, +} + +impl Step { + fn send_command<D, Pinout, P>( + &self, + spi: &mut Spi<Enabled, D, Pinout, 8>, + data_command: &mut Pin<P, FunctionSio<SioOutput>, PullDown>, + ) where + D: SpiDevice, + Pinout: ValidSpiPinout<D>, + P: PinId, + { + let _ = data_command.set_low(); + let _ = spi.write(&[self.command]); + let _ = data_command.set_high(); + let _ = spi.write(&self.params[0..self.param_cnt]); + } +} + +const NOP: u8 = 0x00; + +const SWRESET: Step = Step { + param_cnt: 0, + command: 0x01, + params: [0, 0, 0, 0], + delay: Some(150), +}; +const SLPOUT: Step = Step { + param_cnt: 0, + command: 0x11, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const COLMOD: u8 = 0x3a; +const MADCTL: Step = Step { + param_cnt: 0, + command: 0x36, + params: [0, 0, 0, 0], + delay: None, +}; +const CASET: u8 = 0x2a; +const RASET: u8 = 0x2b; +const INVON: Step = Step { + param_cnt: 0, + command: 0x21, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const NORON: Step = Step { + param_cnt: 0, + command: 0x13, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const DISPON: Step = Step { + param_cnt: 0, + command: 0x29, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const RAMWR: u8 = 0x2c; + +// Adafruit setup instructions +// SWRESET (0x01), 150ms delay +// SLPOUT (0x11), 10ms delay +// COLMOD (0x3a) 0x55 (65K RGB, 16bit/pixel), 10ms delay +// MADCTL (0x36) 0x00, +// memory data access control, RGB +// CASET 0x00, 0, 0, 170, +// column address set, 4 parameters +// 0x00, 0x00 indicates xstart is 0 +// 0x00, 170 indicates xend is 170 +// RASET 0x00, 0, 320 >> 8, 320 & 0xFF, +// row address set, 4 parameters +// 0x00, 0x00 indicates ystart is 0 +// 3230 >> 8, 320 & 0xff indicates that 320 is the last y address +// INVON, 10ms delay +// invert the display +// NORON, 10ms delay +// normal display mode +// DISPON, 10ms delay +// turn the display on + +const SETUP_PROGRAM: [Step; 9] = [ + SWRESET, + SLPOUT, + Step { + param_cnt: 1, + command: COLMOD, + params: [0x55, 0, 0, 0], + delay: Some(10), + }, + MADCTL, + Step { + param_cnt: 4, + command: CASET, + params: [0, 0, 0, 170], + delay: None, + }, + Step { + param_cnt: 4, + command: RASET, + params: [0, 0, (340 as u16 >> 8) as u8, (320 as u16 & 0xff) as u8], + delay: None, + }, + INVON, + NORON, + DISPON, +]; + +#[entry] +unsafe fn main() -> ! { + // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico. + let mut peripherals = pac::Peripherals::take().unwrap(); + + // SIO inidcates "Single Cycle IO". I don't know what this means, but it could mean that this + // is a class of IO operations that can be run in a single clock cycle, such as switching a + // GPIO pin on or off. + let sio = Sio::new(peripherals.SIO); + + // Many of the following systems require a watchdog. I do not know what this does, either, but + // it may be some failsafe software that will reset operations if the watchdog detects a lack + // of activity. + let mut watchdog = Watchdog::new(peripherals.WATCHDOG); + + // Here we grab the GPIO pins in bank 0. + let pins = Pins::new( + peripherals.IO_BANK0, + peripherals.PADS_BANK0, + sio.gpio_bank0, + &mut peripherals.RESETS, + ); + + // Initialize an abstraction of the clock system with a batch of standard hardware clocks. + let clocks = init_clocks_and_plls( + XOSC_CRYSTAL_FREQ, + peripherals.XOSC, + peripherals.CLOCKS, + peripherals.PLL_SYS, + peripherals.PLL_USB, + &mut peripherals.RESETS, + &mut watchdog, + ) + .ok() + .unwrap(); + + // An abstraction for a timer which we can use to delay the code. + let mut timer = Timer::new(peripherals.TIMER, &mut peripherals.RESETS, &clocks); + + // Grab the clock and data pins for SPI1. For Clock pins and for Data pins, there are only two + // pins each on the Pico which can function for SPI1. + let spi_clk = pins.gpio2.into_function(); + let spi_sdo = pins.gpio3.into_function(); + // let spi_sdi = pins.gpio4.into_function(); + // Chip select 1 means the chip is not enabled + let mut board_select = pins.gpio13.into_function(); + let mut data_command = pins.gpio14.into_function(); + let mut reset = pins.gpio15.into_function(); + + let _ = reset.set_low(); + let _ = board_select.set_high(); + let _ = data_command.set_high(); + + // Now, create the SPI function abstraction for SPI1 with spi_clk and spi_sdo. + let mut spi = Spi::<_, _, _, 8>::new(peripherals.SPI0, (spi_sdo, spi_clk)).init( + &mut peripherals.RESETS, + // The SPI system uses the peripheral clock + clocks.peripheral_clock.freq(), + // Transmit data at a rate of 1Mbit. + 1_u32.MHz(), + // Run with SPI Mode 1. This means that the clock line should start high and that data will + // be sampled starting at the first falling edge. + embedded_hal::spi::MODE_1, + ); + + let _ = reset.set_high(); + timer.delay_ms(10); + let _ = board_select.set_low(); + timer.delay_ms(10); + for step in SETUP_PROGRAM { + step.send_command(&mut spi, &mut data_command); + if let Some(delay) = step.delay { + timer.delay_ms(delay); + } + } + + let _ = data_command.set_low(); + let _ = spi.write(&[RAMWR]); + let _ = data_command.set_high(); + let mut bitmap: [u8; 86700] = [0; 86700]; + for i in 0..28900 { + bitmap[i] = 0x55; + } + let _ = spi.write(&bitmap); + let _ = data_command.set_low(); + let _ = spi.write(&[NOP]); + let _ = board_select.set_high(); + + + loop {} +} From fb0e914edf68b82fbece9afbc5638ac0f2f4ea23 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 11 Nov 2024 09:33:45 -0500 Subject: [PATCH 18/34] This gets the screen working, though not correctly --- pico-st7789/src/main.rs | 60 ++++++++++++++++++++++++----------------- 1 file changed, 36 insertions(+), 24 deletions(-) diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 85068f2..aab31b7 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -17,6 +17,10 @@ use rp_pico::{ const XOSC_CRYSTAL_FREQ: u32 = 12_000_000; // MHz, https://forums.raspberrypi.com/viewtopic.php?t=356764 +const ROWS: usize = 320; +const COLUMNS: usize = 240; +const FRAMEBUF: usize = ROWS * COLUMNS * 3; + struct Step { param_cnt: usize, command: u8, @@ -36,8 +40,10 @@ impl Step { { let _ = data_command.set_low(); let _ = spi.write(&[self.command]); - let _ = data_command.set_high(); - let _ = spi.write(&self.params[0..self.param_cnt]); + if self.param_cnt > 0 { + let _ = data_command.set_high(); + let _ = spi.write(&self.params[0..self.param_cnt]); + } } } @@ -57,9 +63,9 @@ const SLPOUT: Step = Step { }; const COLMOD: u8 = 0x3a; const MADCTL: Step = Step { - param_cnt: 0, + param_cnt: 1, command: 0x36, - params: [0, 0, 0, 0], + params: [0x08, 0, 0, 0], delay: None, }; const CASET: u8 = 0x2a; @@ -111,20 +117,20 @@ const SETUP_PROGRAM: [Step; 9] = [ Step { param_cnt: 1, command: COLMOD, - params: [0x55, 0, 0, 0], + params: [0x66, 0, 0, 0], delay: Some(10), }, MADCTL, Step { param_cnt: 4, command: CASET, - params: [0, 0, 0, 170], + params: [0, 0, 0, 240], delay: None, }, Step { param_cnt: 4, command: RASET, - params: [0, 0, (340 as u16 >> 8) as u8, (320 as u16 & 0xff) as u8], + params: [0, 0, (320 >> 8) as u8, (320 & 0xff) as u8], delay: None, }, INVON, @@ -178,8 +184,8 @@ unsafe fn main() -> ! { // let spi_sdi = pins.gpio4.into_function(); // Chip select 1 means the chip is not enabled let mut board_select = pins.gpio13.into_function(); - let mut data_command = pins.gpio14.into_function(); - let mut reset = pins.gpio15.into_function(); + let mut data_command = pins.gpio15.into_function(); + let mut reset = pins.gpio14.into_function(); let _ = reset.set_low(); let _ = board_select.set_high(); @@ -191,10 +197,10 @@ unsafe fn main() -> ! { // The SPI system uses the peripheral clock clocks.peripheral_clock.freq(), // Transmit data at a rate of 1Mbit. - 1_u32.MHz(), + 32_u32.MHz(), // Run with SPI Mode 1. This means that the clock line should start high and that data will // be sampled starting at the first falling edge. - embedded_hal::spi::MODE_1, + embedded_hal::spi::MODE_3, ); let _ = reset.set_high(); @@ -208,18 +214,24 @@ unsafe fn main() -> ! { } } - let _ = data_command.set_low(); - let _ = spi.write(&[RAMWR]); - let _ = data_command.set_high(); - let mut bitmap: [u8; 86700] = [0; 86700]; - for i in 0..28900 { - bitmap[i] = 0x55; + timer.delay_ms(1000); + + let mut bitmap: [u8; FRAMEBUF] = [0; FRAMEBUF]; + + let mut i = 0; + loop { + let _ = board_select.set_low(); + let _ = data_command.set_low(); + let _ = spi.write(&[RAMWR]); + let _ = data_command.set_high(); + let _ = spi.write(&bitmap); + let _ = board_select.set_high(); + + let color = i << 2; + bitmap = [color; FRAMEBUF]; + + i = if i >= 64 { 0 } else { i + 1 }; + + timer.delay_ms(10); } - let _ = spi.write(&bitmap); - let _ = data_command.set_low(); - let _ = spi.write(&[NOP]); - let _ = board_select.set_high(); - - - loop {} } From 54dd0049150f55d0f7c2c7c5bdf1e69d5c4f3028 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Wed, 26 Feb 2025 23:59:10 -0500 Subject: [PATCH 19/34] Tweak the hell out of the code until it shows a small square in the center of the screen --- pico-st7789/src/main.rs | 38 +++++++++++++++++++++++++++----------- rust-toolchain | 2 +- 2 files changed, 28 insertions(+), 12 deletions(-) diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index aab31b7..705929d 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -18,7 +18,7 @@ use rp_pico::{ const XOSC_CRYSTAL_FREQ: u32 = 12_000_000; // MHz, https://forums.raspberrypi.com/viewtopic.php?t=356764 const ROWS: usize = 320; -const COLUMNS: usize = 240; +const COLUMNS: usize = 170; const FRAMEBUF: usize = ROWS * COLUMNS * 3; struct Step { @@ -65,7 +65,7 @@ const COLMOD: u8 = 0x3a; const MADCTL: Step = Step { param_cnt: 1, command: 0x36, - params: [0x08, 0, 0, 0], + params: [0x00, 0, 0, 0], delay: None, }; const CASET: u8 = 0x2a; @@ -111,7 +111,7 @@ const RAMWR: u8 = 0x2c; // DISPON, 10ms delay // turn the display on -const SETUP_PROGRAM: [Step; 9] = [ +const SETUP_PROGRAM: [Step; 8] = [ SWRESET, SLPOUT, Step { @@ -124,15 +124,17 @@ const SETUP_PROGRAM: [Step; 9] = [ Step { param_cnt: 4, command: CASET, - params: [0, 0, 0, 240], + params: [0, 35, 0, 204], delay: None, }, + /* Step { param_cnt: 4, command: RASET, params: [0, 0, (320 >> 8) as u8, (320 & 0xff) as u8], delay: None, }, + */ INVON, NORON, DISPON, @@ -177,6 +179,8 @@ unsafe fn main() -> ! { // An abstraction for a timer which we can use to delay the code. let mut timer = Timer::new(peripherals.TIMER, &mut peripherals.RESETS, &clocks); + let mut led = pins.gpio16.into_function(); + // Grab the clock and data pins for SPI1. For Clock pins and for Data pins, there are only two // pins each on the Pico which can function for SPI1. let spi_clk = pins.gpio2.into_function(); @@ -216,22 +220,34 @@ unsafe fn main() -> ! { timer.delay_ms(1000); - let mut bitmap: [u8; FRAMEBUF] = [0; FRAMEBUF]; + let mut framebuf: [u8; FRAMEBUF] = [0; FRAMEBUF]; - let mut i = 0; + let mut strength = 0; loop { + led.set_high(); let _ = board_select.set_low(); let _ = data_command.set_low(); let _ = spi.write(&[RAMWR]); let _ = data_command.set_high(); - let _ = spi.write(&bitmap); + let _ = spi.write(&framebuf); let _ = board_select.set_high(); - let color = i << 2; - bitmap = [color; FRAMEBUF]; - - i = if i >= 64 { 0 } else { i + 1 }; + for x in 80..90 { + for y in 155..165 { + write_pixel(&mut framebuf, x, y, (0, 0, 63)); + } + } timer.delay_ms(10); + led.set_low(); + + timer.delay_ms(1000); } } + +fn write_pixel(framebuf: &mut[u8], x: usize, y: usize, color: (u8, u8, u8)) { + framebuf[(y * COLUMNS + x) * 3 + 0] = color.0 << 2; + framebuf[(y * COLUMNS + x) * 3 + 1] = color.1 << 2; + framebuf[(y * COLUMNS + x) * 3 + 2] = color.2 << 2; +} + diff --git a/rust-toolchain b/rust-toolchain index b96bf13..f5d4290 100644 --- a/rust-toolchain +++ b/rust-toolchain @@ -1,4 +1,4 @@ [toolchain] -channel = "1.81.0" +channel = "1.85.0" targets = [ "wasm32-unknown-unknown", "thumbv6m-none-eabi" ] components = [ "rustfmt", "rust-analyzer", "clippy" ] From f004aa3514f84917b3a543f1270f2f39372b13c3 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Thu, 27 Feb 2025 00:18:32 -0500 Subject: [PATCH 20/34] Use the onboard LED and try to transmit at 2MB --- pico-st7789/src/main.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 705929d..acc1e3e 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -179,7 +179,7 @@ unsafe fn main() -> ! { // An abstraction for a timer which we can use to delay the code. let mut timer = Timer::new(peripherals.TIMER, &mut peripherals.RESETS, &clocks); - let mut led = pins.gpio16.into_function(); + let mut led = pins.led.into_function(); // Grab the clock and data pins for SPI1. For Clock pins and for Data pins, there are only two // pins each on the Pico which can function for SPI1. @@ -201,7 +201,7 @@ unsafe fn main() -> ! { // The SPI system uses the peripheral clock clocks.peripheral_clock.freq(), // Transmit data at a rate of 1Mbit. - 32_u32.MHz(), + 2_u32.MHz(), // Run with SPI Mode 1. This means that the clock line should start high and that data will // be sampled starting at the first falling edge. embedded_hal::spi::MODE_3, From a69a864dca08ae2b95df9063df7e1bb509f82ebc Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Thu, 27 Feb 2025 08:50:57 -0500 Subject: [PATCH 21/34] Rename framebuf --- pico-st7789/src/main.rs | 22 +++++++++++++++++----- 1 file changed, 17 insertions(+), 5 deletions(-) diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index acc1e3e..61534c4 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -21,6 +21,13 @@ const ROWS: usize = 320; const COLUMNS: usize = 170; const FRAMEBUF: usize = ROWS * COLUMNS * 3; +/* +struct Frame { + columns: usize, + buf: [u8; FRAMEBUF], +} +*/ + struct Step { param_cnt: usize, command: u8, @@ -220,7 +227,13 @@ unsafe fn main() -> ! { timer.delay_ms(1000); - let mut framebuf: [u8; FRAMEBUF] = [0; FRAMEBUF]; + let mut frame: [u8; FRAMEBUF] = [0; FRAMEBUF]; + /* + let mut frame = Frame { + columns: COLUMNS, + buf: [0; FRAMEBUF], + }; + */ let mut strength = 0; loop { @@ -229,12 +242,12 @@ unsafe fn main() -> ! { let _ = data_command.set_low(); let _ = spi.write(&[RAMWR]); let _ = data_command.set_high(); - let _ = spi.write(&framebuf); + let _ = spi.write(&frame); let _ = board_select.set_high(); for x in 80..90 { for y in 155..165 { - write_pixel(&mut framebuf, x, y, (0, 0, 63)); + write_pixel(&mut frame, x, y, (0, 0, 63)); } } @@ -245,9 +258,8 @@ unsafe fn main() -> ! { } } -fn write_pixel(framebuf: &mut[u8], x: usize, y: usize, color: (u8, u8, u8)) { +fn write_pixel(framebuf: &mut [u8], x: usize, y: usize, color: (u8, u8, u8)) { framebuf[(y * COLUMNS + x) * 3 + 0] = color.0 << 2; framebuf[(y * COLUMNS + x) * 3 + 1] = color.1 << 2; framebuf[(y * COLUMNS + x) * 3 + 2] = color.2 << 2; } - From 45dc19c32943be553b30badcc8f3c2518f769627 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Thu, 27 Feb 2025 08:57:16 -0500 Subject: [PATCH 22/34] Move board control into a self-contained object --- pico-st7789/src/main.rs | 145 +++------------------------- pico-st7789/src/st7789.rs | 196 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 210 insertions(+), 131 deletions(-) create mode 100644 pico-st7789/src/st7789.rs diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 61534c4..532810c 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -15,6 +15,10 @@ use rp_pico::{ pac, Pins, }; +mod st7789; +use st7789::{ST7789Display, SETUP_PROGRAM}; +pub use st7789::Step; + const XOSC_CRYSTAL_FREQ: u32 = 12_000_000; // MHz, https://forums.raspberrypi.com/viewtopic.php?t=356764 const ROWS: usize = 320; @@ -28,125 +32,6 @@ struct Frame { } */ -struct Step { - param_cnt: usize, - command: u8, - params: [u8; 4], - delay: Option<u32>, -} - -impl Step { - fn send_command<D, Pinout, P>( - &self, - spi: &mut Spi<Enabled, D, Pinout, 8>, - data_command: &mut Pin<P, FunctionSio<SioOutput>, PullDown>, - ) where - D: SpiDevice, - Pinout: ValidSpiPinout<D>, - P: PinId, - { - let _ = data_command.set_low(); - let _ = spi.write(&[self.command]); - if self.param_cnt > 0 { - let _ = data_command.set_high(); - let _ = spi.write(&self.params[0..self.param_cnt]); - } - } -} - -const NOP: u8 = 0x00; - -const SWRESET: Step = Step { - param_cnt: 0, - command: 0x01, - params: [0, 0, 0, 0], - delay: Some(150), -}; -const SLPOUT: Step = Step { - param_cnt: 0, - command: 0x11, - params: [0, 0, 0, 0], - delay: Some(10), -}; -const COLMOD: u8 = 0x3a; -const MADCTL: Step = Step { - param_cnt: 1, - command: 0x36, - params: [0x00, 0, 0, 0], - delay: None, -}; -const CASET: u8 = 0x2a; -const RASET: u8 = 0x2b; -const INVON: Step = Step { - param_cnt: 0, - command: 0x21, - params: [0, 0, 0, 0], - delay: Some(10), -}; -const NORON: Step = Step { - param_cnt: 0, - command: 0x13, - params: [0, 0, 0, 0], - delay: Some(10), -}; -const DISPON: Step = Step { - param_cnt: 0, - command: 0x29, - params: [0, 0, 0, 0], - delay: Some(10), -}; -const RAMWR: u8 = 0x2c; - -// Adafruit setup instructions -// SWRESET (0x01), 150ms delay -// SLPOUT (0x11), 10ms delay -// COLMOD (0x3a) 0x55 (65K RGB, 16bit/pixel), 10ms delay -// MADCTL (0x36) 0x00, -// memory data access control, RGB -// CASET 0x00, 0, 0, 170, -// column address set, 4 parameters -// 0x00, 0x00 indicates xstart is 0 -// 0x00, 170 indicates xend is 170 -// RASET 0x00, 0, 320 >> 8, 320 & 0xFF, -// row address set, 4 parameters -// 0x00, 0x00 indicates ystart is 0 -// 3230 >> 8, 320 & 0xff indicates that 320 is the last y address -// INVON, 10ms delay -// invert the display -// NORON, 10ms delay -// normal display mode -// DISPON, 10ms delay -// turn the display on - -const SETUP_PROGRAM: [Step; 8] = [ - SWRESET, - SLPOUT, - Step { - param_cnt: 1, - command: COLMOD, - params: [0x66, 0, 0, 0], - delay: Some(10), - }, - MADCTL, - Step { - param_cnt: 4, - command: CASET, - params: [0, 35, 0, 204], - delay: None, - }, - /* - Step { - param_cnt: 4, - command: RASET, - params: [0, 0, (320 >> 8) as u8, (320 & 0xff) as u8], - delay: None, - }, - */ - INVON, - NORON, - DISPON, -]; - #[entry] unsafe fn main() -> ! { // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico. @@ -214,36 +99,34 @@ unsafe fn main() -> ! { embedded_hal::spi::MODE_3, ); + let mut display = ST7789Display::new(board_select, data_command, spi); + let _ = reset.set_high(); timer.delay_ms(10); - let _ = board_select.set_low(); - timer.delay_ms(10); for step in SETUP_PROGRAM { - step.send_command(&mut spi, &mut data_command); - if let Some(delay) = step.delay { - timer.delay_ms(delay); - } + let mut display = display.acquire(); + display.send_command(&step, &mut timer); } timer.delay_ms(1000); let mut frame: [u8; FRAMEBUF] = [0; FRAMEBUF]; - /* - let mut frame = Frame { - columns: COLUMNS, - buf: [0; FRAMEBUF], - }; - */ let mut strength = 0; loop { led.set_high(); + /* let _ = board_select.set_low(); let _ = data_command.set_low(); let _ = spi.write(&[RAMWR]); let _ = data_command.set_high(); let _ = spi.write(&frame); let _ = board_select.set_high(); + */ + { + let display = display.acquire(); + display.send_buf(&frame); + } for x in 80..90 { for y in 155..165 { diff --git a/pico-st7789/src/st7789.rs b/pico-st7789/src/st7789.rs new file mode 100644 index 0000000..4e19441 --- /dev/null +++ b/pico-st7789/src/st7789.rs @@ -0,0 +1,196 @@ +use embedded_hal::{delay::DelayNs, digital::OutputPin, spi::SpiBus}; +use rp_pico::hal::{ + gpio::{FunctionSio, Pin, PinId, PullDown, SioOutput}, + spi::{Enabled, SpiDevice, ValidSpiPinout}, + Spi, Timer, +}; + +pub struct Step { + param_cnt: usize, + command: u8, + params: [u8; 4], + delay: Option<u32>, +} + +impl Step { + pub fn send_command<D, Pinout, P>( + &self, + spi: &mut Spi<Enabled, D, Pinout, 8>, + data_command: &mut Pin<P, FunctionSio<SioOutput>, PullDown>, + ) where + D: SpiDevice, + Pinout: ValidSpiPinout<D>, + P: PinId, + { + let _ = data_command.set_low(); + let _ = spi.write(&[self.command]); + if self.param_cnt > 0 { + let _ = data_command.set_high(); + let _ = spi.write(&self.params[0..self.param_cnt]); + } + } +} + +const NOP: u8 = 0x00; + +const SWRESET: Step = Step { + param_cnt: 0, + command: 0x01, + params: [0, 0, 0, 0], + delay: Some(150), +}; +const SLPOUT: Step = Step { + param_cnt: 0, + command: 0x11, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const COLMOD: u8 = 0x3a; +const MADCTL: Step = Step { + param_cnt: 1, + command: 0x36, + params: [0x00, 0, 0, 0], + delay: None, +}; +const CASET: u8 = 0x2a; +const RASET: u8 = 0x2b; +const INVON: Step = Step { + param_cnt: 0, + command: 0x21, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const NORON: Step = Step { + param_cnt: 0, + command: 0x13, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const DISPON: Step = Step { + param_cnt: 0, + command: 0x29, + params: [0, 0, 0, 0], + delay: Some(10), +}; +const RAMWR: u8 = 0x2c; + +// Adafruit setup instructions +// SWRESET (0x01), 150ms delay +// SLPOUT (0x11), 10ms delay +// COLMOD (0x3a) 0x55 (65K RGB, 16bit/pixel), 10ms delay +// MADCTL (0x36) 0x00, +// memory data access control, RGB +// CASET 0x00, 0, 0, 170, +// column address set, 4 parameters +// 0x00, 0x00 indicates xstart is 0 +// 0x00, 170 indicates xend is 170 +// RASET 0x00, 0, 320 >> 8, 320 & 0xFF, +// row address set, 4 parameters +// 0x00, 0x00 indicates ystart is 0 +// 3230 >> 8, 320 & 0xff indicates that 320 is the last y address +// INVON, 10ms delay +// invert the display +// NORON, 10ms delay +// normal display mode +// DISPON, 10ms delay +// turn the display on + +pub const SETUP_PROGRAM: [Step; 8] = [ + SWRESET, + SLPOUT, + Step { + param_cnt: 1, + command: COLMOD, + params: [0x66, 0, 0, 0], + delay: Some(10), + }, + MADCTL, + Step { + param_cnt: 4, + command: CASET, + params: [0, 35, 0, 204], + delay: None, + }, + /* + Step { + param_cnt: 4, + command: RASET, + params: [0, 0, (320 >> 8) as u8, (320 & 0xff) as u8], + delay: None, + }, + */ + INVON, + NORON, + DISPON, +]; + +pub struct ST7789Display< + BoardSelectId: PinId, + DataCommandId: PinId, + D: SpiDevice, + Pinout: ValidSpiPinout<D>, +> { + inner: ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout>, +} + +impl<BoardSelectId: PinId, DataCommandId: PinId, D: SpiDevice, Pinout: ValidSpiPinout<D>> + ST7789Display<BoardSelectId, DataCommandId, D, Pinout> +{ + pub fn new( + board_select: Pin<BoardSelectId, FunctionSio<SioOutput>, PullDown>, + data_command: Pin<DataCommandId, FunctionSio<SioOutput>, PullDown>, + spi: Spi<Enabled, D, Pinout, 8>, + ) -> Self { + Self { + inner: ST7789DisplayEnabled { + board_select, + data_command, + spi, + }, + } + } + + pub fn acquire( + &mut self, + ) -> &mut ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout> { + self.inner.board_select.set_low(); + &mut self.inner + } +} + +pub struct ST7789DisplayEnabled< + BoardSelectId: PinId, + DataCommandId: PinId, + D: SpiDevice, + Pinout: ValidSpiPinout<D>, +> { + board_select: Pin<BoardSelectId, FunctionSio<SioOutput>, PullDown>, + data_command: Pin<DataCommandId, FunctionSio<SioOutput>, PullDown>, + spi: Spi<Enabled, D, Pinout, 8>, +} + +impl<BoardSelectId: PinId, DataCommandId: PinId, D: SpiDevice, Pinout: ValidSpiPinout<D>> + ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout> +{ + pub fn send_command(&mut self, step: &Step, timer: &mut Timer) { + step.send_command(&mut self.spi, &mut self.data_command); + if let Some(delay) = step.delay { + timer.delay_ms(delay); + } + } + + pub fn send_buf(&mut self, frame: &[u8]) { + let _ = self.data_command.set_low(); + let _ = self.spi.write(&[RAMWR]); + let _ = self.data_command.set_high(); + let _ = self.spi.write(&frame); + } +} + +impl<BoardSelectId: PinId, DataCommandId: PinId, D: SpiDevice, Pinout: ValidSpiPinout<D>> Drop + for ST7789DisplayEnabled<BoardSelectId, DataCommandId, D, Pinout> +{ + fn drop(&mut self) { + self.board_select.set_high(); + } +} From 21c6f30a7d84768e508b1b9d7df48f5ac6c78d72 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Thu, 27 Feb 2025 09:46:46 -0500 Subject: [PATCH 23/34] Add an (unused) DISPOFF step --- pico-st7789/src/main.rs | 10 +--------- pico-st7789/src/st7789.rs | 8 ++++++++ 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 532810c..cdb6130 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -93,7 +93,7 @@ unsafe fn main() -> ! { // The SPI system uses the peripheral clock clocks.peripheral_clock.freq(), // Transmit data at a rate of 1Mbit. - 2_u32.MHz(), + 1_u32.MHz(), // Run with SPI Mode 1. This means that the clock line should start high and that data will // be sampled starting at the first falling edge. embedded_hal::spi::MODE_3, @@ -115,14 +115,6 @@ unsafe fn main() -> ! { let mut strength = 0; loop { led.set_high(); - /* - let _ = board_select.set_low(); - let _ = data_command.set_low(); - let _ = spi.write(&[RAMWR]); - let _ = data_command.set_high(); - let _ = spi.write(&frame); - let _ = board_select.set_high(); - */ { let display = display.acquire(); display.send_buf(&frame); diff --git a/pico-st7789/src/st7789.rs b/pico-st7789/src/st7789.rs index 4e19441..c2bb0e2 100644 --- a/pico-st7789/src/st7789.rs +++ b/pico-st7789/src/st7789.rs @@ -66,6 +66,12 @@ const NORON: Step = Step { params: [0, 0, 0, 0], delay: Some(10), }; +const DISPOFF: Step = Step { + param_cnt: 0, + command: 0x28, + params: [0, 0, 0, 0], + delay: Some(10), +}; const DISPON: Step = Step { param_cnt: 0, command: 0x29, @@ -180,10 +186,12 @@ impl<BoardSelectId: PinId, DataCommandId: PinId, D: SpiDevice, Pinout: ValidSpiP } pub fn send_buf(&mut self, frame: &[u8]) { + // let _ = DISPOFF.send_command(&mut self.spi, &mut self.data_command); let _ = self.data_command.set_low(); let _ = self.spi.write(&[RAMWR]); let _ = self.data_command.set_high(); let _ = self.spi.write(&frame); + // let _ = DISPON.send_command(&mut self.spi, &mut self.data_command); } } From 85e5d0bb5e99acbfe2b2f9716dc0e7fb6a1ec9cd Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 3 Mar 2025 10:24:50 -0500 Subject: [PATCH 24/34] Worked out a font and a canvas I'm adding a 16-segment-based font here, and have encoded the numbers 0-10. I've also worked out a way to make a Canvas structure not crash the pico. --- Cargo.lock | 52 ++++- pico-st7789/Cargo.toml | 1 + pico-st7789/src/drawing.rs | 125 ++++++++++++ pico-st7789/src/font.rs | 398 +++++++++++++++++++++++++++++++++++++ pico-st7789/src/main.rs | 88 +++++--- 5 files changed, 634 insertions(+), 30 deletions(-) create mode 100644 pico-st7789/src/drawing.rs create mode 100644 pico-st7789/src/font.rs diff --git a/Cargo.lock b/Cargo.lock index bb828ca..1772741 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1,6 +1,6 @@ # This file is automatically @generated by Cargo. # It is not intended for manual editing. -version = 3 +version = 4 [[package]] name = "addr2line" @@ -693,6 +693,12 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "const-default" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b396d1f76d455557e1218ec8066ae14bba60b4b36ecd55577ba979f5db7ecaa" + [[package]] name = "const-oid" version = "0.9.6" @@ -1057,6 +1063,18 @@ dependencies = [ "serde 1.0.218", ] +[[package]] +name = "embedded-alloc" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f2de9133f68db0d4627ad69db767726c99ff8585272716708227008d3f1bddd" +dependencies = [ + "const-default", + "critical-section", + "linked_list_allocator", + "rlsf", +] + [[package]] name = "embedded-dma" version = "0.2.0" @@ -2870,6 +2888,12 @@ dependencies = [ "fixed", ] +[[package]] +name = "linked_list_allocator" +version = "0.10.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9afa463f5405ee81cdb9cc2baf37e08ec7e4c8209442b5d72c04cfb2cd6e6286" + [[package]] name = "linux-raw-sys" version = "0.4.15" @@ -3540,6 +3564,7 @@ name = "pico-st7789" version = "0.1.0" dependencies = [ "cortex-m-rt", + "embedded-alloc", "embedded-hal 1.0.0", "fugit", "panic-halt", @@ -4131,6 +4156,18 @@ dependencies = [ "thiserror 1.0.69", ] +[[package]] +name = "rlsf" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "222fb240c3286247ecdee6fa5341e7cdad0ffdf8e7e401d9937f2d58482a20bf" +dependencies = [ + "cfg-if", + "const-default", + "libc", + "svgbobdoc", +] + [[package]] name = "rp-pico" version = "0.9.0" @@ -4844,6 +4881,19 @@ version = "2.6.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" +[[package]] +name = "svgbobdoc" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f2c04b93fc15d79b39c63218f15e3fdffaa4c227830686e3b7c5f41244eb3e50" +dependencies = [ + "base64 0.9.3", + "proc-macro2", + "quote", + "syn 1.0.109", + "unicode-width", +] + [[package]] name = "syn" version = "1.0.109" diff --git a/pico-st7789/Cargo.toml b/pico-st7789/Cargo.toml index 6d8012e..001c708 100644 --- a/pico-st7789/Cargo.toml +++ b/pico-st7789/Cargo.toml @@ -5,6 +5,7 @@ edition = "2021" [dependencies] cortex-m-rt = "0.7.3" +embedded-alloc = "0.6.0" embedded-hal = "1.0.0" fugit = "0.3.7" panic-halt = "1.0.0" diff --git a/pico-st7789/src/drawing.rs b/pico-st7789/src/drawing.rs new file mode 100644 index 0000000..e02e868 --- /dev/null +++ b/pico-st7789/src/drawing.rs @@ -0,0 +1,125 @@ +// a a a +// f b +// f b +// f b +// g g g +// e c +// e c +// e c +// d d d + +use alloc::{boxed::Box, vec::Vec}; + +pub struct Canvas<'a> { + pub buf: &'a mut [u8], + pub width: usize, +} + +impl<'a> Canvas<'a> { + pub fn new(buf: &'a mut [u8], columns: usize) -> Self { + Self { + buf, + width: columns, + } + } + + pub fn set_pixel(&mut self, x: usize, y: usize, color: (u8, u8, u8)) { + self.buf[(y * self.width + x) * 3 + 0] = color.0 << 2; + self.buf[(y * self.width + x) * 3 + 1] = color.1 << 2; + self.buf[(y * self.width + x) * 3 + 2] = color.2 << 2; + } +} + +/* +pub const DIGIT_WIDTH: usize = 5; +pub const DIGIT_HEIGHT: usize = 9; + +pub const ZERO_SEVEN_SEGMENT: [bool; 7] = [true, true, true, true, true, true, false]; + +pub const ONE_SEVEN_SEGMENT: [bool; 7] = [false, true, true, false, false, false, false]; + +pub const TWO_SEVEN_SEGMENT: [bool; 7] = [true, true, false, true, true, false, true]; + +pub const THREE_SEVEN_SEGMENT: [bool; 7] = [true, true, true, true, false, false, true]; + +pub const FOUR_SEVEN_SEGMENT: [bool; 7] = [false, true, true, false, false, true, true]; + +pub const FIVE_SEVEN_SEGMENT: [bool; 7] = [true, false, true, true, false, true, true]; + +pub const SIX_SEVEN_SEGMENT: [bool; 7] = [true, false, true, true, true, true, true]; + +pub const SEVEN_SEVEN_SEGMENT: [bool; 7] = [true, true, true, false, false, false, false]; + +pub const EIGHT_SEVEN_SEGMENT: [bool; 7] = [true, true, true, true, true, true, true]; + +pub const NINE_SEVEN_SEGMENT: [bool; 7] = [true, true, true, false, false, true, true]; + +pub const SEVEN_SEGMENT_FONT: [[bool; 7]; 10] = [ + ZERO_SEVEN_SEGMENT, + ONE_SEVEN_SEGMENT, + TWO_SEVEN_SEGMENT, + THREE_SEVEN_SEGMENT, + FOUR_SEVEN_SEGMENT, + FIVE_SEVEN_SEGMENT, + SIX_SEVEN_SEGMENT, + SEVEN_SEVEN_SEGMENT, + EIGHT_SEVEN_SEGMENT, + NINE_SEVEN_SEGMENT, +]; +*/ + +/* +pub fn draw_pixel(frame: &mut Canvas, x: usize, y: usize, color: (u8, u8, u8)) { + frame.buf[(y * frame.width + x) * 3 + 0] = color.0 << 2; + frame.buf[(y * frame.width + x) * 3 + 1] = color.1 << 2; + frame.buf[(y * frame.width + x) * 3 + 2] = color.2 << 2; +} +*/ + +/* +pub fn draw_seven_segment( + digit: u8, + frame: &mut [u8], + width: usize, + x: usize, + y: usize, + color: (u8, u8, u8), +) { + let segments = SEVEN_SEGMENT_FONT[digit as usize]; + if segments[0] { + write_pixel(frame, x + 1, y, color); + write_pixel(frame, x + 2, y, color); + write_pixel(frame, x + 3, y, color); + } + if segments[1] { + write_pixel(frame, x + 4, y + 1, color); + write_pixel(frame, x + 4, y + 2, color); + write_pixel(frame, x + 4, y + 3, color); + } + if segments[2] { + write_pixel(frame, x + 4, y + 5, color); + write_pixel(frame, x + 4, y + 6, color); + write_pixel(frame, x + 4, y + 7, color); + } + if segments[3] { + write_pixel(frame, x + 1, y + 8, color); + write_pixel(frame, x + 2, y + 8, color); + write_pixel(frame, x + 3, y + 8, color); + } + if segments[4] { + write_pixel(frame, x, y + 5, color); + write_pixel(frame, x, y + 6, color); + write_pixel(frame, x, y + 7, color); + } + if segments[5] { + write_pixel(frame, x, y + 1, color); + write_pixel(frame, x, y + 2, color); + write_pixel(frame, x, y + 3, color); + } + if segments[6] { + write_pixel(frame, x + 1, y + 4, color); + write_pixel(frame, x + 2, y + 4, color); + write_pixel(frame, x + 3, y + 4, color); + } +} +*/ diff --git a/pico-st7789/src/font.rs b/pico-st7789/src/font.rs new file mode 100644 index 0000000..a2f4c3e --- /dev/null +++ b/pico-st7789/src/font.rs @@ -0,0 +1,398 @@ +use alloc::collections::btree_map::BTreeMap; + +use crate::drawing::Canvas; + +pub trait Font<A> { + fn glyph(&self, c: char) -> &A; +} + +pub trait Glyph { + fn draw(&self, canvas: &mut Canvas, x: usize, y: usize, color: (u8, u8, u8)); +} + +// Sixteen Segments +// +// a1 a2 +// f h i jb +// f i j b +// f hij b +// g1 g2 +// e klm c +// e k l m c +// ek l mc +// d1 d2 + +pub struct SixteenSegmentGlyph { + a1: bool, + a2: bool, + b: bool, + c: bool, + d1: bool, + d2: bool, + e: bool, + f: bool, + g1: bool, + g2: bool, + h: bool, + i: bool, + j: bool, + k: bool, + l: bool, + m: bool, +} + +pub struct SixteenSegmentFont(BTreeMap<char, SixteenSegmentGlyph>); + +impl SixteenSegmentFont { + pub fn new() -> Self { + let mut font = BTreeMap::new(); + font.insert( + '*', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: true, + g2: true, + h: true, + i: true, + j: true, + k: true, + l: true, + m: true, + }, + ); + font.insert( + '0', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: true, + k: true, + l: false, + m: false, + }, + ); + font.insert( + '1', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: true, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '2', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: false, + d1: true, + d2: true, + e: true, + f: false, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '3', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: false, + f: false, + g1: false, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '4', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: false, + d2: false, + e: false, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '5', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: false, + d1: true, + d2: true, + e: false, + f: true, + g1: true, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: true, + }, + ); + font.insert( + '6', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '7', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '8', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '9', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: false, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + + Self(font) + } +} + +impl Font<SixteenSegmentGlyph> for SixteenSegmentFont { + fn glyph(&self, c: char) -> &SixteenSegmentGlyph { + self.0.get(&c).unwrap() + } +} + +impl Glyph for SixteenSegmentGlyph { + fn draw(&self, canvas: &mut Canvas, x: usize, y: usize, color: (u8, u8, u8)) { + if self.a1 { + canvas.set_pixel(x + 1, y, color); + canvas.set_pixel(x + 2, y, color); + canvas.set_pixel(x + 3, y, color); + } + if self.a2 { + canvas.set_pixel(x + 5, y, color); + canvas.set_pixel(x + 6, y, color); + canvas.set_pixel(x + 7, y, color); + } + if self.b { + canvas.set_pixel(x + 8, y + 1, color); + canvas.set_pixel(x + 8, y + 2, color); + canvas.set_pixel(x + 8, y + 3, color); + canvas.set_pixel(x + 8, y + 4, color); + canvas.set_pixel(x + 8, y + 5, color); + } + if self.c { + canvas.set_pixel(x + 8, y + 7, color); + canvas.set_pixel(x + 8, y + 8, color); + canvas.set_pixel(x + 8, y + 9, color); + canvas.set_pixel(x + 8, y + 10, color); + canvas.set_pixel(x + 8, y + 11, color); + } + + if self.d1 { + canvas.set_pixel(x + 1, y + 12, color); + canvas.set_pixel(x + 2, y + 12, color); + canvas.set_pixel(x + 3, y + 12, color); + } + if self.d2 { + canvas.set_pixel(x + 5, y + 12, color); + canvas.set_pixel(x + 6, y + 12, color); + canvas.set_pixel(x + 7, y + 12, color); + } + + if self.e { + canvas.set_pixel(x, y + 7, color); + canvas.set_pixel(x, y + 8, color); + canvas.set_pixel(x, y + 9, color); + canvas.set_pixel(x, y + 10, color); + canvas.set_pixel(x, y + 11, color); + } + if self.f { + canvas.set_pixel(x, y + 1, color); + canvas.set_pixel(x, y + 2, color); + canvas.set_pixel(x, y + 3, color); + canvas.set_pixel(x, y + 4, color); + canvas.set_pixel(x, y + 5, color); + } + + if self.g1 { + canvas.set_pixel(x + 1, y + 6, color); + canvas.set_pixel(x + 2, y + 6, color); + canvas.set_pixel(x + 3, y + 6, color); + } + if self.g2 { + canvas.set_pixel(x + 5, y + 6, color); + canvas.set_pixel(x + 6, y + 6, color); + canvas.set_pixel(x + 7, y + 6, color); + } + + if self.h { + canvas.set_pixel(x + 1, y + 1, color); + canvas.set_pixel(x + 1, y + 2, color); + canvas.set_pixel(x + 2, y + 3, color); + canvas.set_pixel(x + 3, y + 4, color); + canvas.set_pixel(x + 3, y + 5, color); + } + if self.i { + canvas.set_pixel(x + 4, y + 1, color); + canvas.set_pixel(x + 4, y + 2, color); + canvas.set_pixel(x + 4, y + 3, color); + canvas.set_pixel(x + 4, y + 4, color); + canvas.set_pixel(x + 4, y + 5, color); + } + if self.j { + canvas.set_pixel(x + 7, y + 1, color); + canvas.set_pixel(x + 7, y + 2, color); + canvas.set_pixel(x + 6, y + 3, color); + canvas.set_pixel(x + 5, y + 4, color); + canvas.set_pixel(x + 5, y + 5, color); + } + if self.k { + canvas.set_pixel(x + 3, y + 7, color); + canvas.set_pixel(x + 3, y + 8, color); + canvas.set_pixel(x + 2, y + 9, color); + canvas.set_pixel(x + 1, y + 10, color); + canvas.set_pixel(x + 1, y + 11, color); + } + if self.l { + canvas.set_pixel(x + 4, y + 7, color); + canvas.set_pixel(x + 4, y + 8, color); + canvas.set_pixel(x + 4, y + 9, color); + canvas.set_pixel(x + 4, y + 10, color); + canvas.set_pixel(x + 4, y + 11, color); + } + if self.m { + canvas.set_pixel(x + 5, y + 7, color); + canvas.set_pixel(x + 5, y + 8, color); + canvas.set_pixel(x + 6, y + 9, color); + canvas.set_pixel(x + 7, y + 10, color); + canvas.set_pixel(x + 7, y + 11, color); + } + } +} diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index cdb6130..33deb64 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -1,23 +1,26 @@ #![no_main] #![no_std] -use embedded_hal::{delay::DelayNs, digital::OutputPin, spi::SpiBus}; +extern crate alloc; + +use embedded_alloc::LlffHeap as Heap; +use embedded_hal::{delay::DelayNs, digital::OutputPin}; use fugit::RateExtU32; use panic_halt as _; use rp_pico::{ entry, - hal::{ - clocks::init_clocks_and_plls, - gpio::{FunctionSio, Pin, PinId, PullDown, SioOutput}, - spi::{Enabled, Spi, SpiDevice, ValidSpiPinout}, - Clock, Sio, Timer, Watchdog, - }, + hal::{clocks::init_clocks_and_plls, spi::Spi, Clock, Sio, Timer, Watchdog}, pac, Pins, }; +mod drawing; +use drawing::{Canvas}; + +mod font; +use font::{Font, Glyph, SixteenSegmentFont}; + mod st7789; use st7789::{ST7789Display, SETUP_PROGRAM}; -pub use st7789::Step; const XOSC_CRYSTAL_FREQ: u32 = 12_000_000; // MHz, https://forums.raspberrypi.com/viewtopic.php?t=356764 @@ -25,15 +28,21 @@ const ROWS: usize = 320; const COLUMNS: usize = 170; const FRAMEBUF: usize = ROWS * COLUMNS * 3; -/* -struct Frame { - columns: usize, - buf: [u8; FRAMEBUF], -} -*/ +#[global_allocator] +static HEAP: Heap = Heap::empty(); + #[entry] unsafe fn main() -> ! { + { + use core::mem::MaybeUninit; + const HEAP_SIZE: usize = 1024; + static mut HEAP_MEM: [MaybeUninit<u8>; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; + unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) } + } + + let font_sixteen = SixteenSegmentFont::new(); + // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico. let mut peripherals = pac::Peripherals::take().unwrap(); @@ -92,8 +101,8 @@ unsafe fn main() -> ! { &mut peripherals.RESETS, // The SPI system uses the peripheral clock clocks.peripheral_clock.freq(), - // Transmit data at a rate of 1Mbit. - 1_u32.MHz(), + // Transmit data at a rate of 32Mbit. + 32_u32.MHz(), // Run with SPI Mode 1. This means that the clock line should start high and that data will // be sampled starting at the first falling edge. embedded_hal::spi::MODE_3, @@ -110,31 +119,52 @@ unsafe fn main() -> ! { timer.delay_ms(1000); - let mut frame: [u8; FRAMEBUF] = [0; FRAMEBUF]; + let mut framebuf = [0; FRAMEBUF]; + let mut canvas = Canvas::new(&mut framebuf, COLUMNS); - let mut strength = 0; loop { - led.set_high(); { let display = display.acquire(); - display.send_buf(&frame); + let _ = led.set_high(); + timer.delay_ms(100); + display.send_buf(&canvas.buf); + let _ = led.set_low(); } + /* + draw_seven_segment(0, &mut frame, 0, 0, (255, 255, 255)); + draw_seven_segment(1, &mut frame, 7, 0, (255, 255, 255)); + draw_seven_segment(2, &mut frame, 14, 0, (255, 255, 255)); + draw_seven_segment(3, &mut frame, 21, 0, (255, 255, 255)); + draw_seven_segment(4, &mut frame, 28, 0, (255, 255, 255)); + draw_seven_segment(5, &mut frame, 35, 0, (255, 255, 255)); + draw_seven_segment(6, &mut frame, 42, 0, (255, 255, 255)); + draw_seven_segment(7, &mut frame, 49, 0, (255, 255, 255)); + draw_seven_segment(8, &mut frame, 56, 0, (255, 255, 255)); + draw_seven_segment(9, &mut frame, 63, 0, (255, 255, 255)); + */ + + font_sixteen.glyph('*').draw(&mut canvas, 0, 10, (255, 255, 255)); + font_sixteen.glyph('0').draw(&mut canvas, 11, 10, (255, 255, 255)); + font_sixteen.glyph('1').draw(&mut canvas, 22, 10, (255, 255, 255)); + font_sixteen.glyph('2').draw(&mut canvas, 33, 10, (255, 255, 255)); + font_sixteen.glyph('3').draw(&mut canvas, 44, 10, (255, 255, 255)); + font_sixteen.glyph('4').draw(&mut canvas, 55, 10, (255, 255, 255)); + font_sixteen.glyph('5').draw(&mut canvas, 66, 10, (255, 255, 255)); + font_sixteen.glyph('6').draw(&mut canvas, 77, 10, (255, 255, 255)); + font_sixteen.glyph('7').draw(&mut canvas, 88, 10, (255, 255, 255)); + font_sixteen.glyph('8').draw(&mut canvas, 99, 10, (255, 255, 255)); + font_sixteen.glyph('9').draw(&mut canvas, 110, 10, (255, 255, 255)); + + /* for x in 80..90 { for y in 155..165 { - write_pixel(&mut frame, x, y, (0, 0, 63)); + draw_pixel(&mut frame, x, y, (0, 0, 63)); } } - - timer.delay_ms(10); - led.set_low(); + */ timer.delay_ms(1000); } } -fn write_pixel(framebuf: &mut [u8], x: usize, y: usize, color: (u8, u8, u8)) { - framebuf[(y * COLUMNS + x) * 3 + 0] = color.0 << 2; - framebuf[(y * COLUMNS + x) * 3 + 1] = color.1 << 2; - framebuf[(y * COLUMNS + x) * 3 + 2] = color.2 << 2; -} From aea858dd1738da36510ff2f681555dea3029064c Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Mon, 3 Mar 2025 23:18:28 -0500 Subject: [PATCH 25/34] Do a static buffer allocation for the Canvas --- pico-st7789/src/drawing.rs | 14 ++++++++------ pico-st7789/src/main.rs | 5 ++++- 2 files changed, 12 insertions(+), 7 deletions(-) diff --git a/pico-st7789/src/drawing.rs b/pico-st7789/src/drawing.rs index e02e868..d8e780f 100644 --- a/pico-st7789/src/drawing.rs +++ b/pico-st7789/src/drawing.rs @@ -10,16 +10,18 @@ use alloc::{boxed::Box, vec::Vec}; -pub struct Canvas<'a> { - pub buf: &'a mut [u8], +static mut BUF: [u8; 163200] = [0; 163200]; + +pub struct Canvas { + pub buf: &'static mut [u8; 163200], pub width: usize, } -impl<'a> Canvas<'a> { - pub fn new(buf: &'a mut [u8], columns: usize) -> Self { +impl Canvas { + pub fn new() -> Self { Self { - buf, - width: columns, + buf: unsafe { &mut BUF }, + width: 170, } } diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 33deb64..90d3789 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -119,15 +119,18 @@ unsafe fn main() -> ! { timer.delay_ms(1000); + /* let mut framebuf = [0; FRAMEBUF]; let mut canvas = Canvas::new(&mut framebuf, COLUMNS); + */ + let mut canvas = Canvas::new(); loop { { let display = display.acquire(); let _ = led.set_high(); timer.delay_ms(100); - display.send_buf(&canvas.buf); + display.send_buf(canvas.buf); let _ = led.set_low(); } From 132c85e99d21f8c06bdfb414a140c8173f5f42ac Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Wed, 5 Mar 2025 00:50:17 -0500 Subject: [PATCH 26/34] Fix a bunch of the letters. Add a dot. --- pico-st7789/src/drawing.rs | 295 +++++++++++++++-- pico-st7789/src/font.rs | 657 ++++++++++++++++++++++++++++++++++++- pico-st7789/src/main.rs | 51 ++- 3 files changed, 944 insertions(+), 59 deletions(-) diff --git a/pico-st7789/src/drawing.rs b/pico-st7789/src/drawing.rs index d8e780f..8917586 100644 --- a/pico-st7789/src/drawing.rs +++ b/pico-st7789/src/drawing.rs @@ -8,7 +8,7 @@ // e c // d d d -use alloc::{boxed::Box, vec::Vec}; +use alloc::collections::btree_map::BTreeMap; static mut BUF: [u8; 163200] = [0; 163200]; @@ -32,7 +32,6 @@ impl Canvas { } } -/* pub const DIGIT_WIDTH: usize = 5; pub const DIGIT_HEIGHT: usize = 9; @@ -68,17 +67,13 @@ pub const SEVEN_SEGMENT_FONT: [[bool; 7]; 10] = [ EIGHT_SEVEN_SEGMENT, NINE_SEVEN_SEGMENT, ]; -*/ -/* -pub fn draw_pixel(frame: &mut Canvas, x: usize, y: usize, color: (u8, u8, u8)) { - frame.buf[(y * frame.width + x) * 3 + 0] = color.0 << 2; - frame.buf[(y * frame.width + x) * 3 + 1] = color.1 << 2; - frame.buf[(y * frame.width + x) * 3 + 2] = color.2 << 2; +pub fn write_pixel(framebuf: &mut [u8], width: usize, x: usize, y: usize, color: (u8, u8, u8)) { + framebuf[(y * width + x) * 3 + 0] = color.0 << 2; + framebuf[(y * width + x) * 3 + 1] = color.1 << 2; + framebuf[(y * width + x) * 3 + 2] = color.2 << 2; } -*/ -/* pub fn draw_seven_segment( digit: u8, frame: &mut [u8], @@ -89,39 +84,275 @@ pub fn draw_seven_segment( ) { let segments = SEVEN_SEGMENT_FONT[digit as usize]; if segments[0] { - write_pixel(frame, x + 1, y, color); - write_pixel(frame, x + 2, y, color); - write_pixel(frame, x + 3, y, color); + write_pixel(frame, width, x + 1, y, color); + write_pixel(frame, width, x + 2, y, color); + write_pixel(frame, width, x + 3, y, color); } if segments[1] { - write_pixel(frame, x + 4, y + 1, color); - write_pixel(frame, x + 4, y + 2, color); - write_pixel(frame, x + 4, y + 3, color); + write_pixel(frame, width, x + 4, y + 1, color); + write_pixel(frame, width, x + 4, y + 2, color); + write_pixel(frame, width, x + 4, y + 3, color); } if segments[2] { - write_pixel(frame, x + 4, y + 5, color); - write_pixel(frame, x + 4, y + 6, color); - write_pixel(frame, x + 4, y + 7, color); + write_pixel(frame, width, x + 4, y + 5, color); + write_pixel(frame, width, x + 4, y + 6, color); + write_pixel(frame, width, x + 4, y + 7, color); } if segments[3] { - write_pixel(frame, x + 1, y + 8, color); - write_pixel(frame, x + 2, y + 8, color); - write_pixel(frame, x + 3, y + 8, color); + write_pixel(frame, width, x + 1, y + 8, color); + write_pixel(frame, width, x + 2, y + 8, color); + write_pixel(frame, width, x + 3, y + 8, color); } if segments[4] { - write_pixel(frame, x, y + 5, color); - write_pixel(frame, x, y + 6, color); - write_pixel(frame, x, y + 7, color); + write_pixel(frame, width, x, y + 5, color); + write_pixel(frame, width, x, y + 6, color); + write_pixel(frame, width, x, y + 7, color); } if segments[5] { - write_pixel(frame, x, y + 1, color); - write_pixel(frame, x, y + 2, color); - write_pixel(frame, x, y + 3, color); + write_pixel(frame, width, x, y + 1, color); + write_pixel(frame, width, x, y + 2, color); + write_pixel(frame, width, x, y + 3, color); } if segments[6] { - write_pixel(frame, x + 1, y + 4, color); - write_pixel(frame, x + 2, y + 4, color); - write_pixel(frame, x + 3, y + 4, color); + write_pixel(frame, width, x + 1, y + 4, color); + write_pixel(frame, width, x + 2, y + 4, color); + write_pixel(frame, width, x + 3, y + 4, color); + } +} + +// Sixteen Segments +// +// a1 a2 +// f h i jb +// f i j b +// f hij b +// g1 g2 +// e klm c +// e k l m c +// ek l mc +// d1 d2 + +pub struct SixteenSegment { + a1: bool, + a2: bool, + b: bool, + c: bool, + d1: bool, + d2: bool, + e: bool, + f: bool, + g1: bool, + g2: bool, + h: bool, + i: bool, + j: bool, + k: bool, + l: bool, + m: bool, +} + +pub fn sixteen_segment_font() -> BTreeMap<char, SixteenSegment> { + let mut font = BTreeMap::new(); + font.insert( + '*', + SixteenSegment { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: true, + g2: true, + h: true, + i: true, + j: true, + k: true, + l: true, + m: true, + }, + ); + font.insert( + '0', + SixteenSegment { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '1', + SixteenSegment { + a1: false, + a2: false, + b: true, + c: true, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: true, + k: false, + l: false, + m: false, + }, + ); + font.insert( + '2', + SixteenSegment { + a1: true, + a2: true, + b: false, + c: false, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: true, + k: false, + l: false, + m: false, + }, + ); + + font +} + +pub fn draw_sixteen_segment( + c: char, + font: &BTreeMap<char, SixteenSegment>, + frame: &mut [u8], + width: usize, + x: usize, + y: usize, + color: (u8, u8, u8), +) { + let segments = font.get(&c).unwrap(); + if segments.a1 { + write_pixel(frame, width, x + 1, y, color); + write_pixel(frame, width, x + 2, y, color); + write_pixel(frame, width, x + 3, y, color); + } + if segments.a2 { + write_pixel(frame, width, x + 5, y, color); + write_pixel(frame, width, x + 6, y, color); + write_pixel(frame, width, x + 7, y, color); + } + if segments.b { + write_pixel(frame, width, x + 8, y + 1, color); + write_pixel(frame, width, x + 8, y + 2, color); + write_pixel(frame, width, x + 8, y + 3, color); + write_pixel(frame, width, x + 8, y + 4, color); + write_pixel(frame, width, x + 8, y + 5, color); + } + if segments.c { + write_pixel(frame, width, x + 8, y + 7, color); + write_pixel(frame, width, x + 8, y + 8, color); + write_pixel(frame, width, x + 8, y + 9, color); + write_pixel(frame, width, x + 8, y + 10, color); + write_pixel(frame, width, x + 8, y + 11, color); + } + + if segments.d1 { + write_pixel(frame, width, x + 1, y + 12, color); + write_pixel(frame, width, x + 2, y + 12, color); + write_pixel(frame, width, x + 3, y + 12, color); + } + if segments.d2 { + write_pixel(frame, width, x + 5, y + 12, color); + write_pixel(frame, width, x + 6, y + 12, color); + write_pixel(frame, width, x + 7, y + 12, color); + } + + if segments.e { + write_pixel(frame, width, x, y + 7, color); + write_pixel(frame, width, x, y + 8, color); + write_pixel(frame, width, x, y + 9, color); + write_pixel(frame, width, x, y + 10, color); + write_pixel(frame, width, x, y + 11, color); + } + if segments.f { + write_pixel(frame, width, x, y + 1, color); + write_pixel(frame, width, x, y + 2, color); + write_pixel(frame, width, x, y + 3, color); + write_pixel(frame, width, x, y + 4, color); + write_pixel(frame, width, x, y + 5, color); + } + + if segments.g1 { + write_pixel(frame, width, x + 1, y + 6, color); + write_pixel(frame, width, x + 2, y + 6, color); + write_pixel(frame, width, x + 3, y + 6, color); + } + if segments.g2 { + write_pixel(frame, width, x + 5, y + 6, color); + write_pixel(frame, width, x + 6, y + 6, color); + write_pixel(frame, width, x + 7, y + 6, color); + } + + if segments.h { + write_pixel(frame, width, x + 1, y + 1, color); + write_pixel(frame, width, x + 1, y + 2, color); + write_pixel(frame, width, x + 2, y + 3, color); + write_pixel(frame, width, x + 3, y + 4, color); + write_pixel(frame, width, x + 3, y + 5, color); + } + if segments.i { + write_pixel(frame, width, x + 4, y + 1, color); + write_pixel(frame, width, x + 4, y + 2, color); + write_pixel(frame, width, x + 4, y + 3, color); + write_pixel(frame, width, x + 4, y + 4, color); + write_pixel(frame, width, x + 4, y + 5, color); + } + if segments.j { + write_pixel(frame, width, x + 7, y + 1, color); + write_pixel(frame, width, x + 7, y + 2, color); + write_pixel(frame, width, x + 6, y + 3, color); + write_pixel(frame, width, x + 5, y + 4, color); + write_pixel(frame, width, x + 5, y + 5, color); + } + if segments.k { + write_pixel(frame, width, x + 3, y + 7, color); + write_pixel(frame, width, x + 3, y + 8, color); + write_pixel(frame, width, x + 2, y + 9, color); + write_pixel(frame, width, x + 1, y + 10, color); + write_pixel(frame, width, x + 1, y + 11, color); + } + if segments.l { + write_pixel(frame, width, x + 4, y + 7, color); + write_pixel(frame, width, x + 4, y + 8, color); + write_pixel(frame, width, x + 4, y + 9, color); + write_pixel(frame, width, x + 4, y + 10, color); + write_pixel(frame, width, x + 4, y + 11, color); + } + if segments.m { + write_pixel(frame, width, x + 5, y + 7, color); + write_pixel(frame, width, x + 5, y + 8, color); + write_pixel(frame, width, x + 6, y + 9, color); + write_pixel(frame, width, x + 7, y + 10, color); + write_pixel(frame, width, x + 7, y + 11, color); } } -*/ diff --git a/pico-st7789/src/font.rs b/pico-st7789/src/font.rs index a2f4c3e..d7b3d5a 100644 --- a/pico-st7789/src/font.rs +++ b/pico-st7789/src/font.rs @@ -39,6 +39,7 @@ pub struct SixteenSegmentGlyph { k: bool, l: bool, m: bool, + dot: bool, } pub struct SixteenSegmentFont(BTreeMap<char, SixteenSegmentGlyph>); @@ -47,24 +48,69 @@ impl SixteenSegmentFont { pub fn new() -> Self { let mut font = BTreeMap::new(); font.insert( - '*', + ' ', SixteenSegmentGlyph { - a1: true, - a2: true, + a1: false, + a2: false, + b: false, + c: false, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + '!', + SixteenSegmentGlyph { + a1: false, + a2: false, b: true, c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: true, - g2: true, - h: true, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: true, + }, + ); + font.insert( + '"', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: false, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, i: true, - j: true, - k: true, - l: true, - m: true, + j: false, + k: false, + l: false, + m: false, + dot: false, }, ); font.insert( @@ -86,6 +132,7 @@ impl SixteenSegmentFont { k: true, l: false, m: false, + dot: false, }, ); font.insert( @@ -107,6 +154,7 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, }, ); font.insert( @@ -128,6 +176,7 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, }, ); font.insert( @@ -149,6 +198,7 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, }, ); font.insert( @@ -170,6 +220,7 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, }, ); font.insert( @@ -191,6 +242,7 @@ impl SixteenSegmentFont { k: false, l: false, m: true, + dot: false, }, ); font.insert( @@ -212,6 +264,7 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, }, ); font.insert( @@ -233,6 +286,7 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, }, ); font.insert( @@ -254,6 +308,7 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, }, ); font.insert( @@ -275,9 +330,581 @@ impl SixteenSegmentFont { k: false, l: false, m: false, + dot: false, + }, + ); + font.insert( + 'A', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: false, + d2: false, + e: true, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'B', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: false, + f: false, + g1: false, + g2: true, + h: false, + i: true, + j: false, + k: false, + l: true, + m: false, + dot: false, + }, + ); + font.insert( + 'C', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: false, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'D', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: true, + j: false, + k: false, + l: true, + m: false, + dot: false, + }, + ); + font.insert( + 'E', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: false, + d1: true, + d2: true, + e: true, + f: true, + g1: true, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'F', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: false, + d1: false, + d2: false, + e: true, + f: true, + g1: true, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'G', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'H', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: false, + d2: false, + e: true, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'I', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: false, + d1: true, + d2: true, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: true, + j: false, + k: false, + l: true, + m: false, + dot: false, + }, + ); + font.insert( + 'J', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'K', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: false, + c: false, + d1: false, + d2: false, + e: true, + f: true, + g1: true, + g2: false, + h: false, + i: false, + j: true, + k: false, + l: false, + m: true, + dot: false, + }, + ); + font.insert( + 'L', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: false, + c: false, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'M', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: false, + d2: false, + e: true, + f: true, + g1: false, + g2: false, + h: true, + i: false, + j: true, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'N', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: false, + d2: false, + e: true, + f: true, + g1: false, + g2: false, + h: true, + i: false, + j: false, + k: false, + l: false, + m: true, + dot: false, + }, + ); + font.insert( + 'O', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'P', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: false, + d1: false, + d2: false, + e: true, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'Q', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: true, + dot: false, + }, + ); + font.insert( + 'R', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: true, + c: false, + d1: false, + d2: false, + e: true, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: true, + dot: false, + }, + ); + font.insert( + 'S', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: true, + d1: true, + d2: true, + e: false, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'T', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: false, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: true, + j: false, + k: false, + l: true, + m: false, + dot: false, + }, + ); + font.insert( + 'U', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: true, + d2: true, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'V', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: false, + c: false, + d1: false, + d2: false, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: true, + k: true, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'W', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: false, + d2: false, + e: true, + f: true, + g1: false, + g2: false, + h: false, + i: false, + j: false, + k: true, + l: false, + m: true, + dot: false, + }, + ); + font.insert( + 'X', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: false, + c: false, + d1: false, + d2: false, + e: false, + f: false, + g1: false, + g2: false, + h: true, + i: false, + j: true, + k: true, + l: false, + m: true, + dot: false, + }, + ); + font.insert( + 'Y', + SixteenSegmentGlyph { + a1: false, + a2: false, + b: true, + c: true, + d1: true, + d2: true, + e: false, + f: true, + g1: true, + g2: true, + h: false, + i: false, + j: false, + k: false, + l: false, + m: false, + dot: false, + }, + ); + font.insert( + 'Z', + SixteenSegmentGlyph { + a1: true, + a2: true, + b: false, + c: false, + d1: true, + d2: true, + e: false, + f: false, + g1: false, + g2: false, + h: false, + i: false, + j: true, + k: true, + l: false, + m: false, + dot: false, }, ); - Self(font) } } diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 90d3789..2a11a68 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -36,7 +36,7 @@ static HEAP: Heap = Heap::empty(); unsafe fn main() -> ! { { use core::mem::MaybeUninit; - const HEAP_SIZE: usize = 1024; + const HEAP_SIZE: usize = 16 * 1024; static mut HEAP_MEM: [MaybeUninit<u8>; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) } } @@ -147,18 +147,45 @@ unsafe fn main() -> ! { draw_seven_segment(9, &mut frame, 63, 0, (255, 255, 255)); */ - font_sixteen.glyph('*').draw(&mut canvas, 0, 10, (255, 255, 255)); - font_sixteen.glyph('0').draw(&mut canvas, 11, 10, (255, 255, 255)); - font_sixteen.glyph('1').draw(&mut canvas, 22, 10, (255, 255, 255)); - font_sixteen.glyph('2').draw(&mut canvas, 33, 10, (255, 255, 255)); - font_sixteen.glyph('3').draw(&mut canvas, 44, 10, (255, 255, 255)); - font_sixteen.glyph('4').draw(&mut canvas, 55, 10, (255, 255, 255)); - font_sixteen.glyph('5').draw(&mut canvas, 66, 10, (255, 255, 255)); - font_sixteen.glyph('6').draw(&mut canvas, 77, 10, (255, 255, 255)); - font_sixteen.glyph('7').draw(&mut canvas, 88, 10, (255, 255, 255)); - font_sixteen.glyph('8').draw(&mut canvas, 99, 10, (255, 255, 255)); - font_sixteen.glyph('9').draw(&mut canvas, 110, 10, (255, 255, 255)); + font_sixteen.glyph(' ').draw(&mut canvas, 0, 0, (255, 255, 255)); + font_sixteen.glyph('0').draw(&mut canvas, 11, 0, (255, 255, 255)); + font_sixteen.glyph('1').draw(&mut canvas, 22, 0, (255, 255, 255)); + font_sixteen.glyph('2').draw(&mut canvas, 33, 0, (255, 255, 255)); + font_sixteen.glyph('3').draw(&mut canvas, 44, 0, (255, 255, 255)); + font_sixteen.glyph('4').draw(&mut canvas, 55, 0, (255, 255, 255)); + font_sixteen.glyph('5').draw(&mut canvas, 66, 0, (255, 255, 255)); + font_sixteen.glyph('6').draw(&mut canvas, 77, 0, (255, 255, 255)); + font_sixteen.glyph('7').draw(&mut canvas, 88, 0, (255, 255, 255)); + font_sixteen.glyph('8').draw(&mut canvas, 99, 0, (255, 255, 255)); + font_sixteen.glyph('9').draw(&mut canvas, 110, 0, (255, 255, 255)); + font_sixteen.glyph('A').draw(&mut canvas, 0, 20, (255, 255, 255)); + font_sixteen.glyph('B').draw(&mut canvas, 11, 20, (255, 255, 255)); + font_sixteen.glyph('C').draw(&mut canvas, 22, 20, (255, 255, 255)); + font_sixteen.glyph('D').draw(&mut canvas, 33, 20, (255, 255, 255)); + font_sixteen.glyph('E').draw(&mut canvas, 44, 20, (255, 255, 255)); + font_sixteen.glyph('F').draw(&mut canvas, 55, 20, (255, 255, 255)); + font_sixteen.glyph('G').draw(&mut canvas, 66, 20, (255, 255, 255)); + font_sixteen.glyph('H').draw(&mut canvas, 77, 20, (255, 255, 255)); + font_sixteen.glyph('I').draw(&mut canvas, 88, 20, (255, 255, 255)); + font_sixteen.glyph('J').draw(&mut canvas, 99, 20, (255, 255, 255)); + font_sixteen.glyph('K').draw(&mut canvas, 110, 20, (255, 255, 255)); + font_sixteen.glyph('L').draw(&mut canvas, 121, 20, (255, 255, 255)); + font_sixteen.glyph('M').draw(&mut canvas, 132, 20, (255, 255, 255)); + font_sixteen.glyph('N').draw(&mut canvas, 143, 20, (255, 255, 255)); + font_sixteen.glyph('O').draw(&mut canvas, 154, 20, (255, 255, 255)); + + font_sixteen.glyph('P').draw(&mut canvas, 0, 40, (255, 255, 255)); + font_sixteen.glyph('Q').draw(&mut canvas, 11, 40, (255, 255, 255)); + font_sixteen.glyph('R').draw(&mut canvas, 22, 40, (255, 255, 255)); + font_sixteen.glyph('S').draw(&mut canvas, 33, 40, (255, 255, 255)); + font_sixteen.glyph('T').draw(&mut canvas, 44, 40, (255, 255, 255)); + font_sixteen.glyph('U').draw(&mut canvas, 55, 40, (255, 255, 255)); + font_sixteen.glyph('V').draw(&mut canvas, 66, 40, (255, 255, 255)); + font_sixteen.glyph('W').draw(&mut canvas, 77, 40, (255, 255, 255)); + font_sixteen.glyph('X').draw(&mut canvas, 88, 40, (255, 255, 255)); + font_sixteen.glyph('Y').draw(&mut canvas, 99, 40, (255, 255, 255)); + font_sixteen.glyph('Z').draw(&mut canvas, 110, 40, (255, 255, 255)); /* for x in 80..90 { for y in 155..165 { From 155d2ba18edb9ee875c0412bad9f281183371d94 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Wed, 5 Mar 2025 09:25:15 -0500 Subject: [PATCH 27/34] Use bitflags to represent the font --- Cargo.lock | 35 +- pico-st7789/Cargo.toml | 1 + pico-st7789/src/font.rs | 1052 +++++++++++---------------------------- pico-st7789/src/main.rs | 29 +- 4 files changed, 324 insertions(+), 793 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 1772741..d871276 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -365,7 +365,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "271383c67ccabffb7381723dea0672a673f292304fcb45c01cc648c7a8d58088" dependencies = [ "annotate-snippets", - "bitflags 2.8.0", + "bitflags 2.9.0", "cexpr", "clang-sys", "itertools 0.12.1", @@ -420,9 +420,9 @@ checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" [[package]] name = "bitflags" -version = "2.8.0" +version = "2.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8f68f53c83ab957f72c32642f3868eec03eb974d1fb82e453128456482613d36" +checksum = "5c8214115b7bf84099f1309324e63141d4c5d7cc26862f97a0a857dbefe165bd" dependencies = [ "serde 1.0.218", ] @@ -485,7 +485,7 @@ version = "0.18.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8ca26ef0159422fb77631dc9d17b102f253b876fe1586b03b803e63a309b4ee2" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "cairo-sys-rs", "glib 0.18.5", "libc", @@ -1815,7 +1815,7 @@ version = "0.18.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "233daaf6e83ae6a12a52055f568f9d7cf4671dabb78ff9560ab6da230ce00ee5" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "futures-channel", "futures-core", "futures-executor", @@ -1838,7 +1838,7 @@ version = "0.20.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "707b819af8059ee5395a2de9f2317d87a53dbad8846a2f089f0bb44703f37686" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "futures-channel", "futures-core", "futures-executor", @@ -2837,7 +2837,7 @@ version = "0.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "65f3a4b81b2a2d8c7f300643676202debd1b7c929dbf5c9bb89402ea11d19810" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "cc", "convert_case", "cookie-factory", @@ -3142,7 +3142,7 @@ version = "0.27.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2eb04e9c688eff1c89d72b407f168cf79bb9e867a9d3323ed6c01519eb9cc053" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "cfg-if", "libc", ] @@ -3286,7 +3286,7 @@ version = "0.10.71" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5e14130c6a98cd258fdcb0fb6d744152343ff729cbfcb28c656a9d12b999fbcd" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "cfg-if", "foreign-types", "libc", @@ -3563,6 +3563,7 @@ dependencies = [ name = "pico-st7789" version = "0.1.0" dependencies = [ + "bitflags 2.9.0", "cortex-m-rt", "embedded-alloc", "embedded-hal 1.0.0", @@ -3632,7 +3633,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "08e645ba5c45109106d56610b3ee60eb13a6f2beb8b74f8dc8186cf261788dda" dependencies = [ "anyhow", - "bitflags 2.8.0", + "bitflags 2.9.0", "libc", "libspa", "libspa-sys", @@ -3829,7 +3830,7 @@ checksum = "14cae93065090804185d3b75f0bf93b8eeda30c7a9b4a33d3bdb3988d6229e50" dependencies = [ "bit-set", "bit-vec", - "bitflags 2.8.0", + "bitflags 2.9.0", "lazy_static", "num-traits", "rand 0.8.5", @@ -4068,7 +4069,7 @@ version = "0.5.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "82b568323e98e49e2a0899dcee453dd679fae22d69adf9b11dd508d1549b7e2f" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", ] [[package]] @@ -4299,7 +4300,7 @@ version = "0.38.44" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fdb5bc1ae2baa591800df16c9ca78619bf65c0488b41b96ccec5d11220d8c154" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "errno", "libc", "linux-raw-sys", @@ -4387,7 +4388,7 @@ version = "2.11.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "897b2245f0b511c87893af39b033e5ca9cce68824c4d7e7630b5a1d339658d02" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", "core-foundation", "core-foundation-sys", "libc", @@ -4758,7 +4759,7 @@ checksum = "4560278f0e00ce64938540546f59f590d60beee33fffbd3b9cd47851e5fff233" dependencies = [ "atoi", "base64 0.22.1", - "bitflags 2.8.0", + "bitflags 2.9.0", "byteorder", "bytes", "crc", @@ -4800,7 +4801,7 @@ checksum = "c5b98a57f363ed6764d5b3a12bfedf62f07aa16e1856a7ddc2a0bb190a959613" dependencies = [ "atoi", "base64 0.22.1", - "bitflags 2.8.0", + "bitflags 2.9.0", "byteorder", "crc", "dotenvy", @@ -6035,7 +6036,7 @@ version = "0.33.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3268f3d866458b787f390cf61f4bbb563b922d091359f9608842999eaee3943c" dependencies = [ - "bitflags 2.8.0", + "bitflags 2.9.0", ] [[package]] diff --git a/pico-st7789/Cargo.toml b/pico-st7789/Cargo.toml index 001c708..ff98242 100644 --- a/pico-st7789/Cargo.toml +++ b/pico-st7789/Cargo.toml @@ -4,6 +4,7 @@ version = "0.1.0" edition = "2021" [dependencies] +bitflags = "2.9.0" cortex-m-rt = "0.7.3" embedded-alloc = "0.6.0" embedded-hal = "1.0.0" diff --git a/pico-st7789/src/font.rs b/pico-st7789/src/font.rs index d7b3d5a..d4f8b88 100644 --- a/pico-st7789/src/font.rs +++ b/pico-st7789/src/font.rs @@ -1,4 +1,5 @@ use alloc::collections::btree_map::BTreeMap; +use bitflags::bitflags; use crate::drawing::Canvas; @@ -22,6 +23,7 @@ pub trait Glyph { // ek l mc // d1 d2 +/* pub struct SixteenSegmentGlyph { a1: bool, a2: bool, @@ -41,869 +43,393 @@ pub struct SixteenSegmentGlyph { m: bool, dot: bool, } +*/ + +bitflags! { + pub struct SixteenSegmentGlyph: u32 { + const NONE = 0; + const A1 = 0x0001; + const A2 = 0x0001 << 1; + const B = 0x0001 << 2; + const C = 0x0001 << 3; + const D1 = 0x0001 << 4; + const D2 = 0x0001 << 5; + const E = 0x0001 << 6; + const F = 0x0001 << 7; + const G1 = 0x0001 << 8; + const G2 = 0x0001 << 9; + const H = 0x0001 << 10; + const I = 0x0001 << 11; + const J = 0x0001 << 12; + const K = 0x0001 << 13; + const L = 0x0001 << 14; + const M = 0x0001 << 15; + const DOT = 0x0001 << 16; + } +} + +/* +impl Default for SixteenSegmentGlyph { + fn default() -> Self { + 0_u32 + } +} +*/ pub struct SixteenSegmentFont(BTreeMap<char, SixteenSegmentGlyph>); impl SixteenSegmentFont { pub fn new() -> Self { let mut font = BTreeMap::new(); - font.insert( - ' ', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: false, - c: false, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, - ); + font.insert(' ', SixteenSegmentGlyph::NONE); font.insert( '!', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: true, - }, - ); - font.insert( - '"', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: false, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: true, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B | SixteenSegmentGlyph::C | SixteenSegmentGlyph::DOT, ); + font.insert('"', SixteenSegmentGlyph::B | SixteenSegmentGlyph::I); font.insert( '0', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: true, - k: true, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::J + | SixteenSegmentGlyph::K, ); font.insert( '1', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: true, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B | SixteenSegmentGlyph::C | SixteenSegmentGlyph::J, ); font.insert( '2', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: false, - d1: true, - d2: true, - e: true, - f: false, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( '3', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: false, - f: false, - g1: false, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::G2, ); font.insert( '4', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: false, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( '5', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: false, - d1: true, - d2: true, - e: false, - f: true, - g1: true, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: true, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::M, ); font.insert( '6', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( '7', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C, ); font.insert( '8', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( '9', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: false, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( 'A', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: false, - d2: false, - e: true, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( 'B', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: false, - f: false, - g1: false, - g2: true, - h: false, - i: true, - j: false, - k: false, - l: true, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::G2 + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, ); font.insert( 'C', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: false, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F, ); font.insert( 'D', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: true, - j: false, - k: false, - l: true, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, ); font.insert( 'E', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: false, - d1: true, - d2: true, - e: true, - f: true, - g1: true, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1, ); font.insert( 'F', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: false, - d1: false, - d2: false, - e: true, - f: true, - g1: true, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1, ); font.insert( 'G', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G2, ); font.insert( 'H', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: true, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( 'I', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: false, - d1: true, - d2: true, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: true, - j: false, - k: false, - l: true, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, ); font.insert( 'J', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E, ); font.insert( 'K', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: false, - c: false, - d1: false, - d2: false, - e: true, - f: true, - g1: true, - g2: false, - h: false, - i: false, - j: true, - k: false, - l: false, - m: true, - dot: false, - }, + SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::J + | SixteenSegmentGlyph::M, ); font.insert( 'L', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: false, - c: false, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F, ); font.insert( 'M', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: true, - f: true, - g1: false, - g2: false, - h: true, - i: false, - j: true, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::H + | SixteenSegmentGlyph::J, ); font.insert( 'N', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: true, - f: true, - g1: false, - g2: false, - h: true, - i: false, - j: false, - k: false, - l: false, - m: true, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::H + | SixteenSegmentGlyph::M, ); font.insert( 'O', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F, ); font.insert( 'P', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: false, - d1: false, - d2: false, - e: true, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( 'Q', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: true, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::M, ); font.insert( 'R', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: true, - c: false, - d1: false, - d2: false, - e: true, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: true, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2 + | SixteenSegmentGlyph::M, ); font.insert( 'S', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: true, - d1: true, - d2: true, - e: false, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( 'T', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: false, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: true, - j: false, - k: false, - l: true, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, ); font.insert( 'U', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F, ); font.insert( 'V', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: false, - c: false, - d1: false, - d2: false, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: true, - k: true, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::J + | SixteenSegmentGlyph::K, ); font.insert( 'W', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: true, - l: false, - m: true, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::K + | SixteenSegmentGlyph::M, ); font.insert( 'X', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: false, - c: false, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: true, - i: false, - j: true, - k: true, - l: false, - m: true, - dot: false, - }, + SixteenSegmentGlyph::H + | SixteenSegmentGlyph::J + | SixteenSegmentGlyph::K + | SixteenSegmentGlyph::M, ); font.insert( 'Y', - SixteenSegmentGlyph { - a1: false, - a2: false, - b: true, - c: true, - d1: true, - d2: true, - e: false, - f: true, - g1: true, - g2: true, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, ); font.insert( 'Z', - SixteenSegmentGlyph { - a1: true, - a2: true, - b: false, - c: false, - d1: true, - d2: true, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: true, - k: true, - l: false, - m: false, - dot: false, - }, + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::J + | SixteenSegmentGlyph::K, ); Self(font) } @@ -917,24 +443,24 @@ impl Font<SixteenSegmentGlyph> for SixteenSegmentFont { impl Glyph for SixteenSegmentGlyph { fn draw(&self, canvas: &mut Canvas, x: usize, y: usize, color: (u8, u8, u8)) { - if self.a1 { + if self.contains(SixteenSegmentGlyph::A1) { canvas.set_pixel(x + 1, y, color); canvas.set_pixel(x + 2, y, color); canvas.set_pixel(x + 3, y, color); } - if self.a2 { + if self.contains(SixteenSegmentGlyph::A2) { canvas.set_pixel(x + 5, y, color); canvas.set_pixel(x + 6, y, color); canvas.set_pixel(x + 7, y, color); } - if self.b { + if self.contains(SixteenSegmentGlyph::B) { canvas.set_pixel(x + 8, y + 1, color); canvas.set_pixel(x + 8, y + 2, color); canvas.set_pixel(x + 8, y + 3, color); canvas.set_pixel(x + 8, y + 4, color); canvas.set_pixel(x + 8, y + 5, color); } - if self.c { + if self.contains(SixteenSegmentGlyph::C) { canvas.set_pixel(x + 8, y + 7, color); canvas.set_pixel(x + 8, y + 8, color); canvas.set_pixel(x + 8, y + 9, color); @@ -942,25 +468,25 @@ impl Glyph for SixteenSegmentGlyph { canvas.set_pixel(x + 8, y + 11, color); } - if self.d1 { + if self.contains(SixteenSegmentGlyph::D1) { canvas.set_pixel(x + 1, y + 12, color); canvas.set_pixel(x + 2, y + 12, color); canvas.set_pixel(x + 3, y + 12, color); } - if self.d2 { + if self.contains(SixteenSegmentGlyph::D2) { canvas.set_pixel(x + 5, y + 12, color); canvas.set_pixel(x + 6, y + 12, color); canvas.set_pixel(x + 7, y + 12, color); } - if self.e { + if self.contains(SixteenSegmentGlyph::E) { canvas.set_pixel(x, y + 7, color); canvas.set_pixel(x, y + 8, color); canvas.set_pixel(x, y + 9, color); canvas.set_pixel(x, y + 10, color); canvas.set_pixel(x, y + 11, color); } - if self.f { + if self.contains(SixteenSegmentGlyph::F) { canvas.set_pixel(x, y + 1, color); canvas.set_pixel(x, y + 2, color); canvas.set_pixel(x, y + 3, color); @@ -968,53 +494,53 @@ impl Glyph for SixteenSegmentGlyph { canvas.set_pixel(x, y + 5, color); } - if self.g1 { + if self.contains(SixteenSegmentGlyph::G1) { canvas.set_pixel(x + 1, y + 6, color); canvas.set_pixel(x + 2, y + 6, color); canvas.set_pixel(x + 3, y + 6, color); } - if self.g2 { + if self.contains(SixteenSegmentGlyph::G2) { canvas.set_pixel(x + 5, y + 6, color); canvas.set_pixel(x + 6, y + 6, color); canvas.set_pixel(x + 7, y + 6, color); } - if self.h { + if self.contains(SixteenSegmentGlyph::H) { canvas.set_pixel(x + 1, y + 1, color); canvas.set_pixel(x + 1, y + 2, color); canvas.set_pixel(x + 2, y + 3, color); canvas.set_pixel(x + 3, y + 4, color); canvas.set_pixel(x + 3, y + 5, color); } - if self.i { + if self.contains(SixteenSegmentGlyph::I) { canvas.set_pixel(x + 4, y + 1, color); canvas.set_pixel(x + 4, y + 2, color); canvas.set_pixel(x + 4, y + 3, color); canvas.set_pixel(x + 4, y + 4, color); canvas.set_pixel(x + 4, y + 5, color); } - if self.j { + if self.contains(SixteenSegmentGlyph::J) { canvas.set_pixel(x + 7, y + 1, color); canvas.set_pixel(x + 7, y + 2, color); canvas.set_pixel(x + 6, y + 3, color); canvas.set_pixel(x + 5, y + 4, color); canvas.set_pixel(x + 5, y + 5, color); } - if self.k { + if self.contains(SixteenSegmentGlyph::K) { canvas.set_pixel(x + 3, y + 7, color); canvas.set_pixel(x + 3, y + 8, color); canvas.set_pixel(x + 2, y + 9, color); canvas.set_pixel(x + 1, y + 10, color); canvas.set_pixel(x + 1, y + 11, color); } - if self.l { + if self.contains(SixteenSegmentGlyph::L) { canvas.set_pixel(x + 4, y + 7, color); canvas.set_pixel(x + 4, y + 8, color); canvas.set_pixel(x + 4, y + 9, color); canvas.set_pixel(x + 4, y + 10, color); canvas.set_pixel(x + 4, y + 11, color); } - if self.m { + if self.contains(SixteenSegmentGlyph::M) { canvas.set_pixel(x + 5, y + 7, color); canvas.set_pixel(x + 5, y + 8, color); canvas.set_pixel(x + 6, y + 9, color); diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 2a11a68..d0bdad6 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -14,7 +14,7 @@ use rp_pico::{ }; mod drawing; -use drawing::{Canvas}; +use drawing::Canvas; mod font; use font::{Font, Glyph, SixteenSegmentFont}; @@ -31,12 +31,11 @@ const FRAMEBUF: usize = ROWS * COLUMNS * 3; #[global_allocator] static HEAP: Heap = Heap::empty(); - #[entry] unsafe fn main() -> ! { { use core::mem::MaybeUninit; - const HEAP_SIZE: usize = 16 * 1024; + const HEAP_SIZE: usize = 1024; static mut HEAP_MEM: [MaybeUninit<u8>; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) } } @@ -126,13 +125,6 @@ unsafe fn main() -> ! { let mut canvas = Canvas::new(); loop { - { - let display = display.acquire(); - let _ = led.set_high(); - timer.delay_ms(100); - display.send_buf(canvas.buf); - let _ = led.set_low(); - } /* draw_seven_segment(0, &mut frame, 0, 0, (255, 255, 255)); @@ -147,8 +139,12 @@ unsafe fn main() -> ! { draw_seven_segment(9, &mut frame, 63, 0, (255, 255, 255)); */ - font_sixteen.glyph(' ').draw(&mut canvas, 0, 0, (255, 255, 255)); - font_sixteen.glyph('0').draw(&mut canvas, 11, 0, (255, 255, 255)); + font_sixteen + .glyph(' ') + .draw(&mut canvas, 0, 0, (255, 255, 255)); + font_sixteen + .glyph('0') + .draw(&mut canvas, 11, 0, (255, 255, 255)); font_sixteen.glyph('1').draw(&mut canvas, 22, 0, (255, 255, 255)); font_sixteen.glyph('2').draw(&mut canvas, 33, 0, (255, 255, 255)); font_sixteen.glyph('3').draw(&mut canvas, 44, 0, (255, 255, 255)); @@ -186,6 +182,14 @@ unsafe fn main() -> ! { font_sixteen.glyph('X').draw(&mut canvas, 88, 40, (255, 255, 255)); font_sixteen.glyph('Y').draw(&mut canvas, 99, 40, (255, 255, 255)); font_sixteen.glyph('Z').draw(&mut canvas, 110, 40, (255, 255, 255)); + + { + let display = display.acquire(); + let _ = led.set_high(); + timer.delay_ms(100); + display.send_buf(canvas.buf); + let _ = led.set_low(); + } /* for x in 80..90 { for y in 155..165 { @@ -197,4 +201,3 @@ unsafe fn main() -> ! { timer.delay_ms(1000); } } - From 8288fdbb6b7a50a87dc966e23b016889b4ba5ea5 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Wed, 5 Mar 2025 09:47:00 -0500 Subject: [PATCH 28/34] Refactor the canvas and font --- pico-st7789/src/{drawing.rs => canvas.rs} | 36 ++-- pico-st7789/src/font/mod.rs | 13 ++ .../src/{font.rs => font/sixteen_segment.rs} | 43 +---- pico-st7789/src/main.rs | 178 ++++++++++++++---- 4 files changed, 174 insertions(+), 96 deletions(-) rename pico-st7789/src/{drawing.rs => canvas.rs} (94%) create mode 100644 pico-st7789/src/font/mod.rs rename pico-st7789/src/{font.rs => font/sixteen_segment.rs} (96%) diff --git a/pico-st7789/src/drawing.rs b/pico-st7789/src/canvas.rs similarity index 94% rename from pico-st7789/src/drawing.rs rename to pico-st7789/src/canvas.rs index 8917586..e7d6f30 100644 --- a/pico-st7789/src/drawing.rs +++ b/pico-st7789/src/canvas.rs @@ -10,28 +10,31 @@ use alloc::collections::btree_map::BTreeMap; -static mut BUF: [u8; 163200] = [0; 163200]; +/* +*/ -pub struct Canvas { - pub buf: &'static mut [u8; 163200], - pub width: usize, -} +pub struct RGB { pub r: u8, pub g: u8, pub b: u8 } -impl Canvas { - pub fn new() -> Self { - Self { - buf: unsafe { &mut BUF }, - width: 170, +pub trait Canvas { + fn set_pixel(&mut self, x: usize, y: usize, color: &RGB); + + fn square(&mut self, x1: usize, y1: usize, x2: usize, y2: usize, color: &RGB) { + for x in x1..x2+1 { + self.set_pixel(x, y1, color); + } + for x in x1..x2+1 { + self.set_pixel(x, y2, color); + } + for y in y1..y2+1 { + self.set_pixel(x1, y, color); + } + for y in y1..y2+1 { + self.set_pixel(x2, y, color); } } - - pub fn set_pixel(&mut self, x: usize, y: usize, color: (u8, u8, u8)) { - self.buf[(y * self.width + x) * 3 + 0] = color.0 << 2; - self.buf[(y * self.width + x) * 3 + 1] = color.1 << 2; - self.buf[(y * self.width + x) * 3 + 2] = color.2 << 2; - } } +/* pub const DIGIT_WIDTH: usize = 5; pub const DIGIT_HEIGHT: usize = 9; @@ -356,3 +359,4 @@ pub fn draw_sixteen_segment( write_pixel(frame, width, x + 7, y + 11, color); } } +*/ diff --git a/pico-st7789/src/font/mod.rs b/pico-st7789/src/font/mod.rs new file mode 100644 index 0000000..126176d --- /dev/null +++ b/pico-st7789/src/font/mod.rs @@ -0,0 +1,13 @@ +use crate::canvas::{Canvas, RGB}; + +mod sixteen_segment; +pub use sixteen_segment::SixteenSegmentFont; + +pub trait Font<A> { + fn glyph(&self, c: char) -> &A; +} + +pub trait Glyph { + fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB); +} + diff --git a/pico-st7789/src/font.rs b/pico-st7789/src/font/sixteen_segment.rs similarity index 96% rename from pico-st7789/src/font.rs rename to pico-st7789/src/font/sixteen_segment.rs index d4f8b88..44061ce 100644 --- a/pico-st7789/src/font.rs +++ b/pico-st7789/src/font/sixteen_segment.rs @@ -1,15 +1,9 @@ use alloc::collections::btree_map::BTreeMap; use bitflags::bitflags; -use crate::drawing::Canvas; +use crate::canvas::{Canvas, RGB}; -pub trait Font<A> { - fn glyph(&self, c: char) -> &A; -} - -pub trait Glyph { - fn draw(&self, canvas: &mut Canvas, x: usize, y: usize, color: (u8, u8, u8)); -} +use super::{Font, Glyph}; // Sixteen Segments // @@ -23,28 +17,6 @@ pub trait Glyph { // ek l mc // d1 d2 -/* -pub struct SixteenSegmentGlyph { - a1: bool, - a2: bool, - b: bool, - c: bool, - d1: bool, - d2: bool, - e: bool, - f: bool, - g1: bool, - g2: bool, - h: bool, - i: bool, - j: bool, - k: bool, - l: bool, - m: bool, - dot: bool, -} -*/ - bitflags! { pub struct SixteenSegmentGlyph: u32 { const NONE = 0; @@ -68,14 +40,6 @@ bitflags! { } } -/* -impl Default for SixteenSegmentGlyph { - fn default() -> Self { - 0_u32 - } -} -*/ - pub struct SixteenSegmentFont(BTreeMap<char, SixteenSegmentGlyph>); impl SixteenSegmentFont { @@ -442,7 +406,7 @@ impl Font<SixteenSegmentGlyph> for SixteenSegmentFont { } impl Glyph for SixteenSegmentGlyph { - fn draw(&self, canvas: &mut Canvas, x: usize, y: usize, color: (u8, u8, u8)) { + fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB) { if self.contains(SixteenSegmentGlyph::A1) { canvas.set_pixel(x + 1, y, color); canvas.set_pixel(x + 2, y, color); @@ -549,3 +513,4 @@ impl Glyph for SixteenSegmentGlyph { } } } + diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index d0bdad6..e404ba7 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -13,8 +13,8 @@ use rp_pico::{ pac, Pins, }; -mod drawing; -use drawing::Canvas; +mod canvas; +use canvas::{Canvas, RGB}; mod font; use font::{Font, Glyph, SixteenSegmentFont}; @@ -31,6 +31,30 @@ const FRAMEBUF: usize = ROWS * COLUMNS * 3; #[global_allocator] static HEAP: Heap = Heap::empty(); +static mut BUF: [u8; 163200] = [0; 163200]; + +pub struct FrameBuf { + pub buf: &'static mut [u8; 163200], + pub width: usize, +} + +impl FrameBuf { + pub fn new() -> Self { + Self { + buf: unsafe { &mut BUF }, + width: 170, + } + } +} + +impl Canvas for FrameBuf { + fn set_pixel(&mut self, x: usize, y: usize, color: &RGB) { + self.buf[(y * self.width + x) * 3 + 0] = color.r << 2; + self.buf[(y * self.width + x) * 3 + 1] = color.g << 2; + self.buf[(y * self.width + x) * 3 + 2] = color.b << 2; + } +} + #[entry] unsafe fn main() -> ! { { @@ -122,10 +146,10 @@ unsafe fn main() -> ! { let mut framebuf = [0; FRAMEBUF]; let mut canvas = Canvas::new(&mut framebuf, COLUMNS); */ - let mut canvas = Canvas::new(); + let mut canvas = FrameBuf::new(); + let white = RGB { r: 63, g: 63, b: 63 }; loop { - /* draw_seven_segment(0, &mut frame, 0, 0, (255, 255, 255)); draw_seven_segment(1, &mut frame, 7, 0, (255, 255, 255)); @@ -141,47 +165,119 @@ unsafe fn main() -> ! { font_sixteen .glyph(' ') - .draw(&mut canvas, 0, 0, (255, 255, 255)); + .draw(&mut canvas, 0, 0, &white); font_sixteen .glyph('0') - .draw(&mut canvas, 11, 0, (255, 255, 255)); - font_sixteen.glyph('1').draw(&mut canvas, 22, 0, (255, 255, 255)); - font_sixteen.glyph('2').draw(&mut canvas, 33, 0, (255, 255, 255)); - font_sixteen.glyph('3').draw(&mut canvas, 44, 0, (255, 255, 255)); - font_sixteen.glyph('4').draw(&mut canvas, 55, 0, (255, 255, 255)); - font_sixteen.glyph('5').draw(&mut canvas, 66, 0, (255, 255, 255)); - font_sixteen.glyph('6').draw(&mut canvas, 77, 0, (255, 255, 255)); - font_sixteen.glyph('7').draw(&mut canvas, 88, 0, (255, 255, 255)); - font_sixteen.glyph('8').draw(&mut canvas, 99, 0, (255, 255, 255)); - font_sixteen.glyph('9').draw(&mut canvas, 110, 0, (255, 255, 255)); + .draw(&mut canvas, 11, 0, &white); + font_sixteen + .glyph('1') + .draw(&mut canvas, 22, 0, &white); + font_sixteen + .glyph('2') + .draw(&mut canvas, 33, 0, &white); + font_sixteen + .glyph('3') + .draw(&mut canvas, 44, 0, &white); + font_sixteen + .glyph('4') + .draw(&mut canvas, 55, 0, &white); + font_sixteen + .glyph('5') + .draw(&mut canvas, 66, 0, &white); + font_sixteen + .glyph('6') + .draw(&mut canvas, 77, 0, &white); + font_sixteen + .glyph('7') + .draw(&mut canvas, 88, 0, &white); + font_sixteen + .glyph('8') + .draw(&mut canvas, 99, 0, &white); + font_sixteen + .glyph('9') + .draw(&mut canvas, 110, 0, &white); - font_sixteen.glyph('A').draw(&mut canvas, 0, 20, (255, 255, 255)); - font_sixteen.glyph('B').draw(&mut canvas, 11, 20, (255, 255, 255)); - font_sixteen.glyph('C').draw(&mut canvas, 22, 20, (255, 255, 255)); - font_sixteen.glyph('D').draw(&mut canvas, 33, 20, (255, 255, 255)); - font_sixteen.glyph('E').draw(&mut canvas, 44, 20, (255, 255, 255)); - font_sixteen.glyph('F').draw(&mut canvas, 55, 20, (255, 255, 255)); - font_sixteen.glyph('G').draw(&mut canvas, 66, 20, (255, 255, 255)); - font_sixteen.glyph('H').draw(&mut canvas, 77, 20, (255, 255, 255)); - font_sixteen.glyph('I').draw(&mut canvas, 88, 20, (255, 255, 255)); - font_sixteen.glyph('J').draw(&mut canvas, 99, 20, (255, 255, 255)); - font_sixteen.glyph('K').draw(&mut canvas, 110, 20, (255, 255, 255)); - font_sixteen.glyph('L').draw(&mut canvas, 121, 20, (255, 255, 255)); - font_sixteen.glyph('M').draw(&mut canvas, 132, 20, (255, 255, 255)); - font_sixteen.glyph('N').draw(&mut canvas, 143, 20, (255, 255, 255)); - font_sixteen.glyph('O').draw(&mut canvas, 154, 20, (255, 255, 255)); + font_sixteen + .glyph('A') + .draw(&mut canvas, 0, 20, &white); + font_sixteen + .glyph('B') + .draw(&mut canvas, 11, 20, &white); + font_sixteen + .glyph('C') + .draw(&mut canvas, 22, 20, &white); + font_sixteen + .glyph('D') + .draw(&mut canvas, 33, 20, &white); + font_sixteen + .glyph('E') + .draw(&mut canvas, 44, 20, &white); + font_sixteen + .glyph('F') + .draw(&mut canvas, 55, 20, &white); + font_sixteen + .glyph('G') + .draw(&mut canvas, 66, 20, &white); + font_sixteen + .glyph('H') + .draw(&mut canvas, 77, 20, &white); + font_sixteen + .glyph('I') + .draw(&mut canvas, 88, 20, &white); + font_sixteen + .glyph('J') + .draw(&mut canvas, 99, 20, &white); + font_sixteen + .glyph('K') + .draw(&mut canvas, 110, 20, &white); + font_sixteen + .glyph('L') + .draw(&mut canvas, 121, 20, &white); + font_sixteen + .glyph('M') + .draw(&mut canvas, 132, 20, &white); + font_sixteen + .glyph('N') + .draw(&mut canvas, 143, 20, &white); + font_sixteen + .glyph('O') + .draw(&mut canvas, 154, 20, &white); - font_sixteen.glyph('P').draw(&mut canvas, 0, 40, (255, 255, 255)); - font_sixteen.glyph('Q').draw(&mut canvas, 11, 40, (255, 255, 255)); - font_sixteen.glyph('R').draw(&mut canvas, 22, 40, (255, 255, 255)); - font_sixteen.glyph('S').draw(&mut canvas, 33, 40, (255, 255, 255)); - font_sixteen.glyph('T').draw(&mut canvas, 44, 40, (255, 255, 255)); - font_sixteen.glyph('U').draw(&mut canvas, 55, 40, (255, 255, 255)); - font_sixteen.glyph('V').draw(&mut canvas, 66, 40, (255, 255, 255)); - font_sixteen.glyph('W').draw(&mut canvas, 77, 40, (255, 255, 255)); - font_sixteen.glyph('X').draw(&mut canvas, 88, 40, (255, 255, 255)); - font_sixteen.glyph('Y').draw(&mut canvas, 99, 40, (255, 255, 255)); - font_sixteen.glyph('Z').draw(&mut canvas, 110, 40, (255, 255, 255)); + font_sixteen + .glyph('P') + .draw(&mut canvas, 0, 40, &white); + font_sixteen + .glyph('Q') + .draw(&mut canvas, 11, 40, &white); + font_sixteen + .glyph('R') + .draw(&mut canvas, 22, 40, &white); + font_sixteen + .glyph('S') + .draw(&mut canvas, 33, 40, &white); + font_sixteen + .glyph('T') + .draw(&mut canvas, 44, 40, &white); + font_sixteen + .glyph('U') + .draw(&mut canvas, 55, 40, &white); + font_sixteen + .glyph('V') + .draw(&mut canvas, 66, 40, &white); + font_sixteen + .glyph('W') + .draw(&mut canvas, 77, 40, &white); + font_sixteen + .glyph('X') + .draw(&mut canvas, 88, 40, &white); + font_sixteen + .glyph('Y') + .draw(&mut canvas, 99, 40, &white); + font_sixteen + .glyph('Z') + .draw(&mut canvas, 110, 40, &white); + + canvas.square(10, 70, 160, 310, &white); { let display = display.acquire(); From b444326c1c418e05217dc57761aed0c0b70d9d7a Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Wed, 5 Mar 2025 09:48:33 -0500 Subject: [PATCH 29/34] Add a link to the sixteen-segment font source --- pico-st7789/src/font/sixteen_segment.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/pico-st7789/src/font/sixteen_segment.rs b/pico-st7789/src/font/sixteen_segment.rs index 44061ce..fe021ff 100644 --- a/pico-st7789/src/font/sixteen_segment.rs +++ b/pico-st7789/src/font/sixteen_segment.rs @@ -6,6 +6,7 @@ use crate::canvas::{Canvas, RGB}; use super::{Font, Glyph}; // Sixteen Segments +// https://www.partsnotincluded.com/segmented-led-display-ascii-library/ // // a1 a2 // f h i jb From bc9e24c0c95c78d62d4c358f790a268e34e05f08 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Wed, 5 Mar 2025 09:58:42 -0500 Subject: [PATCH 30/34] Remove dead code --- pico-st7789/src/canvas.rs | 237 -------------------------------------- 1 file changed, 237 deletions(-) diff --git a/pico-st7789/src/canvas.rs b/pico-st7789/src/canvas.rs index e7d6f30..cb2d05b 100644 --- a/pico-st7789/src/canvas.rs +++ b/pico-st7789/src/canvas.rs @@ -122,241 +122,4 @@ pub fn draw_seven_segment( write_pixel(frame, width, x + 3, y + 4, color); } } - -// Sixteen Segments -// -// a1 a2 -// f h i jb -// f i j b -// f hij b -// g1 g2 -// e klm c -// e k l m c -// ek l mc -// d1 d2 - -pub struct SixteenSegment { - a1: bool, - a2: bool, - b: bool, - c: bool, - d1: bool, - d2: bool, - e: bool, - f: bool, - g1: bool, - g2: bool, - h: bool, - i: bool, - j: bool, - k: bool, - l: bool, - m: bool, -} - -pub fn sixteen_segment_font() -> BTreeMap<char, SixteenSegment> { - let mut font = BTreeMap::new(); - font.insert( - '*', - SixteenSegment { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: true, - g2: true, - h: true, - i: true, - j: true, - k: true, - l: true, - m: true, - }, - ); - font.insert( - '0', - SixteenSegment { - a1: true, - a2: true, - b: true, - c: true, - d1: true, - d2: true, - e: true, - f: true, - g1: false, - g2: false, - h: false, - i: false, - j: false, - k: false, - l: false, - m: false, - }, - ); - font.insert( - '1', - SixteenSegment { - a1: false, - a2: false, - b: true, - c: true, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: true, - k: false, - l: false, - m: false, - }, - ); - font.insert( - '2', - SixteenSegment { - a1: true, - a2: true, - b: false, - c: false, - d1: false, - d2: false, - e: false, - f: false, - g1: false, - g2: false, - h: false, - i: false, - j: true, - k: false, - l: false, - m: false, - }, - ); - - font -} - -pub fn draw_sixteen_segment( - c: char, - font: &BTreeMap<char, SixteenSegment>, - frame: &mut [u8], - width: usize, - x: usize, - y: usize, - color: (u8, u8, u8), -) { - let segments = font.get(&c).unwrap(); - if segments.a1 { - write_pixel(frame, width, x + 1, y, color); - write_pixel(frame, width, x + 2, y, color); - write_pixel(frame, width, x + 3, y, color); - } - if segments.a2 { - write_pixel(frame, width, x + 5, y, color); - write_pixel(frame, width, x + 6, y, color); - write_pixel(frame, width, x + 7, y, color); - } - if segments.b { - write_pixel(frame, width, x + 8, y + 1, color); - write_pixel(frame, width, x + 8, y + 2, color); - write_pixel(frame, width, x + 8, y + 3, color); - write_pixel(frame, width, x + 8, y + 4, color); - write_pixel(frame, width, x + 8, y + 5, color); - } - if segments.c { - write_pixel(frame, width, x + 8, y + 7, color); - write_pixel(frame, width, x + 8, y + 8, color); - write_pixel(frame, width, x + 8, y + 9, color); - write_pixel(frame, width, x + 8, y + 10, color); - write_pixel(frame, width, x + 8, y + 11, color); - } - - if segments.d1 { - write_pixel(frame, width, x + 1, y + 12, color); - write_pixel(frame, width, x + 2, y + 12, color); - write_pixel(frame, width, x + 3, y + 12, color); - } - if segments.d2 { - write_pixel(frame, width, x + 5, y + 12, color); - write_pixel(frame, width, x + 6, y + 12, color); - write_pixel(frame, width, x + 7, y + 12, color); - } - - if segments.e { - write_pixel(frame, width, x, y + 7, color); - write_pixel(frame, width, x, y + 8, color); - write_pixel(frame, width, x, y + 9, color); - write_pixel(frame, width, x, y + 10, color); - write_pixel(frame, width, x, y + 11, color); - } - if segments.f { - write_pixel(frame, width, x, y + 1, color); - write_pixel(frame, width, x, y + 2, color); - write_pixel(frame, width, x, y + 3, color); - write_pixel(frame, width, x, y + 4, color); - write_pixel(frame, width, x, y + 5, color); - } - - if segments.g1 { - write_pixel(frame, width, x + 1, y + 6, color); - write_pixel(frame, width, x + 2, y + 6, color); - write_pixel(frame, width, x + 3, y + 6, color); - } - if segments.g2 { - write_pixel(frame, width, x + 5, y + 6, color); - write_pixel(frame, width, x + 6, y + 6, color); - write_pixel(frame, width, x + 7, y + 6, color); - } - - if segments.h { - write_pixel(frame, width, x + 1, y + 1, color); - write_pixel(frame, width, x + 1, y + 2, color); - write_pixel(frame, width, x + 2, y + 3, color); - write_pixel(frame, width, x + 3, y + 4, color); - write_pixel(frame, width, x + 3, y + 5, color); - } - if segments.i { - write_pixel(frame, width, x + 4, y + 1, color); - write_pixel(frame, width, x + 4, y + 2, color); - write_pixel(frame, width, x + 4, y + 3, color); - write_pixel(frame, width, x + 4, y + 4, color); - write_pixel(frame, width, x + 4, y + 5, color); - } - if segments.j { - write_pixel(frame, width, x + 7, y + 1, color); - write_pixel(frame, width, x + 7, y + 2, color); - write_pixel(frame, width, x + 6, y + 3, color); - write_pixel(frame, width, x + 5, y + 4, color); - write_pixel(frame, width, x + 5, y + 5, color); - } - if segments.k { - write_pixel(frame, width, x + 3, y + 7, color); - write_pixel(frame, width, x + 3, y + 8, color); - write_pixel(frame, width, x + 2, y + 9, color); - write_pixel(frame, width, x + 1, y + 10, color); - write_pixel(frame, width, x + 1, y + 11, color); - } - if segments.l { - write_pixel(frame, width, x + 4, y + 7, color); - write_pixel(frame, width, x + 4, y + 8, color); - write_pixel(frame, width, x + 4, y + 9, color); - write_pixel(frame, width, x + 4, y + 10, color); - write_pixel(frame, width, x + 4, y + 11, color); - } - if segments.m { - write_pixel(frame, width, x + 5, y + 7, color); - write_pixel(frame, width, x + 5, y + 8, color); - write_pixel(frame, width, x + 6, y + 9, color); - write_pixel(frame, width, x + 7, y + 10, color); - write_pixel(frame, width, x + 7, y + 11, color); - } -} */ From 958d18b9a8ddc904cc99a39b1fb3a2aa7c60468c Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Fri, 7 Mar 2025 18:48:56 -0500 Subject: [PATCH 31/34] Set up the seven segment font --- pico-st7789/src/canvas.rs | 94 ----------- pico-st7789/src/font/mod.rs | 4 + pico-st7789/src/font/seven_segment.rs | 211 ++++++++++++++++++++++++ pico-st7789/src/font/sixteen_segment.rs | 4 + pico-st7789/src/main.rs | 130 ++++++++++----- 5 files changed, 311 insertions(+), 132 deletions(-) create mode 100644 pico-st7789/src/font/seven_segment.rs diff --git a/pico-st7789/src/canvas.rs b/pico-st7789/src/canvas.rs index cb2d05b..b154ebf 100644 --- a/pico-st7789/src/canvas.rs +++ b/pico-st7789/src/canvas.rs @@ -8,11 +8,6 @@ // e c // d d d -use alloc::collections::btree_map::BTreeMap; - -/* -*/ - pub struct RGB { pub r: u8, pub g: u8, pub b: u8 } pub trait Canvas { @@ -34,92 +29,3 @@ pub trait Canvas { } } -/* -pub const DIGIT_WIDTH: usize = 5; -pub const DIGIT_HEIGHT: usize = 9; - -pub const ZERO_SEVEN_SEGMENT: [bool; 7] = [true, true, true, true, true, true, false]; - -pub const ONE_SEVEN_SEGMENT: [bool; 7] = [false, true, true, false, false, false, false]; - -pub const TWO_SEVEN_SEGMENT: [bool; 7] = [true, true, false, true, true, false, true]; - -pub const THREE_SEVEN_SEGMENT: [bool; 7] = [true, true, true, true, false, false, true]; - -pub const FOUR_SEVEN_SEGMENT: [bool; 7] = [false, true, true, false, false, true, true]; - -pub const FIVE_SEVEN_SEGMENT: [bool; 7] = [true, false, true, true, false, true, true]; - -pub const SIX_SEVEN_SEGMENT: [bool; 7] = [true, false, true, true, true, true, true]; - -pub const SEVEN_SEVEN_SEGMENT: [bool; 7] = [true, true, true, false, false, false, false]; - -pub const EIGHT_SEVEN_SEGMENT: [bool; 7] = [true, true, true, true, true, true, true]; - -pub const NINE_SEVEN_SEGMENT: [bool; 7] = [true, true, true, false, false, true, true]; - -pub const SEVEN_SEGMENT_FONT: [[bool; 7]; 10] = [ - ZERO_SEVEN_SEGMENT, - ONE_SEVEN_SEGMENT, - TWO_SEVEN_SEGMENT, - THREE_SEVEN_SEGMENT, - FOUR_SEVEN_SEGMENT, - FIVE_SEVEN_SEGMENT, - SIX_SEVEN_SEGMENT, - SEVEN_SEVEN_SEGMENT, - EIGHT_SEVEN_SEGMENT, - NINE_SEVEN_SEGMENT, -]; - -pub fn write_pixel(framebuf: &mut [u8], width: usize, x: usize, y: usize, color: (u8, u8, u8)) { - framebuf[(y * width + x) * 3 + 0] = color.0 << 2; - framebuf[(y * width + x) * 3 + 1] = color.1 << 2; - framebuf[(y * width + x) * 3 + 2] = color.2 << 2; -} - -pub fn draw_seven_segment( - digit: u8, - frame: &mut [u8], - width: usize, - x: usize, - y: usize, - color: (u8, u8, u8), -) { - let segments = SEVEN_SEGMENT_FONT[digit as usize]; - if segments[0] { - write_pixel(frame, width, x + 1, y, color); - write_pixel(frame, width, x + 2, y, color); - write_pixel(frame, width, x + 3, y, color); - } - if segments[1] { - write_pixel(frame, width, x + 4, y + 1, color); - write_pixel(frame, width, x + 4, y + 2, color); - write_pixel(frame, width, x + 4, y + 3, color); - } - if segments[2] { - write_pixel(frame, width, x + 4, y + 5, color); - write_pixel(frame, width, x + 4, y + 6, color); - write_pixel(frame, width, x + 4, y + 7, color); - } - if segments[3] { - write_pixel(frame, width, x + 1, y + 8, color); - write_pixel(frame, width, x + 2, y + 8, color); - write_pixel(frame, width, x + 3, y + 8, color); - } - if segments[4] { - write_pixel(frame, width, x, y + 5, color); - write_pixel(frame, width, x, y + 6, color); - write_pixel(frame, width, x, y + 7, color); - } - if segments[5] { - write_pixel(frame, width, x, y + 1, color); - write_pixel(frame, width, x, y + 2, color); - write_pixel(frame, width, x, y + 3, color); - } - if segments[6] { - write_pixel(frame, width, x + 1, y + 4, color); - write_pixel(frame, width, x + 2, y + 4, color); - write_pixel(frame, width, x + 3, y + 4, color); - } -} -*/ diff --git a/pico-st7789/src/font/mod.rs b/pico-st7789/src/font/mod.rs index 126176d..3886765 100644 --- a/pico-st7789/src/font/mod.rs +++ b/pico-st7789/src/font/mod.rs @@ -1,5 +1,8 @@ use crate::canvas::{Canvas, RGB}; +mod seven_segment; +pub use seven_segment::SevenSegmentFont; + mod sixteen_segment; pub use sixteen_segment::SixteenSegmentFont; @@ -9,5 +12,6 @@ pub trait Font<A> { pub trait Glyph { fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB); + fn extents(&self) -> (usize, usize); } diff --git a/pico-st7789/src/font/seven_segment.rs b/pico-st7789/src/font/seven_segment.rs new file mode 100644 index 0000000..94dd8cd --- /dev/null +++ b/pico-st7789/src/font/seven_segment.rs @@ -0,0 +1,211 @@ +use alloc::collections::btree_map::BTreeMap; +use bitflags::bitflags; + +use crate::canvas::{Canvas, RGB}; + +use super::{Font, Glyph}; + +// Seven Segments +// +// a a a +// f b +// f b +// f b +// g g g +// e c +// e c +// e c +// d d d + +bitflags! { + pub struct SevenSegmentGlyph: u8 { + const NONE = 0; + const A = 0x01; + const B = 0x01 << 1; + const C = 0x01 << 2; + const D = 0x01 << 3; + const E = 0x01 << 4; + const F = 0x01 << 5; + const G = 0x01 << 6; + const DOT = 0x01 << 7; + } +} + +pub struct SevenSegmentFont(BTreeMap<char, SevenSegmentGlyph>); + +impl SevenSegmentFont { + pub fn new() -> Self { + let mut font = BTreeMap::new(); + font.insert(' ', SevenSegmentGlyph::NONE); + font.insert( + '0', + SevenSegmentGlyph::A + | SevenSegmentGlyph::B + | SevenSegmentGlyph::C + | SevenSegmentGlyph::D + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F, + ); + font.insert('1', SevenSegmentGlyph::B | SevenSegmentGlyph::C); + font.insert( + '2', + SevenSegmentGlyph::A + | SevenSegmentGlyph::B + | SevenSegmentGlyph::D + | SevenSegmentGlyph::E + | SevenSegmentGlyph::G, + ); + font.insert( + '3', + SevenSegmentGlyph::A + | SevenSegmentGlyph::B + | SevenSegmentGlyph::C + | SevenSegmentGlyph::G + | SevenSegmentGlyph::D, + ); + font.insert( + '4', + SevenSegmentGlyph::B + | SevenSegmentGlyph::C + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + font.insert( + '5', + SevenSegmentGlyph::A + | SevenSegmentGlyph::C + | SevenSegmentGlyph::D + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + font.insert( + '6', + SevenSegmentGlyph::A + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G + | SevenSegmentGlyph::C + | SevenSegmentGlyph::D, + ); + font.insert( + '7', + SevenSegmentGlyph::A | SevenSegmentGlyph::B | SevenSegmentGlyph::C, + ); + font.insert( + '8', + SevenSegmentGlyph::A + | SevenSegmentGlyph::B + | SevenSegmentGlyph::C + | SevenSegmentGlyph::D + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + font.insert( + '9', + SevenSegmentGlyph::A + | SevenSegmentGlyph::B + | SevenSegmentGlyph::C + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + font.insert( + 'A', + SevenSegmentGlyph::A + | SevenSegmentGlyph::B + | SevenSegmentGlyph::C + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + font.insert( + 'B', + SevenSegmentGlyph::C + | SevenSegmentGlyph::D + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + font.insert( + 'C', + SevenSegmentGlyph::A + | SevenSegmentGlyph::D + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F, + ); + font.insert( + 'D', + SevenSegmentGlyph::B + | SevenSegmentGlyph::C + | SevenSegmentGlyph::D + | SevenSegmentGlyph::E + | SevenSegmentGlyph::G, + ); + font.insert( + 'E', + SevenSegmentGlyph::A + | SevenSegmentGlyph::D + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + font.insert( + 'F', + SevenSegmentGlyph::A + | SevenSegmentGlyph::E + | SevenSegmentGlyph::F + | SevenSegmentGlyph::G, + ); + Self(font) + } +} + +impl Font<SevenSegmentGlyph> for SevenSegmentFont { + fn glyph(&self, c: char) -> &SevenSegmentGlyph { + self.0.get(&c).unwrap() + } +} + +impl Glyph for SevenSegmentGlyph { + fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB) { + if self.contains(SevenSegmentGlyph::A) { + canvas.set_pixel(x + 1, y, color); + canvas.set_pixel(x + 2, y, color); + canvas.set_pixel(x + 3, y, color); + } + if self.contains(SevenSegmentGlyph::B) { + canvas.set_pixel(x + 4, y + 1, color); + canvas.set_pixel(x + 4, y + 2, color); + canvas.set_pixel(x + 4, y + 3, color); + } + if self.contains(SevenSegmentGlyph::C) { + canvas.set_pixel(x + 4, y + 5, color); + canvas.set_pixel(x + 4, y + 6, color); + canvas.set_pixel(x + 4, y + 7, color); + } + if self.contains(SevenSegmentGlyph::D) { + canvas.set_pixel(x + 1, y + 8, color); + canvas.set_pixel(x + 2, y + 8, color); + canvas.set_pixel(x + 3, y + 8, color); + } + if self.contains(SevenSegmentGlyph::E) { + canvas.set_pixel(x, y + 5, color); + canvas.set_pixel(x, y + 6, color); + canvas.set_pixel(x, y + 7, color); + } + if self.contains(SevenSegmentGlyph::F) { + canvas.set_pixel(x, y + 1, color); + canvas.set_pixel(x, y + 2, color); + canvas.set_pixel(x, y + 3, color); + } + if self.contains(SevenSegmentGlyph::G) { + canvas.set_pixel(x + 1, y + 4, color); + canvas.set_pixel(x + 2, y + 4, color); + canvas.set_pixel(x + 3, y + 4, color); + } + } + + fn extents(&self) -> (usize, usize) { + (5, 8) + } +} diff --git a/pico-st7789/src/font/sixteen_segment.rs b/pico-st7789/src/font/sixteen_segment.rs index fe021ff..62e1285 100644 --- a/pico-st7789/src/font/sixteen_segment.rs +++ b/pico-st7789/src/font/sixteen_segment.rs @@ -513,5 +513,9 @@ impl Glyph for SixteenSegmentGlyph { canvas.set_pixel(x + 7, y + 11, color); } } + + fn extents(&self) -> (usize, usize) { + (8, 12) + } } diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index e404ba7..68b0554 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -17,7 +17,7 @@ mod canvas; use canvas::{Canvas, RGB}; mod font; -use font::{Font, Glyph, SixteenSegmentFont}; +use font::{Font, Glyph, SevenSegmentFont, SixteenSegmentFont}; mod st7789; use st7789::{ST7789Display, SETUP_PROGRAM}; @@ -64,6 +64,7 @@ unsafe fn main() -> ! { unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) } } + let font_seven = SevenSegmentFont::new(); let font_sixteen = SixteenSegmentFont::new(); // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico. @@ -163,119 +164,172 @@ unsafe fn main() -> ! { draw_seven_segment(9, &mut frame, 63, 0, (255, 255, 255)); */ - font_sixteen + font_seven .glyph(' ') .draw(&mut canvas, 0, 0, &white); - font_sixteen + font_seven .glyph('0') .draw(&mut canvas, 11, 0, &white); - font_sixteen + font_seven .glyph('1') .draw(&mut canvas, 22, 0, &white); - font_sixteen + font_seven .glyph('2') .draw(&mut canvas, 33, 0, &white); - font_sixteen + font_seven .glyph('3') .draw(&mut canvas, 44, 0, &white); - font_sixteen + font_seven .glyph('4') .draw(&mut canvas, 55, 0, &white); - font_sixteen + font_seven .glyph('5') .draw(&mut canvas, 66, 0, &white); - font_sixteen + font_seven .glyph('6') .draw(&mut canvas, 77, 0, &white); - font_sixteen + font_seven .glyph('7') .draw(&mut canvas, 88, 0, &white); - font_sixteen + font_seven .glyph('8') .draw(&mut canvas, 99, 0, &white); - font_sixteen + font_seven .glyph('9') .draw(&mut canvas, 110, 0, &white); - font_sixteen + font_seven .glyph('A') + .draw(&mut canvas, 121, 0, &white); + font_seven + .glyph('B') + .draw(&mut canvas, 132, 0, &white); + font_seven + .glyph('C') + .draw(&mut canvas, 143, 0, &white); + font_seven + .glyph('D') + .draw(&mut canvas, 154, 0, &white); + font_seven + .glyph('E') + .draw(&mut canvas, 165, 0, &white); + font_seven + .glyph('F') + .draw(&mut canvas, 0, 20, &white); + + font_sixteen + .glyph(' ') .draw(&mut canvas, 0, 20, &white); font_sixteen - .glyph('B') + .glyph('0') .draw(&mut canvas, 11, 20, &white); font_sixteen - .glyph('C') + .glyph('1') .draw(&mut canvas, 22, 20, &white); font_sixteen - .glyph('D') + .glyph('2') .draw(&mut canvas, 33, 20, &white); font_sixteen - .glyph('E') + .glyph('3') .draw(&mut canvas, 44, 20, &white); font_sixteen - .glyph('F') + .glyph('4') .draw(&mut canvas, 55, 20, &white); font_sixteen - .glyph('G') + .glyph('5') .draw(&mut canvas, 66, 20, &white); font_sixteen - .glyph('H') + .glyph('6') .draw(&mut canvas, 77, 20, &white); font_sixteen - .glyph('I') + .glyph('7') .draw(&mut canvas, 88, 20, &white); font_sixteen - .glyph('J') + .glyph('8') .draw(&mut canvas, 99, 20, &white); font_sixteen - .glyph('K') + .glyph('9') .draw(&mut canvas, 110, 20, &white); + + font_sixteen + .glyph('A') + .draw(&mut canvas, 0, 40, &white); + font_sixteen + .glyph('B') + .draw(&mut canvas, 11, 40, &white); + font_sixteen + .glyph('C') + .draw(&mut canvas, 22, 40, &white); + font_sixteen + .glyph('D') + .draw(&mut canvas, 33, 40, &white); + font_sixteen + .glyph('E') + .draw(&mut canvas, 44, 40, &white); + font_sixteen + .glyph('F') + .draw(&mut canvas, 55, 40, &white); + font_sixteen + .glyph('G') + .draw(&mut canvas, 66, 40, &white); + font_sixteen + .glyph('H') + .draw(&mut canvas, 77, 40, &white); + font_sixteen + .glyph('I') + .draw(&mut canvas, 88, 40, &white); + font_sixteen + .glyph('J') + .draw(&mut canvas, 99, 40, &white); + font_sixteen + .glyph('K') + .draw(&mut canvas, 110, 40, &white); font_sixteen .glyph('L') - .draw(&mut canvas, 121, 20, &white); + .draw(&mut canvas, 121, 40, &white); font_sixteen .glyph('M') - .draw(&mut canvas, 132, 20, &white); + .draw(&mut canvas, 132, 40, &white); font_sixteen .glyph('N') - .draw(&mut canvas, 143, 20, &white); + .draw(&mut canvas, 143, 40, &white); font_sixteen .glyph('O') - .draw(&mut canvas, 154, 20, &white); + .draw(&mut canvas, 154, 40, &white); font_sixteen .glyph('P') - .draw(&mut canvas, 0, 40, &white); + .draw(&mut canvas, 0, 60, &white); font_sixteen .glyph('Q') - .draw(&mut canvas, 11, 40, &white); + .draw(&mut canvas, 11, 60, &white); font_sixteen .glyph('R') - .draw(&mut canvas, 22, 40, &white); + .draw(&mut canvas, 22, 60, &white); font_sixteen .glyph('S') - .draw(&mut canvas, 33, 40, &white); + .draw(&mut canvas, 33, 60, &white); font_sixteen .glyph('T') - .draw(&mut canvas, 44, 40, &white); + .draw(&mut canvas, 44, 60, &white); font_sixteen .glyph('U') - .draw(&mut canvas, 55, 40, &white); + .draw(&mut canvas, 55, 60, &white); font_sixteen .glyph('V') - .draw(&mut canvas, 66, 40, &white); + .draw(&mut canvas, 66, 60, &white); font_sixteen .glyph('W') - .draw(&mut canvas, 77, 40, &white); + .draw(&mut canvas, 77, 60, &white); font_sixteen .glyph('X') - .draw(&mut canvas, 88, 40, &white); + .draw(&mut canvas, 88, 60, &white); font_sixteen .glyph('Y') - .draw(&mut canvas, 99, 40, &white); + .draw(&mut canvas, 99, 60, &white); font_sixteen .glyph('Z') - .draw(&mut canvas, 110, 40, &white); + .draw(&mut canvas, 110, 60, &white); canvas.square(10, 70, 160, 310, &white); From 9175b9d4ccb57aa58d0cb72f3301c036432800fe Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Fri, 7 Mar 2025 22:23:03 -0500 Subject: [PATCH 32/34] Create a print function and fill out more sixteen-segment glyphs --- pico-st7789/src/canvas.rs | 33 +++- pico-st7789/src/font/sixteen_segment.rs | 174 +++++++++++++++++++++- pico-st7789/src/main.rs | 190 +----------------------- 3 files changed, 207 insertions(+), 190 deletions(-) diff --git a/pico-st7789/src/canvas.rs b/pico-st7789/src/canvas.rs index b154ebf..a695b86 100644 --- a/pico-st7789/src/canvas.rs +++ b/pico-st7789/src/canvas.rs @@ -8,24 +8,47 @@ // e c // d d d -pub struct RGB { pub r: u8, pub g: u8, pub b: u8 } +use crate::font::{Font, Glyph}; + +pub struct RGB { + pub r: u8, + pub g: u8, + pub b: u8, +} pub trait Canvas { fn set_pixel(&mut self, x: usize, y: usize, color: &RGB); fn square(&mut self, x1: usize, y1: usize, x2: usize, y2: usize, color: &RGB) { - for x in x1..x2+1 { + for x in x1..x2 + 1 { self.set_pixel(x, y1, color); } - for x in x1..x2+1 { + for x in x1..x2 + 1 { self.set_pixel(x, y2, color); } - for y in y1..y2+1 { + for y in y1..y2 + 1 { self.set_pixel(x1, y, color); } - for y in y1..y2+1 { + for y in y1..y2 + 1 { self.set_pixel(x2, y, color); } } } +pub fn print<A>( + canvas: &mut impl Canvas, + font: &impl Font<A>, + sx: usize, + sy: usize, + text: &str, + color: &RGB, +) where + A: Glyph, +{ + let mut x = sx; + for c in text.chars().map(|c| font.glyph(c)) { + c.draw(canvas, x, sy, color); + let (dx, _) = c.extents(); + x = x + dx + 1; + } +} diff --git a/pico-st7789/src/font/sixteen_segment.rs b/pico-st7789/src/font/sixteen_segment.rs index 62e1285..eb8cb29 100644 --- a/pico-st7789/src/font/sixteen_segment.rs +++ b/pico-st7789/src/font/sixteen_segment.rs @@ -396,13 +396,182 @@ impl SixteenSegmentFont { | SixteenSegmentGlyph::J | SixteenSegmentGlyph::K, ); + + font.insert( + 'a', + SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::L, + ); + font.insert( + 'b', + SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::L, + ); + font.insert( + 'c', + SixteenSegmentGlyph::D1 | SixteenSegmentGlyph::E | SixteenSegmentGlyph::G1, + ); + font.insert( + 'd', + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::G2 + | SixteenSegmentGlyph::L, + ); + font.insert( + 'e', + SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::K, + ); + font.insert( + 'f', + SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2 + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, + ); + font.insert( + 'g', + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, + ); + font.insert( + 'h', + SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::L, + ); + font.insert('i', SixteenSegmentGlyph::L); + font.insert( + 'j', + SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, + ); + font.insert( + 'k', + SixteenSegmentGlyph::I + | SixteenSegmentGlyph::J + | SixteenSegmentGlyph::L + | SixteenSegmentGlyph::M, + ); + font.insert('l', SixteenSegmentGlyph::E | SixteenSegmentGlyph::F); + font.insert( + 'm', + SixteenSegmentGlyph::C + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::L + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::G2, + ); + font.insert( + 'n', + SixteenSegmentGlyph::E | SixteenSegmentGlyph::L | SixteenSegmentGlyph::G1, + ); + font.insert( + 'o', + SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::L, + ); + font.insert( + 'p', + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::I, + ); + font.insert( + 'q', + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::L, + ); + font.insert('r', SixteenSegmentGlyph::E | SixteenSegmentGlyph::G1); + font.insert( + 's', + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1 + | SixteenSegmentGlyph::L, + ); + font.insert( + 't', + SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G1, + ); + font.insert( + 'u', + SixteenSegmentGlyph::D1 | SixteenSegmentGlyph::E | SixteenSegmentGlyph::L, + ); + font.insert('v', SixteenSegmentGlyph::E | SixteenSegmentGlyph::K); + font.insert( + 'w', + SixteenSegmentGlyph::C + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::K + | SixteenSegmentGlyph::M, + ); + font.insert( + 'x', + SixteenSegmentGlyph::H + | SixteenSegmentGlyph::J + | SixteenSegmentGlyph::K + | SixteenSegmentGlyph::M, + ); + font.insert( + 'y', + SixteenSegmentGlyph::B + | SixteenSegmentGlyph::C + | SixteenSegmentGlyph::I + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::G2, + ); + font.insert( + 'z', + SixteenSegmentGlyph::D1 | SixteenSegmentGlyph::G1 | SixteenSegmentGlyph::K, + ); + font.insert( + '@', + SixteenSegmentGlyph::A1 + | SixteenSegmentGlyph::A2 + | SixteenSegmentGlyph::B + | SixteenSegmentGlyph::D1 + | SixteenSegmentGlyph::D2 + | SixteenSegmentGlyph::E + | SixteenSegmentGlyph::F + | SixteenSegmentGlyph::G2, + ); Self(font) } } impl Font<SixteenSegmentGlyph> for SixteenSegmentFont { fn glyph(&self, c: char) -> &SixteenSegmentGlyph { - self.0.get(&c).unwrap() + self.0.get(&c).unwrap_or(&SixteenSegmentGlyph::NONE) } } @@ -515,7 +684,6 @@ impl Glyph for SixteenSegmentGlyph { } fn extents(&self) -> (usize, usize) { - (8, 12) + (9, 13) } } - diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 68b0554..79da94f 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -14,7 +14,7 @@ use rp_pico::{ }; mod canvas; -use canvas::{Canvas, RGB}; +use canvas::{Canvas, RGB, print}; mod font; use font::{Font, Glyph, SevenSegmentFont, SixteenSegmentFont}; @@ -64,7 +64,6 @@ unsafe fn main() -> ! { unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) } } - let font_seven = SevenSegmentFont::new(); let font_sixteen = SixteenSegmentFont::new(); // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico. @@ -151,187 +150,14 @@ unsafe fn main() -> ! { let white = RGB { r: 63, g: 63, b: 63 }; loop { - /* - draw_seven_segment(0, &mut frame, 0, 0, (255, 255, 255)); - draw_seven_segment(1, &mut frame, 7, 0, (255, 255, 255)); - draw_seven_segment(2, &mut frame, 14, 0, (255, 255, 255)); - draw_seven_segment(3, &mut frame, 21, 0, (255, 255, 255)); - draw_seven_segment(4, &mut frame, 28, 0, (255, 255, 255)); - draw_seven_segment(5, &mut frame, 35, 0, (255, 255, 255)); - draw_seven_segment(6, &mut frame, 42, 0, (255, 255, 255)); - draw_seven_segment(7, &mut frame, 49, 0, (255, 255, 255)); - draw_seven_segment(8, &mut frame, 56, 0, (255, 255, 255)); - draw_seven_segment(9, &mut frame, 63, 0, (255, 255, 255)); - */ + print(&mut canvas, &font_sixteen, 1, 1, " !\"#$%&'<>*+,-./", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_sixteen, 1, 21, "0123456789| = ?", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_sixteen, 1, 41, "@ABCDEFGHIJKLMNO", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_sixteen, 1, 61, "PQRSTUVWXYZ[\\]^_", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_sixteen, 1, 81, "`abcdefghijklmno", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_sixteen, 1, 101, "pqrstuvwxyz{|}", &RGB{ r: 63, g: 63, b: 63 }); - font_seven - .glyph(' ') - .draw(&mut canvas, 0, 0, &white); - font_seven - .glyph('0') - .draw(&mut canvas, 11, 0, &white); - font_seven - .glyph('1') - .draw(&mut canvas, 22, 0, &white); - font_seven - .glyph('2') - .draw(&mut canvas, 33, 0, &white); - font_seven - .glyph('3') - .draw(&mut canvas, 44, 0, &white); - font_seven - .glyph('4') - .draw(&mut canvas, 55, 0, &white); - font_seven - .glyph('5') - .draw(&mut canvas, 66, 0, &white); - font_seven - .glyph('6') - .draw(&mut canvas, 77, 0, &white); - font_seven - .glyph('7') - .draw(&mut canvas, 88, 0, &white); - font_seven - .glyph('8') - .draw(&mut canvas, 99, 0, &white); - font_seven - .glyph('9') - .draw(&mut canvas, 110, 0, &white); - - font_seven - .glyph('A') - .draw(&mut canvas, 121, 0, &white); - font_seven - .glyph('B') - .draw(&mut canvas, 132, 0, &white); - font_seven - .glyph('C') - .draw(&mut canvas, 143, 0, &white); - font_seven - .glyph('D') - .draw(&mut canvas, 154, 0, &white); - font_seven - .glyph('E') - .draw(&mut canvas, 165, 0, &white); - font_seven - .glyph('F') - .draw(&mut canvas, 0, 20, &white); - - font_sixteen - .glyph(' ') - .draw(&mut canvas, 0, 20, &white); - font_sixteen - .glyph('0') - .draw(&mut canvas, 11, 20, &white); - font_sixteen - .glyph('1') - .draw(&mut canvas, 22, 20, &white); - font_sixteen - .glyph('2') - .draw(&mut canvas, 33, 20, &white); - font_sixteen - .glyph('3') - .draw(&mut canvas, 44, 20, &white); - font_sixteen - .glyph('4') - .draw(&mut canvas, 55, 20, &white); - font_sixteen - .glyph('5') - .draw(&mut canvas, 66, 20, &white); - font_sixteen - .glyph('6') - .draw(&mut canvas, 77, 20, &white); - font_sixteen - .glyph('7') - .draw(&mut canvas, 88, 20, &white); - font_sixteen - .glyph('8') - .draw(&mut canvas, 99, 20, &white); - font_sixteen - .glyph('9') - .draw(&mut canvas, 110, 20, &white); - - font_sixteen - .glyph('A') - .draw(&mut canvas, 0, 40, &white); - font_sixteen - .glyph('B') - .draw(&mut canvas, 11, 40, &white); - font_sixteen - .glyph('C') - .draw(&mut canvas, 22, 40, &white); - font_sixteen - .glyph('D') - .draw(&mut canvas, 33, 40, &white); - font_sixteen - .glyph('E') - .draw(&mut canvas, 44, 40, &white); - font_sixteen - .glyph('F') - .draw(&mut canvas, 55, 40, &white); - font_sixteen - .glyph('G') - .draw(&mut canvas, 66, 40, &white); - font_sixteen - .glyph('H') - .draw(&mut canvas, 77, 40, &white); - font_sixteen - .glyph('I') - .draw(&mut canvas, 88, 40, &white); - font_sixteen - .glyph('J') - .draw(&mut canvas, 99, 40, &white); - font_sixteen - .glyph('K') - .draw(&mut canvas, 110, 40, &white); - font_sixteen - .glyph('L') - .draw(&mut canvas, 121, 40, &white); - font_sixteen - .glyph('M') - .draw(&mut canvas, 132, 40, &white); - font_sixteen - .glyph('N') - .draw(&mut canvas, 143, 40, &white); - font_sixteen - .glyph('O') - .draw(&mut canvas, 154, 40, &white); - - font_sixteen - .glyph('P') - .draw(&mut canvas, 0, 60, &white); - font_sixteen - .glyph('Q') - .draw(&mut canvas, 11, 60, &white); - font_sixteen - .glyph('R') - .draw(&mut canvas, 22, 60, &white); - font_sixteen - .glyph('S') - .draw(&mut canvas, 33, 60, &white); - font_sixteen - .glyph('T') - .draw(&mut canvas, 44, 60, &white); - font_sixteen - .glyph('U') - .draw(&mut canvas, 55, 60, &white); - font_sixteen - .glyph('V') - .draw(&mut canvas, 66, 60, &white); - font_sixteen - .glyph('W') - .draw(&mut canvas, 77, 60, &white); - font_sixteen - .glyph('X') - .draw(&mut canvas, 88, 60, &white); - font_sixteen - .glyph('Y') - .draw(&mut canvas, 99, 60, &white); - font_sixteen - .glyph('Z') - .draw(&mut canvas, 110, 60, &white); - - canvas.square(10, 70, 160, 310, &white); + // canvas.square(10, 70, 160, 310, &white); { let display = display.acquire(); From 596de6525f2b4908bc9e4a53ca7178ace1ca6dbd Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 8 Mar 2025 01:23:06 -0500 Subject: [PATCH 33/34] Create a 5x8 bitmap font --- pico-st7789/src/font/bits_5_8.rs | 180 +++++++++++++++++++++++++++++++ pico-st7789/src/font/mod.rs | 3 + pico-st7789/src/main.rs | 15 ++- 3 files changed, 196 insertions(+), 2 deletions(-) create mode 100644 pico-st7789/src/font/bits_5_8.rs diff --git a/pico-st7789/src/font/bits_5_8.rs b/pico-st7789/src/font/bits_5_8.rs new file mode 100644 index 0000000..301e3dd --- /dev/null +++ b/pico-st7789/src/font/bits_5_8.rs @@ -0,0 +1,180 @@ +use alloc::collections::btree_map::BTreeMap; + +use crate::canvas::{Canvas, RGB}; + +use super::{Font, Glyph}; + +pub struct BitmapGlyph([u8; 8]); + +pub struct BitmapFont(BTreeMap<char, BitmapGlyph>); + +impl BitmapFont { + pub fn new() -> Self { + let mut font = BTreeMap::new(); + font.insert(' ', BitmapGlyph([ + 0b10101, + 0b01010, + 0b10101, + 0b01010, + 0b10101, + 0b01010, + 0b10101, + 0b01010, + ])); + font.insert( + '/', + BitmapGlyph([ + 0b00001, + 0b00010, + 0b00010, + 0b00100, + 0b00100, + 0b01000, + 0b01000, + 0b10000, + ]), + ); + font.insert( + '0', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10001, + 0b10101, + 0b10101, + 0b10001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + '1', + BitmapGlyph([ + 0b00001, 0b00011, 0b00101, 0b00001, 0b00001, 0b00001, 0b00001, 0b00001, + ]), + ); + font.insert( + '2', + BitmapGlyph([ + 0b01110, 0b10001, 0b00001, 0b00001, 0b00010, 0b00100, 0b01000, 0b11111, + ]), + ); + font.insert( + '3', + BitmapGlyph([ + 0b01110, 0b10001, 0b00001, 0b01110, 0b00001, 0b00001, 0b10001, 0b01110, + ]), + ); + font.insert( + '4', + BitmapGlyph([ + 0b10001, + 0b10001, + 0b10001, + 0b11111, + 0b00001, + 0b00001, + 0b00001, + 0b00001, + ]), + ); + font.insert( + '5', + BitmapGlyph([ + 0b11111, + 0b10000, + 0b10000, + 0b11111, + 0b00001, + 0b00001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + '6', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10000, + 0b10000, + 0b11110, + 0b10001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + '7', + BitmapGlyph([ + 0b11111, + 0b00001, + 0b00001, + 0b00010, + 0b00110, + 0b00100, + 0b01000, + 0b01000, + ]), + ); + font.insert( + '8', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10001, + 0b01110, + 0b01110, + 0b10001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + '9', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10001, + 0b10001, + 0b01111, + 0b00001, + 0b00001, + 0b01110, + ]), + ); + Self(font) + } +} + +impl Font<BitmapGlyph> for BitmapFont { + fn glyph(&self, c: char) -> &BitmapGlyph { + self.0.get(&c).unwrap_or(self.0.get(&' ').unwrap()) + } +} + +impl Glyph for BitmapGlyph { + fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB) { + for row in 0..8 { + if self.0[row] & (1 << 4) > 0 { + canvas.set_pixel(x, y + row, color); + } + if self.0[row] & (1 << 3) > 0 { + canvas.set_pixel(x + 1, y + row, color); + } + if self.0[row] & (1 << 2) > 0 { + canvas.set_pixel(x + 2, y + row, color); + } + if self.0[row] & (1 << 1) > 0 { + canvas.set_pixel(x + 3, y + row, color); + } + if self.0[row] & 1 > 0 { + canvas.set_pixel(x + 4, y + row, color); + } + } + } + + fn extents(&self) -> (usize, usize) { + (5, 8) + } +} diff --git a/pico-st7789/src/font/mod.rs b/pico-st7789/src/font/mod.rs index 3886765..f20065b 100644 --- a/pico-st7789/src/font/mod.rs +++ b/pico-st7789/src/font/mod.rs @@ -1,5 +1,8 @@ use crate::canvas::{Canvas, RGB}; +mod bits_5_8; +pub use bits_5_8::BitmapFont; + mod seven_segment; pub use seven_segment::SevenSegmentFont; diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 79da94f..5537613 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -17,7 +17,7 @@ mod canvas; use canvas::{Canvas, RGB, print}; mod font; -use font::{Font, Glyph, SevenSegmentFont, SixteenSegmentFont}; +use font::{BitmapFont, Font, Glyph, SevenSegmentFont, SixteenSegmentFont}; mod st7789; use st7789::{ST7789Display, SETUP_PROGRAM}; @@ -59,12 +59,13 @@ impl Canvas for FrameBuf { unsafe fn main() -> ! { { use core::mem::MaybeUninit; - const HEAP_SIZE: usize = 1024; + const HEAP_SIZE: usize = 64 * 1024; static mut HEAP_MEM: [MaybeUninit<u8>; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE]; unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) } } let font_sixteen = SixteenSegmentFont::new(); + let font_bitmap = BitmapFont::new(); // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico. let mut peripherals = pac::Peripherals::take().unwrap(); @@ -157,6 +158,16 @@ unsafe fn main() -> ! { print(&mut canvas, &font_sixteen, 1, 81, "`abcdefghijklmno", &RGB{ r: 63, g: 63, b: 63 }); print(&mut canvas, &font_sixteen, 1, 101, "pqrstuvwxyz{|}", &RGB{ r: 63, g: 63, b: 63 }); + + print(&mut canvas, &font_bitmap, 1, 121, " !\"#$%&'<>*+,-./", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 141, "0123456789| = ?", &RGB{ r: 63, g: 63, b: 63 }); + /* + print(&mut canvas, &font_bitmap, 1, 161, "@ABCDEFGHIJKLMNO", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 181, "PQRSTUVWXYZ[\\]^_", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 201, "`abcdefghijklmno", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 221, "pqrstuvwxyz{|}", &RGB{ r: 63, g: 63, b: 63 }); + */ + // canvas.square(10, 70, 160, 310, &white); { From 36eef459710d391515f5db6244d8a886be9bf533 Mon Sep 17 00:00:00 2001 From: Savanni D'Gerinel <savanni@luminescent-dreams.com> Date: Sat, 8 Mar 2025 01:49:43 -0500 Subject: [PATCH 34/34] Fill out an alphabet and animate the screen --- pico-st7789/src/canvas.rs | 8 + pico-st7789/src/font/bits_5_8.rs | 377 +++++++++++++++++++++++++++++-- pico-st7789/src/main.rs | 28 +-- 3 files changed, 374 insertions(+), 39 deletions(-) diff --git a/pico-st7789/src/canvas.rs b/pico-st7789/src/canvas.rs index a695b86..7a16821 100644 --- a/pico-st7789/src/canvas.rs +++ b/pico-st7789/src/canvas.rs @@ -19,6 +19,14 @@ pub struct RGB { pub trait Canvas { fn set_pixel(&mut self, x: usize, y: usize, color: &RGB); + fn fill(&mut self, x1: usize, y1: usize, x2: usize, y2: usize, color: &RGB) { + for x in x1..x2 + 1 { + for y in y1..y2 + 1 { + self.set_pixel(x, y, color); + } + } + } + fn square(&mut self, x1: usize, y1: usize, x2: usize, y2: usize, color: &RGB) { for x in x1..x2 + 1 { self.set_pixel(x, y1, color); diff --git a/pico-st7789/src/font/bits_5_8.rs b/pico-st7789/src/font/bits_5_8.rs index 301e3dd..ecada60 100644 --- a/pico-st7789/src/font/bits_5_8.rs +++ b/pico-st7789/src/font/bits_5_8.rs @@ -4,7 +4,7 @@ use crate::canvas::{Canvas, RGB}; use super::{Font, Glyph}; -pub struct BitmapGlyph([u8; 8]); +pub struct BitmapGlyph([u8; 7]); pub struct BitmapFont(BTreeMap<char, BitmapGlyph>); @@ -12,15 +12,26 @@ impl BitmapFont { pub fn new() -> Self { let mut font = BTreeMap::new(); font.insert(' ', BitmapGlyph([ - 0b10101, - 0b01010, - 0b10101, - 0b01010, - 0b10101, - 0b01010, - 0b10101, - 0b01010, + 0b00000, + 0b00000, + 0b00000, + 0b00000, + 0b00000, + 0b00000, + 0b00000, ])); + font.insert( + ':', + BitmapGlyph([ + 0b00000, + 0b00100, + 0b00100, + 0b00000, + 0b00100, + 0b00100, + 0b00000, + ]), + ); font.insert( '/', BitmapGlyph([ @@ -28,7 +39,6 @@ impl BitmapFont { 0b00010, 0b00010, 0b00100, - 0b00100, 0b01000, 0b01000, 0b10000, @@ -41,7 +51,6 @@ impl BitmapFont { 0b10001, 0b10001, 0b10101, - 0b10101, 0b10001, 0b10001, 0b01110, @@ -50,19 +59,37 @@ impl BitmapFont { font.insert( '1', BitmapGlyph([ - 0b00001, 0b00011, 0b00101, 0b00001, 0b00001, 0b00001, 0b00001, 0b00001, + 0b00001, + 0b00011, + 0b00101, + 0b00001, + 0b00001, + 0b00001, + 0b00001, ]), ); font.insert( '2', BitmapGlyph([ - 0b01110, 0b10001, 0b00001, 0b00001, 0b00010, 0b00100, 0b01000, 0b11111, + 0b01110, + 0b10001, + 0b00001, + 0b00010, + 0b00100, + 0b01000, + 0b11111, ]), ); font.insert( '3', BitmapGlyph([ - 0b01110, 0b10001, 0b00001, 0b01110, 0b00001, 0b00001, 0b10001, 0b01110, + 0b01110, + 0b10001, + 0b00001, + 0b01110, + 0b00001, + 0b10001, + 0b01110, ]), ); font.insert( @@ -75,7 +102,6 @@ impl BitmapFont { 0b00001, 0b00001, 0b00001, - 0b00001, ]), ); font.insert( @@ -86,7 +112,6 @@ impl BitmapFont { 0b10000, 0b11111, 0b00001, - 0b00001, 0b10001, 0b01110, ]), @@ -97,7 +122,6 @@ impl BitmapFont { 0b01110, 0b10001, 0b10000, - 0b10000, 0b11110, 0b10001, 0b10001, @@ -109,7 +133,6 @@ impl BitmapFont { BitmapGlyph([ 0b11111, 0b00001, - 0b00001, 0b00010, 0b00110, 0b00100, @@ -124,7 +147,6 @@ impl BitmapFont { 0b10001, 0b10001, 0b01110, - 0b01110, 0b10001, 0b10001, 0b01110, @@ -136,13 +158,324 @@ impl BitmapFont { 0b01110, 0b10001, 0b10001, - 0b10001, 0b01111, 0b00001, 0b00001, 0b01110, ]), ); + font.insert( + 'A', + BitmapGlyph([ + 0b00100, + 0b01010, + 0b01010, + 0b10001, + 0b11111, + 0b10001, + 0b10001, + ]), + ); + font.insert( + 'B', + BitmapGlyph([ + 0b11110, + 0b10001, + 0b10001, + 0b1111 , + 0b10001, + 0b10001, + 0b11110, + ]), + ); + font.insert( + 'C', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10000, + 0b10000, + 0b10000, + 0b10001, + 0b01110, + ]), + ); + font.insert( + 'D', + BitmapGlyph([ + 0b11110, + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b11110, + ]), + ); + font.insert( + 'E', + BitmapGlyph([ + 0b11111, + 0b10000, + 0b10000, + 0b11111, + 0b10000, + 0b10000, + 0b11111, + ]), + ); + font.insert( + 'F', + BitmapGlyph([ + 0b11111, + 0b10000, + 0b10000, + 0b11110, + 0b10000, + 0b10000, + 0b10000, + ]), + ); + font.insert( + 'G', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10000, + 0b10011, + 0b10001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + 'H', + BitmapGlyph([ + 0b10001, + 0b10001, + 0b10001, + 0b11111, + 0b10001, + 0b10001, + 0b10001, + ]), + ); + font.insert( + 'I', + BitmapGlyph([ + 0b11111, + 0b00100, + 0b00100, + 0b00100, + 0b00100, + 0b00100, + 0b11111, + ]), + ); + font.insert( + 'J', + BitmapGlyph([ + 0b00111, + 0b00001, + 0b00001, + 0b00001, + 0b00001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + 'K', + BitmapGlyph([ + 0b10001, + 0b10010, + 0b10100, + 0b11000, + 0b10100, + 0b10010, + 0b10001, + ]), + ); + font.insert( + 'L', + BitmapGlyph([ + 0b10000, + 0b10000, + 0b10000, + 0b10000, + 0b10000, + 0b10000, + 0b11111, + ]), + ); + font.insert( + 'M', + BitmapGlyph([ + 0b10001, + 0b11011, + 0b10101, + 0b10001, + 0b10001, + 0b10001, + 0b10001, + ]), + ); + font.insert( + 'N', + BitmapGlyph([ + 0b10001, + 0b11001, + 0b11001, + 0b10101, + 0b10011, + 0b10011, + 0b10001, + ]), + ); + font.insert( + 'O', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + 'P', + BitmapGlyph([ + 0b11110, + 0b10001, + 0b10001, + 0b11110, + 0b10000, + 0b10000, + 0b10000, + ]), + ); + font.insert( + 'Q', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10001, + 0b10001, + 0b10101, + 0b10011, + 0b01110, + ]), + ); + font.insert( + 'R', + BitmapGlyph([ + 0b11110, + 0b10001, + 0b10001, + 0b11110, + 0b10100, + 0b10010, + 0b10001, + ]), + ); + font.insert( + 'S', + BitmapGlyph([ + 0b01110, + 0b10001, + 0b10000, + 0b01110, + 0b00001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + 'T', + BitmapGlyph([ + 0b11111, + 0b00100, + 0b00100, + 0b00100, + 0b00100, + 0b00100, + 0b00100, + ]), + ); + font.insert( + 'U', + BitmapGlyph([ + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b01110, + ]), + ); + font.insert( + 'V', + BitmapGlyph([ + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b01010, + 0b01010, + 0b00100, + ]), + ); + font.insert( + 'W', + BitmapGlyph([ + 0b10001, + 0b10001, + 0b10001, + 0b10001, + 0b10101, + 0b10101, + 0b01010, + ]), + ); + font.insert( + 'X', + BitmapGlyph([ + 0b10001, + 0b10001, + 0b01010, + 0b00100, + 0b01010, + 0b10001, + 0b10001, + ]), + ); + font.insert( + 'Y', + BitmapGlyph([ + 0b10001, + 0b10001, + 0b01010, + 0b00100, + 0b00100, + 0b00100, + 0b00100, + ]), + ); + font.insert( + 'Z', + BitmapGlyph([ + 0b11111, + 0b00001, + 0b00010, + 0b00100, + 0b01000, + 0b10000, + 0b11111, + ]), + ); Self(font) } } @@ -155,7 +488,7 @@ impl Font<BitmapGlyph> for BitmapFont { impl Glyph for BitmapGlyph { fn draw(&self, canvas: &mut impl Canvas, x: usize, y: usize, color: &RGB) { - for row in 0..8 { + for row in 0..7 { if self.0[row] & (1 << 4) > 0 { canvas.set_pixel(x, y + row, color); } @@ -175,6 +508,6 @@ impl Glyph for BitmapGlyph { } fn extents(&self) -> (usize, usize) { - (5, 8) + (5, 7) } } diff --git a/pico-st7789/src/main.rs b/pico-st7789/src/main.rs index 5537613..ad3a90f 100644 --- a/pico-st7789/src/main.rs +++ b/pico-st7789/src/main.rs @@ -3,6 +3,7 @@ extern crate alloc; +use alloc::fmt::format; use embedded_alloc::LlffHeap as Heap; use embedded_hal::{delay::DelayNs, digital::OutputPin}; use fugit::RateExtU32; @@ -64,7 +65,6 @@ unsafe fn main() -> ! { unsafe { HEAP.init(HEAP_MEM.as_ptr() as usize, HEAP_SIZE) } } - let font_sixteen = SixteenSegmentFont::new(); let font_bitmap = BitmapFont::new(); // rp_pico::pac::Peripherals is a reference to physical hardware defined on the Pico. @@ -149,24 +149,17 @@ unsafe fn main() -> ! { */ let mut canvas = FrameBuf::new(); let white = RGB { r: 63, g: 63, b: 63 }; + let mut count = 0; loop { - print(&mut canvas, &font_sixteen, 1, 1, " !\"#$%&'<>*+,-./", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_sixteen, 1, 21, "0123456789| = ?", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_sixteen, 1, 41, "@ABCDEFGHIJKLMNO", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_sixteen, 1, 61, "PQRSTUVWXYZ[\\]^_", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_sixteen, 1, 81, "`abcdefghijklmno", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_sixteen, 1, 101, "pqrstuvwxyz{|}", &RGB{ r: 63, g: 63, b: 63 }); - - - print(&mut canvas, &font_bitmap, 1, 121, " !\"#$%&'<>*+,-./", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_bitmap, 1, 141, "0123456789| = ?", &RGB{ r: 63, g: 63, b: 63 }); - /* - print(&mut canvas, &font_bitmap, 1, 161, "@ABCDEFGHIJKLMNO", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_bitmap, 1, 181, "PQRSTUVWXYZ[\\]^_", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_bitmap, 1, 201, "`abcdefghijklmno", &RGB{ r: 63, g: 63, b: 63 }); - print(&mut canvas, &font_bitmap, 1, 221, "pqrstuvwxyz{|}", &RGB{ r: 63, g: 63, b: 63 }); - */ + canvas.fill(0, 10, 170, 20, &RGB{r: 0, g: 0, b: 0 }); + print(&mut canvas, &font_bitmap, 1, 10, &format(format_args!("COUNT: {:03}", count)), &RGB{ r: 32, g: 32, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 200, " !\"#$%&'<>*+,-./", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 220, "0123456789|: = ?", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 240, "@ABCDEFGHIJKLMNO", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 260, "PQRSTUVWXYZ[\\]^_", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 280, "`abcdefghijklmno", &RGB{ r: 63, g: 63, b: 63 }); + print(&mut canvas, &font_bitmap, 1, 300, "pqrstuvwxyz{|}", &RGB{ r: 63, g: 63, b: 63 }); // canvas.square(10, 70, 160, 310, &white); @@ -177,6 +170,7 @@ unsafe fn main() -> ! { display.send_buf(canvas.buf); let _ = led.set_low(); } + count = count + 1; /* for x in 80..90 { for y in 155..165 {