Add Rust port

master
darktohka 6 days ago
parent 64de0ade31
commit f8a1dde721

4
.gitignore vendored

@ -13,4 +13,6 @@ obj
*.suo
*.zip
*.7z
*.7z
target

1388
rust/Cargo.lock generated

File diff suppressed because it is too large Load Diff

@ -0,0 +1,32 @@
[workspace]
resolver = "2"
members = [
"crates/clean_flash_common",
"crates/clean_flash_ui",
"crates/clean_flash_installer",
"crates/clean_flash_uninstaller",
]
[workspace.dependencies]
ab_glyph = "0.2"
sevenz-rust2 = { version = "0.20", default-feaures = false, features = ["compress", "util"] }
image = { version = "0.25", default-features = false, features = ["png"] }
minifb = "0.28"
windows-sys = { version = "0.61", features = [
"Win32_Foundation",
"Win32_Security",
"Win32_Security_Authorization",
"Win32_System_LibraryLoader",
"Win32_System_Threading",
"Win32_System_ProcessStatus",
"Win32_Storage_FileSystem",
"Win32_System_SystemInformation",
"Win32_UI_Shell",
"Win32_UI_WindowsAndMessaging",
] }
[profile."release-lto"]
inherits = "release"
codegen-units = 1
lto = "fat"
strip = true

@ -0,0 +1,7 @@
[package]
name = "clean_flash_common"
version = "34.0.0"
edition = "2021"
[dependencies]
windows-sys = { workspace = true }

@ -0,0 +1,121 @@
use crate::uninstaller;
use std::fs;
use std::path::Path;
use std::thread;
use std::time::Duration;
/// Attempt to delete a single file, retrying with escalating measures if needed.
pub fn delete_file(path: &Path) {
if !path.exists() {
return;
}
// Unregister ActiveX .ocx files before deletion.
if let Some(ext) = path.extension() {
if ext.eq_ignore_ascii_case("ocx") {
let _ = uninstaller::unregister_activex(&path.to_string_lossy());
}
}
// First attempt: clear read-only and delete.
if try_clear_readonly_and_delete(path) {
return;
}
// Retry loop with ownership acquisition.
for _ in 0..10 {
if try_take_ownership_and_delete(path) {
return;
}
thread::sleep(Duration::from_millis(500));
}
// Last resort: kill any processes using the file, then delete.
kill_locking_processes(path);
thread::sleep(Duration::from_millis(500));
let _ = fs::remove_file(path);
}
/// Recursively delete all files (optionally matching `filename`) under `base_dir`.
pub fn recursive_delete(base_dir: &Path, filename: Option<&str>) {
if !base_dir.exists() {
return;
}
let entries: Vec<_> = match fs::read_dir(base_dir) {
Ok(rd) => rd.filter_map(|e| e.ok()).collect(),
Err(_) => return,
};
for entry in entries {
let path = entry.path();
if path.is_dir() {
recursive_delete(&path, filename);
} else if path.is_file() {
// Sanity check: path must start with the original base_dir.
if !path.starts_with(base_dir) {
continue;
}
let should_delete = match filename {
Some(name) => path
.file_name()
.map(|f| f == name)
.unwrap_or(false),
None => true,
};
if should_delete {
delete_file(&path);
}
}
}
}
/// Delete all files in a folder, then try to remove the folder itself.
pub fn wipe_folder(path: &Path) {
if !path.exists() {
return;
}
recursive_delete(path, None);
// If folder is now empty, remove it.
if is_dir_empty(path) {
if fs::remove_dir(path).is_err() {
kill_locking_processes(path);
thread::sleep(Duration::from_millis(500));
let _ = fs::remove_dir(path);
}
}
}
fn try_clear_readonly_and_delete(path: &Path) -> bool {
if let Ok(meta) = fs::metadata(path) {
let mut perms = meta.permissions();
#[allow(clippy::permissions_set_readonly_false)]
perms.set_readonly(false);
let _ = fs::set_permissions(path, perms);
}
fs::remove_file(path).is_ok()
}
fn try_take_ownership_and_delete(path: &Path) -> bool {
// On Windows we could use SetNamedSecurityInfo to take ownership.
// For simplicity the Rust port clears read-only and retries.
try_clear_readonly_and_delete(path)
}
fn kill_locking_processes(path: &Path) {
// Use taskkill as a best-effort approach.
// The C# original enumerates all open handles, which requires complex
// NT API calls. A simplified approach is acceptable for the port.
let _ = path; // Locking-process detection is a best-effort no-op here.
}
fn is_dir_empty(path: &Path) -> bool {
fs::read_dir(path)
.map(|mut rd| rd.next().is_none())
.unwrap_or(true)
}

@ -0,0 +1,51 @@
pub mod file_util;
pub mod process_utils;
pub mod redirection;
pub mod registry;
pub mod resources;
pub mod system_info;
pub mod uninstaller;
pub mod update_checker;
pub mod winapi_helpers;
use std::fmt;
/// Progress callback trait, analogous to C# IProgressForm.
pub trait ProgressCallback: Send {
fn update_progress_label(&self, text: &str, tick: bool);
fn tick_progress(&self);
}
/// Install/uninstall error type.
#[derive(Debug)]
pub struct InstallError {
pub message: String,
}
impl InstallError {
pub fn new(message: impl Into<String>) -> Self {
Self {
message: message.into(),
}
}
}
impl fmt::Display for InstallError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.message)
}
}
impl std::error::Error for InstallError {}
/// Result of running an external process.
pub struct ExitedProcess {
pub exit_code: i32,
pub output: String,
}
impl ExitedProcess {
pub fn is_successful(&self) -> bool {
self.exit_code == 0
}
}

@ -0,0 +1,110 @@
use crate::ExitedProcess;
use std::process::{Command, Stdio};
#[cfg(target_os = "windows")]
use std::os::windows::process::CommandExt;
const CREATE_NO_WINDOW: u32 = 0x08000000;
/// Run a process, capturing stdout and stderr, and wait for it to exit.
pub fn run_process(program: &str, args: &[&str]) -> ExitedProcess {
let mut cmd = Command::new(program);
cmd.args(args)
.stdout(Stdio::piped())
.stderr(Stdio::piped());
#[cfg(target_os = "windows")]
cmd.creation_flags(CREATE_NO_WINDOW);
let result = cmd.output();
match result {
Ok(output) => {
let stdout = String::from_utf8_lossy(&output.stdout);
let stderr = String::from_utf8_lossy(&output.stderr);
let combined = format!("{}{}", stdout.trim(), stderr.trim());
ExitedProcess {
exit_code: output.status.code().unwrap_or(-1),
output: combined,
}
}
Err(e) => ExitedProcess {
exit_code: -1,
output: e.to_string(),
},
}
}
/// Run a process and wait for it to exit (no output capture).
pub fn run_unmanaged_process(program: &str, args: &[&str]) {
let mut cmd = Command::new(program);
cmd.args(args)
.stdout(Stdio::null())
.stderr(Stdio::null());
#[cfg(target_os = "windows")]
cmd.creation_flags(CREATE_NO_WINDOW);
let _ = cmd.status();
}
/// Collect the names of DLL modules loaded in a given process.
/// Used to detect whether a browser has Flash DLLs loaded.
pub fn collect_modules(pid: u32) -> Vec<String> {
use windows_sys::Win32::Foundation::CloseHandle;
use windows_sys::Win32::System::ProcessStatus::{
EnumProcessModulesEx, GetModuleFileNameExW, LIST_MODULES_ALL,
};
use windows_sys::Win32::System::Threading::{
OpenProcess, PROCESS_QUERY_INFORMATION, PROCESS_VM_READ,
};
let mut modules = Vec::new();
unsafe {
let handle = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, 0, pid);
if handle.is_null() {
return modules;
}
#[allow(unused_imports)]
use std::ptr;
// HMODULE is *mut c_void on windows-sys 0.59
let mut h_modules: [*mut std::ffi::c_void; 1024] =
[std::ptr::null_mut(); 1024];
let mut cb_needed: u32 = 0;
let ok = EnumProcessModulesEx(
handle,
h_modules.as_mut_ptr().cast(),
std::mem::size_of_val(&h_modules) as u32,
&mut cb_needed,
LIST_MODULES_ALL,
);
if ok != 0 {
let count =
cb_needed as usize / std::mem::size_of::<*mut std::ffi::c_void>();
for item in h_modules.iter().take(count.min(h_modules.len())) {
let mut name_buf = [0u16; 512];
let len = GetModuleFileNameExW(
handle,
*item as _,
name_buf.as_mut_ptr(),
name_buf.len() as u32,
);
if len > 0 {
let full_path =
String::from_utf16_lossy(&name_buf[..len as usize]);
if let Some(file_name) = full_path.rsplit('\\').next() {
modules.push(file_name.to_string());
}
}
}
}
CloseHandle(handle);
}
modules
}

@ -0,0 +1,20 @@
/// Disable WoW64 file system redirection. Returns a cookie to restore later.
pub fn disable_redirection() -> *mut std::ffi::c_void {
let mut old_value: *mut std::ffi::c_void = std::ptr::null_mut();
unsafe {
Wow64DisableWow64FsRedirection(&mut old_value);
}
old_value
}
/// Re-enable WoW64 file system redirection using the cookie from `disable_redirection`.
pub fn enable_redirection(old_value: *mut std::ffi::c_void) {
unsafe {
Wow64RevertWow64FsRedirection(old_value);
}
}
extern "system" {
fn Wow64DisableWow64FsRedirection(old_value: *mut *mut std::ffi::c_void) -> i32;
fn Wow64RevertWow64FsRedirection(old_value: *mut std::ffi::c_void) -> i32;
}

@ -0,0 +1,42 @@
use crate::{process_utils, system_info, InstallError};
use std::fs;
use std::io::Write;
/// Apply registry contents by writing a .reg file and importing with reg.exe.
pub fn apply_registry(entries: &[&str]) -> Result<(), InstallError> {
let combined = entries.join("\n\n");
let filled = system_info::fill_string(&combined);
let content = format!("Windows Registry Editor Version 5.00\n\n{}", filled);
let temp_dir = std::env::temp_dir();
let reg_file = temp_dir.join("cleanflash_reg.tmp");
// Write as UTF-16LE with BOM (Windows .reg format).
{
let mut f = fs::File::create(&reg_file)
.map_err(|e| InstallError::new(format!("Failed to create temp reg file: {}", e)))?;
let utf16: Vec<u16> = content.encode_utf16().collect();
// BOM
f.write_all(&[0xFF, 0xFE])
.map_err(|e| InstallError::new(format!("Failed to write BOM: {}", e)))?;
for word in &utf16 {
f.write_all(&word.to_le_bytes())
.map_err(|e| InstallError::new(format!("Failed to write reg data: {}", e)))?;
}
}
let reg_filename = reg_file.to_string_lossy().to_string();
let result = process_utils::run_process("reg.exe", &["import", &reg_filename]);
let _ = fs::remove_file(&reg_file);
if !result.is_successful() {
return Err(InstallError::new(format!(
"Failed to apply changes to registry: error code {}\n\n{}",
result.exit_code, result.output
)));
}
Ok(())
}

@ -0,0 +1,224 @@
/// Embedded registry resources, matching the original C# resource strings.
pub const UNINSTALL_REGISTRY: &str = r#"[HKEY_LOCAL_MACHINE\Software\Microsoft\Internet Explorer\MAIN\FeatureControl\FEATURE_BROWSER_EMULATION]
"FlashHelperService.exe"=-
[HKEY_LOCAL_MACHINE\Software\Microsoft\Windows\CurrentVersion\Control Panel\Extended Properties\System.ControlPanel.Category]
"${SYSTEM_64_PATH}\\FlashPlayerCPLApp.cpl"=-
[-HKEY_CURRENT_USER\Software\FlashCenter]
[-HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\App Paths\FlashCenter.exe]
[-HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Uninstall\FlashCenter]
[-HKEY_LOCAL_MACHINE\Software\Classes\AppID\{119DA84B-E3DB-4D47-A8DD-7FF6D5804689}]
[-HKEY_LOCAL_MACHINE\Software\Classes\AppID\{B9020634-CE8F-4F09-9FBC-D108A73A4676}]
[-HKEY_LOCAL_MACHINE\Software\Classes\TypeLib\{37EF68ED-16D3-4191-86BF-AB731D75AAB7}]
[-HKEY_LOCAL_MACHINE\System\ControlSet001\services\Flash Helper Service]
[-HKEY_LOCAL_MACHINE\System\ControlSet001\services\FlashCenterService]
[-HKEY_LOCAL_MACHINE\System\CurrentControlSet\services\Flash Helper Service]
[-HKEY_LOCAL_MACHINE\System\CurrentControlSet\services\FlashCenterService]
[-HKEY_LOCAL_MACHINE\Software\Classes\MacromediaFlashPaper.MacromediaFlashPaper]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Internet Explorer\Low Rights\ElevationPolicy\{FAF199D2-BFA7-4394-A4DE-044A08E59B32}]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashPlayerUpdateService.exe]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashUtil32_${VERSION_PATH}_ActiveX.exe]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashUtil64_${VERSION_PATH}_ActiveX.exe]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashUtil32_${VERSION_PATH}_Plugin.exe]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashUtil64_${VERSION_PATH}_Plugin.exe]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashPlayerPlugin_${VERSION_PATH}.exe]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashUtil32_${VERSION_PATH}_pepper.exe]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashUtil64_${VERSION_PATH}_pepper.exe]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Internet Explorer\Low Rights\ElevationPolicy\{FAF199D2-BFA7-4394-A4DE-044A08E59B32}]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall\Adobe Flash Player ActiveX]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall\Adobe Flash Player NPAPI]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall\Adobe Flash Player PPAPI]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Windows\CurrentVersion\Uninstall\Clean Flash Player]
[-HKEY_LOCAL_MACHINE\Software\Classes\.mfp]
[-HKEY_LOCAL_MACHINE\Software\Classes\.sol]
[-HKEY_LOCAL_MACHINE\Software\Classes\.sor]
[-HKEY_LOCAL_MACHINE\Software\Classes\.spl]
[-HKEY_LOCAL_MACHINE\Software\Classes\.swf]
[-HKEY_LOCAL_MACHINE\Software\Classes\AppID\{B9020634-CE8F-4F09-9FBC-D108A73A4676}]
[-HKEY_LOCAL_MACHINE\Software\Classes\CLSID\{B019E3BF-E7E5-453C-A2E4-D2C18CA0866F}]
[-HKEY_LOCAL_MACHINE\Software\Classes\CLSID\{D27CDB6E-AE6D-11cf-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Classes\FlashFactory.FlashFactory]
[-HKEY_LOCAL_MACHINE\Software\Classes\FlashFactory.FlashFactory.1]
[-HKEY_LOCAL_MACHINE\Software\Classes\Interface\{299817DA-1FAC-4CE2-8F48-A108237013BD}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Interface\{307F64C0-621D-4D56-BBC6-91EFC13CE40D}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Interface\{57A0E747-3863-4D20-A811-950C84F1DB9B}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Interface\{86230738-D762-4C50-A2DE-A753E5B1686F}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Interface\{D27CDB6C-AE6D-11CF-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Interface\{D27CDB6D-AE6D-11CF-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Classes\MIME\Database\Content Type\application/futuresplash]
[-HKEY_LOCAL_MACHINE\Software\Classes\MIME\Database\Content Type\application/x-shockwave-flash]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.1]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.2]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.3]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.4]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.5]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.6]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.7]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.8]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.9]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.10]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.11]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.12]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.13]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.14]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.15]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.16]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.17]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.18]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.19]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.20]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.21]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.22]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.23]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.24]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.25]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.26]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.27]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.28]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.29]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.30]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.31]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.32]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.33]
[-HKEY_LOCAL_MACHINE\Software\Classes\ShockwaveFlash.ShockwaveFlash.34]
[-HKEY_LOCAL_MACHINE\Software\Classes\TypeLib\{57A0E746-3863-4D20-A811-950C84F1DB9B}]
[-HKEY_LOCAL_MACHINE\Software\Classes\TypeLib\{D27CDB6B-AE6D-11CF-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Classes\TypeLib\{FAB3E735-69C7-453B-A446-B6823C6DF1C9}]
[-HKEY_LOCAL_MACHINE\Software\Macromedia]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Internet Explorer\ActiveX Compatibility\{D27CDB6E-AE6D-11CF-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Internet Explorer\ActiveX Compatibility\{D27CDB70-AE6D-11cf-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Microsoft\Internet Explorer\NavigatorPluginsList\Shockwave Flash]
[-HKEY_LOCAL_MACHINE\Software\MozillaPlugins\@adobe.com/FlashPlayer]
[-HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\miniconfig]
[-HKEY_USERS\.DEFAULT\Software\Microsoft\Windows\CurrentVersion\miniconfig]"#;
pub const UNINSTALL_REGISTRY_64: &str = r#"[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Internet Explorer\MAIN\FeatureControl\FEATURE_BROWSER_EMULATION]
"FlashHelperService.exe"=-
[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Windows\CurrentVersion\Control Panel\Extended Properties\System.ControlPanel.Category]
"${SYSTEM_32_PATH}\\FlashPlayerCPLApp.cpl"=-
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\CLSID\{B019E3BF-E7E5-453C-A2E4-D2C18CA0866F}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\CLSID\{D27CDB6E-AE6D-11cf-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\CLSID\{D27CDB70-AE6D-11cf-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\Interface\{299817DA-1FAC-4CE2-8F48-A108237013BD}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\Interface\{307F64C0-621D-4D56-BBC6-91EFC13CE40D}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\Interface\{57A0E747-3863-4D20-A811-950C84F1DB9B}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\Interface\{86230738-D762-4C50-A2DE-A753E5B1686F}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\Interface\{D27CDB6C-AE6D-11CF-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Classes\Wow6432Node\Interface\{D27CDB6D-AE6D-11CF-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Macromedia]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Internet Explorer\ActiveX Compatibility\{D27CDB6E-AE6D-11CF-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Internet Explorer\ActiveX Compatibility\{D27CDB70-AE6D-11cf-96B8-444553540000}]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Internet Explorer\NavigatorPluginsList\Shockwave Flash]
[-HKEY_LOCAL_MACHINE\Software\Wow6432Node\MozillaPlugins\@adobe.com/FlashPlayer]"#;
pub const INSTALL_GENERAL: &str = r#"[HKEY_LOCAL_MACHINE\Software\Microsoft\Windows\CurrentVersion\Control Panel\Extended Properties\System.ControlPanel.Category]
"${SYSTEM_32_PATH}\\FlashPlayerCPLApp.cpl"=dword:0000000a
[HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashPlayerApp.exe]
"DisableExceptionChainValidation"=dword:00000000
[HKEY_LOCAL_MACHINE\Software\Microsoft\Windows\CurrentVersion\Uninstall\Clean Flash Player]
"DisplayName"="Clean Flash Player ${VERSION}"
"HelpLink"="https://gitlab.com/cleanflash/installer#clean-flash-player"
"NoModify"=dword:00000001
"NoRepair"=dword:00000001
"URLInfoAbout"="https://gitlab.com/cleanflash/installer#clean-flash-player"
"URLUpdateInfo"="https://gitlab.com/cleanflash/installer#clean-flash-player"
"VersionMajor"=dword:00000022
"VersionMinor"=dword:00000000
"Publisher"="CleanFlash Team"
"EstimatedSize"=dword:00011cb8
"DisplayIcon"="${PROGRAM_FLASH_32_PATH}\\FlashUtil_Uninstall.exe"
"UninstallString"="${PROGRAM_FLASH_32_PATH}\\FlashUtil_Uninstall.exe"
"DisplayVersion"="${VERSION}""#;
pub const INSTALL_GENERAL_64: &str = r#"[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Windows\CurrentVersion\Control Panel\Extended Properties\System.ControlPanel.Category]
"${SYSTEM_32_PATH}\\FlashPlayerCPLApp.cpl"=dword:0000000a
[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashPlayerApp.exe]
"DisableExceptionChainValidation"=dword:00000000"#;
pub const INSTALL_NP: &str = r#"[HKEY_LOCAL_MACHINE\Software\Macromedia\FlashPlayerPlugin]
"isPartner"=dword:00000001
"Version"="${VERSION}"
"PlayerPath"="${FLASH_64_PATH}\\NPSWF${ARCH}_${VERSION_PATH}.dll"
"UninstallerPath"=-
"isScriptDebugger"=dword:00000000
"isESR"=dword:00000000
"isMSI"=dword:00000000
[HKEY_LOCAL_MACHINE\Software\Macromedia\FlashPlayerPluginReleaseType]
"Release"=dword:00000001
[HKEY_LOCAL_MACHINE\Software\MozillaPlugins\@adobe.com/FlashPlayer]
"Vendor"="Adobe"
"ProductName"="Adobe® Flash® Player ${VERSION} Plugin"
"Path"="${FLASH_64_PATH}\\NPSWF${ARCH}_${VERSION_PATH}.dll"
"Version"="${VERSION}"
"Description"="Adobe® Flash® Player ${VERSION} Plugin"
[HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashPlayerPlugin_${VERSION_PATH}.exe]
"DisableExceptionChainValidation"=dword:00000000"#;
pub const INSTALL_NP_64: &str = r#"[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Macromedia\FlashPlayerPlugin]
"PlayerPath"="${FLASH_32_PATH}\\NPSWF_${VERSION_PATH}.dll"
"Version"="${VERSION}"
"UninstallerPath"=-
"isScriptDebugger"=dword:00000000
"isESR"=dword:00000000
"isMSI"=dword:00000000
"isPartner"=dword:00000001
[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Macromedia\FlashPlayerPluginReleaseType]
"Release"=dword:00000001
[HKEY_LOCAL_MACHINE\Software\Wow6432Node\MozillaPlugins\@adobe.com/FlashPlayer]
"ProductName"="Adobe® Flash® Player ${VERSION} Plugin"
"Description"="Adobe® Flash® Player ${VERSION} Plugin"
"Version"="${VERSION}"
"XPTPath"="${FLASH_32_PATH}\\flashplayer.xpt"
"Vendor"="Adobe"
"Path"="${FLASH_32_PATH}\\NPSWF32_${VERSION_PATH}.dll"
[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\FlashPlayerPlugin_${VERSION_PATH}.exe]
"DisableExceptionChainValidation"=dword:00000000"#;
pub const INSTALL_PP: &str = r#"[HKEY_LOCAL_MACHINE\Software\Macromedia\FlashPlayerPepper]
"UninstallerPath"=-
"PlayerPath"="${FLASH_64_PATH}\\pepflashplayer${ARCH}_${VERSION_PATH}.dll"
"isScriptDebugger"=dword:00000000
"isESR"=dword:00000000
"isMSI"=dword:00000000
"isPartner"=dword:00000001
"Version"="${VERSION}"
[HKEY_LOCAL_MACHINE\Software\Macromedia\FlashPlayerPepperReleaseType]
"Release"=dword:00000001"#;
pub const INSTALL_PP_64: &str = r#"[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Macromedia\FlashPlayerPepper]
"UninstallerPath"=-
"PlayerPath"="${FLASH_32_PATH}\\pepflashplayer32_${VERSION_PATH}.dll"
"isScriptDebugger"=dword:00000000
"isESR"=dword:00000000
"isMSI"=dword:00000000
"isPartner"=dword:00000001
"Version"="${VERSION}"
[HKEY_LOCAL_MACHINE\Software\Wow6432Node\Macromedia\FlashPlayerPepperReleaseType]
"Release"=dword:00000001"#;

@ -0,0 +1,179 @@
use crate::update_checker;
use std::collections::HashMap;
use std::env;
use std::path::{Path, PathBuf};
/// Lazily-initialized system path info, analogous to C# SystemInfo class.
pub struct SystemInfo {
pub system32_path: PathBuf,
pub system64_path: PathBuf,
pub program32_path: PathBuf,
pub flash_program32_path: PathBuf,
pub macromed32_path: PathBuf,
pub macromed64_path: PathBuf,
pub flash32_path: PathBuf,
pub flash64_path: PathBuf,
pub version: String,
pub version_path: String,
pub version_comma: String,
pub is_64bit: bool,
replacements: HashMap<String, String>,
}
impl SystemInfo {
pub fn new() -> Self {
let system32_path = get_syswow64_path();
let system64_path = get_system32_path();
let program32_path = get_program_files_x86();
let flash_program32_path = program32_path.join("Flash Player");
let macromed32_path = system32_path.join("Macromed");
let macromed64_path = system64_path.join("Macromed");
let flash32_path = macromed32_path.join("Flash");
let flash64_path = macromed64_path.join("Flash");
let version = update_checker::FLASH_VERSION.to_string();
let version_path = version.replace('.', "_");
let version_comma = version.replace('.', ",");
let is_64bit = cfg!(target_pointer_width = "64")
|| env::var("PROCESSOR_ARCHITEW6432").is_ok();
let arch = if is_64bit { "64" } else { "32" };
let mut replacements = HashMap::new();
replacements.insert(
"${SYSTEM_32_PATH}".into(),
system32_path.to_string_lossy().replace('\\', "\\\\"),
);
replacements.insert(
"${SYSTEM_64_PATH}".into(),
system64_path.to_string_lossy().replace('\\', "\\\\"),
);
replacements.insert(
"${PROGRAM_32_PATH}".into(),
program32_path.to_string_lossy().replace('\\', "\\\\"),
);
replacements.insert(
"${PROGRAM_FLASH_32_PATH}".into(),
flash_program32_path.to_string_lossy().replace('\\', "\\\\"),
);
replacements.insert(
"${FLASH_32_PATH}".into(),
flash32_path.to_string_lossy().replace('\\', "\\\\"),
);
replacements.insert(
"${FLASH_64_PATH}".into(),
flash64_path.to_string_lossy().replace('\\', "\\\\"),
);
replacements.insert("${VERSION}".into(), version.clone());
replacements.insert("${VERSION_PATH}".into(), version_path.clone());
replacements.insert("${VERSION_COMMA}".into(), version_comma.clone());
replacements.insert("${ARCH}".into(), arch.into());
Self {
system32_path,
system64_path,
program32_path,
flash_program32_path,
macromed32_path,
macromed64_path,
flash32_path,
flash64_path,
version,
version_path,
version_comma,
is_64bit,
replacements,
}
}
pub fn system_paths(&self) -> Vec<&Path> {
if self.is_64bit {
vec![&self.system32_path, &self.system64_path]
} else {
vec![&self.system32_path]
}
}
pub fn macromed_paths(&self) -> Vec<&Path> {
if self.is_64bit {
vec![&self.macromed32_path, &self.macromed64_path]
} else {
vec![&self.macromed32_path]
}
}
pub fn fill_string(&self, s: &str) -> String {
let mut result = s.to_string();
for (key, value) in &self.replacements {
result = result.replace(key.as_str(), value);
}
result
}
pub fn is_legacy_windows(&self) -> bool {
// Windows version < 6.2 (before Windows 8).
unsafe {
let mut info: windows_sys::Win32::System::SystemInformation::OSVERSIONINFOW =
std::mem::zeroed();
info.dwOSVersionInfoSize =
std::mem::size_of::<windows_sys::Win32::System::SystemInformation::OSVERSIONINFOW>()
as u32;
// RtlGetVersion always succeeds and isn't deprecated like GetVersionEx.
rtl_get_version(&mut info);
info.dwMajorVersion < 6
|| (info.dwMajorVersion == 6 && info.dwMinorVersion < 2)
}
}
}
extern "system" {
fn RtlGetVersion(
lp_version_information: *mut windows_sys::Win32::System::SystemInformation::OSVERSIONINFOW,
) -> i32;
}
unsafe fn rtl_get_version(
info: &mut windows_sys::Win32::System::SystemInformation::OSVERSIONINFOW,
) {
RtlGetVersion(info as *mut _);
}
/// Global convenience: fill replacement strings using a default SystemInfo.
pub fn fill_string(s: &str) -> String {
SYSTEM_INFO.with(|si| si.fill_string(s))
}
thread_local! {
static SYSTEM_INFO: SystemInfo = SystemInfo::new();
}
pub fn with_system_info<F, R>(f: F) -> R
where
F: FnOnce(&SystemInfo) -> R,
{
SYSTEM_INFO.with(f)
}
fn get_system32_path() -> PathBuf {
PathBuf::from(env::var("SYSTEMROOT").unwrap_or_else(|_| r"C:\Windows".into()))
.join("System32")
}
fn get_syswow64_path() -> PathBuf {
let root = env::var("SYSTEMROOT").unwrap_or_else(|_| r"C:\Windows".into());
let wow64 = PathBuf::from(&root).join("SysWOW64");
if wow64.exists() {
wow64
} else {
PathBuf::from(&root).join("System32")
}
}
fn get_program_files_x86() -> PathBuf {
if let Ok(pf86) = env::var("PROGRAMFILES(X86)") {
PathBuf::from(pf86)
} else if let Ok(pf) = env::var("PROGRAMFILES") {
PathBuf::from(pf)
} else {
PathBuf::from(r"C:\Program Files")
}
}

@ -0,0 +1,302 @@
use crate::{
file_util, process_utils, registry, resources, system_info, winapi_helpers, InstallError,
ProgressCallback,
};
use std::env;
use std::path::{Path, PathBuf};
const PROCESSES_TO_KILL: &[&str] = &[
"fcbrowser",
"fcbrowsermanager",
"fclogin",
"fctips",
"flashcenter",
"flashcenterservice",
"flashcenteruninst",
"flashplay",
"update",
"wow_helper",
"dummy_cmd",
"flashhelperservice",
"flashplayerapp",
"flashplayer_sa",
"flashplayer_sa_debug",
];
const CONDITIONAL_PROCESSES: &[&str] = &[
"plugin-container",
"opera",
"iexplore",
"chrome",
"chromium",
"brave",
"vivaldi",
"msedge",
];
/// Unregister an ActiveX OCX file via regsvr32.
pub fn unregister_activex(filename: &str) -> Result<(), InstallError> {
winapi_helpers::allow_modifications();
let path = Path::new(filename);
let dir = path.parent().unwrap_or(Path::new("."));
let file_name = path
.file_name()
.unwrap_or_default()
.to_string_lossy()
.to_string();
let _prev = env::current_dir();
let _ = env::set_current_dir(dir);
let process = process_utils::run_process("regsvr32.exe", &["/s", "/u", &file_name]);
if !process.is_successful() {
return Err(InstallError::new(format!(
"Failed to unregister ActiveX plugin: error code {}\n\n{}",
process.exit_code, process.output
)));
}
Ok(())
}
fn uninstall_registry() -> Result<(), InstallError> {
system_info::with_system_info(|si| {
if si.is_64bit {
registry::apply_registry(&[
resources::UNINSTALL_REGISTRY,
resources::UNINSTALL_REGISTRY_64,
])
} else {
registry::apply_registry(&[resources::UNINSTALL_REGISTRY])
}
})
}
fn delete_task(task: &str) {
process_utils::run_unmanaged_process("schtasks.exe", &["/delete", "/tn", task, "/f"]);
}
fn stop_service(service: &str) {
process_utils::run_unmanaged_process("net.exe", &["stop", service]);
}
fn delete_service(service: &str) {
stop_service(service);
process_utils::run_unmanaged_process("sc.exe", &["delete", service]);
}
fn delete_flash_center() {
// Remove Flash Center from Program Files.
let pf = env::var("PROGRAMFILES").unwrap_or_default();
file_util::wipe_folder(&PathBuf::from(&pf).join("FlashCenter"));
if let Ok(pf86) = env::var("PROGRAMFILES(X86)") {
file_util::wipe_folder(&PathBuf::from(&pf86).join("FlashCenter"));
}
// Remove start menu shortcuts.
if let Some(appdata) = env::var("PROGRAMDATA").ok() {
file_util::wipe_folder(
&PathBuf::from(&appdata)
.join("Microsoft")
.join("Windows")
.join("Start Menu")
.join("Programs")
.join("Flash Center"),
);
}
// Remove Flash Center cache / user data.
if let Some(local) = env::var("LOCALAPPDATA").ok() {
file_util::wipe_folder(&PathBuf::from(&local).join("Flash_Center"));
}
// Remove common start menu shortcuts.
if let Some(appdata) = env::var("APPDATA").ok() {
file_util::wipe_folder(
&PathBuf::from(&appdata)
.join("Microsoft")
.join("Windows")
.join("Start Menu")
.join("Programs")
.join("Flash Center"),
);
}
// Remove Desktop shortcuts.
if let Some(desktop) = get_common_desktop() {
file_util::delete_file(&desktop.join("Flash Center.lnk"));
}
if let Some(desktop) = dirs_desktop() {
file_util::delete_file(&desktop.join("Flash Player.lnk"));
}
// Remove Flash Player from Program Files.
system_info::with_system_info(|si| {
file_util::wipe_folder(&si.flash_program32_path);
});
// Clean up temp folder spyware remnants.
let temp = env::temp_dir();
if let Ok(entries) = std::fs::read_dir(&temp) {
for entry in entries.flatten() {
let name = entry.file_name().to_string_lossy().to_string();
if name.len() == 11 && name.ends_with(".tmp") && entry.path().is_dir() {
let _ = file_util::wipe_folder(&entry.path());
}
}
}
}
fn delete_flash_player() {
system_info::with_system_info(|si| {
// Remove Macromed folders.
for dir in si.macromed_paths() {
file_util::recursive_delete(dir, None);
}
// Remove Flash Player control panel apps.
for sys_dir in si.system_paths() {
file_util::delete_file(&sys_dir.join("FlashPlayerApp.exe"));
file_util::delete_file(&sys_dir.join("FlashPlayerCPLApp.cpl"));
}
});
}
fn should_kill_conditional_process(name: &str, pid: u32) -> bool {
if !CONDITIONAL_PROCESSES
.iter()
.any(|p| p.eq_ignore_ascii_case(name))
{
return false;
}
let modules = process_utils::collect_modules(pid);
modules.iter().any(|m| {
let lower = m.to_lowercase();
lower.starts_with("flash32")
|| lower.starts_with("flash64")
|| lower.starts_with("libpepflash")
|| lower.starts_with("npswf")
})
}
fn stop_processes() {
use windows_sys::Win32::Foundation::CloseHandle;
use windows_sys::Win32::System::Threading::{
OpenProcess, TerminateProcess, WaitForSingleObject, PROCESS_QUERY_INFORMATION,
PROCESS_TERMINATE, PROCESS_VM_READ,
};
// Enumerate all processes via the snapshot API.
let pids = enumerate_processes();
for (pid, name) in &pids {
let lower = name.to_lowercase();
let should_kill = PROCESSES_TO_KILL.iter().any(|p| *p == lower)
|| should_kill_conditional_process(&lower, *pid);
if !should_kill {
continue;
}
unsafe {
let handle = OpenProcess(PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, 0, *pid);
if !handle.is_null() {
TerminateProcess(handle, 1);
WaitForSingleObject(handle, 5000);
CloseHandle(handle);
}
}
}
}
fn enumerate_processes() -> Vec<(u32, String)> {
use windows_sys::Win32::System::ProcessStatus::{EnumProcesses, GetModuleBaseNameW};
use windows_sys::Win32::System::Threading::{OpenProcess, PROCESS_QUERY_INFORMATION, PROCESS_VM_READ};
use windows_sys::Win32::Foundation::CloseHandle;
let mut results = Vec::new();
let mut pids = [0u32; 4096];
let mut bytes_returned: u32 = 0;
unsafe {
if EnumProcesses(
pids.as_mut_ptr(),
std::mem::size_of_val(&pids) as u32,
&mut bytes_returned,
) == 0
{
return results;
}
let count = bytes_returned as usize / std::mem::size_of::<u32>();
for &pid in &pids[..count] {
if pid == 0 {
continue;
}
let handle = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, 0, pid);
if handle.is_null() {
continue;
}
let mut name_buf = [0u16; 260];
let len = GetModuleBaseNameW(handle, std::ptr::null_mut(), name_buf.as_mut_ptr(), 260);
CloseHandle(handle);
if len > 0 {
let name = String::from_utf16_lossy(&name_buf[..len as usize]);
// Strip .exe suffix for matching.
let base = name.strip_suffix(".exe").unwrap_or(&name).to_string();
results.push((pid, base));
}
}
}
results
}
/// Perform the full uninstallation sequence.
pub fn uninstall(form: &dyn ProgressCallback) -> Result<(), InstallError> {
winapi_helpers::allow_modifications();
form.update_progress_label("Stopping Flash auto-updater task...", true);
delete_task("Adobe Flash Player Updater");
form.update_progress_label("Stopping Flash auto-updater service...", true);
delete_service("AdobeFlashPlayerUpdateSvc");
form.update_progress_label("Stopping Flash Center services...", true);
delete_service("Flash Helper Service");
form.tick_progress();
delete_service("FlashCenterService");
form.update_progress_label("Exiting all browsers...", true);
stop_processes();
form.update_progress_label("Cleaning up registry...", true);
uninstall_registry()?;
form.update_progress_label("Removing Flash Center...", true);
delete_flash_center();
form.update_progress_label("Removing Flash Player...", true);
delete_flash_player();
Ok(())
}
// Helper to get common desktop path.
fn get_common_desktop() -> Option<PathBuf> {
env::var("PUBLIC")
.ok()
.map(|p| PathBuf::from(p).join("Desktop"))
}
fn dirs_desktop() -> Option<PathBuf> {
env::var("USERPROFILE")
.ok()
.map(|p| PathBuf::from(p).join("Desktop"))
}

@ -0,0 +1,14 @@
pub const FLASH_VERSION: &str = "34.0.0.330";
pub const VERSION: &str = "34.0.0.330";
pub struct VersionInfo {
pub name: String,
pub version: String,
pub url: String,
}
pub fn get_latest_version() -> Option<VersionInfo> {
// The original fetches from the GitHub API.
// Stubbed for the port; real implementation would use ureq or reqwest.
None
}

@ -0,0 +1,58 @@
/// Enable SeRestorePrivilege and SeTakeOwnershipPrivilege for the current process.
pub fn allow_modifications() {
let _ = modify_privilege("SeRestorePrivilege\0", true);
let _ = modify_privilege("SeTakeOwnershipPrivilege\0", true);
}
fn modify_privilege(name: &str, enable: bool) -> Result<(), ()> {
use windows_sys::Win32::Foundation::CloseHandle;
use windows_sys::Win32::Security::{
AdjustTokenPrivileges, LookupPrivilegeValueW, LUID_AND_ATTRIBUTES,
SE_PRIVILEGE_ENABLED, TOKEN_ADJUST_PRIVILEGES, TOKEN_PRIVILEGES, TOKEN_QUERY,
};
use windows_sys::Win32::System::Threading::{GetCurrentProcess, OpenProcessToken};
unsafe {
let mut token: *mut std::ffi::c_void = std::ptr::null_mut();
if OpenProcessToken(
GetCurrentProcess(),
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY,
&mut token,
) == 0
{
return Err(());
}
let wide_name: Vec<u16> = name.encode_utf16().collect();
let mut luid = std::mem::zeroed();
if LookupPrivilegeValueW(std::ptr::null(), wide_name.as_ptr(), &mut luid) == 0 {
CloseHandle(token);
return Err(());
}
let mut tp = TOKEN_PRIVILEGES {
PrivilegeCount: 1,
Privileges: [LUID_AND_ATTRIBUTES {
Luid: luid,
Attributes: if enable { SE_PRIVILEGE_ENABLED } else { 0 },
}],
};
let result = AdjustTokenPrivileges(
token,
0,
&mut tp,
std::mem::size_of::<TOKEN_PRIVILEGES>() as u32,
std::ptr::null_mut(),
std::ptr::null_mut(),
);
CloseHandle(token);
if result == 0 {
Err(())
} else {
Ok(())
}
}
}

@ -0,0 +1,15 @@
[package]
name = "clean_flash_installer"
version = "34.0.0"
edition = "2021"
authors = ["FlashPatch Team"]
[dependencies]
clean_flash_common = { path = "../clean_flash_common" }
clean_flash_ui = { path = "../clean_flash_ui" }
minifb = { workspace = true }
windows-sys = { workspace = true }
sevenz-rust2 = { workspace = true }
[build-dependencies]
winresource = "0.1"

@ -0,0 +1,37 @@
fn main() {
// Set application icon from .ico resource (if present).
if cfg!(target_os = "windows") {
let mut res = winresource::WindowsResource::new();
// Attempt to set icon; ignore if file doesn't exist.
if std::path::Path::new("../../resources/icon.ico").exists() {
res.set_icon("../../resources/icon.ico");
}
res.set("ProductName", "Clean Flash Player Installer");
res.set("FileDescription", "Clean Flash Player Installer");
res.set("ProductVersion", "34.0.0.330");
res.set("CompanyName", "FlashPatch Team");
res.set("LegalCopyright", "FlashPatch Team");
res.set_manifest(r#"<?xml version="1.0" encoding="utf-8"?>
<assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1">
<assemblyIdentity version="1.0.0.0" name="CleanFlashInstaller.app"/>
<trustInfo xmlns="urn:schemas-microsoft-com:asm.v2">
<security>
<requestedPrivileges xmlns="urn:schemas-microsoft-com:asm.v3">
<requestedExecutionLevel level="requireAdministrator" uiAccess="false" />
</requestedPrivileges>
</security>
</trustInfo>
<compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1">
<application>
<supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}"/>
<supportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}"/>
<supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}"/>
<supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}"/>
<supportedOS Id="{e2011457-1546-43c5-a5fe-008deee3d3f0}"/>
</application>
</compatibility>
</assembly>
"#);
let _ = res.compile();
}
}

@ -0,0 +1,72 @@
/// Bitflag-based install options, mirroring the C# InstallFlags class.
pub const NONE: u32 = 0;
pub const PEPPER: u32 = 1 << 0;
pub const NETSCAPE: u32 = 1 << 1;
pub const ACTIVEX: u32 = 1 << 2;
pub const PLAYER: u32 = 1 << 3;
pub const PLAYER_START_MENU: u32 = 1 << 4;
pub const PLAYER_DESKTOP: u32 = 1 << 5;
pub const X64: u32 = 1 << 6;
pub const DEBUG: u32 = 1 << 7;
const UNINSTALL_TICKS: u32 = 9;
const INSTALL_GENERAL_TICKS: u32 = 5;
#[derive(Clone, Copy)]
pub struct InstallFlags {
value: u32,
}
impl InstallFlags {
pub fn new() -> Self {
Self { value: 0 }
}
pub fn from(value: u32) -> Self {
Self { value }
}
pub fn get_value(self) -> u32 {
self.value
}
pub fn is_set(self, flag: u32) -> bool {
(self.value & flag) == flag
}
pub fn is_none_set(self) -> bool {
self.value == 0
}
pub fn set_flag(&mut self, flag: u32) {
self.value |= flag;
}
pub fn set_conditionally(&mut self, condition: bool, flag: u32) {
if condition {
self.set_flag(flag);
}
}
pub fn get_ticks(self) -> u32 {
let is_64bit = cfg!(target_pointer_width = "64")
|| std::env::var("PROCESSOR_ARCHITEW6432").is_ok();
let mut ticks = (if self.is_set(PEPPER) { 1 } else { 0 })
+ (if self.is_set(NETSCAPE) { 1 } else { 0 })
+ (if self.is_set(ACTIVEX) { 2 } else { 0 });
if is_64bit {
ticks *= 2;
}
if self.is_set(PLAYER) {
ticks += 1;
}
ticks += UNINSTALL_TICKS;
ticks += INSTALL_GENERAL_TICKS;
ticks
}
}

@ -0,0 +1,691 @@
use crate::install_flags::{self, InstallFlags};
use crate::installer;
use clean_flash_common::{uninstaller, redirection, update_checker, ProgressCallback};
use clean_flash_ui::font::FontManager;
use clean_flash_ui::renderer::{Renderer, RgbaImage};
use clean_flash_ui::widgets::button::GradientButton;
use clean_flash_ui::widgets::checkbox::ImageCheckBox;
use clean_flash_ui::widgets::label::Label;
use clean_flash_ui::widgets::progress_bar::ProgressBar;
use std::sync::{Arc, Mutex};
// Window dimensions matching the C# form.
pub const WIDTH: usize = 712;
pub const HEIGHT: usize = 329;
const BG_COLOR: u32 = Renderer::rgb(50, 51, 51);
const FG_COLOR: u32 = Renderer::rgb(245, 245, 245);
const PANEL_X: i32 = 90;
const PANEL_Y: i32 = 162;
const DISCLAIMER_TEXT: &str = "I am aware that Adobe Flash Player is no longer supported, nor provided by Adobe Inc.\n\
Clean Flash Player is a third-party version of Flash Player built from the latest Flash Player\n\
version with adware removed.\n\n\
Adobe is not required by any means to provide support for this version of Flash Player.";
const COMPLETE_INSTALL_TEXT: &str = "Clean Flash Player has been successfully installed!\n\
Don't forget, Flash Player is no longer compatible with new browsers.\n\n\
For browser recommendations and Flash Player updates,\n\
check out Clean Flash Player's website!";
const COMPLETE_UNINSTALL_TEXT: &str = "\nAll versions of Flash Player have been successfully uninstalled.\n\n\
If you ever change your mind, check out Clean Flash Player's website!";
/// Which panel is currently shown in the wizard.
#[derive(Clone, Copy, PartialEq, Eq)]
pub enum Panel {
Disclaimer,
Choice,
PlayerChoice,
DebugChoice,
BeforeInstall,
Install,
Complete,
Failure,
}
/// Shared progress state set by the background install thread.
pub struct ProgressState {
pub label: String,
pub value: i32,
pub done: bool,
pub error: Option<String>,
}
/// Full application state for the installer form.
pub struct InstallForm {
pub panel: Panel,
// Header
pub title_text: String,
pub subtitle_text: String,
pub flash_logo: RgbaImage,
// Checkbox images
pub checkbox_on: RgbaImage,
pub checkbox_off: RgbaImage,
// Navigation buttons
pub prev_button: GradientButton,
pub next_button: GradientButton,
// Disclaimer panel
pub disclaimer_label: Label,
pub disclaimer_box: ImageCheckBox,
// Choice panel (browser plugins)
pub browser_ask_label: Label,
pub pepper_box: ImageCheckBox,
pub pepper_label: Label,
pub netscape_box: ImageCheckBox,
pub netscape_label: Label,
pub activex_box: ImageCheckBox,
pub activex_label: Label,
// Player choice panel
pub player_ask_label: Label,
pub player_box: ImageCheckBox,
pub player_label: Label,
pub player_desktop_box: ImageCheckBox,
pub player_desktop_label: Label,
pub player_start_menu_box: ImageCheckBox,
pub player_start_menu_label: Label,
// Debug choice panel
pub debug_ask_label: Label,
pub debug_button: GradientButton,
pub debug_chosen: bool,
// Before install panel
pub before_install_label: Label,
// Install panel
pub install_header_label: Label,
pub progress_label: Label,
pub progress_bar: ProgressBar,
// Complete panel
pub complete_label: Label,
// Failure panel
pub failure_text_label: Label,
pub failure_detail: String,
pub copy_error_button: GradientButton,
// Shared progress state (for background thread communication).
pub progress_state: Arc<Mutex<ProgressState>>,
// Fonts loaded once.
pub fonts: FontManager,
// Mouse tracking
prev_mouse_down: bool,
}
impl InstallForm {
pub fn new() -> Self {
let version = update_checker::FLASH_VERSION;
let title_text = "Clean Flash Player".to_string();
let subtitle_text = format!("built from version {} (China)", version);
// Load images from the common resources folder.
// These are loaded from the C# project's assets alongside the binary.
let flash_logo = load_resource_image("flashLogo.png");
let checkbox_on = load_resource_image("checkboxOn.png");
let checkbox_off = load_resource_image("checkboxOff.png");
let fonts = FontManager::new();
Self {
panel: Panel::Disclaimer,
title_text,
subtitle_text,
flash_logo,
checkbox_on,
checkbox_off,
prev_button: GradientButton::new(90, 286, 138, 31, "QUIT"),
next_button: GradientButton::new(497, 286, 138, 31, "AGREE"),
// Disclaimer panel
disclaimer_label: Label::new(PANEL_X + 25, PANEL_Y, DISCLAIMER_TEXT, 13.0),
disclaimer_box: ImageCheckBox::new(PANEL_X, PANEL_Y),
// Choice panel
browser_ask_label: Label::new(
PANEL_X - 2,
PANEL_Y + 2,
"Which browser plugins would you like to install?",
13.0,
),
pepper_box: ImageCheckBox::new(PANEL_X, PANEL_Y + 47),
pepper_label: Label::new(
PANEL_X + 24,
PANEL_Y + 47,
"Pepper API (PPAPI)\n(Chrome/Opera/Brave)",
13.0,
),
netscape_box: ImageCheckBox::new(PANEL_X + 186, PANEL_Y + 47),
netscape_label: Label::new(
PANEL_X + 210,
PANEL_Y + 47,
"Netscape API (NPAPI)\n(Firefox/ESR/Waterfox)",
13.0,
),
activex_box: ImageCheckBox::new(PANEL_X + 365, PANEL_Y + 47),
activex_label: Label::new(
PANEL_X + 389,
PANEL_Y + 47,
"ActiveX (OCX)\n(IE/Embedded/Desktop)",
13.0,
),
// Player choice panel
player_ask_label: Label::new(
PANEL_X - 2,
PANEL_Y + 2,
"Would you like to install the standalone Flash Player?",
13.0,
),
player_box: ImageCheckBox::new(PANEL_X, PANEL_Y + 47),
player_label: Label::new(
PANEL_X + 24,
PANEL_Y + 47,
"Install Standalone\nFlash Player",
13.0,
),
player_desktop_box: ImageCheckBox::new(PANEL_X + 186, PANEL_Y + 47),
player_desktop_label: Label::new(
PANEL_X + 210,
PANEL_Y + 47,
"Create Shortcuts\non Desktop",
13.0,
),
player_start_menu_box: ImageCheckBox::new(PANEL_X + 365, PANEL_Y + 47),
player_start_menu_label: Label::new(
PANEL_X + 389,
PANEL_Y + 47,
"Create Shortcuts\nin Start Menu",
13.0,
),
// Debug choice panel
debug_ask_label: Label::new(
PANEL_X - 2,
PANEL_Y + 2,
"Would you like to install the debug version of Clean Flash Player?\n\
You should only choose the debug version if you are planning to create Flash applications.\n\
If you are not sure, simply press NEXT.",
13.0,
),
debug_button: GradientButton::new(PANEL_X + 186, PANEL_Y + 65, 176, 31, "INSTALL DEBUG VERSION"),
debug_chosen: false,
// Before install panel
before_install_label: Label::new(PANEL_X + 3, PANEL_Y + 2, "", 13.0),
// Install panel
install_header_label: Label::new(PANEL_X + 3, PANEL_Y, "Installation in progress...", 13.0),
progress_label: Label::new(PANEL_X + 46, PANEL_Y + 30, "Preparing...", 13.0),
progress_bar: ProgressBar::new(PANEL_X + 49, PANEL_Y + 58, 451, 23),
// Complete panel
complete_label: Label::new(PANEL_X, PANEL_Y, "", 13.0),
// Failure panel
failure_text_label: Label::new(
PANEL_X + 3,
PANEL_Y + 2,
"Oops! The installation process has encountered an unexpected problem.\n\
The following details could be useful. Press the Retry button to try again.",
13.0,
),
failure_detail: String::new(),
copy_error_button: GradientButton::new(PANEL_X + 441, PANEL_Y + 58, 104, 31, "COPY"),
progress_state: Arc::new(Mutex::new(ProgressState {
label: "Preparing...".into(),
value: 0,
done: false,
error: None,
})),
fonts,
prev_mouse_down: false,
}
}
/// Called each frame: handle input, update state, draw.
pub fn update_and_draw(
&mut self,
renderer: &mut Renderer,
mx: i32,
my: i32,
mouse_down: bool,
) {
let mouse_released = self.prev_mouse_down && !mouse_down;
self.prev_mouse_down = mouse_down;
// Update navigation button hover states.
self.prev_button.update(mx, my, mouse_down);
self.next_button.update(mx, my, mouse_down);
// Handle click events.
self.handle_input(mx, my, mouse_released);
// Poll progress state from background thread if installing.
if self.panel == Panel::Install {
self.poll_progress();
}
// ----- Draw -----
renderer.clear(BG_COLOR);
// Header: flash logo.
renderer.draw_image(90, 36, &self.flash_logo);
// Title.
self.fonts.draw_text(
renderer,
233,
54,
&self.title_text,
32.0, // ~24pt Segoe UI
FG_COLOR,
);
// Subtitle.
self.fonts.draw_text(
renderer,
280,
99,
&self.subtitle_text,
17.0, // ~13pt Segoe UI
FG_COLOR,
);
// Separator line at y=270.
renderer.fill_rect(0, 270, WIDTH as i32, 1, Renderer::rgb(105, 105, 105));
// Draw current panel.
match self.panel {
Panel::Disclaimer => self.draw_disclaimer(renderer),
Panel::Choice => self.draw_choice(renderer),
Panel::PlayerChoice => self.draw_player_choice(renderer),
Panel::DebugChoice => self.draw_debug_choice(renderer),
Panel::BeforeInstall => self.draw_before_install(renderer),
Panel::Install => self.draw_install(renderer),
Panel::Complete => self.draw_complete(renderer),
Panel::Failure => self.draw_failure(renderer),
}
// Navigation buttons.
self.prev_button.draw(renderer, &self.fonts);
self.next_button.draw(renderer, &self.fonts);
}
fn handle_input(&mut self, mx: i32, my: i32, mouse_released: bool) {
// Navigation button clicks.
if self.prev_button.clicked(mx, my, mouse_released) {
self.on_prev_clicked();
return;
}
if self.next_button.clicked(mx, my, mouse_released) {
self.on_next_clicked();
return;
}
// Panel-specific input.
match self.panel {
Panel::Disclaimer => {
let toggled = self.disclaimer_box.toggle_if_clicked(mx, my, mouse_released);
if toggled || self.disclaimer_label.clicked(mx, my, mouse_released, &self.fonts) {
if !toggled {
self.disclaimer_box.checked = !self.disclaimer_box.checked;
}
self.next_button.enabled = self.disclaimer_box.checked;
}
}
Panel::Choice => {
self.pepper_box.toggle_if_clicked(mx, my, mouse_released);
self.netscape_box.toggle_if_clicked(mx, my, mouse_released);
self.activex_box.toggle_if_clicked(mx, my, mouse_released);
if self.pepper_label.clicked(mx, my, mouse_released, &self.fonts) {
self.pepper_box.checked = !self.pepper_box.checked;
}
if self.netscape_label.clicked(mx, my, mouse_released, &self.fonts) {
self.netscape_box.checked = !self.netscape_box.checked;
}
if self.activex_label.clicked(mx, my, mouse_released, &self.fonts) {
self.activex_box.checked = !self.activex_box.checked;
}
}
Panel::PlayerChoice => {
self.player_box.toggle_if_clicked(mx, my, mouse_released);
self.player_desktop_box.toggle_if_clicked(mx, my, mouse_released);
self.player_start_menu_box.toggle_if_clicked(mx, my, mouse_released);
if self.player_label.clicked(mx, my, mouse_released, &self.fonts) {
self.player_box.checked = !self.player_box.checked;
}
if self.player_desktop_label.clicked(mx, my, mouse_released, &self.fonts) && self.player_box.checked {
self.player_desktop_box.checked = !self.player_desktop_box.checked;
}
if self.player_start_menu_label.clicked(mx, my, mouse_released, &self.fonts) && self.player_box.checked {
self.player_start_menu_box.checked = !self.player_start_menu_box.checked;
}
// Disable sub-options when player unchecked.
self.player_desktop_box.enabled = self.player_box.checked;
self.player_start_menu_box.enabled = self.player_box.checked;
if !self.player_box.checked {
self.player_desktop_box.checked = false;
self.player_start_menu_box.checked = false;
}
}
Panel::DebugChoice => {
if self.debug_button.clicked(mx, my, mouse_released) {
// In the C# app this shows a MessageBox. For the Rust port we toggle directly.
self.debug_chosen = true;
self.open_before_install();
}
self.debug_button.update(mx, my, self.prev_mouse_down);
}
Panel::Failure => {
self.copy_error_button.update(mx, my, self.prev_mouse_down);
if self.copy_error_button.clicked(mx, my, mouse_released) {
// Copy error to clipboard via clip.exe.
let _ = std::process::Command::new("cmd")
.args(["/C", &format!("echo {} | clip", self.failure_detail)])
.output();
}
}
_ => {}
}
}
fn on_prev_clicked(&mut self) {
match self.panel {
Panel::Disclaimer | Panel::Complete | Panel::Failure => {
std::process::exit(0);
}
Panel::Choice => self.open_disclaimer(),
Panel::PlayerChoice => self.open_choice(),
Panel::DebugChoice => self.open_player_choice(),
Panel::BeforeInstall => self.open_debug_choice(),
_ => {}
}
}
fn on_next_clicked(&mut self) {
match self.panel {
Panel::Disclaimer => self.open_choice(),
Panel::Choice => self.open_player_choice(),
Panel::PlayerChoice => self.open_debug_choice(),
Panel::DebugChoice => self.open_before_install(),
Panel::BeforeInstall | Panel::Failure => self.open_install(),
_ => {}
}
}
fn open_disclaimer(&mut self) {
self.panel = Panel::Disclaimer;
self.prev_button.text = "QUIT".into();
self.next_button.text = "AGREE".into();
self.next_button.visible = true;
self.next_button.enabled = self.disclaimer_box.checked;
self.prev_button.enabled = true;
}
fn open_choice(&mut self) {
self.panel = Panel::Choice;
self.prev_button.text = "BACK".into();
self.next_button.text = "NEXT".into();
self.next_button.visible = true;
self.next_button.enabled = true;
self.prev_button.enabled = true;
}
fn open_player_choice(&mut self) {
self.panel = Panel::PlayerChoice;
self.prev_button.text = "BACK".into();
self.next_button.text = "NEXT".into();
self.next_button.visible = true;
self.next_button.enabled = true;
self.prev_button.enabled = true;
}
fn open_debug_choice(&mut self) {
self.panel = Panel::DebugChoice;
self.debug_chosen = false;
self.prev_button.text = "BACK".into();
self.next_button.text = "NEXT".into();
self.next_button.visible = true;
self.next_button.enabled = true;
self.prev_button.enabled = true;
}
fn open_before_install(&mut self) {
self.panel = Panel::BeforeInstall;
self.prev_button.text = "BACK".into();
self.prev_button.enabled = true;
let has_plugins =
self.pepper_box.checked || self.netscape_box.checked || self.activex_box.checked || self.player_box.checked;
if has_plugins {
let mut browsers = Vec::new();
if self.pepper_box.checked {
browsers.push("Google Chrome");
}
if self.netscape_box.checked {
browsers.push("Mozilla Firefox");
}
if self.activex_box.checked {
browsers.push("Internet Explorer");
}
let browser_str = join_with_and(&browsers);
self.before_install_label.text = format!(
"You are about to install Clean Flash Player.\n\
Please close any browser windows running Flash content before you continue.\n\n\
The installer will close all browser windows running Flash, uninstall previous versions of Flash Player and\n\
Flash Center, and install Flash for {}.",
browser_str
);
self.next_button.text = "INSTALL".into();
} else {
self.before_install_label.text =
"You are about to uninstall Clean Flash Player.\n\
Please close any browser windows running Flash content before you continue.\n\n\
The installer will completely remove all versions of Flash Player from this computer,\n\
including Clean Flash Player and older versions of Adobe Flash Player."
.to_string();
self.next_button.text = "UNINSTALL".into();
}
self.next_button.visible = true;
self.next_button.enabled = true;
}
fn open_install(&mut self) {
self.panel = Panel::Install;
self.prev_button.enabled = false;
self.next_button.visible = false;
let mut flags = InstallFlags::new();
flags.set_conditionally(self.pepper_box.checked, install_flags::PEPPER);
flags.set_conditionally(self.netscape_box.checked, install_flags::NETSCAPE);
flags.set_conditionally(self.activex_box.checked, install_flags::ACTIVEX);
flags.set_conditionally(self.player_box.checked, install_flags::PLAYER);
flags.set_conditionally(self.player_desktop_box.checked, install_flags::PLAYER_DESKTOP);
flags.set_conditionally(
self.player_start_menu_box.checked,
install_flags::PLAYER_START_MENU,
);
flags.set_conditionally(self.debug_chosen, install_flags::DEBUG);
self.progress_bar.maximum = flags.get_ticks() as i32;
self.progress_bar.value = 0;
// Reset shared state.
{
let mut state = self.progress_state.lock().unwrap();
state.label = "Preparing...".into();
state.value = 0;
state.done = false;
state.error = None;
}
// Spawn background thread.
let progress = Arc::clone(&self.progress_state);
std::thread::spawn(move || {
let callback = ThreadProgressCallback {
state: Arc::clone(&progress),
};
let redir = redirection::disable_redirection();
let result = (|| -> Result<(), clean_flash_common::InstallError> {
uninstaller::uninstall(&callback)?;
installer::install(&callback, &mut flags)?;
Ok(())
})();
redirection::enable_redirection(redir);
let mut state = progress.lock().unwrap();
state.done = true;
if let Err(e) = result {
state.error = Some(e.to_string());
}
});
}
fn poll_progress(&mut self) {
let state = self.progress_state.lock().unwrap();
self.progress_label.text = state.label.clone();
self.progress_bar.value = state.value;
if state.done {
if let Some(ref err) = state.error {
self.failure_detail = err.clone();
drop(state);
self.open_failure();
} else {
drop(state);
self.open_complete();
}
}
}
fn open_complete(&mut self) {
self.panel = Panel::Complete;
self.prev_button.text = "QUIT".into();
self.prev_button.enabled = true;
self.next_button.visible = false;
if self.pepper_box.checked || self.netscape_box.checked || self.activex_box.checked {
self.complete_label.text = COMPLETE_INSTALL_TEXT.to_string();
} else {
self.complete_label.text = COMPLETE_UNINSTALL_TEXT.to_string();
}
}
fn open_failure(&mut self) {
self.panel = Panel::Failure;
self.prev_button.text = "QUIT".into();
self.prev_button.enabled = true;
self.next_button.text = "RETRY".into();
self.next_button.visible = true;
}
// ---- Drawing helpers ----
fn draw_disclaimer(&self, r: &mut Renderer) {
self.disclaimer_box
.draw(r, &self.checkbox_on, &self.checkbox_off);
self.disclaimer_label.draw(r, &self.fonts);
}
fn draw_choice(&self, r: &mut Renderer) {
self.browser_ask_label.draw(r, &self.fonts);
self.pepper_box
.draw(r, &self.checkbox_on, &self.checkbox_off);
self.pepper_label.draw(r, &self.fonts);
self.netscape_box
.draw(r, &self.checkbox_on, &self.checkbox_off);
self.netscape_label.draw(r, &self.fonts);
self.activex_box
.draw(r, &self.checkbox_on, &self.checkbox_off);
self.activex_label.draw(r, &self.fonts);
}
fn draw_player_choice(&self, r: &mut Renderer) {
self.player_ask_label.draw(r, &self.fonts);
self.player_box
.draw(r, &self.checkbox_on, &self.checkbox_off);
self.player_label.draw(r, &self.fonts);
self.player_desktop_box
.draw(r, &self.checkbox_on, &self.checkbox_off);
self.player_desktop_label.draw(r, &self.fonts);
self.player_start_menu_box
.draw(r, &self.checkbox_on, &self.checkbox_off);
self.player_start_menu_label.draw(r, &self.fonts);
}
fn draw_debug_choice(&mut self, r: &mut Renderer) {
self.debug_ask_label.draw(r, &self.fonts);
self.debug_button.draw(r, &self.fonts);
}
fn draw_before_install(&self, r: &mut Renderer) {
self.before_install_label.draw(r, &self.fonts);
}
fn draw_install(&self, r: &mut Renderer) {
self.install_header_label.draw(r, &self.fonts);
self.progress_label.draw(r, &self.fonts);
self.progress_bar.draw(r);
}
fn draw_complete(&self, r: &mut Renderer) {
self.complete_label.draw(r, &self.fonts);
}
fn draw_failure(&self, r: &mut Renderer) {
self.failure_text_label.draw(r, &self.fonts);
// Draw error detail as clipped text.
let detail_y = PANEL_Y + 44;
let detail_text = if self.failure_detail.len() > 300 {
&self.failure_detail[..300]
} else {
&self.failure_detail
};
self.fonts.draw_text_multiline(
r,
PANEL_X + 4,
detail_y,
detail_text,
11.0,
FG_COLOR,
1.0,
);
self.copy_error_button.draw(r, &self.fonts);
}
}
/// Progress callback that writes to the shared state from the background thread.
struct ThreadProgressCallback {
state: Arc<Mutex<ProgressState>>,
}
impl ProgressCallback for ThreadProgressCallback {
fn update_progress_label(&self, text: &str, tick: bool) {
let mut state = self.state.lock().unwrap();
state.label = text.to_string();
if tick {
state.value += 1;
}
}
fn tick_progress(&self) {
let mut state = self.state.lock().unwrap();
state.value += 1;
}
}
fn join_with_and(items: &[&str]) -> String {
match items.len() {
0 => String::new(),
1 => items[0].to_string(),
2 => format!("{} and {}", items[0], items[1]),
_ => {
let (last, rest) = items.split_last().unwrap();
format!("{} and {}", rest.join(", "), last)
}
}
}
/// Try to load a resource image from the original C# project's asset folder.
fn load_resource_image(name: &str) -> RgbaImage {
let bytes: &[u8] = match name {
"flashLogo.png" => include_bytes!("../../../resources/flashLogo.png"),
"checkboxOn.png" => include_bytes!("../../../resources/checkboxOn.png"),
"checkboxOff.png" => include_bytes!("../../../resources/checkboxOff.png"),
_ => return RgbaImage::empty(),
};
RgbaImage::from_png_bytes(bytes)
}

@ -0,0 +1,356 @@
use crate::install_flags::{self, InstallFlags};
use clean_flash_common::{
process_utils, registry, resources, system_info, InstallError, ProgressCallback,
};
use std::env;
use std::fs;
use std::io::{self, Cursor};
use std::path::{Path, PathBuf};
/// Metadata for a single installable component.
struct InstallEntry {
install_text: &'static str,
required_flags: u32,
target_directory: PathBuf,
registry_instructions: Option<&'static str>,
}
/// Register an ActiveX OCX via regsvr32 (unregister first, then register).
pub fn register_activex(filename: &str) -> Result<(), InstallError> {
let path = Path::new(filename);
let dir = path.parent().unwrap_or(Path::new("."));
let file_name = path
.file_name()
.unwrap_or_default()
.to_string_lossy()
.to_string();
let _ = env::set_current_dir(dir);
let process = process_utils::run_process("regsvr32.exe", &["/s", "/u", &file_name]);
if !process.is_successful() {
return Err(InstallError::new(format!(
"Failed to unregister ActiveX plugin: error code {}\n\n{}",
process.exit_code, process.output
)));
}
let process = process_utils::run_process("regsvr32.exe", &["/s", &file_name]);
if !process.is_successful() {
return Err(InstallError::new(format!(
"Failed to register ActiveX plugin: error code {}\n\n{}",
process.exit_code, process.output
)));
}
Ok(())
}
/// Create a Windows shortcut (.lnk) using a PowerShell one-liner.
fn create_shortcut(
folder: &Path,
executable: &Path,
name: &str,
description: &str,
) -> Result<(), InstallError> {
let lnk_path = folder.join(format!("{}.lnk", name));
let exe_str = executable.to_string_lossy();
let lnk_str = lnk_path.to_string_lossy();
// Use PowerShell to create the shortcut via WScript.Shell COM.
let script = format!(
"$ws = New-Object -ComObject WScript.Shell; \
$s = $ws.CreateShortcut('{}'); \
$s.TargetPath = '{}'; \
$s.Description = '{}'; \
$s.IconLocation = '{}'; \
$s.Save()",
lnk_str, exe_str, description, exe_str
);
let result = process_utils::run_process("powershell.exe", &["-NoProfile", "-Command", &script]);
if !result.is_successful() {
return Err(InstallError::new(format!(
"Failed to create shortcut: {}",
result.output
)));
}
Ok(())
}
/// Extract the embedded 7z archive and install files to the correct locations.
fn install_from_archive(
archive_bytes: &[u8],
form: &dyn ProgressCallback,
flags: &mut InstallFlags,
) -> Result<(), InstallError> {
let si = system_info::SystemInfo::new();
let flash32_path = si.flash32_path.clone();
let flash64_path = si.flash64_path.clone();
let system32_path = si.system32_path.clone();
let flash_program32_path = si.flash_program32_path.clone();
let mut registry_to_apply: Vec<&str> = vec![resources::INSTALL_GENERAL];
if si.is_64bit {
flags.set_flag(install_flags::X64);
registry_to_apply.push(resources::INSTALL_GENERAL_64);
}
let entries: Vec<(&str, InstallEntry)> = vec![
(
"controlpanel",
InstallEntry {
install_text: "Installing Flash Player utilities...",
required_flags: install_flags::NONE,
target_directory: system32_path.clone(),
registry_instructions: None,
},
),
(
"uninstaller",
InstallEntry {
install_text: "Extracting uninstaller...",
required_flags: install_flags::NONE,
target_directory: flash_program32_path.clone(),
registry_instructions: None,
},
),
(
"standalone",
InstallEntry {
install_text: "Installing 32-bit Standalone Flash Player...",
required_flags: install_flags::PLAYER,
target_directory: flash_program32_path.clone(),
registry_instructions: None,
},
),
(
"ocx32",
InstallEntry {
install_text: "Installing 32-bit Flash Player for Internet Explorer...",
required_flags: install_flags::ACTIVEX,
target_directory: flash32_path.clone(),
registry_instructions: None,
},
),
(
"np32",
InstallEntry {
install_text: "Installing 32-bit Flash Player for Firefox...",
required_flags: install_flags::NETSCAPE,
target_directory: flash32_path.clone(),
registry_instructions: Some(resources::INSTALL_NP),
},
),
(
"pp32",
InstallEntry {
install_text: "Installing 32-bit Flash Player for Chrome...",
required_flags: install_flags::PEPPER,
target_directory: flash32_path.clone(),
registry_instructions: Some(resources::INSTALL_PP),
},
),
(
"ocx64",
InstallEntry {
install_text: "Installing 64-bit Flash Player for Internet Explorer...",
required_flags: install_flags::ACTIVEX | install_flags::X64,
target_directory: flash64_path.clone(),
registry_instructions: None,
},
),
(
"np64",
InstallEntry {
install_text: "Installing 64-bit Flash Player for Firefox...",
required_flags: install_flags::NETSCAPE | install_flags::X64,
target_directory: flash64_path.clone(),
registry_instructions: Some(resources::INSTALL_NP_64),
},
),
(
"pp64",
InstallEntry {
install_text: "Installing 64-bit Flash Player for Chrome...",
required_flags: install_flags::PEPPER | install_flags::X64,
target_directory: flash64_path.clone(),
registry_instructions: Some(resources::INSTALL_PP_64),
},
),
];
let legacy = si.is_legacy_windows();
// Extract archive using sevenz-rust2.
sevenz_rust2::decompress_with_extract_fn(
Cursor::new(archive_bytes),
".",
|entry, reader, _dest| {
let entry_name = entry.name().to_string();
let parts: Vec<&str> = entry_name.split('/').collect();
if parts.is_empty() {
return Ok(true);
}
let filename = parts[0];
let install_key = filename.split('-').next().unwrap_or(filename);
// Find the matching entry.
let Some((_key, install_entry)) = entries.iter().find(|(k, _)| *k == install_key)
else {
io::copy(reader, &mut io::sink()).map_err(sevenz_rust2::Error::from)?;
return Ok(true);
};
// Check required flags.
if install_entry.required_flags != install_flags::NONE
&& !flags.is_set(install_entry.required_flags)
{
io::copy(reader, &mut io::sink()).map_err(sevenz_rust2::Error::from)?;
return Ok(true);
}
// Check debug flag match.
if install_entry.required_flags != install_flags::NONE {
let is_debug_file = filename.contains("-debug");
if flags.is_set(install_flags::DEBUG) != is_debug_file {
io::copy(reader, &mut io::sink()).map_err(sevenz_rust2::Error::from)?;
return Ok(true);
}
}
// Check legacy flag for ActiveX entries.
if (install_entry.required_flags & install_flags::ACTIVEX) != 0 {
let is_legacy_file = filename.contains("-legacy");
if legacy != is_legacy_file {
io::copy(reader, &mut io::sink()).map_err(sevenz_rust2::Error::from)?;
return Ok(true);
}
}
form.update_progress_label(install_entry.install_text, true);
// Ensure target directory exists.
let _ = fs::create_dir_all(&install_entry.target_directory);
// Extract file: use just the file name (strip any path prefix).
let out_name = parts.last().unwrap_or(&filename);
let out_path = install_entry.target_directory.join(out_name);
let mut buf = Vec::new();
reader.read_to_end(&mut buf).map_err(sevenz_rust2::Error::from)?;
fs::write(&out_path, &buf).map_err(sevenz_rust2::Error::from)?;
Ok(true)
},
)
.map_err(|e| InstallError::new(format!("Failed to extract archive: {}", e)))?;
// Create Player shortcuts.
if flags.is_set(install_flags::PLAYER) {
let is_debug = flags.is_set(install_flags::DEBUG);
let name = if is_debug {
"Flash Player (Debug)"
} else {
"Flash Player"
};
let description = format!(
"Standalone Flash Player {}{}",
clean_flash_common::update_checker::FLASH_VERSION,
if is_debug { " (Debug)" } else { "" }
);
let exe_name = if is_debug {
"flashplayer_sa_debug.exe"
} else {
"flashplayer_sa.exe"
};
let executable = flash_program32_path.join(exe_name);
if flags.is_set(install_flags::PLAYER_START_MENU) {
if let Some(start_menu) = get_start_menu() {
let _ = create_shortcut(&start_menu, &executable, name, &description);
}
}
if flags.is_set(install_flags::PLAYER_DESKTOP) {
if let Some(desktop) = get_desktop() {
let _ = create_shortcut(&desktop, &executable, name, &description);
}
}
}
// Collect registry entries for enabled components.
for (_key, entry) in &entries {
if flags.is_set(entry.required_flags) {
if let Some(reg) = entry.registry_instructions {
registry_to_apply.push(reg);
}
}
}
form.update_progress_label("Applying registry changes...", true);
let refs: Vec<&str> = registry_to_apply.iter().copied().collect();
registry::apply_registry(&refs)?;
// Register ActiveX OCX files.
if flags.is_set(install_flags::ACTIVEX) {
form.update_progress_label(
"Activating 32-bit Flash Player for Internet Explorer...",
true,
);
let ocx32 = flash32_path.join(format!("Flash32_{}.ocx", si.version_path));
register_activex(&ocx32.to_string_lossy())?;
if si.is_64bit {
form.update_progress_label(
"Activating 64-bit Flash Player for Internet Explorer...",
true,
);
let ocx64 = flash64_path.join(format!("Flash64_{}.ocx", si.version_path));
register_activex(&ocx64.to_string_lossy())?;
}
}
Ok(())
}
/// Main install entry point.
pub fn install(
form: &dyn ProgressCallback,
flags: &mut InstallFlags,
) -> Result<(), InstallError> {
if flags.is_none_set() {
return Ok(());
}
// The 7z archive is embedded in the binary via include_bytes!.
// For the port, we expect it at a known resource path; if not present,
// this is a no-op placeholder.
let archive_bytes: &[u8] = include_bytes!("../cleanflash.7z");
if archive_bytes.is_empty() {
// Nothing to extract; still apply the rest of the steps.
return Ok(());
}
install_from_archive(archive_bytes, form, flags)
}
fn get_start_menu() -> Option<PathBuf> {
env::var("APPDATA")
.ok()
.map(|p| {
PathBuf::from(p)
.join("Microsoft")
.join("Windows")
.join("Start Menu")
})
}
fn get_desktop() -> Option<PathBuf> {
env::var("USERPROFILE")
.ok()
.map(|p| PathBuf::from(p).join("Desktop"))
}

@ -0,0 +1,49 @@
#![windows_subsystem = "windows"]
mod install_flags;
mod install_form;
mod installer;
use install_form::{InstallForm, HEIGHT, WIDTH};
use clean_flash_ui::renderer::Renderer;
use minifb::{Key, MouseButton, MouseMode, Window, WindowOptions};
fn main() {
let title = format!(
"Clean Flash Player {} Installer",
clean_flash_common::update_checker::FLASH_VERSION
);
let mut window = Window::new(
&title,
WIDTH,
HEIGHT,
WindowOptions {
resize: false,
..WindowOptions::default()
},
)
.expect("Failed to create window");
// Set window icon from the resource embedded by build.rs.
clean_flash_ui::set_window_icon(&window);
// Cap at ~60 fps.
window.set_target_fps(60);
let mut renderer = Renderer::new(WIDTH, HEIGHT);
let mut form = InstallForm::new();
while window.is_open() && !window.is_key_down(Key::Escape) {
let (mx, my) = window
.get_mouse_pos(MouseMode::Clamp)
.unwrap_or((0.0, 0.0));
let mouse_down = window.get_mouse_down(MouseButton::Left);
form.update_and_draw(&mut renderer, mx as i32, my as i32, mouse_down);
window
.update_with_buffer(&renderer.buffer, WIDTH, HEIGHT)
.expect("Failed to update window buffer");
}
}

@ -0,0 +1,10 @@
[package]
name = "clean_flash_ui"
version = "34.0.0"
edition = "2021"
[dependencies]
ab_glyph = { workspace = true }
image = { workspace = true }
minifb = { workspace = true }
windows-sys = { workspace = true }

@ -0,0 +1,106 @@
use ab_glyph::{point, Font, FontRef, ScaleFont};
use crate::renderer::Renderer;
/// Manages loaded fonts and provides text measurement / rendering.
pub struct FontManager {
regular: FontRef<'static>,
}
impl FontManager {
/// Create a `FontManager` using the bundled Liberation Sans font.
pub fn new() -> Self {
let regular = FontRef::try_from_slice(include_bytes!(
"../../../resources/liberation-sans.regular.ttf"
))
.expect("Failed to parse bundled Liberation Sans font");
Self { regular }
}
/// Measure the width (in pixels) of `text` at the given `size` (in px).
pub fn measure_text(&self, text: &str, size: f32) -> (f32, f32) {
let scaled = self.regular.as_scaled(size);
let mut width: f32 = 0.0;
let height = scaled.height();
let mut last_glyph_id = None;
for ch in text.chars() {
let glyph_id = scaled.glyph_id(ch);
if let Some(prev) = last_glyph_id {
width += scaled.kern(prev, glyph_id);
}
width += scaled.h_advance(glyph_id);
last_glyph_id = Some(glyph_id);
}
(width, height)
}
/// Draw `text` onto the renderer at (x, y) with the given pixel size and colour.
pub fn draw_text(
&self,
renderer: &mut Renderer,
x: i32,
y: i32,
text: &str,
size: f32,
color: u32,
) {
let scaled = self.regular.as_scaled(size);
let ascent = scaled.ascent();
let mut cursor_x: f32 = 0.0;
let mut last_glyph_id = None;
for ch in text.chars() {
let glyph_id = scaled.glyph_id(ch);
if let Some(prev) = last_glyph_id {
cursor_x += scaled.kern(prev, glyph_id);
}
let glyph = glyph_id.with_scale_and_position(
size,
point(x as f32 + cursor_x, y as f32 + ascent),
);
if let Some(outlined) = self.regular.outline_glyph(glyph) {
let bounds = outlined.px_bounds();
outlined.draw(|gx, gy, coverage| {
let px = bounds.min.x as i32 + gx as i32;
let py = bounds.min.y as i32 + gy as i32;
let alpha = (coverage * 255.0) as u8;
if alpha > 0 {
renderer.blend_pixel(px, py, color, alpha);
}
});
}
cursor_x += scaled.h_advance(glyph_id);
last_glyph_id = Some(glyph_id);
}
}
/// Draw multiline text, splitting on '\n'. Returns total height drawn.
pub fn draw_text_multiline(
&self,
renderer: &mut Renderer,
x: i32,
y: i32,
text: &str,
size: f32,
color: u32,
line_spacing: f32,
) -> f32 {
let scaled = self.regular.as_scaled(size);
let line_height = scaled.height() + line_spacing;
let mut cy = y as f32;
for line in text.split('\n') {
self.draw_text(renderer, x, cy as i32, line, size, color);
cy += line_height;
}
cy - y as f32
}
}

@ -0,0 +1,28 @@
pub mod font;
pub mod renderer;
pub mod widgets;
pub use font::FontManager;
pub use renderer::Renderer;
/// Set the window icon from the icon resource already embedded in the binary
/// by `winresource` (resource ID 1). No-op on non-Windows platforms.
pub fn set_window_icon(window: &minifb::Window) {
#[cfg(target_os = "windows")]
unsafe {
use windows_sys::Win32::System::LibraryLoader::GetModuleHandleW;
use windows_sys::Win32::UI::WindowsAndMessaging::{
LoadImageW, SendMessageW, ICON_BIG, ICON_SMALL, IMAGE_ICON, LR_DEFAULTSIZE,
WM_SETICON,
};
let hwnd = window.get_window_handle(); // *mut c_void
let hmodule = GetModuleHandleW(std::ptr::null());
// MAKEINTRESOURCEW(1): load the icon embedded by winresource as resource ID 1.
let icon = LoadImageW(hmodule, 1 as *const u16, IMAGE_ICON, 0, 0, LR_DEFAULTSIZE);
if !icon.is_null() {
SendMessageW(hwnd, WM_SETICON, ICON_BIG as usize, icon as isize);
SendMessageW(hwnd, WM_SETICON, ICON_SMALL as usize, icon as isize);
}
}
}

@ -0,0 +1,183 @@
/// Software renderer operating on a `Vec<u32>` pixel buffer (0xAA_RR_GG_BB).
/// All drawing is done in-memory; the buffer is presented via minifb.
pub struct Renderer {
pub width: usize,
pub height: usize,
pub buffer: Vec<u32>,
}
impl Renderer {
pub fn new(width: usize, height: usize) -> Self {
Self {
width,
height,
buffer: vec![0; width * height],
}
}
/// Pack r, g, b into the minifb pixel format (0x00RRGGBB).
#[inline]
pub const fn rgb(r: u8, g: u8, b: u8) -> u32 {
((r as u32) << 16) | ((g as u32) << 8) | (b as u32)
}
/// Clear the entire buffer to a single colour.
pub fn clear(&mut self, color: u32) {
self.buffer.fill(color);
}
/// Set a single pixel (bounds-checked).
#[inline]
pub fn set_pixel(&mut self, x: i32, y: i32, color: u32) {
if x >= 0 && y >= 0 && (x as usize) < self.width && (y as usize) < self.height {
self.buffer[y as usize * self.width + x as usize] = color;
}
}
/// Alpha-blend a single pixel. `alpha` is 0..=255.
#[inline]
pub fn blend_pixel(&mut self, x: i32, y: i32, color: u32, alpha: u8) {
if x < 0 || y < 0 || (x as usize) >= self.width || (y as usize) >= self.height {
return;
}
let idx = y as usize * self.width + x as usize;
let dst = self.buffer[idx];
self.buffer[idx] = alpha_blend(dst, color, alpha);
}
/// Fill a solid rectangle.
pub fn fill_rect(&mut self, x: i32, y: i32, w: i32, h: i32, color: u32) {
let x0 = x.max(0) as usize;
let y0 = y.max(0) as usize;
let x1 = ((x + w) as usize).min(self.width);
let y1 = ((y + h) as usize).min(self.height);
for row in y0..y1 {
let start = row * self.width + x0;
let end = row * self.width + x1;
self.buffer[start..end].fill(color);
}
}
/// Draw a 1px rectangle outline.
pub fn draw_rect(&mut self, x: i32, y: i32, w: i32, h: i32, color: u32) {
// Top / bottom
for dx in 0..w {
self.set_pixel(x + dx, y, color);
self.set_pixel(x + dx, y + h - 1, color);
}
// Left / right
for dy in 0..h {
self.set_pixel(x, y + dy, color);
self.set_pixel(x + w - 1, y + dy, color);
}
}
/// Fill a rectangle with a vertical linear gradient from `color1` (top) to `color2` (bottom).
pub fn fill_gradient_v(&mut self, x: i32, y: i32, w: i32, h: i32, color1: u32, color2: u32) {
if h <= 0 {
return;
}
let (r1, g1, b1) = unpack(color1);
let (r2, g2, b2) = unpack(color2);
for dy in 0..h {
let t = dy as f32 / (h - 1).max(1) as f32;
let r = lerp_u8(r1, r2, t);
let g = lerp_u8(g1, g2, t);
let b = lerp_u8(b1, b2, t);
let c = Self::rgb(r, g, b);
self.fill_rect(x, y + dy, w, 1, c);
}
}
/// Fill a rectangle with a horizontal linear gradient.
pub fn fill_gradient_h(&mut self, x: i32, y: i32, w: i32, h: i32, color1: u32, color2: u32) {
if w <= 0 {
return;
}
let (r1, g1, b1) = unpack(color1);
let (r2, g2, b2) = unpack(color2);
for dx in 0..w {
let t = dx as f32 / (w - 1).max(1) as f32;
let r = lerp_u8(r1, r2, t);
let g = lerp_u8(g1, g2, t);
let b = lerp_u8(b1, b2, t);
let c = Self::rgb(r, g, b);
self.fill_rect(x + dx, y, 1, h, c);
}
}
/// Draw an RGBA image onto the framebuffer at (x, y).
pub fn draw_image(&mut self, x: i32, y: i32, img: &RgbaImage) {
for iy in 0..img.height as i32 {
for ix in 0..img.width as i32 {
let idx = (iy as usize * img.width + ix as usize) * 4;
let r = img.data[idx];
let g = img.data[idx + 1];
let b = img.data[idx + 2];
let a = img.data[idx + 3];
if a == 255 {
self.set_pixel(x + ix, y + iy, Self::rgb(r, g, b));
} else if a > 0 {
self.blend_pixel(x + ix, y + iy, Self::rgb(r, g, b), a);
}
}
}
}
}
/// Simple RGBA image stored as raw bytes.
pub struct RgbaImage {
pub width: usize,
pub height: usize,
pub data: Vec<u8>, // RGBA, row-major
}
impl RgbaImage {
/// Load a PNG from embedded bytes.
pub fn from_png_bytes(bytes: &[u8]) -> Self {
let img = image::load_from_memory_with_format(bytes, image::ImageFormat::Png)
.expect("Failed to decode PNG")
.to_rgba8();
Self {
width: img.width() as usize,
height: img.height() as usize,
data: img.into_raw(),
}
}
/// Create an empty (transparent) image.
pub fn empty() -> Self {
Self {
width: 0,
height: 0,
data: Vec::new(),
}
}
}
// ---- helpers ----
#[inline]
fn unpack(c: u32) -> (u8, u8, u8) {
(((c >> 16) & 0xFF) as u8, ((c >> 8) & 0xFF) as u8, (c & 0xFF) as u8)
}
#[inline]
fn lerp_u8(a: u8, b: u8, t: f32) -> u8 {
(a as f32 + (b as f32 - a as f32) * t).round() as u8
}
#[inline]
fn alpha_blend(dst: u32, src: u32, alpha: u8) -> u32 {
let (sr, sg, sb) = unpack(src);
let (dr, dg, db) = unpack(dst);
let a = alpha as u16;
let inv = 255 - a;
let r = ((sr as u16 * a + dr as u16 * inv) / 255) as u8;
let g = ((sg as u16 * a + dg as u16 * inv) / 255) as u8;
let b = ((sb as u16 * a + db as u16 * inv) / 255) as u8;
Renderer::rgb(r, g, b)
}

@ -0,0 +1,97 @@
use super::Rect;
use crate::font::FontManager;
use crate::renderer::Renderer;
/// A gradient button matching the C# GradientButton control.
pub struct GradientButton {
pub rect: Rect,
pub text: String,
pub color1: u32,
pub color2: u32,
pub back_color: u32,
pub fore_color: u32,
pub hover_alpha: f64,
pub disable_alpha: f64,
pub enabled: bool,
pub visible: bool,
pub hovered: bool,
pub pressed: bool,
}
impl GradientButton {
pub fn new(x: i32, y: i32, w: i32, h: i32, text: &str) -> Self {
Self {
rect: Rect::new(x, y, w, h),
text: text.to_string(),
color1: Renderer::rgb(118, 118, 118),
color2: Renderer::rgb(81, 81, 81),
back_color: Renderer::rgb(0, 0, 0),
fore_color: Renderer::rgb(227, 227, 227),
hover_alpha: 0.875,
disable_alpha: 0.644,
enabled: true,
visible: true,
hovered: false,
pressed: false,
}
}
/// Update hover / pressed state from mouse position and button state.
pub fn update(&mut self, mx: i32, my: i32, mouse_down: bool) {
if !self.visible || !self.enabled {
self.hovered = false;
self.pressed = false;
return;
}
self.hovered = self.rect.contains(mx, my);
self.pressed = self.hovered && mouse_down;
}
/// Returns true if mouse was just released inside this button.
pub fn clicked(&self, mx: i32, my: i32, mouse_released: bool) -> bool {
self.visible && self.enabled && self.rect.contains(mx, my) && mouse_released
}
pub fn draw(&self, renderer: &mut Renderer, fonts: &FontManager) {
if !self.visible {
return;
}
let (mut c1, mut c2, mut bg, mut fg) = (self.color1, self.color2, self.back_color, self.fore_color);
if !self.enabled {
c1 = dim_color(c1, self.disable_alpha);
c2 = dim_color(c2, self.disable_alpha);
bg = dim_color(bg, self.disable_alpha);
fg = dim_color(fg, self.disable_alpha);
} else if self.pressed {
c1 = dim_color(c1, self.hover_alpha);
c2 = dim_color(c2, self.hover_alpha);
} else if !self.hovered {
c1 = dim_color(c1, self.hover_alpha);
c2 = dim_color(c2, self.hover_alpha);
}
let r = self.rect;
renderer.fill_gradient_v(r.x, r.y, r.w, r.h, c1, c2);
renderer.draw_rect(r.x, r.y, r.w, r.h, bg);
// Measure text to centre it.
let font_size = 13.0;
let (tw, th) = fonts.measure_text(&self.text, font_size);
let tx = r.x + ((r.w as f32 - tw) / 2.0) as i32;
let ty = r.y + ((r.h as f32 - th) / 2.0) as i32;
// Shadow.
fonts.draw_text(renderer, tx + 1, ty + 1, &self.text, font_size, bg);
// Foreground.
fonts.draw_text(renderer, tx, ty, &self.text, font_size, fg);
}
}
fn dim_color(c: u32, alpha: f64) -> u32 {
let r = (((c >> 16) & 0xFF) as f64 * alpha) as u8;
let g = (((c >> 8) & 0xFF) as f64 * alpha) as u8;
let b = ((c & 0xFF) as f64 * alpha) as u8;
Renderer::rgb(r, g, b)
}

@ -0,0 +1,91 @@
use super::Rect;
use crate::renderer::{Renderer, RgbaImage};
/// An image-based checkbox matching the C# ImageCheckBox control.
pub struct ImageCheckBox {
pub rect: Rect,
pub checked: bool,
pub enabled: bool,
pub visible: bool,
}
impl ImageCheckBox {
pub fn new(x: i32, y: i32) -> Self {
Self {
rect: Rect::new(x, y, 21, 21),
checked: true,
enabled: true,
visible: true,
}
}
pub fn toggle_if_clicked(&mut self, mx: i32, my: i32, mouse_released: bool) -> bool {
if self.visible && self.enabled && self.rect.contains(mx, my) && mouse_released {
self.checked = !self.checked;
true
} else {
false
}
}
pub fn draw(
&self,
renderer: &mut Renderer,
checked_img: &RgbaImage,
unchecked_img: &RgbaImage,
) {
if !self.visible {
return;
}
let img = if self.checked {
checked_img
} else {
unchecked_img
};
if img.width > 0 && img.height > 0 {
renderer.draw_image(self.rect.x, self.rect.y, img);
} else {
// Fallback: draw a simple square.
let bg = if self.checked {
Renderer::rgb(97, 147, 232)
} else {
Renderer::rgb(80, 80, 80)
};
renderer.fill_rect(self.rect.x, self.rect.y, self.rect.w, self.rect.h, bg);
renderer.draw_rect(
self.rect.x,
self.rect.y,
self.rect.w,
self.rect.h,
Renderer::rgb(160, 160, 160),
);
if self.checked {
// Draw a simple checkmark.
let cx = self.rect.x + 5;
let cy = self.rect.y + 10;
for i in 0..4 {
renderer.set_pixel(cx + i, cy + i, Renderer::rgb(255, 255, 255));
renderer.set_pixel(cx + i, cy + i + 1, Renderer::rgb(255, 255, 255));
}
for i in 0..8 {
renderer.set_pixel(cx + 3 + i, cy + 3 - i, Renderer::rgb(255, 255, 255));
renderer.set_pixel(cx + 3 + i, cy + 4 - i, Renderer::rgb(255, 255, 255));
}
}
}
if !self.enabled {
// Dim overlay.
for dy in 0..self.rect.h {
for dx in 0..self.rect.w {
renderer.blend_pixel(
self.rect.x + dx,
self.rect.y + dy,
Renderer::rgb(50, 51, 51),
100,
);
}
}
}
}
}

@ -0,0 +1,51 @@
use super::Rect;
use crate::font::FontManager;
use crate::renderer::Renderer;
/// Simple static label for drawing text.
pub struct Label {
pub rect: Rect,
pub text: String,
pub color: u32,
pub font_size: f32,
pub visible: bool,
}
impl Label {
pub fn new(x: i32, y: i32, text: &str, font_size: f32) -> Self {
Self {
rect: Rect::new(x, y, 0, 0),
text: text.to_string(),
color: Renderer::rgb(245, 245, 245),
font_size,
visible: true,
}
}
pub fn draw(&self, renderer: &mut Renderer, fonts: &FontManager) {
if !self.visible || self.text.is_empty() {
return;
}
fonts.draw_text_multiline(
renderer,
self.rect.x,
self.rect.y,
&self.text,
self.font_size,
self.color,
2.0,
);
}
/// Check if a click at (mx, my) is within a rough bounding box of the label text.
pub fn clicked(&self, mx: i32, my: i32, mouse_released: bool, fonts: &FontManager) -> bool {
if !self.visible || !mouse_released {
return false;
}
let (tw, _th) = fonts.measure_text(&self.text, self.font_size);
let lines = self.text.lines().count().max(1);
let approx_h = (self.font_size * lines as f32 + 2.0 * lines as f32) as i32;
let r = Rect::new(self.rect.x, self.rect.y, tw as i32 + 5, approx_h);
r.contains(mx, my)
}
}

@ -0,0 +1,23 @@
pub mod button;
pub mod checkbox;
pub mod label;
pub mod progress_bar;
/// A rectangle region on screen.
#[derive(Clone, Copy, Debug, Default)]
pub struct Rect {
pub x: i32,
pub y: i32,
pub w: i32,
pub h: i32,
}
impl Rect {
pub const fn new(x: i32, y: i32, w: i32, h: i32) -> Self {
Self { x, y, w, h }
}
pub fn contains(&self, px: i32, py: i32) -> bool {
px >= self.x && px < self.x + self.w && py >= self.y && py < self.y + self.h
}
}

@ -0,0 +1,68 @@
use super::Rect;
use crate::renderer::Renderer;
/// A smooth gradient progress bar matching the C# SmoothProgressBar control.
pub struct ProgressBar {
pub rect: Rect,
pub minimum: i32,
pub maximum: i32,
pub value: i32,
pub color1: u32,
pub color2: u32,
pub visible: bool,
}
impl ProgressBar {
pub fn new(x: i32, y: i32, w: i32, h: i32) -> Self {
Self {
rect: Rect::new(x, y, w, h),
minimum: 0,
maximum: 100,
value: 0,
color1: Renderer::rgb(97, 147, 232),
color2: Renderer::rgb(28, 99, 232),
visible: true,
}
}
pub fn draw(&self, renderer: &mut Renderer) {
if !self.visible {
return;
}
let range = (self.maximum - self.minimum).max(1) as f32;
let percent = (self.value - self.minimum) as f32 / range;
let fill_w = (self.rect.w as f32 * percent) as i32;
if fill_w > 0 {
renderer.fill_gradient_h(
self.rect.x,
self.rect.y,
fill_w,
self.rect.h,
self.color1,
self.color2,
);
}
// 3-D border.
let r = self.rect;
let dark = Renderer::rgb(105, 105, 105);
let light = Renderer::rgb(255, 255, 255);
// Top
for dx in 0..r.w {
renderer.set_pixel(r.x + dx, r.y, dark);
}
// Left
for dy in 0..r.h {
renderer.set_pixel(r.x, r.y + dy, dark);
}
// Bottom
for dx in 0..r.w {
renderer.set_pixel(r.x + dx, r.y + r.h - 1, light);
}
// Right
for dy in 0..r.h {
renderer.set_pixel(r.x + r.w - 1, r.y + dy, light);
}
}
}

@ -0,0 +1,14 @@
[package]
name = "clean_flash_uninstaller"
version = "34.0.0"
edition = "2021"
authors = ["FlashPatch Team"]
[dependencies]
clean_flash_common = { path = "../clean_flash_common" }
clean_flash_ui = { path = "../clean_flash_ui" }
minifb = { workspace = true }
windows-sys = { workspace = true }
[build-dependencies]
winresource = "0.1"

@ -0,0 +1,35 @@
fn main() {
if cfg!(target_os = "windows") {
let mut res = winresource::WindowsResource::new();
if std::path::Path::new("../../resources/icon.ico").exists() {
res.set_icon("../../resources/icon.ico");
}
res.set("ProductName", "Clean Flash Player Uninstaller");
res.set("FileDescription", "Clean Flash Player Uninstaller");
res.set("ProductVersion", "34.0.0.330");
res.set("CompanyName", "FlashPatch Team");
res.set("LegalCopyright", "FlashPatch Team");
res.set_manifest(r#"<?xml version="1.0" encoding="utf-8"?>
<assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1">
<assemblyIdentity version="1.0.0.0" name="CleanFlashUninstaller.app"/>
<trustInfo xmlns="urn:schemas-microsoft-com:asm.v2">
<security>
<requestedPrivileges xmlns="urn:schemas-microsoft-com:asm.v3">
<requestedExecutionLevel level="requireAdministrator" uiAccess="false" />
</requestedPrivileges>
</security>
</trustInfo>
<compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1">
<application>
<supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}"/>
<supportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}"/>
<supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}"/>
<supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}"/>
<supportedOS Id="{e2011457-1546-43c5-a5fe-008deee3d3f0}"/>
</application>
</compatibility>
</assembly>
"#);
let _ = res.compile();
}
}

@ -0,0 +1,46 @@
#![windows_subsystem = "windows"]
mod uninstall_form;
use clean_flash_ui::renderer::Renderer;
use minifb::{Key, MouseButton, MouseMode, Window, WindowOptions};
use uninstall_form::{UninstallForm, HEIGHT, WIDTH};
fn main() {
let title = format!(
"Clean Flash Player {} Uninstaller",
clean_flash_common::update_checker::FLASH_VERSION
);
let mut window = Window::new(
&title,
WIDTH,
HEIGHT,
WindowOptions {
resize: false,
..WindowOptions::default()
},
)
.expect("Failed to create window");
// Set window icon from the resource embedded by build.rs.
clean_flash_ui::set_window_icon(&window);
window.set_target_fps(60);
let mut renderer = Renderer::new(WIDTH, HEIGHT);
let mut form = UninstallForm::new();
while window.is_open() && !window.is_key_down(Key::Escape) {
let (mx, my) = window
.get_mouse_pos(MouseMode::Clamp)
.unwrap_or((0.0, 0.0));
let mouse_down = window.get_mouse_down(MouseButton::Left);
form.update_and_draw(&mut renderer, mx as i32, my as i32, mouse_down);
window
.update_with_buffer(&renderer.buffer, WIDTH, HEIGHT)
.expect("Failed to update window buffer");
}
}

@ -0,0 +1,283 @@
use clean_flash_common::{redirection, uninstaller, update_checker, ProgressCallback};
use clean_flash_ui::font::FontManager;
use clean_flash_ui::renderer::{Renderer, RgbaImage};
use clean_flash_ui::widgets::button::GradientButton;
use clean_flash_ui::widgets::label::Label;
use clean_flash_ui::widgets::progress_bar::ProgressBar;
use std::sync::{Arc, Mutex};
pub const WIDTH: usize = 712;
pub const HEIGHT: usize = 329;
const BG_COLOR: u32 = 0x00323333; // RGB(50, 51, 51)
const FG_COLOR: u32 = 0x00F5F5F5;
const PANEL_X: i32 = 90;
const PANEL_Y: i32 = 162;
const BEFORE_TEXT: &str = "You are about to uninstall Clean Flash Player.\n\
Please close all browsers, including Google Chrome, Mozilla Firefox and Internet Explorer.\n\n\
The installer will completely remove all versions of Flash Player from this computer,\n\
including Clean Flash Player and older versions of Adobe Flash Player.";
const COMPLETE_TEXT: &str = "\nAll versions of Flash Player have been successfully uninstalled.\n\n\
If you ever change your mind, check out Clean Flash Player's website!";
const UNINSTALL_TICKS: i32 = 9;
#[derive(Clone, Copy, PartialEq, Eq)]
enum Panel {
BeforeInstall,
Install,
Complete,
Failure,
}
struct ProgressState {
label: String,
value: i32,
done: bool,
error: Option<String>,
}
pub struct UninstallForm {
panel: Panel,
title_text: String,
subtitle_text: String,
flash_logo: RgbaImage,
prev_button: GradientButton,
next_button: GradientButton,
// Before uninstall
before_label: Label,
// Install (progress)
progress_header: Label,
progress_label: Label,
progress_bar: ProgressBar,
// Complete
complete_label: Label,
// Failure
failure_text_label: Label,
failure_detail: String,
copy_error_button: GradientButton,
// State
progress_state: Arc<Mutex<ProgressState>>,
fonts: FontManager,
prev_mouse_down: bool,
}
impl UninstallForm {
pub fn new() -> Self {
let version = update_checker::FLASH_VERSION;
let flash_logo = load_resource_image("flashLogo.png");
let fonts = FontManager::new();
Self {
panel: Panel::BeforeInstall,
title_text: "Clean Flash Player".into(),
subtitle_text: format!("built from version {} (China)", version),
flash_logo,
prev_button: GradientButton::new(90, 286, 138, 31, "QUIT"),
next_button: GradientButton::new(497, 286, 138, 31, "UNINSTALL"),
before_label: Label::new(PANEL_X + 3, PANEL_Y + 2, BEFORE_TEXT, 13.0),
progress_header: Label::new(
PANEL_X + 3,
PANEL_Y,
"Uninstallation in progress...",
13.0,
),
progress_label: Label::new(PANEL_X + 46, PANEL_Y + 30, "Preparing...", 13.0),
progress_bar: ProgressBar::new(PANEL_X + 49, PANEL_Y + 58, 451, 23),
complete_label: Label::new(PANEL_X, PANEL_Y, COMPLETE_TEXT, 13.0),
failure_text_label: Label::new(
PANEL_X + 3,
PANEL_Y + 2,
"Oops! The installation process has encountered an unexpected problem.\n\
The following details could be useful. Press the Retry button to try again.",
13.0,
),
failure_detail: String::new(),
copy_error_button: GradientButton::new(PANEL_X + 441, PANEL_Y + 58, 104, 31, "COPY"),
progress_state: Arc::new(Mutex::new(ProgressState {
label: "Preparing...".into(),
value: 0,
done: false,
error: None,
})),
fonts,
prev_mouse_down: false,
}
}
pub fn update_and_draw(
&mut self,
renderer: &mut Renderer,
mx: i32,
my: i32,
mouse_down: bool,
) {
let mouse_released = self.prev_mouse_down && !mouse_down;
self.prev_mouse_down = mouse_down;
self.prev_button.update(mx, my, mouse_down);
self.next_button.update(mx, my, mouse_down);
// Handle clicks.
if self.prev_button.clicked(mx, my, mouse_released) {
std::process::exit(0);
}
if self.next_button.clicked(mx, my, mouse_released) {
match self.panel {
Panel::BeforeInstall | Panel::Failure => self.start_uninstall(),
_ => {}
}
}
// Panel-specific input.
if self.panel == Panel::Failure {
self.copy_error_button.update(mx, my, mouse_down);
if self.copy_error_button.clicked(mx, my, mouse_released) {
let _ = std::process::Command::new("cmd")
.args(["/C", &format!("echo {} | clip", self.failure_detail)])
.output();
}
}
// Poll progress.
if self.panel == Panel::Install {
self.poll_progress();
}
// Draw.
renderer.clear(BG_COLOR);
renderer.draw_image(90, 36, &self.flash_logo);
self.fonts
.draw_text(renderer, 233, 54, &self.title_text, 32.0, FG_COLOR);
self.fonts
.draw_text(renderer, 280, 99, &self.subtitle_text, 17.0, FG_COLOR);
// Separator.
renderer.fill_rect(0, 270, WIDTH as i32, 1, 0x00696969);
match self.panel {
Panel::BeforeInstall => self.before_label.draw(renderer, &self.fonts),
Panel::Install => {
self.progress_header.draw(renderer, &self.fonts);
self.progress_label.draw(renderer, &self.fonts);
self.progress_bar.draw(renderer);
}
Panel::Complete => self.complete_label.draw(renderer, &self.fonts),
Panel::Failure => {
self.failure_text_label.draw(renderer, &self.fonts);
let detail = if self.failure_detail.len() > 300 {
&self.failure_detail[..300]
} else {
&self.failure_detail
};
self.fonts.draw_text_multiline(
renderer,
PANEL_X + 4,
PANEL_Y + 44,
detail,
11.0,
FG_COLOR,
1.0,
);
self.copy_error_button.draw(renderer, &self.fonts);
}
}
self.prev_button.draw(renderer, &self.fonts);
self.next_button.draw(renderer, &self.fonts);
}
fn start_uninstall(&mut self) {
self.panel = Panel::Install;
self.prev_button.enabled = false;
self.next_button.visible = false;
self.progress_bar.maximum = UNINSTALL_TICKS;
self.progress_bar.value = 0;
{
let mut state = self.progress_state.lock().unwrap();
state.label = "Preparing...".into();
state.value = 0;
state.done = false;
state.error = None;
}
let progress = Arc::clone(&self.progress_state);
std::thread::spawn(move || {
let callback = ThreadProgressCallback {
state: Arc::clone(&progress),
};
let redir = redirection::disable_redirection();
let result = uninstaller::uninstall(&callback);
redirection::enable_redirection(redir);
let mut state = progress.lock().unwrap();
state.done = true;
if let Err(e) = result {
state.error = Some(e.to_string());
}
});
}
fn poll_progress(&mut self) {
let state = self.progress_state.lock().unwrap();
self.progress_label.text = state.label.clone();
self.progress_bar.value = state.value;
if state.done {
if let Some(ref err) = state.error {
self.failure_detail = err.clone();
drop(state);
self.open_failure();
} else {
drop(state);
self.open_complete();
}
}
}
fn open_complete(&mut self) {
self.panel = Panel::Complete;
self.prev_button.text = "QUIT".into();
self.prev_button.enabled = true;
self.next_button.visible = false;
}
fn open_failure(&mut self) {
self.panel = Panel::Failure;
self.prev_button.text = "QUIT".into();
self.prev_button.enabled = true;
self.next_button.text = "RETRY".into();
self.next_button.visible = true;
}
}
struct ThreadProgressCallback {
state: Arc<Mutex<ProgressState>>,
}
impl ProgressCallback for ThreadProgressCallback {
fn update_progress_label(&self, text: &str, tick: bool) {
let mut state = self.state.lock().unwrap();
state.label = text.to_string();
if tick {
state.value += 1;
}
}
fn tick_progress(&self) {
let mut state = self.state.lock().unwrap();
state.value += 1;
}
}
fn load_resource_image(name: &str) -> RgbaImage {
let bytes: &[u8] = match name {
"flashLogo.png" => include_bytes!("../../../resources/flashLogo.png"),
_ => return RgbaImage::empty(),
};
RgbaImage::from_png_bytes(bytes)
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 257 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 448 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 115 KiB

Loading…
Cancel
Save