passepartout-apple/Passepartout/App/PassepartoutApp.swift
Davide 41de48789e
Show in Mac status bar plus Login Item (#773)
Add a status menu via SwiftUI MenuBarExtra where to:

- Show/hide app
- Launch on login via "Login Item" target
- Toggle profiles on/off

Only weird that the login item is not added to the list of "Open at
Login", but to "Allow in the Background", see
https://github.com/pilotmoon/Scroll-Reverser/issues/165

Requires some refactoring to bring AppContext initialization to the
AppDelegate.

Fixes #617
Fixes #482 
Fixes #696 
Fixes #505
2024-10-29 11:40:11 +01:00

110 lines
3.0 KiB
Swift

//
// PassepartoutApp.swift
// Passepartout
//
// Created by Davide De Rosa on 2/22/24.
// Copyright (c) 2024 Davide De Rosa. All rights reserved.
//
// https://github.com/passepartoutvpn
//
// This file is part of Passepartout.
//
// Passepartout 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.
//
// Passepartout 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 Passepartout. If not, see <http://www.gnu.org/licenses/>.
//
import AppUI
import CommonLibrary
import PassepartoutKit
import SwiftUI
@main
struct PassepartoutApp: App {
#if os(iOS)
@UIApplicationDelegateAdaptor
private var appDelegate: AppDelegate
#else
@NSApplicationDelegateAdaptor
private var appDelegate: AppDelegate
#endif
@Environment(\.scenePhase)
private var scenePhase
private var context: AppContext {
appDelegate.context
}
private let appName = BundleConfiguration.mainDisplayName
@StateObject
private var theme = Theme()
#if os(iOS)
var body: some Scene {
WindowGroup {
contentView()
.onOpenURL { url in
ImporterPipe.shared.send([url])
}
}
}
#else
var body: some Scene {
Window(appName, id: appName, content: contentView)
.defaultSize(width: 600, height: 400)
Settings {
SettingsView(profileManager: context.profileManager)
.frame(minWidth: 300, minHeight: 200)
}
MenuBarExtra {
AppMenu()
.withEnvironment(from: context, theme: theme)
} label: {
AppMenuImage(connectionObserver: context.connectionObserver)
.environmentObject(theme)
}
}
#endif
}
private extension PassepartoutApp {
func contentView() -> some View {
AppCoordinator(
profileManager: context.profileManager,
tunnel: context.tunnel,
registry: context.registry
)
.onChange(of: scenePhase) {
switch $0 {
case .active:
Task {
do {
pp_log(.app, .notice, "Prepare tunnel and purge stale data")
try await context.tunnel.prepare(purge: true)
} catch {
pp_log(.app, .fault, "Unable to prepare tunnel: \(error)")
}
}
default:
break
}
}
.themeLockScreen()
.withEnvironment(from: context, theme: theme)
}
}