macOS: add InternalState to cover migrations

This commit is contained in:
Lukas
2026-04-25 17:56:51 +02:00
parent 5b89671d51
commit bfe07bb99e
4 changed files with 132 additions and 21 deletions

View File

@@ -185,6 +185,7 @@
Features/Terminal/ErrorView.swift,
Features/Terminal/TerminalController.swift,
Features/Terminal/TerminalRestorable.swift,
"Features/Terminal/TerminalRestorableState+InteralState.swift",
Features/Terminal/TerminalTabColor.swift,
Features/Terminal/TerminalView.swift,
Features/Terminal/TerminalViewContainer.swift,

View File

@@ -3,15 +3,23 @@ import Cocoa
struct QuickTerminalRestorableState: TerminalRestorable {
static var version: Int { 1 }
let focusedSurface: String?
let surfaceTree: SplitTree<Ghostty.SurfaceView>
let screenStateEntries: QuickTerminalScreenStateCache.Entries
var focusedSurface: String? {
internalState.focusedSurface
}
var surfaceTree: SplitTree<Ghostty.SurfaceView> {
internalState.surfaceTree
}
var screenStateEntries: QuickTerminalScreenStateCache.Entries {
internalState.screenStateEntries
}
private let internalState: InternalState<Ghostty.SurfaceView>
init(from controller: QuickTerminalController) {
controller.saveScreenState(exitFullscreen: true)
self.focusedSurface = controller.focusedSurface?.id.uuidString
self.surfaceTree = controller.surfaceTree
self.screenStateEntries = controller.screenStateCache.stateByDisplay
self.internalState = .init(from: controller)
}
init(copy other: QuickTerminalRestorableState) {
@@ -24,3 +32,37 @@ struct QuickTerminalRestorableState: TerminalRestorable {
return config
}
}
extension QuickTerminalRestorableState {
/// Internal State we use to perform unit tests
///
/// Since we can't really change the type of `QuickTerminalRestorableState`
/// due to `CodableBridge<QuickTerminalRestorableState>` supporting secure coding,
/// we use an internal type to perform migration and tests
struct InternalState<ViewType: NSView & Codable & Identifiable>: Codable {
// MARK: - Version 1 (1.3.0)
let focusedSurface: String?
let surfaceTree: SplitTree<ViewType>
let screenStateEntries: QuickTerminalScreenStateCache.Entries
init(
focusedSurface: String?,
surfaceTree: SplitTree<ViewType>,
screenStateEntries: QuickTerminalScreenStateCache.Entries,
) {
self.focusedSurface = focusedSurface
self.surfaceTree = surfaceTree
self.screenStateEntries = screenStateEntries
}
}
}
extension QuickTerminalRestorableState.InternalState where ViewType == Ghostty.SurfaceView {
init(from controller: QuickTerminalController) {
self.init(
focusedSurface: controller.focusedSurface?.id.uuidString,
surfaceTree: controller.surfaceTree,
screenStateEntries: controller.screenStateCache.stateByDisplay,
)
}
}

View File

@@ -53,26 +53,49 @@ final class TerminalRestorableState: TerminalRestorable {
static var version: Int { 7 }
static var minimumVersion: Int { 5 }
let focusedSurface: String?
let surfaceTree: SplitTree<Ghostty.SurfaceView>
let effectiveFullscreenMode: FullscreenMode?
let tabColor: TerminalTabColor?
let titleOverride: String?
var focusedSurface: String? {
internalState.focusedSurface
}
var surfaceTree: SplitTree<Ghostty.SurfaceView> {
internalState.surfaceTree
}
var effectiveFullscreenMode: FullscreenMode? {
internalState.effectiveFullscreenMode
}
var tabColor: TerminalTabColor? {
internalState.tabColor
}
var titleOverride: String? {
internalState.titleOverride
}
/// Internal State we use to perform unit tests
///
/// Since we can't really change the type of `TerminalRestorableState`
/// due to `CodableBridge<TerminalRestorableState>` supporting secure coding,
/// we use an internal type to perform migration and tests
private let internalState: InternalState<Ghostty.SurfaceView>
init(from controller: TerminalController) {
self.focusedSurface = controller.focusedSurface?.id.uuidString
self.surfaceTree = controller.surfaceTree
self.effectiveFullscreenMode = controller.fullscreenStyle?.fullscreenMode
self.tabColor = (controller.window as? TerminalWindow)?.tabColor
self.titleOverride = controller.titleOverride
internalState = .init(from: controller)
}
required init(copy other: TerminalRestorableState) {
self.surfaceTree = other.surfaceTree
self.focusedSurface = other.focusedSurface
self.effectiveFullscreenMode = other.effectiveFullscreenMode
self.tabColor = other.tabColor
self.titleOverride = other.titleOverride
self.internalState = other.internalState
}
/// This is just wrapper around internalState
///
/// - Important: If you intend to add more things, go to `InternalState`.
init(from decoder: any Decoder) throws {
self.internalState = try InternalState<Ghostty.SurfaceView>(from: decoder)
}
/// This is just wrapper around internalState
///
/// - Important: If you intend to add more things, go to `InternalState`.
func encode(to encoder: any Encoder) throws {
try internalState.encode(to: encoder)
}
}

View File

@@ -0,0 +1,45 @@
import AppKit
extension TerminalRestorableState {
/// Internal State we use to perform unit tests
///
/// Since we can't really change the type of `TerminalRestorableState`
/// due to `CodableBridge<TerminalRestorableState>` supporting secure coding,
/// we use an internal type to perform migration and tests
struct InternalState<ViewType: NSView & Codable & Identifiable>: Codable {
// MARK: - Version 5 (1.2.3)
let focusedSurface: String?
let surfaceTree: SplitTree<ViewType>
// MARK: - Version 7 (1.3.0)
let effectiveFullscreenMode: FullscreenMode?
let tabColor: TerminalTabColor?
let titleOverride: String?
init(
focusedSurface: String?,
surfaceTree: SplitTree<ViewType>,
effectiveFullscreenMode: FullscreenMode?,
tabColor: TerminalTabColor?,
titleOverride: String?,
) {
self.focusedSurface = focusedSurface
self.surfaceTree = surfaceTree
self.effectiveFullscreenMode = effectiveFullscreenMode
self.tabColor = tabColor
self.titleOverride = titleOverride
}
}
}
extension TerminalRestorableState.InternalState where ViewType == Ghostty.SurfaceView {
init(from controller: TerminalController) {
self.init(
focusedSurface: controller.focusedSurface?.id.uuidString,
surfaceTree: controller.surfaceTree,
effectiveFullscreenMode: controller.fullscreenStyle?.fullscreenMode,
tabColor: (controller.window as? TerminalWindow)?.tabColor,
titleOverride: controller.titleOverride,
)
}
}