A general guide to contributing to LOOT may be found on LOOT's website. Information more specific to this repository is found here.
The repository branching structure is pretty simple:
- The
master
branch holds code that's released or that will be in the next release. Code onmaster
is generally pretty stable as CI must pass before anything can be merged into it. - Other branches are generally themed on specific features or groups of changes, and come and go as they are merged into one of the two above, or discarded.
The best way to get started is to comment on something in GitHub's commit log or on the issue tracker (new tracker entries are always welcome).
Surprise pull requests aren't recommended because everything you touched may have been rewritten, making your changes a pain to integrate, or obsolete. There are only generally a few contributors at any one time though, so there's not much chance of painful conflicts requiring resolution, provided you're working off the correct branch.
When you do make a pull request, please do so from a branch which doesn't have the same name as they branch you're requesting your changes to be merged into. It's a lot easier to keep track of what pull request branches do when they're named something like you:specific-cool-feature
rather than you:master
.
LOOT supports translation into other languages, with the following limitations:
- Debug log messages and error messages generated by the libraries LOOT uses cannot be translated.
- Masterlist messages can be translated, but translations must be submitted to the masterlist maintainers for addition. Translating masterlist messages won't be covered here.
- The languages LOOT supports is defined in LOOT's settings, so new translations can be added without updating LOOT. However, LOOT's default settings are hardcoded, so LOOT must be updated to support new translations by default.
The easiest way to contribute translations is to use Weblate. On Weblate you can submit translations for LOOT itself, the LOOT installer and LOOT's masterlist prelude all in one place. Plus, if you're not sure about your translation, you can submit it as a suggestion.
You can also submit translations using Git and GitHub, though we encourage contributors to go through Weblate because it's more user-friendly, and because it helps to avoid synchronisation issues between Weblate and GitHub. You do need an account on Weblate to contribute there, but you can sign in with various identity providers, including GitHub.
If you're having any trouble, feel free to ask a team member for help, such as in the #dev-questions or #translations channels in our Discord or at our dedicated localization issue.
LOOT's translations are licensed under the Creative Commons Zero 1.0 Universal license, effectively putting them into the public domain. By contributing a translation, you agree that your contribution is covered by the same license.
This section assumes you have a basic understanding of how to use a Git client and GitHub to submit your changes, as adding a new translation involves making code changes that can't be done through Weblate. If you don't, don't worry! You can still contribute; simply follow the previously linked guide to learn the basic principles needed.
Keep in mind that you don't need to submit a perfect pull request; just follow the instructions below to the best of your ability and we will correct any coding-related mistakes when we review your submission.
First fork this repository. All file paths given below are relative to its base folder.
First check that an Inno Setup translation exists for your language. Unofficial translations are acceptable, but require a bit of extra handling. If there isn't an official or unofficial translation for Inno Setup, you're better off making a translation and getting it listed on the linked page before continuing.
To add your language as an option in LOOT's installer:
-
Open the installer script at
scripts/installer.iss
in a text editor of your choice. -
If your language only has an unofficial translation, add a
#define <Language>Exists
block for it near the top of the script, like it has been done for Korean and Simplified Chinese. -
Add your language to the
[Languages]
section.The
Name
value should be the POSIX locale code for your language.The
MessagesFile
value should becompiler:Languages\<translation file>,resources/l10n/<locale>/LC_MESSAGES/installer.islu
, where<translation file>
is the filename of the Inno Setup translation that you checked exists and<locale>
is your language's POSIX locale code.If your language only has an unofficial translation, wrap its line in
#ifdef
and#endif
lines, again like it has been done for Korean and Simplified Chinese. -
Save your changes.
To create the installer.islu
translation file for your language, copy resources/l10n/en/LC_MESSAGES/installer.islu
to resources/l10n/<locale>/LC_MESSAGES/installer.islu
, where <locale>
is the POSIX locale code for your language.
If your language's Inno Setup translation is unofficial, also do the following:
- Open the script at
scripts/prepare_installer.ps1
in a text editor of your choice. - Add an entry for your language's translation file to the
$unofficialLanguageFiles
array. - Save your changes.
Once you've completed these steps, follow the instructions further down for updating an existing translation.
To create a loot.po
translation file for your language:
- Download and install the latest version of Poedit.
- If you are starting a new translation, select
Create new...
on the welcome page orFile -> New from POT/PO file...
and select the template file atresources/l10n/template.pot
. Select your language from the drop-down list, specifying a region/dialect if desired, and clickOK
. - Save the translation file with the filename
loot.po
inresources/l10n/<locale>/LC_MESSAGES/
, where<locale>
is your language's POSIX locale code.
LOOT's source code must be updated to recognise the new translation. While you're welcome to do this yourself and include the changes in your translation's pull request, you don't have to if you're not comfortable with them; we can add them later. The files and functions which must be updated are given below:
- In loot_settings.h, add the language's
ISO code and name to the
languages_
initialiser list. - In loot_settings_test.h, update
the
defaultConstructorShouldSetDefaultValues
test so that it includes the added language, updating theactualLanguages.size()
check and adding a newEXPECT_EQ(...)
line for the new language.
See commit 8c28aed7f54dee3a381425b2d4ecf4341309d139 for an example of the relevant changes.
Once you've completed these steps, follow the instructions further down for updating an existing translation.
If you're using Weblate to contribute translations, you can skip these instructions.
- Open
resources/l10n/<locale>/LC_MESSAGES/installer.islu
in your favourite text editor, where<locale>
is your language's POSIX locale code. - If you didn't just create your language's
installer.islu
, openresources/l10n/en/LC_MESSAGES/installer.islu
to check if there are any entries in it that aren't in your language'sinstaller.islu
file: if there are, copy them into your language's file. - Translate the string(s) in your language's
installer.islu
file into your language. Do not change the text before the first=
as that is the message's name. - Save your changes.
- Download and install the latest version of Poedit.
- In Poedit, open the
loot.po
translation file in the relevant subdirectory ofresources/l10n
, then selectTranslation -> Update from POT file...
and select the template file atresources/l10n/template.pot
. ClickOK
in theUpdate summary
dialog if it appears. - Edit the translation file to add or update translations of the programs' text. Strings that were added since the last translation will be missing a translation, and strings that have been changed since the last translation will be highlighted in orange.
- Go to
Translation -> Properties... -> Translation Properties -> Project name and version:
and check that it matches the latest version of LOOT. If it doesn't, please update it. Then go toFile -> Preferences -> General -> Information about the translator
and fill theName:
field with your name/alias. If you don't mind, you can also fill out theEmail:
field so that we or future translators can contact you if need be. - Save the translation file.
Some helpful Poedit settings include:
- Deselecting
File -> Preferences... -> General -> Editing -> Automatically compile MO file when saving
. This file is built by LOOT at runtime and doesn't need to be included in your PR, but you might accidentally submit it alongside your translation. - Selecting
File -> Preferences... -> General -> Editing -> Show summary after updating files
. This will show you a brief summary of strings that were changed/obsoleted when you update from the template.
Some languages may use different words or phrases for different contexts where only one word or phrase may be used for all contexts in English. Some strings are supplied with contextual descriptions to help disambiguate them. While contextual information isn't supplied for all strings by default, it can be added on request.
To request the addition of contextual information to a text string, create an issue for your request in LOOT's source code issue tracker, quoting the string for which you are requesting contextual information. Contextual information cannot be added for installer strings, but we can still answer questions about them.
If available, a string's description can be found within the String information sidebar on the right-hand side of the page, in the Source string description section. That section will not be present for strings that don't have a description.
If available, a string's description can be found in Poedit in the bottom-right corner in Poedit under Notes for translators.
You can also add comments of your own for future translators such as why you translated something a certain way or why you left something untranslated. To do so, right-click a string you want to comment on and select Edit comment
. They can also be seen in the bottom-right corner in Poedit under Comment.
Some strings to be translated may contain special characters. Different types of special character that may be encountered are:
-
Backslashes (
\
). These are used to escape backslashes and double-quotation marks ("
) in the C++ translation strings. Don't add new backslashes into translations, and make sure all backslashes in the original string are retained in the translation. -
Formatting placeholders are used so that LOOT can substitute text or numbers that are generated at runtime into pre-made strings. They appear as a number surrounded by curly braces, e.g.
{0}
.If formatting placeholders are used in the untranslated string, they must all be present in the translated string, or LOOT will encounter an error when it tries to display the translated string. Placeholders can be moved around so that the sentence makes grammatical sense in the target language.
-
A small number of strings include an ampersand (
&
) immediately before another character: these indicate mnemonic shortcuts that can be used for keyboard shortcuts. The ampersand can be moved to a suitable character in the translated text. For example, the string&File
is displayed asFile
and indicates that the user can useAlt + F
to navigate to that menu item, and theF
is underlined when the user holds down theAlt
key. However, the character chosen shouldn't conflict with other shortcuts at the same level. Specifically, menu titles (e.g.,&File
,&Game
) and section headers (e.g.,P&lugins
,F&ilters
) shouldn't share the same shortcuts, nor should menu items within the same menu (e.g.,&Settings...
and&Quit
in the&File
menu). Two menu items in different menus (e.g.,&Copy Content
in the&Game
menu,Copy &Card Content
in the&Plugin
menu) can use the same shortcut, as can a menu title and a menu item, including the menu containing the item (e.g.,Copy &Plugin Name
in the&Plugin
menu).For a better overview you can also look at the following Google Sheet document: LOOT - Mnemonic Shortcuts This document features every string with ampersands (
&
) added to them.Table 1
includes all the original English strings,Table 2
adds the ampersands to them andTable 3
lists every&+character
combination that it finds inTable 2
, plus it will automatically colourize duplicate entries red. You can utilise this document to first translate all the strings fromTable 1
into your own language and then add the ampersands accordingly. Once you are finished and every&+character
combination within the different "Levels" is unique, you can add your translations in Poedit. In order to be able to use the document, open it by clicking on the above link, then rightclick on the sheetEnglish -> Copy to -> New Spreadsheet
which will create your own copy of the document, which you now should be able to change. -
A small number of strings include Markdown hyperlinks (e.g.
[example text](http://example.com/)
). The text between the square brackets should be translated, but the text between the parentheses should not be.
It's occasionally useful to build LOOT using a snapshot build of libloot, e.g. when integrating
unreleased libloot changes. To do this, download a snapshot build artifact from GitHub Actions (you
need to be logged into GitHub to do so) and unzip it to get a 7-zip or XZ-compressed tar file, then
pass -DLIBLOOT_URL=<path to that file>
when running cmake
.
To download and extract a snapshot build artifact during the GitHub Actions CI workflow, change the value of the LIBLOOT_VERSION
environment variable to be the relevant commit hash. That only affects Linux builds, which build libloot from source. Windows builds use a prebuilt binary archive, so add a step to download the appropriate archive, before the step that runs cmake
:
- name: Download libloot snapshot build artifact
shell: bash
run: |
curl -sSfL \
-H 'authorization: Bearer ${{ secrets.GITHUB_TOKEN }}' \
-H 'Accept: application/vnd.github.v3+json' \
-o 'libloot.zip' \
https://api.github.com/repos/loot/libloot/actions/artifacts/39467110/zip
unzip libloot.zip
echo "LIBLOOT_URL=${{ github.workspace }}/$(ls -1 libloot-*.7z)" >> $GITHUB_ENV
Replace 39467110
with the relevant artifact's ID. You can get an artifact's ID from the last
path component in its download URL on the libloot build's GitHub Actions page in your browser: for
example, the page at https://github.com/loot/libloot/actions/runs/542851787
lists
libloot-0.16.1-9-ge97208c_linux-github-actions-Linux.tar.xz
as an artifact, and its download URL
is https://github.com/loot/libloot/suites/1982340583/artifacts/39467110
.
Also append -DLIBLOOT_URL="${{ env.LIBLOOT_URL }}"
to the arguments passed to cmake
.
LOOT's JavaScript code style is codified by the included ESLint configuration, which is based off the Airbnb and Prettier rules.
The Google C++ Style Guide is used as the base, with deviations as listed below. Formatting style is codified in the repository's .clang-format
file, but is not enforced.
- Static variables may contain non-POD types.
- Reference arguments don't need to be
const
(i.e. they can be used for output variables). - Exceptions can be used.
- Unsigned integer types can be used.
- There's no restriction on which Boost libraries can be used.
- Specialising
std::hash
is allowed.
- Constant, enumerator and variable names should use
camelCase
orunderscore_separators
, but they should be consistent within the same scope. - Function names should use
PascalCase
orcamelCase
, but they should be consistent within the same scope.
A pre-commit hook is used to ensure that the translation template file is kept up to date. The easiest way to install it is to configure Git to use the hook's directory as Git's hooks directory:
git config core.hooksPath scripts/git/hooks
or you can reinitialise the repository whenever the pre-commit hook is updated:
git init --template scripts/git
or just copy the file:
cp scripts/git/hooks/pre-commit .git/hooks/