Ocideck/third_party/desktop_multi_window/linux/flutter_window.cc
Brenno de Winter ffcda70966 Extend dual-screen presenter to Windows and Linux
Bring the second-window (beamer) presenter mode to all desktop platforms,
not just macOS:

- Implement the native window_coverScreen / window_close methods for the
  vendored desktop_multi_window plugin on Windows (borderless popup over
  the presentation monitor) and Linux.
- Register the app's plugins for sub-windows in the Windows and Linux
  runners, so video/image rendering works in the audience window there too.
- Gate dual-screen mode through a testable shouldUseDualScreen() helper
  (any desktop platform with >= 2 displays) and cover it with tests.

flutter analyze is clean and all presenter tests pass. Runtime two-screen
behaviour still needs verification on real hardware.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-06 22:03:56 +02:00

120 lines
3.9 KiB
C++
Executable file

#include "flutter_window.h"
#include <cstring>
#include <iostream>
namespace {
bool ReadExternalArgument(FlValue* arguments) {
if (arguments == nullptr ||
fl_value_get_type(arguments) != FL_VALUE_TYPE_MAP) {
return true;
}
FlValue* external = fl_value_lookup_string(arguments, "external");
if (external == nullptr ||
fl_value_get_type(external) != FL_VALUE_TYPE_BOOL) {
return true;
}
return fl_value_get_bool(external);
}
gboolean CloseWindowOnIdle(gpointer data) {
GtkWidget* window = GTK_WIDGET(data);
gtk_widget_destroy(window);
g_object_unref(window);
return G_SOURCE_REMOVE;
}
} // namespace
FlutterWindow::FlutterWindow(const std::string& id,
const std::string& argument,
GtkWidget* window)
: id_(id), window_argument_(argument), window_(window) {}
FlutterWindow::~FlutterWindow() = default;
void FlutterWindow::SetChannel(FlMethodChannel* channel) {
channel_ = channel;
}
void FlutterWindow::NotifyWindowEvent(const gchar* event, FlValue* data) {
if (channel_) {
fl_method_channel_invoke_method(channel_, event, data, nullptr, nullptr, nullptr);
}
}
void FlutterWindow::Show() {
if (window_) {
gtk_widget_show(GTK_WIDGET(window_));
}
}
void FlutterWindow::Hide() {
if (window_) {
gtk_widget_hide(GTK_WIDGET(window_));
}
}
void FlutterWindow::HandleWindowMethod(const gchar* method,
FlValue* arguments,
FlMethodCall* method_call) {
g_autoptr(FlMethodResponse) response = nullptr;
if (strcmp(method, "window_show") == 0) {
Show();
response = FL_METHOD_RESPONSE(fl_method_success_response_new(nullptr));
} else if (strcmp(method, "window_hide") == 0) {
Hide();
response = FL_METHOD_RESPONSE(fl_method_success_response_new(nullptr));
} else if (strcmp(method, "window_close") == 0) {
response = FL_METHOD_RESPONSE(fl_method_success_response_new(nullptr));
if (window_) {
g_idle_add(CloseWindowOnIdle, g_object_ref(window_));
}
} else if (strcmp(method, "window_coverScreen") == 0) {
if (!window_) {
response = FL_METHOD_RESPONSE(
fl_method_error_response_new("-1", "window is not available",
nullptr));
} else {
GtkWindow* window = GTK_WINDOW(window_);
GdkScreen* screen = gtk_window_get_screen(window);
GdkWindow* gdk_window = gtk_widget_get_window(window_);
const gint monitor_count = gdk_screen_get_n_monitors(screen);
gint current_monitor = gdk_window
? gdk_screen_get_monitor_at_window(screen,
gdk_window)
: gdk_screen_get_primary_monitor(screen);
if (current_monitor < 0) {
current_monitor = 0;
}
gint target_monitor = current_monitor;
if (ReadExternalArgument(arguments) && monitor_count > 1) {
for (gint i = 0; i < monitor_count; ++i) {
if (i != current_monitor) {
target_monitor = i;
break;
}
}
}
GdkRectangle bounds;
gdk_screen_get_monitor_geometry(screen, target_monitor, &bounds);
gtk_window_unfullscreen(window);
gtk_window_set_decorated(window, FALSE);
gtk_window_move(window, bounds.x, bounds.y);
gtk_window_resize(window, bounds.width, bounds.height);
gtk_window_fullscreen_on_monitor(window, screen, target_monitor);
gtk_widget_show(window_);
response = FL_METHOD_RESPONSE(fl_method_success_response_new(nullptr));
}
} else {
g_autofree gchar* error_msg = g_strdup_printf("unknown method: %s", method);
response = FL_METHOD_RESPONSE(
fl_method_error_response_new("-1", error_msg, nullptr));
}
fl_method_call_respond(method_call, response, nullptr);
}