zig-raylib-db/src/departure.zig

428 lines
21 KiB
Zig
Raw Normal View History

2024-02-17 08:11:11 +01:00
const std = @import("std");
2024-02-17 02:27:03 +01:00
const raylib = @import("raylib.zig");
const rl = raylib.rl;
2024-02-17 03:31:44 +01:00
const AppState = @import("state.zig");
2024-02-17 08:11:11 +01:00
const Curl = @import("curl.zig");
fn fetchThread(state: *AppState) !void {
std.debug.print("[departure/fetchThread] Started\n", .{});
defer std.debug.print("[departure/fetchThread] Ended\n", .{});
defer state.departure_screen_state.fetch_thread = null;
const allocator = state.allocator;
var station_id_buf = std.BoundedArray(u8, 10){};
var include_tram = false;
var curl = Curl.init() orelse return;
defer curl.deinit();
while (state.departure_screen_state.fetch_thread != null) {
2024-02-17 12:00:58 +01:00
var fetch_anyway = state.departure_screen_state.should_refresh;
if (state.departure_screen_state.last_refresh_time + 60000 < std.time.milliTimestamp()) {
fetch_anyway = true;
}
2024-02-17 08:11:11 +01:00
if (!fetch_anyway and std.mem.eql(u8, station_id_buf.slice(), state.departure_screen_state.station_id.items) and include_tram == state.departure_screen_state.include_tram) {
std.time.sleep(100 * 1000);
continue;
}
station_id_buf.resize(state.departure_screen_state.station_id.items.len) catch continue;
std.mem.copyForwards(u8, station_id_buf.slice(), state.departure_screen_state.station_id.items);
include_tram = state.departure_screen_state.include_tram;
std.debug.print("[departure/fetchThread] Detected update: {s}\n", .{station_id_buf.slice()});
curl.reset();
const departures_base = std.fmt.allocPrintZ(
allocator,
"https://v6.db.transport.rest/stops/{s}/departures",
.{state.departure_screen_state.station_id.items},
) catch continue;
defer allocator.free(departures_base);
var departures_uri = std.Uri.parse(departures_base) catch unreachable;
const query = std.fmt.allocPrint(allocator, "duration=300&bus=false&ferry=false&taxi=false&pretty=false{s}", .{if (include_tram) "" else "&tram=false&subway=false"}) catch continue;
defer allocator.free(query);
departures_uri.query = query;
defer departures_uri.query = null;
std.debug.print("[departure/fetchThread] Making request to: {}\n", .{departures_uri});
const url = try std.fmt.allocPrintZ(allocator, "{}", .{departures_uri});
defer allocator.free(url);
_ = curl.setopt(.url, .{url.ptr});
var result = std.ArrayList(u8).init(allocator);
defer result.deinit();
_ = curl.setopt(.write_function, .{Curl.Utils.array_list_append});
_ = curl.setopt(.write_data, .{&result});
const code = curl.perform();
std.debug.print("[departure/fetchThread] cURL Code: {}\n", .{code});
if (code != 0) continue;
std.debug.print("[departure/fetchThread] Fetched data: <redacted>(len: {})\n", .{result.items.len});
const parsed = std.json.parseFromSlice(std.json.Value, allocator, result.items, .{}) catch |err| {
std.debug.print("[departure/fetchThread] JSON parse error: {}\n", .{err});
continue;
};
if (state.departure_screen_state.fetch_result) |old_result| {
old_result.deinit();
}
state.departure_screen_state.fetch_result = parsed;
state.departure_screen_state.should_refresh = false;
2024-02-17 12:00:58 +01:00
state.departure_screen_state.last_refresh_time = std.time.milliTimestamp();
2024-02-17 08:11:11 +01:00
}
if (state.departure_screen_state.fetch_result) |old_result| {
old_result.deinit();
state.departure_screen_state.fetch_result = null;
}
}
2024-02-17 02:27:03 +01:00
2024-02-17 11:42:51 +01:00
fn draw_db1(state: *AppState) !void {
2024-02-17 08:11:11 +01:00
const allocator = state.allocator;
2024-02-17 11:42:51 +01:00
const ds = &state.departure_screen_state;
2024-02-17 02:27:03 +01:00
2024-02-17 08:11:11 +01:00
const db_blue = raylib.ColorInt(0x18226f);
rl.ClearBackground(if (ds.should_refresh) rl.ORANGE else db_blue);
if (ds.fetch_result) |data| {
if (data.value.object.get("departures")) |departures_raw| {
const departures = departures_raw.array.items;
var not_cancelled = std.ArrayList(std.json.Value).init(allocator);
defer not_cancelled.deinit();
for (departures) |d| {
if (d.object.get("cancelled")) |c| {
switch (c) {
.bool => |b| {
if (b) {
continue;
}
},
else => {},
}
}
not_cancelled.append(d) catch continue;
}
if (not_cancelled.items.len > 0) {
var y: c_int = 16;
// Info area
y += 32 + 16;
const first = not_cancelled.items[0].object;
station_name_blk: {
const station_name = std.fmt.allocPrintZ(allocator, "{s}", .{first.get("stop").?.object.get("name").?.string}) catch break :station_name_blk;
defer allocator.free(station_name);
rl.SetWindowTitle(station_name.ptr);
2024-02-17 11:42:51 +01:00
raylib.DrawRightAlignedTextEx(state.font, station_name.ptr, @floatFromInt(rl.GetScreenWidth() - 4), 4, 14, 0.8, rl.WHITE);
2024-02-17 08:11:11 +01:00
}
const line = try std.fmt.allocPrintZ(allocator, "{s}", .{first.get("line").?.object.get("name").?.string});
defer allocator.free(line);
const destination = try std.fmt.allocPrintZ(allocator, "{s}", .{first.get("direction").?.string});
defer allocator.free(destination);
var next_y = y;
2024-02-17 09:51:10 +01:00
next_y += @intFromFloat(raylib.DrawAndMeasureTextEx(state.font, line.ptr, 16, @floatFromInt(y), 32, 1, rl.WHITE).y);
2024-02-17 08:11:11 +01:00
next_y += 16;
if (ds.platform.items.len == 0) blk: {
if (first.get("platform")) |platform_raw| {
switch (platform_raw) {
.string => |p| {
const platform = std.fmt.allocPrintZ(allocator, "{s}", .{p}) catch break :blk;
defer allocator.free(platform);
2024-02-17 11:42:51 +01:00
const platform_width: c_int = @intFromFloat(rl.MeasureTextEx(state.font, platform.ptr, 40, 1).x);
// Check if platform is different
const is_changed = !std.mem.eql(u8, first.get("plannedPlatform").?.string, p);
if (is_changed) {
rl.DrawRectangle(rl.GetScreenWidth() - platform_width - 16 - 8, y, platform_width + 16, 40, rl.WHITE);
}
raylib.DrawRightAlignedTextEx(state.font, platform.ptr, @floatFromInt(rl.GetScreenWidth() - 16), @floatFromInt(y), 40, 1, if (is_changed) db_blue else rl.WHITE);
2024-02-17 08:11:11 +01:00
},
else => {},
}
}
}
y = next_y;
2024-02-17 09:51:10 +01:00
y += @intFromFloat(raylib.DrawAndMeasureTextEx(
state.font,
destination.ptr,
16,
@floatFromInt(y),
56,
1,
rl.WHITE,
).y);
2024-02-17 08:11:11 +01:00
y += 16;
}
if (not_cancelled.items.len > 1) {
var max_trains: c_int = @intCast(not_cancelled.items.len - 1);
if (max_trains > ds.max_next_trains) max_trains = ds.max_next_trains;
const font_size: c_int = 32;
var x: c_int = 16;
var y = rl.GetScreenHeight() - (font_size + 8) * max_trains - 4;
rl.DrawRectangle(0, y, rl.GetScreenWidth(), rl.GetScreenHeight(), rl.WHITE);
y += 8;
2024-02-17 09:51:10 +01:00
const label_measurement_width = @as(c_int, @intFromFloat(raylib.DrawAndMeasureTextEx(
state.font,
2024-02-17 08:11:11 +01:00
if (max_trains == 1) "Next train: " else "Next trains: ",
@floatFromInt(x),
@floatFromInt(y),
@floatFromInt(font_size),
1,
db_blue,
2024-02-17 09:51:10 +01:00
).x));
2024-02-17 08:11:11 +01:00
x += label_measurement_width;
// Compute line name width
var line_name_width: c_int = 0;
2024-02-17 11:42:51 +01:00
var platform_width: c_int = 0;
2024-02-17 08:11:11 +01:00
for (not_cancelled.items, 0..) |dep_raw, idx| {
if (idx == 0) continue;
if (idx > max_trains) break;
const second = dep_raw.object;
const next_train_line = try std.fmt.allocPrintZ(
allocator,
"{s} ",
.{
second.get("line").?.object.get("name").?.string,
},
);
defer allocator.free(next_train_line);
2024-02-17 09:51:10 +01:00
line_name_width = @max(
line_name_width,
@as(c_int, @intFromFloat(rl.MeasureTextEx(state.font, next_train_line.ptr, @floatFromInt(font_size), 1).x)),
);
2024-02-17 11:42:51 +01:00
if (second.get("platform")) |platform_raw| {
switch (platform_raw) {
.string => |p| {
const platform = std.fmt.allocPrintZ(allocator, "{s}", .{p}) catch continue;
defer allocator.free(platform);
platform_width = @max(
platform_width,
@as(c_int, @intFromFloat(rl.MeasureTextEx(state.font, platform.ptr, @floatFromInt(font_size), 1).x)),
);
},
else => {},
}
}
2024-02-17 08:11:11 +01:00
}
const destionation_x = x + line_name_width;
for (not_cancelled.items, 0..) |dep_raw, idx| {
if (idx == 0) continue;
if (idx > max_trains) break;
const second = dep_raw.object;
const next_train_line = try std.fmt.allocPrintZ(
allocator,
"{s} ",
.{
second.get("line").?.object.get("name").?.string,
},
);
defer allocator.free(next_train_line);
const next_train_direction = try std.fmt.allocPrintZ(
allocator,
"{s}",
.{
second.get("direction").?.string,
},
);
defer allocator.free(next_train_direction);
2024-02-17 09:51:10 +01:00
rl.DrawTextEx(state.font, next_train_line.ptr, .{ .x = @floatFromInt(x), .y = @floatFromInt(y) }, font_size, 1, db_blue);
rl.DrawTextEx(state.font, next_train_direction.ptr, .{ .x = @floatFromInt(destionation_x), .y = @floatFromInt(y) }, font_size, 1, db_blue);
2024-02-17 08:11:11 +01:00
if (ds.platform.items.len == 0) blk: {
if (second.get("platform")) |platform_raw| {
switch (platform_raw) {
.string => |p| {
2024-02-17 11:42:51 +01:00
// Check if platform is different
const is_changed = !std.mem.eql(u8, second.get("plannedPlatform").?.string, p);
if (is_changed) {
rl.DrawRectangle(rl.GetScreenWidth() - platform_width - 16 - 8, y, platform_width + 16, font_size, db_blue);
}
2024-02-17 08:11:11 +01:00
const platform = std.fmt.allocPrintZ(allocator, "{s}", .{p}) catch break :blk;
defer allocator.free(platform);
2024-02-17 11:42:51 +01:00
raylib.DrawRightAlignedTextEx(state.font, platform.ptr, @floatFromInt(rl.GetScreenWidth() - 16), @floatFromInt(y), @floatFromInt(font_size), 1, if (is_changed) rl.WHITE else db_blue);
2024-02-17 08:11:11 +01:00
},
else => {},
}
}
}
y += font_size + 4;
rl.DrawLine(x, y, rl.GetScreenWidth() - 8, y, db_blue);
y += 4;
}
}
}
} else {
rl.DrawText(state.departure_screen_state.station_id.items.ptr, 16, 16, 32, rl.WHITE);
}
2024-02-17 11:42:51 +01:00
}
fn draw_ns(state: *AppState) !void {
const allocator = state.allocator;
const ds = &state.departure_screen_state;
_ = .{ allocator, ds };
const ms = std.time.milliTimestamp();
const language = @rem(@divTrunc(ms, 5000), 2);
const ns_bg1 = raylib.ColorInt(0xE6e6e6);
const ns_bg2 = raylib.ColorInt(0xbdd1d9);
const ns_fg1 = raylib.ColorInt(0x1a4379);
const ns_fg2 = raylib.ColorInt(0x6795af);
const ns_fg3 = raylib.ColorInt(0xab5161);
const header_fs: f32 = 16;
const station_fs: f32 = 28;
const cancel_fs: f32 = 24;
const platform_fs: f32 = 28;
rl.ClearBackground(ns_bg1);
const col1w = rl.MeasureTextEx(state.font, "00:00", station_fs, 1).x;
const header_height = rl.MeasureTextEx(state.font, "Vertrek", header_fs, 1).y;
rl.DrawRectangle(0, 0, rl.GetScreenWidth(), 4 + @as(c_int, @intFromFloat(header_height)) + 4, ns_bg2);
raylib.DrawTextEx(state.font, if (language == 0) "Vertrek" else "Depart", 8, 4, header_fs, 1, ns_fg1);
raylib.DrawTextEx(state.font, if (language == 0) "Naar/Opmerking" else "To/Via", 8 + col1w + 8, 4, header_fs, 1, ns_fg1);
raylib.DrawTextEx(state.font, if (language == 0) "Spoor" else "Platform", @floatFromInt(rl.GetScreenWidth() - 200), 4, header_fs, 1, ns_fg1);
var y = header_height + 8 + 2;
if (ds.fetch_result) |data| {
if (data.value.object.get("departures")) |departures_raw| {
const departures = departures_raw.array.items;
for (departures, 0..) |d, idx| {
const line1h = rl.MeasureTextEx(state.font, "00:00", station_fs, 1).y;
const line2h = rl.MeasureTextEx(state.font, "Cancelled", cancel_fs, 1).y;
const total_height = line1h + 4 + line2h + 2;
if (@mod(idx, 2) == 1) {
// Alternate background
rl.DrawRectangle(0, @intFromFloat(y), rl.GetScreenWidth(), @intFromFloat(total_height), ns_bg2);
}
const train = d.object;
const cancelled = blk: {
if (train.get("cancelled")) |cancelled| {
switch (cancelled) {
.bool => |b| {
break :blk b;
},
else => {},
}
}
break :blk false;
};
raylib.DrawTextEx(state.font, "00:00", 8, y, station_fs, 1, if (cancelled) ns_fg2 else ns_fg1);
const direction = try std.fmt.allocPrintZ(
allocator,
"{s}",
.{
train.get("direction").?.string,
},
);
defer allocator.free(direction);
raylib.DrawTextEx(state.font, direction.ptr, 8 + col1w + 8, y, station_fs, 1, if (cancelled) ns_fg2 else ns_fg1);
// Draw platform square
const square_side = total_height - 4;
const sw = rl.GetScreenWidth();
if (train.get("platform")) |platform_raw| {
switch (platform_raw) {
.string => |p| {
const platform = std.fmt.allocPrintZ(allocator, "{s}", .{p}) catch continue;
defer allocator.free(platform);
rl.DrawRectangle(sw - 200, @intFromFloat(y + 2), 12, 12, if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200, @intFromFloat(y + 2), sw - 200, @intFromFloat(y + 2 + square_side), if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2), sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2 + square_side), if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200, @intFromFloat(y + 2), sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2), if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200, @intFromFloat(y + 2 + square_side), sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2 + square_side), if (cancelled) ns_fg2 else ns_fg1);
const text_size = rl.MeasureTextEx(state.font, platform.ptr, platform_fs, 1);
raylib.DrawTextEx(
state.font,
if (cancelled) "-" else platform.ptr,
@as(f32, @floatFromInt(sw - 200)) + @divTrunc(square_side, 2) - (text_size.x / 2),
y + 2 + @divTrunc(square_side, 2) - (text_size.y / 2),
platform_fs,
1,
if (cancelled) ns_fg2 else ns_fg1,
);
},
else => {
if (cancelled) {
rl.DrawRectangle(sw - 200, @intFromFloat(y + 2), 12, 12, if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200, @intFromFloat(y + 2), sw - 200, @intFromFloat(y + 2 + square_side), if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2), sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2 + square_side), if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200, @intFromFloat(y + 2), sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2), if (cancelled) ns_fg2 else ns_fg1);
rl.DrawLine(sw - 200, @intFromFloat(y + 2 + square_side), sw - 200 + @as(c_int, @intFromFloat(square_side)), @intFromFloat(y + 2 + square_side), if (cancelled) ns_fg2 else ns_fg1);
const text_size = rl.MeasureTextEx(state.font, "-", platform_fs, 1);
raylib.DrawTextEx(
state.font,
"-",
@as(f32, @floatFromInt(sw - 200)) + @divTrunc(square_side, 2) - (text_size.x / 2),
y + 2 + @divTrunc(square_side, 2) - (text_size.y / 2),
platform_fs,
1,
if (cancelled) ns_fg2 else ns_fg1,
);
}
},
}
}
y += line1h + 4;
const cancelled_h = raylib.DrawAndMeasureTextEx(state.font, if (cancelled) (if (language == 0) "Rijdt niet" else "Cancelled") else " ", 8 + col1w + 8, y, cancel_fs, 1, ns_fg3).y;
y += cancelled_h + 4;
}
}
}
}
pub fn render(state: *AppState) !void {
var ds = &state.departure_screen_state;
if (ds.fetch_thread == null) {
ds.fetch_thread = std.Thread.spawn(.{}, fetchThread, .{state}) catch null;
}
while (raylib.GetKeyPressed()) |key| {
switch (key) {
rl.KEY_LEFT => {
state.screen = .home;
},
rl.KEY_R => {
ds.should_refresh = true;
},
rl.KEY_MINUS, rl.KEY_KP_SUBTRACT => {
ds.max_next_trains = @max(1, ds.max_next_trains - 1);
},
rl.KEY_EQUAL, rl.KEY_KP_EQUAL => {
ds.max_next_trains = @min(ds.max_next_trains + 1, if (ds.fetch_result) |fr| @as(c_int, @intCast(fr.value.object.get("departures").?.array.items.len)) else 5);
},
rl.KEY_T => {
ds.include_tram = !ds.include_tram;
},
rl.KEY_ONE => {
ds.render_style = .db1;
},
rl.KEY_THREE => {
ds.render_style = .ns;
},
else => {},
}
}
rl.BeginDrawing();
defer rl.EndDrawing();
switch (ds.render_style) {
.db1 => try draw_db1(state),
.ns => try draw_ns(state),
}
2024-02-17 02:27:03 +01:00
state.close_app = rl.WindowShouldClose();
}