passepartout-apple/Passepartout/Mac/Menu/VPNItemGroup+ViewModel.swift
Davide De Rosa 3f4b55a003 Implement Mac bundle with system menu
Use bundle as a means to provide Mac APIs to Catalyst app.

In order to cross the @objc wall set by the Mac Bundle mechanism,
Swift structures cannot be used directly and must be bridged
through ObjC facades.

Create NSMenu in MVVM style and install it on app launch. Make
sure to do it in AppDelegate.applicationDidFinishLaunching(),
because doing it as early as in PassepartoutApp.init() would
crash Mac code.

Use .representedObject to own view models.

With menu in place, app can be sent to background when main window
is closed. Requires multiple documents support for app not to die
instantly.
2022-07-16 17:39:42 +02:00

82 lines
2.4 KiB
Swift

//
// VPNItemGroup+ViewModel.swift
// Passepartout
//
// Created by Davide De Rosa on 7/3/22.
// Copyright (c) 2022 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 Foundation
import Combine
extension VPNItemGroup {
class ViewModel {
private let vpnManager: LightVPNManager
private let toggleTitleBlock: (Bool) -> String
private let reconnectTitleBlock: () -> String
private var didUpdateState: [(Bool, LightVPNStatus) -> Void] = []
private var subscriptions: Set<AnyCancellable> = []
init(
vpnManager: LightVPNManager,
toggleTitleBlock: @escaping (Bool) -> String,
reconnectTitleBlock: @escaping () -> String
) {
self.vpnManager = vpnManager
self.toggleTitleBlock = toggleTitleBlock
self.reconnectTitleBlock = reconnectTitleBlock
vpnManager.delegate = self
}
var toggleTitle: String {
toggleTitleBlock(vpnManager.isEnabled)
}
var reconnectTitle: String {
reconnectTitleBlock()
}
@objc func toggleVPN() {
vpnManager.toggle()
}
@objc func reconnectVPN() {
vpnManager.reconnect()
}
func subscribeVPNState(_ block: @escaping (Bool, LightVPNStatus) -> Void) {
didUpdateState.append(block)
}
}
}
extension VPNItemGroup.ViewModel: LightVPNManagerDelegate {
func didUpdateState(isEnabled: Bool, vpnStatus: LightVPNStatus) {
didUpdateState.forEach {
$0(isEnabled, vpnStatus)
}
}
}