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
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(...)
- DOMAIN
- The argument to
bindtextdomain("<DOMAIN>", "<LOCALEDIR>")
- The resulting files will be
DOMAIN.pot
,DOMAIN.po
, etc.
- The argument to
- TARGET_NAME
- The name of the target which maintains the gettext files
- Note that the target is a
UTILITY
target, so you cannot usetarget_link_libraries()
, you must useadd_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
- The source-files to search for gettext strings with
- 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
- Add the target to the default, ie.
- 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
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
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" "--msgid-bugs-address=example@website.com"
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
*/