PLGL 1.0.0
dotnet add package PLGL --version 1.0.0
NuGet\Install-Package PLGL -Version 1.0.0
<PackageReference Include="PLGL" Version="1.0.0" />
paket add PLGL --version 1.0.0
#r "nuget: PLGL, 1.0.0"
// Install PLGL as a Cake Addin #addin nuget:?package=PLGL&version=1.0.0 // Install PLGL as a Cake Tool #tool nuget:?package=PLGL&version=1.0.0
(P)rocedural (L)anguage (G)eneration (L)ibrary
No language is justly studied merely as an aid to other purposes. It will in fact better serve other purposes, philological or historical, when it is studied for love, for itself. — J. R. R. Tolkien
1 — Introduction
Procedural Language Generation Library (PLGL) is a code library designed for game developers who want consistent, fictional languages for their game's cultures and peoples, without the time needed to create one. The language author constructs the alphabet, sigma structures, letter pathing, character filtering for deconstruction, and other constraints; then, the generator processes a regular sentence, and returns a new, stylized sentence from your fictional language.
2 — Contents
- Introduction
- Contents
- Examples
- Theory & Process
- 4.1 Deconstruction
- 4.2 Construction
- 4.3 Generating Sentences
- Setting Up
- Future Updates
- Useful Resources
3 — Examples
Qen is more of a scandinavian/english language, whereas Jabanese attempts to mimic Japanese. Qen has complex syllables, and a moderate amount of exclusion rules. Jabanese has simpler syllable structures, yet relies on letter exclusion a bit more. You can convert the generated sentence into hiragana with the built-in ToHiragana()
; otherwise, I recommend using ToRomaji()
.
In English:
So in this pleasant vale we stand again,
The field of Enna, now once more ablaze
With flowers that brighten as thy footstep falls
In the fictional language called Qen (with seed offset at 2):
Fras sil iren yadïnem nigüŋ mel spil nälis,
Lo flapäs ha Enna, om kelbu nölën numëmo
Nob sallesen on sölez egel hän borsüspöŋ ŝlonen
Qen, this time with the seed offset at 15:
Ik öl fonen mëöbür femöm füp be fume,
Lo manäs ha Enna, saŋ nebö fökum belidëf
Fe skenälen sof ürur egel ü mërrublan ëlen
Now in a fictional Japanese language (seed offset at 6 — Romaji and Hiragana):
pa mene ta wadagomo sudere ba ki musaha、
ぱ めね た わだごも すでれ ば き むさは、
chusa kugucho giyu gita、 shoku kazubi dayuke kozopeyo
ちゅさ くぐちょ ぎゆ ぎた、 しょく かづび だゆけ こぞぺよ
sote pumaya shu sushorya chi muki pobihe na。
そて ぷまや しゅ すしょりゃ ち むき ぽびへ な。
4 — Theory & Process
The generation process can be divided into two parts: Deconstruction and construction. Deconstruction breaks down a sentence into segments (specified by custom character filters). This greatly helps the construction process, which is responsible for handling how each filter block is processed. Since filters—and how they function—are defined by the language author, there is immense flexibility.
Using a word as a seed, the code procedurally generates an entirely different word. Because the Random is seeded with the word converted into an integer.
4.1 — Deconstruction.
The deconstruction process loops through the characters in your string, checking if the character matches any characters in any filter. In this case, if it's a letter, it starts counting. When it encounters a character from a different filter, it splits off the string, adds it to the list, and starts counting through the new filter block.
You could write these block separations plainly as: "The| |field| |of| |Enna|,|". It's from this list of character blocks that the constructor operates on (specifically, after they're added to a WordInfo class). You don't have to define every character; however, any unlisted character will be included anyway under the "UNDEFINED" filter, and will appear in the returned string.
After the deconstructor breakes the sentence down (the first pass), the new list of character blocks are looped through again, and the Deconstruct event is called on each. This is the second pass, processing all functions set by the language author. Some circumstances may require a block's filter to be changed, or three blocks to be merged into one. Words such as "let's", or numbers with commas or decimals, or even word flagging. I've included methods that help merge character blocks based on the specified criteria.
4.2 — Construction.
The OnConstruct event is the most crucial to implement. This is where you tell the generator how you want each filter to be processed. The LanguageGenerator class comes with a few common generation methods to speed up language authoring: CONSTRUCT_Hide
, CONSTRUCT_KeepAsIs
, CONSTRUCT_Replace
, CONSTRUCT_Within
, and CONSTRUCT_Generate
. These methods start with CONSTRUCT_
for clarity, so that auto-suggestion groups them together. If you plan to add any custom functionality (and you likely will), here's what KeepAsIs looks like:
public void CONSTRUCT_KeepAsIs(WordInfo word, string filter)
{
//Make sure the filter matches and the word hasn't already been processed.
if (word.Filter.Name.ToUpper() == filter.ToUpper() && word.IsProcessed == false)
{
//Set the final word to what the word started as.
word.WordFinal = word.WordActual;
word.IsProcessed = true;
}
}
The filter check is the most important part. If it's not included, the method is applied to every word. We'll look at CONSTRUCT_Generate
in the next section.
4.3 — Generating Sentences.
The generator starts by finding the root word by extracting any affixes. If none are found, the original word is the root word. If the root matches a key in Lexicon.Roots, the generated word will be set to its value. Then, the Random seed is set to the root.
Next up, the generator must select the syllable structure. Language.OnSyllableSelection is called, excluding any undesired syllables, and the remaining syllables are selected by weight. A custom syllable structure will be set if the word matches a key found in Lexicon.Syllables.
With the syllable structure set, the letters are chosen according to each syllable's letter group. For each letter group, Language.OnLetterSelection is called, excluding any undesired letters, and the remaining letters are selected by weight.
The affixes that were extracted earlier are processed and assembled by order. Language.OnPrefix and Language.OnSuffix is called during this process. This is useful if the affix needs to add a letter to make the word flow easier.
The final word is assembled with its prefixes, generated word, and suffixes put together. The word is memorized, so that it doesn't have to be processed twice (if enabled), and the case of the word is set to match the original word (if enabled). Now you have your new word. Unless you make changes to your language, or adjust the seed offset, it will make the same choices for that word every time.
5 — Setting Up
You should check out the Examples folder for ideas on authoring a language.
- Initial setup.
- Add a class to your project which derives from Language.
- Put all methods in the constructor.
- Fill in your language's metadata: name, description, author.
- Set additional properties found in Language.Options.
- Structuring.
- Add consonants and vowels to your alphabet.
- Add letter groups. These are the building blocks of syllables.
- Add syllables.
- Deconstruction.
- Add character filters. Unlisted charactered are added to Undefined filter when a sentence is processed. Examples:
- Delimiter. Usually just space. Highly recommended.
- Letters. a-z, A-Z. This filter is essentially required.
- Numbers. 0-9. Not required, but recommended.
- Punctuation. Optional, but recommended.
- Flags. Also needs FlagsOpen and FlagsClose during construction.
- Escape. Allows the surrounded block to escape it's filter (e.g, "[Generate]" results in "Generate"). This can be added with flagging, so it's optional.
- Add deconstruct events. This is the second pass, and corrects blocks through a stronger contextual lens. Some suggestions:
- Absorb single apostrophe into Letters, decimals and commas into Numbers, and Letters into Escape filter with
DECONSTRUCT_MergeBlocks
. - For the Flags filter, use
DECONSTRUCT_ContainWithin
.
- Absorb single apostrophe into Letters, decimals and commas into Numbers, and Letters into Escape filter with
- Add character filters. Unlisted charactered are added to Undefined filter when a sentence is processed. Examples:
- Construction. Add construct events (Language.Construct).
- Keep Undefined and Delimiter with
CONSTRUCT_KeepAsIs
. - Set Letters to Generate with
CONSTRUCT_Generate
. This is essential. - Set Punctuation to
Punctuation.Process
. - Set Flags to
Flags.Process
.
- Keep Undefined and Delimiter with
- Other options.
- Add punctuation. Alternative punctuation marks, or a particle system, for stronger language style.
Punctuation.Add
- Add flags (<Hide, Hide>, NoGen, ). There are a few default actions in the Language.Flags class.
- Add punctuation. Alternative punctuation marks, or a particle system, for stronger language style.
6 — Future Updates
- Improve how affixes are handled.
- Stronger control over sigma selection.
- Better control over letters (perhaps with consonant doubling, diphthongs, or some other rules).
- Easier, or less tedious, letter pathing—if at all possible.
- Add generation logging to help authors diagnose and fix their language.
- Add syllable rarity estimation, which returns the most (or least) likely syllables your language generates.
- Custom base conversion for numbers (low priority).
- More supporting methods in Diagnostics.
- Improve existing languages; exclusion rules and affixes, in particular.
- Create new languages.
7 — Useful Resources
- Understanding vowels. https://en.wikipedia.org/wiki/Vowel
- Pulmonic consonants for human-centric phonemes. https://en.wikipedia.org/wiki/Pulmonic_consonant
- English phonology. https://en.wikipedia.org/wiki/English_phonology
- Emic units. https://en.wikipedia.org/wiki/Emic_unit
- "Identifying Types of Affixes in English and Bahasa Indonesia". http://eprints.binadarma.ac.id/12808/1/723-775-1-PB.pdf
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
-
net6.0
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
1.0.0 | 221 | 2/16/2023 |
Initial release.