const GhosttyResources = @This(); const std = @import("std"); const buildpkg = @import("main.zig"); const Config = @import("Config.zig"); const config_vim = @import("../config/vim.zig"); const config_sublime_syntax = @import("../config/sublime_syntax.zig"); const terminfo = @import("../terminfo/main.zig"); const RunStep = std.Build.Step.Run; steps: []*std.Build.Step, pub fn init(b: *std.Build, cfg: *const Config) !GhosttyResources { var steps = std.ArrayList(*std.Build.Step).init(b.allocator); errdefer steps.deinit(); // Terminfo terminfo: { // Encode our terminfo var str = std.ArrayList(u8).init(b.allocator); defer str.deinit(); try terminfo.ghostty.encode(str.writer()); // Write it var wf = b.addWriteFiles(); const source = wf.add("ghostty.terminfo", str.items); if (cfg.emit_terminfo) { const source_install = b.addInstallFile(source, "share/terminfo/ghostty.terminfo"); try steps.append(&source_install.step); } // Windows doesn't have the binaries below. if (cfg.target.result.os.tag == .windows) break :terminfo; // Convert to termcap source format if thats helpful to people and // install it. The resulting value here is the termcap source in case // that is used for other commands. if (cfg.emit_termcap) { const run_step = RunStep.create(b, "infotocap"); run_step.addArg("infotocap"); run_step.addFileArg(source); const out_source = run_step.captureStdOut(); _ = run_step.captureStdErr(); // so we don't see stderr const cap_install = b.addInstallFile(out_source, "share/terminfo/ghostty.termcap"); try steps.append(&cap_install.step); } // Compile the terminfo source into a terminfo database { const run_step = RunStep.create(b, "tic"); run_step.addArgs(&.{ "tic", "-x", "-o" }); const path = run_step.addOutputFileArg("terminfo"); run_step.addFileArg(source); _ = run_step.captureStdErr(); // so we don't see stderr // Ensure that `share/terminfo` is a directory, otherwise the `cp // -R` will create a file named `share/terminfo` const mkdir_step = RunStep.create(b, "make share/terminfo directory"); switch (cfg.target.result.os.tag) { // windows mkdir shouldn't need "-p" .windows => mkdir_step.addArgs(&.{"mkdir"}), else => mkdir_step.addArgs(&.{ "mkdir", "-p" }), } mkdir_step.addArg(b.fmt("{s}/share/terminfo", .{b.install_path})); try steps.append(&mkdir_step.step); // Use cp -R instead of Step.InstallDir because we need to preserve // symlinks in the terminfo database. Zig's InstallDir step doesn't // handle symlinks correctly yet. const copy_step = RunStep.create(b, "copy terminfo db"); copy_step.addArgs(&.{ "cp", "-R" }); copy_step.addFileArg(path); copy_step.addArg(b.fmt("{s}/share", .{b.install_path})); copy_step.step.dependOn(&mkdir_step.step); try steps.append(©_step.step); } } // Shell-integration { const install_step = b.addInstallDirectory(.{ .source_dir = b.path("src/shell-integration"), .install_dir = .{ .custom = "share" }, .install_subdir = b.pathJoin(&.{ "ghostty", "shell-integration" }), .exclude_extensions = &.{".md"}, }); try steps.append(&install_step.step); } // Themes { const upstream = b.dependency("iterm2_themes", .{}); const install_step = b.addInstallDirectory(.{ .source_dir = upstream.path("ghostty"), .install_dir = .{ .custom = "share" }, .install_subdir = b.pathJoin(&.{ "ghostty", "themes" }), .exclude_extensions = &.{".md"}, }); try steps.append(&install_step.step); } // Fish shell completions { const wf = b.addWriteFiles(); _ = wf.add("ghostty.fish", buildpkg.fish_completions); const install_step = b.addInstallDirectory(.{ .source_dir = wf.getDirectory(), .install_dir = .prefix, .install_subdir = "share/fish/vendor_completions.d", }); try steps.append(&install_step.step); } // zsh shell completions { const wf = b.addWriteFiles(); _ = wf.add("_ghostty", buildpkg.zsh_completions); const install_step = b.addInstallDirectory(.{ .source_dir = wf.getDirectory(), .install_dir = .prefix, .install_subdir = "share/zsh/site-functions", }); try steps.append(&install_step.step); } // bash shell completions { const wf = b.addWriteFiles(); _ = wf.add("ghostty.bash", buildpkg.bash_completions); const install_step = b.addInstallDirectory(.{ .source_dir = wf.getDirectory(), .install_dir = .prefix, .install_subdir = "share/bash-completion/completions", }); try steps.append(&install_step.step); } // Vim plugin { const wf = b.addWriteFiles(); _ = wf.add("syntax/ghostty.vim", config_vim.syntax); _ = wf.add("ftdetect/ghostty.vim", config_vim.ftdetect); _ = wf.add("ftplugin/ghostty.vim", config_vim.ftplugin); _ = wf.add("compiler/ghostty.vim", config_vim.compiler); const install_step = b.addInstallDirectory(.{ .source_dir = wf.getDirectory(), .install_dir = .prefix, .install_subdir = "share/vim/vimfiles", }); try steps.append(&install_step.step); } // Neovim plugin // This is just a copy-paste of the Vim plugin, but using a Neovim subdir. // By default, Neovim doesn't look inside share/vim/vimfiles. Some distros // configure it to do that however. Fedora, does not as a counterexample. { const wf = b.addWriteFiles(); _ = wf.add("syntax/ghostty.vim", config_vim.syntax); _ = wf.add("ftdetect/ghostty.vim", config_vim.ftdetect); _ = wf.add("ftplugin/ghostty.vim", config_vim.ftplugin); _ = wf.add("compiler/ghostty.vim", config_vim.compiler); const install_step = b.addInstallDirectory(.{ .source_dir = wf.getDirectory(), .install_dir = .prefix, .install_subdir = "share/nvim/site", }); try steps.append(&install_step.step); } // Sublime syntax highlighting for bat cli tool // NOTE: The current implementation requires symlinking the generated // 'ghostty.sublime-syntax' file from zig-out to the '~.config/bat/syntaxes' // directory. The syntax then needs to be mapped to the correct language in // the config file within the '~.config/bat' directory // (ex: --map-syntax "/Users/user/.config/ghostty/config:Ghostty Config"). { const wf = b.addWriteFiles(); _ = wf.add("ghostty.sublime-syntax", config_sublime_syntax.syntax); const install_step = b.addInstallDirectory(.{ .source_dir = wf.getDirectory(), .install_dir = .prefix, .install_subdir = "share/bat/syntaxes", }); try steps.append(&install_step.step); } // App (Linux) if (cfg.target.result.os.tag == .linux) { // https://developer.gnome.org/documentation/guidelines/maintainer/integrating.html // Desktop file so that we have an icon and other metadata try steps.append(&b.addInstallFile( b.path("dist/linux/app.desktop"), "share/applications/com.mitchellh.ghostty.desktop", ).step); // Right click menu action for Plasma desktop try steps.append(&b.addInstallFile( b.path("dist/linux/ghostty_dolphin.desktop"), "share/kio/servicemenus/com.mitchellh.ghostty.desktop", ).step); // Right click menu action for Nautilus try steps.append(&b.addInstallFile( b.path("dist/linux/ghostty_nautilus.py"), "share/nautilus-python/extensions/com.mitchellh.ghostty.py", ).step); // Various icons that our application can use, including the icon // that will be used for the desktop. try steps.append(&b.addInstallFile( b.path("images/icons/icon_16.png"), "share/icons/hicolor/16x16/apps/com.mitchellh.ghostty.png", ).step); try steps.append(&b.addInstallFile( b.path("images/icons/icon_32.png"), "share/icons/hicolor/32x32/apps/com.mitchellh.ghostty.png", ).step); try steps.append(&b.addInstallFile( b.path("images/icons/icon_128.png"), "share/icons/hicolor/128x128/apps/com.mitchellh.ghostty.png", ).step); try steps.append(&b.addInstallFile( b.path("images/icons/icon_256.png"), "share/icons/hicolor/256x256/apps/com.mitchellh.ghostty.png", ).step); try steps.append(&b.addInstallFile( b.path("images/icons/icon_512.png"), "share/icons/hicolor/512x512/apps/com.mitchellh.ghostty.png", ).step); // Flatpaks only support icons up to 512x512. if (!cfg.flatpak) { try steps.append(&b.addInstallFile( b.path("images/icons/icon_1024.png"), "share/icons/hicolor/1024x1024/apps/com.mitchellh.ghostty.png", ).step); } try steps.append(&b.addInstallFile( b.path("images/icons/icon_16@2x.png"), "share/icons/hicolor/16x16@2/apps/com.mitchellh.ghostty.png", ).step); try steps.append(&b.addInstallFile( b.path("images/icons/icon_32@2x.png"), "share/icons/hicolor/32x32@2/apps/com.mitchellh.ghostty.png", ).step); try steps.append(&b.addInstallFile( b.path("images/icons/icon_128@2x.png"), "share/icons/hicolor/128x128@2/apps/com.mitchellh.ghostty.png", ).step); try steps.append(&b.addInstallFile( b.path("images/icons/icon_256@2x.png"), "share/icons/hicolor/256x256@2/apps/com.mitchellh.ghostty.png", ).step); } return .{ .steps = steps.items }; } pub fn install(self: *const GhosttyResources) void { const b = self.steps[0].owner; for (self.steps) |step| b.getInstallStep().dependOn(step); }