npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@absolutesight/react-gettext

v1.0.5

Published

Gettext implementation for React based project.

Downloads

258

Readme

@absolutesight/react-gettext

A tiny React library that helps to implement internalization in your application using gettext functions. It uses React Context API to expose gettext functions to children components.

Instalation

Note: This library requires React 16.3 or later

npm i react @absolutesight/react-gettext

Usage

To use this library in your application, you need to do a few simple steps:

  1. Prepare translations and define plural form functions.
  2. Add TextDomainContext.Provider provider to the root of your application.
  3. Updated your components to use context functions, provided by TextDomainContext, to translate text messages.

Let's take a closer look at each step. First of all, you to create translation catalogs and prepare plural form functions for every language that you are going to use. Each language needs one catalog and one plural form function.

The translation catalog is an object that contains key/value pairs where keys are original singular messages and values are translations. If you have a message that can have plural forms, the value for it should be an array with translations where each translation corresponds to appropriate plural form. Finally, if you want to use a context with your messages, then it should be prepended to the message itself and separated by using \u0004 (end of transition) character. Here is an example:

{
	"Hello world!": "¡Hola Mundo!", // regular message
	"article": ["artículo", "artículos"], // plural version
	"Logo link\u0004Homepage": "Página principal", // single message with "Logo link" contex
	"Search results count\u0004article": ["artículo", "artículos"], // plural version with "Search results count" context
}

The plural form function is a function that determines the number of a plural form that should be used for a particular translation. For English, the plural form is n != 1 ? 1 : 0, that means to use a translation with 0 index when n == 1, and a translation with 1 index in all other cases. Slavic and arabic languages have more than 2 plural forms and their functions are more complicated. Translate Toolkit has a list of plural forms expressions for many languages that you can use in your project. An example of a plural form function can be the following:

function getPluralForm(n) {
  return n != 1 ? 1 : 0;
}

The next step is to pass translations and plural form function for the current language to the buildTextDomain function. It will create APIs that need to be passed to the TextDomainContext.Provider provider that you need to add to the root of your project:

import React, { Component } from 'react';
import { TextDomainContext, buildTextDomain } from '@absolutesight/react-gettext';

class MyApp extends Component {

    constructor(props) {
        super(props);
        this.state = { textDomain: buildTextDomain(...) };
    }

    render() {
        return (
            <div>
                <TextDomainContext.Provider value={this.state.textDomain}>
                    <ComponentA />
                    ...
                </TextDomainContext>
            </div>
        );
    }

}

Note: Please, pay attention that you need to avoid passing the results of buildTextDomain function directly into TextDomainContext.Provider's value to escape unintentional renders in consumers when a provider’s parent re-renders.

Finally, the last step is to update your descendant components to consume these context APIs. Import TexdomainContext in the child component and assign it to the component contextType static properly. It will expose gettext APIs to that component via this.context field:

import React, { Component } from "react";
import { TextDomainContext } from "@absolutesight/react-gettext";

class ComponentA extends Component {
  render() {
    const { gettext, ngettext, xgettext, nxgettext } = this.context;

    return <div>...</div>;
  }
}

ComponentA.contextType = TextDomainContext;

An example

Let's assume you have following React application:

// app.js
import React, { Component } from "react";
import Header from "./Header";
import Footer from "./Footer";

export default class App extends Component {
  render() {
    return (
      <div id="app">
        <Header />
        ...
        <Footer />
      </div>
    );
  }
}
// Header.js
import React, { Component } from "react";

export default class Header extends Component {
  render() {
    return <h1>Welcome to my application!</h1>;
  }
}

To make it translatable, you need to update your app.js file to use TextDomainContext provider and build textDomain using messages list and plural form function:

  // app.js
  import React, { Component } from 'react';
+ import { TextDomainContext, buildTextDomain } from '@absolutesight/react-gettext';
  import Header from './Header';
  import Footer from './Footer';

  export default class App extends Component {

+     constructor(props) {
+         super(props);
+         this.state = {
+             textDomain: buildTextDomain(
+                 {
+                     'Welcome to my application!': 'Bienvenido a mi aplicación!',
+                     // ...
+                 },
+                 n => n != 1
+             ),
+         };
+     }

      render() {
          return (
              <div id="app">
+                 <TextDomainContext.Provider value={this.state.textDomain}>
                      <Header />
                      ...
                      <Footer />
+                 </TextDomainContext.Provider>
              </div>
          );
      }
  }

After doing it you can start using gettext, ngettext, xgettext and nxgettext functions in your descending components:

  // Header.js
  import React, { Component } from 'react';
+ import { TextDomainContext } from 'absolutesight/react-gettext';

  export default class Header extends Component {

      render() {
+         const { gettext } = this.context;
          return (
-             <h1>Welcome to my application!</h1>
+             <h1>{gettext('Welcome to my application!')}</h1>
          );
      }

  }

+ Header.contextType = TextDomainContext;

Documentation

buildTextDomain(translations, pluralForm)

Builds gettext APIs for TextDomainContext provider that will work with provided translations.

  • translations: an object with translated messages.
  • pluralForm: a function that determines a plural form or a stringular reresentation of it.
const api = buildTextDomain( { ... }, n => n == 1 ? 0 : 1 );
// or
const api = buildTextDomain( { ... }, 'n == 1 ? 0 : 1' );

gettext(message)

The function to translate a string. Accepts original message and returns translation if it exists, otherwise original message.

  • message: a string to be translated.

Example:

// somewhere in your jsx component
this.context.gettext("Some text");

ngettext(singular, plural, n)

The function to translate plural string. Accepts singular and plural messages along with a number to calculate plural form against. Returns translated message based on plural form if it exists, otherwise original message based on n value.

  • singular: a string to be translated when count is not plural
  • plural: a string to be translated when count is plural
  • n: a number to count plural form

Example:

// somewhere in your jsx component
this.context.ngettext("day ago", "days ago", numberOfDays);

xgettext(message, context)

The function to translate a string based on a specific context. Accepts a message to translate and a translation context string. Returns translated message if it exists, otherwise original string.

  • message: A string to be translated.
  • context: A context to search translation in.

Example:

// somewhere in your jsx component
this.context.xgettext("some text", "context where this message is used");

nxgettext(singular, plural, n, context)

The function to translate plural string based on a specific context. Accepts singular and plural messages along with a number to calculate plural form against and context string. Returns translated message based on plural form if it exists, otherwise original message based on n value.

  • singular: a string to be translated when count is not plural
  • plural: a string to be translated when count is plural
  • n: a number to count plural form
  • context: A context to search translation in.

Example:

// somewhere in your jsx component
this.context.nxgettext(
  "day ago",
  "days ago",
  numberOfDays,
  "Article publish date"
);

Legacy API

The initial version of this library had been created when React used the legacy version of Context APIs, thus it played a keystone role in the main approach of how to use this library at that time. However, in the late March of 2018, React 16.3 was released and that API became deprecated, so do the main approach used in this library.

The proper way to use this library is described in the Usage section, this section contains legacy API that will be removed in next versions of the library. We don't encourage you to use it in a new project.

withGettext(translations, pluralForms, options)

Higher-order function which is exported by default from @absolutesight/react-gettext package. It accepts two arguments and returns function to create higher-order component.

  • translations: a hash object or a function which returns hash object where keys are original messages and values are translated messages.
  • pluralForms: a string to calculate plural form (used by Gettext PO) or a function which accepts a number and calculates a plural form number. Pay attentions that plural forms are zero-based what means to get 1st plural form it should return 0, to get 2nd - 1, and so on.
  • options: a hash object with options. Currently supports following options:
    • withRef: an optional boolean flag that determines whether or not to set ref property to a wrapped component what will allow you to get wrapped component instance by calling getWrappedComponent() function of the HOC. By default: FALSE.

Example:

const translations = {
    'Some text': 'Some translated text',
    ...
};

const pluralForms = '(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2)'; // 3 plural forms for Russian, Belarusian, Bosnian, Croatian, Serbian, Ukrainian, etc.

const HOC = withGettext(translations, pluralForms)(App);
function getTranslations() {
    return {
        'Some text': 'Some translated text',
        ...
    };
}

function getPluralForms(n) {
    return n > 1 ? 1 : 0;
}

const HOC = withGettext(getTranslations, getPluralForms)(App);

As an alternative you can pass translations and plural form as properties to higher-order-component, like this:

function getTranslations() {
    return {
        'Some text': 'Some translated text',
        ...
    };
}

function getPluralForms(n) {
    return n > 1 ? 1 : 0;
}

const HOC = withGettext()(App);

...

ReactDOM.render(<HOC translations={getTranslations} plural={getPluralForms}>...</HOC>, ...);

One more alternative is to not create HOC, but use TextDomain component directly. You can import it using import { Textdomain } from '@absolutesight/react-gettext' and use it as a regular component which will provide context functions to translate your messages. Just don't forget to pass translations and plural props to this component when you render it.

Poedit

If you want to use Poedit application to translate your messages, then use the following keywords to properly extract static copy from your javascript files:

gettext;ngettext:1,2;xgettext:1,2c;nxgettext:1,2,4c

Here is an example of a POT file that you can use as a starting point:

msgid ""
msgstr ""
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
"Project-Id-Version: \n"
"POT-Creation-Date: \n"
"PO-Revision-Date: \n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=iso-8859-1\n"
"Content-Transfer-Encoding: 8bit\n"
"X-Poedit-Basepath: ./src\n"
"X-Poedit-KeywordsList: gettext;ngettext:1,2;xgettext:1,2c;nxgettext:1,2,4c\n"
"X-Poedit-SourceCharset: UTF-8\n"

If you prefer using npm scripts, then you can add the following command to your package.json file to extract static copy and generate POT file using CLI commands. Make sure, you have correct project and output paths.

"gettext:extract": "find /path/to/project -name \"*.js\" | xargs xgettext --from-code=UTF-8 --language=JavaScript --keyword=gettext --keyword=ngettext:1,2 --keyword=xgettext:1,2c --keyword=nxgettext:1,2,4c --output=/path/to/project/projectname.pot --sort-by-file --package-name=\"My Project Name\" --package-version=\"1.0.0\""

Contribute

I will provide this information later

License

MIT