diff --git a/Cargo.lock b/Cargo.lock index 9a58a42..e925aac 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1064,6 +1064,15 @@ dependencies = [ "cfg-if", ] +[[package]] +name = "malloc_buf" +version = "0.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62bb907fe88d54d8d9ce32a3cceab4218ed2f6b7d35617cafe9adf84e43919cb" +dependencies = [ + "libc", +] + [[package]] name = "maplit" version = "1.0.2" @@ -1199,6 +1208,15 @@ dependencies = [ "libc", ] +[[package]] +name = "objc2" +version = "0.2.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "612c08b52f6d8b11d8c199d0798bc38ab35c0a95da3be548a60808a4ae56d1f4" +dependencies = [ + "malloc_buf", +] + [[package]] name = "object" version = "0.30.3" @@ -1401,7 +1419,7 @@ name = "pod-gui" version = "0.0.0" dependencies = [ "anyhow", - "bitflags 1.3.2", + "bitflags 2.6.0", "clap", "futures", "futures-util", @@ -1409,6 +1427,7 @@ dependencies = [ "log", "maplit", "midir", + "objc2", "once_cell", "pod-core", "pod-gtk", diff --git a/gui/Cargo.toml b/gui/Cargo.toml index 2b59804..b037bae 100644 --- a/gui/Cargo.toml +++ b/gui/Cargo.toml @@ -34,6 +34,9 @@ pod-mod-xt = { path = "../mod-xt" } pod-mod-bassxt = { path = "../mod-bassxt" } pod-usb = { path = "../usb", optional = true } +[target.'cfg(target_os = "macos")'.dependencies] +objc2 = "0.2.7" + [build-dependencies] git-version = "0.3.5" diff --git a/gui/src/main.rs b/gui/src/main.rs index aedc917..e33b2f0 100644 --- a/gui/src/main.rs +++ b/gui/src/main.rs @@ -8,6 +8,7 @@ mod autodetect; mod check; mod icon; mod usb; +mod platform; use std::collections::HashMap; use std::sync::{Arc, atomic, Mutex}; @@ -49,6 +50,7 @@ use crate::util::{next_thread_id, SenderExt as SenderExt2}; use crate::usb::start_usb; use crate::widgets::*; use crate::widgets::templated::Templated; +use crate::platform::*; const MIDI_OUT_CHANNEL_CAPACITY: usize = 512; const CLOSE_QUIET_DURATION_MS: u64 = 1000; @@ -607,6 +609,15 @@ async fn main() -> Result<()> { let opts: Opts = Opts::from_arg_matches(&cli.get_matches())?; drop(help_text); + if let Some(platform) = &opts.platform { + set_platform_hack_flags(&platform)?; + } + let platform_hack_flags = get_platform_hack_flags(); + sentry::configure_scope(|scope| { + scope.set_tag("platform", &platform_hack_flags); + }); + info!("Platform hacks: {}", &platform_hack_flags); + // glib::set_program_name needs to come before gtk::init! glib::set_program_name(Some(&title)); @@ -637,7 +648,7 @@ async fn main() -> Result<()> { } }); - let exit_code = app.run_with_args::(&[]); + let exit_code = app_run(app); // HACK: instead of dealing with USB thread clean-up (and in case there are any other // threads still running), we just call `process::exit` and let libraries clean up // after themselves @@ -1231,7 +1242,7 @@ fn activate(app: >k::Application, title: &String, opts: Opts, sentry_enabled: r.emit_by_name::<()>("group-changed", &[]); // quit - app.quit(); + app_quit(&app); } } @@ -1253,5 +1264,7 @@ fn activate(app: >k::Application, title: &String, opts: Opts, sentry_enabled: // show the window and do init stuff... window.show_all(); + window.present(); + raise_app_window(); window.resize(1, 1); } \ No newline at end of file diff --git a/gui/src/opts.rs b/gui/src/opts.rs index 5f2a320..6f033f8 100644 --- a/gui/src/opts.rs +++ b/gui/src/opts.rs @@ -3,6 +3,7 @@ use anyhow::Result; use std::fmt::Write; use pod_core::config::configs; use pod_core::midi_io::{MidiInPort, MidiOutPort, MidiPorts}; +use crate::get_platform_hack_flags; use crate::usb::*; #[derive(Parser, Clone)] @@ -57,12 +58,22 @@ pub struct Opts { /// instead of triggering any events on an already-running /// pod-ui application. pub standalone: bool, + + #[clap(short, long, value_name = "FLAGS")] + /// Set active platform hack flags. must be a comma-separated + /// list of platform hack names. To enable a specific hack, it should + /// be specified by name (e.g osx-raise). To disable a specific hack, + /// it should be specified with a `no-` prefix (e.g. no-osx-raise). + pub platform: Option, } pub fn generate_help_text() -> Result { let mut s = String::new(); let tab = " "; + writeln!(s, "Default platform hacks (-p): {}", get_platform_hack_flags())?; + writeln!(s, "")?; + writeln!(s, "Device models (-m):")?; for (i, c) in configs().iter().enumerate() { writeln!(s, "{}[{}] {}", tab, i, &c.name)?; diff --git a/gui/src/platform.rs b/gui/src/platform.rs new file mode 100644 index 0000000..a7bbd44 --- /dev/null +++ b/gui/src/platform.rs @@ -0,0 +1,165 @@ +//! A collection of custom platform-specific hacks that do not make sense, +//! but empirically have been shown to be needed. +//! +//! +use anyhow::*; +use std::sync::OnceLock; +use bitflags::{bitflags, Flags}; + +pub use imp::*; + +bitflags! { + #[derive(Clone)] + pub struct PlatformHackFlags: u8 { + const OSX_RAISE = 0x01; + const CUSTOM_MAIN_LOOP = 0x02; + } +} + +// platform-specific default hack flags + +#[cfg(target_os = "linux")] +static DEFAULT_HACK_FLAGS: PlatformHackFlags = PlatformHackFlags::empty(); + +#[cfg(target_os = "macos")] +const DEFAULT_HACK_FLAGS: PlatformHackFlags = + PlatformHackFlags::CUSTOM_MAIN_LOOP.union(PlatformHackFlags::OSX_RAISE); + +#[cfg(target_os = "windows")] +static DEFAULT_HACK_FLAGS: PlatformHackFlags = PlatformHackFlags::empty(); + +pub(in crate::platform) fn platform_hack_flags() -> &'static mut PlatformHackFlags { + static mut FLAGS: OnceLock = OnceLock::new(); + // Safety: this is NOT sound, but actually using this reference as + // mutable is only even used from the CLI-options parsing code, + // well before it is ever read. + unsafe { + FLAGS.get_or_init(|| DEFAULT_HACK_FLAGS); + FLAGS.get_mut().unwrap() + } +} + +pub fn set_platform_hack_flags(flags_string: &str) -> Result { + let flags = platform_hack_flags(); + + for str in flags_string.split(",") { + let (add, str) = match &str[0..3] { + "no-" => (false, &str[3..]), + _ => (true, str) + }; + let name = str.replace("-", "_").to_ascii_uppercase(); + let flag = PlatformHackFlags::from_name(&name) + .ok_or_else(|| anyhow!("Platform hack {str:?}/{name} not found"))?; + flags.set(flag, add); + } + + Ok(flags.clone()) +} + +pub fn get_platform_hack_flags() -> String { + let flags = platform_hack_flags(); + PlatformHackFlags::FLAGS.iter().flat_map(|flag| { + if !flags.contains(flag.value().clone()) { + return None; + } + Some(flag.name().to_ascii_lowercase().replace("_", "-")) + }) + .collect::>() + .join(",") +} + +mod imp { + use pod_gtk::prelude::*; + use std::sync::{Arc, OnceLock}; + use std::sync::atomic::{AtomicBool, Ordering}; + use bitflags::Flags; + use crate::platform::platform_hack_flags; + use crate::PlatformHackFlags; + + // PlatformHackFlags::CUSTOM_MAIN_LOOP + // + // Due to weird UI hang on macOS (see: https://github.com/arteme/pod-ui/issues/62), + // this is a free-form re-implementation of g_application_run() from + // https://github.com/GNOME/glib/blob/9cb0e9464e6117c4ff84b648851c6ee2f5873d1b/gio/gapplication.c#L2591 + // along with a_application_quit(), as there is no way to catch the + // quit signal otherwise. Can be useful on other platforms. + + fn app_running() -> Arc { + static BOOL: OnceLock> = OnceLock::new(); + BOOL.get_or_init(|| Arc::new(AtomicBool::new(true))).clone() + } + + fn custom_app_run(app: gtk::Application) -> i32 { + let context = glib::MainContext::default(); + let _guard = context.acquire().unwrap(); + + let running = app_running(); + + app.register(None::<&gio::Cancellable>).unwrap(); + app.activate(); + + while running.load(Ordering::SeqCst) { + context.iteration(true); + }; + + 0 + } + + fn custom_app_quit() { + app_running().store(false, Ordering::SeqCst); + } + + pub fn app_run(app: gtk::Application) -> i32 { + if platform_hack_flags().contains(PlatformHackFlags::CUSTOM_MAIN_LOOP) { + custom_app_run(app) + } else { + app.run_with_args::(&[]) + } + } + + pub fn app_quit(app: >k::Application) { + if platform_hack_flags().contains(PlatformHackFlags::CUSTOM_MAIN_LOOP) { + custom_app_quit() + } else { + app.quit() + } + } + + // PlatformHackFlags::OSX_RAISE + + #[cfg(target_os = "macos")] + pub fn raise_app_window() { + if platform_hack_flags().contains(PlatformHackFlags::OSX_RAISE) { + osx::raise(); + } + } + + #[cfg(not(target_os = "macos"))] + pub fn raise_app_window() { + // nop + } + + #[allow(non_snake_case)] + #[cfg(target_os = "macos")] + mod osx { + use objc2::runtime::{Object, Class}; + use objc2::{class, msg_send, sel, sel_impl}; + + /** + * On macOS, gtk_window_present doesn't always bring the window to the foreground. + * Based on this SO article, however, we can remedy the issue with some ObjectiveC + * magic: + * https://stackoverflow.com/questions/47497878/gtk-window-present-does-not-move-window-to-foreground + * + * Use objc2 crate to do just that. + */ + pub fn raise() { + unsafe { + let NSApplication = class!(NSApplication); + let app: *mut Object = msg_send![NSApplication, sharedApplication]; + + let _: () = msg_send![app, activateIgnoringOtherApps:true]; + } + } + } +}