Fluent Templates: A High level Fluent API.

Build & Test crates.io Help Wanted Lines Of Code Documentation

fluent-templates lets you to easily integrate [Fluent] localisation into your Rust application or library. It does this by providing a high level "loader" API that loads fluent strings based on simple language negotiation, and the FluentLoader struct which is a Loader agnostic container type that comes with optional trait implementations for popular templating engines such as handlebars or tera that allow you to be able to use your localisations in your templates with no boilerplate.

Loaders

Currently this crate provides two different kinds of loaders that cover two main use cases.

static_loader!

The easiest way to use fluent-templates is to use the [static_loader!] procedural macro that will create a new [StaticLoader] static variable.

Basic Example

rust fluent_templates::static_loader! { // Declare our `StaticLoader` named `LOCALES`. static LOCALES = { // The directory of localisations and fluent resources. locales: "./tests/locales", // The language to falback on if something is not present. fallback_language: "en-US", // Optional: A fluent resource that is shared with every locale. core_locales: "./tests/locales/core.ftl", }; }

Customise Example

You can also modify each FluentBundle on initialisation to be able to change configuration or add resources from Rust. ```rust use fluentbundle::FluentResource; use fluenttemplates::staticloader; use oncecell::sync::Lazy;

staticloader! { // Declare our StaticLoader named LOCALES. static LOCALES = { // The directory of localisations and fluent resources. locales: "./tests/locales", // The language to falback on if something is not present. fallbacklanguage: "en-US", // Optional: A fluent resource that is shared with every locale. corelocales: "./tests/locales/core.ftl", // Optional: A function that is run over each fluent bundle. customise: |bundle| { // Since this will be called for each locale bundle and // FluentResources need to be either &'static or behind an // Arc it's recommended you use lazily initialised // static variables. static CRATEVERSIONFTL: Lazy = Lazy::new(|| { let ftlstring = String::from( concat!("-crate-version = {}", env!("CARGOPKGVERSION")) );

            FluentResource::try_new(ftl_string).unwrap()
        });

        bundle.add_resource(&CRATE_VERSION_FTL);
    }
};

} ```

Locales Directory

fluent-templates will collect all subdirectories that match a valid Unicode Language Identifier and bundle all fluent files found in those directories and map those resources to the respective identifier. fluent-templates will recurse through each language directory as needed and will respect any .gitignore or .ignore files present.

Example Layout

text locales ├── core.ftl ├── en-US │   └── main.ftl ├── fr │   └── main.ftl ├── zh-CN │   └── main.ftl └── zh-TW └── main.ftl

Looking up fluent resources

You can use the [Loader] trait to lookup a given fluent resource, and provide any additional arguments as needed with lookup_with_args.

Example

``fluent # Inlocales/en-US/main.ftl` hello-world = Hello World! greeting = Hello { $name }!

# In locales/fr/main.ftl hello-world = Bonjour le monde! greeting = Bonjour { $name }!

# In locales/de/main.ftl hello-world = Hallo Welt! greeting = Hallo { $name }! ```

```rust use std::collections::HashMap;

use uniclangid::{LanguageIdentifier, langid}; use fluenttemplates::{Loader, static_loader};

const US_ENGLISH: LanguageIdentifier = langid!("en-US"); const FRENCH: LanguageIdentifier = langid!("fr"); const GERMAN: LanguageIdentifier = langid!("de");

staticloader! { static LOCALES = { locales: "./tests/locales", fallbacklanguage: "en-US", // Removes unicode isolating marks around arguments, you typically // should only set to false when testing. customise: |bundle| bundle.setuseisolating(false), }; }

fn main() { asserteq!("Hello World!", LOCALES.lookup(&USENGLISH, "hello-world")); asserteq!("Bonjour le monde!", LOCALES.lookup(&FRENCH, "hello-world")); asserteq!("Hallo Welt!", LOCALES.lookup(&GERMAN, "hello-world"));

let args = {
    let mut map = HashMap::new();
    map.insert(String::from("name"), "Alice".into());
    map
};

assert_eq!("Hello Alice!", LOCALES.lookup_with_args(&US_ENGLISH, "greeting", &args));
assert_eq!("Bonjour Alice!", LOCALES.lookup_with_args(&FRENCH, "greeting", &args));
assert_eq!("Hallo Alice!", LOCALES.lookup_with_args(&GERMAN, "greeting", &args));

} ```

Tera

With the tera feature you can use FluentLoader as a Tera function. It accepts a key parameter pointing to a fluent resource and lang for what language to get that key for. Optionally you can pass extra arguments to the function as arguments to the resource. fluent-templates will automatically convert argument keys from Tera's snake_case to the fluent's preferred kebab-case arguments.

toml fluent-templates = { version = "*", features = ["tera"] }

```rust use fluenttemplates::{FluentLoader, staticloader};

staticloader! { static LOCALES = { locales: "./tests/locales", fallbacklanguage: "en-US", // Removes unicode isolating marks around arguments, you typically // should only set to false when testing. customise: |bundle| bundle.setuseisolating(false), }; }

fn main() { let mut tera = tera::Tera::default(); let ctx = tera::Context::default(); tera.registerfunction("fluent", FluentLoader::new(&*LOCALES)); asserteq!( "Hello World!", tera.renderstr(r#"{{ fluent(key="hello-world", lang="en-US") }}"#, &ctx).unwrap() ); asserteq!( "Hello Alice!", tera.render_str(r#"{{ fluent(key="greeting", lang="en-US", name="Alice") }}"#, &ctx).unwrap() ); } ```

Handlebars

In handlebars, fluent-templates will read the lang field in your [handlebars::Context] while rendering.

toml fluent-templates = { version = "*", features = ["handlebars"] }

```rust use fluenttemplates::{FluentLoader, staticloader};

staticloader! { static LOCALES = { locales: "./tests/locales", fallbacklanguage: "en-US", // Removes unicode isolating marks around arguments, you typically // should only set to false when testing. customise: |bundle| bundle.setuseisolating(false), }; }

fn main() { let mut handlebars = handlebars::Handlebars::new(); handlebars.registerhelper("fluent", Box::new(FluentLoader::new(&*LOCALES))); let data = serdejson::json!({"lang": "zh-CN"}); asserteq!("Hello World!", handlebars.rendertemplate(r#"{{fluent "hello-world"}}"#, &data).unwrap()); asserteq!("Hello Alice!", handlebars.rendertemplate(r#"{{fluent "greeting" name="Alice"}}"#, &data).unwrap()); } ```

Handlebars helper syntax.

The main helper provided is the {{fluent}} helper. If you have the following Fluent file:

fluent foo-bar = "foo bar" placeholder = this has a placeholder { $variable } placeholder2 = this has { $variable1 } { $variable2 }

You can include the strings in your template with

hbs <!-- will render "foo bar" --> {{fluent "foo-bar"}} <!-- will render "this has a placeholder baz" --> {{fluent "placeholder" variable="baz"}}

You may also use the {{fluentparam}} helper to specify [variables], especially if you need them to be multiline.

hbs {{#fluent "placeholder2"}} {{#fluentparam "variable1"}} first line second line {{/fluentparam}} {{#fluentparam "variable2"}} first line second line {{/fluentparam}} {{/fluent}}

FAQ

Why is there extra characters around the values of arguments?

These are called "Unicode Isolating Marks" that used to allow the text to be bidirectional. You can disable this with FluentBundle::set_isolating_marks being set to false.

rust static_loader! { static LOCALES = { locales: "./tests/locales", fallback_language: "en-US", // Removes unicode isolating marks around arguments. customise: |bundle| bundle.set_use_isolating(false), }; }