From 034e9017be70c5f7b45a51d1fcca86bed5f86993 Mon Sep 17 00:00:00 2001 From: yyc12345 Date: Mon, 13 Oct 2025 15:50:48 +0800 Subject: [PATCH] write shit --- wfassoc/src/lib.rs | 12 +++++- wfassoc/src/winregex.rs | 96 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 107 insertions(+), 1 deletion(-) create mode 100644 wfassoc/src/winregex.rs diff --git a/wfassoc/src/lib.rs b/wfassoc/src/lib.rs index c9a509f..ab44320 100644 --- a/wfassoc/src/lib.rs +++ b/wfassoc/src/lib.rs @@ -4,6 +4,9 @@ #[cfg(not(target_os = "windows"))] compile_error!("Crate wfassoc is only supported on Windows."); +/// The expand of winreg crate according to our module requirements. +mod winregex; + use regex::Regex; use std::fmt::Display; use std::str::FromStr; @@ -249,7 +252,14 @@ impl FileExt { } pub fn iter_open_with(&self, view: View) -> Result, Error> { - todo!() + let viewer = match self.open_view(view)? { + Some(viewer) => viewer, + None => return Ok(std::iter::empty::()), + }; + let it = winregex::iter_sz_keys(&viewer); + let it = winregex::exclude_default_key(it); + + Ok(it.map(|s| ProgId::from(s.as_str()))) } pub fn insert_open_with(&mut self, scope: Scope, prog_id: &ProgId) -> Result<(), Error> { diff --git a/wfassoc/src/winregex.rs b/wfassoc/src/winregex.rs new file mode 100644 index 0000000..02c2faf --- /dev/null +++ b/wfassoc/src/winregex.rs @@ -0,0 +1,96 @@ +use winreg::RegKey; +use winreg::EnumKeys; +use winreg::enums::RegType; +// use thiserror::Error as TeError; + +// #[derive(Debug, TeError)] +// pub enum WinRegExError { +// #[error("{0}")] +// Io(#[from] std::io::Error) +// } + +// pub type WinRegExResult = Result; + +// region Iterate Keys without Error + +pub struct IterKeys<'a> { + iter_keys: EnumKeys<'a>, +} + +impl<'a> IterKeys<'a> { + fn new(regkey: &'a RegKey) -> Self { + Self { iter_keys: regkey.enum_keys() } + } +} + +impl<'a> Iterator for IterKeys<'a> { + type Item = String; + + fn next(&mut self) -> Option { + loop { + match self.iter_keys.next() { + Some(key) => match key { + Ok(key) => return Some(key), + Err(_) => continue, + }, + None => return None, + } + } + } +} + +pub fn iter_keys<'a>(regkey: &'a RegKey) -> IterKeys<'a> { + IterKeys::new(regkey) +} + +// endregion + +// region Iterate REG_SZ Keys + +pub struct IterSzKeys<'a> { + regkey: &'a RegKey, + iter_keys: IterKeys<'a>, +} + +impl<'a> IterSzKeys<'a> { + fn new(regkey: &'a RegKey) -> Self { + Self { regkey, iter_keys: IterKeys::new(regkey) } + } +} + +impl<'a> Iterator for IterSzKeys<'a> { + type Item = String; + + fn next(&mut self) -> Option { + loop { + if let Some(key) = self.iter_keys.next() { + // Check whether given key is REG_SZ. + match self.regkey.get_raw_value(&key) { + Ok(raw_value) => { + if matches!(raw_value.vtype, RegType::REG_SZ) { + return Some(key); + } else { + continue; + } + }, + Err(_) => continue, + } + } else { + return None; + } + } + } +} + +/// Untitled +/// +/// Given RegKey must has KEY_READ permission, otherwise the result may be inaccurate. +pub fn iter_sz_keys<'a>(regkey: &'a RegKey) -> IterSzKeys<'a> { + IterSzKeys::new(regkey) +} + +// endregion + +pub fn exclude_default_key(it: impl Iterator) -> impl Iterator { + it.filter(|x| !x.is_empty()) +}