NGettext.Avalonia 0.0.1-alpha

This is a prerelease version of NGettext.Avalonia.
There is a newer prerelease version of this package available.
See the version list below for details.
dotnet add package NGettext.Avalonia --version 0.0.1-alpha                
NuGet\Install-Package NGettext.Avalonia -Version 0.0.1-alpha                
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="NGettext.Avalonia" Version="0.0.1-alpha" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add NGettext.Avalonia --version 0.0.1-alpha                
#r "nuget: NGettext.Avalonia, 0.0.1-alpha"                
#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 NGettext.Avalonia as a Cake Addin
#addin nuget:?package=NGettext.Avalonia&version=0.0.1-alpha&prerelease

// Install NGettext.Avalonia as a Cake Tool
#tool nuget:?package=NGettext.Avalonia&version=0.0.1-alpha&prerelease                

ngettext-avalonia

Proper internationalization support for Avalonia (via NGettext)

This is a port of the great work of Robert J. Engdahl's ngettext-wpf.

See the change history <a href="CHANGELOG.md">here</a>, including whats new in the latest release and what will be included in the upcomming release.

Build status

LGPL

Getting Started

Get the NuGet from here <a href="https://www.nuget.org/packages/NGettext.Avalonia/">https://www.nuget.org/packages/NGettext.Avalonia/</a>.

NGettext.Avalonia is intended to work with dependency injection. You need to call the following at the entry point of your application:

NGettext.Avalonia.CompositionRoot.Compose("ExampleDomainName");

The "ExampleDomainName" string is the domain name. This means that when the current culture is set to "da-DK" translations will be loaded from "Locale\da-DK\LC_MESSAGES\ExampleDomainName.mo" relative to where your Avalonia app is running (You must include the .mo files in your application and make sure they are copied to the output directory).

Now you can do something like this in XAML:

<Button CommandParameter="en-US" 
        Command="{StaticResource ChangeCultureCommand}" 
        Content="{avalonia:Gettext English}" />

Which demonstrates two features of this library. The most important is the Gettext markup extension which will make sure the Content is set to the translation of "English" with respect to the current culture, and update it when the current culture is changed. The other feature it demonstrates is the ChangeCultureCommand which changes the current culture to the given culture, in this case "en-US".

Have a look at <a href="NGettext.Avalonia.Example/UpdateTranslations.ps1">NGettext.Avalonia.Example\UpdateTranslations.ps1</a> for how to extract msgids from both xaml and cs files.

Note that the script will initially silently fail (i.e. 2> $null) because there is no .po file for the given language. In the gettext world you are supposed to create that with the <a href="https://www.gnu.org/software/gettext/manual/html_node/Creating.html">msginit</a> command that ships with the <a href="https://www.nuget.org/packages/Gettext.Tools/">Gettext.Tools</a> nuget package, or PoEdit can be used to initialize the catalog from the intermediate pot file created. Here is what recently worked for me:

PM> mkdir -p Locale\en-GB\LC_MESSAGES\
PM> msginit --input=obj\result.pot --output-file=Locale\en-GB\LC_MESSAGES\ExampleDomainName.po --locale=en_GB

Conventions

Keep your compiled translations in "Locale\<LOCALE>\LC_MESSAGES\<DOMAIN>.mo". This library will force you to follow this convention. Or rather, NGettext forces you to follow a convention like "<PATH_TO_LOCALES>\<LOCALE>\LC_MESSAGES\<DOMAIN>.mo", and I refined it.

Keep your raw translations in "Locale\<LOCALE>\LC_MESSAGES\<DOMAIN>.po". This is not enforced, but when working with POEdit it will compile the ".mo" file into the correct location when following this convention, and it doesn't remember your previous choice, so stick with the defaults.

There are lots of GNU conventions related to internationalization (i18n) and localization (l10n). One of them is that the notion that the original program is written in US English, so you don't need to translate anything to facilitate internationalization. The original text in US English is called the msgId.

One of the most important GNU convention related to internationalization is providing a context to the translaters so they have a chance to do it right. For instance the English word 'order' has a number of more or less related meanings and thus different translations. For instance in the context of sequential ordering, 'order' translates to 'rækkefølge' in da-DK, but the imperative for placing an order translates to 'bestil'. Here is an example of how that can be fixed:


<Button Command="{StaticResource PlaceOrderCommand}" 
        Content="{avalonia:Gettext Imperative for placing an order|Order}" />

Translaters will rarely think of it, and just translate the first meaning that comes to mind, and as a programmer you might not know which words or sentenses needs a context, I therefore highly recommend to always provide a helpful context.


Support

Reach out to me at one of the following places!

  • Twitter at <a href="https://twitter.com/joergpreiss" target="_blank">@joergpreiss</a>
  • Create a question on <a href="https://stackoverflow.com/questions/ask?tags=ngettext.avalonia">Stack Overflow</a>. Use the tag ngettext.avalonia.
  • or create an <a href="https://github.com/slesa/ngettext-avalonia/issues">issue</a>

Sample Application

In <a href="NGettext.Avalonia.Example/">NGettext.Avalonia.Example/</a> you will find a sample application that demonstrates all the features of this library.

Demo

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.

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
0.3.14-rc 176 12/19/2022
0.3.13-rc 120 12/19/2022
0.0.1-alpha 128 12/18/2022