Skip to content

Function for easy gettext additions to your CMake project

License

Notifications You must be signed in to change notification settings

nrbrook/Gettext-CMake

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

9 Commits
 
 
 
 
 
 

Repository files navigation

Gettext_helpers.cmake

Assist in i18n for CMake projects

This repo provides Gettext_helpers.cmake, which implements the following function:

configure_gettext(
    DOMAIN <domain-name>
    TARGET_NAME <target-name>
    SOURCES <file> ...
    POTFILE_DESTINATION <dir>
    [POFILE_DESTINATION <dir>]
    [GMOFILE_DESTINATION <dir>]
    LANGUAGES <file> ...
    [ALL]
    [INSTALL_DESTINATION <dest>]
    [INSTALL_COMPONENT <dest>]
    [XGETTEXT_ARGS <args> ...
    [MSGMERGE_ARGS <args> ...]
    [MSGINIT_ARGS <args> ...]
    [MSGFMT_ARGS <args> ... ]
    )

configure_gettext creates a target which creates and maintains a .pot file, .po files, and .gmo files, as well as optional install rules

Installation

Copy Gettext_helpers.cmake into your project and ensure its directory is on the CMAKE_MODULE_PATH

list(APPEND CMAKE_MODULE_PATH <Gettext_helpers.cmake location>)

Then simply call include(Gettext_helpers.cmake) and configure_gettext(...)

Required arguments

  • DOMAIN
    • The argument to bindtextdomain("<DOMAIN>", "<LOCALEDIR>")
    • The resulting files will be DOMAIN.pot, DOMAIN.po, etc.
  • TARGET_NAME
    • The name of the target which maintains the gettext files
    • Note that the target is a UTILITY target, so you cannot use target_link_libraries(), you must use add_dependency(main-tgt gettext-tgt)
  • SOURCES
    • The source-files to search for gettext strings with xgettext
    • Note that by default no arguments (outside "--output") are passed to xgettext, so additional keywords such as _ need to be specified with XGETTEXT_ARGS
  • LANGUAGES
    • The languages to support. This will create directories "<POFILE_LOCATION>/<LANG>/" to hold .po files, and the directory structure for installation. See the gettext documentation on how specific you need to be with your language, ("en" vs "en_US" vs "en_US.UTF-8")
  • ALL
    • Add the target to the default, ie. make all will compile it
  • POTFILE_DESTINATION
    • Where to store the .pot file (template file)
    • Note that this file is regenerated on source file update, so do not edit it. Set package-name, etc. with XGETTEXT_ARGS
  • POFILE_DESTINATION
    • Where to store the .po files (language-specific translation files)
    • A directory is created per .po file with the name of the given language
    • defaults to POTFILE_DESTINATION/po/
    • This file is never overwritten, only updated. To provide translations, update this file, but on source-file modification the file will be synced with the .pot file, so make certain you save before rebuilding
  • GMOFILE_DESTINATION
    • Where to store the .gmo files
    • defaults to POFILE_DESTINATION
  • INSTALL_DESTINATION
    • If set, the .gmo files will be installed to this directory in proper hierarchal structure
    • This should be a relative path, ex. share/locale/
  • INSTALL_COMPONENT
    • Which component to install the .gmo files with
  • <PROG>_ARGS
    • A list of arguments to pass to the given program
    • XGETTEXT_ARGS are most likely to be used to specify additional keywords and package details, while the others are not as useful

Note

If errors occur while building, simply delete the generated .pot file and re-run cmake.

A .po file is created using msginit, so prepare for your terminal to be hijaked during a CMake run when a .po file is not found. The .po files should be committed into your project repo, so this will only affect users if they add an additional language

Workflow / Tutorial

When starting a project, internationalization should not be the first priority, but should always be kept in mind. A reccomendation is to put the following in a private (local to the project) header:

#define _(STRING) STRING
#define N_(STRING) STRING
#define P_(SINGULAR, PLURAL, N) (N == 1u ? SINGULAR : PLURAL)

Then, after support is desired, change this to

/* If making an executable */
#include <libintl.h>
#define _(STRING) gettext(STRING)
#define N_(STRING) STRING
#define P_(SINGULAR, PLURAL, N) ngettext(SINGULAR, PLURAL, N)

/* If making a library */
#include <libintl.h>
#define _(STRING) dgettext("domain-name", STRING)
#define N_(STRING) STRING
#define P_(SINGULAR, PLURAL, N) dngettext("domain-name", SINGULAR, PLURAL, N)


# CMakeLists.txt
include(Gettext_helpers.cmake)
config_gettext(
    DOMAIN "domain-name"
    TARGET_NAME "my-app-gettext"
    SOURCES "myapp.c" "util.c"
    POTFILE_DESTINATION "share"
    XGETTEXT_ARGS 
        "--keyword=_" "--keyword=N_" "--keyword=P_:1,2"
        "--package-name=${PROJECT_NAME}" "--package-version=${PROJECT_VERSION}"
        "--copyright-holder=John Doe" "[email protected]"
    LANGUAGES "en_US.UTF-8" "fr_FR" "es")

find_package(Intl REQUIRED)

add_executable(my-app myapp.c util.c)
target_link_libraries(my-app ${Libintl_LIBRARY})
target_include_directories(my-app ${Libintl_INCLUDE_DIRS})
add_dependency(my-app my-app-gettext)

And at the start of the program's execution

bindtextdomain("domain-name", "locale-dir");
/* Only write the following 2 lines if creating an executable */
setlocale(LC_ALL, ""); /* Set locale based on ENV */
textdomain("domain-name");

Note that if "locale-dir" is relative, the program must not change directories, as gettext needs to be able to load .mo files on the fly after a setlocale call.

If you have a GUI program, you may want to enable custom language settings. You can always call setlocale(LC_*, "locale-name") to manually set the locale after, for example, a different option is selected.

On non-Linux platforms, choosing a non-relative "locale-dir" may be impossible, so try to initialize the program by determining where the locale directory is relative to the executable location and making it absolute. Ex.

const char *locale_relative = "../share/locale/";
char *current = get_exe_location();
char *locale_absolute;
asnprintf(&locale_absolute, "%s/%s", current, locale_relative);

bindtextdomain("domain-name", locale_absolute);
/* You can immediately destroy the locale argument, gettext makes a copy */
free(locale_absolute);

/* Now you can freely chdir(), as long as the user doesn't move the
 * locale directory itself
 */

About

Function for easy gettext additions to your CMake project

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • CMake 100.0%