mirror of
https://github.com/XcodesOrg/XcodesApp.git
synced 2026-02-02 11:33:02 +01:00
84 lines
4.5 KiB
Swift
84 lines
4.5 KiB
Swift
import ErrorHandling
|
|
import SwiftUI
|
|
|
|
struct MainWindow: View {
|
|
@EnvironmentObject var appState: AppState
|
|
@State private var selectedXcodeID: Xcode.ID?
|
|
@State private var searchText: String = ""
|
|
@AppStorage("lastUpdated") private var lastUpdated: Double?
|
|
// These two properties should be per-scene state managed by @SceneStorage property wrappers.
|
|
// There's currently a bug with @SceneStorage on macOS, though, where quitting the app will discard the values, which removes a lot of its utility.
|
|
// In the meantime, we're using @AppStorage so that persistence and state restoration works, even though it's not per-scene.
|
|
// FB8979533 SceneStorage doesn't restore value after app is quit by user
|
|
@AppStorage("isShowingInfoPane") private var isShowingInfoPane = false
|
|
@AppStorage("xcodeListCategory") private var category: XcodeListCategory = .all
|
|
|
|
var body: some View {
|
|
HSplitView {
|
|
XcodeListView(selectedXcodeID: $selectedXcodeID, searchText: searchText, category: category)
|
|
.frame(minWidth: 300)
|
|
.layoutPriority(1)
|
|
.alert(item: $appState.xcodeBeingConfirmedForUninstallation) { xcode in
|
|
Alert(title: Text("Uninstall Xcode \(xcode.description)?"),
|
|
message: Text("It will be moved to the Trash, but won't be emptied."),
|
|
primaryButton: .destructive(Text("Uninstall"), action: { self.appState.uninstall(id: xcode.id) }),
|
|
secondaryButton: .cancel(Text("Cancel")))
|
|
}
|
|
InfoPane(selectedXcodeID: selectedXcodeID)
|
|
.frame(minWidth: 300, maxWidth: .infinity)
|
|
.frame(width: isShowingInfoPane ? nil : 0)
|
|
.isHidden(!isShowingInfoPane)
|
|
// This alert isn't intentionally placed here,
|
|
// just trying to put it in a unique part of the hierarchy
|
|
// since you can't have more than one in the same spot.
|
|
.alert(item: $appState.xcodeBeingConfirmedForInstallCancellation) { xcode in
|
|
Alert(title: Text("Are you sure you want to stop the installation of Xcode \(xcode.description)?"),
|
|
message: Text("Any progress will be discarded."),
|
|
primaryButton: .destructive(Text("Stop Installation"), action: { self.appState.cancelInstall(id: xcode.id) }),
|
|
secondaryButton: .cancel(Text("Cancel")))
|
|
}
|
|
}
|
|
.mainToolbar(
|
|
category: $category,
|
|
isShowingInfoPane: $isShowingInfoPane,
|
|
searchText: $searchText
|
|
)
|
|
.navigationSubtitle(subtitleText)
|
|
.frame(minWidth: 600, maxWidth: .infinity, minHeight: 300, maxHeight: .infinity)
|
|
.emittingError($appState.error, recoveryHandler: { _ in })
|
|
.sheet(isPresented: $appState.secondFactorData.isNotNil) {
|
|
secondFactorView(appState.secondFactorData!)
|
|
.environmentObject(appState)
|
|
}
|
|
// I'm expecting to be able to use this modifier on a List row, but using it at the top level here is the only way that has made XcodeCommands work so far.
|
|
// FB8954571 focusedValue(_:_:) on List row doesn't propagate value to @FocusedValue
|
|
.focusedValue(\.selectedXcode, SelectedXcode(appState.allXcodes.first { $0.id == selectedXcodeID }))
|
|
}
|
|
|
|
private var subtitleText: Text {
|
|
if let lastUpdated = lastUpdated.map(Date.init(timeIntervalSince1970:)) {
|
|
return Text("Updated at \(lastUpdated, style: .date) \(lastUpdated, style: .time)")
|
|
} else {
|
|
return Text("")
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
private func secondFactorView(_ secondFactorData: AppState.SecondFactorData) -> some View {
|
|
switch secondFactorData.option {
|
|
case .codeSent:
|
|
SignIn2FAView(isPresented: $appState.secondFactorData.isNotNil, authOptions: secondFactorData.authOptions, sessionData: secondFactorData.sessionData)
|
|
case .smsSent(let trustedPhoneNumber):
|
|
SignInSMSView(isPresented: $appState.secondFactorData.isNotNil, trustedPhoneNumber: trustedPhoneNumber, authOptions: secondFactorData.authOptions, sessionData: secondFactorData.sessionData)
|
|
case .smsPendingChoice:
|
|
SignInPhoneListView(isPresented: $appState.secondFactorData.isNotNil, authOptions: secondFactorData.authOptions, sessionData: secondFactorData.sessionData)
|
|
}
|
|
}
|
|
}
|
|
|
|
struct MainWindow_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
MainWindow()
|
|
}
|
|
}
|