You can download the game now at https://bobcgames.itch.io/yags
If you play, I’m always happy to hear what you think!
You can download the game now at https://bobcgames.itch.io/yags
If you play, I’m always happy to hear what you think!
We have a (tentative) release date for YAGS!
If all goes well, the game will be out on itch.io on Saturday, January 19, 2019 at 10 AM EST.
You can follow my Twitter for that latest.
I spent most of today trying to figure out how to generate an Android build of YAGS. The RenPy documentation is pretty good, except if you want to have your own keystore for signing.
Start by opening RenPy and clicking the “Android” link for your game. If this is your first android build, it’ll download and install a bunch of tools and prompt you to install the Java JDK, if you haven’t already.
After all of that’s done, you can test your game by launching it (from that same "Android" link > Emulation > Phone (or Tablet)). You may need to make some tweaks, especially if you’ve done extensive GUI customization for the standard desktop form factor. In particular, note that the variant "touch" and variant "small" directives can be useful to update your screens, and that gui.rpy has several mobile-specific style overrides (such as font sizes) at the bottom of the file.
Once you’re happy with your game, it’s time to build a package. Before you do this, you may want to create and configure your own keystore and create custom icons (with thanks to Rob Colton for help with the keystore).
First, if you don’t want to use the keystore that RenPy creates for you, you can create your own with keytool:
C:\Users\Bob> cd "C:\Program Files\Java\jdk1.8.0_191\bin"
C:\Program Files\Java\jdk1.8.0_191\bin> keytool -genkey -v -alias yourkeystorealias -keyalg RSA -keysize 2048 -validity 10000
This will prompt you for various values involved in generation. You should set a keystore password, enter your name (i.e. “Bob Conway”), and your company (i.e. “bobcgames”) as the organization unit and organization. Then enter a key password. This will create a .keystore file in your user directory (i.e. C:\Users\Bob\.keystore).
(Alternately, you can use this tool, as recommended by Rob, do to keystore generation visually.)
Next, you’ll want to update your rapt settings. Navigate to renpy-7.x.x-sdk/rapt/project and open local.properties in a text editor. Edit these settings to match the values entered during keystore generation (alias, keystore and key passwords, and keystore location).
Finally, create images for the icons and presplash screen. The official documentation is pretty good about information on this, but note that the files should go in your base directory, not under /game. (This is the same directory where your icon.ico and icon.icns files go.)
You’ll also probably want your foreground image to be larger than the “minimum safe” size of 132×132 px. I just did 250×250 px for mine.
That should be enough to successfully build an apk, which you can then distribute!
Also worth noting: Multi-game persistence does not work with mobile builds. You should case all of your multi-persistence (including initializing the variable) with if not renpy.mobile: checks.
Wolfscade has drawn another piece of YAGS fanart, this time featuring Carlos! This is the third piece overall for YAGS.
After the mess that was tumblr’s new content policy, I decided that it no longer made sense to keep my blog there.
I also took advantage of the opportunity to create a more proper website for myself.
You can now find me at http://bobcgames.com/
For YAGS-specific information, visit http://yags.bobcgames.com/
After a month of work, the barajam game I’ve been working on with @poorlyformed is now out!
You can download it on itch.io here: https://poorlyformed.itch.io/ebae
It’s a lighthearted adventure featuring aliens and sex. And lots and lots of eggplants. 🍆🍆🍆
The game is almost in a shippable state already. We’re only missing some final GUI tweaks (mainly the choice menus), some CGs, and the final background in terms of required things.
I staged music and sound effects throughout the game this past week. Also did some more GUI tweaking and writing cleanup.
@poorlyformed has been hard at work on backgrounds, and we should have the last one shortly… then it’s on to CGs.
We’ve also now got a gallery that unlocks images as you complete the game.
Progress!
One of the things I’ve been more aware of recently is nudity in games, and particularly how that plays (or doesn’t play) with payment processors.
A lot of games, to stay on the safe side, have released their games in a “censored” mode and then released patches to re-enable nudity. I’m not sure how other people do it, but my goal was to make it as simple as possible for users (one file to replace) but also give myself a lot of control over changes.
As always, I don’t pretend that this is the right or best way to do things… only that it works for me.
First off, you have to identify what file(s) and game declarations have nudity or will be affected by censoring. In my case, I have both nude sprites and nudity in my CGs, located in /images/chnew and /images/cg, as well as LayeredImage declarations for my CGs.
We start by defining a new rpy file ndpatch.rpy to hold the aspects that require and control patching. Because of the way my sprites work, the only other thing I need in here is the CG definitions*. We also include some additional constants, for use later.
## Is nudity allowed at all
define nudityIsAllowed = True
## is for uncensored, c is for censored
define nudityverending = ""
## Align the game with the patch version where necessary
## Update expectedndpatchversion in options.rpy
define ndpatchversion = 1
layeredimage person_cg:
always:
"cg/person_base.jpg"
if persistent.bodyhair_person:
"cg/person_hair.png"
if not persistent.nudes:
"cg/person_censor.png"
Next, we define a separate rpa file for everything related to nudity (the sprites, the CGs, and the ndpatch.rpy file itself). This should come first in the build.classify calls since each file will be placed into an archive based on the order the rules are declared, in options.rpy. We also add a corresponding constant that will not be built into ndpatch.rpa.
## Match the patch version in ndpatch.rpy
define expectedndpatchversion = 1
init python:
build.archive("ndpatch", "all")
# Nudity
build.classify("game/ndpatch.rpyc", "ndpatch")
build.classify("game/images/chnew/**", "ndpatch")
build.classify("game/images/cg/**", "ndpatch")
# ...other archive and classify calls...
Now, we make a copy of the entire game project. In that copy, we edit the rpy file to reflect censored nudity.
## Is nudity allowed at all
define nudityIsAllowed = False
## is for uncensored, c is for censored
define nudityverending = "c"
## Align the game with the patch version where necessary
## Update expectedndpatchversion in options.rpy
define ndpatchversion = 1
layeredimage person_cg:
always:
"cg/person_base.jpg"
if persistent.bodyhair_person:
"cg/person_hair.png"
And then we also make sure, in the censored version of the project, we merge the censor in the CGs with the base layer (so even the raw game assets don’t have nudity that can be uncovered via unarchiving the rpa file later). Similarly, any sprite image files that have nudity in them must be edited in the censored version of the project to remove or obscure nudity.**
Technically, this is enough to get censoring working. You would simply build a distribution of the game via the censored version of the project, build a second distribution via the uncensored version, and provide the ndpatch.rpa file from the uncensored version for download. The images will be replaced in the game with uncensored versions when the ndpatch.rpa file is replaced, since they are part of the ndpatch.rpa file.
However, we can do a little better.
First, you can use the nudityverending constant to change how the version of your game displays, to make it obvious whether it’s got nudity or not. For example:
screen about():
...stuff...
text "Version [config.version!t][nudityverending]"
...stuff...
Second, your game may later need to change in a way that makes an older ndpatch.rpa incompatible. For example, if you add a new sprite expression, someone with an older ndpatch.rpa file that tries to use it against a newer game build will see errors when running the game. Similarly, if you delete an asset, someone with a newer ndpatch.rpa file will see errors with an older game build.
This is where the ndpatchversion and expectedndpatchversion constants come in handy. You can ensure they are equal when the game starts up and show an error or warning if they don’t match. For example:
label start:
if ndpatchversion != expectedndpatchversion:
"Your ndpatch.rpa file is the wrong version for this build of the game."
"Please download the latest patch and game from URL."
return
and
screen main_menu():
fixed:
if ndpatchversion > expectedndpatchversion:
text "{size=50}{color=#000}Your ndpatch.rpa is newer than the build of this game (v[config.version]). Please download the latest version of the game before patching.{/color}{/size}" xsize 500 xanchor 0.0 yanchor 0.0 pos(550, 100)
elif ndpatchversion < expectedndpatchversion:
text "{size=50}{color=#000}Your ndpatch.rpa too old for use with this build of the game (v[config.version]). Please download the latest version of the patch and try again.{/color}{/size}" xsize 500 xanchor 0.0 yanchor 0.0 pos(550, 100)
which then lets you bump both constants whenever you make a breaking change (like adding or removing a sprite asset or cg).
Finally, you can use the nudityIsAllowed constant inside the game itself anywhere that you want to handle differently in a censored vs uncensored game. For example, I change the warning that shows at the beginning of the game to reflect explicit vs non-explicit nudity.
There’s probably a way to make this work with a single Renpy project that builds both a ZIP distribution and a separate ndpatch.rpa file. But it seemed easier to me to just duplicate the project. But because of this, you probably want to do this work when your game is otherwise done, or you’ll potentially have to make bug fixes in both copies of the game project.
* Yes, I could have left the CG declarations outside of ndpatch.rpy and simply censored the base image, but that would require me to have a cg/person_censor.png file in both versions of the game, and that felt inefficient for me.
** Identifying and editing relevant images is where most of the work is, really. If you just want to flat out replace images, you don’t even need an ndpatch.rpy file. Simply declaring an ndpatch.rpa archive in options.rpy, and including all image files that will need to be censored in that archive, would be enough.
I’m apparently bad at updating on time. Oops.
On the writing side of things, I’m up to just under 11,000 words, and am calling the script done for now. There’s still some polish I’ll need to do, but I’ll wait until we finish staging the game for that.
I’ve also started working on the GUI. It’s very…. eggplant-ful. And I really like it.
On the art side of things, @poorlyformed has finished another two backgrounds as well as some other random pieces of art for the game.
Hooray progress!
Via @soulsoftea comes the first ever piece of YAGS fanart (Adam, Elliot, and Steve), and via @wolfscade comes piece #2 (chibi Adam!), and I am both blown away by how amazing they are, and touched by the effort put in.
Ahhhh I am so excited to see these.
The placeholder sprites have been unexpectedly popular, so I’m actually going to carry them through in the final game as an unlockable sprite option. But seeing them together with finished Adam in the first piece (much like the game in its current state) is just ridiculous and awesome.
Also now I want to commission more characters in chibi form…