generic io interface

This commit is contained in:
jjanzen 2025-01-22 12:17:02 -06:00
parent 275d7f1463
commit 8ecefa4b3f
3 changed files with 80 additions and 41 deletions

View file

@ -0,0 +1,50 @@
const std = @import("std");
const Action = @import("../actions.zig").Action;
pub const IOInterface = struct {
ptr: *anyopaque,
displayMessageFn: *const fn (ptr: *anyopaque, str: []const u8) anyerror!void,
waitForTickFn: *const fn (ptr: *anyopaque) anyerror!Action,
deinitFn: *const fn (ptr: *anyopaque) void,
pub fn init(ptr: anytype) !IOInterface {
const T = @TypeOf(ptr);
const ptr_info = @typeInfo(T);
const gen = struct {
pub fn displayMessage(pointer: *anyopaque, str: []const u8) anyerror!void {
const self: T = @ptrCast(@alignCast(pointer));
return ptr_info.Pointer.child.displayMessage(self, str);
}
pub fn waitForTick(pointer: *anyopaque) anyerror!Action {
const self: T = @ptrCast(@alignCast(pointer));
return ptr_info.Pointer.child.waitForTick(self);
}
pub fn deinit(pointer: *anyopaque) void {
const self: T = @ptrCast(@alignCast(pointer));
return ptr_info.Pointer.child.deinit(self);
}
};
return .{
.ptr = ptr,
.displayMessageFn = gen.displayMessage,
.waitForTickFn = gen.waitForTick,
.deinitFn = gen.deinit,
};
}
pub fn displayMessage(self: IOInterface, str: []const u8) anyerror!void {
return self.displayMessageFn(self.ptr, str);
}
pub fn waitForTick(self: IOInterface) anyerror!Action {
return self.waitForTickFn(self.ptr);
}
pub fn deinit(self: IOInterface) void {
return self.deinitFn(self.ptr);
}
};

View file

@ -1,5 +1,5 @@
//! This module provides an implementation of IO using the ncurses library.
const io_interface = @import("io_interface.zig");
const IOInterface = @import("io_interface.zig").IOInterface;
const std = @import("std");
const Action = @import("../actions.zig").Action;
const ncurses = @cImport({
@ -120,7 +120,7 @@ pub const IO = struct {
/// Display a message in the message box.
/// Takes a pre-formatted null-terminated string
/// If the message is too wide for the box, display "Message too long" instead.
pub fn displayMessage(self: *IO, str: []const u8) error{OutOfMemory}!void {
pub fn displayMessage(self: *IO, str: []const u8) anyerror!void {
if (self.msgs == null) return;
for (1..MESSAGE_PANEL_WIDTH - 1) |i| {
@ -161,7 +161,7 @@ pub const IO = struct {
/// An interface for user input and time processing.
/// Waits for the end of a tick and returns a tick action.
/// If input is given before the end of the tick, return that instead.
pub fn waitForTick(self: *IO) !Action {
pub fn waitForTick(self: *IO) anyerror!Action {
var new = std.time.milliTimestamp();
while (new - self.prev_tick_time <= TICK_MS) {
@ -236,17 +236,17 @@ pub const IO = struct {
);
}
pub fn init(allocator: std.mem.Allocator) !IO {
pub fn init(allocator: std.mem.Allocator) !IOInterface {
_ = locale.setlocale(locale.LC_ALL, "");
var io = IO{
.allocator = allocator,
.inst = null,
.msgs = null,
.stat = null,
.main = null,
.prev_tick_time = std.time.milliTimestamp(),
};
const io_ptr = try allocator.alloc(IO, 1);
var io = &io_ptr[0];
io.allocator = allocator;
io.inst = null;
io.msgs = null;
io.stat = null;
io.main = null;
io.prev_tick_time = std.time.milliTimestamp();
if (ncurses.initscr() == null) {
return error.CursesInitFail;
@ -275,7 +275,7 @@ pub const IO = struct {
try io.createPanels();
}
return io;
return IOInterface.init(io);
}
fn deleteWindows(self: *IO) void {

View file

@ -2,48 +2,37 @@ const std = @import("std");
const IO = @import("frontend/ncurses.zig").IO;
const Action = @import("actions.zig").Action;
pub fn main() u8 {
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
const allocator = gpa.allocator();
var io = IO.init(allocator) catch |err| {
std.log.err("{}", .{err});
return 1;
};
defer io.deinit();
fn run(allocator: std.mem.Allocator) !void {
var io = try IO.init(allocator);
defer {
io.deinit();
}
io.displayMessage("Initialized") catch {
return 1;
};
try io.displayMessage("Initialized");
var action = Action.illegal;
var tick_count: usize = 0;
while (action != Action.exit) {
const str = std.fmt.allocPrint(allocator, "{}", .{tick_count}) catch {
return 1;
};
const str = try std.fmt.allocPrint(allocator, "{}", .{tick_count});
defer allocator.free(str);
io.displayMessage(str) catch {
return 1;
};
try io.displayMessage(str);
action = io.waitForTick() catch {
return 1;
};
action = try io.waitForTick();
switch (action) {
Action.tick => {
const str2 = std.fmt.allocPrint(allocator, "{}", .{tick_count}) catch {
return 1;
};
defer allocator.free(str2);
io.displayMessage(str2) catch {
return 1;
};
tick_count += 1;
},
else => {},
}
}
}
pub fn main() u8 {
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
const allocator = gpa.allocator();
run(allocator) catch {
return 1;
};
return 0;
}