Compare commits
1 Commits
11ada8521e
...
9b6c1d8411
| Author | SHA1 | Date |
|---|---|---|
|
|
9b6c1d8411 | 3 weeks ago |
File diff suppressed because it is too large
Load Diff
@ -1,70 +0,0 @@
|
|||||||
use argh::FromArgs;
|
|
||||||
|
|
||||||
/// Clean Flash Player Installer
|
|
||||||
///
|
|
||||||
/// Install Clean Flash Player with GUI, TUI, or silent mode.
|
|
||||||
#[derive(FromArgs)]
|
|
||||||
pub struct InstallerArgs {
|
|
||||||
/// use the terminal UI (ratatui) instead of the graphical window
|
|
||||||
#[argh(switch)]
|
|
||||||
pub cli: bool,
|
|
||||||
|
|
||||||
/// run the installation non-interactively (no GUI or TUI)
|
|
||||||
#[argh(switch)]
|
|
||||||
pub install: bool,
|
|
||||||
|
|
||||||
/// suppress all output (only valid with --install)
|
|
||||||
#[argh(switch)]
|
|
||||||
pub silent: bool,
|
|
||||||
|
|
||||||
/// install the Modern Browsers (MV3) native host
|
|
||||||
#[argh(switch)]
|
|
||||||
pub native_host: bool,
|
|
||||||
|
|
||||||
/// install the Pepper API (PPAPI) plugin
|
|
||||||
#[argh(switch)]
|
|
||||||
pub pepper: bool,
|
|
||||||
|
|
||||||
/// install the Netscape API (NPAPI) plugin
|
|
||||||
#[argh(switch)]
|
|
||||||
pub netscape: bool,
|
|
||||||
|
|
||||||
/// install the ActiveX (OCX) plugin
|
|
||||||
#[argh(switch)]
|
|
||||||
pub activex: bool,
|
|
||||||
|
|
||||||
/// install the standalone Flash Player
|
|
||||||
#[argh(switch)]
|
|
||||||
pub player: bool,
|
|
||||||
|
|
||||||
/// create desktop shortcuts for standalone player (requires --player)
|
|
||||||
#[argh(switch)]
|
|
||||||
pub player_desktop: bool,
|
|
||||||
|
|
||||||
/// create start menu shortcuts for standalone player (requires --player)
|
|
||||||
#[argh(switch)]
|
|
||||||
pub player_start_menu: bool,
|
|
||||||
|
|
||||||
/// install the debug version
|
|
||||||
#[argh(switch)]
|
|
||||||
pub debug: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// The mode in which the application should run.
|
|
||||||
pub enum RunMode {
|
|
||||||
Gui,
|
|
||||||
Tui,
|
|
||||||
Silent,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl InstallerArgs {
|
|
||||||
pub fn determine_mode(&self) -> RunMode {
|
|
||||||
if self.install {
|
|
||||||
RunMode::Silent
|
|
||||||
} else if self.cli {
|
|
||||||
RunMode::Tui
|
|
||||||
} else {
|
|
||||||
RunMode::Gui
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,81 +0,0 @@
|
|||||||
use crate::cli_args::InstallerArgs;
|
|
||||||
use crate::install_flags::{self, InstallFlags};
|
|
||||||
use crate::installer;
|
|
||||||
use clean_flash_common::{uninstaller, redirection, update_checker, ProgressCallback, InstallError};
|
|
||||||
|
|
||||||
struct SilentProgressCallback {
|
|
||||||
silent: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ProgressCallback for SilentProgressCallback {
|
|
||||||
fn update_progress_label(&self, text: &str, _tick: bool) {
|
|
||||||
if !self.silent {
|
|
||||||
eprintln!("[*] {}", text);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn tick_progress(&self) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn run_silent_install(args: &InstallerArgs) -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
let version = update_checker::FLASH_VERSION;
|
|
||||||
|
|
||||||
let mut flags = InstallFlags::new();
|
|
||||||
flags.set_conditionally(args.native_host, install_flags::NATIVE_HOST);
|
|
||||||
flags.set_conditionally(args.pepper, install_flags::PEPPER);
|
|
||||||
flags.set_conditionally(args.netscape, install_flags::NETSCAPE);
|
|
||||||
flags.set_conditionally(args.activex, install_flags::ACTIVEX);
|
|
||||||
flags.set_conditionally(args.player, install_flags::PLAYER);
|
|
||||||
flags.set_conditionally(args.player_desktop, install_flags::PLAYER_DESKTOP);
|
|
||||||
flags.set_conditionally(args.player_start_menu, install_flags::PLAYER_START_MENU);
|
|
||||||
flags.set_conditionally(args.debug, install_flags::DEBUG);
|
|
||||||
|
|
||||||
if flags.is_none_set() {
|
|
||||||
if !args.silent {
|
|
||||||
eprintln!("No components selected. Use flags to select what to install:");
|
|
||||||
eprintln!(" --native-host Modern Browsers (MV3)");
|
|
||||||
eprintln!(" --pepper Pepper API (PPAPI)");
|
|
||||||
eprintln!(" --netscape Netscape API (NPAPI)");
|
|
||||||
eprintln!(" --activex ActiveX (OCX)");
|
|
||||||
eprintln!(" --player Standalone Flash Player");
|
|
||||||
eprintln!(" --player-desktop Desktop shortcuts (requires --player)");
|
|
||||||
eprintln!(" --player-start-menu Start Menu shortcuts (requires --player)");
|
|
||||||
eprintln!(" --debug Debug version");
|
|
||||||
eprintln!("\nExample: clean_flash_installer --install --native-host --pepper");
|
|
||||||
}
|
|
||||||
std::process::exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if !args.silent {
|
|
||||||
eprintln!("Clean Flash Player {} - Installation", version);
|
|
||||||
}
|
|
||||||
|
|
||||||
let callback = SilentProgressCallback {
|
|
||||||
silent: args.silent,
|
|
||||||
};
|
|
||||||
|
|
||||||
let redir = redirection::disable_redirection();
|
|
||||||
|
|
||||||
let result = (|| -> Result<(), InstallError> {
|
|
||||||
uninstaller::uninstall(&callback)?;
|
|
||||||
installer::install(&callback, &mut flags)?;
|
|
||||||
Ok(())
|
|
||||||
})();
|
|
||||||
|
|
||||||
redirection::enable_redirection(redir);
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(()) => {
|
|
||||||
if !args.silent {
|
|
||||||
eprintln!("[+] Installation completed successfully.");
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
if !args.silent {
|
|
||||||
eprintln!("[!] Installation failed: {}", e);
|
|
||||||
}
|
|
||||||
std::process::exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,639 +0,0 @@
|
|||||||
use crate::install_flags::{self, InstallFlags};
|
|
||||||
use crate::installer;
|
|
||||||
use clean_flash_common::{uninstaller, redirection, update_checker, ProgressCallback, InstallError};
|
|
||||||
use crossterm::{
|
|
||||||
event::{self, Event, KeyCode, KeyEventKind},
|
|
||||||
terminal::{disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen},
|
|
||||||
execute,
|
|
||||||
};
|
|
||||||
use ratatui::{
|
|
||||||
prelude::*,
|
|
||||||
widgets::{Block, Borders, Gauge, List, ListItem, Paragraph, Wrap},
|
|
||||||
};
|
|
||||||
use std::io::{self, stdout};
|
|
||||||
use std::sync::{Arc, Mutex};
|
|
||||||
|
|
||||||
const FLASH_VERSION: &str = update_checker::FLASH_VERSION;
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, PartialEq, Eq)]
|
|
||||||
enum Panel {
|
|
||||||
Disclaimer,
|
|
||||||
Choice,
|
|
||||||
PlayerChoice,
|
|
||||||
DebugChoice,
|
|
||||||
BeforeInstall,
|
|
||||||
Install,
|
|
||||||
Complete,
|
|
||||||
Failure,
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ProgressState {
|
|
||||||
label: String,
|
|
||||||
value: u32,
|
|
||||||
maximum: u32,
|
|
||||||
done: bool,
|
|
||||||
error: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
struct TuiInstallState {
|
|
||||||
panel: Panel,
|
|
||||||
// Disclaimer
|
|
||||||
disclaimer_accepted: bool,
|
|
||||||
// Choice
|
|
||||||
native_host: bool,
|
|
||||||
pepper: bool,
|
|
||||||
netscape: bool,
|
|
||||||
activex: bool,
|
|
||||||
choice_cursor: usize,
|
|
||||||
// Player choice
|
|
||||||
player: bool,
|
|
||||||
player_desktop: bool,
|
|
||||||
player_start_menu: bool,
|
|
||||||
player_cursor: usize,
|
|
||||||
// Debug
|
|
||||||
debug_chosen: bool,
|
|
||||||
// Progress
|
|
||||||
progress: Arc<Mutex<ProgressState>>,
|
|
||||||
// Failure
|
|
||||||
failure_detail: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TuiInstallState {
|
|
||||||
fn new() -> Self {
|
|
||||||
Self {
|
|
||||||
panel: Panel::Disclaimer,
|
|
||||||
disclaimer_accepted: false,
|
|
||||||
native_host: true,
|
|
||||||
pepper: false,
|
|
||||||
netscape: false,
|
|
||||||
activex: false,
|
|
||||||
choice_cursor: 0,
|
|
||||||
player: false,
|
|
||||||
player_desktop: false,
|
|
||||||
player_start_menu: false,
|
|
||||||
player_cursor: 0,
|
|
||||||
debug_chosen: false,
|
|
||||||
progress: Arc::new(Mutex::new(ProgressState {
|
|
||||||
label: "Preparing...".into(),
|
|
||||||
value: 0,
|
|
||||||
maximum: 10,
|
|
||||||
done: false,
|
|
||||||
error: None,
|
|
||||||
})),
|
|
||||||
failure_detail: String::new(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn choice_items(&self) -> Vec<(&str, bool, bool)> {
|
|
||||||
let is_windows = cfg!(target_os = "windows");
|
|
||||||
vec![
|
|
||||||
("Modern Browsers (MV3) - Chrome/Firefox/Edge", self.native_host, true),
|
|
||||||
("Pepper API (PPAPI) - Chrome/Opera/Brave", self.pepper, is_windows),
|
|
||||||
("Netscape API (NPAPI) - Firefox/ESR/Waterfox", self.netscape, is_windows),
|
|
||||||
("ActiveX (OCX) - IE/Embedded/Desktop", self.activex, is_windows),
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
fn player_items(&self) -> Vec<(&str, bool, bool)> {
|
|
||||||
vec![
|
|
||||||
("Install Standalone Flash Player", self.player, true),
|
|
||||||
("Create Desktop Shortcuts", self.player_desktop, self.player),
|
|
||||||
("Create Start Menu Shortcuts", self.player_start_menu, self.player),
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
fn toggle_choice(&mut self) {
|
|
||||||
let is_windows = cfg!(target_os = "windows");
|
|
||||||
match self.choice_cursor {
|
|
||||||
0 => self.native_host = !self.native_host,
|
|
||||||
1 if is_windows => self.pepper = !self.pepper,
|
|
||||||
2 if is_windows => self.netscape = !self.netscape,
|
|
||||||
3 if is_windows => self.activex = !self.activex,
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn toggle_player(&mut self) {
|
|
||||||
match self.player_cursor {
|
|
||||||
0 => {
|
|
||||||
self.player = !self.player;
|
|
||||||
if !self.player {
|
|
||||||
self.player_desktop = false;
|
|
||||||
self.player_start_menu = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
1 if self.player => self.player_desktop = !self.player_desktop,
|
|
||||||
2 if self.player => self.player_start_menu = !self.player_start_menu,
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn build_flags(&self) -> InstallFlags {
|
|
||||||
let mut flags = InstallFlags::new();
|
|
||||||
flags.set_conditionally(self.native_host, install_flags::NATIVE_HOST);
|
|
||||||
flags.set_conditionally(self.pepper, install_flags::PEPPER);
|
|
||||||
flags.set_conditionally(self.netscape, install_flags::NETSCAPE);
|
|
||||||
flags.set_conditionally(self.activex, install_flags::ACTIVEX);
|
|
||||||
flags.set_conditionally(self.player, install_flags::PLAYER);
|
|
||||||
flags.set_conditionally(self.player_desktop, install_flags::PLAYER_DESKTOP);
|
|
||||||
flags.set_conditionally(self.player_start_menu, install_flags::PLAYER_START_MENU);
|
|
||||||
flags.set_conditionally(self.debug_chosen, install_flags::DEBUG);
|
|
||||||
flags
|
|
||||||
}
|
|
||||||
|
|
||||||
fn start_install(&mut self) {
|
|
||||||
self.panel = Panel::Install;
|
|
||||||
let mut flags = self.build_flags();
|
|
||||||
let max_ticks = flags.get_ticks();
|
|
||||||
|
|
||||||
{
|
|
||||||
let mut state = self.progress.lock().unwrap();
|
|
||||||
state.label = "Preparing...".into();
|
|
||||||
state.value = 0;
|
|
||||||
state.maximum = max_ticks;
|
|
||||||
state.done = false;
|
|
||||||
state.error = None;
|
|
||||||
}
|
|
||||||
|
|
||||||
let progress = Arc::clone(&self.progress);
|
|
||||||
std::thread::spawn(move || {
|
|
||||||
let callback = ThreadProgressCallback {
|
|
||||||
state: Arc::clone(&progress),
|
|
||||||
};
|
|
||||||
|
|
||||||
let redir = redirection::disable_redirection();
|
|
||||||
let result = (|| -> Result<(), InstallError> {
|
|
||||||
uninstaller::uninstall(&callback)?;
|
|
||||||
installer::install(&callback, &mut flags)?;
|
|
||||||
Ok(())
|
|
||||||
})();
|
|
||||||
redirection::enable_redirection(redir);
|
|
||||||
|
|
||||||
let mut state = progress.lock().unwrap();
|
|
||||||
if let Err(e) = result {
|
|
||||||
state.error = Some(e.to_string());
|
|
||||||
}
|
|
||||||
state.done = true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn summary_text(&self) -> String {
|
|
||||||
let mut parts = Vec::new();
|
|
||||||
if self.native_host { parts.push("Modern Browsers (via extension)"); }
|
|
||||||
if self.pepper { parts.push("Google Chrome"); }
|
|
||||||
if self.netscape { parts.push("Mozilla Firefox"); }
|
|
||||||
if self.activex { parts.push("Internet Explorer"); }
|
|
||||||
if self.player { parts.push("Standalone Player"); }
|
|
||||||
|
|
||||||
if parts.is_empty() {
|
|
||||||
"You are about to uninstall Clean Flash Player.\n\
|
|
||||||
The installer will completely remove all versions of Flash Player."
|
|
||||||
.to_string()
|
|
||||||
} else {
|
|
||||||
format!(
|
|
||||||
"You are about to install Clean Flash Player for:\n {}\n\n\
|
|
||||||
The installer will close browsers running Flash, uninstall previous versions, and install.",
|
|
||||||
parts.join(", ")
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn run_tui_installer() -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
enable_raw_mode()?;
|
|
||||||
let mut stdout = stdout();
|
|
||||||
execute!(stdout, EnterAlternateScreen)?;
|
|
||||||
let backend = CrosstermBackend::new(stdout);
|
|
||||||
let mut terminal = Terminal::new(backend)?;
|
|
||||||
|
|
||||||
let result = run_app(&mut terminal);
|
|
||||||
|
|
||||||
disable_raw_mode()?;
|
|
||||||
execute!(terminal.backend_mut(), LeaveAlternateScreen)?;
|
|
||||||
terminal.show_cursor()?;
|
|
||||||
|
|
||||||
result
|
|
||||||
}
|
|
||||||
|
|
||||||
fn run_app(terminal: &mut Terminal<CrosstermBackend<io::Stdout>>) -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
let mut state = TuiInstallState::new();
|
|
||||||
|
|
||||||
#[cfg(not(target_os = "windows"))]
|
|
||||||
{
|
|
||||||
state.pepper = false;
|
|
||||||
state.netscape = false;
|
|
||||||
state.activex = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
loop {
|
|
||||||
terminal.draw(|f| draw_ui(f, &state))?;
|
|
||||||
|
|
||||||
// While installing, poll both events (with timeout) and progress
|
|
||||||
if state.panel == Panel::Install {
|
|
||||||
// Poll progress
|
|
||||||
{
|
|
||||||
let ps = state.progress.lock().unwrap();
|
|
||||||
if ps.done {
|
|
||||||
if let Some(ref err) = ps.error {
|
|
||||||
state.failure_detail = err.clone();
|
|
||||||
drop(ps);
|
|
||||||
state.panel = Panel::Failure;
|
|
||||||
} else {
|
|
||||||
drop(ps);
|
|
||||||
state.panel = Panel::Complete;
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if event::poll(std::time::Duration::from_millis(100))? {
|
|
||||||
if let Event::Key(key) = event::read()? {
|
|
||||||
if key.kind == KeyEventKind::Press && (key.code == KeyCode::Char('q') || key.code == KeyCode::Esc) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Event::Key(key) = event::read()? {
|
|
||||||
if key.kind != KeyEventKind::Press {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
match key.code {
|
|
||||||
KeyCode::Char('q') | KeyCode::Esc => {
|
|
||||||
match state.panel {
|
|
||||||
Panel::Install => {} // can't quit during install
|
|
||||||
_ => break,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
KeyCode::Enter => {
|
|
||||||
match state.panel {
|
|
||||||
Panel::Disclaimer => {
|
|
||||||
if state.disclaimer_accepted {
|
|
||||||
state.panel = Panel::Choice;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Panel::Choice => {
|
|
||||||
if cfg!(target_os = "windows") {
|
|
||||||
state.panel = Panel::PlayerChoice;
|
|
||||||
} else {
|
|
||||||
state.panel = Panel::BeforeInstall;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Panel::PlayerChoice => state.panel = Panel::DebugChoice,
|
|
||||||
Panel::DebugChoice => state.panel = Panel::BeforeInstall,
|
|
||||||
Panel::BeforeInstall => state.start_install(),
|
|
||||||
Panel::Complete => break,
|
|
||||||
Panel::Failure => state.start_install(), // retry
|
|
||||||
Panel::Install => {} // handled by progress polling above
|
|
||||||
}
|
|
||||||
}
|
|
||||||
KeyCode::Backspace => {
|
|
||||||
match state.panel {
|
|
||||||
Panel::Choice => state.panel = Panel::Disclaimer,
|
|
||||||
Panel::PlayerChoice => state.panel = Panel::Choice,
|
|
||||||
Panel::DebugChoice => state.panel = Panel::PlayerChoice,
|
|
||||||
Panel::BeforeInstall => {
|
|
||||||
if cfg!(target_os = "windows") {
|
|
||||||
state.panel = Panel::DebugChoice;
|
|
||||||
} else {
|
|
||||||
state.panel = Panel::Choice;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
KeyCode::Char(' ') => {
|
|
||||||
match state.panel {
|
|
||||||
Panel::Disclaimer => {
|
|
||||||
state.disclaimer_accepted = !state.disclaimer_accepted;
|
|
||||||
}
|
|
||||||
Panel::Choice => state.toggle_choice(),
|
|
||||||
Panel::PlayerChoice => state.toggle_player(),
|
|
||||||
Panel::DebugChoice => {
|
|
||||||
state.debug_chosen = !state.debug_chosen;
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
KeyCode::Up => {
|
|
||||||
match state.panel {
|
|
||||||
Panel::Choice => {
|
|
||||||
if state.choice_cursor > 0 {
|
|
||||||
state.choice_cursor -= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Panel::PlayerChoice => {
|
|
||||||
if state.player_cursor > 0 {
|
|
||||||
state.player_cursor -= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
KeyCode::Down => {
|
|
||||||
match state.panel {
|
|
||||||
Panel::Choice => {
|
|
||||||
let max = state.choice_items().len().saturating_sub(1);
|
|
||||||
if state.choice_cursor < max {
|
|
||||||
state.choice_cursor += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Panel::PlayerChoice => {
|
|
||||||
let max = state.player_items().len().saturating_sub(1);
|
|
||||||
if state.player_cursor < max {
|
|
||||||
state.player_cursor += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_ui(f: &mut Frame, state: &TuiInstallState) {
|
|
||||||
let area = f.area();
|
|
||||||
|
|
||||||
// Layout: title area, main panel, footer
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([
|
|
||||||
Constraint::Length(3), // title
|
|
||||||
Constraint::Min(5), // content
|
|
||||||
Constraint::Length(3), // footer
|
|
||||||
])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
// Title bar
|
|
||||||
let title = format!(
|
|
||||||
" Clean Flash Player {} Installer ",
|
|
||||||
FLASH_VERSION
|
|
||||||
);
|
|
||||||
let title_block = Block::default()
|
|
||||||
.borders(Borders::ALL)
|
|
||||||
.border_style(Style::default().fg(Color::Cyan))
|
|
||||||
.title(title)
|
|
||||||
.title_alignment(Alignment::Center);
|
|
||||||
f.render_widget(title_block, chunks[0]);
|
|
||||||
|
|
||||||
// Content
|
|
||||||
let content_block = Block::default()
|
|
||||||
.borders(Borders::ALL)
|
|
||||||
.border_style(Style::default().fg(Color::DarkGray));
|
|
||||||
|
|
||||||
let inner = content_block.inner(chunks[1]);
|
|
||||||
f.render_widget(content_block, chunks[1]);
|
|
||||||
|
|
||||||
match state.panel {
|
|
||||||
Panel::Disclaimer => draw_disclaimer(f, inner, state),
|
|
||||||
Panel::Choice => draw_choice(f, inner, state),
|
|
||||||
Panel::PlayerChoice => draw_player_choice(f, inner, state),
|
|
||||||
Panel::DebugChoice => draw_debug_choice(f, inner, state),
|
|
||||||
Panel::BeforeInstall => draw_before_install(f, inner, state),
|
|
||||||
Panel::Install => draw_install(f, inner, state),
|
|
||||||
Panel::Complete => draw_complete(f, inner, state),
|
|
||||||
Panel::Failure => draw_failure(f, inner, state),
|
|
||||||
}
|
|
||||||
|
|
||||||
// Footer
|
|
||||||
let footer_text = match state.panel {
|
|
||||||
Panel::Disclaimer => "[Space] Toggle accept [Enter] Next [q/Esc] Quit",
|
|
||||||
Panel::Choice => "[↑↓] Move [Space] Toggle [Enter] Next [Backspace] Back [q] Quit",
|
|
||||||
Panel::PlayerChoice => "[↑↓] Move [Space] Toggle [Enter] Next [Backspace] Back [q] Quit",
|
|
||||||
Panel::DebugChoice => "[Space] Toggle debug [Enter] Next [Backspace] Back [q] Quit",
|
|
||||||
Panel::BeforeInstall => "[Enter] Install [Backspace] Back [q] Quit",
|
|
||||||
Panel::Install => "Installing...",
|
|
||||||
Panel::Complete => "[Enter/q] Quit",
|
|
||||||
Panel::Failure => "[Enter] Retry [q] Quit",
|
|
||||||
};
|
|
||||||
let footer = Paragraph::new(footer_text)
|
|
||||||
.alignment(Alignment::Center)
|
|
||||||
.block(Block::default().borders(Borders::ALL).border_style(Style::default().fg(Color::DarkGray)));
|
|
||||||
f.render_widget(footer, chunks[2]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_disclaimer(f: &mut Frame, area: Rect, state: &TuiInstallState) {
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([Constraint::Min(3), Constraint::Length(1)])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let text = "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 \
|
|
||||||
version with adware removed.\n\n\
|
|
||||||
Adobe is not required by any means to provide support for this version of Flash Player.";
|
|
||||||
|
|
||||||
let paragraph = Paragraph::new(text)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(paragraph, chunks[0]);
|
|
||||||
|
|
||||||
let checkbox = if state.disclaimer_accepted {
|
|
||||||
"[x] I accept the above disclaimer"
|
|
||||||
} else {
|
|
||||||
"[ ] I accept the above disclaimer"
|
|
||||||
};
|
|
||||||
let cb_style = if state.disclaimer_accepted {
|
|
||||||
Style::default().fg(Color::Green)
|
|
||||||
} else {
|
|
||||||
Style::default().fg(Color::Yellow)
|
|
||||||
};
|
|
||||||
let cb = Paragraph::new(checkbox).style(cb_style);
|
|
||||||
f.render_widget(cb, chunks[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_choice(f: &mut Frame, area: Rect, state: &TuiInstallState) {
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([Constraint::Length(2), Constraint::Min(3)])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let header = Paragraph::new("Which browser plugins would you like to install?")
|
|
||||||
.style(Style::default().fg(Color::White).add_modifier(Modifier::BOLD));
|
|
||||||
f.render_widget(header, chunks[0]);
|
|
||||||
|
|
||||||
let items: Vec<ListItem> = state
|
|
||||||
.choice_items()
|
|
||||||
.iter()
|
|
||||||
.enumerate()
|
|
||||||
.map(|(i, (label, checked, enabled))| {
|
|
||||||
let marker = if *checked { "[x]" } else { "[ ]" };
|
|
||||||
let text = format!("{} {}", marker, label);
|
|
||||||
let style = if !enabled {
|
|
||||||
Style::default().fg(Color::DarkGray)
|
|
||||||
} else if i == state.choice_cursor {
|
|
||||||
Style::default().fg(Color::Cyan).add_modifier(Modifier::BOLD)
|
|
||||||
} else {
|
|
||||||
Style::default().fg(Color::White)
|
|
||||||
};
|
|
||||||
ListItem::new(text).style(style)
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let list = List::new(items);
|
|
||||||
f.render_widget(list, chunks[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_player_choice(f: &mut Frame, area: Rect, state: &TuiInstallState) {
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([Constraint::Length(2), Constraint::Min(3)])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let header = Paragraph::new("Would you like to install the standalone Flash Player?")
|
|
||||||
.style(Style::default().fg(Color::White).add_modifier(Modifier::BOLD));
|
|
||||||
f.render_widget(header, chunks[0]);
|
|
||||||
|
|
||||||
let items: Vec<ListItem> = state
|
|
||||||
.player_items()
|
|
||||||
.iter()
|
|
||||||
.enumerate()
|
|
||||||
.map(|(i, (label, checked, enabled))| {
|
|
||||||
let marker = if *checked { "[x]" } else { "[ ]" };
|
|
||||||
let text = format!("{} {}", marker, label);
|
|
||||||
let style = if !enabled {
|
|
||||||
Style::default().fg(Color::DarkGray)
|
|
||||||
} else if i == state.player_cursor {
|
|
||||||
Style::default().fg(Color::Cyan).add_modifier(Modifier::BOLD)
|
|
||||||
} else {
|
|
||||||
Style::default().fg(Color::White)
|
|
||||||
};
|
|
||||||
ListItem::new(text).style(style)
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let list = List::new(items);
|
|
||||||
f.render_widget(list, chunks[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_debug_choice(f: &mut Frame, area: Rect, state: &TuiInstallState) {
|
|
||||||
let text = "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 Enter to continue.";
|
|
||||||
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([Constraint::Min(3), Constraint::Length(1)])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let paragraph = Paragraph::new(text)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(paragraph, chunks[0]);
|
|
||||||
|
|
||||||
let checkbox = if state.debug_chosen {
|
|
||||||
"[x] Install debug version"
|
|
||||||
} else {
|
|
||||||
"[ ] Install debug version"
|
|
||||||
};
|
|
||||||
let cb_style = if state.debug_chosen {
|
|
||||||
Style::default().fg(Color::Yellow)
|
|
||||||
} else {
|
|
||||||
Style::default().fg(Color::White)
|
|
||||||
};
|
|
||||||
let cb = Paragraph::new(checkbox).style(cb_style);
|
|
||||||
f.render_widget(cb, chunks[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_before_install(f: &mut Frame, area: Rect, state: &TuiInstallState) {
|
|
||||||
let text = state.summary_text();
|
|
||||||
let paragraph = Paragraph::new(text)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(paragraph, area);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_install(f: &mut Frame, area: Rect, state: &TuiInstallState) {
|
|
||||||
let ps = state.progress.lock().unwrap();
|
|
||||||
let label = ps.label.clone();
|
|
||||||
let ratio = if ps.maximum > 0 {
|
|
||||||
(ps.value as f64 / ps.maximum as f64).min(1.0)
|
|
||||||
} else {
|
|
||||||
0.0
|
|
||||||
};
|
|
||||||
drop(ps);
|
|
||||||
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([
|
|
||||||
Constraint::Length(2),
|
|
||||||
Constraint::Length(1),
|
|
||||||
Constraint::Length(3),
|
|
||||||
Constraint::Min(0),
|
|
||||||
])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let header = Paragraph::new("Installation in progress...")
|
|
||||||
.style(Style::default().fg(Color::White).add_modifier(Modifier::BOLD));
|
|
||||||
f.render_widget(header, chunks[0]);
|
|
||||||
|
|
||||||
let status = Paragraph::new(label).style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(status, chunks[1]);
|
|
||||||
|
|
||||||
let gauge = Gauge::default()
|
|
||||||
.block(Block::default().borders(Borders::ALL))
|
|
||||||
.gauge_style(Style::default().fg(Color::Cyan).bg(Color::DarkGray))
|
|
||||||
.ratio(ratio);
|
|
||||||
f.render_widget(gauge, chunks[2]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_complete(f: &mut Frame, area: Rect, _state: &TuiInstallState) {
|
|
||||||
let text = "Clean Flash Player has been successfully installed!\n\n\
|
|
||||||
Don't forget, Flash Player is no longer compatible with new browsers.\n\
|
|
||||||
For browser recommendations and Flash Player updates, check out Clean Flash Player's website!";
|
|
||||||
|
|
||||||
let paragraph = Paragraph::new(text)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::Green));
|
|
||||||
f.render_widget(paragraph, area);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_failure(f: &mut Frame, area: Rect, state: &TuiInstallState) {
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([Constraint::Length(3), Constraint::Min(2)])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let header = Paragraph::new(
|
|
||||||
"Oops! The installation process has encountered an unexpected problem.\nPress Enter to retry or q to quit.",
|
|
||||||
)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::Red).add_modifier(Modifier::BOLD));
|
|
||||||
f.render_widget(header, chunks[0]);
|
|
||||||
|
|
||||||
let detail_text = if state.failure_detail.len() > 500 {
|
|
||||||
&state.failure_detail[..500]
|
|
||||||
} else {
|
|
||||||
&state.failure_detail
|
|
||||||
};
|
|
||||||
let detail = Paragraph::new(detail_text.to_string())
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(detail, chunks[1]);
|
|
||||||
}
|
|
||||||
@ -1,38 +0,0 @@
|
|||||||
use argh::FromArgs;
|
|
||||||
|
|
||||||
/// Clean Flash Player Uninstaller
|
|
||||||
///
|
|
||||||
/// Uninstall Clean Flash Player with GUI, TUI, or silent mode.
|
|
||||||
#[derive(FromArgs)]
|
|
||||||
pub struct UninstallerArgs {
|
|
||||||
/// use the terminal UI (ratatui) instead of the graphical window
|
|
||||||
#[argh(switch)]
|
|
||||||
pub cli: bool,
|
|
||||||
|
|
||||||
/// run the uninstallation non-interactively (no GUI or TUI)
|
|
||||||
#[argh(switch)]
|
|
||||||
pub uninstall: bool,
|
|
||||||
|
|
||||||
/// suppress all output (only valid with --uninstall)
|
|
||||||
#[argh(switch)]
|
|
||||||
pub silent: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// The mode in which the application should run.
|
|
||||||
pub enum RunMode {
|
|
||||||
Gui,
|
|
||||||
Tui,
|
|
||||||
Silent,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl UninstallerArgs {
|
|
||||||
pub fn determine_mode(&self) -> RunMode {
|
|
||||||
if self.uninstall {
|
|
||||||
RunMode::Silent
|
|
||||||
} else if self.cli {
|
|
||||||
RunMode::Tui
|
|
||||||
} else {
|
|
||||||
RunMode::Gui
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,47 +0,0 @@
|
|||||||
use crate::cli_args::UninstallerArgs;
|
|
||||||
use clean_flash_common::{uninstaller, redirection, update_checker, ProgressCallback, InstallError};
|
|
||||||
|
|
||||||
struct SilentProgressCallback {
|
|
||||||
silent: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ProgressCallback for SilentProgressCallback {
|
|
||||||
fn update_progress_label(&self, text: &str, _tick: bool) {
|
|
||||||
if !self.silent {
|
|
||||||
eprintln!("[*] {}", text);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn tick_progress(&self) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn run_silent_uninstall(args: &UninstallerArgs) -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
let version = update_checker::FLASH_VERSION;
|
|
||||||
|
|
||||||
if !args.silent {
|
|
||||||
eprintln!("Clean Flash Player {} - Silent Uninstall", version);
|
|
||||||
}
|
|
||||||
|
|
||||||
let callback = SilentProgressCallback {
|
|
||||||
silent: args.silent,
|
|
||||||
};
|
|
||||||
|
|
||||||
let redir = redirection::disable_redirection();
|
|
||||||
let result: Result<(), InstallError> = uninstaller::uninstall(&callback);
|
|
||||||
redirection::enable_redirection(redir);
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(()) => {
|
|
||||||
if !args.silent {
|
|
||||||
eprintln!("[+] Uninstallation completed successfully.");
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
if !args.silent {
|
|
||||||
eprintln!("[!] Uninstallation failed: {}", e);
|
|
||||||
}
|
|
||||||
std::process::exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,306 +0,0 @@
|
|||||||
use clean_flash_common::{uninstaller, redirection, update_checker, ProgressCallback, InstallError};
|
|
||||||
use crossterm::{
|
|
||||||
event::{self, Event, KeyCode, KeyEventKind},
|
|
||||||
terminal::{disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen},
|
|
||||||
execute,
|
|
||||||
};
|
|
||||||
use ratatui::{
|
|
||||||
prelude::*,
|
|
||||||
widgets::{Block, Borders, Gauge, Paragraph, Wrap},
|
|
||||||
};
|
|
||||||
use std::io::{self, stdout};
|
|
||||||
use std::sync::{Arc, Mutex};
|
|
||||||
|
|
||||||
const FLASH_VERSION: &str = update_checker::FLASH_VERSION;
|
|
||||||
const UNINSTALL_TICKS: u32 = 10;
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, PartialEq, Eq)]
|
|
||||||
enum Panel {
|
|
||||||
Confirm,
|
|
||||||
Uninstall,
|
|
||||||
Complete,
|
|
||||||
Failure,
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ProgressState {
|
|
||||||
label: String,
|
|
||||||
value: u32,
|
|
||||||
maximum: u32,
|
|
||||||
done: bool,
|
|
||||||
error: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
struct TuiUninstallState {
|
|
||||||
panel: Panel,
|
|
||||||
progress: Arc<Mutex<ProgressState>>,
|
|
||||||
failure_detail: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TuiUninstallState {
|
|
||||||
fn new() -> Self {
|
|
||||||
Self {
|
|
||||||
panel: Panel::Confirm,
|
|
||||||
progress: Arc::new(Mutex::new(ProgressState {
|
|
||||||
label: "Preparing...".into(),
|
|
||||||
value: 0,
|
|
||||||
maximum: UNINSTALL_TICKS,
|
|
||||||
done: false,
|
|
||||||
error: None,
|
|
||||||
})),
|
|
||||||
failure_detail: String::new(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn start_uninstall(&mut self) {
|
|
||||||
self.panel = Panel::Uninstall;
|
|
||||||
|
|
||||||
{
|
|
||||||
let mut state = self.progress.lock().unwrap();
|
|
||||||
state.label = "Preparing...".into();
|
|
||||||
state.value = 0;
|
|
||||||
state.maximum = UNINSTALL_TICKS;
|
|
||||||
state.done = false;
|
|
||||||
state.error = None;
|
|
||||||
}
|
|
||||||
|
|
||||||
let progress = Arc::clone(&self.progress);
|
|
||||||
std::thread::spawn(move || {
|
|
||||||
let callback = ThreadProgressCallback {
|
|
||||||
state: Arc::clone(&progress),
|
|
||||||
};
|
|
||||||
|
|
||||||
let redir = redirection::disable_redirection();
|
|
||||||
let result: Result<(), InstallError> = uninstaller::uninstall(&callback);
|
|
||||||
redirection::enable_redirection(redir);
|
|
||||||
|
|
||||||
let mut state = progress.lock().unwrap();
|
|
||||||
if let Err(e) = result {
|
|
||||||
state.error = Some(e.to_string());
|
|
||||||
}
|
|
||||||
state.done = 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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn run_tui_uninstaller() -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
enable_raw_mode()?;
|
|
||||||
let mut stdout = stdout();
|
|
||||||
execute!(stdout, EnterAlternateScreen)?;
|
|
||||||
let backend = CrosstermBackend::new(stdout);
|
|
||||||
let mut terminal = Terminal::new(backend)?;
|
|
||||||
|
|
||||||
let result = run_app(&mut terminal);
|
|
||||||
|
|
||||||
disable_raw_mode()?;
|
|
||||||
execute!(terminal.backend_mut(), LeaveAlternateScreen)?;
|
|
||||||
terminal.show_cursor()?;
|
|
||||||
|
|
||||||
result
|
|
||||||
}
|
|
||||||
|
|
||||||
fn run_app(terminal: &mut Terminal<CrosstermBackend<io::Stdout>>) -> Result<(), Box<dyn std::error::Error>> {
|
|
||||||
let mut state = TuiUninstallState::new();
|
|
||||||
|
|
||||||
loop {
|
|
||||||
terminal.draw(|f| draw_ui(f, &state))?;
|
|
||||||
|
|
||||||
if state.panel == Panel::Uninstall {
|
|
||||||
// Poll progress
|
|
||||||
{
|
|
||||||
let ps = state.progress.lock().unwrap();
|
|
||||||
if ps.done {
|
|
||||||
if let Some(ref err) = ps.error {
|
|
||||||
state.failure_detail = err.clone();
|
|
||||||
drop(ps);
|
|
||||||
state.panel = Panel::Failure;
|
|
||||||
} else {
|
|
||||||
drop(ps);
|
|
||||||
state.panel = Panel::Complete;
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if event::poll(std::time::Duration::from_millis(100))? {
|
|
||||||
if let Event::Key(key) = event::read()? {
|
|
||||||
if key.kind == KeyEventKind::Press && (key.code == KeyCode::Char('q') || key.code == KeyCode::Esc) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Event::Key(key) = event::read()? {
|
|
||||||
if key.kind != KeyEventKind::Press {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
match key.code {
|
|
||||||
KeyCode::Char('q') | KeyCode::Esc => break,
|
|
||||||
KeyCode::Enter => {
|
|
||||||
match state.panel {
|
|
||||||
Panel::Confirm => state.start_uninstall(),
|
|
||||||
Panel::Complete => break,
|
|
||||||
Panel::Failure => state.start_uninstall(), // retry
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_ui(f: &mut Frame, state: &TuiUninstallState) {
|
|
||||||
let area = f.area();
|
|
||||||
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([
|
|
||||||
Constraint::Length(3),
|
|
||||||
Constraint::Min(5),
|
|
||||||
Constraint::Length(3),
|
|
||||||
])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
// Title
|
|
||||||
let title = format!(
|
|
||||||
" Clean Flash Player {} Uninstaller ",
|
|
||||||
FLASH_VERSION
|
|
||||||
);
|
|
||||||
let title_block = Block::default()
|
|
||||||
.borders(Borders::ALL)
|
|
||||||
.border_style(Style::default().fg(Color::Cyan))
|
|
||||||
.title(title)
|
|
||||||
.title_alignment(Alignment::Center);
|
|
||||||
f.render_widget(title_block, chunks[0]);
|
|
||||||
|
|
||||||
// Content
|
|
||||||
let content_block = Block::default()
|
|
||||||
.borders(Borders::ALL)
|
|
||||||
.border_style(Style::default().fg(Color::DarkGray));
|
|
||||||
let inner = content_block.inner(chunks[1]);
|
|
||||||
f.render_widget(content_block, chunks[1]);
|
|
||||||
|
|
||||||
match state.panel {
|
|
||||||
Panel::Confirm => draw_confirm(f, inner),
|
|
||||||
Panel::Uninstall => draw_uninstall(f, inner, state),
|
|
||||||
Panel::Complete => draw_complete(f, inner),
|
|
||||||
Panel::Failure => draw_failure(f, inner, state),
|
|
||||||
}
|
|
||||||
|
|
||||||
// Footer
|
|
||||||
let footer_text = match state.panel {
|
|
||||||
Panel::Confirm => "[Enter] Uninstall [q/Esc] Quit",
|
|
||||||
Panel::Uninstall => "Uninstalling...",
|
|
||||||
Panel::Complete => "[Enter/q] Quit",
|
|
||||||
Panel::Failure => "[Enter] Retry [q] Quit",
|
|
||||||
};
|
|
||||||
let footer = Paragraph::new(footer_text)
|
|
||||||
.alignment(Alignment::Center)
|
|
||||||
.block(Block::default().borders(Borders::ALL).border_style(Style::default().fg(Color::DarkGray)));
|
|
||||||
f.render_widget(footer, chunks[2]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_confirm(f: &mut Frame, area: Rect) {
|
|
||||||
let text = "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.\n\n\
|
|
||||||
Press Enter to proceed.";
|
|
||||||
|
|
||||||
let paragraph = Paragraph::new(text)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(paragraph, area);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_uninstall(f: &mut Frame, area: Rect, state: &TuiUninstallState) {
|
|
||||||
let ps = state.progress.lock().unwrap();
|
|
||||||
let label = ps.label.clone();
|
|
||||||
let ratio = if ps.maximum > 0 {
|
|
||||||
(ps.value as f64 / ps.maximum as f64).min(1.0)
|
|
||||||
} else {
|
|
||||||
0.0
|
|
||||||
};
|
|
||||||
drop(ps);
|
|
||||||
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([
|
|
||||||
Constraint::Length(2),
|
|
||||||
Constraint::Length(1),
|
|
||||||
Constraint::Length(3),
|
|
||||||
Constraint::Min(0),
|
|
||||||
])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let header = Paragraph::new("Uninstallation in progress...")
|
|
||||||
.style(Style::default().fg(Color::White).add_modifier(Modifier::BOLD));
|
|
||||||
f.render_widget(header, chunks[0]);
|
|
||||||
|
|
||||||
let status = Paragraph::new(label).style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(status, chunks[1]);
|
|
||||||
|
|
||||||
let gauge = Gauge::default()
|
|
||||||
.block(Block::default().borders(Borders::ALL))
|
|
||||||
.gauge_style(Style::default().fg(Color::Cyan).bg(Color::DarkGray))
|
|
||||||
.ratio(ratio);
|
|
||||||
f.render_widget(gauge, chunks[2]);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_complete(f: &mut Frame, area: Rect) {
|
|
||||||
let text = "All versions of Flash Player have been successfully uninstalled.\n\n\
|
|
||||||
If you ever change your mind, check out Clean Flash Player's website!";
|
|
||||||
|
|
||||||
let paragraph = Paragraph::new(text)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::Green));
|
|
||||||
f.render_widget(paragraph, area);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn draw_failure(f: &mut Frame, area: Rect, state: &TuiUninstallState) {
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints([Constraint::Length(3), Constraint::Min(2)])
|
|
||||||
.split(area);
|
|
||||||
|
|
||||||
let header = Paragraph::new(
|
|
||||||
"Oops! The uninstallation process has encountered an unexpected problem.\nPress Enter to retry or q to quit.",
|
|
||||||
)
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::Red).add_modifier(Modifier::BOLD));
|
|
||||||
f.render_widget(header, chunks[0]);
|
|
||||||
|
|
||||||
let detail_text = if state.failure_detail.len() > 500 {
|
|
||||||
&state.failure_detail[..500]
|
|
||||||
} else {
|
|
||||||
&state.failure_detail
|
|
||||||
};
|
|
||||||
let detail = Paragraph::new(detail_text.to_string())
|
|
||||||
.wrap(Wrap { trim: false })
|
|
||||||
.style(Style::default().fg(Color::White));
|
|
||||||
f.render_widget(detail, chunks[1]);
|
|
||||||
}
|
|
||||||
@ -1,24 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
|
||||||
<plist version="1.0">
|
|
||||||
<dict>
|
|
||||||
<key>CFBundleName</key>
|
|
||||||
<string>Clean Flash Installer</string>
|
|
||||||
<key>CFBundleDisplayName</key>
|
|
||||||
<string>Clean Flash Player Installer</string>
|
|
||||||
<key>CFBundleIdentifier</key>
|
|
||||||
<string>com.flashpatch.clean-flash-installer</string>
|
|
||||||
<key>CFBundleVersion</key>
|
|
||||||
<string>34.0.0.330</string>
|
|
||||||
<key>CFBundleShortVersionString</key>
|
|
||||||
<string>34.0.0.330</string>
|
|
||||||
<key>CFBundlePackageType</key>
|
|
||||||
<string>APPL</string>
|
|
||||||
<key>CFBundleExecutable</key>
|
|
||||||
<string>clean_flash_installer</string>
|
|
||||||
<key>NSHighResolutionCapable</key>
|
|
||||||
<true/>
|
|
||||||
<key>LSMinimumSystemVersion</key>
|
|
||||||
<string>10.13</string>
|
|
||||||
</dict>
|
|
||||||
</plist>
|
|
||||||
@ -1,24 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
|
||||||
<plist version="1.0">
|
|
||||||
<dict>
|
|
||||||
<key>CFBundleName</key>
|
|
||||||
<string>Clean Flash Uninstaller</string>
|
|
||||||
<key>CFBundleDisplayName</key>
|
|
||||||
<string>Clean Flash Player Uninstaller</string>
|
|
||||||
<key>CFBundleIdentifier</key>
|
|
||||||
<string>com.flashpatch.clean-flash-uninstaller</string>
|
|
||||||
<key>CFBundleVersion</key>
|
|
||||||
<string>34.0.0.330</string>
|
|
||||||
<key>CFBundleShortVersionString</key>
|
|
||||||
<string>34.0.0.330</string>
|
|
||||||
<key>CFBundlePackageType</key>
|
|
||||||
<string>APPL</string>
|
|
||||||
<key>CFBundleExecutable</key>
|
|
||||||
<string>clean_flash_uninstaller</string>
|
|
||||||
<key>NSHighResolutionCapable</key>
|
|
||||||
<true/>
|
|
||||||
<key>LSMinimumSystemVersion</key>
|
|
||||||
<string>10.13</string>
|
|
||||||
</dict>
|
|
||||||
</plist>
|
|
||||||
Loading…
Reference in New Issue