/* * This file is part of modulo. * * Copyright (C) 2020-2021 Federico Terzi * * modulo is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * modulo is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with modulo. If not, see . */ use std::os::raw::{c_int}; use std::{ffi::CString, sync::Mutex}; use crate::sys::interop::{ WIZARD_DETECTED_OS_UNKNOWN, WIZARD_DETECTED_OS_WAYLAND, WIZARD_DETECTED_OS_X11, }; use crate::sys::util::convert_to_cstring_or_null; use crate::{ sys::interop::{ WizardMetadata, WIZARD_MIGRATE_RESULT_CLEAN_FAILURE, WIZARD_MIGRATE_RESULT_DIRTY_FAILURE, WIZARD_MIGRATE_RESULT_SUCCESS, WIZARD_MIGRATE_RESULT_UNKNOWN_FAILURE, }, wizard::{WizardHandlers, WizardOptions}, }; lazy_static! { static ref HANDLERS: Mutex> = Mutex::new(None); } pub fn show(options: WizardOptions) -> bool { let c_version = CString::new(options.version).expect("unable to convert version to CString"); let (_c_window_icon_path, c_window_icon_path_ptr) = convert_to_cstring_or_null(options.window_icon_path); let (_c_welcome_image, c_welcome_image_path_ptr) = convert_to_cstring_or_null(options.welcome_image_path); let (_c_accessibility_image_1_path, c_accessibility_image_1_path_ptr) = convert_to_cstring_or_null(options.accessibility_image_1_path); let (_c_accessibility_image_2_path, c_accessibility_image_2_path_ptr) = convert_to_cstring_or_null(options.accessibility_image_2_path); extern "C" fn is_legacy_version_running() -> c_int { let lock = HANDLERS .lock() .expect("unable to acquire lock in is_legacy_version_running method"); let handlers_ref = (*lock).as_ref().expect("unable to unwrap handlers"); if let Some(handler_ref) = handlers_ref.is_legacy_version_running.as_ref() { if (*handler_ref)() { 1 } else { 0 } } else { -1 } } extern "C" fn backup_and_migrate() -> c_int { let lock = HANDLERS .lock() .expect("unable to acquire lock in backup_and_migrate method"); let handlers_ref = (*lock).as_ref().expect("unable to unwrap handlers"); if let Some(handler_ref) = handlers_ref.backup_and_migrate.as_ref() { match (*handler_ref)() { crate::wizard::MigrationResult::Success => WIZARD_MIGRATE_RESULT_SUCCESS, crate::wizard::MigrationResult::CleanFailure => WIZARD_MIGRATE_RESULT_CLEAN_FAILURE, crate::wizard::MigrationResult::DirtyFailure => WIZARD_MIGRATE_RESULT_DIRTY_FAILURE, crate::wizard::MigrationResult::UnknownFailure => WIZARD_MIGRATE_RESULT_UNKNOWN_FAILURE, } } else { WIZARD_MIGRATE_RESULT_UNKNOWN_FAILURE } } extern "C" fn add_to_path() -> c_int { let lock = HANDLERS .lock() .expect("unable to acquire lock in add_to_path method"); let handlers_ref = (*lock).as_ref().expect("unable to unwrap handlers"); if let Some(handler_ref) = handlers_ref.add_to_path.as_ref() { if (*handler_ref)() { 1 } else { 0 } } else { -1 } } extern "C" fn enable_accessibility() -> c_int { let lock = HANDLERS .lock() .expect("unable to acquire lock in enable_accessibility method"); let handlers_ref = (*lock).as_ref().expect("unable to unwrap handlers"); if let Some(handler_ref) = handlers_ref.enable_accessibility.as_ref() { (*handler_ref)(); 1 } else { -1 } } extern "C" fn is_accessibility_enabled() -> c_int { let lock = HANDLERS .lock() .expect("unable to acquire lock in is_accessibility_enabled method"); let handlers_ref = (*lock).as_ref().expect("unable to unwrap handlers"); if let Some(handler_ref) = handlers_ref.is_accessibility_enabled.as_ref() { if (*handler_ref)() { 1 } else { 0 } } else { -1 } } extern "C" fn on_completed() { let lock = HANDLERS .lock() .expect("unable to acquire lock in on_completed method"); let handlers_ref = (*lock).as_ref().expect("unable to unwrap handlers"); if let Some(handler_ref) = handlers_ref.on_completed.as_ref() { (*handler_ref)() } } { let mut lock = HANDLERS.lock().expect("unable to acquire handlers lock"); *lock = Some(options.handlers) } let wizard_metadata = WizardMetadata { version: c_version.as_ptr(), is_welcome_page_enabled: if options.is_welcome_page_enabled { 1 } else { 0 }, is_move_bundle_page_enabled: if options.is_move_bundle_page_enabled { 1 } else { 0 }, is_legacy_version_page_enabled: if options.is_legacy_version_page_enabled { 1 } else { 0 }, is_wrong_edition_page_enabled: if options.is_wrong_edition_page_enabled { 1 } else { 0 }, is_migrate_page_enabled: if options.is_migrate_page_enabled { 1 } else { 0 }, is_add_path_page_enabled: if options.is_add_path_page_enabled { 1 } else { 0 }, is_accessibility_page_enabled: if options.is_accessibility_page_enabled { 1 } else { 0 }, window_icon_path: c_window_icon_path_ptr, welcome_image_path: c_welcome_image_path_ptr, accessibility_image_1_path: c_accessibility_image_1_path_ptr, accessibility_image_2_path: c_accessibility_image_2_path_ptr, detected_os: match options.detected_os { crate::wizard::DetectedOS::Unknown => WIZARD_DETECTED_OS_UNKNOWN, crate::wizard::DetectedOS::X11 => WIZARD_DETECTED_OS_X11, crate::wizard::DetectedOS::Wayland => WIZARD_DETECTED_OS_WAYLAND, }, is_legacy_version_running, backup_and_migrate, add_to_path, enable_accessibility, is_accessibility_enabled, on_completed, }; let successful = unsafe { super::interop::interop_show_wizard(&wizard_metadata) }; successful == 1 }