mirror of
https://gitlab.com/asus-linux/asusctl.git
synced 2026-02-06 00:15:04 +01:00
Compare commits
89 Commits
6.2.0
...
f4dc4dfea9
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f4dc4dfea9 | ||
|
|
1238e090a5 | ||
|
|
f92545b0bb | ||
|
|
bf0ea88f01 | ||
|
|
4db901ce17 | ||
|
|
e105e6bd86 | ||
|
|
465f472c25 | ||
|
|
b04fed9a6d | ||
|
|
0311c74049 | ||
|
|
c08355f5c2 | ||
|
|
9ee154cd1c | ||
|
|
14c4b0501d | ||
|
|
eab7e89fa3 | ||
|
|
2c61cd8da6 | ||
|
|
c7de3fb0c8 | ||
|
|
6e83884c0a | ||
|
|
7edb77b41f | ||
|
|
737ffa522c | ||
|
|
0311cfb1f9 | ||
|
|
b0ee27fb74 | ||
|
|
d4eca0c93e | ||
|
|
a9f4aac875 | ||
|
|
8377056580 | ||
|
|
d541581012 | ||
|
|
72ef6dea07 | ||
|
|
bfadc39400 | ||
|
|
754d82d031 | ||
|
|
8ae072ea82 | ||
|
|
cec2168591 | ||
|
|
8d6acc5975 | ||
|
|
b20ecf5378 | ||
|
|
ade839e981 | ||
|
|
d625c279a6 | ||
|
|
5ea14be3fa | ||
|
|
64c2e55db4 | ||
|
|
5303bfc1ad | ||
|
|
635f1dc9b9 | ||
|
|
33a4dba8fe | ||
|
|
e4680c9543 | ||
|
|
60c4818381 | ||
|
|
9cd48dc101 | ||
|
|
8d954c16fa | ||
|
|
3665d9cc8e | ||
|
|
d7ddee246a | ||
|
|
e85938b34d | ||
|
|
b7d480dd39 | ||
|
|
77640d1637 | ||
|
|
32da2a2da0 | ||
|
|
7981a85ff5 | ||
|
|
e3035adf98 | ||
|
|
be17e0b388 | ||
|
|
20cbddb6fa | ||
|
|
e9c5315bda | ||
|
|
b521a9ffc1 | ||
|
|
5e48923db1 | ||
|
|
392436808d | ||
|
|
3d9a08d7e0 | ||
|
|
ff103f98af | ||
|
|
d05182ae64 | ||
|
|
a4957a6eeb | ||
|
|
dda750cf33 | ||
|
|
0b5e04393a | ||
|
|
c9c9a022a4 | ||
|
|
aa063c20fd | ||
|
|
8551908452 | ||
|
|
5ecb174b8f | ||
|
|
fa266bff5b | ||
|
|
f53f1f360f | ||
|
|
da19216b78 | ||
|
|
46efd4190f | ||
|
|
ed3022e25e | ||
|
|
7c10d6c6d9 | ||
|
|
af5f3a5c71 | ||
|
|
7d5ec5f2c7 | ||
|
|
1c8acf6de3 | ||
|
|
7b644e7ad6 | ||
|
|
0f02fe868c | ||
|
|
abd3100e30 | ||
|
|
6f651c2b85 | ||
|
|
93ec5d1bce | ||
|
|
5aea7f51c0 | ||
|
|
3a206eb76f | ||
|
|
4449838282 | ||
|
|
58d740f77a | ||
|
|
f0488d9750 | ||
|
|
f1b9ae6f71 | ||
|
|
db5de3b854 | ||
|
|
7a3d39b8f1 | ||
|
|
7a5d6325c0 |
5
.gitignore
vendored
5
.gitignore
vendored
@@ -20,3 +20,8 @@ desktop-extensions/gnome*/@types/gir-generated
|
|||||||
desktop-extensions/gnome*/node_modules
|
desktop-extensions/gnome*/node_modules
|
||||||
desktop-extensions/gnome*/schemas/gschemas.compiled
|
desktop-extensions/gnome*/schemas/gschemas.compiled
|
||||||
desktop-extensions/gnome*/*.zip
|
desktop-extensions/gnome*/*.zip
|
||||||
|
|
||||||
|
# agents and reference
|
||||||
|
CLAUDE.md
|
||||||
|
AGENTS.md
|
||||||
|
/reference
|
||||||
|
|||||||
@@ -90,7 +90,7 @@ pages:
|
|||||||
- rm -rf public
|
- rm -rf public
|
||||||
- mkdir public
|
- mkdir public
|
||||||
- cp -R ci-target/doc/* public
|
- cp -R ci-target/doc/* public
|
||||||
- cp extra/index.html public
|
- if [ -f extra/index.html ]; then cp extra/index.html public; else echo "no extra/index.html to copy"; fi
|
||||||
artifacts:
|
artifacts:
|
||||||
paths:
|
paths:
|
||||||
- public
|
- public
|
||||||
|
|||||||
25
CHANGELOG.md
25
CHANGELOG.md
@@ -1,10 +1,31 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
## [Unreleased]
|
## [6.3.2]
|
||||||
|
|
||||||
|
### Changes
|
||||||
|
- Improve the notification area, @shevchenko0013 strikes again!
|
||||||
|
- Improve firmware attributes handling
|
||||||
|
|
||||||
|
## [6.3.1]
|
||||||
|
|
||||||
|
### Changes
|
||||||
|
- Removed a lighting mode that is unavailable in windows to G835L: thanks to @shevchenko0013 again!
|
||||||
|
- Added translations for Ukranian language, thanks @shevchenko0013!
|
||||||
|
- Added LEDs definition for G615LR, thanks @btnrv
|
||||||
|
- Fix improper usage of Quiet when only LowPower is available
|
||||||
|
|
||||||
|
## [6.3.0]
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- Added support for TUF keyboard powerstate control
|
||||||
|
- Improved AniMe Matrix support thanks to @Seom1177 !
|
||||||
|
- Fixed a bug with one-shot battery change, thanks @bitr8 !
|
||||||
|
- Changed the CLI interface of asusctl to be less confusing
|
||||||
|
- Added support for G835L, thanks to @shevchenko0013 !
|
||||||
|
|
||||||
## [6.2.0]
|
## [6.2.0]
|
||||||
|
|
||||||
## Changed
|
### Changed
|
||||||
- Added aura support for FX607V: thanks @jomp16
|
- Added aura support for FX607V: thanks @jomp16
|
||||||
- Added testing support for G835LW
|
- Added testing support for G835LW
|
||||||
- Added support for GU605C models slash lighting: thanks @Otters
|
- Added support for GU605C models slash lighting: thanks @Otters
|
||||||
|
|||||||
673
Cargo.lock
generated
673
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
@@ -1,9 +1,12 @@
|
|||||||
[workspace.package]
|
[workspace.package]
|
||||||
version = "6.2.0"
|
version = "6.3.1"
|
||||||
rust-version = "1.82"
|
rust-version = "1.82"
|
||||||
license = "MPL-2.0"
|
license = "MPL-2.0"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
authors = ["Luke <luke@ljones.dev>"]
|
authors = [
|
||||||
|
"Luke <luke@ljones.dev>",
|
||||||
|
"Denis Benato <benato.denis96@gmail.com>"
|
||||||
|
]
|
||||||
repository = "https://gitlab.com/asus-linux/asusctl"
|
repository = "https://gitlab.com/asus-linux/asusctl"
|
||||||
homepage = "https://gitlab.com/asus-linux/asusctl"
|
homepage = "https://gitlab.com/asus-linux/asusctl"
|
||||||
description = "Laptop feature control for ASUS ROG laptops and others"
|
description = "Laptop feature control for ASUS ROG laptops and others"
|
||||||
@@ -44,7 +47,7 @@ smol = "^2.0"
|
|||||||
mio = "0.8.11"
|
mio = "0.8.11"
|
||||||
|
|
||||||
futures-util = "0.3.31"
|
futures-util = "0.3.31"
|
||||||
zbus = "5.5.0"
|
zbus = "5.13.1"
|
||||||
logind-zbus = { version = "5.2.0" } #, default-features = false, features = ["non_blocking"] }
|
logind-zbus = { version = "5.2.0" } #, default-features = false, features = ["non_blocking"] }
|
||||||
|
|
||||||
serde = { version = "^1.0", features = ["serde_derive"] }
|
serde = { version = "^1.0", features = ["serde_derive"] }
|
||||||
|
|||||||
@@ -13,9 +13,7 @@ Now includes a GUI, `rog-control-center`.
|
|||||||
|
|
||||||
Due to on-going driver work the minimum suggested kernel version is always **the latest*, as improvements and fixes are continuous.
|
Due to on-going driver work the minimum suggested kernel version is always **the latest*, as improvements and fixes are continuous.
|
||||||
|
|
||||||
Support for some new features is not avilable unless you run a patched kernel with the work I am doing [in this github repo](https://github.com/flukejones/linux/tree/wip/ally-6.13). Use the linked branch, or `wip/ally-6.12`. Everything that is done here is upstreamed eventually (a long process).
|
Support for TDP is tied to the new asus-armoury driver: available mainline since linux 6.19: everything older is not supported.
|
||||||
|
|
||||||
Z13 devices will need [these](https://lore.kernel.org/linux-input/20240416090402.31057-1-luke@ljones.dev/T/#t)
|
|
||||||
|
|
||||||
## X11 support
|
## X11 support
|
||||||
|
|
||||||
@@ -180,3 +178,7 @@ Reference to any ASUS products, services, processes, or other information and/or
|
|||||||
The use of ROG and ASUS trademarks within this website and associated tools and libraries is only to provide a recognisable identifier to users to enable them to associate that these tools will work with ASUS ROG laptops.
|
The use of ROG and ASUS trademarks within this website and associated tools and libraries is only to provide a recognisable identifier to users to enable them to associate that these tools will work with ASUS ROG laptops.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## AI Disaclaimer
|
||||||
|
|
||||||
|
Portions of this code have been written by various AI tools and reviewed by the maintainer exaclty as with every other contribution.
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ env_logger.workspace = true
|
|||||||
ron.workspace = true
|
ron.workspace = true
|
||||||
gumdrop.workspace = true
|
gumdrop.workspace = true
|
||||||
zbus.workspace = true
|
zbus.workspace = true
|
||||||
|
argh = "0.1"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
rog_dbus = { path = "../rog-dbus" }
|
rog_dbus = { path = "../rog-dbus" }
|
||||||
|
|||||||
547
asusctl/examples/anime-led-scan.rs
Normal file
547
asusctl/examples/anime-led-scan.rs
Normal file
@@ -0,0 +1,547 @@
|
|||||||
|
//! LED scanning tool for discovering AniMe Matrix buffer-to-LED mappings.
|
||||||
|
//!
|
||||||
|
//! This tool lights up one buffer index at a time, allowing you to observe
|
||||||
|
//! which physical LED corresponds to each buffer position. This is essential
|
||||||
|
//! for mapping new device types like G835L where the exact layout is unknown.
|
||||||
|
//!
|
||||||
|
//! You might want to use it slowly, as it sometimes doesn't work properly.
|
||||||
|
//! Maybe there's better ways to make this reliable but for now it works for my use case.
|
||||||
|
//!
|
||||||
|
//! # Usage
|
||||||
|
//! ```
|
||||||
|
//! cargo run --example anime-led-scan -- [options]
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! # Controls
|
||||||
|
//! - `n` or `Enter`: Next index
|
||||||
|
//! - `p` or `Backspace`: Previous index
|
||||||
|
//! - `j` followed by number: Jump to specific index
|
||||||
|
//! - `+` / `-`: Adjust step size (default 1)
|
||||||
|
//! - `s`: Save current index to notes file
|
||||||
|
//! - `r`: Mark current index as row start
|
||||||
|
//! - `q` or `Ctrl+C`: Quit
|
||||||
|
//!
|
||||||
|
//! # Output
|
||||||
|
//! Creates a `led-scan-notes.txt` file with recorded observations.
|
||||||
|
|
||||||
|
use std::env;
|
||||||
|
use std::fs::OpenOptions;
|
||||||
|
use std::io::{self, BufRead, Write};
|
||||||
|
|
||||||
|
use rog_anime::usb::{get_anime_type, Brightness};
|
||||||
|
use rog_anime::{AnimeDataBuffer, AnimeType};
|
||||||
|
use rog_dbus::zbus_anime::AnimeProxyBlocking;
|
||||||
|
use zbus::blocking::Connection;
|
||||||
|
|
||||||
|
/// Saved device state for restoration on exit
|
||||||
|
struct SavedState {
|
||||||
|
builtins_enabled: bool,
|
||||||
|
brightness: Brightness,
|
||||||
|
display_enabled: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_help(scan_len: usize, buffer_len: usize) {
|
||||||
|
println!("\n=== LED Scan Tool ===");
|
||||||
|
println!(
|
||||||
|
"Scan range: 0-{} (buffer size: {})",
|
||||||
|
scan_len - 1,
|
||||||
|
buffer_len
|
||||||
|
);
|
||||||
|
println!("Commands:");
|
||||||
|
println!(" n, Enter - Next index");
|
||||||
|
println!(" p, Backspace - Previous index");
|
||||||
|
println!(" j <num> - Jump to index");
|
||||||
|
println!(" + / - - Increase/decrease step size");
|
||||||
|
println!(" s - Save note for current index");
|
||||||
|
println!(" r - Mark as row start");
|
||||||
|
println!(" a - Auto-scan (runs through all indices)");
|
||||||
|
println!(" f - Fill all buffer bytes");
|
||||||
|
println!(" f <start> <end> - Fill range (inclusive)");
|
||||||
|
println!(" p1/p2/p3 - Fill pane 1/2/3 only (each is 627 bytes)");
|
||||||
|
println!(" hold - Hold current LED (press Enter to release)");
|
||||||
|
println!(" hold <s> <e> - Hold range (press Enter to release)");
|
||||||
|
println!(" c - Clear display");
|
||||||
|
println!(" row - Step through rows (G835L, provisional)");
|
||||||
|
println!(" row <n> - Show specific row (G835L, provisional)");
|
||||||
|
println!(" allrows - Light all rows sequentially (G835L)");
|
||||||
|
println!(" rowmap - Print the full row mapping (G835L)");
|
||||||
|
println!(" h - Show this help");
|
||||||
|
println!(" q - Quit and restore state");
|
||||||
|
println!();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn save_note(index: usize, note: &str) -> io::Result<()> {
|
||||||
|
let mut file = OpenOptions::new()
|
||||||
|
.create(true)
|
||||||
|
.append(true)
|
||||||
|
.open("led-scan-notes.txt")?;
|
||||||
|
writeln!(file, "Index {}: {}", index, note)?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn write_single_led(
|
||||||
|
proxy: &AnimeProxyBlocking,
|
||||||
|
anime_type: AnimeType,
|
||||||
|
index: usize,
|
||||||
|
brightness: u8,
|
||||||
|
) {
|
||||||
|
let mut buffer = AnimeDataBuffer::new(anime_type);
|
||||||
|
let data = buffer.data_mut();
|
||||||
|
if index < data.len() {
|
||||||
|
data[index] = brightness;
|
||||||
|
}
|
||||||
|
if let Err(e) = proxy.write(buffer) {
|
||||||
|
eprintln!("Error writing to device: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn clear_display(proxy: &AnimeProxyBlocking, anime_type: AnimeType) {
|
||||||
|
let buffer = AnimeDataBuffer::new(anime_type);
|
||||||
|
let _ = proxy.write(buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fill_display(proxy: &AnimeProxyBlocking, anime_type: AnimeType, brightness: u8) {
|
||||||
|
let mut buffer = AnimeDataBuffer::new(anime_type);
|
||||||
|
let data = buffer.data_mut();
|
||||||
|
for byte in data.iter_mut() {
|
||||||
|
*byte = brightness;
|
||||||
|
}
|
||||||
|
if let Err(e) = proxy.write(buffer) {
|
||||||
|
eprintln!("Error writing to device: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Fill a range of LEDs. Both start and end are INCLUSIVE.
|
||||||
|
fn fill_range(
|
||||||
|
proxy: &AnimeProxyBlocking,
|
||||||
|
anime_type: AnimeType,
|
||||||
|
start: usize,
|
||||||
|
end: usize,
|
||||||
|
brightness: u8,
|
||||||
|
) {
|
||||||
|
let mut buffer = AnimeDataBuffer::new(anime_type);
|
||||||
|
let data = buffer.data_mut();
|
||||||
|
for i in start..=end.min(data.len().saturating_sub(1)) {
|
||||||
|
data[i] = brightness;
|
||||||
|
}
|
||||||
|
if let Err(e) = proxy.write(buffer) {
|
||||||
|
eprintln!("Error writing to device: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fill_pane(proxy: &AnimeProxyBlocking, anime_type: AnimeType, pane: usize, brightness: u8) {
|
||||||
|
const PANE_LEN: usize = 627;
|
||||||
|
let start = pane * PANE_LEN;
|
||||||
|
let end = start + PANE_LEN - 1;
|
||||||
|
fill_range(proxy, anime_type, start, end, brightness);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// G835L row pattern (PROVISIONAL - needs hardware verification):
|
||||||
|
/// - Rows 0-1: 1 LED each
|
||||||
|
/// - Rows 2-3: 2 LEDs each
|
||||||
|
/// - ... (pairs of rows with same length)
|
||||||
|
/// - Rows 26-27: 14 LEDs each
|
||||||
|
/// - Rows 28+: 15 LEDs each (constant)
|
||||||
|
///
|
||||||
|
/// Returns (start_index, end_index_inclusive, row_length)
|
||||||
|
fn g835l_row_bounds(row: usize) -> (usize, usize, usize) {
|
||||||
|
let triangle_rows = 28;
|
||||||
|
let triangle_leds = 210;
|
||||||
|
|
||||||
|
if row < triangle_rows {
|
||||||
|
let length = row / 2 + 1;
|
||||||
|
let mut start = 0usize;
|
||||||
|
for r in 0..row {
|
||||||
|
start += r / 2 + 1;
|
||||||
|
}
|
||||||
|
(start, start + length - 1, length)
|
||||||
|
} else {
|
||||||
|
let rows_after_triangle = row - triangle_rows;
|
||||||
|
let start = triangle_leds + rows_after_triangle * 15;
|
||||||
|
(start, start + 14, 15)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn g835l_total_rows() -> usize {
|
||||||
|
28 + 40
|
||||||
|
}
|
||||||
|
|
||||||
|
fn save_state(proxy: &AnimeProxyBlocking) -> SavedState {
|
||||||
|
SavedState {
|
||||||
|
builtins_enabled: proxy.builtins_enabled().unwrap_or(false),
|
||||||
|
brightness: proxy.brightness().unwrap_or(Brightness::Med),
|
||||||
|
display_enabled: proxy.enable_display().unwrap_or(true),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn restore_state(proxy: &AnimeProxyBlocking, state: &SavedState) {
|
||||||
|
let _ = proxy.set_builtins_enabled(state.builtins_enabled);
|
||||||
|
let _ = proxy.set_brightness(state.brightness);
|
||||||
|
let _ = proxy.set_enable_display(state.display_enabled);
|
||||||
|
let _ = proxy.run_main_loop(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let args: Vec<String> = env::args().collect();
|
||||||
|
|
||||||
|
let mut start_index = 0usize;
|
||||||
|
let mut brightness = 200u8;
|
||||||
|
let mut scan_limit: Option<usize> = None;
|
||||||
|
|
||||||
|
let mut i = 1;
|
||||||
|
while i < args.len() {
|
||||||
|
match args[i].as_str() {
|
||||||
|
"--start" | "-s" => {
|
||||||
|
if i + 1 < args.len() {
|
||||||
|
start_index = args[i + 1].parse().unwrap_or(0);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"--brightness" | "-b" => {
|
||||||
|
if i + 1 < args.len() {
|
||||||
|
brightness = args[i + 1].parse().unwrap_or(200);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"--limit" | "-l" => {
|
||||||
|
if i + 1 < args.len() {
|
||||||
|
scan_limit = args[i + 1].parse().ok();
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"--help" | "-h" => {
|
||||||
|
println!("LED Scan Tool for AniMe Matrix");
|
||||||
|
println!();
|
||||||
|
println!("Usage: anime-led-scan [options]");
|
||||||
|
println!();
|
||||||
|
println!("Options:");
|
||||||
|
println!(" -s, --start <N> Start at index N (default: 0)");
|
||||||
|
println!(" -b, --brightness <N> LED brightness 0-255 (default: 200)");
|
||||||
|
println!(" -l, --limit <N> Cap scan range to N indices (e.g. 810 for G835L)");
|
||||||
|
println!(" -h, --help Show this help");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
let conn = match Connection::system() {
|
||||||
|
Ok(c) => c,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Failed to connect to D-Bus: {}", e);
|
||||||
|
eprintln!("Make sure asusd is running.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let proxy = match AnimeProxyBlocking::new(&conn) {
|
||||||
|
Ok(p) => p,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Failed to create Anime proxy: {}", e);
|
||||||
|
eprintln!("Make sure asusd supports your device.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let anime_type = get_anime_type();
|
||||||
|
let buffer_len = anime_type.data_length();
|
||||||
|
let scan_len = scan_limit.unwrap_or(buffer_len).min(buffer_len);
|
||||||
|
|
||||||
|
println!("=== LED Scan Tool ===");
|
||||||
|
println!("Device type: {:?}", anime_type);
|
||||||
|
println!("Buffer length: {} bytes", buffer_len);
|
||||||
|
println!("Scan range: 0-{}", scan_len - 1);
|
||||||
|
println!("Brightness: {}", brightness);
|
||||||
|
println!();
|
||||||
|
|
||||||
|
// Save current state for restoration
|
||||||
|
let saved_state = save_state(&proxy);
|
||||||
|
println!("Saved device state for restoration on exit.");
|
||||||
|
|
||||||
|
// Stop system animations
|
||||||
|
if let Err(e) = proxy.run_main_loop(false) {
|
||||||
|
eprintln!("Warning: Could not stop main loop: {}", e);
|
||||||
|
}
|
||||||
|
println!("Stopped system animations.");
|
||||||
|
|
||||||
|
print_help(scan_len, buffer_len);
|
||||||
|
|
||||||
|
let mut current_index = start_index.min(scan_len - 1);
|
||||||
|
let mut step = 1usize;
|
||||||
|
|
||||||
|
write_single_led(&proxy, anime_type, current_index, brightness);
|
||||||
|
println!(">>> Index: {} (step: {})", current_index, step);
|
||||||
|
|
||||||
|
let stdin = io::stdin();
|
||||||
|
let mut input = String::new();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
input.clear();
|
||||||
|
print!("> ");
|
||||||
|
io::stdout().flush().unwrap();
|
||||||
|
|
||||||
|
if stdin.lock().read_line(&mut input).is_err() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
let cmd = input.trim();
|
||||||
|
|
||||||
|
match cmd {
|
||||||
|
"q" | "quit" | "exit" => {
|
||||||
|
clear_display(&proxy, anime_type);
|
||||||
|
restore_state(&proxy, &saved_state);
|
||||||
|
println!("Restored device state. Goodbye!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
"n" | "" => {
|
||||||
|
current_index = (current_index + step).min(scan_len - 1);
|
||||||
|
write_single_led(&proxy, anime_type, current_index, brightness);
|
||||||
|
println!(">>> Index: {} (step: {})", current_index, step);
|
||||||
|
}
|
||||||
|
"p" => {
|
||||||
|
current_index = current_index.saturating_sub(step);
|
||||||
|
write_single_led(&proxy, anime_type, current_index, brightness);
|
||||||
|
println!(">>> Index: {} (step: {})", current_index, step);
|
||||||
|
}
|
||||||
|
"+" => {
|
||||||
|
step = step.saturating_mul(2).max(1);
|
||||||
|
println!("Step size: {}", step);
|
||||||
|
}
|
||||||
|
"-" => {
|
||||||
|
step = step.saturating_div(2).max(1);
|
||||||
|
println!("Step size: {}", step);
|
||||||
|
}
|
||||||
|
"r" => {
|
||||||
|
if let Err(e) = save_note(current_index, "ROW START") {
|
||||||
|
eprintln!("Error saving note: {}", e);
|
||||||
|
} else {
|
||||||
|
println!("Saved: Index {} marked as ROW START", current_index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"h" | "help" | "?" => {
|
||||||
|
print_help(scan_len, buffer_len);
|
||||||
|
}
|
||||||
|
cmd if cmd.starts_with('j') => {
|
||||||
|
let num_str = cmd.trim_start_matches('j').trim();
|
||||||
|
if let Ok(idx) = num_str.parse::<usize>() {
|
||||||
|
if idx < scan_len {
|
||||||
|
current_index = idx;
|
||||||
|
write_single_led(&proxy, anime_type, current_index, brightness);
|
||||||
|
println!(">>> Index: {} (step: {})", current_index, step);
|
||||||
|
} else {
|
||||||
|
println!("Index {} out of range (max: {})", idx, scan_len - 1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println!("Usage: j <number>");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cmd if cmd.starts_with('s') && !cmd.starts_with("show") => {
|
||||||
|
let note = cmd.trim_start_matches('s').trim();
|
||||||
|
let note = if note.is_empty() { "observed" } else { note };
|
||||||
|
if let Err(e) = save_note(current_index, note) {
|
||||||
|
eprintln!("Error saving note: {}", e);
|
||||||
|
} else {
|
||||||
|
println!("Saved note for index {}", current_index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"a" => {
|
||||||
|
println!("Auto-scan mode (0 to {})...", scan_len - 1);
|
||||||
|
let delay = std::time::Duration::from_millis(10);
|
||||||
|
for idx in current_index..scan_len {
|
||||||
|
write_single_led(&proxy, anime_type, idx, brightness);
|
||||||
|
print!("\rIndex: {} / {} ", idx, scan_len - 1);
|
||||||
|
io::stdout().flush().unwrap();
|
||||||
|
std::thread::sleep(delay);
|
||||||
|
current_index = idx;
|
||||||
|
}
|
||||||
|
println!();
|
||||||
|
println!("Auto-scan complete. Current index: {}", current_index);
|
||||||
|
}
|
||||||
|
"c" => {
|
||||||
|
clear_display(&proxy, anime_type);
|
||||||
|
println!("Display cleared");
|
||||||
|
}
|
||||||
|
"f" => {
|
||||||
|
fill_display(&proxy, anime_type, brightness);
|
||||||
|
println!("All buffer bytes filled at brightness {}", brightness);
|
||||||
|
}
|
||||||
|
"p1" => {
|
||||||
|
fill_pane(&proxy, anime_type, 0, brightness);
|
||||||
|
println!("Pane 1 (indices 0-626) filled");
|
||||||
|
}
|
||||||
|
"p2" => {
|
||||||
|
fill_pane(&proxy, anime_type, 1, brightness);
|
||||||
|
println!("Pane 2 (indices 627-1253) filled");
|
||||||
|
}
|
||||||
|
"p3" => {
|
||||||
|
fill_pane(&proxy, anime_type, 2, brightness);
|
||||||
|
println!("Pane 3 (indices 1254-1880) filled");
|
||||||
|
}
|
||||||
|
cmd if cmd.starts_with("f ") => {
|
||||||
|
let parts: Vec<&str> = cmd.split_whitespace().collect();
|
||||||
|
if parts.len() == 3 {
|
||||||
|
if let (Ok(start), Ok(end)) =
|
||||||
|
(parts[1].parse::<usize>(), parts[2].parse::<usize>())
|
||||||
|
{
|
||||||
|
fill_range(&proxy, anime_type, start, end, brightness);
|
||||||
|
println!("Filled indices {} to {}", start, end);
|
||||||
|
} else {
|
||||||
|
println!("Usage: f <start> <end>");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println!("Usage: f <start> <end>");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"show" => {
|
||||||
|
write_single_led(&proxy, anime_type, current_index, brightness);
|
||||||
|
println!(">>> Index: {} (step: {})", current_index, step);
|
||||||
|
}
|
||||||
|
"row" => {
|
||||||
|
if anime_type != AnimeType::G835L {
|
||||||
|
println!("Warning: Row commands use G835L mapping (provisional). You can add to this code to support other types. `examples/anime-led-scan.rs[402:425]`");
|
||||||
|
}
|
||||||
|
println!("Row stepping mode. Press Enter for next row, 'q' to quit.");
|
||||||
|
let total = g835l_total_rows();
|
||||||
|
for row_num in 0..total {
|
||||||
|
let (start, end, len) = g835l_row_bounds(row_num);
|
||||||
|
if end >= scan_len {
|
||||||
|
println!("Row {} exceeds scan limit, stopping.", row_num);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
println!("Row {}: indices {}-{} ({} LEDs)", row_num, start, end, len);
|
||||||
|
fill_range(&proxy, anime_type, start, end, brightness);
|
||||||
|
input.clear();
|
||||||
|
print!("(Enter=next, q=quit) > ");
|
||||||
|
io::stdout().flush().unwrap();
|
||||||
|
if stdin.lock().read_line(&mut input).is_err() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if input.trim() == "q" {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
clear_display(&proxy, anime_type);
|
||||||
|
}
|
||||||
|
println!("Row stepping done.");
|
||||||
|
}
|
||||||
|
cmd if cmd.starts_with("row ") => {
|
||||||
|
if anime_type != AnimeType::G835L {
|
||||||
|
println!("Warning: Row commands use G835L mapping (provisional).");
|
||||||
|
}
|
||||||
|
let row_str = cmd.trim_start_matches("row ").trim();
|
||||||
|
if let Ok(row_num) = row_str.parse::<usize>() {
|
||||||
|
let total = g835l_total_rows();
|
||||||
|
if row_num < total {
|
||||||
|
let (start, end, len) = g835l_row_bounds(row_num);
|
||||||
|
if end < scan_len {
|
||||||
|
println!("Row {}: indices {}-{} ({} LEDs)", row_num, start, end, len);
|
||||||
|
fill_range(&proxy, anime_type, start, end, brightness);
|
||||||
|
} else {
|
||||||
|
println!("Row {} exceeds scan limit", row_num);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println!("Row {} out of range (max: {})", row_num, total - 1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println!("Usage: row <number>");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"allrows" => {
|
||||||
|
if anime_type != AnimeType::G835L {
|
||||||
|
println!("Warning: Row commands use G835L mapping (provisional).");
|
||||||
|
}
|
||||||
|
println!("Lighting all rows sequentially (200ms each)...");
|
||||||
|
let total = g835l_total_rows();
|
||||||
|
let delay = std::time::Duration::from_millis(200);
|
||||||
|
for row_num in 0..total {
|
||||||
|
let (start, end, len) = g835l_row_bounds(row_num);
|
||||||
|
if end >= scan_len {
|
||||||
|
println!("\nRow {} exceeds scan limit, stopping.", row_num);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
print!(
|
||||||
|
"\rRow {}/{}: indices {}-{} ({} LEDs) ",
|
||||||
|
row_num,
|
||||||
|
total - 1,
|
||||||
|
start,
|
||||||
|
end,
|
||||||
|
len
|
||||||
|
);
|
||||||
|
io::stdout().flush().unwrap();
|
||||||
|
fill_range(&proxy, anime_type, start, end, brightness);
|
||||||
|
std::thread::sleep(delay);
|
||||||
|
clear_display(&proxy, anime_type);
|
||||||
|
}
|
||||||
|
println!("\nDone.");
|
||||||
|
}
|
||||||
|
"rowmap" => {
|
||||||
|
if anime_type != AnimeType::G835L {
|
||||||
|
println!("Warning: Row map is for G835L (provisional).");
|
||||||
|
}
|
||||||
|
println!("G835L Row Map:");
|
||||||
|
let total = g835l_total_rows();
|
||||||
|
for row_num in 0..total {
|
||||||
|
let (start, end, len) = g835l_row_bounds(row_num);
|
||||||
|
let marker = if end >= scan_len {
|
||||||
|
" (exceeds limit)"
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
};
|
||||||
|
println!(
|
||||||
|
" Row {:2}: indices {:4}-{:4} ({:2} LEDs){}",
|
||||||
|
row_num, start, end, len, marker
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"hold" => {
|
||||||
|
// Single write, wait for Enter to release
|
||||||
|
println!("Holding index {}. Press Enter to release...", current_index);
|
||||||
|
write_single_led(&proxy, anime_type, current_index, brightness);
|
||||||
|
input.clear();
|
||||||
|
let _ = stdin.lock().read_line(&mut input);
|
||||||
|
clear_display(&proxy, anime_type);
|
||||||
|
println!("Released.");
|
||||||
|
}
|
||||||
|
cmd if cmd.starts_with("hold ") => {
|
||||||
|
let arg = cmd.trim_start_matches("hold ").trim();
|
||||||
|
let (start, end): (usize, usize) = match arg {
|
||||||
|
"p1" | "1" => (0, 626),
|
||||||
|
"p2" | "2" => (627, 1253),
|
||||||
|
_ => {
|
||||||
|
let parts: Vec<&str> = arg.split_whitespace().collect();
|
||||||
|
if parts.len() == 2 {
|
||||||
|
if let (Ok(s), Ok(e)) = (parts[0].parse(), parts[1].parse()) {
|
||||||
|
(s, e)
|
||||||
|
} else {
|
||||||
|
println!("Usage: hold p1, hold p2, or hold <start> <end>");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println!("Usage: hold p1, hold p2, or hold <start> <end>");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
println!("Holding range {}-{}. Press Enter to release...", start, end);
|
||||||
|
fill_range(&proxy, anime_type, start, end, brightness);
|
||||||
|
input.clear();
|
||||||
|
let _ = stdin.lock().read_line(&mut input);
|
||||||
|
clear_display(&proxy, anime_type);
|
||||||
|
println!("Released.");
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
if let Ok(idx) = cmd.parse::<usize>() {
|
||||||
|
if idx < scan_len {
|
||||||
|
current_index = idx;
|
||||||
|
write_single_led(&proxy, anime_type, current_index, brightness);
|
||||||
|
println!(">>> Index: {} (step: {})", current_index, step);
|
||||||
|
} else {
|
||||||
|
println!("Index {} out of range (max: {})", idx, scan_len - 1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println!("Unknown command: '{}'. Type 'h' for help.", cmd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,154 +1,151 @@
|
|||||||
use gumdrop::Options;
|
use argh::FromArgs;
|
||||||
use rog_anime::usb::{AnimAwake, AnimBooting, AnimShutdown, AnimSleeping, Brightness};
|
use rog_anime::usb::{AnimAwake, AnimBooting, AnimShutdown, AnimSleeping};
|
||||||
use rog_anime::AnimeType;
|
use rog_anime::AnimeType;
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "anime", description = "anime commands")]
|
||||||
pub struct AnimeCommand {
|
pub struct AnimeCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "override the display type")]
|
||||||
pub help: bool,
|
|
||||||
#[options(meta = "", help = "override the display type")]
|
|
||||||
pub override_type: Option<AnimeType>,
|
pub override_type: Option<AnimeType>,
|
||||||
#[options(meta = "", help = "enable/disable the display")]
|
#[argh(option, description = "enable/disable the display")]
|
||||||
pub enable_display: Option<bool>,
|
pub enable_display: Option<bool>,
|
||||||
#[options(meta = "", help = "enable/disable the builtin run/powersave animation")]
|
#[argh(
|
||||||
pub enable_powersave_anim: Option<bool>,
|
option,
|
||||||
#[options(
|
description = "enable/disable the builtin run/powersave animation"
|
||||||
meta = "",
|
|
||||||
help = "set global base brightness value <Off, Low, Med, High>"
|
|
||||||
)]
|
)]
|
||||||
pub brightness: Option<Brightness>,
|
pub enable_powersave_anim: Option<bool>,
|
||||||
#[options(help = "clear the display")]
|
#[argh(
|
||||||
|
option,
|
||||||
|
description = "set global base brightness value <off, low, med, high>"
|
||||||
|
)]
|
||||||
|
pub brightness: Option<rog_anime::usb::Brightness>,
|
||||||
|
#[argh(switch, description = "clear the display")]
|
||||||
pub clear: bool,
|
pub clear: bool,
|
||||||
#[options(
|
#[argh(
|
||||||
no_short,
|
option,
|
||||||
meta = "",
|
description = "turn the anime off when external power is unplugged"
|
||||||
help = "turn the anime off when external power is unplugged"
|
|
||||||
)]
|
)]
|
||||||
pub off_when_unplugged: Option<bool>,
|
pub off_when_unplugged: Option<bool>,
|
||||||
#[options(
|
#[argh(option, description = "turn the anime off when the laptop suspends")]
|
||||||
no_short,
|
|
||||||
meta = "",
|
|
||||||
help = "turn the anime off when the laptop suspends"
|
|
||||||
)]
|
|
||||||
pub off_when_suspended: Option<bool>,
|
pub off_when_suspended: Option<bool>,
|
||||||
#[options(
|
#[argh(option, description = "turn the anime off when the lid is closed")]
|
||||||
no_short,
|
|
||||||
meta = "",
|
|
||||||
help = "turn the anime off when the lid is closed"
|
|
||||||
)]
|
|
||||||
pub off_when_lid_closed: Option<bool>,
|
pub off_when_lid_closed: Option<bool>,
|
||||||
#[options(no_short, meta = "", help = "Off with his head!!!")]
|
#[argh(option, description = "off with his head!!!")]
|
||||||
pub off_with_his_head: Option<bool>,
|
pub off_with_his_head: Option<bool>,
|
||||||
#[options(command)]
|
#[argh(subcommand)]
|
||||||
pub command: Option<AnimeActions>,
|
pub command: Option<AnimeActions>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
/// Anime subcommands (image, gif, builtins, etc.)
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand)]
|
||||||
pub enum AnimeActions {
|
pub enum AnimeActions {
|
||||||
#[options(help = "display a PNG image")]
|
|
||||||
Image(AnimeImage),
|
Image(AnimeImage),
|
||||||
#[options(help = "display a diagonal/pixel-perfect PNG")]
|
|
||||||
PixelImage(AnimeImageDiagonal),
|
PixelImage(AnimeImageDiagonal),
|
||||||
#[options(help = "display an animated GIF")]
|
|
||||||
Gif(AnimeGif),
|
Gif(AnimeGif),
|
||||||
#[options(help = "display an animated diagonal/pixel-perfect GIF")]
|
|
||||||
PixelGif(AnimeGifDiagonal),
|
PixelGif(AnimeGifDiagonal),
|
||||||
#[options(help = "change which builtin animations are shown")]
|
|
||||||
SetBuiltins(Builtins),
|
SetBuiltins(Builtins),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "set-builtins",
|
||||||
|
description = "change which builtin animations are shown"
|
||||||
|
)]
|
||||||
pub struct Builtins {
|
pub struct Builtins {
|
||||||
#[options(help = "print help message")]
|
#[argh(
|
||||||
pub help: bool,
|
option,
|
||||||
#[options(
|
description = "default is used if unspecified, <default:GlitchConstruction, StaticEmergence>"
|
||||||
meta = "",
|
|
||||||
help = "Default is used if unspecified, <default:GlitchConstruction, StaticEmergence>"
|
|
||||||
)]
|
)]
|
||||||
pub boot: AnimBooting,
|
pub boot: AnimBooting,
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "Default is used if unspecified, <default:BinaryBannerScroll, RogLogoGlitch>"
|
description = "default is used if unspecified, <default:BinaryBannerScroll, RogLogoGlitch>"
|
||||||
)]
|
)]
|
||||||
pub awake: AnimAwake,
|
pub awake: AnimAwake,
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "Default is used if unspecified, <default:BannerSwipe, Starfield>"
|
description = "default is used if unspecified, <default:BannerSwipe, Starfield>"
|
||||||
)]
|
)]
|
||||||
pub sleep: AnimSleeping,
|
pub sleep: AnimSleeping,
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "Default is used if unspecified, <default:GlitchOut, SeeYa>"
|
description = "default is used if unspecified, <default:GlitchOut, SeeYa>"
|
||||||
)]
|
)]
|
||||||
pub shutdown: AnimShutdown,
|
pub shutdown: AnimShutdown,
|
||||||
#[options(meta = "", help = "set/apply the animations <true/false>")]
|
#[argh(option, description = "set/apply the animations <true/false>")]
|
||||||
pub set: Option<bool>,
|
pub set: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "image", description = "display a PNG image")]
|
||||||
pub struct AnimeImage {
|
pub struct AnimeImage {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "full path to the png to display")]
|
||||||
pub help: bool,
|
|
||||||
#[options(meta = "", help = "full path to the png to display")]
|
|
||||||
pub path: String,
|
pub path: String,
|
||||||
#[options(meta = "", default = "1.0", help = "scale 1.0 == normal")]
|
#[argh(option, default = "1.0", description = "scale 1.0 == normal")]
|
||||||
pub scale: f32,
|
pub scale: f32,
|
||||||
#[options(meta = "", default = "0.0", help = "x position (float)")]
|
#[argh(option, default = "0.0", description = "x position (float)")]
|
||||||
pub x_pos: f32,
|
pub x_pos: f32,
|
||||||
#[options(meta = "", default = "0.0", help = "y position (float)")]
|
#[argh(option, default = "0.0", description = "y position (float)")]
|
||||||
pub y_pos: f32,
|
pub y_pos: f32,
|
||||||
#[options(meta = "", default = "0.0", help = "the angle in radians")]
|
#[argh(option, default = "0.0", description = "the angle in radians")]
|
||||||
pub angle: f32,
|
pub angle: f32,
|
||||||
#[options(meta = "", default = "1.0", help = "brightness 0.0-1.0")]
|
#[argh(option, default = "1.0", description = "brightness 0.0-1.0")]
|
||||||
pub bright: f32,
|
pub bright: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "pixel-image",
|
||||||
|
description = "display a diagonal/pixel-perfect PNG"
|
||||||
|
)]
|
||||||
pub struct AnimeImageDiagonal {
|
pub struct AnimeImageDiagonal {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "full path to the png to display")]
|
||||||
pub help: bool,
|
|
||||||
#[options(meta = "", help = "full path to the png to display")]
|
|
||||||
pub path: String,
|
pub path: String,
|
||||||
#[options(meta = "", default = "1.0", help = "brightness 0.0-1.0")]
|
#[argh(option, default = "1.0", description = "brightness 0.0-1.0")]
|
||||||
pub bright: f32,
|
pub bright: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "gif", description = "display an animated GIF")]
|
||||||
pub struct AnimeGif {
|
pub struct AnimeGif {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "full path to the gif to display")]
|
||||||
pub help: bool,
|
|
||||||
#[options(meta = "", help = "full path to the png to display")]
|
|
||||||
pub path: String,
|
pub path: String,
|
||||||
#[options(meta = "", default = "1.0", help = "scale 1.0 == normal")]
|
#[argh(option, default = "1.0", description = "scale 1.0 == normal")]
|
||||||
pub scale: f32,
|
pub scale: f32,
|
||||||
#[options(meta = "", default = "0.0", help = "x position (float)")]
|
#[argh(option, default = "0.0", description = "x position (float)")]
|
||||||
pub x_pos: f32,
|
pub x_pos: f32,
|
||||||
#[options(meta = "", default = "0.0", help = "y position (float)")]
|
#[argh(option, default = "0.0", description = "y position (float)")]
|
||||||
pub y_pos: f32,
|
pub y_pos: f32,
|
||||||
#[options(meta = "", default = "0.0", help = "the angle in radians")]
|
#[argh(option, default = "0.0", description = "the angle in radians")]
|
||||||
pub angle: f32,
|
pub angle: f32,
|
||||||
#[options(meta = "", default = "1.0", help = "brightness 0.0-1.0")]
|
#[argh(option, default = "1.0", description = "brightness 0.0-1.0")]
|
||||||
pub bright: f32,
|
pub bright: f32,
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
default = "1",
|
default = "0",
|
||||||
help = "how many loops to play - 0 is infinite"
|
description = "how many loops to play - 0 is infinite"
|
||||||
)]
|
)]
|
||||||
pub loops: u32,
|
pub loops: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "pixel-gif",
|
||||||
|
description = "display an animated diagonal/pixel-perfect GIF"
|
||||||
|
)]
|
||||||
pub struct AnimeGifDiagonal {
|
pub struct AnimeGifDiagonal {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "full path to the gif to display")]
|
||||||
pub help: bool,
|
|
||||||
#[options(meta = "", help = "full path to the png to display")]
|
|
||||||
pub path: String,
|
pub path: String,
|
||||||
#[options(meta = "", default = "1.0", help = "brightness 0.0-1.0")]
|
#[argh(option, default = "1.0", description = "brightness 0.0-1.0")]
|
||||||
pub bright: f32,
|
pub bright: f32,
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
default = "1",
|
default = "0",
|
||||||
help = "how many loops to play - 0 is infinite"
|
description = "how many loops to play - 0 is infinite"
|
||||||
)]
|
)]
|
||||||
pub loops: u32,
|
pub loops: u32,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,68 +1,67 @@
|
|||||||
|
use std::fmt;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use gumdrop::Options;
|
use argh::FromArgs;
|
||||||
use rog_aura::error::Error;
|
use rog_aura::error::Error;
|
||||||
use rog_aura::{AuraEffect, AuraModeNum, AuraZone, Colour, Direction, Speed};
|
use rog_aura::{AuraEffect, AuraModeNum, AuraZone, Colour, Direction, Speed};
|
||||||
|
|
||||||
#[derive(Options, Debug)]
|
#[derive(FromArgs, Debug, Clone)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "aura-power-old",
|
||||||
|
description = "aura power (old ROGs and TUF laptops)"
|
||||||
|
)]
|
||||||
pub struct LedPowerCommand1 {
|
pub struct LedPowerCommand1 {
|
||||||
#[options(help = "print help message")]
|
#[argh(
|
||||||
pub help: bool,
|
option,
|
||||||
#[options(meta = "", help = "Control if LEDs enabled while awake <true/false>")]
|
description = "control if LEDs enabled while awake <true/false>"
|
||||||
|
)]
|
||||||
pub awake: Option<bool>,
|
pub awake: Option<bool>,
|
||||||
#[options(help = "Use with awake option, if excluded defaults to false")]
|
|
||||||
|
#[argh(
|
||||||
|
switch,
|
||||||
|
description = "use with awake option; if excluded defaults to false"
|
||||||
|
)]
|
||||||
pub keyboard: bool,
|
pub keyboard: bool,
|
||||||
#[options(help = "Use with awake option, if excluded defaults to false")]
|
|
||||||
|
#[argh(
|
||||||
|
switch,
|
||||||
|
description = "use with awake option; if excluded defaults to false"
|
||||||
|
)]
|
||||||
pub lightbar: bool,
|
pub lightbar: bool,
|
||||||
#[options(meta = "", help = "Control boot animations <true/false>")]
|
|
||||||
|
#[argh(option, description = "control boot animations <true/false>")]
|
||||||
pub boot: Option<bool>,
|
pub boot: Option<bool>,
|
||||||
#[options(meta = "", help = "Control suspend animations <true/false>")]
|
|
||||||
|
#[argh(option, description = "control suspend animations <true/false>")]
|
||||||
pub sleep: Option<bool>,
|
pub sleep: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options, Debug)]
|
#[derive(FromArgs, Debug, Clone)]
|
||||||
|
#[argh(subcommand, name = "aura-power", description = "aura power")]
|
||||||
pub struct LedPowerCommand2 {
|
pub struct LedPowerCommand2 {
|
||||||
#[options(help = "print help message")]
|
#[argh(subcommand)]
|
||||||
pub help: bool,
|
|
||||||
#[options(command)]
|
|
||||||
pub command: Option<SetAuraZoneEnabled>,
|
pub command: Option<SetAuraZoneEnabled>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options, Debug)]
|
/// Subcommands to enable/disable specific aura zones
|
||||||
|
#[derive(FromArgs, Debug, Clone)]
|
||||||
|
#[argh(subcommand)]
|
||||||
pub enum SetAuraZoneEnabled {
|
pub enum SetAuraZoneEnabled {
|
||||||
/// Applies to both old and new models
|
Keyboard(KeyboardPower),
|
||||||
#[options(help = "")]
|
Logo(LogoPower),
|
||||||
Keyboard(AuraPowerStates),
|
Lightbar(LightbarPower),
|
||||||
#[options(help = "")]
|
Lid(LidPower),
|
||||||
Logo(AuraPowerStates),
|
RearGlow(RearGlowPower),
|
||||||
#[options(help = "")]
|
Ally(AllyPower),
|
||||||
Lightbar(AuraPowerStates),
|
|
||||||
#[options(help = "")]
|
|
||||||
Lid(AuraPowerStates),
|
|
||||||
#[options(help = "")]
|
|
||||||
RearGlow(AuraPowerStates),
|
|
||||||
#[options(help = "")]
|
|
||||||
Ally(AuraPowerStates),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Options)]
|
/// Keyboard brightness argument helper
|
||||||
pub struct AuraPowerStates {
|
#[derive(Debug, Clone)]
|
||||||
#[options(help = "print help message")]
|
|
||||||
pub help: bool,
|
|
||||||
#[options(help = "defaults to false if option unused")]
|
|
||||||
pub boot: bool,
|
|
||||||
#[options(help = "defaults to false if option unused")]
|
|
||||||
pub awake: bool,
|
|
||||||
#[options(help = "defaults to false if option unused")]
|
|
||||||
pub sleep: bool,
|
|
||||||
#[options(help = "defaults to false if option unused")]
|
|
||||||
pub shutdown: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Options)]
|
|
||||||
pub struct LedBrightness {
|
pub struct LedBrightness {
|
||||||
level: Option<u8>,
|
level: Option<u8>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LedBrightness {
|
impl LedBrightness {
|
||||||
pub fn new(level: Option<u8>) -> Self {
|
pub fn new(level: Option<u8>) -> Self {
|
||||||
LedBrightness { level }
|
LedBrightness { level }
|
||||||
@@ -72,176 +71,302 @@ impl LedBrightness {
|
|||||||
self.level
|
self.level
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FromStr for LedBrightness {
|
impl FromStr for LedBrightness {
|
||||||
type Err = Error;
|
type Err = Error;
|
||||||
|
|
||||||
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
let s = s.to_lowercase();
|
let s = s.to_lowercase();
|
||||||
match s.as_str() {
|
match s.as_str() {
|
||||||
"off" => Ok(LedBrightness { level: Some(0x00) }),
|
"off" => Ok(Self::new(Some(0x00))),
|
||||||
"low" => Ok(LedBrightness { level: Some(0x01) }),
|
"low" => Ok(Self::new(Some(0x01))),
|
||||||
"med" => Ok(LedBrightness { level: Some(0x02) }),
|
"med" => Ok(Self::new(Some(0x02))),
|
||||||
"high" => Ok(LedBrightness { level: Some(0x03) }),
|
"high" => Ok(Self::new(Some(0x03))),
|
||||||
_ => {
|
_ => Err(Error::ParseBrightness),
|
||||||
print!("Invalid argument, must be one of: off, low, med, high");
|
|
||||||
Err(Error::ParseBrightness)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[allow(clippy::to_string_trait_impl)]
|
|
||||||
impl ToString for LedBrightness {
|
impl fmt::Display for LedBrightness {
|
||||||
fn to_string(&self) -> String {
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
let s = match self.level {
|
let s = match self.level {
|
||||||
Some(0x00) => "low",
|
Some(0x00) => "off",
|
||||||
Some(0x01) => "med",
|
Some(0x01) => "low",
|
||||||
Some(0x02) => "high",
|
Some(0x02) => "med",
|
||||||
|
Some(0x03) => "high",
|
||||||
_ => "unknown",
|
_ => "unknown",
|
||||||
};
|
};
|
||||||
s.to_owned()
|
write!(f, "{}", s)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Options, Default)]
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "keyboard",
|
||||||
|
description = "set power states for keyboard zone"
|
||||||
|
)]
|
||||||
|
pub struct KeyboardPower {
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub boot: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub awake: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub sleep: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub shutdown: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "logo",
|
||||||
|
description = "set power states for logo zone"
|
||||||
|
)]
|
||||||
|
pub struct LogoPower {
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub boot: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub awake: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub sleep: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub shutdown: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "lightbar",
|
||||||
|
description = "set power states for lightbar zone"
|
||||||
|
)]
|
||||||
|
pub struct LightbarPower {
|
||||||
|
#[argh(switch, description = "enable power while device is booting")]
|
||||||
|
pub boot: bool,
|
||||||
|
#[argh(switch, description = "enable power while device is awake")]
|
||||||
|
pub awake: bool,
|
||||||
|
#[argh(switch, description = "enable power while device is sleeping")]
|
||||||
|
pub sleep: bool,
|
||||||
|
#[argh(
|
||||||
|
switch,
|
||||||
|
description = "enable power while device is shutting down or hibernating"
|
||||||
|
)]
|
||||||
|
pub shutdown: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "lid",
|
||||||
|
description = "set power states for lid zone"
|
||||||
|
)]
|
||||||
|
pub struct LidPower {
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub boot: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub awake: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub sleep: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub shutdown: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "rear-glow",
|
||||||
|
description = "set power states for rear glow zone"
|
||||||
|
)]
|
||||||
|
pub struct RearGlowPower {
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub boot: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub awake: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub sleep: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub shutdown: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "ally",
|
||||||
|
description = "set power states for ally zone"
|
||||||
|
)]
|
||||||
|
pub struct AllyPower {
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub boot: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub awake: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub sleep: bool,
|
||||||
|
#[argh(switch, description = "defaults to false if option unused")]
|
||||||
|
pub shutdown: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Single speed-based effect
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "rainbow-cycle",
|
||||||
|
description = "single speed-based effect"
|
||||||
|
)]
|
||||||
pub struct SingleSpeed {
|
pub struct SingleSpeed {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "set the speed: low, med, high")]
|
||||||
help: bool,
|
|
||||||
#[options(no_long, meta = "WORD", help = "set the speed: low, med, high")]
|
|
||||||
pub speed: Speed,
|
pub speed: Speed,
|
||||||
#[options(
|
|
||||||
no_long,
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "set the zone for this effect e.g, 0, 1, one, logo, lightbar-left"
|
default = "AuraZone::None",
|
||||||
|
description = "set the zone for this effect e.g. 0, 1, one, logo, lightbar-left"
|
||||||
)]
|
)]
|
||||||
pub zone: AuraZone,
|
pub zone: AuraZone,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Options, Default)]
|
/// Single speed effect with direction
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "rainbow-wave",
|
||||||
|
description = "single speed effect with direction"
|
||||||
|
)]
|
||||||
pub struct SingleSpeedDirection {
|
pub struct SingleSpeedDirection {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "set the direction: up, down, left, right")]
|
||||||
help: bool,
|
|
||||||
#[options(no_long, meta = "", help = "set the direction: up, down, left, right")]
|
|
||||||
pub direction: Direction,
|
pub direction: Direction,
|
||||||
#[options(no_long, meta = "", help = "set the speed: low, med, high")]
|
|
||||||
|
#[argh(option, description = "set the speed: low, med, high")]
|
||||||
pub speed: Speed,
|
pub speed: Speed,
|
||||||
#[options(
|
|
||||||
no_long,
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "set the zone for this effect e.g, 0, 1, one, logo, lightbar-left"
|
default = "AuraZone::None",
|
||||||
|
description = "set the zone for this effect e.g. 0, 1, one, logo, lightbar-left"
|
||||||
)]
|
)]
|
||||||
pub zone: AuraZone,
|
pub zone: AuraZone,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, Options)]
|
/// Static single-colour effect
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "static",
|
||||||
|
description = "static single-colour effect"
|
||||||
|
)]
|
||||||
pub struct SingleColour {
|
pub struct SingleColour {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, short = 'c', description = "set the RGB value e.g. ff00ff")]
|
||||||
help: bool,
|
|
||||||
#[options(no_long, meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
pub colour: Colour,
|
pub colour: Colour,
|
||||||
#[options(
|
|
||||||
no_long,
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "set the zone for this effect e.g, 0, 1, one, logo, lightbar-left"
|
default = "AuraZone::None",
|
||||||
|
description = "set the zone for this effect e.g. 0, 1, one, logo, lightbar-left"
|
||||||
)]
|
)]
|
||||||
pub zone: AuraZone,
|
pub zone: AuraZone,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, Options)]
|
/// Single-colour effect with speed
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "highlight",
|
||||||
|
description = "single-colour effect with speed"
|
||||||
|
)]
|
||||||
pub struct SingleColourSpeed {
|
pub struct SingleColourSpeed {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, short = 'c', description = "set the RGB value e.g. ff00ff")]
|
||||||
help: bool,
|
|
||||||
#[options(no_long, meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
pub colour: Colour,
|
pub colour: Colour,
|
||||||
#[options(no_long, meta = "", help = "set the speed: low, med, high")]
|
|
||||||
|
#[argh(option, description = "set the speed: low, med, high")]
|
||||||
pub speed: Speed,
|
pub speed: Speed,
|
||||||
#[options(
|
|
||||||
no_long,
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "set the zone for this effect e.g, 0, 1, one, logo, lightbar-left"
|
default = "AuraZone::None",
|
||||||
|
description = "set the zone for this effect e.g. 0, 1, one, logo, lightbar-left"
|
||||||
)]
|
)]
|
||||||
pub zone: AuraZone,
|
pub zone: AuraZone,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Options, Default)]
|
/// Two-colour breathing effect
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "breathe",
|
||||||
|
description = "two-colour breathing effect"
|
||||||
|
)]
|
||||||
pub struct TwoColourSpeed {
|
pub struct TwoColourSpeed {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "set the first RGB value e.g. ff00ff")]
|
||||||
help: bool,
|
|
||||||
#[options(no_long, meta = "", help = "set the first RGB value e.g, ff00ff")]
|
|
||||||
pub colour: Colour,
|
pub colour: Colour,
|
||||||
#[options(no_long, meta = "", help = "set the second RGB value e.g, ff00ff")]
|
|
||||||
|
#[argh(option, description = "set the second RGB value e.g. ff00ff")]
|
||||||
pub colour2: Colour,
|
pub colour2: Colour,
|
||||||
#[options(no_long, meta = "", help = "set the speed: low, med, high")]
|
|
||||||
|
#[argh(option, description = "set the speed: low, med, high")]
|
||||||
pub speed: Speed,
|
pub speed: Speed,
|
||||||
#[options(
|
|
||||||
no_long,
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "set the zone for this effect e.g, 0, 1, one, logo, lightbar-left"
|
default = "AuraZone::None",
|
||||||
|
description = "set the zone for this effect e.g. 0, 1, one, logo, lightbar-left"
|
||||||
)]
|
)]
|
||||||
pub zone: AuraZone,
|
pub zone: AuraZone,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, Options)]
|
/// Multi-zone colour settings
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
|
#[argh(description = "multi-zone colour settings")]
|
||||||
pub struct MultiZone {
|
pub struct MultiZone {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, short = 'a', description = "set the RGB value e.g. ff00ff")]
|
||||||
help: bool,
|
|
||||||
#[options(short = "a", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
pub colour1: Colour,
|
pub colour1: Colour,
|
||||||
#[options(short = "b", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
|
#[argh(option, short = 'b', description = "set the RGB value e.g. ff00ff")]
|
||||||
pub colour2: Colour,
|
pub colour2: Colour,
|
||||||
#[options(short = "c", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
|
#[argh(option, short = 'c', description = "set the RGB value e.g. ff00ff")]
|
||||||
pub colour3: Colour,
|
pub colour3: Colour,
|
||||||
#[options(short = "d", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
|
#[argh(option, short = 'd', description = "set the RGB value e.g. ff00ff")]
|
||||||
pub colour4: Colour,
|
pub colour4: Colour,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, Options)]
|
/// Multi-colour with speed
|
||||||
|
#[derive(FromArgs, Debug, Clone, Default)]
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
|
#[argh(description = "multi-colour with speed")]
|
||||||
pub struct MultiColourSpeed {
|
pub struct MultiColourSpeed {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, short = 'a', description = "set the RGB value e.g. ff00ff")]
|
||||||
help: bool,
|
|
||||||
#[options(short = "a", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
pub colour1: Colour,
|
pub colour1: Colour,
|
||||||
#[options(short = "b", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
|
#[argh(option, short = 'b', description = "set the RGB value e.g. ff00ff")]
|
||||||
pub colour2: Colour,
|
pub colour2: Colour,
|
||||||
#[options(short = "c", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
|
#[argh(option, short = 'c', description = "set the RGB value e.g. ff00ff")]
|
||||||
pub colour3: Colour,
|
pub colour3: Colour,
|
||||||
#[options(short = "d", meta = "", help = "set the RGB value e.g, ff00ff")]
|
|
||||||
|
#[argh(option, short = 'd', description = "set the RGB value e.g. ff00ff")]
|
||||||
pub colour4: Colour,
|
pub colour4: Colour,
|
||||||
#[options(no_long, meta = "", help = "set the speed: low, med, high")]
|
|
||||||
|
#[argh(option, description = "set the speed: low, med, high")]
|
||||||
pub speed: Speed,
|
pub speed: Speed,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Byte value for setting the built-in mode.
|
/// Builtin aura effects
|
||||||
///
|
#[derive(FromArgs, Debug)]
|
||||||
/// Enum corresponds to the required integer value
|
#[argh(subcommand)]
|
||||||
// NOTE: The option names here must match those in rog-aura crate
|
|
||||||
#[derive(Options)]
|
|
||||||
pub enum SetAuraBuiltin {
|
pub enum SetAuraBuiltin {
|
||||||
#[options(help = "set a single static colour")]
|
Static(SingleColour), // 0
|
||||||
Static(SingleColour), // 0
|
Breathe(TwoColourSpeed), // 1
|
||||||
#[options(help = "pulse between one or two colours")]
|
RainbowCycle(SingleSpeed), // 2
|
||||||
Breathe(TwoColourSpeed), // 1
|
|
||||||
#[options(help = "strobe through all colours")]
|
|
||||||
RainbowCycle(SingleSpeed), // 2
|
|
||||||
#[options(help = "rainbow cycling in one of four directions")]
|
|
||||||
RainbowWave(SingleSpeedDirection), // 3
|
RainbowWave(SingleSpeedDirection), // 3
|
||||||
#[options(help = "rain pattern mimicking raindrops")]
|
Stars(TwoColourSpeed), // 4
|
||||||
Stars(TwoColourSpeed), // 4
|
Rain(SingleSpeed), // 5
|
||||||
#[options(help = "rain pattern of three preset colours")]
|
Highlight(SingleColourSpeed), // 6
|
||||||
Rain(SingleSpeed), // 5
|
Laser(SingleColourSpeed), // 7
|
||||||
#[options(help = "pressed keys are highlighted to fade")]
|
Ripple(SingleColourSpeed), // 8
|
||||||
Highlight(SingleColourSpeed), // 6
|
Pulse(SingleColour), // 10
|
||||||
#[options(help = "pressed keys generate horizontal laser")]
|
Comet(SingleColour), // 11
|
||||||
Laser(SingleColourSpeed), // 7
|
Flash(SingleColour), // 12
|
||||||
#[options(help = "pressed keys ripple outwards like a splash")]
|
|
||||||
Ripple(SingleColourSpeed), // 8
|
|
||||||
#[options(help = "set a rapid pulse")]
|
|
||||||
Pulse(SingleColour), // 10
|
|
||||||
#[options(help = "set a vertical line zooming from left")]
|
|
||||||
Comet(SingleColour), // 11
|
|
||||||
#[options(help = "set a wide vertical line zooming from left")]
|
|
||||||
Flash(SingleColour), // 12
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for SetAuraBuiltin {
|
impl Default for SetAuraBuiltin {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use gumdrop::Options;
|
use argh::FromArgs;
|
||||||
use rog_platform::platform::PlatformProfile;
|
use rog_platform::platform::PlatformProfile;
|
||||||
|
|
||||||
use crate::anime_cli::AnimeCommand;
|
use crate::anime_cli::AnimeCommand;
|
||||||
@@ -7,128 +7,308 @@ use crate::fan_curve_cli::FanCurveCommand;
|
|||||||
use crate::scsi_cli::ScsiCommand;
|
use crate::scsi_cli::ScsiCommand;
|
||||||
use crate::slash_cli::SlashCommand;
|
use crate::slash_cli::SlashCommand;
|
||||||
|
|
||||||
#[derive(Default, Options)]
|
#[derive(FromArgs, Default, Debug)]
|
||||||
|
/// asusctl command-line options
|
||||||
pub struct CliStart {
|
pub struct CliStart {
|
||||||
#[options(help_flag, help = "print help message")]
|
#[argh(subcommand)]
|
||||||
pub help: bool,
|
pub command: CliCommand,
|
||||||
#[options(help = "show program version number")]
|
|
||||||
pub version: bool,
|
|
||||||
#[options(help = "show supported functions of this laptop")]
|
|
||||||
pub show_supported: bool,
|
|
||||||
#[options(meta = "", help = "<off, low, med, high>")]
|
|
||||||
pub kbd_bright: Option<LedBrightness>,
|
|
||||||
#[options(help = "Toggle to next keyboard brightness")]
|
|
||||||
pub next_kbd_bright: bool,
|
|
||||||
#[options(help = "Toggle to previous keyboard brightness")]
|
|
||||||
pub prev_kbd_bright: bool,
|
|
||||||
#[options(meta = "", help = "Set your battery charge limit <20-100>")]
|
|
||||||
pub chg_limit: Option<u8>,
|
|
||||||
#[options(help = "Toggle one-shot battery charge to 100%")]
|
|
||||||
pub one_shot_chg: bool,
|
|
||||||
#[options(command)]
|
|
||||||
pub command: Option<CliCommand>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
/// Top-level subcommands for asusctl
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand)]
|
||||||
pub enum CliCommand {
|
pub enum CliCommand {
|
||||||
#[options(help = "Set the keyboard lighting from built-in modes")]
|
|
||||||
Aura(LedModeCommand),
|
Aura(LedModeCommand),
|
||||||
#[options(help = "Set the LED power states")]
|
|
||||||
AuraPowerOld(LedPowerCommand1),
|
AuraPowerOld(LedPowerCommand1),
|
||||||
#[options(help = "Set the LED power states")]
|
|
||||||
AuraPower(LedPowerCommand2),
|
AuraPower(LedPowerCommand2),
|
||||||
#[options(help = "Set or select platform_profile")]
|
Brightness(BrightnessCommand),
|
||||||
Profile(ProfileCommand),
|
Profile(ProfileCommand),
|
||||||
#[options(help = "Set, select, or modify fan curves if supported")]
|
|
||||||
FanCurve(FanCurveCommand),
|
FanCurve(FanCurveCommand),
|
||||||
#[options(help = "Set the graphics mode (obsoleted by supergfxctl)")]
|
|
||||||
Graphics(GraphicsCommand),
|
|
||||||
#[options(name = "anime", help = "Manage AniMe Matrix")]
|
|
||||||
Anime(AnimeCommand),
|
Anime(AnimeCommand),
|
||||||
#[options(name = "slash", help = "Manage Slash Ledbar")]
|
|
||||||
Slash(SlashCommand),
|
Slash(SlashCommand),
|
||||||
#[options(name = "scsi", help = "Manage SCSI external drive")]
|
|
||||||
Scsi(ScsiCommand),
|
Scsi(ScsiCommand),
|
||||||
#[options(
|
|
||||||
help = "Change platform settings. This is a new interface exposed by the asus-armoury \
|
|
||||||
driver, some of the settings will be the same as the older platform interface"
|
|
||||||
)]
|
|
||||||
Armoury(ArmouryCommand),
|
Armoury(ArmouryCommand),
|
||||||
#[options(name = "backlight", help = "Set screen backlight levels")]
|
|
||||||
Backlight(BacklightCommand),
|
Backlight(BacklightCommand),
|
||||||
|
Battery(BatteryCommand),
|
||||||
|
Info(InfoCommand),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Options)]
|
impl Default for CliCommand {
|
||||||
|
fn default() -> Self {
|
||||||
|
CliCommand::Info(InfoCommand::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "profile", description = "profile management")]
|
||||||
pub struct ProfileCommand {
|
pub struct ProfileCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(subcommand)]
|
||||||
pub help: bool,
|
pub command: ProfileSubCommand,
|
||||||
|
|
||||||
#[options(help = "toggle to next profile in list")]
|
|
||||||
pub next: bool,
|
|
||||||
|
|
||||||
#[options(help = "list available profiles")]
|
|
||||||
pub list: bool,
|
|
||||||
|
|
||||||
#[options(help = "get profile")]
|
|
||||||
pub profile_get: bool,
|
|
||||||
|
|
||||||
#[options(meta = "", help = "set the active profile")]
|
|
||||||
pub profile_set: Option<PlatformProfile>,
|
|
||||||
|
|
||||||
#[options(short = "a", meta = "", help = "set the profile to use on AC power")]
|
|
||||||
pub profile_set_ac: Option<PlatformProfile>,
|
|
||||||
|
|
||||||
#[options(
|
|
||||||
short = "b",
|
|
||||||
meta = "",
|
|
||||||
help = "set the profile to use on battery power"
|
|
||||||
)]
|
|
||||||
pub profile_set_bat: Option<PlatformProfile>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand)]
|
||||||
|
pub enum ProfileSubCommand {
|
||||||
|
Next(ProfileNextCommand),
|
||||||
|
List(ProfileListCommand),
|
||||||
|
Get(ProfileGetCommand),
|
||||||
|
Set(ProfileSetCommand),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for ProfileSubCommand {
|
||||||
|
fn default() -> Self {
|
||||||
|
ProfileSubCommand::List(ProfileListCommand::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "next",
|
||||||
|
description = "toggle to next profile in list"
|
||||||
|
)]
|
||||||
|
pub struct ProfileNextCommand {}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(subcommand, name = "list", description = "list available profiles")]
|
||||||
|
pub struct ProfileListCommand {}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(subcommand, name = "get", description = "get profile")]
|
||||||
|
pub struct ProfileGetCommand {}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(subcommand, name = "set", description = "set profile")]
|
||||||
|
pub struct ProfileSetCommand {
|
||||||
|
#[argh(positional, description = "profile to set")]
|
||||||
|
pub profile: PlatformProfile,
|
||||||
|
|
||||||
|
#[argh(
|
||||||
|
switch,
|
||||||
|
short = 'a',
|
||||||
|
description = "set the profile to use on AC power"
|
||||||
|
)]
|
||||||
|
pub ac: bool,
|
||||||
|
|
||||||
|
#[argh(
|
||||||
|
switch,
|
||||||
|
short = 'b',
|
||||||
|
description = "set the profile to use on battery power"
|
||||||
|
)]
|
||||||
|
pub battery: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(subcommand, name = "aura", description = "led mode commands")]
|
||||||
pub struct LedModeCommand {
|
pub struct LedModeCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(switch, description = "switch to next aura mode")]
|
||||||
pub help: bool,
|
|
||||||
#[options(help = "switch to next aura mode")]
|
|
||||||
pub next_mode: bool,
|
pub next_mode: bool,
|
||||||
#[options(help = "switch to previous aura mode")]
|
|
||||||
|
#[argh(switch, description = "switch to previous aura mode")]
|
||||||
pub prev_mode: bool,
|
pub prev_mode: bool,
|
||||||
#[options(command)]
|
|
||||||
|
#[argh(subcommand)]
|
||||||
pub command: Option<SetAuraBuiltin>,
|
pub command: Option<SetAuraBuiltin>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug, Default)]
|
||||||
pub struct GraphicsCommand {
|
#[argh(
|
||||||
#[options(help = "print help message")]
|
subcommand,
|
||||||
pub help: bool,
|
name = "armoury",
|
||||||
}
|
description = "armoury / firmware attributes"
|
||||||
|
)]
|
||||||
#[derive(Options, Debug)]
|
|
||||||
pub struct ArmouryCommand {
|
pub struct ArmouryCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(subcommand)]
|
||||||
pub help: bool,
|
pub command: ArmourySubCommand,
|
||||||
#[options(
|
|
||||||
free,
|
|
||||||
help = "append each value name followed by the value to set. `-1` sets to default"
|
|
||||||
)]
|
|
||||||
pub free: Vec<String>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand)]
|
||||||
|
pub enum ArmourySubCommand {
|
||||||
|
Set(ArmouryPropertySetCommand),
|
||||||
|
Get(ArmouryPropertyGetCommand),
|
||||||
|
List(ArmouryPropertyListCommand),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for ArmourySubCommand {
|
||||||
|
fn default() -> Self {
|
||||||
|
ArmourySubCommand::List(ArmouryPropertyListCommand::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "set",
|
||||||
|
description = "set an asus-armoury firmware-attribute"
|
||||||
|
)]
|
||||||
|
pub struct ArmouryPropertySetCommand {
|
||||||
|
#[argh(
|
||||||
|
positional,
|
||||||
|
description = "name of the attribute to set (see asus-armoury list for available properties)"
|
||||||
|
)]
|
||||||
|
pub property: String,
|
||||||
|
|
||||||
|
#[argh(positional, description = "value to set for the given attribute")]
|
||||||
|
pub value: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "list",
|
||||||
|
description = "list all firmware-attributes supported by asus-armoury"
|
||||||
|
)]
|
||||||
|
pub struct ArmouryPropertyListCommand {}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "get",
|
||||||
|
description = "get a firmware-attribute from asus-armoury"
|
||||||
|
)]
|
||||||
|
pub struct ArmouryPropertyGetCommand {
|
||||||
|
#[argh(
|
||||||
|
positional,
|
||||||
|
description = "name of the property to get (see asus-armoury list for available properties)"
|
||||||
|
)]
|
||||||
|
pub property: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(subcommand, name = "backlight", description = "backlight options")]
|
||||||
pub struct BacklightCommand {
|
pub struct BacklightCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "set screen brightness <0-100>")]
|
||||||
pub help: bool,
|
|
||||||
#[options(meta = "", help = "Set screen brightness <0-100>")]
|
|
||||||
pub screenpad_brightness: Option<i32>,
|
pub screenpad_brightness: Option<i32>,
|
||||||
#[options(
|
|
||||||
meta = "",
|
#[argh(
|
||||||
help = "Set screenpad gamma brightness 0.5 - 2.2, 1.0 == linear"
|
option,
|
||||||
|
description = "set screenpad gamma brightness 0.5 - 2.2, 1.0 == linear"
|
||||||
)]
|
)]
|
||||||
pub screenpad_gamma: Option<f32>,
|
pub screenpad_gamma: Option<f32>,
|
||||||
#[options(
|
|
||||||
meta = "",
|
#[argh(
|
||||||
help = "Set screenpad brightness to sync with primary display"
|
option,
|
||||||
|
description = "set screenpad brightness to sync with primary display"
|
||||||
)]
|
)]
|
||||||
pub sync_screenpad_brightness: Option<bool>,
|
pub sync_screenpad_brightness: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "battery", description = "battery options")]
|
||||||
|
pub struct BatteryCommand {
|
||||||
|
#[argh(subcommand)]
|
||||||
|
pub command: BatterySubCommand,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand)]
|
||||||
|
pub enum BatterySubCommand {
|
||||||
|
Limit(BatteryLimitCommand),
|
||||||
|
OneShot(BatteryOneShotCommand),
|
||||||
|
Info(BatteryInfoCommand),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for BatterySubCommand {
|
||||||
|
fn default() -> Self {
|
||||||
|
BatterySubCommand::OneShot(BatteryOneShotCommand::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "limit",
|
||||||
|
description = "set battery charge limit <20-100>"
|
||||||
|
)]
|
||||||
|
pub struct BatteryLimitCommand {
|
||||||
|
#[argh(positional, description = "charge limit percentage 20-100")]
|
||||||
|
pub limit: u8,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "oneshot",
|
||||||
|
description = "one-shot full charge (optional percent)"
|
||||||
|
)]
|
||||||
|
pub struct BatteryOneShotCommand {
|
||||||
|
#[argh(positional, description = "optional target percent (defaults to 100)")]
|
||||||
|
pub percent: Option<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "info",
|
||||||
|
description = "show current battery charge limit"
|
||||||
|
)]
|
||||||
|
pub struct BatteryInfoCommand {}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "info",
|
||||||
|
description = "show program version and system info"
|
||||||
|
)]
|
||||||
|
pub struct InfoCommand {
|
||||||
|
#[argh(switch, description = "show supported functions of this laptop")]
|
||||||
|
pub show_supported: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "leds", description = "keyboard brightness control")]
|
||||||
|
pub struct BrightnessCommand {
|
||||||
|
#[argh(subcommand)]
|
||||||
|
pub command: BrightnessSubCommand,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand)]
|
||||||
|
pub enum BrightnessSubCommand {
|
||||||
|
Set(BrightnessSetCommand),
|
||||||
|
Get(BrightnessGetCommand),
|
||||||
|
Next(BrightnessNextCommand),
|
||||||
|
Prev(BrightnessPrevCommand),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for BrightnessSubCommand {
|
||||||
|
fn default() -> Self {
|
||||||
|
BrightnessSubCommand::Get(BrightnessGetCommand::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "set",
|
||||||
|
description = "set keyboard brightness <off, low, med, high>"
|
||||||
|
)]
|
||||||
|
pub struct BrightnessSetCommand {
|
||||||
|
#[argh(positional, description = "brightness level: off, low, med, high")]
|
||||||
|
pub level: LedBrightness,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "get",
|
||||||
|
description = "get current keyboard brightness"
|
||||||
|
)]
|
||||||
|
pub struct BrightnessGetCommand {}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "next",
|
||||||
|
description = "toggle to next keyboard brightness"
|
||||||
|
)]
|
||||||
|
pub struct BrightnessNextCommand {}
|
||||||
|
|
||||||
|
#[derive(FromArgs, Debug, Default)]
|
||||||
|
#[argh(
|
||||||
|
subcommand,
|
||||||
|
name = "prev",
|
||||||
|
description = "toggle to previous keyboard brightness"
|
||||||
|
)]
|
||||||
|
pub struct BrightnessPrevCommand {}
|
||||||
|
|||||||
@@ -1,49 +1,44 @@
|
|||||||
use gumdrop::Options;
|
use argh::FromArgs;
|
||||||
use rog_platform::platform::PlatformProfile;
|
use rog_platform::platform::PlatformProfile;
|
||||||
use rog_profiles::fan_curve_set::CurveData;
|
use rog_profiles::fan_curve_set::CurveData;
|
||||||
use rog_profiles::FanCurvePU;
|
use rog_profiles::FanCurvePU;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Options)]
|
#[derive(FromArgs, Debug, Clone)]
|
||||||
|
#[argh(subcommand, name = "fan-curve", description = "fan curve commands")]
|
||||||
pub struct FanCurveCommand {
|
pub struct FanCurveCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(switch, description = "get enabled fan profiles")]
|
||||||
pub help: bool,
|
|
||||||
|
|
||||||
#[options(help = "get enabled fan profiles")]
|
|
||||||
pub get_enabled: bool,
|
pub get_enabled: bool,
|
||||||
|
|
||||||
#[options(help = "set the active profile's fan curve to default")]
|
#[argh(switch, description = "set the active profile's fan curve to default")]
|
||||||
pub default: bool,
|
pub default: bool,
|
||||||
|
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "profile to modify fan-curve for. Shows data if no options provided"
|
description = "profile to modify fan-curve for. shows data if no options provided"
|
||||||
)]
|
)]
|
||||||
pub mod_profile: Option<PlatformProfile>,
|
pub mod_profile: Option<PlatformProfile>,
|
||||||
|
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "enable or disable <true/false> fan all curves for a profile. `--mod_profile` \
|
description = "enable or disable <true/false> fan all curves for a profile; --mod_profile required"
|
||||||
required"
|
|
||||||
)]
|
)]
|
||||||
pub enable_fan_curves: Option<bool>,
|
pub enable_fan_curves: Option<bool>,
|
||||||
|
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "enable or disable <true/false> a single fan curve for a profile. `--mod_profile` \
|
description = "enable or disable <true/false> a single fan curve for a profile; --mod_profile and --fan required"
|
||||||
and `--fan` required"
|
|
||||||
)]
|
)]
|
||||||
pub enable_fan_curve: Option<bool>,
|
pub enable_fan_curve: Option<bool>,
|
||||||
|
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "select fan <cpu/gpu/mid> to modify. `--mod_profile` required"
|
description = "select fan <cpu/gpu/mid> to modify; --mod_profile required"
|
||||||
)]
|
)]
|
||||||
pub fan: Option<FanCurvePU>,
|
pub fan: Option<FanCurvePU>,
|
||||||
|
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "data format = 30c:1%,49c:2%,59c:3%,69c:4%,79c:31%,89c:49%,99c:56%,109c:58%. \
|
description = "data format = 30c:1%,49c:2%,...; --mod-profile required. If '%' is omitted the fan range is 0-255"
|
||||||
`--mod-profile` required. If '%' is omitted the fan range is 0-255"
|
|
||||||
)]
|
)]
|
||||||
pub data: Option<CurveData>,
|
pub data: Option<CurveData>,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
use std::convert::TryFrom;
|
use std::convert::TryFrom;
|
||||||
use std::env::args;
|
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::process::Command;
|
use std::process::Command;
|
||||||
use std::thread::sleep;
|
use std::thread::sleep;
|
||||||
@@ -8,12 +7,11 @@ use anime_cli::{AnimeActions, AnimeCommand};
|
|||||||
use aura_cli::{LedPowerCommand1, LedPowerCommand2};
|
use aura_cli::{LedPowerCommand1, LedPowerCommand2};
|
||||||
use dmi_id::DMIID;
|
use dmi_id::DMIID;
|
||||||
use fan_curve_cli::FanCurveCommand;
|
use fan_curve_cli::FanCurveCommand;
|
||||||
use gumdrop::{Opt, Options};
|
|
||||||
use log::{error, info, LevelFilter};
|
use log::{error, info, LevelFilter};
|
||||||
use rog_anime::usb::get_anime_type;
|
use rog_anime::usb::get_anime_type;
|
||||||
use rog_anime::{AnimTime, AnimeDataBuffer, AnimeDiagonal, AnimeGif, AnimeImage, AnimeType, Vec2};
|
use rog_anime::{AnimTime, AnimeDataBuffer, AnimeDiagonal, AnimeGif, AnimeImage, AnimeType, Vec2};
|
||||||
use rog_aura::keyboard::{AuraPowerState, LaptopAuraPower};
|
use rog_aura::keyboard::{AuraPowerState, LaptopAuraPower};
|
||||||
use rog_aura::{self, AuraDeviceType, AuraEffect, PowerZones};
|
use rog_aura::{self, AuraEffect, PowerZones};
|
||||||
use rog_dbus::asus_armoury::AsusArmouryProxyBlocking;
|
use rog_dbus::asus_armoury::AsusArmouryProxyBlocking;
|
||||||
use rog_dbus::list_iface_blocking;
|
use rog_dbus::list_iface_blocking;
|
||||||
use rog_dbus::scsi_aura::ScsiAuraProxyBlocking;
|
use rog_dbus::scsi_aura::ScsiAuraProxyBlocking;
|
||||||
@@ -32,7 +30,6 @@ use scsi_cli::ScsiCommand;
|
|||||||
use zbus::blocking::proxy::ProxyImpl;
|
use zbus::blocking::proxy::ProxyImpl;
|
||||||
use zbus::blocking::Connection;
|
use zbus::blocking::Connection;
|
||||||
|
|
||||||
use crate::aura_cli::{AuraPowerStates, LedBrightness};
|
|
||||||
use crate::cli_opts::*;
|
use crate::cli_opts::*;
|
||||||
use crate::slash_cli::SlashCommand;
|
use crate::slash_cli::SlashCommand;
|
||||||
|
|
||||||
@@ -56,22 +53,7 @@ fn main() {
|
|||||||
.format_timestamp(None)
|
.format_timestamp(None)
|
||||||
.init();
|
.init();
|
||||||
|
|
||||||
let self_version = env!("CARGO_PKG_VERSION");
|
let parsed: CliStart = argh::from_env();
|
||||||
println!("Starting version {self_version}");
|
|
||||||
let args: Vec<String> = args().skip(1).collect();
|
|
||||||
|
|
||||||
let missing_argument_k = gumdrop::Error::missing_argument(Opt::Short('k'));
|
|
||||||
let parsed = match CliStart::parse_args_default(&args) {
|
|
||||||
Ok(p) => p,
|
|
||||||
Err(err) if err.to_string() == missing_argument_k.to_string() => CliStart {
|
|
||||||
kbd_bright: Some(LedBrightness::new(None)),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
Err(err) => {
|
|
||||||
println!("Error: {}", err);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let conn = Connection::system().unwrap();
|
let conn = Connection::system().unwrap();
|
||||||
if let Ok(platform_proxy) = PlatformProxyBlocking::new(&conn).map_err(|e| {
|
if let Ok(platform_proxy) = PlatformProxyBlocking::new(&conn).map_err(|e| {
|
||||||
@@ -90,6 +72,7 @@ fn main() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let self_version = env!("CARGO_PKG_VERSION");
|
||||||
if asusd_version != self_version {
|
if asusd_version != self_version {
|
||||||
println!("Version mismatch: asusctl = {self_version}, asusd = {asusd_version}");
|
println!("Version mismatch: asusctl = {self_version}, asusd = {asusd_version}");
|
||||||
return;
|
return;
|
||||||
@@ -110,12 +93,6 @@ fn main() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if parsed.version {
|
|
||||||
println!("asusctl v{}", env!("CARGO_PKG_VERSION"));
|
|
||||||
println!();
|
|
||||||
print_info();
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Err(err) = do_parsed(&parsed, &supported_interfaces, &supported_properties, conn) {
|
if let Err(err) = do_parsed(&parsed, &supported_interfaces, &supported_properties, conn) {
|
||||||
print_error_help(&*err, &supported_interfaces, &supported_properties);
|
print_error_help(&*err, &supported_interfaces, &supported_properties);
|
||||||
}
|
}
|
||||||
@@ -142,9 +119,9 @@ fn print_info() {
|
|||||||
let dmi = DMIID::new().unwrap_or_default();
|
let dmi = DMIID::new().unwrap_or_default();
|
||||||
let board_name = dmi.board_name;
|
let board_name = dmi.board_name;
|
||||||
let prod_family = dmi.product_family;
|
let prod_family = dmi.product_family;
|
||||||
println!("asusctl version: {}", env!("CARGO_PKG_VERSION"));
|
println!("Software version: {}", env!("CARGO_PKG_VERSION"));
|
||||||
println!(" Product family: {}", prod_family.trim());
|
println!(" Product family: {}", prod_family.trim());
|
||||||
println!(" Board name: {}", board_name.trim());
|
println!(" Board name: {}", board_name.trim());
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_service(name: &str) -> bool {
|
fn check_service(name: &str) -> bool {
|
||||||
@@ -209,149 +186,63 @@ fn do_parsed(
|
|||||||
conn: Connection,
|
conn: Connection,
|
||||||
) -> Result<(), Box<dyn std::error::Error>> {
|
) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
match &parsed.command {
|
match &parsed.command {
|
||||||
Some(CliCommand::Aura(mode)) => handle_led_mode(mode)?,
|
CliCommand::Aura(mode) => handle_led_mode(mode)?,
|
||||||
Some(CliCommand::AuraPowerOld(pow)) => handle_led_power1(pow)?,
|
CliCommand::AuraPowerOld(pow) => handle_led_power1(pow)?,
|
||||||
Some(CliCommand::AuraPower(pow)) => handle_led_power2(pow)?,
|
CliCommand::AuraPower(pow) => handle_led_power2(pow)?,
|
||||||
Some(CliCommand::Profile(cmd)) => {
|
CliCommand::Brightness(cmd) => handle_brightness(cmd)?,
|
||||||
handle_throttle_profile(&conn, supported_properties, cmd)?
|
CliCommand::Profile(cmd) => handle_throttle_profile(&conn, supported_properties, cmd)?,
|
||||||
}
|
CliCommand::FanCurve(cmd) => handle_fan_curve(&conn, cmd)?,
|
||||||
Some(CliCommand::FanCurve(cmd)) => {
|
CliCommand::Anime(cmd) => handle_anime(cmd)?,
|
||||||
handle_fan_curve(&conn, cmd)?;
|
CliCommand::Slash(cmd) => handle_slash(cmd)?,
|
||||||
}
|
CliCommand::Scsi(cmd) => handle_scsi(cmd)?,
|
||||||
Some(CliCommand::Graphics(_)) => do_gfx(),
|
CliCommand::Armoury(cmd) => handle_armoury_command(cmd)?,
|
||||||
Some(CliCommand::Anime(cmd)) => handle_anime(cmd)?,
|
CliCommand::Backlight(cmd) => handle_backlight(cmd)?,
|
||||||
Some(CliCommand::Slash(cmd)) => handle_slash(cmd)?,
|
CliCommand::Battery(cmd) => handle_battery(cmd, &conn)?,
|
||||||
Some(CliCommand::Scsi(cmd)) => handle_scsi(cmd)?,
|
CliCommand::Info(info_opt) => {
|
||||||
Some(CliCommand::Armoury(cmd)) => handle_armoury_command(cmd)?,
|
handle_info(info_opt, supported_interfaces, supported_properties)?
|
||||||
Some(CliCommand::Backlight(cmd)) => handle_backlight(cmd)?,
|
|
||||||
None => {
|
|
||||||
if (!parsed.show_supported
|
|
||||||
&& parsed.kbd_bright.is_none()
|
|
||||||
&& parsed.chg_limit.is_none()
|
|
||||||
&& !parsed.next_kbd_bright
|
|
||||||
&& !parsed.prev_kbd_bright
|
|
||||||
&& !parsed.one_shot_chg)
|
|
||||||
|| parsed.help
|
|
||||||
{
|
|
||||||
println!("{}", CliStart::usage());
|
|
||||||
println!();
|
|
||||||
if let Some(cmdlist) = CliStart::command_list() {
|
|
||||||
let dev_type =
|
|
||||||
if let Ok(proxy) = find_iface::<AuraProxyBlocking>("xyz.ljones.Aura") {
|
|
||||||
// TODO: commands on all?
|
|
||||||
proxy
|
|
||||||
.first()
|
|
||||||
.unwrap()
|
|
||||||
.device_type()
|
|
||||||
.unwrap_or(AuraDeviceType::Unknown)
|
|
||||||
} else {
|
|
||||||
AuraDeviceType::Unknown
|
|
||||||
};
|
|
||||||
let commands: Vec<String> = cmdlist.lines().map(|s| s.to_owned()).collect();
|
|
||||||
for command in commands.iter().filter(|command| {
|
|
||||||
if command.trim().starts_with("fan-curve")
|
|
||||||
&& !supported_interfaces.contains(&"xyz.ljones.FanCurves".to_string())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if command.trim().starts_with("aura")
|
|
||||||
&& !supported_interfaces.contains(&"xyz.ljones.Aura".to_string())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if command.trim().starts_with("anime")
|
|
||||||
&& !supported_interfaces.contains(&"xyz.ljones.Anime".to_string())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if command.trim().starts_with("slash")
|
|
||||||
&& !supported_interfaces.contains(&"xyz.ljones.Slash".to_string())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if command.trim().starts_with("platform")
|
|
||||||
&& !supported_interfaces.contains(&"xyz.ljones.Platform".to_string())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if command.trim().starts_with("armoury")
|
|
||||||
&& !supported_interfaces.contains(&"xyz.ljones.AsusArmoury".to_string())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if command.trim().starts_with("backlight")
|
|
||||||
&& !supported_interfaces.contains(&"xyz.ljones.Backlight".to_string())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if !dev_type.is_old_laptop()
|
|
||||||
&& !dev_type.is_tuf_laptop()
|
|
||||||
&& command.trim().starts_with("aura-power-old")
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if !dev_type.is_new_laptop() && command.trim().starts_with("aura-power") {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
true
|
|
||||||
}) {
|
|
||||||
println!("{}", command);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
println!("\nExtra help can be requested on any command or subcommand:");
|
|
||||||
println!(" asusctl aura --help");
|
|
||||||
println!(" asusctl aura static --help");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(brightness) = &parsed.kbd_bright {
|
Ok(())
|
||||||
if let Ok(aura) = find_iface::<AuraProxyBlocking>("xyz.ljones.Aura") {
|
}
|
||||||
for aura in aura.iter() {
|
|
||||||
match brightness.level() {
|
fn handle_battery(
|
||||||
None => {
|
cmd: &BatteryCommand,
|
||||||
let level = aura.brightness()?;
|
conn: &Connection,
|
||||||
println!("Current keyboard led brightness: {level:?}");
|
) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
}
|
match &cmd.command {
|
||||||
Some(level) => aura.set_brightness(rog_aura::LedBrightness::from(level))?,
|
BatterySubCommand::Limit(l) => {
|
||||||
}
|
let proxy = PlatformProxyBlocking::new(conn)?;
|
||||||
|
proxy.set_charge_control_end_threshold(l.limit)?;
|
||||||
|
}
|
||||||
|
BatterySubCommand::OneShot(o) => {
|
||||||
|
let proxy = PlatformProxyBlocking::new(conn)?;
|
||||||
|
if let Some(p) = o.percent {
|
||||||
|
proxy.set_charge_control_end_threshold(p)?;
|
||||||
}
|
}
|
||||||
} else {
|
proxy.one_shot_full_charge()?;
|
||||||
println!("No aura interface found");
|
}
|
||||||
|
BatterySubCommand::Info(_) => {
|
||||||
|
let proxy = PlatformProxyBlocking::new(conn)?;
|
||||||
|
let limit = proxy.charge_control_end_threshold()?;
|
||||||
|
println!("Current battery charge limit: {}%", limit);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if parsed.next_kbd_bright {
|
Ok(())
|
||||||
if let Ok(aura) = find_iface::<AuraProxyBlocking>("xyz.ljones.Aura") {
|
}
|
||||||
for aura in aura.iter() {
|
|
||||||
let brightness = aura.brightness()?;
|
|
||||||
aura.set_brightness(brightness.next())?;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
println!("No aura interface found");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if parsed.prev_kbd_bright {
|
fn handle_info(
|
||||||
if let Ok(aura) = find_iface::<AuraProxyBlocking>("xyz.ljones.Aura") {
|
info_opt: &InfoCommand,
|
||||||
for aura in aura.iter() {
|
supported_interfaces: &[String],
|
||||||
let brightness = aura.brightness()?;
|
supported_properties: &[Properties],
|
||||||
aura.set_brightness(brightness.prev())?;
|
) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
}
|
println!("asusctl v{}", env!("CARGO_PKG_VERSION"));
|
||||||
} else {
|
println!();
|
||||||
println!("No aura interface found");
|
print_info();
|
||||||
}
|
println!();
|
||||||
}
|
|
||||||
|
|
||||||
if parsed.show_supported {
|
if info_opt.show_supported {
|
||||||
println!("Supported Core Functions:\n{:#?}", supported_interfaces);
|
println!("Supported Core Functions:\n{:#?}", supported_interfaces);
|
||||||
println!(
|
println!(
|
||||||
"Supported Platform Properties:\n{:#?}",
|
"Supported Platform Properties:\n{:#?}",
|
||||||
@@ -372,35 +263,14 @@ fn do_parsed(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(chg_limit) = parsed.chg_limit {
|
|
||||||
let proxy = PlatformProxyBlocking::new(&conn)?;
|
|
||||||
proxy.set_charge_control_end_threshold(chg_limit)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
if parsed.one_shot_chg {
|
|
||||||
let proxy = PlatformProxyBlocking::new(&conn)?;
|
|
||||||
proxy.one_shot_full_charge()?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn do_gfx() {
|
|
||||||
println!(
|
|
||||||
"Please use supergfxctl for graphics switching. supergfxctl is the result of making \
|
|
||||||
asusctl graphics switching generic so all laptops can use it"
|
|
||||||
);
|
|
||||||
println!("This command will be removed in future");
|
|
||||||
}
|
|
||||||
|
|
||||||
fn handle_backlight(cmd: &BacklightCommand) -> Result<(), Box<dyn std::error::Error>> {
|
fn handle_backlight(cmd: &BacklightCommand) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
if (cmd.screenpad_brightness.is_none()
|
if cmd.screenpad_brightness.is_none()
|
||||||
&& cmd.screenpad_gamma.is_none()
|
&& cmd.screenpad_gamma.is_none()
|
||||||
&& cmd.sync_screenpad_brightness.is_none())
|
&& cmd.sync_screenpad_brightness.is_none()
|
||||||
|| cmd.help
|
|
||||||
{
|
{
|
||||||
println!("Missing arg or command\n\n{}", cmd.self_usage());
|
|
||||||
|
|
||||||
let backlights = find_iface::<BacklightProxyBlocking>("xyz.ljones.Backlight")?;
|
let backlights = find_iface::<BacklightProxyBlocking>("xyz.ljones.Backlight")?;
|
||||||
for backlight in backlights {
|
for backlight in backlights {
|
||||||
println!("Current screenpad settings:");
|
println!("Current screenpad settings:");
|
||||||
@@ -433,8 +303,50 @@ fn handle_backlight(cmd: &BacklightCommand) -> Result<(), Box<dyn std::error::Er
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn handle_brightness(cmd: &BrightnessCommand) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
let Ok(aura_proxies) = find_iface::<AuraProxyBlocking>("xyz.ljones.Aura") else {
|
||||||
|
println!("No aura interface found");
|
||||||
|
return Ok(());
|
||||||
|
};
|
||||||
|
|
||||||
|
match &cmd.command {
|
||||||
|
BrightnessSubCommand::Set(s) => {
|
||||||
|
for aura in aura_proxies.iter() {
|
||||||
|
if let Some(level) = s.level.level() {
|
||||||
|
aura.set_brightness(rog_aura::LedBrightness::from(level))?;
|
||||||
|
} else {
|
||||||
|
let current = aura.brightness()?;
|
||||||
|
println!("Current keyboard led brightness: {current:?}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BrightnessSubCommand::Get(_) => {
|
||||||
|
for aura in aura_proxies.iter() {
|
||||||
|
let level = aura.brightness()?;
|
||||||
|
println!("Current keyboard led brightness: {level:?}");
|
||||||
|
}
|
||||||
|
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
BrightnessSubCommand::Next(_) => {
|
||||||
|
for aura in aura_proxies.iter() {
|
||||||
|
let brightness = aura.brightness()?;
|
||||||
|
aura.set_brightness(brightness.next())?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BrightnessSubCommand::Prev(_) => {
|
||||||
|
for aura in aura_proxies.iter() {
|
||||||
|
let brightness = aura.brightness()?;
|
||||||
|
aura.set_brightness(brightness.prev())?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
if (cmd.command.is_none()
|
if cmd.command.is_none()
|
||||||
&& cmd.enable_display.is_none()
|
&& cmd.enable_display.is_none()
|
||||||
&& cmd.enable_powersave_anim.is_none()
|
&& cmd.enable_powersave_anim.is_none()
|
||||||
&& cmd.brightness.is_none()
|
&& cmd.brightness.is_none()
|
||||||
@@ -442,13 +354,9 @@ fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
&& cmd.off_when_suspended.is_none()
|
&& cmd.off_when_suspended.is_none()
|
||||||
&& cmd.off_when_unplugged.is_none()
|
&& cmd.off_when_unplugged.is_none()
|
||||||
&& cmd.off_with_his_head.is_none()
|
&& cmd.off_with_his_head.is_none()
|
||||||
&& !cmd.clear)
|
&& !cmd.clear
|
||||||
|| cmd.help
|
|
||||||
{
|
{
|
||||||
println!("Missing arg or command\n\n{}", cmd.self_usage());
|
println!("Missing arg or command; run 'asusctl anime --help' for usage");
|
||||||
if let Some(lst) = cmd.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let animes = find_iface::<AnimeProxyBlocking>("xyz.ljones.Anime").map_err(|e| {
|
let animes = find_iface::<AnimeProxyBlocking>("xyz.ljones.Anime").map_err(|e| {
|
||||||
@@ -495,11 +403,10 @@ fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
if let Some(action) = cmd.command.as_ref() {
|
if let Some(action) = cmd.command.as_ref() {
|
||||||
match action {
|
match action {
|
||||||
AnimeActions::Image(image) => {
|
AnimeActions::Image(image) => {
|
||||||
if image.help_requested() || image.path.is_empty() {
|
if image.path.is_empty() {
|
||||||
println!("Missing arg or command\n\n{}", image.self_usage());
|
println!(
|
||||||
if let Some(lst) = image.self_command_list() {
|
"Missing arg or command; run 'asusctl anime image --help' for usage"
|
||||||
println!("\n{}", lst);
|
);
|
||||||
}
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
verify_brightness(image.bright);
|
verify_brightness(image.bright);
|
||||||
@@ -516,11 +423,8 @@ fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
proxy.write(<AnimeDataBuffer>::try_from(&matrix)?)?;
|
proxy.write(<AnimeDataBuffer>::try_from(&matrix)?)?;
|
||||||
}
|
}
|
||||||
AnimeActions::PixelImage(image) => {
|
AnimeActions::PixelImage(image) => {
|
||||||
if image.help_requested() || image.path.is_empty() {
|
if image.path.is_empty() {
|
||||||
println!("Missing arg or command\n\n{}", image.self_usage());
|
println!("Missing arg or command; run 'asusctl anime pixel-image --help' for usage");
|
||||||
if let Some(lst) = image.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
verify_brightness(image.bright);
|
verify_brightness(image.bright);
|
||||||
@@ -535,11 +439,10 @@ fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
proxy.write(matrix.into_data_buffer(anime_type)?)?;
|
proxy.write(matrix.into_data_buffer(anime_type)?)?;
|
||||||
}
|
}
|
||||||
AnimeActions::Gif(gif) => {
|
AnimeActions::Gif(gif) => {
|
||||||
if gif.help_requested() || gif.path.is_empty() {
|
if gif.path.is_empty() {
|
||||||
println!("Missing arg or command\n\n{}", gif.self_usage());
|
println!(
|
||||||
if let Some(lst) = gif.self_command_list() {
|
"Missing arg or command; run 'asusctl anime gif --help' for usage"
|
||||||
println!("\n{}", lst);
|
);
|
||||||
}
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
verify_brightness(gif.bright);
|
verify_brightness(gif.bright);
|
||||||
@@ -569,11 +472,8 @@ fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
AnimeActions::PixelGif(gif) => {
|
AnimeActions::PixelGif(gif) => {
|
||||||
if gif.help_requested() || gif.path.is_empty() {
|
if gif.path.is_empty() {
|
||||||
println!("Missing arg or command\n\n{}", gif.self_usage());
|
println!("Missing arg or command; run 'asusctl anime pixel-gif --help' for usage");
|
||||||
if let Some(lst) = gif.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
verify_brightness(gif.bright);
|
verify_brightness(gif.bright);
|
||||||
@@ -600,14 +500,8 @@ fn handle_anime(cmd: &AnimeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
AnimeActions::SetBuiltins(builtins) => {
|
AnimeActions::SetBuiltins(builtins) => {
|
||||||
if builtins.help_requested() || builtins.set.is_none() {
|
if builtins.set.is_none() {
|
||||||
println!(
|
println!("Missing arg; run 'asusctl anime set-builtins --help' for usage");
|
||||||
"\nAny unspecified args will be set to default (first shown var)\n"
|
|
||||||
);
|
|
||||||
println!("\n{}", builtins.self_usage());
|
|
||||||
if let Some(lst) = builtins.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -634,24 +528,19 @@ fn verify_brightness(brightness: f32) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn handle_slash(cmd: &SlashCommand) -> Result<(), Box<dyn std::error::Error>> {
|
fn handle_slash(cmd: &SlashCommand) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
if (cmd.brightness.is_none()
|
if cmd.brightness.is_none()
|
||||||
&& cmd.interval.is_none()
|
&& cmd.interval.is_none()
|
||||||
&& cmd.show_on_boot.is_none()
|
&& cmd.show_on_boot.is_none()
|
||||||
&& cmd.show_on_shutdown.is_none()
|
&& cmd.show_on_shutdown.is_none()
|
||||||
&& cmd.show_on_sleep.is_none()
|
&& cmd.show_on_sleep.is_none()
|
||||||
&& cmd.show_on_battery.is_none()
|
&& cmd.show_on_battery.is_none()
|
||||||
&& cmd.show_battery_warning.is_none()
|
&& cmd.show_battery_warning.is_none()
|
||||||
// && cmd.show_on_lid_closed.is_none()
|
|
||||||
&& cmd.mode.is_none()
|
&& cmd.mode.is_none()
|
||||||
&& !cmd.list
|
&& !cmd.list
|
||||||
&& !cmd.enable
|
&& !cmd.enable
|
||||||
&& !cmd.disable)
|
&& !cmd.disable
|
||||||
|| cmd.help
|
|
||||||
{
|
{
|
||||||
println!("Missing arg or command\n\n{}", cmd.self_usage());
|
println!("Missing arg or command; run 'asusctl slash --help' for usage");
|
||||||
if let Some(lst) = cmd.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let slashes = find_iface::<SlashProxyBlocking>("xyz.ljones.Slash")?;
|
let slashes = find_iface::<SlashProxyBlocking>("xyz.ljones.Slash")?;
|
||||||
@@ -702,13 +591,8 @@ fn handle_slash(cmd: &SlashCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn handle_scsi(cmd: &ScsiCommand) -> Result<(), Box<dyn std::error::Error>> {
|
fn handle_scsi(cmd: &ScsiCommand) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
if (!cmd.list && cmd.enable.is_none() && cmd.mode.is_none() && cmd.colours.is_empty())
|
if !cmd.list && cmd.enable.is_none() && cmd.mode.is_none() && cmd.colours.is_empty() {
|
||||||
|| cmd.help
|
println!("Missing arg or command; run 'asusctl scsi --help' for usage");
|
||||||
{
|
|
||||||
println!("Missing arg or command\n\n{}", cmd.self_usage());
|
|
||||||
if let Some(lst) = cmd.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let scsis = find_iface::<ScsiAuraProxyBlocking>("xyz.ljones.ScsiAura")?;
|
let scsis = find_iface::<ScsiAuraProxyBlocking>("xyz.ljones.ScsiAura")?;
|
||||||
@@ -774,38 +658,15 @@ fn handle_scsi(cmd: &ScsiCommand) -> Result<(), Box<dyn std::error::Error>> {
|
|||||||
|
|
||||||
fn handle_led_mode(mode: &LedModeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
fn handle_led_mode(mode: &LedModeCommand) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
if mode.command.is_none() && !mode.prev_mode && !mode.next_mode {
|
if mode.command.is_none() && !mode.prev_mode && !mode.next_mode {
|
||||||
if !mode.help {
|
println!("Missing arg or command; run 'asusctl aura --help' for usage");
|
||||||
println!("Missing arg or command\n");
|
// print available modes when possible
|
||||||
}
|
if let Ok(aura) = find_iface::<AuraProxyBlocking>("xyz.ljones.Aura") {
|
||||||
println!("{}\n", mode.self_usage());
|
|
||||||
println!("Commands available");
|
|
||||||
|
|
||||||
if let Some(cmdlist) = LedModeCommand::command_list() {
|
|
||||||
let commands: Vec<String> = cmdlist.lines().map(|s| s.to_owned()).collect();
|
|
||||||
// TODO: multiple rgb check
|
|
||||||
let aura = find_iface::<AuraProxyBlocking>("xyz.ljones.Aura")?;
|
|
||||||
let modes = aura.first().unwrap().supported_basic_modes()?;
|
let modes = aura.first().unwrap().supported_basic_modes()?;
|
||||||
for command in commands.iter().filter(|command| {
|
println!("Available modes:");
|
||||||
for mode in &modes {
|
for m in modes {
|
||||||
let mut mode = <&str>::from(mode).to_string();
|
println!(" {:?}", m);
|
||||||
if let Some(pos) = mode.chars().skip(1).position(|c| c.is_uppercase()) {
|
|
||||||
mode.insert(pos + 1, '-');
|
|
||||||
}
|
|
||||||
if command.trim().starts_with(&mode.to_lowercase()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// TODO
|
|
||||||
// if !supported.basic_zones.is_empty() && command.trim().starts_with("multi") {
|
|
||||||
// return true;
|
|
||||||
// }
|
|
||||||
false
|
|
||||||
}) {
|
|
||||||
println!("{}", command);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
println!("\nHelp can also be requested on modes, e.g: static --help");
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -837,10 +698,6 @@ fn handle_led_mode(mode: &LedModeCommand) -> Result<(), Box<dyn std::error::Erro
|
|||||||
aura.set_led_mode(modes[pos])?;
|
aura.set_led_mode(modes[pos])?;
|
||||||
}
|
}
|
||||||
} else if let Some(mode) = mode.command.as_ref() {
|
} else if let Some(mode) = mode.command.as_ref() {
|
||||||
if mode.help_requested() {
|
|
||||||
println!("{}", mode.self_usage());
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
for aura in aura {
|
for aura in aura {
|
||||||
aura.set_led_mode_data(<AuraEffect>::from(mode))?;
|
aura.set_led_mode_data(<AuraEffect>::from(mode))?;
|
||||||
}
|
}
|
||||||
@@ -863,10 +720,7 @@ fn handle_led_power1(power: &LedPowerCommand1) -> Result<(), Box<dyn std::error:
|
|||||||
&& !power.keyboard
|
&& !power.keyboard
|
||||||
&& !power.lightbar
|
&& !power.lightbar
|
||||||
{
|
{
|
||||||
if !power.help {
|
println!("Missing arg or command; run 'asusctl aura-power-old --help' for usage");
|
||||||
println!("Missing arg or command\n");
|
|
||||||
}
|
|
||||||
println!("{}\n", power.self_usage());
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -918,51 +772,47 @@ fn handle_led_power2(power: &LedPowerCommand2) -> Result<(), Box<dyn std::error:
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if power.command().is_none() {
|
if power.command.is_none() {
|
||||||
if !power.help {
|
println!("Missing arg or command; run 'asusctl aura-power --help' for usage");
|
||||||
println!("Missing arg or command\n");
|
|
||||||
}
|
|
||||||
println!("{}\n", power.self_usage());
|
|
||||||
println!("Commands available");
|
println!("Commands available");
|
||||||
|
|
||||||
if let Some(cmdlist) = LedPowerCommand2::command_list() {
|
|
||||||
let commands: Vec<String> = cmdlist.lines().map(|s| s.to_owned()).collect();
|
|
||||||
for command in &commands {
|
|
||||||
println!("{}", command);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
println!("\nHelp can also be requested on commands, e.g: boot --help");
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(pow) = power.command.as_ref() {
|
if let Some(_pow) = power.command.as_ref() {
|
||||||
if pow.help_requested() {
|
|
||||||
println!("{}", pow.self_usage());
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut states = aura.led_power()?;
|
let mut states = aura.led_power()?;
|
||||||
let mut set = |zone: PowerZones, set_to: &AuraPowerStates| {
|
let mut set =
|
||||||
for state in states.states.iter_mut() {
|
|zone: PowerZones, boot_v: bool, awake_v: bool, sleep_v: bool, shutdown_v: bool| {
|
||||||
if state.zone == zone {
|
for state in states.states.iter_mut() {
|
||||||
state.boot = set_to.boot;
|
if state.zone == zone {
|
||||||
state.awake = set_to.awake;
|
state.boot = boot_v;
|
||||||
state.sleep = set_to.sleep;
|
state.awake = awake_v;
|
||||||
state.shutdown = set_to.shutdown;
|
state.sleep = sleep_v;
|
||||||
break;
|
state.shutdown = shutdown_v;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(cmd) = &power.command {
|
if let Some(cmd) = &power.command {
|
||||||
match cmd {
|
match cmd {
|
||||||
aura_cli::SetAuraZoneEnabled::Keyboard(k) => set(PowerZones::Keyboard, k),
|
aura_cli::SetAuraZoneEnabled::Keyboard(k) => {
|
||||||
aura_cli::SetAuraZoneEnabled::Logo(l) => set(PowerZones::Logo, l),
|
set(PowerZones::Keyboard, k.boot, k.awake, k.sleep, k.shutdown)
|
||||||
aura_cli::SetAuraZoneEnabled::Lightbar(l) => set(PowerZones::Lightbar, l),
|
}
|
||||||
aura_cli::SetAuraZoneEnabled::Lid(l) => set(PowerZones::Lid, l),
|
aura_cli::SetAuraZoneEnabled::Logo(l) => {
|
||||||
aura_cli::SetAuraZoneEnabled::RearGlow(r) => set(PowerZones::RearGlow, r),
|
set(PowerZones::Logo, l.boot, l.awake, l.sleep, l.shutdown)
|
||||||
aura_cli::SetAuraZoneEnabled::Ally(r) => set(PowerZones::Ally, r),
|
}
|
||||||
|
aura_cli::SetAuraZoneEnabled::Lightbar(l) => {
|
||||||
|
set(PowerZones::Lightbar, l.boot, l.awake, l.sleep, l.shutdown)
|
||||||
|
}
|
||||||
|
aura_cli::SetAuraZoneEnabled::Lid(l) => {
|
||||||
|
set(PowerZones::Lid, l.boot, l.awake, l.sleep, l.shutdown)
|
||||||
|
}
|
||||||
|
aura_cli::SetAuraZoneEnabled::RearGlow(r) => {
|
||||||
|
set(PowerZones::RearGlow, r.boot, r.awake, r.sleep, r.shutdown)
|
||||||
|
}
|
||||||
|
aura_cli::SetAuraZoneEnabled::Ally(r) => {
|
||||||
|
set(PowerZones::Ally, r.boot, r.awake, r.sleep, r.shutdown)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -983,51 +833,37 @@ fn handle_throttle_profile(
|
|||||||
return Err(ProfileError::NotSupported.into());
|
return Err(ProfileError::NotSupported.into());
|
||||||
}
|
}
|
||||||
|
|
||||||
if !cmd.next
|
|
||||||
&& !cmd.list
|
|
||||||
&& cmd.profile_set.is_none()
|
|
||||||
&& !cmd.profile_get
|
|
||||||
&& cmd.profile_set_ac.is_none()
|
|
||||||
&& cmd.profile_set_bat.is_none()
|
|
||||||
{
|
|
||||||
if !cmd.help {
|
|
||||||
println!("Missing arg or command\n");
|
|
||||||
}
|
|
||||||
println!("{}", ProfileCommand::usage());
|
|
||||||
|
|
||||||
if let Some(lst) = cmd.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let proxy = PlatformProxyBlocking::new(conn)?;
|
let proxy = PlatformProxyBlocking::new(conn)?;
|
||||||
let current = proxy.platform_profile()?;
|
let current = proxy.platform_profile()?;
|
||||||
let choices = proxy.platform_profile_choices()?;
|
let choices = proxy.platform_profile_choices()?;
|
||||||
|
|
||||||
if cmd.next {
|
match &cmd.command {
|
||||||
proxy.set_platform_profile(PlatformProfile::next(current, &choices))?;
|
crate::cli_opts::ProfileSubCommand::Next(_) => {
|
||||||
} else if let Some(profile) = cmd.profile_set {
|
proxy.set_platform_profile(PlatformProfile::next(current, &choices))?;
|
||||||
proxy.set_platform_profile(profile)?;
|
}
|
||||||
} else if let Some(profile) = cmd.profile_set_ac {
|
crate::cli_opts::ProfileSubCommand::Set(s) => {
|
||||||
proxy.set_platform_profile_on_ac(profile)?;
|
if !s.ac && !s.battery {
|
||||||
} else if let Some(profile) = cmd.profile_set_bat {
|
proxy.set_platform_profile(s.profile)?;
|
||||||
proxy.set_platform_profile_on_battery(profile)?;
|
} else {
|
||||||
}
|
if s.ac {
|
||||||
|
proxy.set_platform_profile_on_ac(s.profile)?;
|
||||||
if cmd.list {
|
}
|
||||||
for p in &choices {
|
if s.battery {
|
||||||
println!("{:?}", p);
|
proxy.set_platform_profile_on_battery(s.profile)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
crate::cli_opts::ProfileSubCommand::List(_) => {
|
||||||
|
for p in &choices {
|
||||||
|
println!("{:?}", p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
crate::cli_opts::ProfileSubCommand::Get(_) => {
|
||||||
|
println!("Active profile: {current:?}");
|
||||||
|
println!();
|
||||||
|
println!("AC profile {:?}", proxy.platform_profile_on_ac()?);
|
||||||
|
println!("Battery profile {:?}", proxy.platform_profile_on_battery()?);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if cmd.profile_get {
|
|
||||||
println!("Active profile is {current:?}");
|
|
||||||
println!("Profile on AC is {:?}", proxy.platform_profile_on_ac()?);
|
|
||||||
println!(
|
|
||||||
"Profile on Battery is {:?}",
|
|
||||||
proxy.platform_profile_on_battery()?
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -1044,14 +880,7 @@ fn handle_fan_curve(
|
|||||||
};
|
};
|
||||||
|
|
||||||
if !cmd.get_enabled && !cmd.default && cmd.mod_profile.is_none() {
|
if !cmd.get_enabled && !cmd.default && cmd.mod_profile.is_none() {
|
||||||
if !cmd.help {
|
println!("Missing arg or command; run 'asusctl fan-curve --help' for usage");
|
||||||
println!("Missing arg or command\n");
|
|
||||||
}
|
|
||||||
println!("{}", FanCurveCommand::usage());
|
|
||||||
|
|
||||||
if let Some(lst) = cmd.self_command_list() {
|
|
||||||
println!("\n{}", lst);
|
|
||||||
}
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1190,33 +1019,33 @@ fn print_firmware_attr(attr: &AsusArmouryProxyBlocking) -> Result<(), Box<dyn st
|
|||||||
|
|
||||||
#[allow(clippy::manual_is_multiple_of, clippy::nonminimal_bool)]
|
#[allow(clippy::manual_is_multiple_of, clippy::nonminimal_bool)]
|
||||||
fn handle_armoury_command(cmd: &ArmouryCommand) -> Result<(), Box<dyn std::error::Error>> {
|
fn handle_armoury_command(cmd: &ArmouryCommand) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
{
|
// If nested subcommand provided, handle set/get/list.
|
||||||
if cmd.free.is_empty() || (cmd.free.len() % 2 != 0) || cmd.help {
|
match &cmd.command {
|
||||||
const USAGE: &str = "Usage: asusctl platform panel_overdrive 1 nv_dynamic_boost 5";
|
ArmourySubCommand::List(_) => {
|
||||||
if cmd.free.len() % 2 != 0 {
|
if let Ok(attrs) = find_iface::<AsusArmouryProxyBlocking>("xyz.ljones.AsusArmoury") {
|
||||||
println!(
|
for attr in attrs.iter() {
|
||||||
"Incorrect number of args, each attribute label must be paired with a setting:"
|
|
||||||
);
|
|
||||||
println!("{USAGE}");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Ok(attr) = find_iface::<AsusArmouryProxyBlocking>("xyz.ljones.AsusArmoury") {
|
|
||||||
println!("\n{USAGE}\n");
|
|
||||||
println!("Available firmware attributes: ");
|
|
||||||
for attr in attr.iter() {
|
|
||||||
print_firmware_attr(attr)?;
|
print_firmware_attr(attr)?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Ok(());
|
Ok(())
|
||||||
}
|
}
|
||||||
|
ArmourySubCommand::Get(g) => {
|
||||||
if let Ok(attr) = find_iface::<AsusArmouryProxyBlocking>("xyz.ljones.AsusArmoury") {
|
if let Ok(attrs) = find_iface::<AsusArmouryProxyBlocking>("xyz.ljones.AsusArmoury") {
|
||||||
for cmd in cmd.free.chunks(2) {
|
for attr in attrs.iter() {
|
||||||
for attr in attr.iter() {
|
|
||||||
let name = attr.name()?;
|
let name = attr.name()?;
|
||||||
if <&str>::from(name) == cmd[0] {
|
if <&str>::from(name) == g.property {
|
||||||
let mut value: i32 = cmd[1].parse()?;
|
print_firmware_attr(attr)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
ArmourySubCommand::Set(s) => {
|
||||||
|
if let Ok(attrs) = find_iface::<AsusArmouryProxyBlocking>("xyz.ljones.AsusArmoury") {
|
||||||
|
for attr in attrs.iter() {
|
||||||
|
let name = attr.name()?;
|
||||||
|
if <&str>::from(name) == s.property {
|
||||||
|
let mut value: i32 = s.value;
|
||||||
if value == -1 {
|
if value == -1 {
|
||||||
info!("Setting to default");
|
info!("Setting to default");
|
||||||
value = attr.default_value()?;
|
value = attr.default_value()?;
|
||||||
@@ -1226,7 +1055,7 @@ fn handle_armoury_command(cmd: &ArmouryCommand) -> Result<(), Box<dyn std::error
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,35 +1,30 @@
|
|||||||
use gumdrop::Options;
|
use argh::FromArgs;
|
||||||
use rog_scsi::{AuraMode, Colour, Direction, Speed};
|
use rog_scsi::{AuraMode, Colour, Direction, Speed};
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "scsi", description = "scsi LED commands")]
|
||||||
pub struct ScsiCommand {
|
pub struct ScsiCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(option, description = "enable the SCSI drive LEDs")]
|
||||||
pub help: bool,
|
|
||||||
|
|
||||||
#[options(help = "Enable the SCSI drive LEDs")]
|
|
||||||
pub enable: Option<bool>,
|
pub enable: Option<bool>,
|
||||||
|
|
||||||
#[options(meta = "", help = "Set LED mode (so 'list' for all options)")]
|
#[argh(option, description = "set LED mode (use 'list' for all options)")]
|
||||||
pub mode: Option<AuraMode>,
|
pub mode: Option<AuraMode>,
|
||||||
|
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "Set LED mode speed <slowest, slow, med, fast, fastest> (does not apply to all)"
|
description = "set LED mode speed <slowest, slow, med, fast, fastest>"
|
||||||
)]
|
)]
|
||||||
pub speed: Option<Speed>,
|
pub speed: Option<Speed>,
|
||||||
|
|
||||||
#[options(
|
#[argh(option, description = "set LED mode direction <forward, reverse>")]
|
||||||
meta = "",
|
|
||||||
help = "Set LED mode direction <forward, reverse> (does not apply to all)"
|
|
||||||
)]
|
|
||||||
pub direction: Option<Direction>,
|
pub direction: Option<Direction>,
|
||||||
|
|
||||||
#[options(
|
#[argh(
|
||||||
meta = "",
|
option,
|
||||||
help = "Set LED colours <hex>, specify up to 4 with repeated arg"
|
description = "set LED colours <hex>, specify up to 4 with repeated arg"
|
||||||
)]
|
)]
|
||||||
pub colours: Vec<Colour>,
|
pub colours: Vec<Colour>,
|
||||||
|
|
||||||
#[options(help = "list available animations")]
|
#[argh(switch, description = "list available animations")]
|
||||||
pub list: bool,
|
pub list: bool,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,37 +1,34 @@
|
|||||||
use gumdrop::Options;
|
use argh::FromArgs;
|
||||||
use rog_slash::SlashMode;
|
use rog_slash::SlashMode;
|
||||||
|
|
||||||
#[derive(Options)]
|
#[derive(FromArgs, Debug)]
|
||||||
|
#[argh(subcommand, name = "slash", description = "slash ledbar commands")]
|
||||||
pub struct SlashCommand {
|
pub struct SlashCommand {
|
||||||
#[options(help = "print help message")]
|
#[argh(switch, description = "enable the Slash Ledbar")]
|
||||||
pub help: bool,
|
|
||||||
#[options(help = "Enable the Slash Ledbar")]
|
|
||||||
pub enable: bool,
|
pub enable: bool,
|
||||||
#[options(help = "Disable the Slash Ledbar")]
|
#[argh(switch, description = "disable the Slash Ledbar")]
|
||||||
pub disable: bool,
|
pub disable: bool,
|
||||||
#[options(short = "l", meta = "", help = "Set brightness value <0-255>")]
|
#[argh(option, short = 'l', description = "set brightness value <0-255>")]
|
||||||
pub brightness: Option<u8>,
|
pub brightness: Option<u8>,
|
||||||
#[options(meta = "", help = "Set interval value <0-5>")]
|
#[argh(option, description = "set interval value <0-5>")]
|
||||||
pub interval: Option<u8>,
|
pub interval: Option<u8>,
|
||||||
#[options(meta = "", help = "Set SlashMode (so 'list' for all options)")]
|
#[argh(option, description = "set SlashMode (use 'list' for options)")]
|
||||||
pub mode: Option<SlashMode>,
|
pub mode: Option<SlashMode>,
|
||||||
#[options(help = "list available animations")]
|
#[argh(switch, description = "list available animations")]
|
||||||
pub list: bool,
|
pub list: bool,
|
||||||
|
|
||||||
#[options(short = "B", meta = "", help = "Show the animation on boot")]
|
#[argh(option, short = 'B', description = "show the animation on boot")]
|
||||||
pub show_on_boot: Option<bool>,
|
pub show_on_boot: Option<bool>,
|
||||||
#[options(short = "S", meta = "", help = "Show the animation on shutdown")]
|
#[argh(option, short = 'S', description = "show the animation on shutdown")]
|
||||||
pub show_on_shutdown: Option<bool>,
|
pub show_on_shutdown: Option<bool>,
|
||||||
#[options(short = "s", meta = "", help = "Show the animation on sleep")]
|
#[argh(option, short = 's', description = "show the animation on sleep")]
|
||||||
pub show_on_sleep: Option<bool>,
|
pub show_on_sleep: Option<bool>,
|
||||||
#[options(short = "b", meta = "", help = "Show the animation on battery")]
|
#[argh(option, short = 'b', description = "show the animation on battery")]
|
||||||
pub show_on_battery: Option<bool>,
|
pub show_on_battery: Option<bool>,
|
||||||
// #[options(short = "L", meta = "", help = "Show the animation on lid closed")]
|
#[argh(
|
||||||
// pub show_on_lid_closed: Option<bool>,
|
option,
|
||||||
#[options(
|
short = 'w',
|
||||||
short = "w",
|
description = "show the low-battery warning animation"
|
||||||
meta = "",
|
|
||||||
help = "Show the low-battery warning animation"
|
|
||||||
)]
|
)]
|
||||||
pub show_battery_warning: Option<bool>,
|
pub show_battery_warning: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,10 @@
|
|||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
use config_traits::StdConfig;
|
use config_traits::StdConfig;
|
||||||
use log::{debug, error, info};
|
use log::{debug, error, info, warn};
|
||||||
use rog_platform::asus_armoury::{AttrValue, Attribute, FirmwareAttribute, FirmwareAttributes};
|
use rog_platform::asus_armoury::{
|
||||||
|
AttrValue, Attribute, FirmwareAttribute, FirmwareAttributeType, FirmwareAttributes,
|
||||||
|
};
|
||||||
use rog_platform::platform::{PlatformProfile, RogPlatform};
|
use rog_platform::platform::{PlatformProfile, RogPlatform};
|
||||||
use rog_platform::power::AsusPower;
|
use rog_platform::power::AsusPower;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
@@ -168,68 +170,64 @@ impl ArmouryAttributeRegistry {
|
|||||||
impl crate::Reloadable for AsusArmouryAttribute {
|
impl crate::Reloadable for AsusArmouryAttribute {
|
||||||
async fn reload(&mut self) -> Result<(), RogError> {
|
async fn reload(&mut self) -> Result<(), RogError> {
|
||||||
info!("Reloading {}", self.attr.name());
|
info!("Reloading {}", self.attr.name());
|
||||||
let name: FirmwareAttribute = self.attr.name().into();
|
let attribute: FirmwareAttribute = self.attr.name().into();
|
||||||
|
let name = self.attr.name();
|
||||||
|
|
||||||
// Treat dGPU attributes the same as PPT attributes for power-profile
|
let config = self.config.lock().await;
|
||||||
// behaviour so they follow AC/DC tuning groups.
|
let apply_value = match attribute.property_type() {
|
||||||
if name.is_ppt() || name.is_dgpu() {
|
FirmwareAttributeType::Ppt => {
|
||||||
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
||||||
let power_plugged = self
|
let power_plugged = self
|
||||||
.power
|
.power
|
||||||
.get_online()
|
.get_online()
|
||||||
.map_err(|e| {
|
.map_err(|e| {
|
||||||
error!("Could not get power status: {e:?}");
|
error!("Could not get power status: {e:?}");
|
||||||
e
|
e
|
||||||
})
|
|
||||||
.unwrap_or_default()
|
|
||||||
== 1;
|
|
||||||
|
|
||||||
let apply_value = {
|
|
||||||
let config = self.config.lock().await;
|
|
||||||
config
|
|
||||||
.select_tunings_ref(power_plugged, profile)
|
|
||||||
.and_then(|tuning| {
|
|
||||||
if tuning.enabled {
|
|
||||||
tuning.group.get(&self.name()).copied()
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
};
|
.unwrap_or_default()
|
||||||
|
== 1;
|
||||||
|
|
||||||
if let Some(tune) = apply_value {
|
let apply_value = {
|
||||||
self.attr
|
config.select_tunings_ref(power_plugged, profile).and_then(
|
||||||
.set_current_value(&AttrValue::Integer(tune))
|
|tuning| match tuning.enabled {
|
||||||
.map_err(|e| {
|
true => tuning.group.get(&self.name()).copied(),
|
||||||
error!("Could not set {} value: {e:?}", self.attr.name());
|
false => None,
|
||||||
self.attr.base_path_exists();
|
},
|
||||||
e
|
)
|
||||||
})?;
|
};
|
||||||
info!("Set {} to {:?}", self.attr.name(), tune);
|
|
||||||
|
apply_value.map_or(AttrValue::None, AttrValue::Integer)
|
||||||
}
|
}
|
||||||
} else {
|
FirmwareAttributeType::Gpu => {
|
||||||
// Handle non-PPT attributes (boolean and other settings)
|
info!("Reload called on GPU attribute {name}: doing nothing");
|
||||||
if let Some(saved_value) = self.config.lock().await.armoury_settings.get(&name) {
|
AttrValue::None
|
||||||
self.attr
|
|
||||||
.set_current_value(&AttrValue::Integer(*saved_value))
|
|
||||||
.map_err(|e| {
|
|
||||||
error!("Could not set {} value: {e:?}", self.attr.name());
|
|
||||||
self.attr.base_path_exists();
|
|
||||||
e
|
|
||||||
})?;
|
|
||||||
info!(
|
|
||||||
"Restored armoury setting {} to {:?}",
|
|
||||||
self.attr.name(),
|
|
||||||
saved_value
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
_ => {
|
||||||
|
info!("Reload called on firmware attribute {name}");
|
||||||
|
match config.armoury_settings.get(&attribute) {
|
||||||
|
Some(saved_value) => AttrValue::Integer(*saved_value),
|
||||||
|
None => AttrValue::None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
self.attr.set_current_value(&apply_value).map_err(|e| {
|
||||||
|
error!("Could not set {} value: {e:?}", self.attr.name());
|
||||||
|
self.attr.base_path_exists();
|
||||||
|
e
|
||||||
|
})?;
|
||||||
|
|
||||||
|
info!(
|
||||||
|
"Restored asus-armoury setting {} to {:?}",
|
||||||
|
self.attr.name(),
|
||||||
|
apply_value
|
||||||
|
);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// If return is `-1` on a property then there is avilable value for that
|
/// If return is `-1` on a property then there is available value for that
|
||||||
/// property
|
/// property
|
||||||
#[interface(name = "xyz.ljones.AsusArmoury")]
|
#[interface(name = "xyz.ljones.AsusArmoury")]
|
||||||
impl AsusArmouryAttribute {
|
impl AsusArmouryAttribute {
|
||||||
@@ -279,7 +277,7 @@ impl AsusArmouryAttribute {
|
|||||||
|
|
||||||
async fn restore_default(&self) -> fdo::Result<()> {
|
async fn restore_default(&self) -> fdo::Result<()> {
|
||||||
self.attr.restore_default()?;
|
self.attr.restore_default()?;
|
||||||
if self.name().is_ppt() || self.name().is_dgpu() {
|
if self.name().property_type() == FirmwareAttributeType::Ppt {
|
||||||
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
||||||
let power_plugged = self
|
let power_plugged = self
|
||||||
.power
|
.power
|
||||||
@@ -338,7 +336,7 @@ impl AsusArmouryAttribute {
|
|||||||
|
|
||||||
#[zbus(property)]
|
#[zbus(property)]
|
||||||
async fn current_value(&self) -> fdo::Result<i32> {
|
async fn current_value(&self) -> fdo::Result<i32> {
|
||||||
if self.name().is_ppt() || self.name().is_dgpu() {
|
if self.name().property_type() == FirmwareAttributeType::Ppt {
|
||||||
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
||||||
let power_plugged = self
|
let power_plugged = self
|
||||||
.power
|
.power
|
||||||
@@ -373,68 +371,64 @@ impl AsusArmouryAttribute {
|
|||||||
|
|
||||||
#[zbus(property)]
|
#[zbus(property)]
|
||||||
async fn set_current_value(&mut self, value: i32) -> fdo::Result<()> {
|
async fn set_current_value(&mut self, value: i32) -> fdo::Result<()> {
|
||||||
if self.name().is_ppt() || self.name().is_dgpu() {
|
let name = self.attr.name();
|
||||||
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
let apply_value = match self.name().property_type() {
|
||||||
let power_plugged = self
|
FirmwareAttributeType::Ppt => {
|
||||||
.power
|
let profile: PlatformProfile = self.platform.get_platform_profile()?.into();
|
||||||
.get_online()
|
let power_plugged = self
|
||||||
.map_err(|e| {
|
.power
|
||||||
error!("Could not get power status: {e:?}");
|
.get_online()
|
||||||
e
|
|
||||||
})
|
|
||||||
.unwrap_or_default();
|
|
||||||
|
|
||||||
let mut config = self.config.lock().await;
|
|
||||||
let tuning = config.select_tunings(power_plugged == 1, profile);
|
|
||||||
|
|
||||||
if let Some(tune) = tuning.group.get_mut(&self.name()) {
|
|
||||||
*tune = value;
|
|
||||||
} else {
|
|
||||||
tuning.group.insert(self.name(), value);
|
|
||||||
debug!("Store tuning config for {} = {:?}", self.attr.name(), value);
|
|
||||||
}
|
|
||||||
if tuning.enabled {
|
|
||||||
self.attr
|
|
||||||
.set_current_value(&AttrValue::Integer(value))
|
|
||||||
.map_err(|e| {
|
.map_err(|e| {
|
||||||
error!("Could not set value: {e:?}");
|
error!("Could not get power status: {e:?}");
|
||||||
e
|
e
|
||||||
})?;
|
})
|
||||||
}
|
.unwrap_or_default();
|
||||||
} else {
|
|
||||||
self.attr
|
|
||||||
.set_current_value(&AttrValue::Integer(value))
|
|
||||||
.map_err(|e| {
|
|
||||||
error!("Could not set value: {e:?}");
|
|
||||||
e
|
|
||||||
})?;
|
|
||||||
|
|
||||||
let has_attr = self
|
let mut config = self.config.lock().await;
|
||||||
.config
|
let tuning = config.select_tunings(power_plugged == 1, profile);
|
||||||
.lock()
|
|
||||||
.await
|
if let Some(tune) = tuning.group.get_mut(&self.name()) {
|
||||||
.armoury_settings
|
*tune = value;
|
||||||
.contains_key(&self.name());
|
} else {
|
||||||
if has_attr {
|
tuning.group.insert(self.name(), value);
|
||||||
if let Some(setting) = self
|
debug!("Store tuning config for {name} = {:?}", value);
|
||||||
.config
|
}
|
||||||
.lock()
|
|
||||||
.await
|
match tuning.enabled {
|
||||||
.armoury_settings
|
true => {
|
||||||
.get_mut(&self.name())
|
debug!("Tuning is enabled: setting value to PPT property {name} = {value}");
|
||||||
{
|
AttrValue::Integer(value)
|
||||||
*setting = value
|
}
|
||||||
|
false => {
|
||||||
|
warn!("Tuning is disabled: skipping setting value to PPT property {name}");
|
||||||
|
AttrValue::None
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
debug!("Adding config for {}", self.attr.name());
|
|
||||||
self.config
|
|
||||||
.lock()
|
|
||||||
.await
|
|
||||||
.armoury_settings
|
|
||||||
.insert(self.name(), value);
|
|
||||||
debug!("Set config for {} = {:?}", self.attr.name(), value);
|
|
||||||
}
|
}
|
||||||
}
|
_ => {
|
||||||
|
let mut settings = self.config.lock().await;
|
||||||
|
settings
|
||||||
|
.armoury_settings
|
||||||
|
.entry(self.name())
|
||||||
|
.and_modify(|setting| {
|
||||||
|
debug!("Set config for {name} = {value}");
|
||||||
|
*setting = value;
|
||||||
|
})
|
||||||
|
.or_insert_with(|| {
|
||||||
|
debug!("Adding config for {name} = {value}");
|
||||||
|
value
|
||||||
|
});
|
||||||
|
|
||||||
|
AttrValue::Integer(value)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
self.attr.set_current_value(&apply_value).map_err(|e| {
|
||||||
|
error!("Could not set value {value} to attribute {name}: {e:?}");
|
||||||
|
e
|
||||||
|
})?;
|
||||||
|
|
||||||
|
// write config after setting value
|
||||||
self.config.lock().await.write();
|
self.config.lock().await.write();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@@ -501,7 +495,7 @@ pub async fn set_config_or_default(
|
|||||||
) {
|
) {
|
||||||
for attr in attrs.attributes().iter() {
|
for attr in attrs.attributes().iter() {
|
||||||
let name: FirmwareAttribute = attr.name().into();
|
let name: FirmwareAttribute = attr.name().into();
|
||||||
if name.is_ppt() || name.is_dgpu() {
|
if name.property_type() == FirmwareAttributeType::Ppt {
|
||||||
let tuning = config.select_tunings(power_plugged, profile);
|
let tuning = config.select_tunings(power_plugged, profile);
|
||||||
if !tuning.enabled {
|
if !tuning.enabled {
|
||||||
debug!("Tuning group is not enabled, skipping");
|
debug!("Tuning group is not enabled, skipping");
|
||||||
|
|||||||
@@ -71,7 +71,13 @@ impl AuraZbus {
|
|||||||
#[zbus(property)]
|
#[zbus(property)]
|
||||||
async fn set_brightness(&mut self, brightness: LedBrightness) -> Result<(), ZbErr> {
|
async fn set_brightness(&mut self, brightness: LedBrightness) -> Result<(), ZbErr> {
|
||||||
if let Some(bl) = self.0.backlight.as_ref() {
|
if let Some(bl) = self.0.backlight.as_ref() {
|
||||||
return Ok(bl.lock().await.set_brightness(brightness.into())?);
|
let res = bl.lock().await.set_brightness(brightness.into());
|
||||||
|
if res.is_ok() {
|
||||||
|
let mut config = self.0.config.lock().await;
|
||||||
|
config.brightness = brightness;
|
||||||
|
config.write();
|
||||||
|
}
|
||||||
|
return Ok(res?);
|
||||||
}
|
}
|
||||||
Err(ZbErr::Failed("No sysfs brightness control".to_string()))
|
Err(ZbErr::Failed("No sysfs brightness control".to_string()))
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,6 +8,12 @@ use serde::{Deserialize, Serialize};
|
|||||||
|
|
||||||
const CONFIG_FILE: &str = "asusd.ron";
|
const CONFIG_FILE: &str = "asusd.ron";
|
||||||
|
|
||||||
|
/// Default value for base_charge_control_end_threshold when not present in config.
|
||||||
|
/// Returns 0 so restore_charge_limit() skips restoration for upgraded configs.
|
||||||
|
fn default_base_charge_limit() -> u8 {
|
||||||
|
0
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Default, Clone, Deserialize, Serialize, PartialEq)]
|
#[derive(Default, Clone, Deserialize, Serialize, PartialEq)]
|
||||||
pub struct Tuning {
|
pub struct Tuning {
|
||||||
pub enabled: bool,
|
pub enabled: bool,
|
||||||
@@ -19,8 +25,8 @@ type Tunings = HashMap<PlatformProfile, Tuning>;
|
|||||||
pub struct Config {
|
pub struct Config {
|
||||||
// The current charge limit applied
|
// The current charge limit applied
|
||||||
pub charge_control_end_threshold: u8,
|
pub charge_control_end_threshold: u8,
|
||||||
/// Save charge limit for restoring
|
/// Save charge limit for restoring after one-shot full charge
|
||||||
#[serde(skip)]
|
#[serde(default = "default_base_charge_limit")]
|
||||||
pub base_charge_control_end_threshold: u8,
|
pub base_charge_control_end_threshold: u8,
|
||||||
pub disable_nvidia_powerd_on_battery: bool,
|
pub disable_nvidia_powerd_on_battery: bool,
|
||||||
/// An optional command/script to run when power is changed to AC
|
/// An optional command/script to run when power is changed to AC
|
||||||
@@ -86,6 +92,9 @@ impl Default for Config {
|
|||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Self {
|
Self {
|
||||||
charge_control_end_threshold: 100,
|
charge_control_end_threshold: 100,
|
||||||
|
// NOTE: This is intentionally 100 (not 0 like the serde default).
|
||||||
|
// New installs get 100 (no limit). Upgraded configs missing this
|
||||||
|
// field get 0 via serde, which skips restore_charge_limit().
|
||||||
base_charge_control_end_threshold: 100,
|
base_charge_control_end_threshold: 100,
|
||||||
disable_nvidia_powerd_on_battery: true,
|
disable_nvidia_powerd_on_battery: true,
|
||||||
ac_command: Default::default(),
|
ac_command: Default::default(),
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ use std::sync::Arc;
|
|||||||
|
|
||||||
use config_traits::StdConfig;
|
use config_traits::StdConfig;
|
||||||
use log::{debug, error, info, warn};
|
use log::{debug, error, info, warn};
|
||||||
use rog_platform::asus_armoury::{AttrValue, FirmwareAttribute, FirmwareAttributes};
|
use rog_platform::asus_armoury::{
|
||||||
|
AttrValue, FirmwareAttribute, FirmwareAttributeType, FirmwareAttributes,
|
||||||
|
};
|
||||||
use rog_platform::cpu::{CPUControl, CPUGovernor, CPUEPP};
|
use rog_platform::cpu::{CPUControl, CPUGovernor, CPUEPP};
|
||||||
use rog_platform::platform::{PlatformProfile, Properties, RogPlatform};
|
use rog_platform::platform::{PlatformProfile, Properties, RogPlatform};
|
||||||
use rog_platform::power::AsusPower;
|
use rog_platform::power::AsusPower;
|
||||||
@@ -459,8 +461,18 @@ impl CtrlPlatform {
|
|||||||
#[zbus(signal_context)] ctxt: SignalEmitter<'_>,
|
#[zbus(signal_context)] ctxt: SignalEmitter<'_>,
|
||||||
policy: PlatformProfile,
|
policy: PlatformProfile,
|
||||||
) -> Result<(), FdoErr> {
|
) -> Result<(), FdoErr> {
|
||||||
self.config.lock().await.platform_profile_on_battery = policy;
|
// If the requested profile isn't available on this platform, and it's
|
||||||
self.set_platform_profile(ctxt, policy).await?;
|
// `Quiet`, fall back to `LowPower` so we don't write an unavailable
|
||||||
|
// profile into the config file.
|
||||||
|
let mut chosen = policy;
|
||||||
|
if let Ok(choices) = self.platform.get_platform_profile_choices() {
|
||||||
|
if chosen == PlatformProfile::Quiet && !choices.contains(&PlatformProfile::Quiet) {
|
||||||
|
chosen = PlatformProfile::LowPower;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self.config.lock().await.platform_profile_on_battery = chosen;
|
||||||
|
self.set_platform_profile(ctxt, chosen).await?;
|
||||||
self.config.lock().await.write();
|
self.config.lock().await.write();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@@ -488,8 +500,16 @@ impl CtrlPlatform {
|
|||||||
#[zbus(signal_context)] ctxt: SignalEmitter<'_>,
|
#[zbus(signal_context)] ctxt: SignalEmitter<'_>,
|
||||||
policy: PlatformProfile,
|
policy: PlatformProfile,
|
||||||
) -> Result<(), FdoErr> {
|
) -> Result<(), FdoErr> {
|
||||||
self.config.lock().await.platform_profile_on_ac = policy;
|
// Mirror the same fallback behavior for AC profile changes.
|
||||||
self.set_platform_profile(ctxt, policy).await?;
|
let mut chosen = policy;
|
||||||
|
if let Ok(choices) = self.platform.get_platform_profile_choices() {
|
||||||
|
if chosen == PlatformProfile::Quiet && !choices.contains(&PlatformProfile::Quiet) {
|
||||||
|
chosen = PlatformProfile::LowPower;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self.config.lock().await.platform_profile_on_ac = chosen;
|
||||||
|
self.set_platform_profile(ctxt, chosen).await?;
|
||||||
self.config.lock().await.write();
|
self.config.lock().await.write();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@@ -599,7 +619,7 @@ impl CtrlPlatform {
|
|||||||
|
|
||||||
for attr in self.attributes.attributes() {
|
for attr in self.attributes.attributes() {
|
||||||
let name: FirmwareAttribute = attr.name().into();
|
let name: FirmwareAttribute = attr.name().into();
|
||||||
if name.is_ppt() {
|
if name.property_type() == FirmwareAttributeType::Ppt {
|
||||||
// reset stored value
|
// reset stored value
|
||||||
if let Some(tune) = self
|
if let Some(tune) = self
|
||||||
.config
|
.config
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ Environment=IS_SERVICE=1
|
|||||||
# Reduce noisy span logs while keeping useful debug info for asusd and related crates.
|
# Reduce noisy span logs while keeping useful debug info for asusd and related crates.
|
||||||
# Keep global level at info but allow debug for our crates; silence tracing::span (very noisy)
|
# Keep global level at info but allow debug for our crates; silence tracing::span (very noisy)
|
||||||
# RUST_LOG format: <module>=<level>,... (levels: error,warn,info,debug,trace)
|
# RUST_LOG format: <module>=<level>,... (levels: error,warn,info,debug,trace)
|
||||||
Environment=RUST_LOG="info,asusd=debug,rog_platform=debug,tracing::span=error"
|
Environment=RUST_LOG="info,asusd=debug,rog_platform=debug,tracing::span=error,zbus::object_server=error,zbus::connection::handshake::common=error,zbus::connection::handshake::client=error"
|
||||||
# required to prevent init issues with hid_asus and MCU
|
# required to prevent init issues with hid_asus and MCU
|
||||||
ExecStartPre=/bin/sleep 1
|
ExecStartPre=/bin/sleep 1
|
||||||
ExecStart=/usr/bin/asusd
|
ExecStart=/usr/bin/asusd
|
||||||
|
|||||||
@@ -20,7 +20,7 @@
|
|||||||
%global debug_package %{nil}
|
%global debug_package %{nil}
|
||||||
%endif
|
%endif
|
||||||
|
|
||||||
%define version 6.2.0
|
%define version 6.3.1
|
||||||
%define specrelease %{?dist}
|
%define specrelease %{?dist}
|
||||||
%define pkg_release 1%{specrelease}
|
%define pkg_release 1%{specrelease}
|
||||||
|
|
||||||
@@ -147,6 +147,15 @@ install -D -m 0644 LICENSE %{buildroot}%{_datadir}/asusctl/LICENSE
|
|||||||
|
|
||||||
desktop-file-validate %{buildroot}%{_datadir}/applications/rog-control-center.desktop
|
desktop-file-validate %{buildroot}%{_datadir}/applications/rog-control-center.desktop
|
||||||
|
|
||||||
|
%post
|
||||||
|
%systemd_post asusd.service
|
||||||
|
|
||||||
|
%preun
|
||||||
|
%systemd_preun asusd.service
|
||||||
|
|
||||||
|
%postun
|
||||||
|
%systemd_postun_with_restart asusd.service
|
||||||
|
|
||||||
%files
|
%files
|
||||||
%license LICENSE
|
%license LICENSE
|
||||||
%{_bindir}/asusd
|
%{_bindir}/asusd
|
||||||
|
|||||||
23
extra/index.html
Normal file
23
extra/index.html
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta name="viewport" content="width=device-width,initial-scale=1">
|
||||||
|
<title>asusctl docs</title>
|
||||||
|
<!-- Redirect to the generated crate docs -->
|
||||||
|
<meta http-equiv="refresh" content="0;url=asusctl/index.html">
|
||||||
|
<link rel="canonical" href="asusctl/index.html">
|
||||||
|
<style>
|
||||||
|
body { font-family: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial; color:#222; display:flex; align-items:center; justify-content:center; height:100vh; margin:0 }
|
||||||
|
.box { text-align:center }
|
||||||
|
a { color: #0366d6 }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="box">
|
||||||
|
<h1>asusctl documentation</h1>
|
||||||
|
<p>Redirecting to the generated docs — if your browser doesn't redirect automatically, <a href="asusctl/index.html">click here</a>.</p>
|
||||||
|
<p>If you expected a different landing page, update <code>extra/index.html</code> accordingly.</p>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
0
rog-anime/data/g835l/images/.gitkeep
Normal file
0
rog-anime/data/g835l/images/.gitkeep
Normal file
BIN
rog-anime/data/g835l/templates/custom-image-template.png
Normal file
BIN
rog-anime/data/g835l/templates/custom-image-template.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 518 B |
@@ -64,7 +64,7 @@ pub enum AnimeType {
|
|||||||
GA402,
|
GA402,
|
||||||
GU604,
|
GU604,
|
||||||
G635L,
|
G635L,
|
||||||
G835LW,
|
G835L,
|
||||||
#[default]
|
#[default]
|
||||||
Unsupported,
|
Unsupported,
|
||||||
}
|
}
|
||||||
@@ -73,14 +73,21 @@ impl FromStr for AnimeType {
|
|||||||
type Err = AnimeError;
|
type Err = AnimeError;
|
||||||
|
|
||||||
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
|
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
|
||||||
Ok(match s.to_uppercase().as_str() {
|
let dmi = s.to_uppercase();
|
||||||
"GA401" => Self::GA401,
|
|
||||||
"GA402" => Self::GA402,
|
if dmi.contains("GA401") {
|
||||||
"GU604" => Self::GU604,
|
return Ok(Self::GA401);
|
||||||
"G635L" => Self::G635L,
|
} else if dmi.contains("GA402") {
|
||||||
"G835LW" => Self::G835LW,
|
return Ok(Self::GA402);
|
||||||
_ => Self::Unsupported,
|
} else if dmi.contains("GU604") {
|
||||||
})
|
return Ok(Self::GU604);
|
||||||
|
} else if dmi.contains("G635L") {
|
||||||
|
return Ok(Self::G635L);
|
||||||
|
} else if dmi.contains("G835L") {
|
||||||
|
return Ok(Self::G835L);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Self::Unsupported)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -93,8 +100,10 @@ impl AnimeType {
|
|||||||
AnimeType::GA402
|
AnimeType::GA402
|
||||||
} else if board_name.contains("GU604V") {
|
} else if board_name.contains("GU604V") {
|
||||||
AnimeType::GU604
|
AnimeType::GU604
|
||||||
} else if board_name.contains("G635L") || board_name.contains("G635L") {
|
} else if board_name.contains("G635L") {
|
||||||
AnimeType::G635L
|
AnimeType::G635L
|
||||||
|
} else if board_name.contains("G835L") {
|
||||||
|
AnimeType::G835L
|
||||||
} else {
|
} else {
|
||||||
AnimeType::Unsupported
|
AnimeType::Unsupported
|
||||||
}
|
}
|
||||||
@@ -104,7 +113,9 @@ impl AnimeType {
|
|||||||
pub fn width(&self) -> usize {
|
pub fn width(&self) -> usize {
|
||||||
match self {
|
match self {
|
||||||
AnimeType::GU604 => 70,
|
AnimeType::GU604 => 70,
|
||||||
AnimeType::G835LW => 74,
|
// TODO: Find G635L W*H
|
||||||
|
AnimeType::G635L => 68,
|
||||||
|
AnimeType::G835L => 68,
|
||||||
_ => 74,
|
_ => 74,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -114,7 +125,8 @@ impl AnimeType {
|
|||||||
match self {
|
match self {
|
||||||
AnimeType::GA401 => 36,
|
AnimeType::GA401 => 36,
|
||||||
AnimeType::GU604 => 43,
|
AnimeType::GU604 => 43,
|
||||||
AnimeType::G835LW => 39,
|
AnimeType::G635L => 34,
|
||||||
|
AnimeType::G835L => 34,
|
||||||
_ => 39,
|
_ => 39,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -123,8 +135,9 @@ impl AnimeType {
|
|||||||
pub fn data_length(&self) -> usize {
|
pub fn data_length(&self) -> usize {
|
||||||
match self {
|
match self {
|
||||||
AnimeType::GA401 => PANE_LEN * 2,
|
AnimeType::GA401 => PANE_LEN * 2,
|
||||||
AnimeType::GU604 => PANE_LEN * 3,
|
// G835L has 810 LEDs: 210 (triangle) + 600 (staggered rectangle)
|
||||||
AnimeType::G835LW => PANE_LEN * 3,
|
AnimeType::G635L => 810, // TODO: This is provisional until we have a G635L to test on
|
||||||
|
AnimeType::G835L => 810,
|
||||||
_ => PANE_LEN * 3,
|
_ => PANE_LEN * 3,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -188,29 +201,35 @@ impl TryFrom<AnimeDataBuffer> for AnimePacketType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let mut buffers = match anime.anime {
|
let mut buffers = match anime.anime {
|
||||||
AnimeType::GA401 => vec![[0; 640]; 2],
|
AnimeType::GA401 | AnimeType::G635L | AnimeType::G835L => vec![[0; 640]; 2],
|
||||||
AnimeType::GA402
|
AnimeType::GA402 | AnimeType::GU604 | AnimeType::Unsupported => {
|
||||||
| AnimeType::GU604
|
|
||||||
| AnimeType::G635L
|
|
||||||
| AnimeType::G835LW
|
|
||||||
| AnimeType::Unsupported => {
|
|
||||||
vec![[0; 640]; 3]
|
vec![[0; 640]; 3]
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
for (idx, chunk) in anime.data.as_slice().chunks(PANE_LEN).enumerate() {
|
// G835L has different packing: 627 bytes in pane 1, 183 bytes in pane 2
|
||||||
buffers[idx][BLOCK_START..BLOCK_END].copy_from_slice(chunk);
|
if anime.anime == AnimeType::G835L || anime.anime == AnimeType::G635L {
|
||||||
|
let data = anime.data.as_slice();
|
||||||
|
// Pane 1: first 627 bytes
|
||||||
|
let pane1_len = PANE_LEN.min(data.len());
|
||||||
|
buffers[0][BLOCK_START..BLOCK_START + pane1_len].copy_from_slice(&data[..pane1_len]);
|
||||||
|
// Pane 2: remaining bytes (183)
|
||||||
|
if data.len() > PANE_LEN {
|
||||||
|
let pane2_len = data.len() - PANE_LEN;
|
||||||
|
buffers[1][BLOCK_START..BLOCK_START + pane2_len].copy_from_slice(&data[PANE_LEN..]);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (idx, chunk) in anime.data.as_slice().chunks(PANE_LEN).enumerate() {
|
||||||
|
buffers[idx][BLOCK_START..BLOCK_START + chunk.len()].copy_from_slice(chunk);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
buffers[0][..7].copy_from_slice(&USB_PREFIX1);
|
buffers[0][..7].copy_from_slice(&USB_PREFIX1);
|
||||||
buffers[1][..7].copy_from_slice(&USB_PREFIX2);
|
buffers[1][..7].copy_from_slice(&USB_PREFIX2);
|
||||||
|
|
||||||
if matches!(
|
if matches!(
|
||||||
anime.anime,
|
anime.anime,
|
||||||
AnimeType::GA402
|
AnimeType::GA402 | AnimeType::GU604 | AnimeType::Unsupported
|
||||||
| AnimeType::GU604
|
|
||||||
| AnimeType::G635L
|
|
||||||
| AnimeType::G835LW
|
|
||||||
| AnimeType::Unsupported
|
|
||||||
) {
|
) {
|
||||||
buffers[2][..7].copy_from_slice(&USB_PREFIX3);
|
buffers[2][..7].copy_from_slice(&USB_PREFIX3);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -146,6 +146,8 @@ impl AnimeDiagonal {
|
|||||||
match anime_type {
|
match anime_type {
|
||||||
AnimeType::GA401 => self.to_ga401_packets(),
|
AnimeType::GA401 => self.to_ga401_packets(),
|
||||||
AnimeType::GU604 => self.to_gu604_packets(),
|
AnimeType::GU604 => self.to_gu604_packets(),
|
||||||
|
AnimeType::G635L => self.to_g835l_packets(), // TODO: Verify with G635L model
|
||||||
|
AnimeType::G835L => self.to_g835l_packets(),
|
||||||
_ => self.to_ga402_packets(),
|
_ => self.to_ga402_packets(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -381,4 +383,80 @@ impl AnimeDiagonal {
|
|||||||
|
|
||||||
AnimeDataBuffer::from_vec(crate::AnimeType::GA402, buf)
|
AnimeDataBuffer::from_vec(crate::AnimeType::GA402, buf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// G835L diagonal packing - inverted geometry (rows grow then constant)
|
||||||
|
/// Triangle (rows 0-27): pairs grow from 1→14 LEDs
|
||||||
|
/// Rectangle (rows 28-67): constant 15 LEDs
|
||||||
|
///
|
||||||
|
/// Diagonal PNG layout for G835L:
|
||||||
|
/// - Image height = 34 (row pairs)
|
||||||
|
/// - Image width = 68 (half-step X grid)
|
||||||
|
/// - Even/odd rows are interleaved in X (staggered by 0.5 LED = 1 px)
|
||||||
|
fn to_g835l_packets(&self) -> Result<AnimeDataBuffer> {
|
||||||
|
use log::debug;
|
||||||
|
|
||||||
|
let mut buf = vec![0u8; AnimeType::G835L.data_length()];
|
||||||
|
let mut buf_idx = 0usize;
|
||||||
|
|
||||||
|
debug!(
|
||||||
|
"G835L packing: image dimensions {}x{}, buffer size {}",
|
||||||
|
self.1.first().map(|r| r.len()).unwrap_or(0),
|
||||||
|
self.1.len(),
|
||||||
|
buf.len()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Helper: get row length for G835L
|
||||||
|
fn row_length(row: usize) -> usize {
|
||||||
|
if row < 28 {
|
||||||
|
row / 2 + 1
|
||||||
|
} else {
|
||||||
|
15
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper: starting X (in LED units) for the row
|
||||||
|
fn first_x(row: usize) -> usize {
|
||||||
|
if row < 28 {
|
||||||
|
0
|
||||||
|
} else {
|
||||||
|
(row - 28) / 2
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Process all 68 rows
|
||||||
|
for row in 0..68 {
|
||||||
|
let len = row_length(row);
|
||||||
|
let img_y = row / 2;
|
||||||
|
let base_x = first_x(row);
|
||||||
|
let stagger = row % 2;
|
||||||
|
|
||||||
|
for i in 0..len {
|
||||||
|
// Half-step X grid: even rows on even pixels, odd rows on odd pixels.
|
||||||
|
let img_x = (base_x + i) * 2 + stagger;
|
||||||
|
|
||||||
|
// Read from image, clamp to bounds
|
||||||
|
let val = if img_y < self.1.len() && img_x < self.1[img_y].len() {
|
||||||
|
self.1[img_y][img_x]
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
};
|
||||||
|
|
||||||
|
// Log first LED of each row for debugging
|
||||||
|
if i == 0 {
|
||||||
|
debug!(
|
||||||
|
"Row {}: len={}, first LED at img[{}][{}] = {}",
|
||||||
|
row, len, img_y, img_x, val
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if buf_idx < buf.len() {
|
||||||
|
buf[buf_idx] = val;
|
||||||
|
}
|
||||||
|
buf_idx += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
debug!("G835L packing complete: {} bytes written", buf_idx);
|
||||||
|
AnimeDataBuffer::from_vec(AnimeType::G835L, buf)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -137,8 +137,8 @@ impl AnimeImage {
|
|||||||
fn scale_y(anime_type: AnimeType) -> f32 {
|
fn scale_y(anime_type: AnimeType) -> f32 {
|
||||||
match anime_type {
|
match anime_type {
|
||||||
AnimeType::GA401 => 0.3,
|
AnimeType::GA401 => 0.3,
|
||||||
AnimeType::GU604 => 0.28,
|
AnimeType::GA402 => 0.283,
|
||||||
_ => 0.283,
|
_ => 0.28,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -149,6 +149,7 @@ impl AnimeImage {
|
|||||||
/// square grid, so `first_x` is the x position on that grid where the
|
/// square grid, so `first_x` is the x position on that grid where the
|
||||||
/// LED is actually positioned in relation to the Y.
|
/// LED is actually positioned in relation to the Y.
|
||||||
///
|
///
|
||||||
|
/// For GA401/GA402/GU604 (shrinking pattern - diagonal cuts in from left):
|
||||||
/// ```text
|
/// ```text
|
||||||
/// +------------+
|
/// +------------+
|
||||||
/// | |
|
/// | |
|
||||||
@@ -162,6 +163,19 @@ impl AnimeImage {
|
|||||||
/// ^ ------+
|
/// ^ ------+
|
||||||
/// first_x
|
/// first_x
|
||||||
/// ```
|
/// ```
|
||||||
|
///
|
||||||
|
/// For G835L (inverted pattern - triangle grows then rectangle shifts):
|
||||||
|
/// ```text
|
||||||
|
/// ● <- Row 0: first_x = 0, width = 1
|
||||||
|
/// ● <- Row 1: first_x = 0 (stagger), width = 1
|
||||||
|
/// ● ● <- Row 2: first_x = 0, width = 2
|
||||||
|
/// ● ● <- Row 3: first_x = 0 (stagger), width = 2
|
||||||
|
/// ... <- Triangle continues, first_x = 0 for rows 0-27
|
||||||
|
/// ● ● ● ● ● ● ● ● ● ● ● ● ● ● ● <- Row 28+: first_x grows
|
||||||
|
/// ● ● ● ● ● ● ● ● ● ● ● ● ● ● ● <- Rectangle shifts right
|
||||||
|
/// ```
|
||||||
|
/// Triangle (rows 0-27): first_x = 0 (no cumulative shift)
|
||||||
|
/// Rectangle (rows 28-67): first_x = (y - 28) / 2 (shifts right)
|
||||||
fn first_x(anime_type: AnimeType, y: u32) -> u32 {
|
fn first_x(anime_type: AnimeType, y: u32) -> u32 {
|
||||||
match anime_type {
|
match anime_type {
|
||||||
AnimeType::GA401 => {
|
AnimeType::GA401 => {
|
||||||
@@ -179,6 +193,16 @@ impl AnimeImage {
|
|||||||
// and then their offset grows by one every two rows
|
// and then their offset grows by one every two rows
|
||||||
(y - 9) / 2
|
(y - 9) / 2
|
||||||
}
|
}
|
||||||
|
AnimeType::G835L => {
|
||||||
|
// G835L has inverted geometry - triangle at top-left, rectangle shifts right
|
||||||
|
// Triangle (rows 0-27): no cumulative shift, just alternating stagger
|
||||||
|
// Rectangle (rows 28-67): shifts right by ~0.5px per row
|
||||||
|
if y < 28 {
|
||||||
|
0
|
||||||
|
} else {
|
||||||
|
(y - 28) / 2
|
||||||
|
}
|
||||||
|
}
|
||||||
_ => {
|
_ => {
|
||||||
// first 11 rows start at zero
|
// first 11 rows start at zero
|
||||||
if y <= 11 {
|
if y <= 11 {
|
||||||
@@ -221,6 +245,16 @@ impl AnimeImage {
|
|||||||
}
|
}
|
||||||
38 - Self::first_x(anime_type, y) + y % 2
|
38 - Self::first_x(anime_type, y) + y % 2
|
||||||
}
|
}
|
||||||
|
AnimeType::G835L => {
|
||||||
|
// G835L rows GROW then stay constant (inverted from other devices)
|
||||||
|
// Triangle (rows 0-27): pairs of rows with same length, 1→14
|
||||||
|
// Rectangle (rows 28-67): constant 15 LEDs
|
||||||
|
if y < 28 {
|
||||||
|
y / 2 + 1
|
||||||
|
} else {
|
||||||
|
15
|
||||||
|
}
|
||||||
|
}
|
||||||
_ => {
|
_ => {
|
||||||
if y <= 11 {
|
if y <= 11 {
|
||||||
return 34;
|
return 34;
|
||||||
@@ -235,8 +269,8 @@ impl AnimeImage {
|
|||||||
match anime_type {
|
match anime_type {
|
||||||
// 33.0 = Longest row LED count (physical) plus half-pixel offset
|
// 33.0 = Longest row LED count (physical) plus half-pixel offset
|
||||||
AnimeType::GA401 => (33.0 + 0.5) * Self::scale_x(anime_type),
|
AnimeType::GA401 => (33.0 + 0.5) * Self::scale_x(anime_type),
|
||||||
|
|
||||||
AnimeType::GU604 => (38.0 + 0.5) * Self::scale_x(anime_type),
|
AnimeType::GU604 => (38.0 + 0.5) * Self::scale_x(anime_type),
|
||||||
|
AnimeType::G835L => (33.0 + 0.5) * Self::scale_x(anime_type),
|
||||||
_ => (35.0 + 0.5) * Self::scale_x(anime_type),
|
_ => (35.0 + 0.5) * Self::scale_x(anime_type),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -246,6 +280,7 @@ impl AnimeImage {
|
|||||||
match anime_type {
|
match anime_type {
|
||||||
AnimeType::GA401 => 55,
|
AnimeType::GA401 => 55,
|
||||||
AnimeType::GU604 => 62,
|
AnimeType::GU604 => 62,
|
||||||
|
AnimeType::G835L => 68,
|
||||||
_ => 61,
|
_ => 61,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -256,6 +291,7 @@ impl AnimeImage {
|
|||||||
// 54.0 = End column LED count (physical) plus one dead pixel
|
// 54.0 = End column LED count (physical) plus one dead pixel
|
||||||
AnimeType::GA401 => (54.0 + 1.0) * Self::scale_y(anime_type),
|
AnimeType::GA401 => (54.0 + 1.0) * Self::scale_y(anime_type),
|
||||||
AnimeType::GU604 => 62.0 * Self::scale_y(anime_type),
|
AnimeType::GU604 => 62.0 * Self::scale_y(anime_type),
|
||||||
|
AnimeType::G835L => 68.0 * Self::scale_y(anime_type),
|
||||||
// GA402 may not have dead pixels and require only the physical LED count
|
// GA402 may not have dead pixels and require only the physical LED count
|
||||||
_ => 61.0 * Self::scale_y(anime_type),
|
_ => 61.0 * Self::scale_y(anime_type),
|
||||||
}
|
}
|
||||||
@@ -269,8 +305,8 @@ impl AnimeImage {
|
|||||||
1 | 3 => 35, // Some rows are padded
|
1 | 3 => 35, // Some rows are padded
|
||||||
_ => 36 - y / 2,
|
_ => 36 - y / 2,
|
||||||
},
|
},
|
||||||
AnimeType::GU604 => AnimeImage::width(anime_type, y),
|
|
||||||
// GA402 does not have padding, equivalent to width
|
// Other devices don't have dead pixels
|
||||||
_ => AnimeImage::width(anime_type, y),
|
_ => AnimeImage::width(anime_type, y),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -405,13 +441,35 @@ impl AnimeImage {
|
|||||||
let transform =
|
let transform =
|
||||||
Mat3::from_scale_angle_translation(self.scale, self.angle, self.translation);
|
Mat3::from_scale_angle_translation(self.scale, self.angle, self.translation);
|
||||||
|
|
||||||
let pos_in_leds = Mat3::from_translation(Vec2::new(20.0, 20.0));
|
let pos_in_leds = Mat3::from_translation(self.led_center());
|
||||||
// Get LED-to-image coords
|
// Get LED-to-image coords
|
||||||
let led_from_px = pos_in_leds * led_from_cm * transform * cm_from_px * center;
|
let led_from_px = pos_in_leds * led_from_cm * transform * cm_from_px * center;
|
||||||
|
|
||||||
led_from_px.inverse()
|
led_from_px.inverse()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn led_center(&self) -> Vec2 {
|
||||||
|
if self.anime_type != AnimeType::G835L {
|
||||||
|
return Vec2::new(20.0, 20.0);
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut min = Vec2::splat(f32::INFINITY);
|
||||||
|
let mut max = Vec2::splat(f32::NEG_INFINITY);
|
||||||
|
for led in self.led_pos.iter().flatten() {
|
||||||
|
let pos = Vec2::new(led.x(), led.y());
|
||||||
|
min = min.min(pos);
|
||||||
|
max = max.max(pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
if min.x.is_finite() {
|
||||||
|
let mut center = (min + max) * 0.5;
|
||||||
|
center.y += 1.0;
|
||||||
|
center
|
||||||
|
} else {
|
||||||
|
Vec2::new(20.0, 20.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Generate the base image from inputs. The result can be displayed as is
|
/// Generate the base image from inputs. The result can be displayed as is
|
||||||
/// or updated via scale, position, or angle then displayed again after
|
/// or updated via scale, position, or angle then displayed again after
|
||||||
/// `update()`.
|
/// `update()`.
|
||||||
|
|||||||
@@ -253,6 +253,8 @@ pub fn get_anime_type() -> AnimeType {
|
|||||||
AnimeType::GU604
|
AnimeType::GU604
|
||||||
} else if board_name.contains("G635L") {
|
} else if board_name.contains("G635L") {
|
||||||
AnimeType::G635L
|
AnimeType::G635L
|
||||||
|
} else if board_name.contains("G835L") {
|
||||||
|
AnimeType::G835L
|
||||||
} else {
|
} else {
|
||||||
AnimeType::Unsupported
|
AnimeType::Unsupported
|
||||||
}
|
}
|
||||||
|
|||||||
BIN
rog-anime/tests/data/g835l-diagonal-fullbright.png
Normal file
BIN
rog-anime/tests/data/g835l-diagonal-fullbright.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 205 B |
BIN
rog-anime/tests/data/g835l-diagonal.gif
Normal file
BIN
rog-anime/tests/data/g835l-diagonal.gif
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 87 KiB |
BIN
rog-anime/tests/data/g835l-diagonal.png
Normal file
BIN
rog-anime/tests/data/g835l-diagonal.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 156 B |
556
rog-anime/tests/g835l.rs
Normal file
556
rog-anime/tests/g835l.rs
Normal file
@@ -0,0 +1,556 @@
|
|||||||
|
// TODO: This is a provisional copy paste of GA401
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use std::path::PathBuf;
|
||||||
|
|
||||||
|
use rog_anime::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn g835l_image_edge_packet_check() {
|
||||||
|
let pkt0_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x01, 0x00, 0x73, 0x02, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0xff, 0xff, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
let pkt1_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x74, 0x02, 0x73, 0x02, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
|
||||||
|
let mut matrix = AnimeImage::new(
|
||||||
|
Vec2::new(1.0, 1.0),
|
||||||
|
0.0,
|
||||||
|
Vec2::default(),
|
||||||
|
0.0,
|
||||||
|
vec![Pixel::default(); 1000],
|
||||||
|
100,
|
||||||
|
AnimeType::G835L,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
matrix.edge_outline();
|
||||||
|
let data = AnimeDataBuffer::try_from(&matrix).unwrap();
|
||||||
|
let pkt = AnimePacketType::try_from(data).unwrap();
|
||||||
|
|
||||||
|
assert_eq!(pkt[0], pkt0_check);
|
||||||
|
assert_eq!(pkt[1], pkt1_check);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn g835l_diagonal_packet_check() {
|
||||||
|
let pkt0_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x01, 0x00, 0x73, 0x02, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0xff, 0xff, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
let pkt1_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x74, 0x02, 0x73, 0x02, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
|
||||||
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
|
||||||
|
let mut path = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
||||||
|
path.push("tests/data/g835l-diagonal.png");
|
||||||
|
|
||||||
|
let matrix = AnimeDiagonal::from_png(&path, None, 255.0, AnimeType::G835L).unwrap();
|
||||||
|
let data = matrix.into_data_buffer(AnimeType::G835L).unwrap();
|
||||||
|
let pkt = AnimePacketType::try_from(data).unwrap();
|
||||||
|
|
||||||
|
assert_eq!(pkt[0], pkt0_check);
|
||||||
|
assert_eq!(pkt[1], pkt1_check);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn g835l_diagonal_fullbright_packet_check() {
|
||||||
|
let pkt0_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x01, 0x00, 0x73, 0x02, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
let pkt1_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x74, 0x02, 0x73, 0x02, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
||||||
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
|
||||||
|
let mut path = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
||||||
|
path.push("tests/data/g835l-diagonal-fullbright.png");
|
||||||
|
|
||||||
|
let matrix = AnimeDiagonal::from_png(&path, None, 255.0, AnimeType::G835L).unwrap();
|
||||||
|
let data = matrix.into_data_buffer(AnimeType::G835L).unwrap();
|
||||||
|
let pkt = AnimePacketType::try_from(data).unwrap();
|
||||||
|
|
||||||
|
assert_eq!(pkt[0], pkt0_check);
|
||||||
|
assert_eq!(pkt[1], pkt1_check);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn g835l_diagonal_gif_wave_packet_check() {
|
||||||
|
let pkt0_frame0_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x01, 0x00, 0x73, 0x02, 0x7f, 0x92, 0x84, 0xa2, 0x95, 0xac, 0x8a,
|
||||||
|
0xa0, 0xaf, 0x97, 0xa4, 0xa8, 0x91, 0x9c, 0x9e, 0x99, 0x98, 0x98, 0x91, 0x82, 0x99,
|
||||||
|
0x97, 0x8d, 0x7c, 0x67, 0x99, 0x8d, 0x7a, 0x63, 0x4a, 0xa0, 0x92, 0x7d, 0x63, 0x48,
|
||||||
|
0x2e, 0x9a, 0x83, 0x67, 0x4a, 0x2e, 0x17, 0xa6, 0x8d, 0x70, 0x50, 0x32, 0x19, 0x08,
|
||||||
|
0x9a, 0x7b, 0x5a, 0x3b, 0x20, 0x0c, 0x02, 0xa8, 0x89, 0x67, 0x46, 0x2a, 0x15, 0x09,
|
||||||
|
0x07, 0x97, 0x75, 0x54, 0x37, 0x21, 0x13, 0x10, 0x16, 0xa6, 0x84, 0x63, 0x45, 0x2e,
|
||||||
|
0x20, 0x1c, 0x21, 0x2e, 0x91, 0x71, 0x54, 0x3d, 0x2f, 0x2a, 0x2e, 0x3a, 0x4c, 0x9d,
|
||||||
|
0x7d, 0x61, 0x4c, 0x3e, 0x39, 0x3d, 0x49, 0x5a, 0x6d, 0x87, 0x6d, 0x58, 0x4c, 0x48,
|
||||||
|
0x4c, 0x58, 0x69, 0x7c, 0x8e, 0x8e, 0x75, 0x63, 0x58, 0x55, 0x5a, 0x67, 0x78, 0x8b,
|
||||||
|
0x9d, 0xab, 0x7a, 0x69, 0x60, 0x5f, 0x66, 0x74, 0x86, 0x9a, 0xac, 0xba, 0xc1, 0x7a,
|
||||||
|
0x6c, 0x65, 0x66, 0x6f, 0x7f, 0x93, 0xa7, 0xba, 0xc9, 0xd0, 0xce, 0x6a, 0x65, 0x69,
|
||||||
|
0x74, 0x86, 0x9c, 0xb2, 0xc7, 0xd6, 0xde, 0xdd, 0xd2, 0x63, 0x61, 0x67, 0x75, 0x89,
|
||||||
|
0xa1, 0xba, 0xd0, 0xe1, 0xea, 0xea, 0xe0, 0xcc, 0x59, 0x61, 0x71, 0x88, 0xa2, 0xbd,
|
||||||
|
0xd5, 0xe8, 0xf3, 0xf4, 0xec, 0xd9, 0xbf, 0x4e, 0x58, 0x6a, 0x82, 0x9e, 0xbc, 0xd6,
|
||||||
|
0xeb, 0xf8, 0xfc, 0xf4, 0xe4, 0xca, 0xac, 0x4b, 0x5f, 0x79, 0x97, 0xb6, 0xd3, 0xea,
|
||||||
|
0xf9, 0xfe, 0xf9, 0xea, 0xd3, 0xb6, 0x96, 0x3d, 0x51, 0x6c, 0x8c, 0xad, 0xcb, 0xe4,
|
||||||
|
0xf6, 0xfd, 0xfa, 0xed, 0xd8, 0xbd, 0x9f, 0x82, 0x43, 0x5e, 0x7f, 0xa1, 0xc0, 0xdb,
|
||||||
|
0xee, 0xf7, 0xf6, 0xeb, 0xd8, 0xc0, 0xa4, 0x89, 0x71, 0x50, 0x70, 0x92, 0xb3, 0xcf,
|
||||||
|
0xe3, 0xee, 0xee, 0xe6, 0xd5, 0xbe, 0xa4, 0x8c, 0x76, 0x67, 0x62, 0x84, 0xa5, 0xc1,
|
||||||
|
0xd5, 0xe1, 0xe3, 0xdc, 0xcd, 0xb8, 0xa1, 0x8a, 0x77, 0x6a, 0x64, 0x76, 0x96, 0xb2,
|
||||||
|
0xc7, 0xd3, 0xd5, 0xcf, 0xc1, 0xae, 0x99, 0x84, 0x73, 0x68, 0x65, 0x6a, 0x89, 0xa4,
|
||||||
|
0xb8, 0xc4, 0xc6, 0xc0, 0xb3, 0xa1, 0x8d, 0x7a, 0x6b, 0x63, 0x62, 0x69, 0x78, 0x98,
|
||||||
|
0xab, 0xb5, 0xb8, 0xb1, 0xa4, 0x93, 0x7f, 0x6e, 0x60, 0x59, 0x5b, 0x64, 0x75, 0x8c,
|
||||||
|
0xa0, 0xa9, 0xaa, 0xa3, 0x95, 0x83, 0x70, 0x5f, 0x53, 0x4d, 0x50, 0x5b, 0x6e, 0x87,
|
||||||
|
0xa4, 0xa0, 0x9f, 0x96, 0x87, 0x74, 0x61, 0x50, 0x44, 0x3f, 0x42, 0x4f, 0x64, 0x7e,
|
||||||
|
0x9d, 0xbc, 0x97, 0x8c, 0x7c, 0x68, 0x53, 0x41, 0x34, 0x2f, 0x33, 0x41, 0x56, 0x73,
|
||||||
|
0x93, 0xb3, 0xd1, 0x86, 0x74, 0x5e, 0x47, 0x34, 0x27, 0x21, 0x25, 0x32, 0x48, 0x65,
|
||||||
|
0x86, 0xa7, 0xc6, 0xe0, 0x70, 0x57, 0x3f, 0x2a, 0x1b, 0x14, 0x17, 0x24, 0x39, 0x56,
|
||||||
|
0x77, 0x99, 0xb9, 0xd4, 0xe6, 0x56, 0x3b, 0x24, 0x13, 0x0a, 0x0c, 0x17, 0x2c, 0x48,
|
||||||
|
0x69, 0x8b, 0xab, 0xc6, 0xd9, 0xe3, 0x3b, 0x22, 0x0f, 0x04, 0x04, 0x0e, 0x21, 0x3c,
|
||||||
|
0x5b, 0x7d, 0x9c, 0xb7, 0xca, 0xd5, 0xd6, 0x25, 0x0f, 0x03, 0x00, 0x08, 0x19, 0x32,
|
||||||
|
0x50, 0x70, 0x8f, 0xa9, 0xbc, 0xc6, 0xc7, 0xc0, 0x14, 0x05, 0x01, 0x06, 0x15, 0x2c,
|
||||||
|
0x49, 0x67, 0x84, 0x9c, 0xae, 0xb7, 0xb8, 0xb0, 0xa2, 0x0c, 0x05, 0x09, 0x16, 0x2b,
|
||||||
|
0x45, 0x61, 0x7c, 0x93, 0xa3, 0xab, 0xaa, 0xa2, 0x93, 0x81, 0x0e, 0x10, 0x1b, 0x2e,
|
||||||
|
0x46, 0x60, 0x78, 0x8d, 0x9b, 0xa1, 0x9e, 0x94, 0x85, 0x72, 0x5f, 0x1b, 0x24, 0x35,
|
||||||
|
0x4b, 0x62, 0x79, 0x8b, 0x96, 0x9a, 0x96, 0x8a, 0x79, 0x65, 0x50, 0x3f, 0x30, 0x40,
|
||||||
|
0x54, 0x6a, 0x7e, 0x8d, 0x97, 0x98, 0x91, 0x84, 0x70, 0x5a, 0x44, 0x32, 0x26, 0x4d,
|
||||||
|
0x60, 0x74, 0x87, 0x94, 0x9b, 0x9a, 0x91, 0x81, 0x6c, 0x53, 0x3c, 0x28, 0x1a, 0x15,
|
||||||
|
0x6e, 0x82, 0x93, 0x9f, 0xa4, 0xa1, 0x96, 0x83, 0x6b, 0x51, 0x37, 0x21, 0x11, 0x0a,
|
||||||
|
0x0e, 0x91, 0xa1, 0xac, 0xb0, 0xab, 0x9e, 0x89, 0x6f, 0x53, 0x37, 0x1e, 0x0d, 0x04,
|
||||||
|
0x05, 0x11, 0xb0, 0xbb, 0xbe, 0xb8, 0xaa, 0x93, 0x77, 0x59, 0x3a, 0x20, 0x0c, 0x01,
|
||||||
|
0x01, 0x0a, 0x1d, 0xca, 0xcd, 0xc7, 0xb7, 0xa0, 0x83, 0x62, 0x42, 0x26, 0x10, 0x03,
|
||||||
|
0x01, 0x08, 0x19, 0x31, 0xdc, 0xd6, 0xc6, 0xae, 0x90, 0x6f, 0x4e, 0x30, 0x18, 0x0a,
|
||||||
|
0x05, 0x0a, 0x19, 0x2f, 0x4a, 0xe4, 0xd4, 0xbd, 0x9f, 0x7d, 0x5b, 0x3c, 0x24, 0x14,
|
||||||
|
0x0d, 0x11, 0x1e, 0x32, 0x4a, 0x64, 0xe1, 0xcb, 0xad, 0x8c, 0x6a, 0x4b, 0x31, 0x20,
|
||||||
|
0x19, 0x1b, 0x26, 0x38, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
let pkt1_frame0_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x74, 0x02, 0x73, 0x02, 0x4f, 0x66, 0x7c, 0xd6, 0xba, 0x99, 0x78,
|
||||||
|
0x59, 0x40, 0x2f, 0x27, 0x28, 0x32, 0x42, 0x57, 0x6d, 0x80, 0x8f, 0xc4, 0xa5, 0x85,
|
||||||
|
0x67, 0x4f, 0x3e, 0x35, 0x37, 0x40, 0x4f, 0x63, 0x77, 0x89, 0x96, 0x9b, 0xad, 0x8f,
|
||||||
|
0x73, 0x5c, 0x4c, 0x44, 0x46, 0x4f, 0x5e, 0x71, 0x84, 0x94, 0xa0, 0xa3, 0x9f, 0x96,
|
||||||
|
0x7b, 0x66, 0x58, 0x52, 0x54, 0x5e, 0x6e, 0x80, 0x93, 0xa2, 0xac, 0xaf, 0xa9, 0x9a,
|
||||||
|
0x80, 0x6d, 0x61, 0x5c, 0x60, 0x6c, 0x7c, 0x8f, 0xa2, 0xb2, 0xbb, 0xbd, 0xb5, 0xa5,
|
||||||
|
0x8e, 0x70, 0x66, 0x64, 0x6a, 0x77, 0x89, 0x9d, 0xb1, 0xc1, 0xca, 0xcb, 0xc4, 0xb3,
|
||||||
|
0x9a, 0x7c, 0x66, 0x67, 0x6f, 0x7e, 0x92, 0xa8, 0xbd, 0xce, 0xd9, 0xda, 0xd3, 0xc1,
|
||||||
|
0xa8, 0x8a, 0x68, 0x65, 0x70, 0x82, 0x98, 0xb0, 0xc7, 0xd9, 0xe5, 0xe8, 0xe1, 0xd0,
|
||||||
|
0xb7, 0x98, 0x76, 0x54, 0x6d, 0x81, 0x99, 0xb3, 0xcd, 0xe1, 0xee, 0xf3, 0xed, 0xdd,
|
||||||
|
0xc5, 0xa6, 0x85, 0x63, 0x45, 0x7b, 0x96, 0xb3, 0xce, 0xe5, 0xf4, 0xfa, 0xf6, 0xe8,
|
||||||
|
0xd1, 0xb4, 0x93, 0x71, 0x53, 0x3b, 0x8f, 0xad, 0xcb, 0xe4, 0xf5, 0xfe, 0xfc, 0xef,
|
||||||
|
0xda, 0xbe, 0x9f, 0x7f, 0x61, 0x4a, 0x3b, 0xa5, 0xc4, 0xdf, 0xf2, 0xfd, 0xfd, 0xf2,
|
||||||
|
0xdf, 0xc5, 0xa8, 0x89, 0x6e, 0x57, 0x49, 0x43, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
let pkt0_frame16_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x01, 0x00, 0x73, 0x02, 0x95, 0x88, 0x92, 0x75, 0x81, 0x60, 0x8f,
|
||||||
|
0x6d, 0x4c, 0x7c, 0x5a, 0x3c, 0x8a, 0x69, 0x4a, 0x32, 0x77, 0x59, 0x41, 0x32, 0x83,
|
||||||
|
0x66, 0x50, 0x40, 0x3a, 0x72, 0x5c, 0x4e, 0x49, 0x4c, 0x79, 0x66, 0x59, 0x56, 0x5a,
|
||||||
|
0x65, 0x6c, 0x61, 0x5f, 0x66, 0x72, 0x84, 0x6d, 0x66, 0x66, 0x6e, 0x7d, 0x90, 0xa5,
|
||||||
|
0x65, 0x68, 0x72, 0x83, 0x99, 0xaf, 0xc5, 0x61, 0x65, 0x72, 0x86, 0x9d, 0xb6, 0xcd,
|
||||||
|
0xdf, 0x5f, 0x6e, 0x84, 0x9d, 0xb9, 0xd2, 0xe6, 0xf2, 0x55, 0x66, 0x7d, 0x99, 0xb7,
|
||||||
|
0xd2, 0xe8, 0xf7, 0xfc, 0x5a, 0x73, 0x91, 0xb1, 0xce, 0xe6, 0xf7, 0xfe, 0xfb, 0x4c,
|
||||||
|
0x67, 0x86, 0xa7, 0xc6, 0xe0, 0xf3, 0xfc, 0xfb, 0xf0, 0x58, 0x78, 0x9a, 0xbb, 0xd6,
|
||||||
|
0xeb, 0xf6, 0xf7, 0xed, 0xdc, 0x4a, 0x6a, 0x8c, 0xad, 0xca, 0xdf, 0xec, 0xee, 0xe7,
|
||||||
|
0xd7, 0xc1, 0x5b, 0x7d, 0x9f, 0xbc, 0xd2, 0xdf, 0xe3, 0xdd, 0xcf, 0xbb, 0xa4, 0x4f,
|
||||||
|
0x70, 0x90, 0xad, 0xc3, 0xd0, 0xd5, 0xd0, 0xc3, 0xb1, 0x9b, 0x87, 0x64, 0x83, 0x9f,
|
||||||
|
0xb4, 0xc1, 0xc6, 0xc1, 0xb5, 0xa3, 0x90, 0x7c, 0x6d, 0x5c, 0x79, 0x93, 0xa7, 0xb3,
|
||||||
|
0xb7, 0xb2, 0xa6, 0x95, 0x81, 0x6f, 0x61, 0x59, 0x73, 0x8b, 0x9d, 0xa7, 0xa9, 0xa3,
|
||||||
|
0x97, 0x85, 0x72, 0x60, 0x53, 0x4c, 0x4e, 0x70, 0x86, 0x96, 0x9e, 0x9f, 0x97, 0x89,
|
||||||
|
0x77, 0x63, 0x51, 0x44, 0x3e, 0x40, 0x4b, 0x86, 0x93, 0x99, 0x97, 0x8e, 0x7e, 0x6a,
|
||||||
|
0x55, 0x43, 0x35, 0x2f, 0x31, 0x3d, 0x51, 0x8a, 0x95, 0x99, 0x94, 0x88, 0x76, 0x61,
|
||||||
|
0x4a, 0x36, 0x27, 0x20, 0x22, 0x2e, 0x43, 0x5f, 0x9b, 0x9c, 0x96, 0x87, 0x73, 0x5b,
|
||||||
|
0x43, 0x2d, 0x1c, 0x14, 0x15, 0x20, 0x34, 0x50, 0x71, 0xa4, 0x9b, 0x8b, 0x74, 0x5a,
|
||||||
|
0x3f, 0x27, 0x15, 0x0b, 0x0a, 0x14, 0x27, 0x42, 0x62, 0x84, 0xa5, 0x92, 0x7a, 0x5d,
|
||||||
|
0x40, 0x26, 0x11, 0x05, 0x03, 0x0b, 0x1d, 0x36, 0x55, 0x76, 0x96, 0x9d, 0x83, 0x64,
|
||||||
|
0x45, 0x29, 0x12, 0x04, 0x00, 0x06, 0x15, 0x2d, 0x4b, 0x6a, 0x89, 0xa4, 0x8f, 0x6f,
|
||||||
|
0x4e, 0x30, 0x18, 0x07, 0x01, 0x05, 0x12, 0x28, 0x44, 0x62, 0x7f, 0x98, 0xab, 0x7c,
|
||||||
|
0x5a, 0x3b, 0x21, 0x0f, 0x06, 0x08, 0x14, 0x27, 0x41, 0x5c, 0x77, 0x8f, 0xa0, 0xa9,
|
||||||
|
0x68, 0x48, 0x2d, 0x1a, 0x10, 0x10, 0x19, 0x2b, 0x42, 0x5c, 0x74, 0x89, 0x98, 0xa0,
|
||||||
|
0x9e, 0x57, 0x3b, 0x27, 0x1c, 0x1b, 0x23, 0x33, 0x48, 0x5f, 0x76, 0x88, 0x95, 0x9a,
|
||||||
|
0x97, 0x8c, 0x4a, 0x36, 0x2a, 0x28, 0x2f, 0x3e, 0x51, 0x67, 0x7b, 0x8c, 0x96, 0x99,
|
||||||
|
0x93, 0x86, 0x73, 0x44, 0x39, 0x37, 0x3e, 0x4b, 0x5e, 0x72, 0x85, 0x93, 0x9b, 0x9c,
|
||||||
|
0x94, 0x84, 0x6f, 0x57, 0x48, 0x46, 0x4d, 0x5b, 0x6c, 0x80, 0x91, 0x9e, 0xa4, 0xa3,
|
||||||
|
0x99, 0x87, 0x70, 0x55, 0x3b, 0x54, 0x5c, 0x6a, 0x7c, 0x8f, 0xa0, 0xac, 0xb1, 0xad,
|
||||||
|
0xa1, 0x8e, 0x74, 0x58, 0x3b, 0x22, 0x69, 0x78, 0x8b, 0x9e, 0xaf, 0xbb, 0xbf, 0xbb,
|
||||||
|
0xad, 0x98, 0x7d, 0x5e, 0x40, 0x24, 0x0f, 0x84, 0x98, 0xac, 0xbe, 0xca, 0xce, 0xc9,
|
||||||
|
0xbb, 0xa5, 0x89, 0x68, 0x48, 0x2b, 0x14, 0x05, 0xa2, 0xb8, 0xcb, 0xd8, 0xdd, 0xd8,
|
||||||
|
0xca, 0xb3, 0x96, 0x75, 0x54, 0x35, 0x1c, 0x0c, 0x05, 0xc0, 0xd5, 0xe3, 0xea, 0xe6,
|
||||||
|
0xd8, 0xc2, 0xa5, 0x84, 0x61, 0x42, 0x28, 0x16, 0x0e, 0x10, 0xdb, 0xec, 0xf4, 0xf1,
|
||||||
|
0xe5, 0xd0, 0xb3, 0x92, 0x70, 0x50, 0x36, 0x23, 0x1a, 0x1b, 0x24, 0xf0, 0xfa, 0xfa,
|
||||||
|
0xef, 0xdb, 0xc0, 0xa0, 0x7e, 0x5f, 0x45, 0x32, 0x28, 0x28, 0x31, 0x40, 0xfc, 0xfe,
|
||||||
|
0xf5, 0xe3, 0xc9, 0xab, 0x8b, 0x6c, 0x53, 0x41, 0x37, 0x37, 0x3f, 0x4e, 0x61, 0xfd,
|
||||||
|
0xf7, 0xe7, 0xcf, 0xb3, 0x94, 0x78, 0x60, 0x4f, 0x46, 0x46, 0x4e, 0x5d, 0x6f, 0x82,
|
||||||
|
0xf4, 0xe7, 0xd1, 0xb7, 0x9b, 0x80, 0x6a, 0x5a, 0x53, 0x54, 0x5d, 0x6c, 0x7e, 0x91,
|
||||||
|
0xa1, 0xe2, 0xce, 0xb6, 0x9c, 0x84, 0x70, 0x62, 0x5d, 0x60, 0x6a, 0x7a, 0x8d, 0xa0,
|
||||||
|
0xb1, 0xbb, 0xc8, 0xb2, 0x9a, 0x84, 0x72, 0x67, 0x64, 0x69, 0x75, 0x86, 0x9b, 0xaf,
|
||||||
|
0xbf, 0xca, 0xcd, 0xa9, 0x93, 0x7f, 0x70, 0x67, 0x66, 0x6d, 0x7c, 0x8f, 0xa5, 0xbb,
|
||||||
|
0xcd, 0xd8, 0xdc, 0xd5, 0x89, 0x76, 0x69, 0x63, 0x64, 0x6e, 0x7e, 0x94, 0xac, 0xc4,
|
||||||
|
0xd8, 0xe5, 0xe9, 0xe3, 0xd4, 0x6b, 0x5f, 0x5a, 0x5e, 0x6a, 0x7d, 0x95, 0xaf, 0xc9,
|
||||||
|
0xdf, 0xed, 0xf3, 0xef, 0xe1, 0xca, 0x52, 0x4f, 0x54, 0x62, 0x77, 0x91, 0xae, 0xca,
|
||||||
|
0xe1, 0xf2, 0xfa, 0xf8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
let pkt1_frame16_check = [
|
||||||
|
0x5e, 0xc0, 0x02, 0x74, 0x02, 0x73, 0x02, 0xeb, 0xd6, 0xb9, 0x41, 0x47, 0x56, 0x6d,
|
||||||
|
0x89, 0xa8, 0xc6, 0xe0, 0xf3, 0xfd, 0xfd, 0xf2, 0xde, 0xc4, 0xa4, 0x39, 0x49, 0x61,
|
||||||
|
0x7e, 0x9f, 0xbe, 0xda, 0xef, 0xfb, 0xfd, 0xf5, 0xe3, 0xca, 0xad, 0x8f, 0x3a, 0x53,
|
||||||
|
0x71, 0x92, 0xb3, 0xd1, 0xe7, 0xf5, 0xf9, 0xf3, 0xe3, 0xcd, 0xb2, 0x95, 0x7b, 0x44,
|
||||||
|
0x62, 0x84, 0xa6, 0xc4, 0xdc, 0xec, 0xf1, 0xed, 0xdf, 0xcb, 0xb2, 0x98, 0x80, 0x6d,
|
||||||
|
0x54, 0x76, 0x97, 0xb6, 0xcf, 0xdf, 0xe6, 0xe3, 0xd7, 0xc5, 0xae, 0x96, 0x80, 0x6f,
|
||||||
|
0x66, 0x68, 0x89, 0xa7, 0xc0, 0xd1, 0xd8, 0xd6, 0xcc, 0xbb, 0xa6, 0x90, 0x7c, 0x6e,
|
||||||
|
0x66, 0x67, 0x7c, 0x9a, 0xb2, 0xc2, 0xc9, 0xc8, 0xbe, 0xae, 0x9a, 0x86, 0x74, 0x68,
|
||||||
|
0x63, 0x65, 0x70, 0x8e, 0xa4, 0xb4, 0xba, 0xb8, 0xaf, 0x9f, 0x8c, 0x79, 0x69, 0x5e,
|
||||||
|
0x5b, 0x60, 0x6d, 0x81, 0x9a, 0xa8, 0xad, 0xaa, 0x9f, 0x90, 0x7d, 0x6a, 0x5b, 0x52,
|
||||||
|
0x50, 0x57, 0x66, 0x7c, 0x96, 0x9e, 0xa2, 0x9d, 0x92, 0x81, 0x6e, 0x5b, 0x4c, 0x43,
|
||||||
|
0x42, 0x4a, 0x5b, 0x73, 0x8f, 0xae, 0x9a, 0x94, 0x86, 0x74, 0x60, 0x4c, 0x3d, 0x34,
|
||||||
|
0x33, 0x3c, 0x4d, 0x66, 0x85, 0xa5, 0xc4, 0x8e, 0x7f, 0x6a, 0x54, 0x40, 0x2f, 0x25,
|
||||||
|
0x24, 0x2d, 0x3f, 0x58, 0x78, 0x99, 0xba, 0xd6, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
];
|
||||||
|
|
||||||
|
let mut path = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
||||||
|
path.push("tests/data/g835l-diagonal.gif");
|
||||||
|
|
||||||
|
let gif =
|
||||||
|
AnimeGif::from_diagonal_gif(&path, AnimTime::Count(1), 1.0, AnimeType::G835L).unwrap();
|
||||||
|
assert_eq!(gif.frame_count(), 48);
|
||||||
|
|
||||||
|
let pkt = AnimePacketType::try_from(gif.frames()[0].frame().clone()).unwrap();
|
||||||
|
assert_eq!(pkt[0], pkt0_frame0_check);
|
||||||
|
assert_eq!(pkt[1], pkt1_frame0_check);
|
||||||
|
|
||||||
|
let pkt = AnimePacketType::try_from(gif.frames()[16].frame().clone()).unwrap();
|
||||||
|
assert_eq!(pkt[0], pkt0_frame16_check);
|
||||||
|
assert_eq!(pkt[1], pkt1_frame16_check);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -38,7 +38,16 @@
|
|||||||
(
|
(
|
||||||
device_name: "FA617NS",
|
device_name: "FA617NS",
|
||||||
product_id: "",
|
product_id: "",
|
||||||
layout_name: "fx505d",
|
layout_name: "fa507",
|
||||||
|
basic_modes: [Static, Breathe, Pulse],
|
||||||
|
basic_zones: [],
|
||||||
|
advanced_type: r#None,
|
||||||
|
power_zones: [Keyboard],
|
||||||
|
),
|
||||||
|
(
|
||||||
|
device_name: "FA617NT",
|
||||||
|
product_id: "",
|
||||||
|
layout_name: "fa507",
|
||||||
basic_modes: [Static, Breathe, Pulse],
|
basic_modes: [Static, Breathe, Pulse],
|
||||||
basic_zones: [],
|
basic_zones: [],
|
||||||
advanced_type: r#None,
|
advanced_type: r#None,
|
||||||
@@ -47,7 +56,16 @@
|
|||||||
(
|
(
|
||||||
device_name: "FA617XS",
|
device_name: "FA617XS",
|
||||||
product_id: "",
|
product_id: "",
|
||||||
layout_name: "fx505d",
|
layout_name: "fa507",
|
||||||
|
basic_modes: [Static, Breathe, Pulse],
|
||||||
|
basic_zones: [],
|
||||||
|
advanced_type: r#None,
|
||||||
|
power_zones: [Keyboard],
|
||||||
|
),
|
||||||
|
(
|
||||||
|
device_name: "FA617XT",
|
||||||
|
product_id: "",
|
||||||
|
layout_name: "fa507",
|
||||||
basic_modes: [Static, Breathe, Pulse],
|
basic_modes: [Static, Breathe, Pulse],
|
||||||
basic_zones: [],
|
basic_zones: [],
|
||||||
advanced_type: r#None,
|
advanced_type: r#None,
|
||||||
@@ -296,6 +314,15 @@
|
|||||||
advanced_type: r#None,
|
advanced_type: r#None,
|
||||||
power_zones: [Keyboard, Lightbar],
|
power_zones: [Keyboard, Lightbar],
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
device_name: "G614JU",
|
||||||
|
product_id: "",
|
||||||
|
layout_name: "g634j-per-key",
|
||||||
|
basic_modes: [Static, Breathe, RainbowCycle, RainbowWave, Star, Rain, Highlight, Laser, Ripple, Pulse, Comet, Flash],
|
||||||
|
basic_zones: [],
|
||||||
|
advanced_type: PerKey,
|
||||||
|
power_zones: [Keyboard, Lightbar, Logo],
|
||||||
|
),
|
||||||
(
|
(
|
||||||
device_name: "G614JZ",
|
device_name: "G614JZ",
|
||||||
product_id: "",
|
product_id: "",
|
||||||
@@ -305,6 +332,15 @@
|
|||||||
advanced_type: r#None,
|
advanced_type: r#None,
|
||||||
power_zones: [Keyboard, Lightbar],
|
power_zones: [Keyboard, Lightbar],
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
device_name: "G615LR",
|
||||||
|
product_id: "",
|
||||||
|
layout_name: "g634j-per-key",
|
||||||
|
basic_modes: [Static, Breathe, RainbowCycle, RainbowWave],
|
||||||
|
basic_zones: [Key1, Key2, Key3, Key4],
|
||||||
|
advanced_type: r#None,
|
||||||
|
power_zones: [Keyboard, Lightbar],
|
||||||
|
),
|
||||||
(
|
(
|
||||||
device_name: "G634J",
|
device_name: "G634J",
|
||||||
product_id: "",
|
product_id: "",
|
||||||
@@ -539,6 +575,15 @@
|
|||||||
advanced_type: PerKey,
|
advanced_type: PerKey,
|
||||||
power_zones: [Keyboard, Lightbar, Logo, RearGlow],
|
power_zones: [Keyboard, Lightbar, Logo, RearGlow],
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
device_name: "G835L",
|
||||||
|
product_id: "",
|
||||||
|
layout_name: "g814ji-per-key",
|
||||||
|
basic_modes: [Static, Breathe, RainbowCycle, RainbowWave, Star, Rain, Highlight, Laser, Ripple, Comet, Flash],
|
||||||
|
basic_zones: [],
|
||||||
|
advanced_type: PerKey,
|
||||||
|
power_zones: [Keyboard, Lightbar, Logo],
|
||||||
|
),
|
||||||
(
|
(
|
||||||
device_name: "GA401I",
|
device_name: "GA401I",
|
||||||
product_id: "",
|
product_id: "",
|
||||||
|
|||||||
@@ -359,6 +359,12 @@ impl From<AuraEffect> for AuraModeNum {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "dbus")]
|
||||||
|
impl zbus::zvariant::Basic for AuraModeNum {
|
||||||
|
const SIGNATURE_CHAR: char = 'u';
|
||||||
|
const SIGNATURE_STR: &'static str = "u";
|
||||||
|
}
|
||||||
|
|
||||||
/// Base effects have no zoning, while multizone is 1-4
|
/// Base effects have no zoning, while multizone is 1-4
|
||||||
#[cfg_attr(
|
#[cfg_attr(
|
||||||
feature = "dbus",
|
feature = "dbus",
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ mocking = []
|
|||||||
x11 = ["slint/backend-winit-x11"]
|
x11 = ["slint/backend-winit-x11"]
|
||||||
# Optional tokio debug feature does not require nightly; remove RUSTFLAGS note.
|
# Optional tokio debug feature does not require nightly; remove RUSTFLAGS note.
|
||||||
tokio-debug = ["console-subscriber"]
|
tokio-debug = ["console-subscriber"]
|
||||||
|
rog_ally = []
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
console-subscriber = { version = "^0.4", optional = true }
|
console-subscriber = { version = "^0.4", optional = true }
|
||||||
|
|||||||
@@ -107,7 +107,6 @@ async fn main() -> Result<()> {
|
|||||||
let board_name = dmi.board_name;
|
let board_name = dmi.board_name;
|
||||||
let prod_family = dmi.product_family;
|
let prod_family = dmi.product_family;
|
||||||
info!("Running on {board_name}, product: {prod_family}");
|
info!("Running on {board_name}, product: {prod_family}");
|
||||||
let is_rog_ally = board_name == "RC71L" || board_name == "RC72L" || prod_family == "ROG Ally";
|
|
||||||
|
|
||||||
let args: Vec<String> = args().skip(1).collect();
|
let args: Vec<String> = args().skip(1).collect();
|
||||||
|
|
||||||
@@ -138,6 +137,18 @@ async fn main() -> Result<()> {
|
|||||||
config.start_fullscreen = false;
|
config.start_fullscreen = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let is_rog_ally = {
|
||||||
|
#[cfg(feature = "rog_ally")]
|
||||||
|
{
|
||||||
|
board_name == "RC71L" || board_name == "RC72L" || prod_family == "ROG Ally"
|
||||||
|
}
|
||||||
|
#[cfg(not(feature = "rog_ally"))]
|
||||||
|
{
|
||||||
|
false
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#[cfg(feature = "rog_ally")]
|
||||||
if is_rog_ally {
|
if is_rog_ally {
|
||||||
config.notifications.enabled = false;
|
config.notifications.enabled = false;
|
||||||
config.enable_tray_icon = false;
|
config.enable_tray_icon = false;
|
||||||
@@ -145,6 +156,7 @@ async fn main() -> Result<()> {
|
|||||||
config.startup_in_background = false;
|
config.startup_in_background = false;
|
||||||
config.start_fullscreen = true;
|
config.start_fullscreen = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
config.write();
|
config.write();
|
||||||
|
|
||||||
let enable_tray_icon = config.enable_tray_icon;
|
let enable_tray_icon = config.enable_tray_icon;
|
||||||
@@ -203,76 +215,77 @@ async fn main() -> Result<()> {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
.ok();
|
.ok();
|
||||||
} else {
|
|
||||||
// save as a var, don't hold the lock the entire time or deadlocks happen
|
continue;
|
||||||
if let Ok(app_state) = app_state.lock() {
|
}
|
||||||
state = *app_state;
|
|
||||||
|
// save as a var, don't hold the lock the entire time or deadlocks happen
|
||||||
|
if let Ok(app_state) = app_state.lock() {
|
||||||
|
state = *app_state;
|
||||||
|
}
|
||||||
|
|
||||||
|
// This sleep is required to give the event loop time to react
|
||||||
|
sleep(Duration::from_millis(300));
|
||||||
|
if state == AppState::MainWindowShouldOpen {
|
||||||
|
if let Ok(mut app_state) = app_state.lock() {
|
||||||
|
*app_state = AppState::MainWindowOpen;
|
||||||
}
|
}
|
||||||
|
|
||||||
// This sleep is required to give the event loop time to react
|
let config_copy = config.clone();
|
||||||
sleep(Duration::from_millis(300));
|
let app_state_copy = app_state.clone();
|
||||||
if state == AppState::MainWindowShouldOpen {
|
slint::invoke_from_event_loop(move || {
|
||||||
if let Ok(mut app_state) = app_state.lock() {
|
UI.with(|ui| {
|
||||||
*app_state = AppState::MainWindowOpen;
|
let app_state_copy = app_state_copy.clone();
|
||||||
}
|
let mut ui = ui.borrow_mut();
|
||||||
|
if let Some(ui) = ui.as_mut() {
|
||||||
|
ui.window().show().unwrap();
|
||||||
|
ui.window().on_close_requested(move || {
|
||||||
|
if let Ok(mut app_state) = app_state_copy.lock() {
|
||||||
|
*app_state = AppState::MainWindowClosed;
|
||||||
|
}
|
||||||
|
slint::CloseRequestResponse::HideWindow
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
let config_copy_2 = config_copy.clone();
|
||||||
|
let newui = setup_window(config_copy);
|
||||||
|
newui.window().on_close_requested(move || {
|
||||||
|
if let Ok(mut app_state) = app_state_copy.lock() {
|
||||||
|
*app_state = AppState::MainWindowClosed;
|
||||||
|
}
|
||||||
|
slint::CloseRequestResponse::HideWindow
|
||||||
|
});
|
||||||
|
|
||||||
let config_copy = config.clone();
|
let ui_copy = newui.as_weak();
|
||||||
let app_state_copy = app_state.clone();
|
newui
|
||||||
slint::invoke_from_event_loop(move || {
|
.window()
|
||||||
UI.with(|ui| {
|
.set_rendering_notifier(move |s, _| {
|
||||||
let app_state_copy = app_state_copy.clone();
|
if let slint::RenderingState::RenderingSetup = s {
|
||||||
let mut ui = ui.borrow_mut();
|
let config = config_copy_2.clone();
|
||||||
if let Some(ui) = ui.as_mut() {
|
ui_copy
|
||||||
ui.window().show().unwrap();
|
.upgrade_in_event_loop(move |w| {
|
||||||
ui.window().on_close_requested(move || {
|
let fullscreen =
|
||||||
if let Ok(mut app_state) = app_state_copy.lock() {
|
config.lock().is_ok_and(|c| c.start_fullscreen);
|
||||||
*app_state = AppState::MainWindowClosed;
|
if fullscreen && !w.window().is_fullscreen() {
|
||||||
|
w.window().set_fullscreen(fullscreen);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.ok();
|
||||||
}
|
}
|
||||||
slint::CloseRequestResponse::HideWindow
|
})
|
||||||
});
|
.ok();
|
||||||
} else {
|
ui.replace(newui);
|
||||||
let config_copy_2 = config_copy.clone();
|
|
||||||
let newui = setup_window(config_copy);
|
|
||||||
newui.window().on_close_requested(move || {
|
|
||||||
if let Ok(mut app_state) = app_state_copy.lock() {
|
|
||||||
*app_state = AppState::MainWindowClosed;
|
|
||||||
}
|
|
||||||
slint::CloseRequestResponse::HideWindow
|
|
||||||
});
|
|
||||||
|
|
||||||
let ui_copy = newui.as_weak();
|
|
||||||
newui
|
|
||||||
.window()
|
|
||||||
.set_rendering_notifier(move |s, _| {
|
|
||||||
if let slint::RenderingState::RenderingSetup = s {
|
|
||||||
let config = config_copy_2.clone();
|
|
||||||
ui_copy
|
|
||||||
.upgrade_in_event_loop(move |w| {
|
|
||||||
let fullscreen = config
|
|
||||||
.lock()
|
|
||||||
.is_ok_and(|c| c.start_fullscreen);
|
|
||||||
if fullscreen && !w.window().is_fullscreen() {
|
|
||||||
w.window().set_fullscreen(fullscreen);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.ok();
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.ok();
|
|
||||||
ui.replace(newui);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.unwrap();
|
|
||||||
} else if state == AppState::QuitApp {
|
|
||||||
slint::quit_event_loop().unwrap();
|
|
||||||
exit(0);
|
|
||||||
} else if state != AppState::MainWindowOpen {
|
|
||||||
if let Ok(config) = config.lock() {
|
|
||||||
if !config.run_in_background {
|
|
||||||
slint::quit_event_loop().unwrap();
|
|
||||||
exit(0);
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
} else if state == AppState::QuitApp {
|
||||||
|
slint::quit_event_loop().unwrap();
|
||||||
|
exit(0);
|
||||||
|
} else if state != AppState::MainWindowOpen {
|
||||||
|
if let Ok(config) = config.lock() {
|
||||||
|
if !config.run_in_background {
|
||||||
|
slint::quit_event_loop().unwrap();
|
||||||
|
exit(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,15 +9,11 @@ use std::process::Command;
|
|||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc, Mutex};
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
use futures_util::StreamExt;
|
|
||||||
use log::{debug, error, info, warn};
|
use log::{debug, error, info, warn};
|
||||||
use notify_rust::{Hint, Notification, Timeout, Urgency};
|
use notify_rust::{Hint, Notification, Timeout};
|
||||||
use rog_platform::platform::GpuMode;
|
|
||||||
use rog_platform::power::AsusPower;
|
use rog_platform::power::AsusPower;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use supergfxctl::actions::UserActionRequired as GfxUserAction;
|
use supergfxctl::pci_device::GfxPower;
|
||||||
use supergfxctl::pci_device::{GfxMode, GfxPower};
|
|
||||||
use supergfxctl::zbus_proxy::DaemonProxy as SuperProxy;
|
|
||||||
use tokio::runtime::Runtime;
|
use tokio::runtime::Runtime;
|
||||||
use tokio::task::JoinHandle;
|
use tokio::task::JoinHandle;
|
||||||
|
|
||||||
@@ -145,12 +141,8 @@ pub fn start_notifications(
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let enabled_notifications_copy = config.clone();
|
info!("Attempting to start plain dgpu status monitor");
|
||||||
let no_supergfx = move |e: &zbus::Error| {
|
start_dpu_status_mon(config.clone());
|
||||||
error!("zbus signal: receive_notify_gfx_status: {e}");
|
|
||||||
warn!("Attempting to start plain dgpu status monitor");
|
|
||||||
start_dpu_status_mon(enabled_notifications_copy.clone());
|
|
||||||
};
|
|
||||||
|
|
||||||
// GPU MUX Mode notif
|
// GPU MUX Mode notif
|
||||||
// TODO: need to get armoury attrs and iter to find
|
// TODO: need to get armoury attrs and iter to find
|
||||||
@@ -189,95 +181,9 @@ pub fn start_notifications(
|
|||||||
// Ok::<(), zbus::Error>(())
|
// Ok::<(), zbus::Error>(())
|
||||||
// });
|
// });
|
||||||
|
|
||||||
let enabled_notifications_copy = config.clone();
|
|
||||||
// GPU Mode change/action notif
|
|
||||||
tokio::spawn(async move {
|
|
||||||
let conn = zbus::Connection::system().await.inspect_err(|e| {
|
|
||||||
no_supergfx(e);
|
|
||||||
})?;
|
|
||||||
let proxy = SuperProxy::builder(&conn).build().await.inspect_err(|e| {
|
|
||||||
no_supergfx(e);
|
|
||||||
})?;
|
|
||||||
let _ = proxy.mode().await.inspect_err(|e| {
|
|
||||||
no_supergfx(e);
|
|
||||||
})?;
|
|
||||||
|
|
||||||
let proxy_copy = proxy.clone();
|
|
||||||
let enabled_notifications_copy_action = enabled_notifications_copy.clone();
|
|
||||||
let mut p = proxy.receive_notify_action().await?;
|
|
||||||
tokio::spawn(async move {
|
|
||||||
info!("Started zbus signal thread: receive_notify_action");
|
|
||||||
while let Some(e) = p.next().await {
|
|
||||||
if let Ok(out) = e.args() {
|
|
||||||
// Respect user notification settings for gpu actions
|
|
||||||
if let Ok(cfg) = enabled_notifications_copy_action.lock() {
|
|
||||||
if !cfg.notifications.enabled || !cfg.notifications.receive_notify_gfx {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let action = out.action();
|
|
||||||
let mode = convert_gfx_mode(proxy.mode().await.unwrap_or_default());
|
|
||||||
match action {
|
|
||||||
supergfxctl::actions::UserActionRequired::Reboot => {
|
|
||||||
do_mux_notification("Graphics mode change requires reboot", &mode)
|
|
||||||
}
|
|
||||||
_ => do_gfx_action_notif(<&str>::from(action), *action, mode),
|
|
||||||
}
|
|
||||||
.map_err(|e| {
|
|
||||||
error!("zbus signal: do_gfx_action_notif: {e}");
|
|
||||||
e
|
|
||||||
})
|
|
||||||
.ok();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut p = proxy_copy.receive_notify_gfx_status().await?;
|
|
||||||
tokio::spawn(async move {
|
|
||||||
info!("Started zbus signal thread: receive_notify_gfx_status");
|
|
||||||
let mut last_status = GfxPower::Unknown;
|
|
||||||
while let Some(e) = p.next().await {
|
|
||||||
if let Ok(out) = e.args() {
|
|
||||||
let status = out.status;
|
|
||||||
if status != GfxPower::Unknown && status != last_status {
|
|
||||||
if let Ok(config) = enabled_notifications_copy.lock() {
|
|
||||||
if !config.notifications.receive_notify_gfx_status
|
|
||||||
|| !config.notifications.enabled
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Required check because status cycles through
|
|
||||||
// active/unknown/suspended
|
|
||||||
do_gpu_status_notif("dGPU status changed:", &status)
|
|
||||||
.show_async()
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.on_close(|_| ());
|
|
||||||
}
|
|
||||||
last_status = status;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
Ok::<(), zbus::Error>(())
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(vec![blocking])
|
Ok(vec![blocking])
|
||||||
}
|
}
|
||||||
|
|
||||||
fn convert_gfx_mode(gfx: GfxMode) -> GpuMode {
|
|
||||||
match gfx {
|
|
||||||
GfxMode::Hybrid => GpuMode::Optimus,
|
|
||||||
GfxMode::Integrated => GpuMode::Integrated,
|
|
||||||
GfxMode::NvidiaNoModeset => GpuMode::Optimus,
|
|
||||||
GfxMode::Vfio => GpuMode::Vfio,
|
|
||||||
GfxMode::AsusEgpu => GpuMode::Egpu,
|
|
||||||
GfxMode::AsusMuxDgpu => GpuMode::Ultimate,
|
|
||||||
GfxMode::None => GpuMode::Error,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn base_notification<T>(message: &str, data: &T) -> Notification
|
fn base_notification<T>(message: &str, data: &T) -> Notification
|
||||||
where
|
where
|
||||||
T: Display,
|
T: Display,
|
||||||
@@ -303,97 +209,3 @@ fn do_gpu_status_notif(message: &str, data: &GfxPower) -> Notification {
|
|||||||
notif.icon(icon);
|
notif.icon(icon);
|
||||||
notif
|
notif
|
||||||
}
|
}
|
||||||
|
|
||||||
fn do_gfx_action_notif(message: &str, action: GfxUserAction, mode: GpuMode) -> Result<()> {
|
|
||||||
if matches!(action, GfxUserAction::Reboot) {
|
|
||||||
do_mux_notification("Graphics mode change requires reboot", &mode).ok();
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut notif = Notification::new();
|
|
||||||
notif
|
|
||||||
.appname(NOTIF_HEADER)
|
|
||||||
.summary(&format!("Changing to {mode}. {message}"))
|
|
||||||
//.hint(Hint::Resident(true))
|
|
||||||
.hint(Hint::Category("device".into()))
|
|
||||||
.urgency(Urgency::Critical)
|
|
||||||
// For user-action notifications keep them visible if they require interaction
|
|
||||||
// but for non-interactive actions we prefer they auto-hide like other notifs.
|
|
||||||
.timeout(Timeout::Milliseconds(6000))
|
|
||||||
.icon("dialog-warning")
|
|
||||||
.hint(Hint::Transient(true));
|
|
||||||
|
|
||||||
if matches!(action, GfxUserAction::Logout) {
|
|
||||||
notif.action("gfx-mode-session-action", "Logout");
|
|
||||||
let handle = notif.show()?;
|
|
||||||
if let Ok(desktop) = std::env::var("XDG_CURRENT_DESKTOP") {
|
|
||||||
if desktop.to_lowercase() == "gnome" {
|
|
||||||
handle.wait_for_action(|id| {
|
|
||||||
if id == "gfx-mode-session-action" {
|
|
||||||
let mut cmd = Command::new("gnome-session-quit");
|
|
||||||
cmd.spawn().ok();
|
|
||||||
} else if id == "__closed" {
|
|
||||||
// TODO: cancel the switching
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else if desktop.to_lowercase() == "kde" {
|
|
||||||
handle.wait_for_action(|id| {
|
|
||||||
if id == "gfx-mode-session-action" {
|
|
||||||
let mut cmd = Command::new("qdbus");
|
|
||||||
cmd.args([
|
|
||||||
"org.kde.ksmserver", "/KSMServer", "logout", "1", "0", "0",
|
|
||||||
]);
|
|
||||||
cmd.spawn().ok();
|
|
||||||
} else if id == "__closed" {
|
|
||||||
// TODO: cancel the switching
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
// todo: handle alternatives
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
notif.show()?;
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Actual `GpuMode` unused as data is never correct until switched by reboot
|
|
||||||
fn do_mux_notification(message: &str, m: &GpuMode) -> Result<()> {
|
|
||||||
let mut notif = base_notification(message, &m.to_string());
|
|
||||||
notif
|
|
||||||
.action("gfx-mode-session-action", "Reboot")
|
|
||||||
.urgency(Urgency::Critical)
|
|
||||||
.icon("system-reboot-symbolic")
|
|
||||||
.hint(Hint::Transient(true));
|
|
||||||
let handle = notif.show()?;
|
|
||||||
|
|
||||||
std::thread::spawn(|| {
|
|
||||||
if let Ok(desktop) = std::env::var("XDG_CURRENT_DESKTOP") {
|
|
||||||
if desktop.to_lowercase() == "gnome" {
|
|
||||||
handle.wait_for_action(|id| {
|
|
||||||
if id == "gfx-mode-session-action" {
|
|
||||||
let mut cmd = Command::new("gnome-session-quit");
|
|
||||||
cmd.arg("--reboot");
|
|
||||||
cmd.spawn().ok();
|
|
||||||
} else if id == "__closed" {
|
|
||||||
// TODO: cancel the switching
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else if desktop.to_lowercase() == "kde" {
|
|
||||||
handle.wait_for_action(|id| {
|
|
||||||
if id == "gfx-mode-session-action" {
|
|
||||||
let mut cmd = Command::new("qdbus");
|
|
||||||
cmd.args([
|
|
||||||
"org.kde.ksmserver", "/KSMServer", "logout", "1", "1", "0",
|
|
||||||
]);
|
|
||||||
cmd.spawn().ok();
|
|
||||||
} else if id == "__closed" {
|
|
||||||
// TODO: cancel the switching
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -170,7 +170,7 @@ pub fn init_tray(_supported_properties: Vec<Properties>, config: Arc<Mutex<Confi
|
|||||||
|
|
||||||
// TODO: return an error to the UI
|
// TODO: return an error to the UI
|
||||||
let mut tray;
|
let mut tray;
|
||||||
match tray_init.spawn_without_dbus_name().await {
|
match tray_init.disable_dbus_name(true).spawn().await {
|
||||||
Ok(t) => tray = t,
|
Ok(t) => tray = t,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::error!(
|
log::error!(
|
||||||
|
|||||||
@@ -102,8 +102,9 @@ pub fn setup_window(config: Arc<Mutex<Config>>) -> MainWindow {
|
|||||||
available.contains(&"xyz.ljones.Aura".to_string()),
|
available.contains(&"xyz.ljones.Aura".to_string()),
|
||||||
available.contains(&"xyz.ljones.Anime".to_string()),
|
available.contains(&"xyz.ljones.Anime".to_string()),
|
||||||
available.contains(&"xyz.ljones.FanCurves".to_string()),
|
available.contains(&"xyz.ljones.FanCurves".to_string()),
|
||||||
true,
|
true, // GPU Configuration
|
||||||
true,
|
true, // App Settings
|
||||||
|
true, // About
|
||||||
]
|
]
|
||||||
.into(),
|
.into(),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -168,15 +168,24 @@ pub fn setup_fan_curve_page(ui: &MainWindow, _config: Arc<Mutex<Config>>) {
|
|||||||
|
|
||||||
update_fan_data(handle, balanced, perf, quiet);
|
update_fan_data(handle, balanced, perf, quiet);
|
||||||
|
|
||||||
|
let choices_for_ui = platform_profile_choices.clone();
|
||||||
let handle_next1 = handle_copy.clone();
|
let handle_next1 = handle_copy.clone();
|
||||||
if let Err(e) = handle_copy.upgrade_in_event_loop(move |handle| {
|
if let Err(e) = handle_copy.upgrade_in_event_loop(move |handle| {
|
||||||
let global = handle.global::<FanPageData>();
|
let global = handle.global::<FanPageData>();
|
||||||
let fans1 = fans.clone();
|
let fans1 = fans.clone();
|
||||||
|
let choices = choices_for_ui.clone();
|
||||||
global.on_set_profile_default(move |profile| {
|
global.on_set_profile_default(move |profile| {
|
||||||
let fans = fans1.clone();
|
let fans = fans1.clone();
|
||||||
let handle_next = handle_next1.clone();
|
let handle_next = handle_next1.clone();
|
||||||
|
let choices = choices.clone();
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if fans.set_curves_to_defaults(profile.into()).await.is_err() {
|
let mut target: PlatformProfile = profile.into();
|
||||||
|
if target == PlatformProfile::Quiet
|
||||||
|
&& !choices.contains(&PlatformProfile::Quiet)
|
||||||
|
{
|
||||||
|
target = PlatformProfile::LowPower;
|
||||||
|
}
|
||||||
|
if fans.set_curves_to_defaults(target).await.is_err() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let Ok(balanced) = fans
|
let Ok(balanced) = fans
|
||||||
|
|||||||
@@ -40,6 +40,10 @@ pub fn setup_system_page(ui: &MainWindow, _config: Arc<Mutex<Config>>) {
|
|||||||
ui.global::<SystemPageData>().set_platform_profile(-1);
|
ui.global::<SystemPageData>().set_platform_profile(-1);
|
||||||
ui.global::<SystemPageData>().set_panel_overdrive(-1);
|
ui.global::<SystemPageData>().set_panel_overdrive(-1);
|
||||||
ui.global::<SystemPageData>().set_boot_sound(-1);
|
ui.global::<SystemPageData>().set_boot_sound(-1);
|
||||||
|
ui.global::<SystemPageData>().set_kbd_leds_awake(-1);
|
||||||
|
ui.global::<SystemPageData>().set_kbd_leds_sleep(-1);
|
||||||
|
ui.global::<SystemPageData>().set_kbd_leds_boot(-1);
|
||||||
|
ui.global::<SystemPageData>().set_kbd_leds_shutdown(-1);
|
||||||
ui.global::<SystemPageData>().set_screen_auto_brightness(-1);
|
ui.global::<SystemPageData>().set_screen_auto_brightness(-1);
|
||||||
ui.global::<SystemPageData>().set_mcu_powersave(-1);
|
ui.global::<SystemPageData>().set_mcu_powersave(-1);
|
||||||
ui.global::<SystemPageData>().set_mini_led_mode(-1);
|
ui.global::<SystemPageData>().set_mini_led_mode(-1);
|
||||||
@@ -669,6 +673,26 @@ pub fn setup_system_page_callbacks(ui: &MainWindow, _states: Arc<Mutex<Config>>)
|
|||||||
setup_callback!(boot_sound, handle, attr, i32);
|
setup_callback!(boot_sound, handle, attr, i32);
|
||||||
setup_external!(boot_sound, i32, handle, attr, value)
|
setup_external!(boot_sound, i32, handle, attr, value)
|
||||||
}
|
}
|
||||||
|
FirmwareAttribute::KbdLedsAwake => {
|
||||||
|
init_property!(kbd_leds_awake, handle, value, i32);
|
||||||
|
setup_callback!(kbd_leds_awake, handle, attr, i32);
|
||||||
|
setup_external!(kbd_leds_awake, i32, handle, attr, value)
|
||||||
|
}
|
||||||
|
FirmwareAttribute::KbdLedsSleep => {
|
||||||
|
init_property!(kbd_leds_sleep, handle, value, i32);
|
||||||
|
setup_callback!(kbd_leds_sleep, handle, attr, i32);
|
||||||
|
setup_external!(kbd_leds_sleep, i32, handle, attr, value)
|
||||||
|
}
|
||||||
|
FirmwareAttribute::KbdLedsBoot => {
|
||||||
|
init_property!(kbd_leds_boot, handle, value, i32);
|
||||||
|
setup_callback!(kbd_leds_boot, handle, attr, i32);
|
||||||
|
setup_external!(kbd_leds_boot, i32, handle, attr, value)
|
||||||
|
}
|
||||||
|
FirmwareAttribute::KbdLedsShutdown => {
|
||||||
|
init_property!(kbd_leds_shutdown, handle, value, i32);
|
||||||
|
setup_callback!(kbd_leds_shutdown, handle, attr, i32);
|
||||||
|
setup_external!(kbd_leds_shutdown, i32, handle, attr, value)
|
||||||
|
}
|
||||||
FirmwareAttribute::ScreenAutoBrightness => {
|
FirmwareAttribute::ScreenAutoBrightness => {
|
||||||
init_property!(screen_auto_brightness, handle, value, i32);
|
init_property!(screen_auto_brightness, handle, value, i32);
|
||||||
setup_callback!(screen_auto_brightness, handle, attr, i32);
|
setup_callback!(screen_auto_brightness, handle, attr, i32);
|
||||||
|
|||||||
Binary file not shown.
810
rog-control-center/translations/uk_UA/rog-control-center.po
Normal file
810
rog-control-center/translations/uk_UA/rog-control-center.po
Normal file
@@ -0,0 +1,810 @@
|
|||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: rog-control-center\n"
|
||||||
|
"POT-Creation-Date: 2026-01-16 22:25+0000\n"
|
||||||
|
"PO-Revision-Date: 2024-07-28 12:00+0300\n"
|
||||||
|
"Last-Translator: Mykola Shevchenko\n"
|
||||||
|
"Language-Team: Ukrainian\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
|
"Language: uk_UA\n"
|
||||||
|
"Plural-Forms: nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);\n"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:55
|
||||||
|
msgctxt "MainWindow"
|
||||||
|
msgid "ROG"
|
||||||
|
msgstr "ROG"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:57
|
||||||
|
msgctxt "Menu1"
|
||||||
|
msgid "System Control"
|
||||||
|
msgstr "Системні"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:58
|
||||||
|
msgctxt "Menu2"
|
||||||
|
msgid "Keyboard Aura"
|
||||||
|
msgstr "Aura клавіатури"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:59
|
||||||
|
msgctxt "Menu3"
|
||||||
|
msgid "AniMe Matrix"
|
||||||
|
msgstr "AniMe матриця"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:60
|
||||||
|
msgctxt "Menu4"
|
||||||
|
msgid "Fan Curves"
|
||||||
|
msgstr "Криві вентиляторів"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:61
|
||||||
|
msgctxt "Menu5"
|
||||||
|
msgid "App Settings"
|
||||||
|
msgstr "Налаштування"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:62
|
||||||
|
msgctxt "Menu6"
|
||||||
|
msgid "About"
|
||||||
|
msgstr "Про додаток"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/main_window.slint:74
|
||||||
|
msgctxt "MainWindow"
|
||||||
|
msgid "Quit App"
|
||||||
|
msgstr "Вийти"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:6
|
||||||
|
msgctxt "Anime Brightness"
|
||||||
|
msgid "Off"
|
||||||
|
msgstr "Вимкнено"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:7
|
||||||
|
msgctxt "Anime Brightness"
|
||||||
|
msgid "Low"
|
||||||
|
msgstr "Низька"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:8
|
||||||
|
msgctxt "Anime Brightness"
|
||||||
|
msgid "Med"
|
||||||
|
msgstr "Середня"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:9
|
||||||
|
msgctxt "Anime Brightness"
|
||||||
|
msgid "High"
|
||||||
|
msgstr "Висока"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:23
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "Glitch Construction"
|
||||||
|
msgstr "Збій конструкції"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:23
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "Static Emergence"
|
||||||
|
msgstr "Статична поява"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:25
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "Binary Banner Scroll"
|
||||||
|
msgstr "Прокрутка бінарного банера"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:25
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "Rog Logo Glitch"
|
||||||
|
msgstr "Збій логотипу Rog"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:27
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "Banner Swipe"
|
||||||
|
msgstr "Прогортання банера"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:27
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "Starfield"
|
||||||
|
msgstr "Зоряне поле"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:29
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "Glitch Out"
|
||||||
|
msgstr "Збій"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:29
|
||||||
|
msgctxt "AnimePageData"
|
||||||
|
msgid "See Ya"
|
||||||
|
msgstr "Бувай"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:50
|
||||||
|
msgctxt "Anime Brightness"
|
||||||
|
msgid "Brightness"
|
||||||
|
msgstr "Яскравість"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:66
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Enable display"
|
||||||
|
msgstr "Увімкнути показ"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:74 rog-control-center/ui/pages/anime.slint:97
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Advanced"
|
||||||
|
msgstr "Розширені"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:89
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Use built-in animations"
|
||||||
|
msgstr "Використовувати вбудовані анімації"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:146
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Set which builtin animations are played"
|
||||||
|
msgstr "Встановити, які вбудовані анімації відтворювати"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:150
|
||||||
|
msgctxt "Anime built-in selection"
|
||||||
|
msgid "Boot Animation"
|
||||||
|
msgstr "Анімація завантаження"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:160
|
||||||
|
msgctxt "Anime built-in selection"
|
||||||
|
msgid "Running Animation"
|
||||||
|
msgstr "Анімація роботи"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:170
|
||||||
|
msgctxt "Anime built-in selection"
|
||||||
|
msgid "Sleep Animation"
|
||||||
|
msgstr "Анімація сну"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:180
|
||||||
|
msgctxt "Anime built-in selection"
|
||||||
|
msgid "Shutdown Animation"
|
||||||
|
msgstr "Анімація вимкнення"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:220
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Advanced Display Settings"
|
||||||
|
msgstr "Розширені налаштування показу"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:225
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Off when lid closed"
|
||||||
|
msgstr "Вимикати при закритій кришці"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:234
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Off when suspended"
|
||||||
|
msgstr "Вимикати в режимі сну"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/anime.slint:243
|
||||||
|
msgctxt "PageAnime"
|
||||||
|
msgid "Off when on battery"
|
||||||
|
msgstr "Вимикати при роботі від батареї"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:20
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "Balanced"
|
||||||
|
msgstr "Збалансований"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:20 rog-control-center/ui/pages/system.slint:27
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "Performance"
|
||||||
|
msgstr "Продуктивний"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:20
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "Quiet"
|
||||||
|
msgstr "Тихий"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:20
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "LowPower"
|
||||||
|
msgstr "Низьке споживання"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:26
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "Default"
|
||||||
|
msgstr "За замовчуванням"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:28
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "BalancePerformance"
|
||||||
|
msgstr "Баланс-Продуктивність"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:29
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "BalancePower"
|
||||||
|
msgstr "Баланс-Енергозбереження"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:30
|
||||||
|
msgctxt "SystemPageData"
|
||||||
|
msgid "Power"
|
||||||
|
msgstr "Енергозбереження"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:159
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Power settings"
|
||||||
|
msgstr "Налаштування живлення"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:164
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Charge limit"
|
||||||
|
msgstr "Ліміт заряду"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:179
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Platform Profile"
|
||||||
|
msgstr "Профіль платформи"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:189
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Advanced"
|
||||||
|
msgstr "Розширені"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:209
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Screenpad brightness"
|
||||||
|
msgstr "Яскравість екранної панелі"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:233
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Sync with primary"
|
||||||
|
msgstr "Синхронізувати з основним"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:253
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Armoury settings"
|
||||||
|
msgstr "Налаштування Armoury"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:253
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Keyboard Power Management"
|
||||||
|
msgstr "Керування живленням клавіатури"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:263
|
||||||
|
msgctxt "no_asus_armoury_driver_1"
|
||||||
|
msgid "The asus-armoury driver is not loaded"
|
||||||
|
msgstr "Драйвер asus-armoury не завантажено"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:269
|
||||||
|
msgctxt "no_asus_armoury_driver_2"
|
||||||
|
msgid "For advanced features you will require a kernel with this driver added."
|
||||||
|
msgstr "Для розширених функцій вам знадобиться ядро з доданим цим драйвером."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:280
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Panel Overdrive"
|
||||||
|
msgstr "Розгін матриці"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:288
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "MiniLED Mode"
|
||||||
|
msgstr "Режим MiniLED"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:296
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "POST boot sound"
|
||||||
|
msgstr "Звук при завантаженні"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:312
|
||||||
|
msgctxt "ppt_warning"
|
||||||
|
msgid "The following settings are not applied until the toggle is enabled."
|
||||||
|
msgstr "Наступні налаштування не застосовуються, доки перемикач не буде увімкнено."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:317 rog-control-center/ui/pages/system.slint:324
|
||||||
|
msgctxt "ppt_group_enabled"
|
||||||
|
msgid "Enable Tuning"
|
||||||
|
msgstr "Увімкнути налаштування"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:334 rog-control-center/ui/pages/system.slint:335
|
||||||
|
msgctxt "ppt_pl1_spl"
|
||||||
|
msgid "CPU Sustained Power Limit"
|
||||||
|
msgstr "Тривалий ліміт потужності CPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:336
|
||||||
|
msgctxt "ppt_pl1_spl_help"
|
||||||
|
msgid ""
|
||||||
|
"Long-term CPU power limit that affects sustained workload performance. "
|
||||||
|
"Higher values may increase heat and power consumption."
|
||||||
|
msgstr ""
|
||||||
|
"Довготривалий ліміт потужності CPU, що впливає на продуктивність при тривалих навантаженнях."
|
||||||
|
"Вищі значення можуть збільшити нагрівання та споживання енергії."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:352 rog-control-center/ui/pages/system.slint:353
|
||||||
|
msgctxt "ppt_pl2_sppt"
|
||||||
|
msgid "CPU Turbo Power Limit"
|
||||||
|
msgstr "Турбо ліміт потужності CPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:354
|
||||||
|
msgctxt "ppt_pl2_sppt_help"
|
||||||
|
msgid ""
|
||||||
|
"Short-term CPU power limit for boost periods. Controls maximum power during "
|
||||||
|
"brief high-performance bursts."
|
||||||
|
msgstr ""
|
||||||
|
"Короткочасний ліміт потужності CPU для періодів прискорення. Контролює максимальну"
|
||||||
|
"потужність під час коротких сплесків високої продуктивності."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:370 rog-control-center/ui/pages/system.slint:371
|
||||||
|
msgctxt "ppt_pl3_fppt"
|
||||||
|
msgid "CPU Fast Burst Power Limit"
|
||||||
|
msgstr "Швидкий ліміт потужності CPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:372
|
||||||
|
msgctxt "ppt_pl3_fppt_help"
|
||||||
|
msgid ""
|
||||||
|
"Ultra-short duration power limit for instantaneous CPU bursts. Affects "
|
||||||
|
"responsiveness during sudden workload spikes."
|
||||||
|
msgstr ""
|
||||||
|
"Надкороткий ліміт потужності для миттєвих сплесків CPU."
|
||||||
|
"Впливає на чутливість під час раптових піків навантаження."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:387 rog-control-center/ui/pages/system.slint:388
|
||||||
|
msgctxt "ppt_fppt"
|
||||||
|
msgid "Fast Package Power Limit"
|
||||||
|
msgstr "Швидкий ліміт потужності пакета"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:389
|
||||||
|
msgctxt "ppt_fppt_help"
|
||||||
|
msgid ""
|
||||||
|
"Ultra-short duration power limit for system package. Controls maximum power "
|
||||||
|
"during millisecond-scale load spikes."
|
||||||
|
msgstr ""
|
||||||
|
"Надкороткий ліміт потужності для системного пакета. Контролює максимальну"
|
||||||
|
"потужність під час пікових навантажень мілісекундного масштабу."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:405 rog-control-center/ui/pages/system.slint:406
|
||||||
|
msgctxt "ppt_apu_sppt"
|
||||||
|
msgid "APU Sustained Power Limit"
|
||||||
|
msgstr "Тривалий ліміт потужності APU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:407
|
||||||
|
msgctxt "ppt_apu_sppt_help"
|
||||||
|
msgid ""
|
||||||
|
"Long-term power limit for integrated graphics and CPU combined. Affects "
|
||||||
|
"sustained performance of APU-based workloads."
|
||||||
|
msgstr ""
|
||||||
|
"Довготривалий ліміт потужності для інтегрованої графіки та CPU разом."
|
||||||
|
"Впливає на тривалу продуктивність навантажень на базі APU."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:423 rog-control-center/ui/pages/system.slint:424
|
||||||
|
msgctxt "ppt_platform_sppt"
|
||||||
|
msgid "Platform Sustained Power Limit"
|
||||||
|
msgstr "Тривалий ліміт потужності платформи"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:425
|
||||||
|
msgctxt "ppt_platform_sppt_help"
|
||||||
|
msgid ""
|
||||||
|
"Overall system power limit for sustained operations. Controls total platform "
|
||||||
|
"power consumption over extended periods."
|
||||||
|
msgstr ""
|
||||||
|
"Загальний ліміт потужності системи для тривалих операцій. Контролює загальне"
|
||||||
|
"споживання енергії платформою протягом тривалих періодів."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:441 rog-control-center/ui/pages/system.slint:442
|
||||||
|
msgctxt "nv_dynamic_boost"
|
||||||
|
msgid "GPU Power Boost"
|
||||||
|
msgstr "Прискорення потужності GPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:443
|
||||||
|
msgctxt "nv_dynamic_boost_help"
|
||||||
|
msgid ""
|
||||||
|
"Additional power allocation for GPU dynamic boost. Higher values increase "
|
||||||
|
"GPU performance but generate more heat."
|
||||||
|
msgstr ""
|
||||||
|
"Додаткове виділення потужності для динамічного прискорення GPU. Вищі значення"
|
||||||
|
"збільшують продуктивність GPU, але генерують більше тепла."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:459 rog-control-center/ui/pages/system.slint:460
|
||||||
|
msgctxt "nv_temp_target"
|
||||||
|
msgid "GPU Temperature Limit"
|
||||||
|
msgstr "Ліміт температури GPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:461
|
||||||
|
msgctxt "nv_temp_target_help"
|
||||||
|
msgid ""
|
||||||
|
"Maximum GPU temperature threshold in Celsius. GPU will throttle to maintain "
|
||||||
|
"temperature below this limit."
|
||||||
|
msgstr ""
|
||||||
|
"Максимальний поріг температури GPU у градусах Цельсія. GPU буде знижувати"
|
||||||
|
"частоту для підтримки температури нижче цього ліміту."
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:513
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Energy Performance Preference linked to Throttle Policy"
|
||||||
|
msgstr "Перевага енергоефективності пов'язана з Політикою Тротлінгу"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:517
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Change EPP based on Throttle Policy"
|
||||||
|
msgstr "Змінювати EPP на основі Політики Тротлінгу"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:525
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "EPP for Balanced Policy"
|
||||||
|
msgstr "EPP для Збалансованої Політики"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:535
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "EPP for Performance Policy"
|
||||||
|
msgstr "EPP для Політики Продуктивності"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:545
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "EPP for Quiet Policy"
|
||||||
|
msgstr "EPP для Тихої Політики"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:563
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Throttle Policy for power state"
|
||||||
|
msgstr "Політика Тротлінгу для стану живлення"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:569
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Throttle Policy on Battery"
|
||||||
|
msgstr "Політика Тротлінгу при живленні від батареї"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:579 rog-control-center/ui/pages/system.slint:600
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Enabled"
|
||||||
|
msgstr "Увімкнено"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/system.slint:590
|
||||||
|
msgctxt "PageSystem"
|
||||||
|
msgid "Throttle Policy on AC"
|
||||||
|
msgstr "Політика Тротлінгу при живленні від мережі"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:28
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Brightness"
|
||||||
|
msgstr "Яскравість"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:39
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Aura mode"
|
||||||
|
msgstr "Режим Aura"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:59
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Colour 1"
|
||||||
|
msgstr "Колір 1"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:85
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Colour 2"
|
||||||
|
msgstr "Колір 2"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:119
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Zone"
|
||||||
|
msgstr "Зона"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:142
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Direction"
|
||||||
|
msgstr "Напрямок"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:164
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Speed"
|
||||||
|
msgstr "Швидкість"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:185
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Power Settings"
|
||||||
|
msgstr "Налаштування живлення"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/aura.slint:270
|
||||||
|
msgctxt "PageAura"
|
||||||
|
msgid "Power Zones"
|
||||||
|
msgstr "Зони живлення"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/app_settings.slint:26
|
||||||
|
msgctxt "PageAppSettings"
|
||||||
|
msgid "Run in background after closing"
|
||||||
|
msgstr "Працювати у фоні після закриття"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/app_settings.slint:34
|
||||||
|
msgctxt "PageAppSettings"
|
||||||
|
msgid "Start app in background (UI closed)"
|
||||||
|
msgstr "Запускати у фоні (без інтерфейсу)"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/app_settings.slint:42
|
||||||
|
msgctxt "PageAppSettings"
|
||||||
|
msgid "Enable system tray icon"
|
||||||
|
msgstr "Увімкнути іконку в треї"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/app_settings.slint:50
|
||||||
|
msgctxt "PageAppSettings"
|
||||||
|
msgid "Enable dGPU notifications"
|
||||||
|
msgstr "Увімкнути сповіщення про dGPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:26
|
||||||
|
msgctxt "FanTab"
|
||||||
|
msgid "This fan is not avilable on this machine"
|
||||||
|
msgstr "Цей вентилятор недоступний на цьому пристрої"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:34
|
||||||
|
msgctxt "FanTab"
|
||||||
|
msgid "Enabled"
|
||||||
|
msgstr "Увімкнено"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:43
|
||||||
|
msgctxt "FanTab"
|
||||||
|
msgid "Apply"
|
||||||
|
msgstr "Застосувати"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:51
|
||||||
|
msgctxt "FanTab"
|
||||||
|
msgid "Cancel"
|
||||||
|
msgstr "Скасувати"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:59
|
||||||
|
msgctxt "FanTab"
|
||||||
|
msgid "Factory Default (all fans)"
|
||||||
|
msgstr "Заводські налаштування (всі вентилятори)"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:72
|
||||||
|
msgctxt "PageFans"
|
||||||
|
msgid "Balanced"
|
||||||
|
msgstr "Збалансований"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:75 rog-control-center/ui/pages/fans.slint:134 rog-control-center/ui/pages/fans.slint:193
|
||||||
|
msgctxt "PageFans"
|
||||||
|
msgid "CPU"
|
||||||
|
msgstr "CPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:93 rog-control-center/ui/pages/fans.slint:152 rog-control-center/ui/pages/fans.slint:211
|
||||||
|
msgctxt "PageFans"
|
||||||
|
msgid "Mid"
|
||||||
|
msgstr "Середній"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:111 rog-control-center/ui/pages/fans.slint:170 rog-control-center/ui/pages/fans.slint:229
|
||||||
|
msgctxt "PageFans"
|
||||||
|
msgid "GPU"
|
||||||
|
msgstr "GPU"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:131
|
||||||
|
msgctxt "PageFans"
|
||||||
|
msgid "Performance"
|
||||||
|
msgstr "Продуктивний"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/pages/fans.slint:190
|
||||||
|
msgctxt "PageFans"
|
||||||
|
msgid "Quiet"
|
||||||
|
msgstr "Тихий"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/common.slint:126
|
||||||
|
msgctxt "confirm_reset"
|
||||||
|
msgid "Are you sure you want to reset this?"
|
||||||
|
msgstr "Ви впевнені, що хочете скинути це?"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:33
|
||||||
|
msgctxt "AuraPowerGroup"
|
||||||
|
msgid "Boot"
|
||||||
|
msgstr "Запуск"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:43
|
||||||
|
msgctxt "AuraPowerGroup"
|
||||||
|
msgid "Awake"
|
||||||
|
msgstr "Робота"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:53
|
||||||
|
msgctxt "AuraPowerGroup"
|
||||||
|
msgid "Sleep"
|
||||||
|
msgstr "Сон"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:63
|
||||||
|
msgctxt "AuraPowerGroup"
|
||||||
|
msgid "Shutdown"
|
||||||
|
msgstr "Вимкнення"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:102
|
||||||
|
msgctxt "AuraPowerGroupOld"
|
||||||
|
msgid "Zone Selection"
|
||||||
|
msgstr "Вибір зони"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:114
|
||||||
|
msgctxt "AuraPowerGroupOld"
|
||||||
|
msgid "Boot"
|
||||||
|
msgstr "Завантаження"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:124
|
||||||
|
msgctxt "AuraPowerGroupOld"
|
||||||
|
msgid "Awake"
|
||||||
|
msgstr "Робота"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/widgets/aura_power.slint:134
|
||||||
|
msgctxt "AuraPowerGroupOld"
|
||||||
|
msgid "Sleep"
|
||||||
|
msgstr "Сон"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:52
|
||||||
|
msgctxt "Aura power zone"
|
||||||
|
msgid "Logo"
|
||||||
|
msgstr "Логотип"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:53 rog-control-center/ui/types/aura_types.slint:63
|
||||||
|
msgctxt "Aura power zone"
|
||||||
|
msgid "Keyboard"
|
||||||
|
msgstr "Клавіатура"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:54 rog-control-center/ui/types/aura_types.slint:64
|
||||||
|
msgctxt "Aura power zone"
|
||||||
|
msgid "Lightbar"
|
||||||
|
msgstr "Світлова панель"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:55
|
||||||
|
msgctxt "Aura power zone"
|
||||||
|
msgid "Lid"
|
||||||
|
msgstr "Кришка"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:56
|
||||||
|
msgctxt "Aura power zone"
|
||||||
|
msgid "Rear Glow"
|
||||||
|
msgstr "Заднє світіння"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:57 rog-control-center/ui/types/aura_types.slint:65
|
||||||
|
msgctxt "Aura power zone"
|
||||||
|
msgid "Keyboard and Lightbar"
|
||||||
|
msgstr "Клавіатура та світлова панель"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:58
|
||||||
|
msgctxt "Aura power zone"
|
||||||
|
msgid "Ally"
|
||||||
|
msgstr "Ally"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:68
|
||||||
|
msgctxt "Aura brightness"
|
||||||
|
msgid "Off"
|
||||||
|
msgstr "Вимкнено"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:69
|
||||||
|
msgctxt "Aura brightness"
|
||||||
|
msgid "Low"
|
||||||
|
msgstr "Низька"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:70
|
||||||
|
msgctxt "Aura brightness"
|
||||||
|
msgid "Med"
|
||||||
|
msgstr "Середня"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:71
|
||||||
|
msgctxt "Aura brightness"
|
||||||
|
msgid "High"
|
||||||
|
msgstr "Висока"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:76 rog-control-center/ui/types/aura_types.slint:91
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Static"
|
||||||
|
msgstr "Статичний"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:77 rog-control-center/ui/types/aura_types.slint:92
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Breathe"
|
||||||
|
msgstr "Дихання"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:78 rog-control-center/ui/types/aura_types.slint:93
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Strobe"
|
||||||
|
msgstr "Стробоскоп"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:79
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Rainbow"
|
||||||
|
msgstr "Веселка"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:80
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Star"
|
||||||
|
msgstr "Зірка"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:81
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Rain"
|
||||||
|
msgstr "Дощ"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:82
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Highlight"
|
||||||
|
msgstr "Підсвічування"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:83
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Laser"
|
||||||
|
msgstr "Лазер"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:84
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Ripple"
|
||||||
|
msgstr "Хвиля"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:85
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Nothing"
|
||||||
|
msgstr "Нічого"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:86
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Pulse"
|
||||||
|
msgstr "Пульс"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:87
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Comet"
|
||||||
|
msgstr "Комета"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:88
|
||||||
|
msgctxt "Basic aura mode"
|
||||||
|
msgid "Flash"
|
||||||
|
msgstr "Спалах"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:100
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "None"
|
||||||
|
msgstr "Немає"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:101
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "Key1"
|
||||||
|
msgstr "Клавіша 1"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:102
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "Key2"
|
||||||
|
msgstr "Клавіша 2"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:103
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "Key3"
|
||||||
|
msgstr "Клавіша 3"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:104
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "Key4"
|
||||||
|
msgstr "Клавіша 4"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:105
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "Logo"
|
||||||
|
msgstr "Логотип"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:106
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "Lightbar Left"
|
||||||
|
msgstr "Світлова панель зліва"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:107
|
||||||
|
msgctxt "Aura zone"
|
||||||
|
msgid "Lightbar Right"
|
||||||
|
msgstr "Світлова панель справа"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:111
|
||||||
|
msgctxt "Aura direction"
|
||||||
|
msgid "Right"
|
||||||
|
msgstr "Вправо"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:112
|
||||||
|
msgctxt "Aura direction"
|
||||||
|
msgid "Left"
|
||||||
|
msgstr "Вліво"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:113
|
||||||
|
msgctxt "Aura direction"
|
||||||
|
msgid "Up"
|
||||||
|
msgstr "Вгору"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:114
|
||||||
|
msgctxt "Aura direction"
|
||||||
|
msgid "Down"
|
||||||
|
msgstr "Вниз"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:118
|
||||||
|
msgctxt "Aura speed"
|
||||||
|
msgid "Low"
|
||||||
|
msgstr "Низька"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:119
|
||||||
|
msgctxt "Aura speed"
|
||||||
|
msgid "Medium"
|
||||||
|
msgstr "Середня"
|
||||||
|
|
||||||
|
#: rog-control-center/ui/types/aura_types.slint:120
|
||||||
|
msgctxt "Aura speed"
|
||||||
|
msgid "High"
|
||||||
|
msgstr "Висока"
|
||||||
|
|
||||||
@@ -7,6 +7,7 @@ import { PageFans } from "pages/fans.slint";
|
|||||||
import { PageAnime, AnimePageData } from "pages/anime.slint";
|
import { PageAnime, AnimePageData } from "pages/anime.slint";
|
||||||
import { RogItem } from "widgets/common.slint";
|
import { RogItem } from "widgets/common.slint";
|
||||||
import { PageAura } from "pages/aura.slint";
|
import { PageAura } from "pages/aura.slint";
|
||||||
|
import { PageGPU } from "pages/gpu.slint";
|
||||||
import { Node } from "widgets/graph.slint";
|
import { Node } from "widgets/graph.slint";
|
||||||
export { Node }
|
export { Node }
|
||||||
import { FanPageData, FanType, Profile } from "types/fan_types.slint";
|
import { FanPageData, FanType, Profile } from "types/fan_types.slint";
|
||||||
@@ -24,7 +25,15 @@ export component MainWindow inherits Window {
|
|||||||
default-font-size: 14px;
|
default-font-size: 14px;
|
||||||
default-font-weight: 400;
|
default-font-weight: 400;
|
||||||
icon: @image-url("../data/rog-control-center.png");
|
icon: @image-url("../data/rog-control-center.png");
|
||||||
in property <[bool]> sidebar_items_avilable: [true, true, true, true, true, true];
|
in property <[bool]> sidebar_items_avilable: [
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
true, // GPU Configuration
|
||||||
|
true, // App Settings
|
||||||
|
true, // About
|
||||||
|
];
|
||||||
private property <bool> show_notif;
|
private property <bool> show_notif;
|
||||||
private property <bool> fade_cover;
|
private property <bool> fade_cover;
|
||||||
private property <bool> toast: false;
|
private property <bool> toast: false;
|
||||||
@@ -58,8 +67,9 @@ export component MainWindow inherits Window {
|
|||||||
@tr("Menu2" => "Keyboard Aura"),
|
@tr("Menu2" => "Keyboard Aura"),
|
||||||
@tr("Menu3" => "AniMe Matrix"),
|
@tr("Menu3" => "AniMe Matrix"),
|
||||||
@tr("Menu4" => "Fan Curves"),
|
@tr("Menu4" => "Fan Curves"),
|
||||||
@tr("Menu5" => "App Settings"),
|
@tr("Menu5" => "GPU Configuration"),
|
||||||
@tr("Menu6" => "About"),
|
@tr("Menu6" => "App Settings"),
|
||||||
|
@tr("Menu7" => "About"),
|
||||||
];
|
];
|
||||||
available: root.sidebar_items_avilable;
|
available: root.sidebar_items_avilable;
|
||||||
}
|
}
|
||||||
@@ -89,26 +99,61 @@ export component MainWindow inherits Window {
|
|||||||
height: root.height + 12px;
|
height: root.height + 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
aura := PageAura {
|
/*if(side-bar.current-item == 1):*/ aura := PageAura {
|
||||||
width: root.width - side-bar.width;
|
width: root.width - side-bar.width;
|
||||||
visible: side-bar.current-item == 1;
|
visible: side-bar.current-item == 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(side-bar.current-item == 2): PageAnime {
|
if(side-bar.current-item == 2): PageAnime {
|
||||||
width: root.width - side-bar.width;
|
width: root.width - side-bar.width;
|
||||||
|
visible: side-bar.current-item == 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
fans := PageFans {
|
if(side-bar.current-item == 3): fans := PageFans {
|
||||||
width: root.width - side-bar.width;
|
width: root.width - side-bar.width;
|
||||||
visible: side-bar.current-item == 3;
|
visible: side-bar.current-item == 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(side-bar.current-item == 4): PageAppSettings {
|
if(side-bar.current-item == 4): PageGPU {
|
||||||
width: root.width - side-bar.width;
|
width: root.width - side-bar.width;
|
||||||
|
visible: side-bar.current-item == 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(side-bar.current-item == 5): PageAbout {
|
if(side-bar.current-item == 5): PageAppSettings {
|
||||||
width: root.width - side-bar.width;
|
width: root.width - side-bar.width;
|
||||||
|
visible: side-bar.current-item == 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(side-bar.current-item == 6): PageAbout {
|
||||||
|
width: root.width - side-bar.width;
|
||||||
|
visible: side-bar.current-item == 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
if toast: Rectangle {
|
||||||
|
x: 0px;
|
||||||
|
y: root.height - self.height;
|
||||||
|
width: root.width - side-bar.width;
|
||||||
|
height: 40px;
|
||||||
|
opacity: 1.0;
|
||||||
|
background: Palette.selection-background;
|
||||||
|
clip: true;
|
||||||
|
TouchArea {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
clicked => {
|
||||||
|
toast = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
background: Palette.control-background;
|
||||||
|
Text {
|
||||||
|
color: Palette.control-foreground;
|
||||||
|
text: root.toast_text;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -133,31 +178,6 @@ export component MainWindow inherits Window {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if toast: Rectangle {
|
|
||||||
x: 0px;
|
|
||||||
y: 0px;
|
|
||||||
width: root.width;
|
|
||||||
height: 32px;
|
|
||||||
opacity: 1.0;
|
|
||||||
background: Colors.grey;
|
|
||||||
TouchArea {
|
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
clicked => {
|
|
||||||
toast = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Rectangle {
|
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
background: Palette.control-background;
|
|
||||||
Text {
|
|
||||||
color: Palette.control-foreground;
|
|
||||||
text: root.toast_text;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// // TODO: or use Dialogue
|
// // TODO: or use Dialogue
|
||||||
if show_notif: Rectangle {
|
if show_notif: Rectangle {
|
||||||
@@ -190,7 +210,7 @@ export component MainWindow inherits Window {
|
|||||||
y: 0px;
|
y: 0px;
|
||||||
width: root.width;
|
width: root.width;
|
||||||
height: root.height;
|
height: root.height;
|
||||||
|
|
||||||
//padding only has effect on layout elements
|
//padding only has effect on layout elements
|
||||||
//padding: 10px;
|
//padding: 10px;
|
||||||
|
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ export component PageAbout inherits VerticalLayout {
|
|||||||
|
|
||||||
Text {
|
Text {
|
||||||
wrap: TextWrap.word-wrap;
|
wrap: TextWrap.word-wrap;
|
||||||
text: "You will require a kernel built with my work from here: https://github.com/flukejones/linux";
|
text: "You need to use kernel version 6.19 to use this software";
|
||||||
}
|
}
|
||||||
|
|
||||||
Text {
|
Text {
|
||||||
@@ -43,10 +43,6 @@ export component PageAbout inherits VerticalLayout {
|
|||||||
text: "- [ ] Slash control";
|
text: "- [ ] Slash control";
|
||||||
}
|
}
|
||||||
|
|
||||||
Text {
|
|
||||||
text: "- [ ] Supergfx control";
|
|
||||||
}
|
|
||||||
|
|
||||||
Text {
|
Text {
|
||||||
text: "- [ ] Screenpad controls";
|
text: "- [ ] Screenpad controls";
|
||||||
}
|
}
|
||||||
|
|||||||
102
rog-control-center/ui/pages/gpu.slint
Normal file
102
rog-control-center/ui/pages/gpu.slint
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
import { Palette, TabWidget, Button, CheckBox } from "std-widgets.slint";
|
||||||
|
import { Graph, Node } from "../widgets/graph.slint";
|
||||||
|
import { SystemToggle } from "../widgets/common.slint";
|
||||||
|
import { Profile, FanType, FanPageData } from "../types/fan_types.slint";
|
||||||
|
import { Palette, HorizontalBox , VerticalBox, ScrollView, Slider, Button, Switch, ComboBox, GroupBox, StandardButton} from "std-widgets.slint";
|
||||||
|
|
||||||
|
export global GPUPageData {
|
||||||
|
// GPU mode and device state
|
||||||
|
in-out property <int> gpu_mux_mode: 1; // 0 = Ultra/Discreet, 1 = Integrated/Optimus
|
||||||
|
in-out property <int> dgpu_disabled: 0; // 1 == dGPU disabled
|
||||||
|
in-out property <int> egpu_enabled: 0; // 1 == eGPU (XGMobile) enabled
|
||||||
|
callback cb_gpu_mux_mode(int);
|
||||||
|
callback cb_dgpu_disabled(int);
|
||||||
|
callback cb_egpu_enabled(int);
|
||||||
|
}
|
||||||
|
|
||||||
|
export component PageGPU inherits Rectangle {
|
||||||
|
|
||||||
|
clip: true;
|
||||||
|
ScrollView {
|
||||||
|
VerticalLayout {
|
||||||
|
padding: 10px;
|
||||||
|
spacing: 10px;
|
||||||
|
alignment: LayoutAlignment.start;
|
||||||
|
Rectangle {
|
||||||
|
background: Palette.alternate-background;
|
||||||
|
border-color: Palette.accent-background;
|
||||||
|
border-width: 3px;
|
||||||
|
border-radius: 10px;
|
||||||
|
height: 40px;
|
||||||
|
Text {
|
||||||
|
font-size: 18px;
|
||||||
|
color: Palette.control-foreground;
|
||||||
|
horizontal-alignment: TextHorizontalAlignment.center;
|
||||||
|
text: @tr("GPU Configuration");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GroupBox {
|
||||||
|
HorizontalLayout {
|
||||||
|
spacing: 10px;
|
||||||
|
|
||||||
|
// Ultra (discreet) mode button - disabled if dGPU is marked disabled
|
||||||
|
Rectangle {
|
||||||
|
width: 120px;
|
||||||
|
height: 36px;
|
||||||
|
border-radius: 6px;
|
||||||
|
border-color: Palette.border;
|
||||||
|
border-width: 2px;
|
||||||
|
background: GPUPageData.gpu_mux_mode == 0 ? Palette.accent-background : Palette.control-background;
|
||||||
|
opacity: GPUPageData.dgpu_disabled == 1 ? 0.5 : 1.0;
|
||||||
|
|
||||||
|
Text {
|
||||||
|
color: Palette.control-foreground;
|
||||||
|
horizontal-alignment: TextHorizontalAlignment.center;
|
||||||
|
vertical-alignment: TextVerticalAlignment.center;
|
||||||
|
text: @tr("Ultra");
|
||||||
|
}
|
||||||
|
|
||||||
|
TouchArea {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
clicked => {
|
||||||
|
if (GPUPageData.dgpu_disabled != 1 && GPUPageData.gpu_mux_mode != 0) {
|
||||||
|
GPUPageData.cb_gpu_mux_mode(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Integrated (Optimus) mode button
|
||||||
|
Rectangle {
|
||||||
|
width: 120px;
|
||||||
|
height: 36px;
|
||||||
|
border-radius: 6px;
|
||||||
|
border-color: Palette.border;
|
||||||
|
border-width: 2px;
|
||||||
|
background: GPUPageData.gpu_mux_mode == 1 ? Palette.accent-background : Palette.control-background;
|
||||||
|
|
||||||
|
Text {
|
||||||
|
color: Palette.control-foreground;
|
||||||
|
horizontal-alignment: TextHorizontalAlignment.center;
|
||||||
|
vertical-alignment: TextVerticalAlignment.center;
|
||||||
|
text: @tr("Integrated");
|
||||||
|
}
|
||||||
|
|
||||||
|
TouchArea {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
clicked => {
|
||||||
|
if (GPUPageData.gpu_mux_mode != 1) {
|
||||||
|
GPUPageData.cb_gpu_mux_mode(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -51,6 +51,14 @@ export global SystemPageData {
|
|||||||
callback cb_panel_overdrive(int);
|
callback cb_panel_overdrive(int);
|
||||||
in-out property <int> boot_sound;
|
in-out property <int> boot_sound;
|
||||||
callback cb_boot_sound(int);
|
callback cb_boot_sound(int);
|
||||||
|
in-out property <int> kbd_leds_awake;
|
||||||
|
callback cb_kbd_leds_awake(int);
|
||||||
|
in-out property <int> kbd_leds_sleep;
|
||||||
|
callback cb_kbd_leds_sleep(int);
|
||||||
|
in-out property <int> kbd_leds_boot;
|
||||||
|
callback cb_kbd_leds_boot(int);
|
||||||
|
in-out property <int> kbd_leds_shutdown;
|
||||||
|
callback cb_kbd_leds_shutdown(int);
|
||||||
in-out property <int> screen_auto_brightness;
|
in-out property <int> screen_auto_brightness;
|
||||||
callback cb_screen_auto_brightness(int);
|
callback cb_screen_auto_brightness(int);
|
||||||
in-out property <int> mcu_powersave;
|
in-out property <int> mcu_powersave;
|
||||||
@@ -245,6 +253,66 @@ export component PageSystem inherits Rectangle {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if SystemPageData.kbd_leds_awake != -1 ||
|
||||||
|
SystemPageData.kbd_leds_sleep != -1 ||
|
||||||
|
SystemPageData.kbd_leds_boot != -1 ||
|
||||||
|
SystemPageData.kbd_leds_shutdown != -1: VerticalLayout {
|
||||||
|
padding: 0px;
|
||||||
|
spacing: 0px;
|
||||||
|
alignment: LayoutAlignment.start;
|
||||||
|
Rectangle {
|
||||||
|
background: Palette.alternate-background;
|
||||||
|
border-color: Palette.accent-background;
|
||||||
|
border-width: 3px;
|
||||||
|
border-radius: 10px;
|
||||||
|
height: 40px;
|
||||||
|
Text {
|
||||||
|
font-size: 18px;
|
||||||
|
color: Palette.control-foreground;
|
||||||
|
horizontal-alignment: TextHorizontalAlignment.center;
|
||||||
|
text: @tr("Keyboard Power Management");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GroupBox {
|
||||||
|
|
||||||
|
HorizontalLayout {
|
||||||
|
spacing: 10px;
|
||||||
|
if SystemPageData.kbd_leds_awake != -1: SystemToggleInt {
|
||||||
|
text: @tr("Keyboard Awake Effect");
|
||||||
|
checked_int <=> SystemPageData.kbd_leds_awake;
|
||||||
|
toggled => {
|
||||||
|
SystemPageData.cb_kbd_leds_awake(SystemPageData.kbd_leds_awake)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if SystemPageData.kbd_leds_sleep != -1: SystemToggleInt {
|
||||||
|
text: @tr("Keyboard Sleep Effect");
|
||||||
|
checked_int <=> SystemPageData.kbd_leds_sleep;
|
||||||
|
toggled => {
|
||||||
|
SystemPageData.cb_kbd_leds_sleep(SystemPageData.kbd_leds_sleep)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if SystemPageData.kbd_leds_boot != -1: SystemToggleInt {
|
||||||
|
text: @tr("Keyboard Boot Effect");
|
||||||
|
checked_int <=> SystemPageData.kbd_leds_boot;
|
||||||
|
toggled => {
|
||||||
|
SystemPageData.cb_kbd_leds_boot(SystemPageData.kbd_leds_boot)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if SystemPageData.kbd_leds_shutdown != -1: SystemToggleInt {
|
||||||
|
text: @tr("Keyboard Shutdown Effect");
|
||||||
|
checked_int <=> SystemPageData.kbd_leds_shutdown;
|
||||||
|
toggled => {
|
||||||
|
SystemPageData.cb_kbd_leds_shutdown(SystemPageData.kbd_leds_shutdown)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
background: Palette.alternate-background;
|
background: Palette.alternate-background;
|
||||||
border-color: Palette.accent-background;
|
border-color: Palette.accent-background;
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ use zbus::proxy;
|
|||||||
#[proxy(
|
#[proxy(
|
||||||
interface = "xyz.ljones.Anime",
|
interface = "xyz.ljones.Anime",
|
||||||
default_service = "xyz.ljones.Asusd",
|
default_service = "xyz.ljones.Asusd",
|
||||||
default_path = "/xyz/ljones"
|
default_path = "/xyz/ljones/aura/anime"
|
||||||
)]
|
)]
|
||||||
pub trait Anime {
|
pub trait Anime {
|
||||||
/// DeviceState method
|
/// DeviceState method
|
||||||
|
|||||||
@@ -253,8 +253,19 @@ impl FirmwareAttributes {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Default)]
|
||||||
|
pub enum FirmwareAttributeType {
|
||||||
|
#[default]
|
||||||
|
Immediate,
|
||||||
|
TUFKeyboard,
|
||||||
|
Ppt,
|
||||||
|
Gpu,
|
||||||
|
Bios,
|
||||||
|
}
|
||||||
|
|
||||||
macro_rules! define_attribute_getters {
|
macro_rules! define_attribute_getters {
|
||||||
($($attr:ident),*) => {
|
// Accept a list of attribute idents and an optional `types { .. }` block
|
||||||
|
( $( $attr:ident ),* $(,)? $( ; types { $( $tattr:ident : $ptype:ident ),* $(,)? } )? ) => {
|
||||||
impl FirmwareAttributes {
|
impl FirmwareAttributes {
|
||||||
$(
|
$(
|
||||||
pub fn $attr(&self) -> Option<&Attribute> {
|
pub fn $attr(&self) -> Option<&Attribute> {
|
||||||
@@ -268,6 +279,17 @@ macro_rules! define_attribute_getters {
|
|||||||
});
|
});
|
||||||
)*
|
)*
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl FirmwareAttribute {
|
||||||
|
pub fn property_type(&self) -> FirmwareAttributeType {
|
||||||
|
match <&str>::from(*self) {
|
||||||
|
$(
|
||||||
|
$( stringify!($tattr) => FirmwareAttributeType::$ptype, )*
|
||||||
|
)?
|
||||||
|
_ => FirmwareAttributeType::Immediate,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -286,6 +308,10 @@ define_attribute_getters!(
|
|||||||
dgpu_tgp,
|
dgpu_tgp,
|
||||||
charge_mode,
|
charge_mode,
|
||||||
boot_sound,
|
boot_sound,
|
||||||
|
kbd_leds_awake,
|
||||||
|
kbd_leds_sleep,
|
||||||
|
kbd_leds_boot,
|
||||||
|
kbd_leds_shutdown,
|
||||||
mcu_powersave,
|
mcu_powersave,
|
||||||
panel_od,
|
panel_od,
|
||||||
panel_hd_mode,
|
panel_hd_mode,
|
||||||
@@ -295,6 +321,38 @@ define_attribute_getters!(
|
|||||||
gpu_mux_mode,
|
gpu_mux_mode,
|
||||||
mini_led_mode,
|
mini_led_mode,
|
||||||
screen_auto_brightness
|
screen_auto_brightness
|
||||||
|
; types {
|
||||||
|
ppt_pl1_spl: Ppt,
|
||||||
|
ppt_pl2_sppt: Ppt,
|
||||||
|
ppt_apu_sppt: Ppt,
|
||||||
|
ppt_platform_sppt: Ppt,
|
||||||
|
ppt_fppt: Ppt,
|
||||||
|
nv_dynamic_boost: Ppt,
|
||||||
|
nv_temp_target: Ppt,
|
||||||
|
dgpu_base_tgp: Ppt,
|
||||||
|
dgpu_tgp: Ppt,
|
||||||
|
|
||||||
|
gpu_mux_mode: Gpu,
|
||||||
|
egpu_connected: Gpu,
|
||||||
|
egpu_enable: Gpu,
|
||||||
|
dgpu_disable: Gpu,
|
||||||
|
|
||||||
|
boot_sound: Bios,
|
||||||
|
|
||||||
|
mcu_powersave: Immediate,
|
||||||
|
|
||||||
|
screen_auto_brightness: Immediate,
|
||||||
|
mini_led_mode: Immediate,
|
||||||
|
panel_hd_mode: Immediate,
|
||||||
|
panel_od: Immediate,
|
||||||
|
|
||||||
|
kbd_leds_awake: TUFKeyboard,
|
||||||
|
kbd_leds_sleep: TUFKeyboard,
|
||||||
|
kbd_leds_boot: TUFKeyboard,
|
||||||
|
kbd_leds_shutdown: TUFKeyboard,
|
||||||
|
|
||||||
|
charge_mode: Immediate,
|
||||||
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
/// CamelCase names of the properties. Intended for use with DBUS
|
/// CamelCase names of the properties. Intended for use with DBUS
|
||||||
@@ -342,29 +400,10 @@ pub enum FirmwareAttribute {
|
|||||||
PptEnabled = 24,
|
PptEnabled = 24,
|
||||||
None = 25,
|
None = 25,
|
||||||
ScreenAutoBrightness = 26,
|
ScreenAutoBrightness = 26,
|
||||||
}
|
KbdLedsAwake = 27,
|
||||||
|
KbdLedsSleep = 28,
|
||||||
impl FirmwareAttribute {
|
KbdLedsBoot = 29,
|
||||||
pub fn is_ppt(&self) -> bool {
|
KbdLedsShutdown = 30,
|
||||||
matches!(
|
|
||||||
self,
|
|
||||||
FirmwareAttribute::PptPl1Spl
|
|
||||||
| FirmwareAttribute::PptPl2Sppt
|
|
||||||
| FirmwareAttribute::PptPl3Fppt
|
|
||||||
| FirmwareAttribute::PptFppt
|
|
||||||
| FirmwareAttribute::PptApuSppt
|
|
||||||
| FirmwareAttribute::PptPlatformSppt
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_dgpu(&self) -> bool {
|
|
||||||
matches!(
|
|
||||||
self,
|
|
||||||
FirmwareAttribute::NvDynamicBoost
|
|
||||||
| FirmwareAttribute::NvTempTarget
|
|
||||||
| FirmwareAttribute::DgpuTgp
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<&str> for FirmwareAttribute {
|
impl From<&str> for FirmwareAttribute {
|
||||||
@@ -386,6 +425,10 @@ impl From<&str> for FirmwareAttribute {
|
|||||||
"nv_tgp" => Self::DgpuTgp,
|
"nv_tgp" => Self::DgpuTgp,
|
||||||
"charge_mode" => Self::ChargeMode,
|
"charge_mode" => Self::ChargeMode,
|
||||||
"boot_sound" => Self::BootSound,
|
"boot_sound" => Self::BootSound,
|
||||||
|
"kbd_leds_awake" => Self::KbdLedsAwake,
|
||||||
|
"kbd_leds_sleep" => Self::KbdLedsSleep,
|
||||||
|
"kbd_leds_boot" => Self::KbdLedsBoot,
|
||||||
|
"kbd_leds_shutdown" => Self::KbdLedsShutdown,
|
||||||
"mcu_powersave" => Self::McuPowersave,
|
"mcu_powersave" => Self::McuPowersave,
|
||||||
"panel_overdrive" => Self::PanelOverdrive,
|
"panel_overdrive" => Self::PanelOverdrive,
|
||||||
"panel_hd_mode" => Self::PanelHdMode,
|
"panel_hd_mode" => Self::PanelHdMode,
|
||||||
@@ -431,6 +474,10 @@ impl From<FirmwareAttribute> for &str {
|
|||||||
FirmwareAttribute::DgpuDisable => "dgpu_disable",
|
FirmwareAttribute::DgpuDisable => "dgpu_disable",
|
||||||
FirmwareAttribute::GpuMuxMode => "gpu_mux_mode",
|
FirmwareAttribute::GpuMuxMode => "gpu_mux_mode",
|
||||||
FirmwareAttribute::MiniLedMode => "mini_led_mode",
|
FirmwareAttribute::MiniLedMode => "mini_led_mode",
|
||||||
|
FirmwareAttribute::KbdLedsAwake => "kbd_leds_awake",
|
||||||
|
FirmwareAttribute::KbdLedsSleep => "kbd_leds_sleep",
|
||||||
|
FirmwareAttribute::KbdLedsBoot => "kbd_leds_boot",
|
||||||
|
FirmwareAttribute::KbdLedsShutdown => "kbd_leds_shutdown",
|
||||||
FirmwareAttribute::PendingReboot => "pending_reboot",
|
FirmwareAttribute::PendingReboot => "pending_reboot",
|
||||||
FirmwareAttribute::ScreenAutoBrightness => "screen_auto_brightness",
|
FirmwareAttribute::ScreenAutoBrightness => "screen_auto_brightness",
|
||||||
FirmwareAttribute::None => "none",
|
FirmwareAttribute::None => "none",
|
||||||
|
|||||||
@@ -177,6 +177,12 @@ pub enum AuraMode {
|
|||||||
DoubleFade = 14,
|
DoubleFade = 14,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "dbus")]
|
||||||
|
impl zbus::zvariant::Basic for AuraMode {
|
||||||
|
const SIGNATURE_CHAR: char = 'u';
|
||||||
|
const SIGNATURE_STR: &'static str = "u";
|
||||||
|
}
|
||||||
|
|
||||||
impl AuraMode {
|
impl AuraMode {
|
||||||
pub fn list() -> [String; 15] {
|
pub fn list() -> [String; 15] {
|
||||||
[
|
[
|
||||||
|
|||||||
@@ -12,12 +12,13 @@ use crate::usb::{PROD_ID1, PROD_ID1_STR, PROD_ID2, PROD_ID2_STR};
|
|||||||
|
|
||||||
#[derive(Default, Debug, Copy, Clone, PartialEq, Eq, Deserialize, Serialize)]
|
#[derive(Default, Debug, Copy, Clone, PartialEq, Eq, Deserialize, Serialize)]
|
||||||
pub enum SlashType {
|
pub enum SlashType {
|
||||||
GA403,
|
GA403_2024,
|
||||||
GA403W,
|
GA403_2025,
|
||||||
GA605,
|
GA605_2024,
|
||||||
GU605,
|
GA605_2025,
|
||||||
GU605C,
|
GU605_2024,
|
||||||
G614F,
|
GU605_2025,
|
||||||
|
G614_2025,
|
||||||
#[default]
|
#[default]
|
||||||
Unsupported,
|
Unsupported,
|
||||||
}
|
}
|
||||||
@@ -25,24 +26,26 @@ pub enum SlashType {
|
|||||||
impl SlashType {
|
impl SlashType {
|
||||||
pub const fn prod_id(&self) -> u16 {
|
pub const fn prod_id(&self) -> u16 {
|
||||||
match self {
|
match self {
|
||||||
SlashType::GA403W => PROD_ID2,
|
SlashType::GA403_2025 => PROD_ID2,
|
||||||
SlashType::GA403 => PROD_ID1,
|
SlashType::GA403_2024 => PROD_ID1,
|
||||||
SlashType::GA605 => PROD_ID2,
|
SlashType::GA605_2025 => PROD_ID2,
|
||||||
SlashType::GU605 => PROD_ID1,
|
SlashType::GA605_2024 => PROD_ID2,
|
||||||
SlashType::GU605C => PROD_ID2,
|
SlashType::GU605_2025 => PROD_ID2,
|
||||||
SlashType::G614F => PROD_ID2,
|
SlashType::GU605_2024 => PROD_ID1,
|
||||||
|
SlashType::G614_2025 => PROD_ID2,
|
||||||
SlashType::Unsupported => 0,
|
SlashType::Unsupported => 0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const fn prod_id_str(&self) -> &str {
|
pub const fn prod_id_str(&self) -> &str {
|
||||||
match self {
|
match self {
|
||||||
SlashType::GA403W => PROD_ID2_STR,
|
SlashType::GA403_2025 => PROD_ID2_STR,
|
||||||
SlashType::GA403 => PROD_ID1_STR,
|
SlashType::GA403_2024 => PROD_ID1_STR,
|
||||||
SlashType::GA605 => PROD_ID2_STR,
|
SlashType::GA605_2025 => PROD_ID2_STR,
|
||||||
SlashType::GU605 => PROD_ID1_STR,
|
SlashType::GA605_2024 => PROD_ID2_STR,
|
||||||
SlashType::GU605C => PROD_ID2_STR,
|
SlashType::GU605_2025 => PROD_ID2_STR,
|
||||||
SlashType::G614F => PROD_ID2_STR,
|
SlashType::GU605_2024 => PROD_ID1_STR,
|
||||||
|
SlashType::G614_2025 => PROD_ID2_STR,
|
||||||
SlashType::Unsupported => "",
|
SlashType::Unsupported => "",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -50,17 +53,24 @@ impl SlashType {
|
|||||||
pub fn from_dmi() -> Self {
|
pub fn from_dmi() -> Self {
|
||||||
let board_name = DMIID::new().unwrap_or_default().board_name.to_uppercase();
|
let board_name = DMIID::new().unwrap_or_default().board_name.to_uppercase();
|
||||||
if board_name.contains("G614F") {
|
if board_name.contains("G614F") {
|
||||||
SlashType::G614F
|
SlashType::G614_2025
|
||||||
} else if board_name.contains("GA403W") {
|
} else if [
|
||||||
SlashType::GA403W
|
"GA403W", "GA403UH", "GA403UM", "GA403UP",
|
||||||
|
]
|
||||||
|
.iter()
|
||||||
|
.any(|s| board_name.contains(s))
|
||||||
|
{
|
||||||
|
SlashType::GA403_2025
|
||||||
} else if board_name.contains("GA403") {
|
} else if board_name.contains("GA403") {
|
||||||
SlashType::GA403
|
SlashType::GA403_2024
|
||||||
|
} else if board_name.contains("GA605K") {
|
||||||
|
SlashType::GA605_2025
|
||||||
} else if board_name.contains("GA605") {
|
} else if board_name.contains("GA605") {
|
||||||
SlashType::GA605
|
SlashType::GA605_2024
|
||||||
} else if board_name.contains("GU605C") {
|
} else if board_name.contains("GU605C") {
|
||||||
SlashType::GU605C
|
SlashType::GU605_2025
|
||||||
} else if board_name.contains("GU605") {
|
} else if board_name.contains("GU605") {
|
||||||
SlashType::GU605
|
SlashType::GU605_2024
|
||||||
} else {
|
} else {
|
||||||
SlashType::Unsupported
|
SlashType::Unsupported
|
||||||
}
|
}
|
||||||
@@ -72,12 +82,13 @@ impl FromStr for SlashType {
|
|||||||
|
|
||||||
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
|
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
|
||||||
Ok(match s.to_uppercase().as_str() {
|
Ok(match s.to_uppercase().as_str() {
|
||||||
"GA403W" => Self::GA403W,
|
"GA403_2025" => Self::GA403_2025,
|
||||||
"GA403" => Self::GA403,
|
"GA403_2024" => Self::GA403_2024,
|
||||||
"GA605" => Self::GA605,
|
"GA605_2025" => Self::GA605_2025,
|
||||||
"GU605C" => Self::GU605C,
|
"GA605_2024" => Self::GA605_2024,
|
||||||
"GU605" => Self::GU605,
|
"GU605_2025" => Self::GU605_2025,
|
||||||
"G614FR" => Self::G614F,
|
"GU605_2024" => Self::GU605_2024,
|
||||||
|
"G614_2025" => Self::G614_2025,
|
||||||
_ => Self::Unsupported,
|
_ => Self::Unsupported,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -86,6 +97,7 @@ impl FromStr for SlashType {
|
|||||||
#[cfg_attr(feature = "dbus", derive(Type, Value, OwnedValue))]
|
#[cfg_attr(feature = "dbus", derive(Type, Value, OwnedValue))]
|
||||||
#[derive(Debug, Default, PartialEq, Eq, Copy, Clone, Deserialize, Serialize)]
|
#[derive(Debug, Default, PartialEq, Eq, Copy, Clone, Deserialize, Serialize)]
|
||||||
pub enum SlashMode {
|
pub enum SlashMode {
|
||||||
|
Static = 0x06,
|
||||||
Bounce = 0x10,
|
Bounce = 0x10,
|
||||||
Slash = 0x12,
|
Slash = 0x12,
|
||||||
Loading = 0x13,
|
Loading = 0x13,
|
||||||
@@ -109,6 +121,7 @@ impl FromStr for SlashMode {
|
|||||||
|
|
||||||
fn from_str(s: &str) -> Result<Self, SlashError> {
|
fn from_str(s: &str) -> Result<Self, SlashError> {
|
||||||
match s {
|
match s {
|
||||||
|
"Static" => Ok(SlashMode::Static),
|
||||||
"Bounce" => Ok(SlashMode::Bounce),
|
"Bounce" => Ok(SlashMode::Bounce),
|
||||||
"Slash" => Ok(SlashMode::Slash),
|
"Slash" => Ok(SlashMode::Slash),
|
||||||
"Loading" => Ok(SlashMode::Loading),
|
"Loading" => Ok(SlashMode::Loading),
|
||||||
@@ -132,6 +145,7 @@ impl FromStr for SlashMode {
|
|||||||
impl Display for SlashMode {
|
impl Display for SlashMode {
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
let str = match &self {
|
let str = match &self {
|
||||||
|
SlashMode::Static => String::from("Static"),
|
||||||
SlashMode::Bounce => String::from("Bounce"),
|
SlashMode::Bounce => String::from("Bounce"),
|
||||||
SlashMode::Slash => String::from("Slash"),
|
SlashMode::Slash => String::from("Slash"),
|
||||||
SlashMode::Loading => String::from("Loading"),
|
SlashMode::Loading => String::from("Loading"),
|
||||||
@@ -153,8 +167,9 @@ impl Display for SlashMode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl SlashMode {
|
impl SlashMode {
|
||||||
pub fn list() -> [String; 15] {
|
pub fn list() -> [String; 16] {
|
||||||
[
|
[
|
||||||
|
SlashMode::Static.to_string(),
|
||||||
SlashMode::Bounce.to_string(),
|
SlashMode::Bounce.to_string(),
|
||||||
SlashMode::Slash.to_string(),
|
SlashMode::Slash.to_string(),
|
||||||
SlashMode::Loading.to_string(),
|
SlashMode::Loading.to_string(),
|
||||||
|
|||||||
@@ -39,17 +39,24 @@ pub fn get_slash_type() -> SlashType {
|
|||||||
.unwrap_or_default();
|
.unwrap_or_default();
|
||||||
let board_name = dmi.board_name.to_uppercase();
|
let board_name = dmi.board_name.to_uppercase();
|
||||||
if board_name.contains("G614F") {
|
if board_name.contains("G614F") {
|
||||||
SlashType::G614F
|
SlashType::G614_2025
|
||||||
} else if board_name.contains("GA403W") {
|
} else if [
|
||||||
SlashType::GA403W
|
"GA403W", "GA403UH", "GA403UM", "GA403UP",
|
||||||
|
]
|
||||||
|
.iter()
|
||||||
|
.any(|s| board_name.contains(s))
|
||||||
|
{
|
||||||
|
SlashType::GA403_2025
|
||||||
} else if board_name.contains("GA403") {
|
} else if board_name.contains("GA403") {
|
||||||
SlashType::GA403
|
SlashType::GA403_2024
|
||||||
|
} else if board_name.contains("GA605K") {
|
||||||
|
SlashType::GA605_2025
|
||||||
} else if board_name.contains("GA605") {
|
} else if board_name.contains("GA605") {
|
||||||
SlashType::GA605
|
SlashType::GA605_2024
|
||||||
} else if board_name.contains("GU605C") {
|
} else if board_name.contains("GU605C") {
|
||||||
SlashType::GU605C
|
SlashType::GU605_2025
|
||||||
} else if board_name.contains("GU605") {
|
} else if board_name.contains("GU605") {
|
||||||
SlashType::GU605
|
SlashType::GU605_2024
|
||||||
} else {
|
} else {
|
||||||
SlashType::Unsupported
|
SlashType::Unsupported
|
||||||
}
|
}
|
||||||
@@ -57,12 +64,13 @@ pub fn get_slash_type() -> SlashType {
|
|||||||
|
|
||||||
pub const fn report_id(slash_type: SlashType) -> u8 {
|
pub const fn report_id(slash_type: SlashType) -> u8 {
|
||||||
match slash_type {
|
match slash_type {
|
||||||
SlashType::GA403W => REPORT_ID_19B6,
|
SlashType::GA403_2025 => REPORT_ID_19B6,
|
||||||
SlashType::GA403 => REPORT_ID_193B,
|
SlashType::GA403_2024 => REPORT_ID_193B,
|
||||||
SlashType::GA605 => REPORT_ID_19B6,
|
SlashType::GA605_2025 => REPORT_ID_19B6,
|
||||||
SlashType::G614F => REPORT_ID_19B6,
|
SlashType::GA605_2024 => REPORT_ID_19B6,
|
||||||
SlashType::GU605 => REPORT_ID_193B,
|
SlashType::GU605_2025 => REPORT_ID_19B6,
|
||||||
SlashType::GU605C => REPORT_ID_19B6,
|
SlashType::GU605_2024 => REPORT_ID_193B,
|
||||||
|
SlashType::G614_2025 => REPORT_ID_19B6,
|
||||||
SlashType::Unsupported => REPORT_ID_19B6,
|
SlashType::Unsupported => REPORT_ID_19B6,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
1
rust-toolchain
Normal file
1
rust-toolchain
Normal file
@@ -0,0 +1 @@
|
|||||||
|
stable
|
||||||
70
simulators/src/animatrix/map_g635l.rs
Normal file
70
simulators/src/animatrix/map_g635l.rs
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
use super::Row;
|
||||||
|
|
||||||
|
// TODO: This is a placeholder for G635L map
|
||||||
|
pub const G635L: [Row; 63] = [
|
||||||
|
Row(0x01, 7, 32, 0),
|
||||||
|
Row(0x01, 7 + 34, 32, 0),
|
||||||
|
Row(0x01, 7 + 68, 32, 0),
|
||||||
|
Row(0x01, 7 + 102, 32, 0), // 34 len
|
||||||
|
Row(0x01, 7 + 136, 32, 0),
|
||||||
|
Row(0x01, 7 + 170, 34, 0),
|
||||||
|
Row(0x01, 7 + 204, 34, 0),
|
||||||
|
Row(0x01, 7 + 238, 34, 0),
|
||||||
|
Row(0x01, 7 + 272, 34, 0),
|
||||||
|
Row(0x01, 7 + 306, 34, 0),
|
||||||
|
Row(0x01, 7 + 340, 34, 0),
|
||||||
|
Row(0x01, 7 + 374, 34, 0),
|
||||||
|
Row(0x01, 7 + 408, 33, 1),
|
||||||
|
Row(0x01, 7 + 441, 33, 1),
|
||||||
|
Row(0x01, 7 + 474, 32, 2),
|
||||||
|
Row(0x01, 7 + 506, 32, 2),
|
||||||
|
Row(0x01, 7 + 538, 31, 3),
|
||||||
|
Row(0x01, 7 + 569, 31, 3),
|
||||||
|
Row(0x01, 7 + 600, 28, 4),
|
||||||
|
//
|
||||||
|
Row(0x74, 7 + 1, 3, 28 + 4), // adds to end of previous
|
||||||
|
Row(0x74, 7 + 3, 30, 4),
|
||||||
|
Row(0x74, 7 + 33, 29, 5),
|
||||||
|
Row(0x74, 7 + 62, 29, 5),
|
||||||
|
Row(0x74, 7 + 91, 28, 6),
|
||||||
|
Row(0x74, 7 + 119, 28, 6),
|
||||||
|
Row(0x74, 7 + 147, 27, 7),
|
||||||
|
Row(0x74, 7 + 174, 27, 7),
|
||||||
|
Row(0x74, 7 + 202, 26, 9),
|
||||||
|
Row(0x74, 7 + 228, 26, 9),
|
||||||
|
Row(0x74, 7 + 254, 25, 10),
|
||||||
|
Row(0x74, 7 + 278, 25, 9), // WEIRD OFFSET
|
||||||
|
Row(0x74, 7 + 303, 24, 10),
|
||||||
|
Row(0x74, 7 + 327, 24, 10),
|
||||||
|
Row(0x74, 7 + 351, 23, 11),
|
||||||
|
Row(0x74, 7 + 374, 23, 11),
|
||||||
|
Row(0x74, 7 + 397, 22, 12),
|
||||||
|
Row(0x74, 7 + 419, 22, 12),
|
||||||
|
Row(0x74, 7 + 441, 21, 13),
|
||||||
|
Row(0x74, 7 + 462, 21, 13),
|
||||||
|
Row(0x74, 7 + 483, 20, 14),
|
||||||
|
Row(0x74, 7 + 503, 20, 14),
|
||||||
|
Row(0x74, 7 + 523, 19, 15),
|
||||||
|
Row(0x74, 7 + 542, 19, 15),
|
||||||
|
Row(0x74, 7 + 561, 18, 16),
|
||||||
|
Row(0x74, 7 + 579, 18, 16),
|
||||||
|
Row(0x74, 7 + 597, 17, 17),
|
||||||
|
Row(0x74, 7 + 614, 13, 17),
|
||||||
|
//
|
||||||
|
Row(0xe7, 7 + 1, 4, 13 + 18), // adds to end of previous
|
||||||
|
Row(0xe7, 7 + 4, 16, 18),
|
||||||
|
Row(0xe7, 7 + 20, 16, 18),
|
||||||
|
Row(0xe7, 7 + 36, 15, 19),
|
||||||
|
Row(0xe7, 7 + 51, 15, 19),
|
||||||
|
Row(0xe7, 7 + 66, 14, 20),
|
||||||
|
Row(0xe7, 7 + 80, 12, 20), // too long? 14
|
||||||
|
Row(0xe7, 7 + 94, 13, 21),
|
||||||
|
Row(0xe7, 7 + 107, 13, 21),
|
||||||
|
Row(0xe7, 7 + 120, 12, 12), // Actual display end
|
||||||
|
Row(0xe7, 7 + 132, 12, 22),
|
||||||
|
Row(0xe7, 7 + 144, 11, 23),
|
||||||
|
Row(0xe7, 7 + 155, 11, 23),
|
||||||
|
Row(0xe7, 7 + 166, 10, 24),
|
||||||
|
Row(0xe7, 7 + 176, 10, 24),
|
||||||
|
Row(0xe7, 7 + 186, 9, 25),
|
||||||
|
];
|
||||||
74
simulators/src/animatrix/map_g835l.rs
Normal file
74
simulators/src/animatrix/map_g835l.rs
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
use super::Row;
|
||||||
|
|
||||||
|
// G835L layout: 68 rows (triangle + rectangle), 2 packets
|
||||||
|
pub const G835L: [Row; 69] = [
|
||||||
|
Row(0x01, 7, 0, 0),
|
||||||
|
Row(0x01, 8, 0, 0),
|
||||||
|
Row(0x01, 9, 1, 0),
|
||||||
|
Row(0x01, 11, 1, 0),
|
||||||
|
Row(0x01, 13, 2, 0),
|
||||||
|
Row(0x01, 16, 2, 0),
|
||||||
|
Row(0x01, 19, 3, 0),
|
||||||
|
Row(0x01, 23, 3, 0),
|
||||||
|
Row(0x01, 27, 4, 0),
|
||||||
|
Row(0x01, 32, 4, 0),
|
||||||
|
Row(0x01, 37, 5, 0),
|
||||||
|
Row(0x01, 43, 5, 0),
|
||||||
|
Row(0x01, 49, 6, 0),
|
||||||
|
Row(0x01, 56, 6, 0),
|
||||||
|
Row(0x01, 63, 7, 0),
|
||||||
|
Row(0x01, 71, 7, 0),
|
||||||
|
Row(0x01, 79, 8, 0),
|
||||||
|
Row(0x01, 88, 8, 0),
|
||||||
|
Row(0x01, 97, 9, 0),
|
||||||
|
Row(0x01, 107, 9, 0),
|
||||||
|
Row(0x01, 117, 10, 0),
|
||||||
|
Row(0x01, 128, 10, 0),
|
||||||
|
Row(0x01, 139, 11, 0),
|
||||||
|
Row(0x01, 151, 11, 0),
|
||||||
|
Row(0x01, 163, 12, 0),
|
||||||
|
Row(0x01, 176, 12, 0),
|
||||||
|
Row(0x01, 189, 13, 0),
|
||||||
|
Row(0x01, 203, 13, 0),
|
||||||
|
Row(0x01, 217, 14, 0),
|
||||||
|
Row(0x01, 232, 14, 0),
|
||||||
|
Row(0x01, 247, 14, 1),
|
||||||
|
Row(0x01, 262, 14, 1),
|
||||||
|
Row(0x01, 277, 14, 2),
|
||||||
|
Row(0x01, 292, 14, 2),
|
||||||
|
Row(0x01, 307, 14, 3),
|
||||||
|
Row(0x01, 322, 14, 3),
|
||||||
|
Row(0x01, 337, 14, 4),
|
||||||
|
Row(0x01, 352, 14, 4),
|
||||||
|
Row(0x01, 367, 14, 5),
|
||||||
|
Row(0x01, 382, 14, 5),
|
||||||
|
Row(0x01, 397, 14, 6),
|
||||||
|
Row(0x01, 412, 14, 6),
|
||||||
|
Row(0x01, 427, 14, 7),
|
||||||
|
Row(0x01, 442, 14, 7),
|
||||||
|
Row(0x01, 457, 14, 8),
|
||||||
|
Row(0x01, 472, 14, 8),
|
||||||
|
Row(0x01, 487, 14, 9),
|
||||||
|
Row(0x01, 502, 14, 9),
|
||||||
|
Row(0x01, 517, 14, 10),
|
||||||
|
Row(0x01, 532, 14, 10),
|
||||||
|
Row(0x01, 547, 14, 11),
|
||||||
|
Row(0x01, 562, 14, 11),
|
||||||
|
Row(0x01, 577, 14, 12),
|
||||||
|
Row(0x01, 592, 14, 12),
|
||||||
|
Row(0x01, 607, 14, 13),
|
||||||
|
Row(0x01, 622, 11, 13),
|
||||||
|
Row(0x74, 7, 2, 25),
|
||||||
|
Row(0x74, 10, 14, 14),
|
||||||
|
Row(0x74, 25, 14, 14),
|
||||||
|
Row(0x74, 40, 14, 15),
|
||||||
|
Row(0x74, 55, 14, 15),
|
||||||
|
Row(0x74, 70, 14, 16),
|
||||||
|
Row(0x74, 85, 14, 16),
|
||||||
|
Row(0x74, 100, 14, 17),
|
||||||
|
Row(0x74, 115, 14, 17),
|
||||||
|
Row(0x74, 130, 14, 18),
|
||||||
|
Row(0x74, 145, 14, 18),
|
||||||
|
Row(0x74, 160, 14, 19),
|
||||||
|
Row(0x74, 175, 14, 19),
|
||||||
|
];
|
||||||
@@ -1,9 +1,13 @@
|
|||||||
use rog_anime::AnimeType;
|
use rog_anime::AnimeType;
|
||||||
|
|
||||||
|
use self::map_g635l::G635L;
|
||||||
|
use self::map_g835l::G835L;
|
||||||
use self::map_ga401::GA401;
|
use self::map_ga401::GA401;
|
||||||
use self::map_ga402::GA402;
|
use self::map_ga402::GA402;
|
||||||
use self::map_gu604::GU604;
|
use self::map_gu604::GU604;
|
||||||
|
|
||||||
|
mod map_g635l;
|
||||||
|
mod map_g835l;
|
||||||
mod map_ga401;
|
mod map_ga401;
|
||||||
mod map_ga402;
|
mod map_ga402;
|
||||||
mod map_gu604;
|
mod map_gu604;
|
||||||
@@ -39,11 +43,13 @@ pub struct AniMatrix {
|
|||||||
impl AniMatrix {
|
impl AniMatrix {
|
||||||
pub fn new(model: AnimeType) -> Self {
|
pub fn new(model: AnimeType) -> Self {
|
||||||
let led_shape = match model {
|
let led_shape = match model {
|
||||||
|
// TODO: Verify how this reacts on G635L and G835L
|
||||||
|
// These are all doing the same thing. Can be simplified
|
||||||
AnimeType::GA401 => LedShape {
|
AnimeType::GA401 => LedShape {
|
||||||
vertical: 2,
|
vertical: 2,
|
||||||
horizontal: 5,
|
horizontal: 5,
|
||||||
},
|
},
|
||||||
AnimeType::GA402 | AnimeType::G635L | AnimeType::G835LW | AnimeType::Unsupported => {
|
AnimeType::GA402 | AnimeType::G635L | AnimeType::G835L | AnimeType::Unsupported => {
|
||||||
LedShape {
|
LedShape {
|
||||||
vertical: 2,
|
vertical: 2,
|
||||||
horizontal: 5,
|
horizontal: 5,
|
||||||
@@ -58,10 +64,10 @@ impl AniMatrix {
|
|||||||
// Do a hard mapping of each (derived from wireshardk captures)
|
// Do a hard mapping of each (derived from wireshardk captures)
|
||||||
let rows = match model {
|
let rows = match model {
|
||||||
AnimeType::GA401 => GA401.to_vec(),
|
AnimeType::GA401 => GA401.to_vec(),
|
||||||
AnimeType::GA402 | AnimeType::G635L | AnimeType::G835LW | AnimeType::Unsupported => {
|
AnimeType::GA402 | AnimeType::Unsupported => GA402.to_vec(),
|
||||||
GA402.to_vec()
|
|
||||||
}
|
|
||||||
AnimeType::GU604 => GU604.to_vec(),
|
AnimeType::GU604 => GU604.to_vec(),
|
||||||
|
AnimeType::G635L => G635L.to_vec(),
|
||||||
|
AnimeType::G835L => G835L.to_vec(),
|
||||||
};
|
};
|
||||||
|
|
||||||
Self { rows, led_shape }
|
Self { rows, led_shape }
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ use std::env;
|
|||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use log::error;
|
|
||||||
use rog_anime::usb::{PROD_ID, VENDOR_ID};
|
use rog_anime::usb::{PROD_ID, VENDOR_ID};
|
||||||
use rog_anime::{AnimeType, USB_PREFIX2};
|
use rog_anime::{AnimeType, USB_PREFIX2};
|
||||||
use sdl2::event::Event;
|
use sdl2::event::Event;
|
||||||
@@ -20,6 +19,7 @@ pub struct VirtAnimeMatrix {
|
|||||||
animatrix: AniMatrix,
|
animatrix: AniMatrix,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: This isn't working
|
||||||
impl VirtAnimeMatrix {
|
impl VirtAnimeMatrix {
|
||||||
pub fn new(model: AnimeType) -> Self {
|
pub fn new(model: AnimeType) -> Self {
|
||||||
VirtAnimeMatrix {
|
VirtAnimeMatrix {
|
||||||
@@ -86,8 +86,12 @@ impl VirtAnimeMatrix {
|
|||||||
]
|
]
|
||||||
.to_vec(),
|
.to_vec(),
|
||||||
})
|
})
|
||||||
.map_err(|err| error!("Could not create virtual device: {:?}", err))
|
.unwrap_or_else(|err| {
|
||||||
.expect("Could not create virtual device"),
|
panic!(
|
||||||
|
"Could not create virtual device: {err:?}. \
|
||||||
|
Try loading the uhid module and ensure you have the necessary permissions."
|
||||||
|
)
|
||||||
|
}),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -114,7 +118,7 @@ impl VirtAnimeMatrix {
|
|||||||
fn main() -> Result<(), Box<dyn Error>> {
|
fn main() -> Result<(), Box<dyn Error>> {
|
||||||
let args: Vec<String> = env::args().collect();
|
let args: Vec<String> = env::args().collect();
|
||||||
if args.len() <= 1 {
|
if args.len() <= 1 {
|
||||||
println!("Must supply arg, one of <GA401, GA402, GU604>");
|
println!("Must supply arg, one of <GA401, GA402, GU604, G835L>");
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
let anime_type = AnimeType::from_str(&args[1])?;
|
let anime_type = AnimeType::from_str(&args[1])?;
|
||||||
|
|||||||
Reference in New Issue
Block a user