diff options
| author | s-ol <s-ol@users.noreply.github.com> | 2020-01-24 14:32:29 +0000 |
|---|---|---|
| committer | s-ol <s-ol@users.noreply.github.com> | 2020-01-25 15:06:20 +0000 |
| commit | 84a3bbf2bffa04d6bb0d3b038473467dd573c9d3 (patch) | |
| tree | 7059479d35048b9cdce421706e2f98ca26c78a28 /src/control.zig | |
| parent | hot reloading (diff) | |
| download | glsl-view-84a3bbf2bffa04d6bb0d3b038473467dd573c9d3.tar.gz glsl-view-84a3bbf2bffa04d6bb0d3b038473467dd573c9d3.zip | |
basic OSC support
Diffstat (limited to 'src/control.zig')
| -rw-r--r-- | src/control.zig | 77 |
1 files changed, 77 insertions, 0 deletions
diff --git a/src/control.zig b/src/control.zig new file mode 100644 index 0000000..4937bb7 --- /dev/null +++ b/src/control.zig @@ -0,0 +1,77 @@ +const c = @import("c.zig"); +const gl = @import("gl.zig"); +const std = @import("std"); + +const param_prefix = "/param/"; + +pub const ControlServer = struct { + server: c.lo_server, + shader: gl.ShaderProgram, + + pub fn init(allocator: *std.mem.Allocator) !*ControlServer { + var self: *ControlServer = try allocator.create(ControlServer); + + self.server = c.lo_server_new("1234", handle_error); + _ = c.lo_server_add_method(self.server, null, null, handle_method, @ptrCast(*c_void, self)); + + return self; + } + + pub fn update(self: *ControlServer, shader: gl.ShaderProgram) void { + self.shader = shader; + while (c.lo_server_recv_noblock(self.server, 0) > 0) {} + } + + pub fn destroy(self: ControlServer) void { + c.lo_server_free(self.server); + } + + fn handle_error(num: c_int, msg: [*c]const u8, where: [*c]const u8) callconv(.C) void { + std.debug.warn("OSC error {} @ {s}: {s}\n", .{ num, @ptrCast([*:0]const u8, where), @ptrCast([*:0]const u8, msg) }); + } + + fn handle_method(_path: [*c]const u8, _types: [*c]const u8, argv: [*c][*c]c.lo_arg, argc: c_int, msg: c.lo_message, userdata: ?*c_void) callconv(.C) c_int { + const self = @ptrCast(*ControlServer, @alignCast(@alignOf(ControlServer), userdata.?)); + const path = _path[0..c.strlen(_path)]; + const types = _types[0..@intCast(u32, argc)]; + + if (std.mem.startsWith(u8, path, param_prefix)) { + const param_name = path[param_prefix.len..]; + + const uniform = self.shader.uniformLocation(@ptrCast([*]const u8, param_name.ptr)) catch { + std.debug.warn("param {s} doesn't exist\n", .{param_name}); + return 0; + }; + + if (argc == 0) { + std.debug.warn("param get: {s}\n", .{param_name}); + return 0; + } + + if (std.mem.eql(u8, types, "f")) { + self.shader.setUniform1f(uniform, argv[0].*.f); + } else if (std.mem.eql(u8, types, "ff")) { + self.shader.setUniform2f(uniform, argv[0].*.f, argv[1].*.f); + } else if (std.mem.eql(u8, types, "fff")) { + self.shader.setUniform3f(uniform, argv[0].*.f, argv[1].*.f, argv[2].*.f); + } else if (std.mem.eql(u8, types, "ffff")) { + self.shader.setUniform4f(uniform, argv[0].*.f, argv[1].*.f, argv[2].*.f, argv[3].*.f); + } else if (std.mem.eql(u8, types, "i")) { + self.shader.setUniform1i(uniform, argv[0].*.i); + } else { + std.debug.warn("unsupported types: {s}\n", .{types}); + } + return 0; + } + + std.debug.warn("unhandled OSC message {s} ({s})\n", .{ path, types }); + return 1; + } + + fn handle_bundle_start(time: c.lo_timetag, userdata: ?*c_void) callconv(.C) c_int { + return 1; + } + fn handle_bundle_end(userdata: ?*c_void) callconv(.C) c_int { + return 1; + } +}; |
