mirror of
https://github.com/neon443/ShhShell.git
synced 2026-03-11 13:26:16 +00:00
206 lines
5.7 KiB
Swift
206 lines
5.7 KiB
Swift
//
|
|
// ConnectionView.swift
|
|
// ShhShell
|
|
//
|
|
// Created by neon443 on 20/06/2025.
|
|
//
|
|
|
|
import SwiftUI
|
|
|
|
struct ConnectionView: View {
|
|
@ObservedObject var handler: SSHHandler
|
|
@ObservedObject var hostsManager: HostsManager
|
|
@ObservedObject var keyManager: KeyManager
|
|
|
|
@State private var shellView: ShellView? = nil
|
|
|
|
@State var pubkeyStr: String = ""
|
|
@State var privkeyStr: String = ""
|
|
|
|
@State var showTerminal: Bool = false
|
|
|
|
@State var hostKeyChangedAlert: Bool = false
|
|
|
|
var body: some View {
|
|
ZStack {
|
|
hostsManager.selectedTheme.background.suiColor.opacity(0.7)
|
|
.ignoresSafeArea(.all)
|
|
List {
|
|
Section {
|
|
ScrollView(.horizontal) {
|
|
HStack {
|
|
ForEach(HostSymbol.allCases, id: \.self) { symbol in
|
|
ZStack {
|
|
if handler.host.symbol == symbol {
|
|
RoundedRectangle(cornerRadius: 10)
|
|
.fill(.gray.opacity(0.5))
|
|
}
|
|
SymbolPreview(symbol: symbol, label: handler.host.label)
|
|
.padding(5)
|
|
}
|
|
.frame(width: 60, height: 60)
|
|
.onTapGesture {
|
|
withAnimation { handler.host.symbol = symbol }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
HStack {
|
|
SymbolPreview(symbol: handler.host.symbol, label: handler.host.label)
|
|
.id(handler.host)
|
|
.frame(width: 60, height: 60)
|
|
|
|
TextBox(label: "Icon Text", text: $handler.host.label, prompt: "a few letters in the icon")
|
|
}
|
|
}
|
|
Section {
|
|
Text("\(handler.state)")
|
|
.foregroundStyle(handler.state.color)
|
|
|
|
TextBox(label: "Name", text: $handler.host.name, prompt: "defaults to host address")
|
|
|
|
TextBox(label: "Address", text: $handler.host.address, prompt: "required")
|
|
|
|
TextBox(label: "Port", text: Binding(
|
|
get: { String(handler.host.port) },
|
|
set: {
|
|
if let input = Int($0) {
|
|
handler.host.port = input
|
|
}
|
|
}),
|
|
prompt: "most likely 22",
|
|
keyboardType: .numberPad
|
|
)
|
|
}
|
|
|
|
Section {
|
|
TextBox(label: "Username", text: $handler.host.username, prompt: "required")
|
|
|
|
TextBox(label: "Password", text: $handler.host.password, prompt: "not required if using publickeys", secure: true)
|
|
|
|
Picker("Private key", selection: $handler.host.privateKeyID) {
|
|
Text("None")
|
|
.tag(nil as UUID?)
|
|
ForEach(keyManager.keypairs) { keypair in
|
|
Text(keypair.label)
|
|
.tag(keypair.id as UUID?)
|
|
}
|
|
}
|
|
|
|
TextBox(label: "Publickey", text: $pubkeyStr, prompt: "in openssh format")
|
|
.onChange(of: pubkeyStr) { _ in
|
|
let newStr = pubkeyStr.replacingOccurrences(of: "\r\n", with: "")
|
|
handler.host.publicKey = Data(newStr.utf8)
|
|
}
|
|
.onSubmit {
|
|
let newStr = pubkeyStr.replacingOccurrences(of: "\r\n", with: "")
|
|
handler.host.publicKey = Data(newStr.utf8)
|
|
}
|
|
|
|
TextBox(label: "Privatekey", text: $privkeyStr, prompt: "required if using publickeys", secure: true)
|
|
.onSubmit {
|
|
let newStr = privkeyStr.replacingOccurrences(of: "\r\n", with: "")
|
|
handler.host.privateKey = Data(newStr.utf8)
|
|
}
|
|
.onChange(of: privkeyStr) { _ in
|
|
let newStr = privkeyStr.replacingOccurrences(of: "\r\n", with: "")
|
|
handler.host.privateKey = Data(newStr.utf8)
|
|
}
|
|
|
|
TextBox(label: "Passphrase", text: $handler.host.passphrase, prompt: "optional")
|
|
}
|
|
|
|
Button() {
|
|
showTerminal.toggle()
|
|
} label: {
|
|
Label("Show Terminal", systemImage: "apple.terminal")
|
|
}
|
|
.disabled(!checkShell(handler.state))
|
|
|
|
Button() {
|
|
handler.testExec()
|
|
} label: {
|
|
if let testResult = handler.testSuceeded {
|
|
Image(systemName: testResult ? "checkmark.circle" : "xmark.circle")
|
|
.modifier(foregroundColorStyle(testResult ? .green : .red))
|
|
} else {
|
|
Label("Test Connection", systemImage: "checkmark")
|
|
}
|
|
}
|
|
}
|
|
.scrollContentBackground(.hidden)
|
|
.transition(.opacity)
|
|
.onChange(of: handler.host.key) { _ in
|
|
guard let previousKnownHost = hostsManager.getHostMatching(handler.host) else { return }
|
|
guard handler.host.key == previousKnownHost.key else {
|
|
hostKeyChangedAlert = true
|
|
return
|
|
}
|
|
}
|
|
.onDisappear {
|
|
hostsManager.updateHost(handler.host)
|
|
}
|
|
.task {
|
|
if let publicKeyData = handler.host.publicKey {
|
|
pubkeyStr = String(data: publicKeyData, encoding: .utf8) ?? ""
|
|
}
|
|
if let privateKeyData = handler.host.privateKey {
|
|
privkeyStr = String(data: privateKeyData, encoding: .utf8) ?? ""
|
|
}
|
|
}
|
|
.onAppear {
|
|
if shellView == nil {
|
|
shellView = ShellView(handler: handler, hostsManager: hostsManager)
|
|
}
|
|
}
|
|
.onAppear {
|
|
hostsManager.addHostIfNeeded(handler.host)
|
|
}
|
|
.alert("Hostkey changed", isPresented: $hostKeyChangedAlert) {
|
|
Button("Accept New Hostkey", role: .destructive) {
|
|
hostsManager.updateHost(handler.host)
|
|
handler.go()
|
|
}
|
|
|
|
Button("Disconnect", role: .cancel) {
|
|
handler.disconnect()
|
|
handler.host.key = hostsManager.getHostMatching(handler.host)?.key
|
|
}
|
|
} message: {
|
|
Text("Expected \(handler.host.key ?? "nil")\nbut recieved \(handler.getHostkey() ?? "nil") from the server")
|
|
}
|
|
.toolbar {
|
|
ToolbarItem() {
|
|
Button() {
|
|
handler.go()
|
|
showTerminal = checkShell(handler.state)
|
|
} label: {
|
|
Label(
|
|
handler.connected ? "Disconnect" : "Connect",
|
|
systemImage: handler.connected ? "xmark.app.fill" : "power"
|
|
)
|
|
}
|
|
.disabled(handler.hostInvalid())
|
|
}
|
|
}
|
|
.fullScreenCover(isPresented: $showTerminal) {
|
|
if let shellView {
|
|
shellView
|
|
} else {
|
|
Text("no shellview")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#Preview {
|
|
ConnectionView(
|
|
handler: SSHHandler(host: Host.debug),
|
|
hostsManager: HostsManager(),
|
|
keyManager: KeyManager()
|
|
)
|
|
}
|