mirror of
https://github.com/ghostty-org/ghostty.git
synced 2026-06-16 16:41:23 +00:00
I mostly did this to familiarize myself with the codebase and figured it doesn't hurt to cover this with tests if I add more in this area, despite receiving indirect coverage elsewhere.
245 lines
7.1 KiB
Zig
245 lines
7.1 KiB
Zig
const std = @import("std");
|
|
const inputpkg = @import("../input.zig");
|
|
const state = &@import("../global.zig").state;
|
|
const c = @import("../main_c.zig");
|
|
|
|
const Config = @import("Config.zig");
|
|
const c_get = @import("c_get.zig");
|
|
const edit = @import("edit.zig");
|
|
const Key = @import("key.zig").Key;
|
|
|
|
const log = std.log.scoped(.config);
|
|
|
|
/// Create a new configuration filled with the initial default values.
|
|
export fn ghostty_config_new() ?*Config {
|
|
const result = state.alloc.create(Config) catch |err| {
|
|
log.err("error allocating config err={}", .{err});
|
|
return null;
|
|
};
|
|
|
|
result.* = Config.default(state.alloc) catch |err| {
|
|
log.err("error creating config err={}", .{err});
|
|
state.alloc.destroy(result);
|
|
return null;
|
|
};
|
|
|
|
return result;
|
|
}
|
|
|
|
export fn ghostty_config_free(ptr: ?*Config) void {
|
|
if (ptr) |v| {
|
|
v.deinit();
|
|
state.alloc.destroy(v);
|
|
}
|
|
}
|
|
|
|
/// Deep clone the configuration.
|
|
export fn ghostty_config_clone(self: *Config) ?*Config {
|
|
const result = state.alloc.create(Config) catch |err| {
|
|
log.err("error allocating config err={}", .{err});
|
|
return null;
|
|
};
|
|
|
|
result.* = self.clone(state.alloc) catch |err| {
|
|
log.err("error cloning config err={}", .{err});
|
|
state.alloc.destroy(result);
|
|
return null;
|
|
};
|
|
|
|
return result;
|
|
}
|
|
|
|
/// Load the configuration from the CLI args.
|
|
export fn ghostty_config_load_cli_args(self: *Config) void {
|
|
self.loadCliArgs(state.alloc) catch |err| {
|
|
log.err("error loading config err={}", .{err});
|
|
};
|
|
}
|
|
|
|
/// Load the configuration from the default file locations. This
|
|
/// is usually done first. The default file locations are locations
|
|
/// such as the home directory.
|
|
export fn ghostty_config_load_default_files(self: *Config) void {
|
|
self.loadDefaultFiles(state.alloc) catch |err| {
|
|
log.err("error loading config err={}", .{err});
|
|
};
|
|
}
|
|
|
|
/// Load the configuration from a specific file path.
|
|
/// The path must be null-terminated.
|
|
export fn ghostty_config_load_file(self: *Config, path: [*:0]const u8) void {
|
|
const path_slice = std.mem.span(path);
|
|
self.loadFile(state.alloc, path_slice) catch |err| {
|
|
log.err("error loading config from file path={s} err={}", .{ path_slice, err });
|
|
};
|
|
}
|
|
|
|
/// Load the configuration from the user-specified configuration
|
|
/// file locations in the previously loaded configuration. This will
|
|
/// recursively continue to load up to a built-in limit.
|
|
export fn ghostty_config_load_recursive_files(self: *Config) void {
|
|
self.loadRecursiveFiles(state.alloc) catch |err| {
|
|
log.err("error loading config err={}", .{err});
|
|
};
|
|
}
|
|
|
|
export fn ghostty_config_finalize(self: *Config) void {
|
|
self.finalize() catch |err| {
|
|
log.err("error finalizing config err={}", .{err});
|
|
};
|
|
}
|
|
|
|
export fn ghostty_config_get(
|
|
self: *Config,
|
|
ptr: *anyopaque,
|
|
key_str: [*]const u8,
|
|
len: usize,
|
|
) bool {
|
|
@setEvalBranchQuota(10_000);
|
|
const key = std.meta.stringToEnum(Key, key_str[0..len]) orelse return false;
|
|
return c_get.get(self, key, ptr);
|
|
}
|
|
|
|
export fn ghostty_config_trigger(
|
|
self: *Config,
|
|
str: [*]const u8,
|
|
len: usize,
|
|
) inputpkg.Binding.Trigger.C {
|
|
return config_trigger_(self, str[0..len]) catch |err| err: {
|
|
log.err("error finding trigger err={}", .{err});
|
|
break :err .{};
|
|
};
|
|
}
|
|
|
|
fn config_trigger_(
|
|
self: *Config,
|
|
str: []const u8,
|
|
) !inputpkg.Binding.Trigger.C {
|
|
const action = try inputpkg.Binding.Action.parse(str);
|
|
const trigger: inputpkg.Binding.Trigger = self.keybind.set.getTrigger(action) orelse .{};
|
|
return trigger.cval();
|
|
}
|
|
|
|
export fn ghostty_config_diagnostics_count(self: *Config) u32 {
|
|
return @intCast(self._diagnostics.items().len);
|
|
}
|
|
|
|
export fn ghostty_config_get_diagnostic(self: *Config, idx: u32) Diagnostic {
|
|
const items = self._diagnostics.items();
|
|
if (idx >= items.len) return .{};
|
|
const message = self._diagnostics.precompute.messages.items[idx];
|
|
return .{ .message = message.ptr };
|
|
}
|
|
|
|
export fn ghostty_config_open_path() c.String {
|
|
const path = edit.openPath(state.alloc) catch |err| {
|
|
log.err("error opening config in editor err={}", .{err});
|
|
return .empty;
|
|
};
|
|
|
|
return .fromSlice(path);
|
|
}
|
|
|
|
/// Sync with ghostty_diagnostic_s
|
|
const Diagnostic = extern struct {
|
|
message: [*:0]const u8 = "",
|
|
};
|
|
|
|
test "ghostty_config_get: bool" {
|
|
const testing = std.testing;
|
|
const alloc = testing.allocator;
|
|
|
|
var cfg = try Config.default(alloc);
|
|
defer cfg.deinit();
|
|
cfg.maximize = true;
|
|
|
|
var out = false;
|
|
const key = "maximize";
|
|
try testing.expect(ghostty_config_get(&cfg, &out, key, key.len));
|
|
try testing.expect(out);
|
|
}
|
|
|
|
test "ghostty_config_get: enum" {
|
|
const testing = std.testing;
|
|
const alloc = testing.allocator;
|
|
|
|
var cfg = try Config.default(alloc);
|
|
defer cfg.deinit();
|
|
cfg.@"window-theme" = .dark;
|
|
|
|
var out: [*:0]const u8 = undefined;
|
|
const key = "window-theme";
|
|
try testing.expect(ghostty_config_get(&cfg, @ptrCast(&out), key, key.len));
|
|
const str = std.mem.sliceTo(out, 0);
|
|
try testing.expectEqualStrings("dark", str);
|
|
}
|
|
|
|
test "ghostty_config_get: optional null returns false" {
|
|
const testing = std.testing;
|
|
const alloc = testing.allocator;
|
|
|
|
var cfg = try Config.default(alloc);
|
|
defer cfg.deinit();
|
|
cfg.@"unfocused-split-fill" = null;
|
|
|
|
var out: Config.Color.C = undefined;
|
|
const key = "unfocused-split-fill";
|
|
try testing.expect(!ghostty_config_get(&cfg, @ptrCast(&out), key, key.len));
|
|
}
|
|
|
|
test "ghostty_config_get: unknown key returns false" {
|
|
const testing = std.testing;
|
|
const alloc = testing.allocator;
|
|
|
|
var cfg = try Config.default(alloc);
|
|
defer cfg.deinit();
|
|
|
|
var out = false;
|
|
const key = "not-a-real-key";
|
|
try testing.expect(!ghostty_config_get(&cfg, &out, key, key.len));
|
|
}
|
|
|
|
test "ghostty_config_get: optional string null returns true" {
|
|
const testing = std.testing;
|
|
const alloc = testing.allocator;
|
|
|
|
var cfg = try Config.default(alloc);
|
|
defer cfg.deinit();
|
|
cfg.title = null;
|
|
|
|
var out: ?[*:0]const u8 = undefined;
|
|
const key = "title";
|
|
try testing.expect(ghostty_config_get(&cfg, @ptrCast(&out), key, key.len));
|
|
try testing.expect(out == null);
|
|
}
|
|
|
|
test "ghostty_config_get: float" {
|
|
const testing = std.testing;
|
|
const alloc = testing.allocator;
|
|
|
|
var cfg = try Config.default(alloc);
|
|
defer cfg.deinit();
|
|
cfg.@"background-opacity" = 0.42;
|
|
|
|
var out: f64 = 0;
|
|
const key = "background-opacity";
|
|
try testing.expect(ghostty_config_get(&cfg, &out, key, key.len));
|
|
try testing.expectApproxEqAbs(@as(f64, 0.42), out, 0.000001);
|
|
}
|
|
|
|
test "ghostty_config_get: struct cval conversion" {
|
|
const testing = std.testing;
|
|
const alloc = testing.allocator;
|
|
|
|
var cfg = try Config.default(alloc);
|
|
defer cfg.deinit();
|
|
cfg.background = .{ .r = 12, .g = 34, .b = 56 };
|
|
|
|
var out: Config.Color.C = undefined;
|
|
const key = "background";
|
|
try testing.expect(ghostty_config_get(&cfg, @ptrCast(&out), key, key.len));
|
|
try testing.expectEqual(@as(u8, 12), out.r);
|
|
try testing.expectEqual(@as(u8, 34), out.g);
|
|
try testing.expectEqual(@as(u8, 56), out.b);
|
|
}
|