This is a documentation for Board Game Arena: play board games online !
ผลต่างระหว่างรุ่นของ "BGA Studio Cookbook"
Victoria la (คุย | ส่วนร่วม) |
Victoria la (คุย | ส่วนร่วม) |
||
บรรทัดที่ 69: | บรรทัดที่ 69: | ||
We can notice that resource and money are uncountable, and don't need to be track individually so we can replace our mapping to | We can notice that resource and money are uncountable, and don't need to be track individually so we can replace our mapping to | ||
* (resource type/money,player home zone, count) | * (resource type/money,player home zone, count) | ||
And vp stored already for us in player table, so we can remove it from that list. | |||
Now when we get to encode it we can see that everything can be encoded as (object,zone,state) form, where object and zone is string and state is integer. The resource mapping is slightly different semantically so you can go with two table, or counting using same table with state been used as count for resources. | Now when we get to encode it we can see that everything can be encoded as (object,zone,state) form, where object and zone is string and state is integer. The resource mapping is slightly different semantically so you can go with two table, or counting using same table with state been used as count for resources. | ||
บรรทัดที่ 81: | บรรทัดที่ 83: | ||
PRIMARY KEY (`token_key`) | PRIMARY KEY (`token_key`) | ||
) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ||
{| class="wikitable" | |||
|+token | |||
! token_key | |||
! token_location | |||
! token_state | |||
|- | |||
|meeple_red_1 | |||
|home_red | |||
|0 | |||
|- | |||
|dice_black_2 | |||
|board_guard | |||
|1 | |||
|- | |||
|dice_green_1 | |||
|board_action_mayor | |||
|3 | |||
|- | |||
|bread | |||
|home_red | |||
|5 | |||
|} | |||
Variant 2.1: Additional resource table, using same location as token table | Variant 2.1: Additional resource table, using same location as token table | ||
บรรทัดที่ 90: | บรรทัดที่ 117: | ||
PRIMARY KEY (`resource_type`,`resource_location`) | PRIMARY KEY (`resource_type`,`resource_location`) | ||
) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ||
{| class="wikitable" | |||
|+resource | |||
! resource_type | |||
! resource_location | |||
! resource_count | |||
|- | |||
|bread | |||
|home_red | |||
|5 | |||
|} | |||
Variant 2.2: Additional resource table, resource count for each player id | Variant 2.2: Additional resource table, resource count for each player id | ||
บรรทัดที่ 102: | บรรทัดที่ 140: | ||
ALTER TABLE resource ADD CONSTRAINT fk_player_id FOREIGN KEY (player_id) REFERENCES player(player_id); | ALTER TABLE resource ADD CONSTRAINT fk_player_id FOREIGN KEY (player_id) REFERENCES player(player_id); | ||
{| class="wikitable" | |||
|+resource | |||
! player_id | |||
! resource_key | |||
! resource_count | |||
|- | |||
|123456 | |||
|bread | |||
|5 | |||
|} | |||
Variant 3: More normalised | Variant 3: More normalised | ||
บรรทัดที่ 115: | บรรทัดที่ 163: | ||
PRIMARY KEY (`token_id`) | PRIMARY KEY (`token_id`) | ||
) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ||
{| class="wikitable" | |||
|+token | |||
! token_id | |||
! token_type | |||
! token_arg | |||
! token_location | |||
! token_state | |||
|- | |||
|22 | |||
|meeple | |||
|123456 | |||
|home_123456 | |||
|0 | |||
|- | |||
|23 | |||
|dice | |||
|2 | |||
|board_guard | |||
|1 | |||
|- | |||
|26 | |||
|dice | |||
|1 | |||
|board_action_mayor | |||
|3 | |||
|- | |||
|49 | |||
|bread | |||
|0 | |||
|home_123456 | |||
|5 | |||
|} | |||
Advantages of this would be is a bit more straightforward to do some queries in db, disadvantage its hard to read (as you can compare with previous example, you | |||
cannot just look at say, ah I know what it means). Another questionable advantage is it allows you to do id randomisation, so it hard to do crafted queries to | |||
cheat, the down side of that you cannot understand it either, and handcraft db states for debugging or testing. | |||
=== Database for The card game === | |||
Lets say you have a standard card game, player have hidden cards in hand, you can draw card from draw deck, play card on tableau and discard to discard pile. | |||
We have to design database for such game. | |||
In real word to "save" the game we take a picture a play area, save cards from it, then put away draw deck, discard and hand of each player separately and mark it, also we will record current scoring (if any) and who's turn was it. | |||
* Framework handles state machine transition, so you don't have to worry about database design for that (i.e. who's turn it is, what phase of the game we are at, you still have to design it but part of state machine step) | |||
* Also framework supports basic player information, color, order around the table, basic scoring, etc, so you don't have to worry about it either | |||
* The only thing you need in our database is state of the "board", which is "where each pieces is, and in what state", or (position,rotation) pair. | |||
Lets see what we have for that: | |||
* The card state is very simple, its usually "face up/face down", "tapped/untapped", "right side up/up side down" | |||
* As position go we never need real coordinates x,y,z. We need to know what "zone" card was, and depending on the zone it may sometimes need an extra "z" or "x" as card order. The zone position usually static or irrelevant. | |||
* So our model is: we have cards, which have some attributes, at any given point in time they belong to a "zone", and can also have order and state | |||
* Now for mapping we should consider what information changes and what information is static, later is always candidate for material file | |||
* For dynamic information we should try to reduce amount of fields we need | |||
** we need at least a field for card, so its one | |||
** we need to know what zone cards belong to, its 2 | |||
** and we have possibly few other fields, if you look closely at you game you may find out that most of the zone only need one attribute at a time, i.e. draw pile always have cards face down, hand always face up, also for hand and discard order does not matter at all (but for draw it does matter). So in majority of cases we can get away with one single extra integer field representing state or order | |||
* In real database both card and zone will be integers as primary keys referring to additional tables, but in our case its total overkill, so they can be strings as easily | |||
Variant 1: Minimalistic | |||
<pre> | |||
CREATE TABLE IF NOT EXISTS `card` ( | |||
`card_key` varchar(32) unsigned NOT NULL, | |||
`card_location` varchar(32) NOT NULL, | |||
`card_state` int(11) NOT NULL, | |||
PRIMARY KEY (`card_id`) | |||
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ; | |||
</pre> | |||
Variant 2: More normalised | |||
This version supported by Deck php class, so unless you want to rewrite db access layer go with this one | |||
<pre> | |||
CREATE TABLE IF NOT EXISTS `card` ( | |||
`card_id` int(10) unsigned NOT NULL AUTO_INCREMENT, | |||
`card_type` varchar(16) NOT NULL, | |||
`card_type_arg` int(11) NOT NULL, | |||
`card_location` varchar(16) NOT NULL, | |||
`card_location_arg` int(11) NOT NULL, | |||
PRIMARY KEY (`card_id`) | |||
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ; | |||
</pre> | |||
Note: if you using this schema, some zones/locations have special semantic. The 'hand' location is actually multiple locations - one per player, but player id is encoded as card_location_arg. If 'hand' in your game is ordered, visible or can have some other card states, you cannot use hand location (replacement is hand_<player_id> or hand_<color_id>) | |||
== Assorted Stuff == | == Assorted Stuff == | ||
tbd | tbd |
รุ่นแก้ไขเมื่อ 02:09, 19 มิถุนายน 2560
WORK IN PROGRESS
This page is collection of design and implementation recipes for BGA Studio framework. For tooling and usage recipes see Tools and tips of BGA Studio.
Visual Effects, Layout and Animation
Create all pieces statically
Ingredients: ggg_ggg.tpl, ggg.css, ggg.view.php (optional)
- Create ALL game pieces in html template (.tpl)
- ALL pieces should have unique id, and it should be meaningful, i.e. meeple_red_1d
- Do not use inline styling
- Id of player's specific pieces should use some sort of 'color' identification, since player id cannot be used in static layout, you can use english color name, hex 6 char value, or color "number" (1,2,3...)
- Pieces should have separated class for its color, type, etc, so it can be easily styled in groups. In example below you now can style all meeples, all red meeples or all red tokens, or all "first" meeples
in .tpl file:
<div id="home_red" class="home red"> <div id="meeple_red_1" class="meeple red n1"></div> <div id="meeple_red_2" class="meeple red n2"></div> </div>
in .css file:
.meeple { width: 32px; height: 39px; background-image: url(img/78_64_stand_meeples.png); background-size: 352px; } .meeple.red { background-position: 30% 0%; }
- There should be straight forward mapping between server id and js id (or 1:1)
- You place objects in different zones of the layout, and setup css to take care of layout
.home .meeple{ display: inline-block; }
- If you need to have a temporary object that look like original you can use dojo.clone (and change id to some temp id)
- If there is lots of repetition or zone grid you can use template generator, but inject style declaration in css instead of inline style for flexibility
Note:
- If you use this model you cannot use premade js components such as Stock and Zone
Game Model and Database design
Database for The euro game
Lets say we have a game with workers, dice, tokens, board, resources, money and vp. Workers and dice can be placed in various zones on the board, and you can get resources, money, tokens and vp in your home zone. Also tokens can be flipped or not flipped.
Now lets try to map it, we have
- (meeple,zone)
- (die, zone, sideup)
- (resource cube/money token/vp token,player home zone)
- (token, player home zone, flip state)
We can notice that resource and money are uncountable, and don't need to be track individually so we can replace our mapping to
- (resource type/money,player home zone, count)
And vp stored already for us in player table, so we can remove it from that list.
Now when we get to encode it we can see that everything can be encoded as (object,zone,state) form, where object and zone is string and state is integer. The resource mapping is slightly different semantically so you can go with two table, or counting using same table with state been used as count for resources.
So the piece mapping for non-grid based games can be in most case represented by (string: token_key, string: location, int: state), example of such database schema can be found here: dbmodel.sql and class implementing access to it here table.game.php.
Variant 1: Minimalistic
CREATE TABLE IF NOT EXISTS `token` ( `token_key` varchar(32) NOT NULL, `token_location` varchar(32) NOT NULL, `token_state` int(10), PRIMARY KEY (`token_key`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
token_key | token_location | token_state |
---|---|---|
meeple_red_1 | home_red | 0 |
dice_black_2 | board_guard | 1 |
dice_green_1 | board_action_mayor | 3 |
bread | home_red | 5 |
Variant 2.1: Additional resource table, using same location as token table
CREATE TABLE IF NOT EXISTS `resource` ( `resource_type` varchar(32) NOT NULL, `resource_location` varchar(32) NOT NULL, `resource_count` int(10) signed NOT NULL, PRIMARY KEY (`resource_type`,`resource_location`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
resource_type | resource_location | resource_count |
---|---|---|
bread | home_red | 5 |
Variant 2.2: Additional resource table, resource count for each player id
CREATE TABLE IF NOT EXISTS `resource` ( `player_id` int(10) unsigned NOT NULL, `resource_key` varchar(32) NOT NULL, `resource_count` int(10) signed NOT NULL, PRIMARY KEY (`player_id`,`resource_key`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
ALTER TABLE resource ADD CONSTRAINT fk_player_id FOREIGN KEY (player_id) REFERENCES player(player_id);
player_id | resource_key | resource_count |
---|---|---|
123456 | bread | 5 |
Variant 3: More normalised
This version is similar to "card" table from hearts tutorial, you can also use exact cards database schema and Deck implementation for most purposes (even you not dealing with cards).
CREATE TABLE IF NOT EXISTS `token` ( `token_id` int(10) unsigned NOT NULL AUTO_INCREMENT, `token_type` varchar(16) NOT NULL, `token_arg` int(11) NOT NULL, `token_location` varchar(32) NOT NULL, `token_state` int(10), PRIMARY KEY (`token_id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
token_id | token_type | token_arg | token_location | token_state |
---|---|---|---|---|
22 | meeple | 123456 | home_123456 | 0 |
23 | dice | 2 | board_guard | 1 |
26 | dice | 1 | board_action_mayor | 3 |
49 | bread | 0 | home_123456 | 5 |
Advantages of this would be is a bit more straightforward to do some queries in db, disadvantage its hard to read (as you can compare with previous example, you cannot just look at say, ah I know what it means). Another questionable advantage is it allows you to do id randomisation, so it hard to do crafted queries to cheat, the down side of that you cannot understand it either, and handcraft db states for debugging or testing.
Database for The card game
Lets say you have a standard card game, player have hidden cards in hand, you can draw card from draw deck, play card on tableau and discard to discard pile. We have to design database for such game.
In real word to "save" the game we take a picture a play area, save cards from it, then put away draw deck, discard and hand of each player separately and mark it, also we will record current scoring (if any) and who's turn was it.
- Framework handles state machine transition, so you don't have to worry about database design for that (i.e. who's turn it is, what phase of the game we are at, you still have to design it but part of state machine step)
- Also framework supports basic player information, color, order around the table, basic scoring, etc, so you don't have to worry about it either
- The only thing you need in our database is state of the "board", which is "where each pieces is, and in what state", or (position,rotation) pair.
Lets see what we have for that:
- The card state is very simple, its usually "face up/face down", "tapped/untapped", "right side up/up side down"
- As position go we never need real coordinates x,y,z. We need to know what "zone" card was, and depending on the zone it may sometimes need an extra "z" or "x" as card order. The zone position usually static or irrelevant.
- So our model is: we have cards, which have some attributes, at any given point in time they belong to a "zone", and can also have order and state
- Now for mapping we should consider what information changes and what information is static, later is always candidate for material file
- For dynamic information we should try to reduce amount of fields we need
- we need at least a field for card, so its one
- we need to know what zone cards belong to, its 2
- and we have possibly few other fields, if you look closely at you game you may find out that most of the zone only need one attribute at a time, i.e. draw pile always have cards face down, hand always face up, also for hand and discard order does not matter at all (but for draw it does matter). So in majority of cases we can get away with one single extra integer field representing state or order
- In real database both card and zone will be integers as primary keys referring to additional tables, but in our case its total overkill, so they can be strings as easily
Variant 1: Minimalistic
CREATE TABLE IF NOT EXISTS `card` ( `card_key` varchar(32) unsigned NOT NULL, `card_location` varchar(32) NOT NULL, `card_state` int(11) NOT NULL, PRIMARY KEY (`card_id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;
Variant 2: More normalised
This version supported by Deck php class, so unless you want to rewrite db access layer go with this one
CREATE TABLE IF NOT EXISTS `card` ( `card_id` int(10) unsigned NOT NULL AUTO_INCREMENT, `card_type` varchar(16) NOT NULL, `card_type_arg` int(11) NOT NULL, `card_location` varchar(16) NOT NULL, `card_location_arg` int(11) NOT NULL, PRIMARY KEY (`card_id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;
Note: if you using this schema, some zones/locations have special semantic. The 'hand' location is actually multiple locations - one per player, but player id is encoded as card_location_arg. If 'hand' in your game is ordered, visible or can have some other card states, you cannot use hand location (replacement is hand_<player_id> or hand_<color_id>)
Assorted Stuff
tbd