PLGL 1.0.0

dotnet add package PLGL --version 1.0.0                
NuGet\Install-Package PLGL -Version 1.0.0                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="PLGL" Version="1.0.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add PLGL --version 1.0.0                
#r "nuget: PLGL, 1.0.0"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// 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

  1. Introduction
  2. Contents
  3. Examples
  4. Theory & Process
  5. Setting Up
  6. Future Updates
  7. 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.

plgl

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.

plgl2

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.

  1. 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.
  2. Structuring.
    • Add consonants and vowels to your alphabet.
    • Add letter groups. These are the building blocks of syllables.
    • Add syllables.
  3. 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.
  4. 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.
  5. 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.

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

Product 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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • 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.