This file is here to provide some information to anyone who wants to assist on the patch, or people looking to create stuff like translations.
To prepare for this work, please first set up the actual game and the patch in their existing form. Make sure they work. This will allow you to test your changes.
For your build environment, I strongly recommend using GitHub Actions (through the "Actions" tab in your fork). There are two workflows: one which produces a publicly downloadable permanent release (that one is triggered manually), and one which produces a temporary download for testing you can find on the workflow's page (that one is triggered on every push).
If your release builds are failing, go to Settings -> Actions -> General, and make sure "Workflow Permissions" are set to "Read & Write".
If you are building locally, make sure you have at least the following:
- OS: Ubuntu 20.04 or later (Windows users should use the WSL version of Ubuntu). Other operating systems are likely going to work fine, but it's untested.
- Python 3.8 or later, to generate exefs patches
- Ruby 2.7 or later, to generate the romfs patch
- A
zip
command, to generate the patch archives. - Bash, to run the
build.sh
script.
Also, make sure to run the following commands after cloning the repo, or your build will fail:
git submodule init
git submodule update
Set all these things up and run build.sh
. You are going to end up with the same patch archives as what's available in the release section.
Follow the normal installation procedure from that point to update your patch and test your changes.
If you are building locally and using an emulator to test, you may define the $UMINEKO_TARGET
(for Ryujinx) or $UMINEKO_TARGET_YUZU
environment variables as the path to your Ryujinx/yuzu folders,
e.g. /mnt/c/Users/<your username>/AppData/Roaming/Ryujinx
or /mnt/c/Users/<your username>/AppData/Roaming/yuzu
.
If done correctly, then building the patch will automatically copy it to your emulator folder instead of creating archives for manual extraction.
Finally, Docker is also supported for building. To use this option, install Docker and run ./docker_build.sh --build-image
(Linux-like) or docker_build.bat --build-image
(Windows) to create the ~70MB image; this only has to be run once or when you want to update the image. Run ./docker_build.sh
(Linux-like) or docker_build.bat
(Windows) with no arguments to build the patch (building locally with $UMINEKO_TARGET
or $UMINEKO_TARGET_YUZU
will also work, although if using the .bat
file, you will need to use Windows-style paths).
Test any lines of the script you've changed. The quickest way is to select the appropriate Episode, then open the backlog with the X button and find your lines. Jump to them, check if everything looks fine, then continue your work. If you insist on using save files and playing through the game normally, note that any error in the script can cause the game to go haywire and erase your save data completely. Make backups!
The main script is stored in the script.rb
file. Note that this file contains a mix of text and binary data, so you need to use a sane editor for changing it that preserves NUL characters.
I use Visual Studio Code, but there may be others that work fine.
As a general rule, do not touch any line except those that start with s.ins 0x86
. These are the lines responsible for displaying text,
and they're the only ones you should be changing.
I currently do not parse any of the other instructions into anything sensible, so what anything else even does is a mystery to me.
Inside the 0x86
lines, you will also find a single call to s.layout()
containing the actual text. This is what you should be editing.
All strings in the file use single-quote syntax, so make sure to escape apostrophes inside the text (i.e. replace '
with \'
before you do anything).
Inside the text, you will find several "tags". These change text behaviour in one way or another. All tags start with @
.
Some tags exist on their own, like @r
is a complete tag.
Some tags take arguments. For those, the syntax is @varg1|arg2|arg3.
(the tag is @v
and the arguments are arg1
, arg2
, and arg3
; the .
terminates the tag).
The tags that are used in Umineko, with example arguments, are as follows:
@r
forces a line break. Additionally, separates the nametag from the actual displayed text.@w500.
waits for 500 milliseconds before continuing the text.@k
waits for a user to click before continuing in the middle of a line (this is automatic on end of line).@e
ends the line without waiting for a click. If used, this must be the last two characters of the line.@t
causes the text before and after it to appear at the same time, generally used when multiple characters talk over each other.@v29/52200086.
plays the voice line located in the filevoice/29/52200086.nxa
.@c900.
changes the text color to red. The number here is a strange decimal RGB code, ranging from000
(black) to999
(white).@c.
is the same thing as@c999.
, i.e. it changes the text color to white.@z70.
changes the font size to 70% of the normal size. The default size, therefore, is@z100.
.@{text@}
displays "text" in bold.@[text@]
displays "text" instantly, regardless of the user's text speed setting.@btop text.@<bottom text@>
causes furigana to render, with "bottom text" being the main, large text at the bottom, and "top text" being the smaller, informative text at the top. This is used mainly for pronounciation guides. Note that "top text", annoyingly, may not include any other tags or the.
character.@|
runs an external piece of code. This can do virtually anything, but is generally used for sound effect playback or mid-line sprite changes. This code is defined outside of the line itself, and it's not currently known how to change it. Because of this, it's very important to keep the amount and placement of@|
tags the same between translated and original lines, as modifying these can break the whole game.@y
waits for the code that was launched by@|
to finish executing. It almost always directly follows@|
, and just like the above, its placement should not be modified from the original line.
Many languages, even Latin-based ones, include characters that are not found in English. The game does not correctly handle these by default, so some extra files are required.
You may find these in the locale_base
folder. Simply take all of the files and copy them over to the main repo folder (where script.rb
is). After the files are copied, open replace_chars.txt
in a text editor and list any non-English characters that your language has, one per line. The game should now work with your language's character set.
If that fails and the localized font does not correctly render your language's characters, you will need to manually generate new font and manifest files, and structure them the same way as what's in the locale_base
folder. The repack/fnt
folder contains tooling necessary to do this, but do note that replacing this game's fonts is a long and annoying process. I recommend against doing it unless it is completely necessary.
On a final note, please be aware that the @b
tag (for furigana) will not work correctly with languages that contain special characters. You may choose to write the appropriate descriptions in parentheses instead.
exefs_texts.txt
contains text that is hardcoded in the Switch executable of the game. This is mainly used for UI text, such as confirmation prompts, but it also unfortunately includes BERNKASTEL's game during EP8.
This file is a TSV file. Each row has the format offset<tab>English text<tab>Japanese text
.
Out of these, the only one you should be changing is the "English text".
The offset is used to replace the correct portion of the executable file, so changing that will result in the game completely breaking.
The Japanese text is used for length validation. The English text must be the same size as the Japanese text, or lower, in bytes, when encoded as UTF-8. If the English text is larger (in bytes) than the Japanese version, the replacement will fail.
There is nothing that can be done about this, unfortunately. This will not cause problems translating into English, since one Japanese character will basically be the same size as three English chars, however, translating to other languages may prove problematic because of this limitation.
As a workaround, do note that the entire text for Bern's game is also mirrored in the actual script file. You may choose to inform your users of the ability to use the backlog to browse through the script, instead of translating the exefs text, if your language cannot be reasonably made work with the length requirements.
You will need to build and install enter_extractor for this.
The game includes two types of image files: .pic
, which is a simple, single-image format used for backgrounds and such,
and .txa
, which is a more complex multi-image texture format.
There is also .bup
used by sprites, but there's no conceivable need to ever edit those.
You will need just several commands to make image editing work.
- To convert a PIC file to an editable PNG, use
EnterExtractor file.pic file.png
. Edit the file with your favourite image editor. - To convert a PNG file back to a PIC, use
EnterExtractor file_original.pic file_new.pic -replace file.png
. You need the original PIC file from the base game for this process -- that'sfile_original.pic
. Thefile_new.pic
will be the edited version that EnterExtractor generates. - To convert a TXA file to a bunch of images, use
EnterExtractor file.txa prefix
. This will generate images namedprefix_<name from txa>.png
. Edit each of those with your favorite editor. - To convert the PNGs back to a TXA, use
EnterExtractor file_original.txa file_new.txa -replace prefix
. This likewise requires the original TXA file from the base game for it to work.
Note that PIC images can be palleted, and TXA images must be. Unpalleted TXAs will render incorrectly. Unpalleted PICs will just take an enormous amount of disk space, but will otherwise work fine.
For this paletting process to work, the PNG needs to be saved with 256 colors only. This will work fine for all the images in the game that include text, so just remember to check the appropriate setting in your image editor.