Skip to content

Commit d915e17

Browse files
authored
Prototype scaled dungeon w anim background (#3)
* Prototype of scaled background, a dungeon twice as large. Additionally, working on background animation so the field doesn't fill like it's simply floating * A nice looking starfield is rendering for the background layer. * More comments, added a BoundedArrayQueue generic over a fixed size array and also added more code enhancements
1 parent be3d903 commit d915e17

16 files changed

+271
-99
lines changed

README.md

+14-13
Original file line numberDiff line numberDiff line change
@@ -20,14 +20,14 @@ This is a near exact **Zig port** of the [original DungeonRush `C-based`](https:
2020
* 8-bit style music and many sound sfx
2121

2222
## How to play
23-
* Each round starts with a random hero in the middle
24-
* Move the arrow keys: up, down, left, right to control your character
23+
* Each round starts with a random hero in the middle of the dungeon
24+
* Move the arrow keys: `up`, `down`, `left`, `right` to control your character
2525
* Make your party stronger by collecting more heroes
26-
* Collect weapon and loot drops to become more powerful and heal up
27-
* Don't run into walls, bad guys or yourself
28-
* Your heroes will intelligently attack nearby enemies
29-
* To advance to the next round, you must collect a certain amount of heros
30-
* Be careful of giant bosses, they are very strong
26+
* Collect weapon and loot drops to become more powerful and heal your party
27+
* Don't run into walls, bad guys or yourself or you will 💀
28+
* Your heroes will intelligently attack nearby enemies when they are near
29+
* To advance to the next round, you must collect a certain amount of heroes
30+
* Be careful of giant bosses, they can be very powerful
3131

3232
## Port Goals
3333
* To re-create a moderately complex game, fully in Zig and to get better at the language.
@@ -40,7 +40,7 @@ This is a near exact **Zig port** of the [original DungeonRush `C-based`](https:
4040
1. Phase 1: port the C code almost as-is to minimize bugs introduced
4141
* ✅ Deprecate usage of `c.malloc`/`c.free` in all cases
4242
* ✅ Deprecate use of `c.qsort` with a `callconv(.C)` callback in favor of Zig's sorting
43-
* Deprecate all [*c] style pointers
43+
* Deprecate all [*c] style pointers
4444
* ✅ Deprecate C-style multi-pointers
4545
* Find and improve `const` correctness where applicable
4646
2. Phase 2: Ziggify
@@ -50,23 +50,22 @@ This is a near exact **Zig port** of the [original DungeonRush `C-based`](https:
5050
* Utilize `defer`/`errdefer` for effective cleanup
5151
* Migrate to a Zig-based SDL wrapper, for a nicer SDL experience
5252
* Ensure all errors are accounted for, utilize `try`
53-
* Use build.zig.zon
53+
* Use `build.zig.zon` file for any dependencies
5454
* ✅ Setup Github to build the project regularly
5555
3. Phase 3: Code Clean-up/Refactor
5656
* ✅ Gamepad controller support added
5757
* Remove duplicate code
5858
* Make code even more idiomatic for Zig
5959
* Make the code more maintainable
6060
* Use less globals
61-
* Fix namespace issues
6261
* Remove redundant naming like some enumerations have their container name as the prefix
6362
* ✅ Use some Zig based collections like the `generic` LinkList over the original C ADT style
64-
* Bonus: Introduce unit-tests
6563
* Get building for other OSes (w/ community contributions)
6664
* Migrate hardcoded textures, music + sound sfx out of the source code and into config files
6765
* This will allow the game to be easily skinned, for a whole new experience.
6866
4. Phase 4: ???
6967
* I'd love to port this to Raylib.
68+
* Bonus: Introduce unit-tests
7069

7170
## More baddies?
7271
* [pixel-sprite-mixer](https://kingbell.itch.io/pixel-sprite-mixer)
@@ -104,5 +103,7 @@ dumb shit like that.
104103
* This would make the game a little more interesting because you can pick up cerrtain heroes
105104
as part of the strategy to help you cope with the certain level.
106105
* Introduce themed levels, like a frost level where most or all baddies are cold-based.
107-
* Introduce upgrades along the way, pick up treason, loot upgrade your heroes.
108-
* Gamepad support, it might be fun to not always use the damn keyboard.
106+
* Introduce upgrades along the way, loot upgrade your heroes.
107+
* ✅ Gamepad support, it might be fun to not always use the damn keyboard.
108+
* Weather effects, snow, rain, wind, fog-of-war, etc.
109+
* Show temporary animated scores when certain events occur like picking up loot

build.zig

+27
Original file line numberDiff line numberDiff line change
@@ -11,14 +11,41 @@ pub fn build(b: *std.Build) void {
1111
.optimize = optimize,
1212
});
1313

14+
//exe.linkSystemLibrary("objc");
15+
// exe.linkFramework("Cocoa");
16+
// exe.linkFramework("CoreAudio");
17+
// exe.linkFramework("Carbon");
18+
// exe.linkFramework("Metal");
19+
// exe.linkFramework("QuartzCore");
20+
// exe.linkFramework("AudioToolbox");
21+
// exe.linkFramework("ForceFeedback");
22+
// exe.linkFramework("GameController");
23+
// exe.linkFramework("CoreHaptics");
24+
// exe.linkFramework("IOKit");
25+
26+
// exe.linkSystemLibrary("iconv");
27+
1428
// Link SDL2 and friends.
1529
exe.linkSystemLibrary("SDL2");
1630
exe.linkSystemLibrary("SDL2_mixer");
1731
exe.linkSystemLibrary("SDL2_image");
1832
exe.linkSystemLibrary("SDL2_ttf");
1933
exe.linkSystemLibrary("SDL2_net");
2034

35+
// exe.linkSystemLibrary2("SDL2", .{ .preferred_link_mode = .static });
36+
// exe.linkSystemLibrary2("SDL2_mixer", .{ .preferred_link_mode = .static });
37+
// exe.linkSystemLibrary2("SDL2_image", .{ .preferred_link_mode = .static });
38+
// exe.linkSystemLibrary2("SDL2_ttf", .{ .preferred_link_mode = .static });
39+
// exe.linkSystemLibrary2("SDL2_net", .{ .preferred_link_mode = .static });
40+
41+
// exe.linkSystemLibrary("freetype");
42+
// exe.linkSystemLibrary("harfbuzz");
43+
// exe.linkSystemLibrary("bz2");
44+
// exe.linkSystemLibrary("zlib");
45+
// exe.linkSystemLibrary("graphite2");
46+
2147
// Link C
48+
//exe.linkSystemLibrary2("c", .{ .preferred_link_mode = .static });
2249
exe.linkSystemLibrary("c");
2350

2451
// This declares intent for the executable to be installed into the

res/drawable/starfield

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
purple_staff 0 0 320 320 1

res/drawable/starfield.png

41 KB
Loading

zsrc/ai.zig

+2-2
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,7 @@ pub fn getPowerfulPlayer() c_int {
7474
var id: c_int = -1;
7575

7676
for (0..@intCast(gm.playersCount)) |i| {
77-
const num = gm.spriteSnake[i].?.num;
77+
const num = gm.spriteSnake[i].?.num();
7878
if (num > maxNum) {
7979
maxNum = num;
8080
mxCount = 1;
@@ -85,7 +85,7 @@ pub fn getPowerfulPlayer() c_int {
8585
}
8686

8787
if (id != -1 and mxCount == 1) {
88-
if (@as(f64, @floatFromInt(gm.spriteSnake[@intCast(id)].?.num)) >= AI_LOCK_LIMIT) {
88+
if (@as(f64, @floatFromInt(gm.spriteSnake[@intCast(id)].?.num())) >= AI_LOCK_LIMIT) {
8989
return id;
9090
} else {
9191
return -1;

zsrc/audio.zig

+4-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,8 @@ const BGM_FADE_DURATION = 800;
3030
var nowBgmId: ?usize = null;
3131

3232
pub fn playBgm(id: usize) void {
33-
if (id == nowBgmId) {
33+
// NOTE: temporary disabling music, as it's getting old.
34+
if (true or id == nowBgmId) {
3435
return;
3536
}
3637

@@ -48,6 +49,8 @@ pub fn stopBgm() void {
4849
nowBgmId = null;
4950
}
5051

52+
/// randomBgm selects a random background music track from 1.. because track
53+
/// 0 is reserved for the game menu.
5154
pub fn randomBgm() void {
5255
const r: usize = @intCast(hlp.randInt(1, res.bgmsPath.len - 1));
5356
playBgm(r);

zsrc/boundedarrayqueue.zig

+95
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
const std = @import("std");
2+
3+
/// This is a generic queue built on top of a fixed size array.
4+
pub fn BoundedArrayQueue(comptime Child: type, comptime bufferSize: usize) type {
5+
return struct {
6+
const Self = @This();
7+
8+
buffer: [bufferSize]Child = undefined,
9+
count: usize,
10+
11+
/// Creates and initializes an empty queue structure.
12+
pub fn init() Self {
13+
return .{ .count = 0 };
14+
}
15+
16+
/// Only peeks at the top of the queue, but this doesn't do any
17+
/// modification to the queue itself.
18+
pub fn peek(self: *const Self) ?Child {
19+
if (self.count > 0) {
20+
return self.buffer[0];
21+
}
22+
return null;
23+
}
24+
25+
/// This enqueues the value to be at the head of the queue.
26+
pub fn enqueue(self: *Self, value: Child) void {
27+
std.debug.assert(self.count < self.buffer.len);
28+
self.buffer[self.count] = value;
29+
self.count += 1;
30+
}
31+
32+
/// Does the work of shifting over the elements
33+
/// in the queue and returns what was at the top.
34+
pub fn dequeue(self: *Self) ?Child {
35+
if (self.count > 0) {
36+
return self.popAndSlideOne();
37+
}
38+
return null;
39+
}
40+
41+
/// Does the primary work of shifting everything over
42+
/// and returning whatever was the first out.
43+
fn popAndSlideOne(self: *Self) Child {
44+
const item = self.buffer[0];
45+
self.count -= 1;
46+
47+
for (0..self.count + 1) |idx| {
48+
self.buffer[idx] = self.buffer[idx + 1];
49+
}
50+
51+
return item;
52+
}
53+
};
54+
}
55+
56+
fn testQ() void {
57+
var q = BoundedArrayQueue(u8, 10).init();
58+
q.enqueue(8);
59+
q.enqueue(6);
60+
q.enqueue(7);
61+
q.enqueue(5);
62+
q.enqueue(3);
63+
q.enqueue(0);
64+
q.enqueue(9);
65+
66+
std.debug.assert(q.count == 7);
67+
std.debug.assert(q.peek() == 8);
68+
69+
var r = q.dequeue();
70+
std.debug.assert(r == 8);
71+
72+
r = q.dequeue();
73+
std.debug.assert(r == 6);
74+
75+
r = q.dequeue();
76+
std.debug.assert(r == 7);
77+
78+
r = q.dequeue();
79+
std.debug.assert(r == 5);
80+
81+
r = q.dequeue();
82+
std.debug.assert(r == 3);
83+
84+
r = q.dequeue();
85+
std.debug.assert(r == 0);
86+
87+
r = q.dequeue();
88+
std.debug.assert(r == 9);
89+
90+
r = q.dequeue();
91+
std.debug.assert(r == null);
92+
std.debug.assert(q.count == 0);
93+
94+
std.process.exit(0);
95+
}

zsrc/bullet.zig

+1-1
Original file line numberDiff line numberDiff line change
@@ -71,7 +71,7 @@ pub const Bullet = struct {
7171
self.ani.y = self.y;
7272
}
7373

74-
pub fn deinit(self: *Bullet) void {
74+
pub fn deinit(self: *const Bullet) void {
7575
gAllocator.destroy(self);
7676
}
7777
};

zsrc/game.zig

+16-17
Original file line numberDiff line numberDiff line change
@@ -173,7 +173,7 @@ pub fn appendSpriteToSnake(
173173
y: c_int,
174174
direction: tps.Direction,
175175
) void {
176-
snake.num += 1;
176+
//snake.num += 1;
177177
snake.score.got += 1;
178178
var newX = x;
179179
var newY = y;
@@ -531,7 +531,7 @@ fn makeSnakeAttack(snake: *pl.Snake) void {
531531

532532
fn isWin() bool {
533533
if (playersCount != 1) return false;
534-
return spriteSnake[0].?.num >= GAME_WIN_NUM;
534+
return spriteSnake[0].?.num() >= GAME_WIN_NUM;
535535
}
536536

537537
const GameStatus = enum {
@@ -1175,7 +1175,8 @@ fn initGame(localPlayers: c_int, remotePlayers: c_int, localFirst: bool) !void {
11751175
}
11761176
try ren.initInfo();
11771177
// create map
1178-
mp.initRandomMap(0.7, 7, GAME_TRAP_RATE);
1178+
//mp.initRandomMap(0.7, 7, GAME_TRAP_RATE); // Original
1179+
mp.initRandomMap(0.6, 3, GAME_TRAP_RATE);
11791180

11801181
clearItemMap();
11811182

@@ -1489,7 +1490,6 @@ fn makeSnakeCross(snake: *pl.Snake) bool {
14891490
// So it goes from: {character-type}_run_anim -> {character-type}_hit_anim
14901491
// NOTE: Only some characters have the _hit_anim
14911492
var deathPtr: usize = @intFromPtr(sprite.ani.origin);
1492-
14931493
if (isPlayer(snake)) deathPtr += (@sizeOf(tps.Texture) * 1);
14941494

14951495
dropItem(sprite);
@@ -1543,7 +1543,7 @@ fn makeSnakeCross(snake: *pl.Snake) bool {
15431543
ren.clearBindInAnimationsList(sprite, ren.RENDER_LIST_SPRITE_ID);
15441544
tps.removeAnimationFromLinkList(&ren.animationsList[ren.RENDER_LIST_SPRITE_ID], sprite.ani);
15451545
//sprite.ani = null;
1546-
snake.num -= 1;
1546+
//snake.num -= 1;
15471547
}
15481548
}
15491549
}
@@ -1567,7 +1567,8 @@ fn makeSnakeCross(snake: *pl.Snake) bool {
15671567
const currSprite: *spr.Sprite = @alignCast(@ptrCast(q.?.data));
15681568
currSprite.direction = prevSprite.direction;
15691569
currSprite.face = prevSprite.face;
1570-
currSprite.posBuffer = prevSprite.posBuffer;
1570+
//currSprite.posBuffer = prevSprite.posBuffer;
1571+
currSprite.posQueue = prevSprite.posQueue;
15711572
currSprite.x = prevSprite.x;
15721573
currSprite.y = prevSprite.y;
15731574
}
@@ -1746,17 +1747,15 @@ fn moveSnake(snake: *pl.Snake) void {
17461747
const sprite: *spr.Sprite = @ptrCast(@alignCast(node.data));
17471748

17481749
for (0..@intCast(step)) |_| {
1749-
const b = &sprite.posBuffer;
1750-
var firstSlot = b.buffer[0];
1751-
1752-
while (b.size > 0 and sprite.x == firstSlot.x and sprite.y == firstSlot.y) {
1753-
tps.changeSpriteDirection(node, firstSlot.direction);
1754-
b.size -= 1;
1755-
for (0..@intCast(b.size)) |i| {
1756-
b.buffer[i] = b.buffer[i + 1];
1757-
}
1758-
firstSlot = b.buffer[0];
1750+
const b = &sprite.posQueue;
1751+
var firstSlot = b.peek();
1752+
1753+
while (firstSlot != null and b.count > 0 and sprite.x == firstSlot.?.x and sprite.y == firstSlot.?.y) {
1754+
tps.changeSpriteDirection(node, firstSlot.?.direction);
1755+
_ = b.dequeue();
1756+
firstSlot = b.peek();
17591757
}
1758+
17601759
moveSprite(sprite, 1);
17611760
}
17621761
}
@@ -1890,7 +1889,7 @@ fn gameLoop() !GameStatus {
18901889
// Cull snakes that have no more soldiers.
18911890
var i: usize = @intCast(playersCount);
18921891
while (i < spritesCount) : (i += 1) {
1893-
if (spriteSnake[i].?.num == 0) {
1892+
if (spriteSnake[i].?.num() == 0) {
18941893
destroySnake(spriteSnake[i].?);
18951894
spriteSnake[i] = null;
18961895
var j = i;

zsrc/map.zig

+8-3
Original file line numberDiff line numberDiff line change
@@ -433,9 +433,11 @@ pub fn initRandomMap(floorPercent: f64, smoothTimes: c_int, trapRate: f64) void
433433
}
434434

435435
pub fn pushMapToRender() void {
436+
std.log.info("pushMapToRender()...", .{});
436437
// Aliases to cutdown on long names.
437438
const cpa = ren.createAndPushAnimation;
438439

440+
// This code figures out the map boundaries and lays down the map walls (edges) only.
439441
for (0..res.n) |i| {
440442
for (0..res.m) |j| {
441443
const i_cInt: c_int = @intCast(i);
@@ -699,13 +701,16 @@ pub fn pushMapToRender() void {
699701
}
700702
}
701703

702-
for (0..(res.SCREEN_WIDTH / res.UNIT)) |i| {
703-
for (0..(res.SCREEN_HEIGHT / res.UNIT)) |j| {
704+
// This code lays down the map ground basically, anything within wall boundaries that
705+
// is not animated (like traps).
706+
for (0..res.n) |i| {
707+
for (0..res.m) |j| {
704708
if (!hasMap[i][j]) {
705709
continue;
706710
}
707711
const node = tps.createLinkNode(gm.map[i][j].ani);
708-
std.debug.assert(node.data != null);
712+
// This line is not necessary, it's always guaranteed to have a populated data field.
713+
// std.debug.assert(node.data != null);
709714
tps.pushLinkNode(&ren.animationsList[ren.RENDER_LIST_MAP_ID], node);
710715
}
711716
}

0 commit comments

Comments
 (0)