When a second display is connected (macOS), presenting now opens a borderless audience window on the beamer showing the slide, while the main window shows the presenter view (current/next slide, speaker notes, clock, controls) on the laptop. The two windows stay in sync over method channels: navigation, blank screen, audio-complete and beamer clicks are forwarded between them, and media plays only on the beamer to avoid double audio. Falls back to the existing single-window presenter when there is one display or the second window can't be created. - Vendors a fork of desktop_multi_window in third_party/ that re-adds the native macOS window geometry/fullscreen calls (coverScreen, setFrame, close) the published 0.3.0 dropped; wired via a path dependency. - Registers the app's plugins for sub-windows in MainFlutterWindow so video/image rendering works on the beamer. - Routes the multi_window dart entrypoint to a minimal AudienceWindowApp. Compiles (flutter analyze + macOS debug build) and all tests pass; runtime two-screen behaviour still needs verification on real hardware. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
38 lines
1.3 KiB
Dart
38 lines
1.3 KiB
Dart
import 'dart:convert';
|
|
import 'dart:io';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
import 'package:window_manager/window_manager.dart';
|
|
import 'app.dart';
|
|
import 'widgets/presentation/audience_window.dart';
|
|
|
|
void main(List<String> args) async {
|
|
WidgetsFlutterBinding.ensureInitialized();
|
|
|
|
// Secondary windows (e.g. the audience/beamer slide window) are launched by
|
|
// desktop_multi_window with these entrypoint arguments. They run a minimal app
|
|
// and must not touch the main window's window_manager setup.
|
|
if (args.isNotEmpty && args.first == 'multi_window') {
|
|
final raw = args.length >= 3 ? args[2] : '';
|
|
final parsed = raw.isEmpty ? const {} : jsonDecode(raw);
|
|
final map = Map<String, dynamic>.from(parsed as Map);
|
|
runApp(AudienceWindowApp(args: map));
|
|
return;
|
|
}
|
|
|
|
if (!kIsWeb && (Platform.isMacOS || Platform.isWindows || Platform.isLinux)) {
|
|
await windowManager.ensureInitialized();
|
|
const options = WindowOptions(
|
|
minimumSize: Size(1000, 650),
|
|
title: 'OciDeck',
|
|
);
|
|
windowManager.waitUntilReadyToShow(options, () async {
|
|
await windowManager.show();
|
|
await windowManager.focus();
|
|
await windowManager.setPreventClose(true);
|
|
});
|
|
}
|
|
|
|
runApp(const ProviderScope(child: OciDeckApp()));
|
|
}
|