Thumbnail / Header for article: Unstable Universe (Part 2)

Unstable Universe (Part 2)

Welcome to part 2 of the technical devlog.

Havenā€™t read the other entries? Go to the devlog overview.

Step 5: Determine all other nodes

This is the big one.

For each game, I needed:

  • A diverse set of node types (e.g. each different action type must appear at least once)

  • Following certain requirements (e.g. node X must appear at least 3 times)

  • Placed at proper locations on the board (e.g. harder nodes must be placed more towards the center)

Step 1: determine the nodes weā€™ll use. This is only activated if you have expansions enabled, because then the computer needs to choose a subset out of a large number of nodes. Iā€™ll elaborate on this later.

Step 2: create the full list of nodes beforehand. Thatā€™s usually the best way to go with random generation. Ensure that each type is considered at least once, then fill the list randomly until itā€™s equal to the number of nodes on the board.

I ended up with these special settings that each node could have:

  • Prob: probability of drawing this node (required).

  • Min: when this node is selected, it is immediately added to the list ā€œminā€ times.

  • Max: whenever this node is placed, we count how many already exist. If weā€™ve reached the maximum, we remove the node from the options (so itā€™s never placed again).

  • MinDistanceFromEdge: self-explanatory.

  • MaxDistanceFromEdge: self-explanatory.

  • MaxSequence: how many of this type may be in sequence. Having nodes in sequence is usually boring or overpowered, and makes the board unbalanced (because all the nodes of type X are all in the same corner). By default, only two are allowed in sequence, with many nodes going down to 1. Iā€™ll explain the code below.

Remark: Nodes such as ā€œCrittersā€ require you to collect X nodes of a different type before youā€™re allowed to pass them. They are (near) impossible to pass if you reach them early in the game, so thatā€™s why I introduced the ā€œminDistanceFromEdgeā€ => to place these away from the starting nodes. (To ensure this very rarely happens I even added an extra constraint in the code: donā€™t allow nodes with a number on them right after a starting node.)

Whenever you need to find all nodes that share something (they are in sequence, they are connected, whatever), but you have no clue how many there are or where they are, your best bet is usually a recursive function.

To check if a certain node placement would create too many identical nodes in sequence, I execute this function:

  • Add the node to a shared list and mark it as ā€œvisitedā€.

  • Go through all connections.

    • If the connection has the same type as us, call this function again, but on that node.

The code for this particular part looks as follows:

 1//
 2// somewhere else, where we check if a node is allowed
 3//
 4var sequenceLength = this.checkSequenceRecursive(p, type);
 5if(sequenceLength > maxSequence) { return false; }
 6
 7
 8//
 9// the big function
10//
11function checkSequenceRecursive(p, type) {
12  var sequenceSums = 1; // also take into account the node itself
13  p.sequenceCounter = this.curSequence + 1;
14
15  for(var c = 0; c < p.connections.length; c++) {
16    var conn = p.connections[c];
17    
18    // node is the wrong type, or already checked? Continue!
19    if(conn.type != type || conn.sequenceCounter > this.curSequence) { continue; }
20
21    // otherwise, this node extends the sequence, so call this function again on THAT node
22    sequenceSums += this.checkSequenceRecursive(conn, type)
23  }
24
25  return sequenceSums;
26}

Remark: I use a trick to make this faster. Normally, youā€™d go through all nodes and mark them ā€œunvisitedā€ before running such an algorithm, but I found that slow. I keep a global counter that increases every time this algorithm is run. If a nodeā€™s sequenceCounter is smaller than the global one, it means it hasnā€™t been considered for this algorithm yet!

(See the main code for the game for all the finer details.)

Step 3: go through all the nodes and assign their type: pick the first type in the list, remove it from the list and give it to the node.

However, a nodeā€™s special settings might forbid placing this type here. If thatā€™s the case, we temporarily skip it and try the next one on the list. Repeat until we find a matching type. (If we find nothing that fits, the algorithm begrudgingly assigns a random type from the list.)

 1function determineNodeTypes() {
 2  // now go through all points 
 3  var nodeTypes = []; // ACTUALLY a list with all the nodes we actually want to place; explained later
 4  var nodeCounter = 0;
 5  for(var i = 0; i < this.points.length; i++) {
 6    var p = this.points[i];
 7
 8    // (only update those that haven't yet been updated, otherwise we override center node and mission nodes)
 9    if(p.type != 'Regular') { continue; }
10
11    // as long as we encounter nodes that are not allowed, try the next point type on the list
12    // (if we've exhausted the whole list, reset to 0 and just pick that anyway)
13    var nodeAllowed = false, counter = -1, tempType = null;
14    do {
15      counter++;
16
17      if(counter >= nodeTypes.length) {
18        tempType = nodeTypes[0]
19        break;
20      }
21
22      tempType = nodeTypes[counter]
23      nodeAllowed = this.checkIfNodeAllowed(p, tempType);
24    } while(!nodeAllowed);
25
26    p.type = tempType;
27    nodeTypes.splice(counter, 1);
28  }
29}
30 
31function checkIfNodeAllowed(p, type) {
32  // this is the huge dictionary that holds all node types (key = node name, value = data about it)
33  var nodeData = NODES[type]
34
35  //
36  // some nodes are not allowed at the edge
37  //
38  if(p.edgePoint && nodeData.forbiddenOnEdge) { return false; }
39
40  //
41  // some nodes have a fixed minimum distance from the edge
42  //
43  var minDistance = nodeData.minDistanceFromEdge || -1;
44  if(this.distanceToEdge(p) <= minDistance) { return false; }
45
46  //
47  // and some even have a fixed MAXIMUM distance from the edge
48  // (such as water, because I don't want people teleporting to the center of the board)
49  //
50  var maxDistanceFromEdge = nodeData.maxDistanceFromEdge || Infinity;
51  if(this.distanceToEdge(p) > maxDistanceFromEdge) { return false; }
52
53
54  //
55  // many nodes have a maximum on the number of them that may be "in sequence" ( = grouped together)
56  //
57  var maxSequence = nodeData.maxSequence || 2;
58
59  this.curSequence++;
60  var sequenceLength = this.checkSequenceRecursive(p, type);
61  if(sequenceLength > maxSequence) { return false; }
62
63  //
64  // a node with a number is NOT allowed right after a starting node
65  //
66  if(nodeData.needsNumber && this.connectedToStartingSquare(p)) { return false; }
67
68  return true;
69}

Hereā€™s the closest image I have of the board at this point in development. Notice how elements are evenly spread out and varied across the board:

Determining node types, following restrictions (old board)
Determining node types, following restrictions (old board)

Step 6: Placing Power Dots

The concept of ā€œPower Dotsā€ is actually the thing that holds this whole game together.

If you allow cutting into the game board, thereā€™s going to be a ton of imprecision. When is a node ā€œdestroyedā€ or ā€œcut offā€? When only half of it is visible? When itā€™s completely gone? When itā€™s been cut at least one time?

There were no satisfying answers to this question, until I realized that the node itself wasnā€™t important. The edges were important.

And so I invented the idea of power dots: these are placed around a node, in the space between edges. (If possible, the board tries to make sure they never overlap.) If such a dot is cut off ā€“ which is a clear yes/no answer ā€“ itā€™s lost. You only own a node as long as you own one of its power dots.

The power dots are the small (semi-)circles around each node:

Power dots pointed out on board
Power dots pointed out on board

Finding a good placement algorithm took a few tries, but I eventually settled on the following:

  • Determine the angle of all the edges (of this particular node).

  • Repeat until we find a fitting location:

    • Pick a random edge. Check the angle between this one and the next one.

    • If the angle is large enough, we should be able to fit a power dot between them.

    • Place it randomly within the available space.

    • (From now on, ā€œpretendā€ the new power dot is also an edge, so we donā€™t get overlapping power dots.)

We repeat this algorithm as many times as we need power dots. (On higher player counts, there are more power dots around each node on average.)

The loop also terminates once weā€™ve exhausted all possible angles. This simply means that weā€™ve checked the space between every edge in our list and the next one, and found no space large enough. (It would be unwise to wait until literally all angles have been tried, because there are infinitely many of them.)

This is the code:

 1function placePowerDots() {
 2  const powerDotRadius; // size of power dots; depends on, well, whatever you decide to set it to
 3  var minPD, maxPD // minimum and maximum number of power dots; depends on player count
 4
 5  var tempNumPowerDots = 0;
 6  for(var i = 0; i < this.points.length; i++) {
 7    var p = this.points[i];
 8    p.powerDots = [];
 9
10    tempNumPowerDots = Math.floor(Math.random()*(maxPD - minPD + 1)) + minPD;
11
12    // make a COPY of the angles list, because we're going to be modifying it here, 
13    // and we don't want to accidentally modify the original
14    var allAngles = p.edgeAngles.slice();
15    
16    // How does it work?
17    // We pick a random edge and calculate distance to next edge. If that is big enough, we can place the dot somewhere in between
18    // The dot also counts as an edge, so we add it to allAngles
19    for(var pd = 0; pd < tempNumPowerDots; pd++) {
20      var curAngleIndex = Math.floor(RNG() * allAngles.length);
21      var foundFreeSpace = false;
22      var numTries = 0;
23
24      allAngles.sort();
25
26      do {
27        var ang = allAngles[curAngleIndex];
28        var nextAng = allAngles[(curAngleIndex + 1) % allAngles.length]
29
30        var spaceBetween = (nextAng - ang + 2*Math.PI) % (2*Math.PI); 
31        if(ang == nextAng) { spaceBetween = 2*Math.PI; }
32
33        // yes, enough space! place it in here
34        var res = false;
35        if(spaceBetween > 2*powerDotRadius) {
36          var randAngle = ang + Math.random() * (spaceBetween-2*powerDotRadius) + powerDotRadius;
37
38          var res = this.createPowerDot(randAngle, p, allAngles)
39
40          if(res) {
41            foundFreeSpace = true;
42            break;
43          }
44        }
45
46        // no, not enough space, try again with the next edge!
47        if(!res) {
48          curAngleIndex = (curAngleIndex + 1) % allAngles.length;
49          numTries++;
50        }
51
52      // we're certain there is no free space if we've tried all angles we have
53      } while(!foundFreeSpace && numTries <= allAngles.length);
54    }
55  }
56}

(Iā€™ve left out some exceptions in case of nodes at the boundary of the paper - which canā€™t use all angles of course, because some are off the board - or when no power dot could be placed at all. For simplicityā€™s sake.)

Step 7: Polishing & Visuals!

Weā€™re almost done! Already!

You might be thinking that the algorithms for this game arenā€™t that difficult at all, despite my claims at the beginning, but thatā€™s because I only wrote down the end result. I can spend days if not weeks pondering how to best solve a problem, trying five different approaches, until I find the right one for this game.

(If you want to read about the failed attempts or early versions, they are in the regular devlog. I donā€™t see a point in explaining failed algorithms and showing their code in such a technical devlog. Iā€™d basically be steering readers in the wrong direction all the time.)

Example of a game board, base game only (no expansions or extras)
Example of a game board, base game only (no expansions or extras)

Finishing Touch #1: I mentioned that certain nodes, like Critters, need a number on them. You need to collect that many nodes of some type before you can enter. Itā€™s only logical that this number depends on how many nodes of that type are even on the board.

So I count that, divide by the player count (as any player could snatch those resources), and divide by distance to the center node. (If youā€™re further away, you should have really low numbers, as players donā€™t have that many resources yet.)

Finishing Touch #2: until this point, Iā€™d allowed my game engine (Phaser v3) to draw some circles and lines. That looked fine, but was too abstract and didnā€™t fit the theme of the game.

I changed it to the following:

  • I draw a perfect circle/rectangle for each node and color it depending on their type.

  • I also draw a perfect line between the nodes (behind them in the visual hierarchy).

  • I drew several rough/sketchy outlines and imported them. They are placed around the nodes and rotated randomly for extra sketchiness!

  • Then I placed each node icon inside the circle and rotated that randomly as well.

I find this a nice balance: the basic parts of the game are drawn by the computer, which makes them ā€œperfectā€ and ā€œcrispā€ (and saves me a lot of time and effort). But the other half, the details and the icons, are drawn by me in a looser and more organic style. Combining the two strikes a nice balance between readability/clarity and playfulness/interestingness (is that even a word?).

Finishing Touch #3: one of the more common missions in the game is to own (at least) one node in each ā€œquadrantā€ of the board ( = top left, top right, bottom left, bottom right). As youā€™re playing, people commented it was hard to see at a glance where each node was. So I added two straight lines through the center node to clearly mark the quadrants.

Finishing Touch #4: edge nodes didnā€™t have much space for icons. To solve this, I offset every edge node 10px from the edge by default, giving them some breathing space. (This turned out to be more annoying than I thought, because it also messed with other calculations if certain nodes were suddenly somewhere else. But I got it working after some frustrating hours.)

Conclusion (for now)

And thatā€™s it! Those are all the algorithms, code and ideas that went into creating a random generator of game boards for Unstable Universe.

The first cutting boardgame ever made! (As far as I can tell.) The boards turned out great, the game is a lot of fun, and there are many possibilities with the cutting mechanic that are simply impossible in other games. (Think about re-attachingĀ­ parts of the board somewhere else or drawing your own edge, which can then be moved over or cut.)

But for all those thoughts and comments, visit the regular devlog!

I always leave out the finer details, the exceptions, the failed attempts, etcetera. So donā€™t think that I did all this in one go, finishing this whole system in a single afternoon. Certainly not, it took months! (I also worked on many other things at the same time, but still.)

Why do I say this? Because I used to be discouraged, when I was younger, whenever I saw an artist create something cool or brilliant and pretending they did it in a day. Without any effort, or failed attempts, or doubts. I especially hate it when they donā€™t explain why they do something. With these devlogs, I try to give inspiration and explain how to solve certain problems, without going too much into the details.

That said, there are still some expansions that we need to talk about!