mirror of
https://github.com/ghostty-org/ghostty.git
synced 2026-04-06 07:38:21 +00:00
Fixes #11029 (probably) If you renamed the app bundle, the prior check would infinite loop due to the combination of two bugs: invalid termination checks and hardcoding "Ghostty.app"
144 lines
5.4 KiB
Swift
144 lines
5.4 KiB
Swift
import AppKit
|
|
|
|
class DockTilePlugin: NSObject, NSDockTilePlugIn {
|
|
// WARNING: An instance of this class is alive as long as Ghostty's icon is
|
|
// in the doc (running or not!), so keep any state and processing to a
|
|
// minimum to respect resource usage.
|
|
|
|
private let pluginBundle = Bundle(for: DockTilePlugin.self)
|
|
|
|
// Separate defaults based on debug vs release builds so we can test icons
|
|
// without messing up releases.
|
|
#if DEBUG
|
|
private let ghosttyUserDefaults = UserDefaults(suiteName: "com.mitchellh.ghostty.debug")
|
|
#else
|
|
private let ghosttyUserDefaults = UserDefaults(suiteName: "com.mitchellh.ghostty")
|
|
#endif
|
|
|
|
private var iconChangeObserver: Any?
|
|
|
|
/// The URL to the enclosing app bundle, determined from the plugin bundle path.
|
|
var ghosttyAppURL: URL? {
|
|
Self.appBundleURL(for: pluginBundle.bundleURL)
|
|
}
|
|
|
|
/// Determine the enclosing app bundle for the dock tile plugin bundle.
|
|
///
|
|
/// We intentionally avoid matching a specific bundle name (such as
|
|
/// "Ghostty.app") so renaming the app in Finder still works.
|
|
static func appBundleURL(for pluginBundleURL: URL) -> URL? {
|
|
var url = pluginBundleURL
|
|
while true {
|
|
if url.pathExtension.compare("app", options: .caseInsensitive) == .orderedSame {
|
|
return url
|
|
}
|
|
|
|
let parent = url.deletingLastPathComponent()
|
|
if parent.path == url.path {
|
|
// Safety stop: this should only happen at filesystem root.
|
|
return nil
|
|
}
|
|
|
|
url = parent
|
|
}
|
|
}
|
|
|
|
/// The primary NSDockTilePlugin function.
|
|
func setDockTile(_ dockTile: NSDockTile?) {
|
|
// If no dock tile or no access to Ghostty defaults, we can't do anything.
|
|
guard let dockTile, let ghosttyUserDefaults else {
|
|
iconChangeObserver = nil
|
|
return
|
|
}
|
|
|
|
// Try to restore the previous icon on launch.
|
|
iconDidChange(ghosttyUserDefaults.appIcon, dockTile: dockTile)
|
|
|
|
// Setup a new observer for when the icon changes so we can update. This message
|
|
// is sent by the primary Ghostty app.
|
|
iconChangeObserver = DistributedNotificationCenter
|
|
.default()
|
|
.publisher(for: .ghosttyIconDidChange)
|
|
.map { [weak self] _ in self?.ghosttyUserDefaults?.appIcon }
|
|
.receive(on: DispatchQueue.global())
|
|
.sink { [weak self] newIcon in self?.iconDidChange(newIcon, dockTile: dockTile) }
|
|
}
|
|
|
|
private func iconDidChange(_ newIcon: AppIcon?, dockTile: NSDockTile) {
|
|
guard let appIcon = newIcon?.image(in: pluginBundle) else {
|
|
resetIcon(dockTile: dockTile)
|
|
return
|
|
}
|
|
|
|
if let appBundleURL = self.ghosttyAppURL {
|
|
let appBundlePath = appBundleURL.path
|
|
NSWorkspace.shared.setIcon(appIcon, forFile: appBundlePath)
|
|
NSWorkspace.shared.noteFileSystemChanged(appBundlePath)
|
|
}
|
|
|
|
dockTile.setIcon(appIcon)
|
|
}
|
|
|
|
/// Reset the application icon and dock tile icon to the default.
|
|
private func resetIcon(dockTile: NSDockTile) {
|
|
let appBundlePath = self.ghosttyAppURL?.path
|
|
let appIcon: NSImage
|
|
if #available(macOS 26.0, *) {
|
|
// Reset to the default (glassy) icon.
|
|
if let appBundlePath {
|
|
NSWorkspace.shared.setIcon(nil, forFile: appBundlePath)
|
|
}
|
|
|
|
#if DEBUG
|
|
// Use the `Blueprint` icon to distinguish Debug from Release builds.
|
|
appIcon = pluginBundle.image(forResource: "BlueprintImage")!
|
|
#else
|
|
// Get the composed icon from the app bundle.
|
|
if let appBundlePath,
|
|
let iconRep = NSWorkspace.shared.icon(forFile: appBundlePath)
|
|
.bestRepresentation(
|
|
for: CGRect(origin: .zero, size: dockTile.size),
|
|
context: nil,
|
|
hints: nil
|
|
) {
|
|
appIcon = NSImage(size: dockTile.size)
|
|
appIcon.addRepresentation(iconRep)
|
|
} else {
|
|
// If something unexpected happens on macOS 26,
|
|
// fall back to a bundled icon.
|
|
appIcon = pluginBundle.image(forResource: "AppIconImage")!
|
|
}
|
|
#endif
|
|
} else {
|
|
// Use the bundled icon to keep the corner radius consistent with pre-Tahoe apps.
|
|
appIcon = pluginBundle.image(forResource: "AppIconImage")!
|
|
if let appBundlePath {
|
|
NSWorkspace.shared.setIcon(appIcon, forFile: appBundlePath)
|
|
}
|
|
}
|
|
|
|
// Notify Finder/Dock so icon caches refresh immediately.
|
|
if let appBundlePath {
|
|
NSWorkspace.shared.noteFileSystemChanged(appBundlePath)
|
|
}
|
|
dockTile.setIcon(appIcon)
|
|
}
|
|
}
|
|
|
|
private extension NSDockTile {
|
|
func setIcon(_ newIcon: NSImage) {
|
|
// Update the Dock tile on the main thread.
|
|
DispatchQueue.main.async {
|
|
let iconView = NSImageView(frame: CGRect(origin: .zero, size: self.size))
|
|
iconView.wantsLayer = true
|
|
iconView.image = newIcon
|
|
self.contentView = iconView
|
|
self.display()
|
|
}
|
|
}
|
|
}
|
|
|
|
// This is required because of the DispatchQueue call above. This doesn't
|
|
// feel right but I don't know a better way to solve this.
|
|
extension NSDockTile: @unchecked @retroactive Sendable {}
|