Ludum Linguarum is a set of tools and recipes to extract localized content (currently just text) from a set of supported games, which can then be turned into flash cards for use with the following spaced repetition software:
- Anki, one of the most popular spaced repetition software programs, with clients for all major desktop and mobile platforms.
- SuperMemo
- Mnemosyne
- AnyMemo
In other words, you can spice up your language learning regimen by drilling yourself with flash cards containing text from video games!
The list of 64 currently supported games can be found here. Currently only PC games are supported, although there's no technical reason other platforms couldn't be supported -- it's just simpler to work on PC versions of titles.
Ludum Linguarum can run on Windows, Linux, and OS X! (Some titles may only be supported under Windows.)
This walkthrough covers setting up Ludum Linguarum, the Anki desktop client, importing text from a game, exporting it to a format that Anki can recognize, and then importing it into Anki and using the flashcards to drill.
- Download and extract the latest version of Ludum Linguarum, from the list of releases above.
- Download the Anki desktop client and install it.
- Ensure that you have at least one of the supported games installed.
- Open up a command prompt window, in the directory to which you extracted Ludum Linguarum.
- Type
LudumLinguarum list-supported-games
to find the name of the game that you want to extract. - Type
LudumLinguarum import --game="<game name>" --game-dir="<root directory of game>"
to import the game text. (Some games that only support a single language with their installed data may require you to specify the language with--language-tag=<language tag>
.) - Type
LudumLinguarum export-anki --game="<game name>" --recognition-language="<first language>" --production-language="<second language>" --export-path="<path to export file>" --production-word-limit=5
. The "recognition language" is the two-letter language code for the language you are more familiar with, and the "production language" is the language code for the one you're trying to learn. Examples are "en" for English, "fr" for French, "de" for German, "ja" for Japanese, and so on. (Flash cards will be generated for translating in both directions.)export-path
is the output text file, which will be imported into Anki in the next step, and 'production-word-limit' restricts the length of text to 5 whitespace-separated words in the target language, which can be useful for limiting the flash cards to shorter phrases or vocabulary. - Open up the Anki desktop client. Click the Create Deck button at the bottom, and enter a name for the flashcard deck.
- Click the Import File button at the bottom. Select the file you exported in step 7.
- In the Import dialog, change Type to "Basic (optional reversed card)". Deck should be set to the deck you created in step 8. Fields separated by should be set to Tab, and the Field Mapping should be field 1 as Front, field 2 as Back, and field 3 as Add Reverse. (All of these except Type and Deck should be already correct, per the defaults.)
- Click the Import button. You'll probably see a warning about duplicate entries -- this is normal.
- Now, click on the deck name in the list, and the Study Now button. Now you're ready to practice! I recommend reading the Anki documentation to find more information on how to use the Anki client.
I also recommend setting up an AnkiWeb account, as this will let you sync your progress between the desktop and mobile clients. This makes it easy to practice on the go!
This walkthrough covers setting up Ludum Linguarum, importing text from a game, exporting it to a format that will work with SuperMemo's online importer, and then importing it and using the flashcards to drill.
- Download and extract the latest version of Ludum Linguarum, from the list of releases above.
- Ensure that you have at least one of the supported games installed.
- Open up a command prompt window, in the directory to which you extracted Ludum Linguarum.
- Type
LudumLinguarum list-supported-games
to find the name of the game that you want to extract. - Type
LudumLinguarum import --game="<game name>" --game-dir="<root directory of game>"
to import the game text. (Some games that only support a single language with their installed data may require you to specify the language with--language-tag=<language tag>
.) - Type
LudumLinguarum export-supermemo --game="<game name>" --recognition-language="<first language>" --production-language="<second language>" --export-path="<path to export file>" --production-word-limit=5
. The "recognition language" is the two-letter language code for the language you are more familiar with, and the "production language" is the language code for the one you're trying to learn. Examples are "en" for English, "fr" for French, "de" for German, "ja" for Japanese, and so on. (Flash cards will be generated for translating in both directions.)export-path
specifies the output text file (or files -- your export will be split up into files of 99 cards each, which matches SuperMemo's limit of how many cards you can import at once), which will be imported into SuperMemo in the next step, and 'production-word-limit' restricts the length of text to 5 whitespace-separated words in the target language. This can be useful for limiting the flash cards to shorter phrases or vocabulary. - Open up the SuperMemo website. Select the Course Editor from the drop-down menu.
- Click on the + button at the top (next to the list of courses) to create a new course, and give it a name.
- For each file generated in Step 6, do the following:
- Click the Import button at the bottom. Select Tab under "Question and Answer".
- Open up the file.
- Copy all of the text from the file.
- Paste it into the SuperMemo import dialog.
- Verify that the card contents are correct, and hit the Import button at the bottom.
- Exit, and select the new deck from SuperMemo's main menu. Now you're ready to practice! You can read more about how to most effectively use SuperMemo on their main web site.
This walkthrough importing text from a game, exporting it to a format that Mnemosyne can recognize, and then importing it into Mnemosyne and using the flashcards to drill.
- Download and extract the latest version of Ludum Linguarum, from the list of releases above.
- Download the Mnemosyne client and install it.
- Ensure that you have at least one of the supported games installed.
- Open up a command prompt window, in the directory to which you extracted Ludum Linguarum.
- Type
LudumLinguarum list-supported-games
to find the name of the game that you want to extract. - Type
LudumLinguarum import --game="<game name>" --game-dir="<root directory of game>"
to import the game text. (Some games that only support a single language with their installed data may require you to specify the language with--language-tag=<language tag>
.) - Type
LudumLinguarum export-mnemosyne --game="<game name>" --recognition-language="<first language>" --production-language="<second language>" --export-path="<path to export file>" --production-word-limit=5
. The "recognition language" is the two-letter language code for the language you are more familiar with, and the "production language" is the language code for the one you're trying to learn. Examples are "en" for English, "fr" for French, "de" for German, "ja" for Japanese, and so on. (Flash cards will be generated for translating in both directions.)export-path
is the output text file, which will be imported into Anki in the next step, and 'production-word-limit' restricts the length of text to 5 whitespace-separated words in the target language, which can be useful for limiting the flash cards to shorter phrases or vocabulary. - Open up the Mnemosyne client.
- Select the Import menu item under the File menu. Select the file you exported in step 7, and make sure that the file format selected is Tab-separated text files. You may want to add a tag to these cards, so you can bulk-modify them after import.
- Click the Ok button. After the import is complete, you may want to select the cards (under Cards -> Browse Cards...), and change the card type from Front-to-back only to Front-to-back and back-to-front.
- You are now ready to study with Mnemosyne! Read its documentation to find more information about how to use the client.
-
If you want to focus on particular types of cards (i.e. short vocabulary words, instead of translating dialogue), don't hesitate to use Anki's "bury" or "delete" functions -- "bury" will send the card to the bottom of the deck, while "delete" will completely remove it. Also, depending on whether a game's extraction recipe breaks it down this way, you may be able to filter on extraction by "lesson" name -- see the command-line help for more details about this.
-
Depending on the game, sometimes the extraction isn't 100% clean. Please let me know if you run into problems -- and in the meantime, you can always delete incorrect cards from your decks so you can continue working with them in the meantime.
Yes! Lots of free and commercial software and services are based around these ideas, including Duolingo, Course Hero, and Memrise. I would strongly encourage you to try it out if you are skeptical -- you'll find it starts to improve your recall almost immediately, and daily practice will build up your vocabulary quite quickly.
You should use this as a fun supplement to other language learning resources, like classes, study groups, etc. Using flash cards generated by this isn't going to teach you the very basics of a language, pronunciation, grammar, etc. -- it is a very effective way of expanding your vocabulary, and seeing a huge corpus of text in a foreign language. If you've already played the game whose text you've exported (in a language you understand), it can be quite helpful in rapidly learning the deck. It's not a requirement, though -- you can just as easily use this to enable you to play a game 'cold' in another language!
Check the list of currently supported games here. The release notes file also lists supported games, along with the version number in which they were implemented.
No. It's copyrighted content. Not gonna happen.
Thanks for asking! The answer is, "maybe," depending on how difficult it is to extract the content from the game.
I've drawn up a list of games that have been investigated and are either definitively unsupportable (i.e. their localized text is all stored as bitmaps, etc.), or are unlikely to be supported due to the difficulty in reverse-engineering their file formats.
Some characteristics that might make it easier to support a particular game:
- game files are unencrypted/uncompressed/compressed in a well-known format like ZIP
- game text is in a well-structured format (XML, JSON, etc.) or in a format with existing framework/library support (.NET satellite localization assemblies)
- game text is in a well-documented format (like the Bioware talk tables)
- the game is a sequel/prequel to another game that's already supported, or uses the same game engine as another already-supported title
I am definitely on the lookout for games that meet these criteria, as I can add support for them quickly. If you know of games that definitely meet these criteria, please contact me (see below) and let me know!
If a game doesn't match any of these characteristics, it doesn't mean that it's impossible to add support, but it may be very difficult or time-consuming to do. Code contributions and pull requests are always welcome, so if you are technically inclined and sufficiently motivated, you can do it yourself!
Look at the OneOffGamesPlugin project within the source for some small, self-contained examples. Essentially, the plugin registers a bunch of game names that it recognizes, and then gets called by the main application to extract the text, and generate the "cards" to be written to the SQLite database that contains all of the extracted content. SimpleGames.fs contains the simplest implementations of extraction -- some of them are just a few lines of code!
The project has a Twitter account, @LudumLinguarum. You can also contact me (Erik Novales) at @yankeefinn.
GitHub Actions |
---|
Package | Stable | Prerelease |
---|---|---|
LudumLinguarum |
Make sure the following requirements are installed on your system:
- dotnet SDK 3.0 or higher
- Mono if you're on Linux or macOS.
or
CONFIGURATION
will set the configuration of the dotnet commands. If not set, it will default to Release.CONFIGURATION=Debug ./build.sh
will result in-c
additions to commands such as indotnet build -c Debug
GITHUB_TOKEN
will be used to upload release notes and NuGet packages to GitHub.- Be sure to set this before releasing
DISABLE_COVERAGE
Will disable running code coverage metrics. AltCover can have severe performance degradation so it's worth disabling when looking to do a quicker feedback loop.DISABLE_COVERAGE=1 ./build.sh
> build.cmd <optional buildtarget> // on windows
$ ./build.sh <optional buildtarget>// on unix
Clean
- Cleans artifact and temp directories.DotnetRestore
- Runs dotnet restore on the solution file.DotnetBuild
- Runs dotnet build on the solution file.DotnetTest
- Runs dotnet test on the solution file.GenerateCoverageReport
- Code coverage is run duringDotnetTest
and this generates a report via ReportGenerator.WatchApp
- Runs dotnet watch on the application. Useful for rapid feedback loops.WatchTests
- Runs dotnet watch with the test projects. Useful for rapid feedback loops.GenerateAssemblyInfo
- Generates AssemblyInfo for libraries.CreatePackages
- Runs the packaging task from dotnet-packaging. This creates applications forwin-x64
,osx-x64
andlinux-x64
- Runtime Identifiers.- Bundles the
win-x64
application in a .zip file. - Bundles the
osx-x64
application in a .tar.gz file. - Bundles the
linux-x64
application in a .tar.gz file.
- Bundles the
GitRelease
- Creates a commit message with the Release Notes and a git tag via the version in theRelease Notes
.GitHubRelease
- Publishes a GitHub Release with the Release Notes and any NuGet packages.FormatCode
- Runs Fantomas on the solution file.Release
- Task that runs all release type tasks such asGitRelease
andGitHubRelease
. Make sure to read Releasing to setup your environment correctly for releases.
git add .
git commit -m "Scaffold"
git remote add origin https://github.com/user/MyCoolNewApp.git
git push -u origin master
-
- You can then set the
GITHUB_TOKEN
to upload release notes and artifacts to github - Otherwise it will fallback to username/password
- You can then set the
-
Then update the
CHANGELOG.md
with an "Unreleased" section containing release notes for this version, in KeepAChangelog format.
NOTE: Its highly recommend to add a link to the Pull Request next to the release note that it affects. The reason for this is when the RELEASE
target is run, it will add these new notes into the body of git commit. GitHub will notice the links and will update the Pull Request with what commit referenced it saying "added a commit that referenced this pull request". Since the build script automates the commit message, it will say "Bump Version to x.y.z". The benefit of this is when users goto a Pull Request, it will be clear when and which version those code changes released. Also when reading the CHANGELOG
, if someone is curious about how or why those changes were made, they can easily discover the work and discussions.
Here's an example of adding an "Unreleased" section to a CHANGELOG.md
with a 0.1.0
section already released.
## [Unreleased]
### Added
- Does cool stuff!
### Fixed
- Fixes that silly oversight
## [0.1.0] - 2017-03-17
First release
### Added
- This release already has lots of features
[Unreleased]: https://github.com/user/MyCoolNewApp.git/compare/v0.1.0...HEAD
[0.1.0]: https://github.com/user/MyCoolNewApp.git/releases/tag/v0.1.0
- You can then use the
Release
target, specifying the version number either in theRELEASE_VERSION
environment variable, or else as a parameter after the target name. This will:- update
CHANGELOG.md
, moving changes from theUnreleased
section into a new0.2.0
section- if there were any prerelease versions of 0.2.0 in the changelog, it will also collect their changes into the final 0.2.0 entry
- make a commit bumping the version:
Bump version to 0.2.0
and adds the new changelog section to the commit's body - push a git tag
- create a GitHub release for that git tag
- update
macOS/Linux Parameter:
./build.sh Release 0.2.0
macOS/Linux Environment Variable:
RELEASE_VERSION=0.2.0 ./build.sh Release
Copyright (c) 2007‒2017, Stephan Tolksdorf. All rights reserved.
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
This software is provided by the copyright holders “as is” and any express or implied warranties, including, but not limited to, the implied warranties of merchantability and fitness for a particular purpose are disclaimed. In no event shall the copyright holders be liable for any direct, indirect, incidental, special, exemplary, or consequential damages (including, but not limited to, procurement of substitute goods or services; loss of use, data, or profits; or business interruption) however caused and on any theory of liability, whether in contract, strict liability, or tort (including negligence or otherwise) arising in any way out of the use of this software, even if advised of the possibility of such damage.
MIT License
Copyright (c) 2022 Greg Dennis
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.