experteer-gettext 2.0.4
Sign up to get free protection for your applications and to get access to all the features.
- data/COPYING +55 -0
- data/ChangeLog +57 -0
- data/ChangeLog-1 +2016 -0
- data/NEWS-1 +414 -0
- data/README.rdoc +251 -0
- data/Rakefile +260 -0
- data/VERSION +2 -0
- data/bin/rgettext +24 -0
- data/bin/rmsgfmt +24 -0
- data/bin/rmsgmerge +24 -0
- data/gettext.gemspec +494 -0
- data/lib/gettext/cgi.rb +39 -0
- data/lib/gettext/class_info.rb +67 -0
- data/lib/gettext/core_ext/iconv.rb +112 -0
- data/lib/gettext/core_ext/string.rb +84 -0
- data/lib/gettext/locale_path.rb +123 -0
- data/lib/gettext/mofile.rb +330 -0
- data/lib/gettext/parser/erb.rb +54 -0
- data/lib/gettext/parser/glade.rb +100 -0
- data/lib/gettext/parser/ruby.rb +193 -0
- data/lib/gettext/poparser.rb +355 -0
- data/lib/gettext/textdomain.rb +179 -0
- data/lib/gettext/textdomain_group.rb +24 -0
- data/lib/gettext/textdomain_manager.rb +212 -0
- data/lib/gettext/tools/rgettext.rb +236 -0
- data/lib/gettext/tools/rmsgfmt.rb +84 -0
- data/lib/gettext/tools/rmsgmerge.rb +498 -0
- data/lib/gettext/tools.rb +195 -0
- data/lib/gettext/translation_target.rb +93 -0
- data/lib/gettext/utils.rb +37 -0
- data/lib/gettext/version.rb +12 -0
- data/lib/gettext.rb +312 -0
- data/po/bg/rgettext.po +130 -0
- data/po/bs/rgettext.po +131 -0
- data/po/ca/rgettext.po +129 -0
- data/po/cs/rgettext.po +132 -0
- data/po/de/rgettext.po +137 -0
- data/po/el/rgettext.po +128 -0
- data/po/eo/rgettext.po +129 -0
- data/po/es/rgettext.po +130 -0
- data/po/et/rgettext.po +127 -0
- data/po/fr/rgettext.po +133 -0
- data/po/hr/rgettext.po +131 -0
- data/po/hu/rgettext.po +129 -0
- data/po/it/rgettext.po +130 -0
- data/po/ja/rgettext.po +129 -0
- data/po/ko/rgettext.po +129 -0
- data/po/lv/rgettext.po +130 -0
- data/po/nb/rgettext.po +131 -0
- data/po/nl/rgettext.po +130 -0
- data/po/pt_BR/rgettext.po +132 -0
- data/po/rgettext.pot +121 -0
- data/po/ru/rgettext.po +131 -0
- data/po/sr/rgettext.po +130 -0
- data/po/sv/rgettext.po +123 -0
- data/po/ua/rgettext.po +134 -0
- data/po/vi/rgettext.po +128 -0
- data/po/zh/rgettext.po +129 -0
- data/po/zh_TW/rgettext.po +128 -0
- data/samples/README +16 -0
- data/samples/cgi/README +43 -0
- data/samples/cgi/Rakefile +27 -0
- data/samples/cgi/cookie.cgi +64 -0
- data/samples/cgi/gettext.css +116 -0
- data/samples/cgi/helloerb.rhtml +28 -0
- data/samples/cgi/helloerb1.cgi +58 -0
- data/samples/cgi/helloerb2.cgi +51 -0
- data/samples/cgi/hellolib.rb +20 -0
- data/samples/cgi/http.rb +51 -0
- data/samples/cgi/index.cgi +111 -0
- data/samples/cgi/other.rhtml +20 -0
- data/samples/cgi/po/bg/helloerb1.po +59 -0
- data/samples/cgi/po/bg/helloerb2.po +51 -0
- data/samples/cgi/po/bg/hellolib.po +23 -0
- data/samples/cgi/po/bg/main.po +83 -0
- data/samples/cgi/po/bs/helloerb1.po +59 -0
- data/samples/cgi/po/bs/helloerb2.po +51 -0
- data/samples/cgi/po/bs/hellolib.po +23 -0
- data/samples/cgi/po/bs/main.po +83 -0
- data/samples/cgi/po/ca/helloerb1.po +59 -0
- data/samples/cgi/po/ca/helloerb2.po +51 -0
- data/samples/cgi/po/ca/hellolib.po +23 -0
- data/samples/cgi/po/ca/main.po +83 -0
- data/samples/cgi/po/cs/helloerb1.po +61 -0
- data/samples/cgi/po/cs/helloerb2.po +52 -0
- data/samples/cgi/po/cs/hellolib.po +25 -0
- data/samples/cgi/po/cs/main.po +85 -0
- data/samples/cgi/po/de/helloerb1.po +61 -0
- data/samples/cgi/po/de/helloerb2.po +52 -0
- data/samples/cgi/po/de/hellolib.po +24 -0
- data/samples/cgi/po/de/main.po +86 -0
- data/samples/cgi/po/el/helloerb1.po +60 -0
- data/samples/cgi/po/el/helloerb2.po +51 -0
- data/samples/cgi/po/el/hellolib.po +23 -0
- data/samples/cgi/po/el/main.po +84 -0
- data/samples/cgi/po/eo/helloerb1.po +60 -0
- data/samples/cgi/po/eo/helloerb2.po +52 -0
- data/samples/cgi/po/eo/hellolib.po +24 -0
- data/samples/cgi/po/eo/main.po +84 -0
- data/samples/cgi/po/es/helloerb1.po +59 -0
- data/samples/cgi/po/es/helloerb2.po +50 -0
- data/samples/cgi/po/es/hellolib.po +22 -0
- data/samples/cgi/po/es/main.po +83 -0
- data/samples/cgi/po/fr/helloerb1.po +59 -0
- data/samples/cgi/po/fr/helloerb2.po +51 -0
- data/samples/cgi/po/fr/hellolib.po +22 -0
- data/samples/cgi/po/fr/main.po +85 -0
- data/samples/cgi/po/helloerb1.pot +60 -0
- data/samples/cgi/po/helloerb2.pot +52 -0
- data/samples/cgi/po/hellolib.pot +24 -0
- data/samples/cgi/po/hr/helloerb1.po +59 -0
- data/samples/cgi/po/hr/helloerb2.po +51 -0
- data/samples/cgi/po/hr/hellolib.po +23 -0
- data/samples/cgi/po/hr/main.po +83 -0
- data/samples/cgi/po/hu/helloerb1.po +59 -0
- data/samples/cgi/po/hu/helloerb2.po +51 -0
- data/samples/cgi/po/hu/hellolib.po +23 -0
- data/samples/cgi/po/hu/main.po +82 -0
- data/samples/cgi/po/it/helloerb1.po +60 -0
- data/samples/cgi/po/it/helloerb2.po +52 -0
- data/samples/cgi/po/it/hellolib.po +24 -0
- data/samples/cgi/po/it/main.po +84 -0
- data/samples/cgi/po/ja/helloerb1.po +60 -0
- data/samples/cgi/po/ja/helloerb2.po +52 -0
- data/samples/cgi/po/ja/hellolib.po +24 -0
- data/samples/cgi/po/ja/main.po +85 -0
- data/samples/cgi/po/ko/helloerb1.po +59 -0
- data/samples/cgi/po/ko/helloerb2.po +51 -0
- data/samples/cgi/po/ko/hellolib.po +23 -0
- data/samples/cgi/po/ko/main.po +84 -0
- data/samples/cgi/po/lv/helloerb1.po +65 -0
- data/samples/cgi/po/lv/helloerb2.po +52 -0
- data/samples/cgi/po/lv/hellolib.po +24 -0
- data/samples/cgi/po/lv/main.po +77 -0
- data/samples/cgi/po/main.pot +80 -0
- data/samples/cgi/po/nb/helloerb1.po +60 -0
- data/samples/cgi/po/nb/helloerb2.po +52 -0
- data/samples/cgi/po/nb/hellolib.po +24 -0
- data/samples/cgi/po/nb/main.po +84 -0
- data/samples/cgi/po/nl/helloerb1.po +61 -0
- data/samples/cgi/po/nl/helloerb2.po +52 -0
- data/samples/cgi/po/nl/hellolib.po +24 -0
- data/samples/cgi/po/nl/main.po +86 -0
- data/samples/cgi/po/pt_BR/helloerb1.po +59 -0
- data/samples/cgi/po/pt_BR/helloerb2.po +51 -0
- data/samples/cgi/po/pt_BR/hellolib.po +23 -0
- data/samples/cgi/po/pt_BR/main.po +84 -0
- data/samples/cgi/po/ru/helloerb1.po +58 -0
- data/samples/cgi/po/ru/helloerb2.po +50 -0
- data/samples/cgi/po/ru/hellolib.po +22 -0
- data/samples/cgi/po/ru/main.po +82 -0
- data/samples/cgi/po/sr/helloerb1.po +60 -0
- data/samples/cgi/po/sr/helloerb2.po +52 -0
- data/samples/cgi/po/sr/hellolib.po +24 -0
- data/samples/cgi/po/sr/main.po +80 -0
- data/samples/cgi/po/ua/helloerb1.po +62 -0
- data/samples/cgi/po/ua/helloerb2.po +54 -0
- data/samples/cgi/po/ua/hellolib.po +26 -0
- data/samples/cgi/po/ua/main.po +84 -0
- data/samples/cgi/po/vi/helloerb1.po +65 -0
- data/samples/cgi/po/vi/helloerb2.po +52 -0
- data/samples/cgi/po/vi/hellolib.po +24 -0
- data/samples/cgi/po/vi/main.po +77 -0
- data/samples/cgi/po/zh/helloerb1.po +60 -0
- data/samples/cgi/po/zh/helloerb2.po +52 -0
- data/samples/cgi/po/zh/hellolib.po +24 -0
- data/samples/cgi/po/zh/main.po +80 -0
- data/samples/cgi/po/zh_TW/helloerb1.po +67 -0
- data/samples/cgi/po/zh_TW/helloerb2.po +54 -0
- data/samples/cgi/po/zh_TW/hellolib.po +26 -0
- data/samples/cgi/po/zh_TW/main.po +79 -0
- data/samples/hello.rb +36 -0
- data/samples/hello2.rb +23 -0
- data/samples/hello_glade2.glade +70 -0
- data/samples/hello_glade2.rb +25 -0
- data/samples/hello_gtk2.rb +27 -0
- data/samples/hello_noop.rb +31 -0
- data/samples/hello_plural.rb +26 -0
- data/samples/hello_tk.rb +19 -0
- data/samples/makemo.rb +4 -0
- data/samples/po/bg/hello.po +24 -0
- data/samples/po/bg/hello2.po +31 -0
- data/samples/po/bg/hello_glade2.po +31 -0
- data/samples/po/bg/hello_gtk.po +23 -0
- data/samples/po/bg/hello_noop.po +27 -0
- data/samples/po/bg/hello_plural.po +25 -0
- data/samples/po/bg/hello_tk.po +23 -0
- data/samples/po/bs/hello.po +23 -0
- data/samples/po/bs/hello2.po +31 -0
- data/samples/po/bs/hello_glade2.po +31 -0
- data/samples/po/bs/hello_gtk.po +23 -0
- data/samples/po/bs/hello_noop.po +27 -0
- data/samples/po/bs/hello_plural.po +26 -0
- data/samples/po/bs/hello_tk.po +23 -0
- data/samples/po/ca/hello.po +23 -0
- data/samples/po/ca/hello2.po +31 -0
- data/samples/po/ca/hello_glade2.po +31 -0
- data/samples/po/ca/hello_gtk.po +23 -0
- data/samples/po/ca/hello_noop.po +27 -0
- data/samples/po/ca/hello_plural.po +25 -0
- data/samples/po/ca/hello_tk.po +23 -0
- data/samples/po/cs/hello.po +23 -0
- data/samples/po/cs/hello2.po +31 -0
- data/samples/po/cs/hello_glade2.po +37 -0
- data/samples/po/cs/hello_gtk.po +23 -0
- data/samples/po/cs/hello_noop.po +27 -0
- data/samples/po/cs/hello_plural.po +26 -0
- data/samples/po/cs/hello_tk.po +23 -0
- data/samples/po/de/hello.po +20 -0
- data/samples/po/de/hello2.po +28 -0
- data/samples/po/de/hello_glade2.po +27 -0
- data/samples/po/de/hello_gtk.po +20 -0
- data/samples/po/de/hello_noop.po +24 -0
- data/samples/po/de/hello_plural.po +25 -0
- data/samples/po/de/hello_tk.po +20 -0
- data/samples/po/el/hello.po +23 -0
- data/samples/po/el/hello2.po +31 -0
- data/samples/po/el/hello_glade2.po +31 -0
- data/samples/po/el/hello_gtk.po +22 -0
- data/samples/po/el/hello_noop.po +27 -0
- data/samples/po/el/hello_plural.po +25 -0
- data/samples/po/el/hello_tk.po +23 -0
- data/samples/po/eo/hello.po +23 -0
- data/samples/po/eo/hello2.po +31 -0
- data/samples/po/eo/hello_glade2.po +32 -0
- data/samples/po/eo/hello_gtk.po +23 -0
- data/samples/po/eo/hello_noop.po +27 -0
- data/samples/po/eo/hello_plural.po +26 -0
- data/samples/po/eo/hello_tk.po +24 -0
- data/samples/po/es/hello.po +21 -0
- data/samples/po/es/hello2.po +28 -0
- data/samples/po/es/hello_glade2.po +28 -0
- data/samples/po/es/hello_gtk.po +20 -0
- data/samples/po/es/hello_noop.po +24 -0
- data/samples/po/es/hello_plural.po +23 -0
- data/samples/po/es/hello_tk.po +20 -0
- data/samples/po/fr/hello.po +18 -0
- data/samples/po/fr/hello2.po +26 -0
- data/samples/po/fr/hello_glade2.po +27 -0
- data/samples/po/fr/hello_gtk.po +18 -0
- data/samples/po/fr/hello_noop.po +22 -0
- data/samples/po/fr/hello_plural.po +21 -0
- data/samples/po/fr/hello_tk.po +18 -0
- data/samples/po/hello.pot +23 -0
- data/samples/po/hello2.pot +31 -0
- data/samples/po/hello_glade2.pot +32 -0
- data/samples/po/hello_gtk.pot +23 -0
- data/samples/po/hello_noop.pot +27 -0
- data/samples/po/hello_plural.pot +26 -0
- data/samples/po/hello_tk.pot +24 -0
- data/samples/po/hr/hello.po +23 -0
- data/samples/po/hr/hello2.po +31 -0
- data/samples/po/hr/hello_glade2.po +31 -0
- data/samples/po/hr/hello_gtk.po +23 -0
- data/samples/po/hr/hello_noop.po +27 -0
- data/samples/po/hr/hello_plural.po +26 -0
- data/samples/po/hr/hello_tk.po +23 -0
- data/samples/po/hu/hello.po +22 -0
- data/samples/po/hu/hello2.po +30 -0
- data/samples/po/hu/hello_glade2.po +31 -0
- data/samples/po/hu/hello_gtk.po +22 -0
- data/samples/po/hu/hello_noop.po +26 -0
- data/samples/po/hu/hello_plural.po +25 -0
- data/samples/po/hu/hello_tk.po +23 -0
- data/samples/po/it/hello.po +20 -0
- data/samples/po/it/hello2.po +28 -0
- data/samples/po/it/hello_glade2.po +28 -0
- data/samples/po/it/hello_gtk.po +21 -0
- data/samples/po/it/hello_noop.po +24 -0
- data/samples/po/it/hello_plural.po +23 -0
- data/samples/po/it/hello_tk.po +21 -0
- data/samples/po/ja/hello.po +20 -0
- data/samples/po/ja/hello2.po +28 -0
- data/samples/po/ja/hello_glade2.po +26 -0
- data/samples/po/ja/hello_gtk.po +19 -0
- data/samples/po/ja/hello_noop.po +23 -0
- data/samples/po/ja/hello_plural.po +21 -0
- data/samples/po/ja/hello_tk.po +19 -0
- data/samples/po/ko/hello.po +18 -0
- data/samples/po/ko/hello2.po +26 -0
- data/samples/po/ko/hello_glade2.po +29 -0
- data/samples/po/ko/hello_gtk.po +18 -0
- data/samples/po/ko/hello_noop.po +22 -0
- data/samples/po/ko/hello_plural.po +25 -0
- data/samples/po/ko/hello_tk.po +19 -0
- data/samples/po/lv/hello.po +24 -0
- data/samples/po/lv/hello2.po +32 -0
- data/samples/po/lv/hello_glade2.po +38 -0
- data/samples/po/lv/hello_gtk.po +24 -0
- data/samples/po/lv/hello_noop.po +28 -0
- data/samples/po/lv/hello_plural.po +26 -0
- data/samples/po/lv/hello_tk.po +24 -0
- data/samples/po/nb/hello.po +23 -0
- data/samples/po/nb/hello2.po +31 -0
- data/samples/po/nb/hello_glade2.po +31 -0
- data/samples/po/nb/hello_gtk.po +23 -0
- data/samples/po/nb/hello_noop.po +27 -0
- data/samples/po/nb/hello_plural.po +26 -0
- data/samples/po/nb/hello_tk.po +24 -0
- data/samples/po/nl/hello.po +24 -0
- data/samples/po/nl/hello2.po +32 -0
- data/samples/po/nl/hello_glade2.po +31 -0
- data/samples/po/nl/hello_gtk.po +24 -0
- data/samples/po/nl/hello_noop.po +28 -0
- data/samples/po/nl/hello_plural.po +25 -0
- data/samples/po/nl/hello_tk.po +24 -0
- data/samples/po/pt_BR/hello.po +21 -0
- data/samples/po/pt_BR/hello2.po +29 -0
- data/samples/po/pt_BR/hello_glade2.po +29 -0
- data/samples/po/pt_BR/hello_gtk.po +21 -0
- data/samples/po/pt_BR/hello_noop.po +25 -0
- data/samples/po/pt_BR/hello_plural.po +23 -0
- data/samples/po/pt_BR/hello_tk.po +21 -0
- data/samples/po/ru/hello.po +22 -0
- data/samples/po/ru/hello2.po +30 -0
- data/samples/po/ru/hello_glade2.po +30 -0
- data/samples/po/ru/hello_gtk.po +22 -0
- data/samples/po/ru/hello_noop.po +26 -0
- data/samples/po/ru/hello_plural.po +28 -0
- data/samples/po/ru/hello_tk.po +22 -0
- data/samples/po/sr/hello.po +22 -0
- data/samples/po/sr/hello2.po +30 -0
- data/samples/po/sr/hello_glade2.po +32 -0
- data/samples/po/sr/hello_gtk.po +22 -0
- data/samples/po/sr/hello_noop.po +26 -0
- data/samples/po/sr/hello_plural.po +26 -0
- data/samples/po/sr/hello_tk.po +24 -0
- data/samples/po/sv/hello.po +20 -0
- data/samples/po/sv/hello2.po +28 -0
- data/samples/po/sv/hello_glade2.po +28 -0
- data/samples/po/sv/hello_gtk.po +20 -0
- data/samples/po/sv/hello_noop.po +24 -0
- data/samples/po/sv/hello_plural.po +23 -0
- data/samples/po/sv/hello_tk.po +20 -0
- data/samples/po/test.rb +11 -0
- data/samples/po/ua/hello.po +22 -0
- data/samples/po/ua/hello2.po +30 -0
- data/samples/po/ua/hello_glade2.po +34 -0
- data/samples/po/ua/hello_gtk.po +22 -0
- data/samples/po/ua/hello_noop.po +26 -0
- data/samples/po/ua/hello_plural.po +29 -0
- data/samples/po/ua/hello_tk.po +26 -0
- data/samples/po/vi/hello.po +23 -0
- data/samples/po/vi/hello2.po +31 -0
- data/samples/po/vi/hello_glade2.po +38 -0
- data/samples/po/vi/hello_gtk.po +23 -0
- data/samples/po/vi/hello_noop.po +27 -0
- data/samples/po/vi/hello_plural.po +26 -0
- data/samples/po/vi/hello_tk.po +24 -0
- data/samples/po/zh/hello.po +23 -0
- data/samples/po/zh/hello2.po +31 -0
- data/samples/po/zh/hello_glade2.po +31 -0
- data/samples/po/zh/hello_gtk.po +23 -0
- data/samples/po/zh/hello_noop.po +27 -0
- data/samples/po/zh/hello_plural.po +25 -0
- data/samples/po/zh/hello_tk.po +23 -0
- data/samples/po/zh_TW/hello.po +26 -0
- data/samples/po/zh_TW/hello2.po +34 -0
- data/samples/po/zh_TW/hello_glade2.po +40 -0
- data/samples/po/zh_TW/hello_gtk.po +25 -0
- data/samples/po/zh_TW/hello_noop.po +30 -0
- data/samples/po/zh_TW/hello_plural.po +28 -0
- data/samples/po/zh_TW/hello_tk.po +26 -0
- data/test/README +1 -0
- data/test/Rakefile +17 -0
- data/test/po/cr/plural.po +23 -0
- data/test/po/da/plural.po +22 -0
- data/test/po/da/plural_error.po +22 -0
- data/test/po/fr/plural.po +28 -0
- data/test/po/fr/plural_error.po +20 -0
- data/test/po/fr/test1.po +23 -0
- data/test/po/fr/test2.po +19 -0
- data/test/po/ir/plural.po +23 -0
- data/test/po/ja/npgettext.po +46 -0
- data/test/po/ja/nsgettext.po +65 -0
- data/test/po/ja/pgettext.po +41 -0
- data/test/po/ja/plural.po +28 -0
- data/test/po/ja/plural_error.po +20 -0
- data/test/po/ja/rubyparser.po +43 -0
- data/test/po/ja/sgettext.po +47 -0
- data/test/po/ja/test1.po +23 -0
- data/test/po/ja/test2.po +19 -0
- data/test/po/ja/test3.po +19 -0
- data/test/po/la/plural.po +23 -0
- data/test/po/la/plural_error.po +21 -0
- data/test/po/li/plural.po +23 -0
- data/test/po/li/plural_error.po +27 -0
- data/test/po/po/plural.po +23 -0
- data/test/po/sl/plural.po +24 -0
- data/test/test_class_info.rb +83 -0
- data/test/test_gettext.rb +311 -0
- data/test/test_locale_path.rb +50 -0
- data/test/test_parser.rb +157 -0
- data/test/test_po_generation.rb +20 -0
- data/test/test_string.rb +65 -0
- data/test/test_textdomain_bind.rb +37 -0
- data/test/test_textdomain_multi.rb +80 -0
- data/test/test_textdomain_toplevel.rb +42 -0
- data/test/test_thread.rb +41 -0
- data/test/test_translation_target.rb +29 -0
- data/test/testlib/N_.rb +66 -0
- data/test/testlib/erb.rhtml +15 -0
- data/test/testlib/erb.rxml +16 -0
- data/test/testlib/gettext.rb +113 -0
- data/test/testlib/gladeparser.glade +183 -0
- data/test/testlib/helper.rb +11 -0
- data/test/testlib/multi_textdomain.rb +131 -0
- data/test/testlib/ngettext.rb +79 -0
- data/test/testlib/npgettext.rb +31 -0
- data/test/testlib/nsgettext.rb +42 -0
- data/test/testlib/pgettext.rb +36 -0
- data/test/testlib/sgettext.rb +46 -0
- data/test/testlib/simple.rb +14 -0
- data/test/tools/files/app.pot +0 -0
- data/test/tools/files/de/app.po +0 -0
- data/test/tools/files/en/app.po +0 -0
- data/test/tools/files/en/test.po +21 -0
- data/test/tools/files/simple_1.po +2 -0
- data/test/tools/files/simple_2.po +2 -0
- data/test/tools/files/simple_translation.rb +1 -0
- data/test/tools/files/version.po +7 -0
- data/test/tools/test.pot +21 -0
- data/test/tools/test_tools.rb +61 -0
- metadata +510 -0
@@ -0,0 +1,195 @@
|
|
1
|
+
=begin
|
2
|
+
tools.rb - Utility functions
|
3
|
+
|
4
|
+
Copyright (C) 2005-2008 Masao Mutoh
|
5
|
+
|
6
|
+
You may redistribute it and/or modify it under the same
|
7
|
+
license terms as Ruby.
|
8
|
+
=end
|
9
|
+
|
10
|
+
require 'rbconfig'
|
11
|
+
if /mingw|mswin|mswin32/ =~ RUBY_PLATFORM
|
12
|
+
ENV['PATH'] = %w(bin lib).collect{|dir|
|
13
|
+
"#{Config::CONFIG["prefix"]}\\lib\\GTK\\#{dir};"
|
14
|
+
}.join('') + ENV['PATH']
|
15
|
+
end
|
16
|
+
|
17
|
+
require 'gettext/tools/rgettext'
|
18
|
+
require 'gettext/tools/rmsgfmt'
|
19
|
+
require 'gettext/mofile'
|
20
|
+
require 'fileutils'
|
21
|
+
|
22
|
+
module GetText
|
23
|
+
bindtextdomain "rgettext"
|
24
|
+
|
25
|
+
BOM_UTF8 = [0xef, 0xbb, 0xbf].pack("c3")
|
26
|
+
|
27
|
+
# Currently, GNU msgmerge doesn't accept BOM.
|
28
|
+
# This mesthod remove the UTF-8 BOM from the po-file.
|
29
|
+
def remove_bom(path) #:nodoc:
|
30
|
+
bom = IO.read(path, 3)
|
31
|
+
if bom == BOM_UTF8
|
32
|
+
data = IO.read(path)[3..-1]
|
33
|
+
File.open(path, "w") {|f| f.write(data)}
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
# Merges two Uniforum style .po files together.
|
38
|
+
#
|
39
|
+
# *Note* This function requires "msgmerge" tool included in GNU GetText. So you need to install GNU GetText.
|
40
|
+
#
|
41
|
+
# The def.po file is an existing PO file with translations which will be taken
|
42
|
+
# over to the newly created file as long as they still match; comments will be preserved,
|
43
|
+
# but extracted comments and file positions will be discarded.
|
44
|
+
#
|
45
|
+
# The ref.pot file is the last created PO file with up-to-date source references but
|
46
|
+
# old translations, or a PO Template file (generally created by rgettext);
|
47
|
+
# any translations or comments in the file will be discarded, however dot
|
48
|
+
# comments and file positions will be preserved. Where an exact match
|
49
|
+
# cannot be found, fuzzy matching is used to produce better results.
|
50
|
+
#
|
51
|
+
# Usually you don't need to call this function directly. Use GetText.update_pofiles instead.
|
52
|
+
#
|
53
|
+
# * defpo: a po-file. translations referring to old sources
|
54
|
+
# * refpo: a po-file. references to new sources
|
55
|
+
# * app_version: the application information which appears "Project-Id-Version: #{app_version}" in the pot/po-files.
|
56
|
+
# * Returns: self
|
57
|
+
def msgmerge(defpo, refpo, app_version, options={})
|
58
|
+
verbose = options.delete(:verbose)
|
59
|
+
puts "msgmerge called" if verbose
|
60
|
+
$stderr.print defpo + " "
|
61
|
+
|
62
|
+
content = merge_po_files(defpo,refpo,options.delete(:msgmerge),verbose)
|
63
|
+
|
64
|
+
if content.empty?
|
65
|
+
# report failure
|
66
|
+
failed_filename = refpo + "~"
|
67
|
+
FileUtils.cp(refpo, failed_filename)
|
68
|
+
$stderr.puts _("Failed to merge with %{defpo}") % {:defpo => defpo}
|
69
|
+
$stderr.puts _("New .pot was copied to %{failed_filename}") %{:failed_filename => failed_filename}
|
70
|
+
raise _("Check these po/pot-files. It may have syntax errors or something wrong.")
|
71
|
+
else
|
72
|
+
# update version and save merged data
|
73
|
+
content.sub!(/(Project-Id-Version\:).*$/, "\\1 #{app_version}\\n\"")
|
74
|
+
File.open(defpo, "w") {|f|f.write(content)}
|
75
|
+
end
|
76
|
+
|
77
|
+
self
|
78
|
+
end
|
79
|
+
|
80
|
+
# Creates mo-files using #{po_root}/#{lang}/*.po an put them to
|
81
|
+
# #{targetdir}/#{targetdir_rule}/.
|
82
|
+
#
|
83
|
+
# This is a convenience function of GetText.rmsgfmt for multiple target files.
|
84
|
+
# * options: options as a Hash.
|
85
|
+
# * verbose: true if verbose mode, otherwise false
|
86
|
+
# * po_root: the root directory of po-files.
|
87
|
+
# * mo_root: the target root directory where the mo-files are stored.
|
88
|
+
# * mo_path_rule: the target directory for each mo-files.
|
89
|
+
def create_mofiles(options = {})
|
90
|
+
options = {:po_root => "./po"}.merge(options)
|
91
|
+
|
92
|
+
Dir.glob(File.join(options[:po_root], "*/*.po")) do |po_file|
|
93
|
+
mo_file = mo_file_from_po_file(po_file,options)
|
94
|
+
$stderr.print %Q[#{po_file} -> #{mo_file} ... ] if options[:verbose]
|
95
|
+
FileUtils.mkdir_p(File.dirname(mo_file))
|
96
|
+
rmsgfmt(po_file, mo_file)
|
97
|
+
$stderr.puts "Done." if options[:verbose]
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
|
102
|
+
# At first, this creates the #{po_root}/#{domainname}.pot file using GetText.rgettext.
|
103
|
+
# In the second step, this updates(merges) the #{po_root}/#{domainname}.pot and all of the
|
104
|
+
# #{po_root}/#{lang}/#{domainname}.po files under "po_root" using "msgmerge".
|
105
|
+
#
|
106
|
+
# *Note* "msgmerge" tool is included in GNU GetText. So you need to install GNU GetText.
|
107
|
+
#
|
108
|
+
# See <HOWTO maintain po/mo files(http://www.yotabanana.com/hiki/ruby-gettext-howto-manage.html)> for more detals.
|
109
|
+
# * domainname: the textdomain name.
|
110
|
+
# * targetfiles: An Array of target files, that should be parsed for messages (See GetText.rgettext for more details).
|
111
|
+
# * app_version: the application information which appears "Project-Id-Version: #{app_version}" in the pot/po-files.
|
112
|
+
# * options: a hash with following possible settings
|
113
|
+
# :lang - update files only for one language - the language specified by this option
|
114
|
+
# :po_root - the root directory of po-files
|
115
|
+
# :msgmerge - an array with the options, passed through to the gnu msgmerge tool
|
116
|
+
# symbols are automatically translated to options with dashes,
|
117
|
+
# example: [:no_wrap, :no_fuzzy_matching, :sort_output] translated to '--no-fuzzy-matching --sort-output'
|
118
|
+
# :verbose - true to show verbose messages. default is false.
|
119
|
+
#
|
120
|
+
# Example: GetText.update_pofiles("myapp", Dir.glob("lib/*.rb"), "myapp 1.0.0", :verbose => true)
|
121
|
+
def update_pofiles(textdomain, files, app_version, options = {})
|
122
|
+
puts options.inspect if options[:verbose]
|
123
|
+
|
124
|
+
#write found messages to tmp.pot
|
125
|
+
temp_pot = "tmp.pot"
|
126
|
+
rgettext(files, temp_pot)
|
127
|
+
|
128
|
+
#merge tmp.pot and existing pot
|
129
|
+
po_root = options.delete(:po_root) || "po"
|
130
|
+
FileUtils.mkdir_p(po_root)
|
131
|
+
msgmerge("#{po_root}/#{textdomain}.pot", temp_pot, app_version, options.dup)
|
132
|
+
|
133
|
+
#update local po-files
|
134
|
+
only_one_language = options.delete(:lang)
|
135
|
+
if only_one_language
|
136
|
+
msgmerge("#{po_root}/#{only_one_language}/#{textdomain}.po", temp_pot, app_version, options.dup)
|
137
|
+
else
|
138
|
+
Dir.glob("#{po_root}/*/#{textdomain}.po") do |po_file|
|
139
|
+
msgmerge(po_file, temp_pot, app_version, options.dup)
|
140
|
+
end
|
141
|
+
end
|
142
|
+
|
143
|
+
File.delete(temp_pot)
|
144
|
+
end
|
145
|
+
|
146
|
+
private
|
147
|
+
|
148
|
+
# Merge 2 po files, using msgmerge
|
149
|
+
def merge_po_files(po_a,po_b,msgmerge_options=[],verbose=false)
|
150
|
+
return File.read(po_b) unless FileTest.exist? po_a
|
151
|
+
|
152
|
+
cmd = ENV["MSGMERGE_PATH"] || "msgmerge"
|
153
|
+
ensure_command_exists(cmd)
|
154
|
+
|
155
|
+
remove_bom(po_a)
|
156
|
+
|
157
|
+
cmd_params = array_to_cli_options(msgmerge_options)
|
158
|
+
to_run = "#{cmd} #{cmd_params} #{po_a} #{po_b}"
|
159
|
+
puts "\nrunning #{to_run}" if verbose
|
160
|
+
`#{to_run}`
|
161
|
+
end
|
162
|
+
|
163
|
+
# convert an array of String/Symbol to cli options
|
164
|
+
def array_to_cli_options(array)
|
165
|
+
[*array].map do |o|
|
166
|
+
o.kind_of?(Symbol) ? "--#{o}".gsub('_','-') : o.to_s
|
167
|
+
end.join(' ')
|
168
|
+
end
|
169
|
+
|
170
|
+
def ensure_command_exists(cmd)
|
171
|
+
`#{cmd} --help`
|
172
|
+
unless $? && $?.success?
|
173
|
+
raise _("`%{cmd}' can not be found. \nInstall GNU Gettext then set PATH or MSGMERGE_PATH correctly.") % {:cmd => cmd}
|
174
|
+
end
|
175
|
+
end
|
176
|
+
|
177
|
+
# where lies the mo file for a given po_file
|
178
|
+
# generare directory unless it exists
|
179
|
+
def mo_file_from_po_file(po_file,options)
|
180
|
+
options = {
|
181
|
+
:mo_root => "./data/locale",
|
182
|
+
:mo_path_rule => "%{lang}/LC_MESSAGES"
|
183
|
+
}.merge(options)
|
184
|
+
|
185
|
+
lang, textdomain = %r[/([^/]+?)/(.*)\.po].match(po_file[options[:po_root].size..-1]).to_a[1,2]
|
186
|
+
|
187
|
+
mo_dir_rule = File.join(options[:mo_root], options[:mo_path_rule])
|
188
|
+
mo_dir = mo_dir_rule % {:lang => lang}
|
189
|
+
File.join(mo_dir, "#{textdomain}.mo")
|
190
|
+
end
|
191
|
+
end
|
192
|
+
|
193
|
+
if __FILE__ == $0
|
194
|
+
GetText.update_pofiles("foo", ARGV, "foo 1.1.0")
|
195
|
+
end
|
@@ -0,0 +1,93 @@
|
|
1
|
+
module GetText
|
2
|
+
class ParseError < StandardError
|
3
|
+
end
|
4
|
+
|
5
|
+
# Contains data related to the expression or sentence that
|
6
|
+
# is to be translated (translation target).
|
7
|
+
# Implements a sort of state machine to assist the parser.
|
8
|
+
class TranslationTarget
|
9
|
+
attr_accessor :type, :msgid, :occurrences # obligatory attributes
|
10
|
+
attr_accessor :plural, :msgctxt, :extracted_comment # optional attributes
|
11
|
+
|
12
|
+
def initialize(new_type)
|
13
|
+
@type = new_type
|
14
|
+
@occurrences = Array.new
|
15
|
+
@param_number = 0
|
16
|
+
end
|
17
|
+
|
18
|
+
# Supports parsing by setting attributes by and by.
|
19
|
+
def set_current_attribute(str)
|
20
|
+
case @param_number
|
21
|
+
when 0
|
22
|
+
set_string_value :msgid, str
|
23
|
+
when 1
|
24
|
+
case type
|
25
|
+
when :plural
|
26
|
+
set_string_value :plural, str
|
27
|
+
when :msgctxt, :msgctxt_plural
|
28
|
+
set_string_value :msgctxt, str
|
29
|
+
else
|
30
|
+
raise ParseError, 'no more string parameters expected'
|
31
|
+
end
|
32
|
+
when 2
|
33
|
+
if :msgctxt_plural
|
34
|
+
set_string_value plural, str
|
35
|
+
else
|
36
|
+
raise ParseError, 'no more string parameters expected'
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def advance_to_next_attribute
|
42
|
+
@param_number += 1
|
43
|
+
end
|
44
|
+
|
45
|
+
# Support for extracted comments. Explanation s.
|
46
|
+
# http://www.gnu.org/software/gettext/manual/gettext.html#Names
|
47
|
+
def add_extracted_comment(new_comment)
|
48
|
+
@extracted_comment = @extracted_comment.to_s + new_comment
|
49
|
+
to_s
|
50
|
+
end
|
51
|
+
|
52
|
+
# Returns a parameter representation suitable for po-files
|
53
|
+
# and other purposes.
|
54
|
+
def escaped(param_name)
|
55
|
+
orig = self.send param_name
|
56
|
+
orig.gsub(/"/, '\"').gsub(/\r/, '')
|
57
|
+
end
|
58
|
+
|
59
|
+
# Checks if the other translation target is mergeable with
|
60
|
+
# the current one. Relevant are msgid and translation context (msgctxt).
|
61
|
+
def matches?(other)
|
62
|
+
other.msgid == self.msgid && other.msgctxt == self.msgctxt
|
63
|
+
end
|
64
|
+
|
65
|
+
# Merges two translation targets with the same msgid and returns the merged
|
66
|
+
# result. If one is declared as plural and the other not, then the one
|
67
|
+
# with the plural wins.
|
68
|
+
def merge(other)
|
69
|
+
return self if other.nil?
|
70
|
+
raise ParseError, "Translation targets do not match: \n" \
|
71
|
+
" self: #{self.inspect}\n other: '#{other.inspect}'" unless matches?(other)
|
72
|
+
if other.plural && !self.plural
|
73
|
+
res = other
|
74
|
+
res.occurrences.concat self.occurrences
|
75
|
+
else
|
76
|
+
res = self
|
77
|
+
res.occurrences.concat other.occurrences
|
78
|
+
end
|
79
|
+
res
|
80
|
+
end
|
81
|
+
|
82
|
+
private
|
83
|
+
|
84
|
+
# sets or extends the value of a translation target params like msgid,
|
85
|
+
# msgctxt etc.
|
86
|
+
# param is symbol with the name of param
|
87
|
+
# value - new value
|
88
|
+
def set_string_value(param, value)
|
89
|
+
send "#{param}=", (send(param) || '') + value.gsub(/\n/, '\n')
|
90
|
+
end
|
91
|
+
end
|
92
|
+
|
93
|
+
end
|
@@ -0,0 +1,37 @@
|
|
1
|
+
=begin
|
2
|
+
utils.rb - Utility functions
|
3
|
+
|
4
|
+
Copyright (C) 2005,2006 Masao Mutoh
|
5
|
+
|
6
|
+
You may redistribute it and/or modify it under the same
|
7
|
+
license terms as Ruby.
|
8
|
+
=end
|
9
|
+
|
10
|
+
require 'gettext/tools'
|
11
|
+
|
12
|
+
warn "'gettext/utils.rb' is deprecated. Use gettext/tools.rb."
|
13
|
+
|
14
|
+
module GetText
|
15
|
+
|
16
|
+
alias :create_mofiles_org :create_mofiles #:nodoc:
|
17
|
+
alias :update_pofiles_org :update_pofiles #:nodoc:
|
18
|
+
|
19
|
+
|
20
|
+
# Deprecated. Use gettext/tools instead.
|
21
|
+
def create_mofiles(verbose = false,
|
22
|
+
podir = "./po", targetdir = "./data/locale",
|
23
|
+
targetpath_rule = "%s/LC_MESSAGES") # :nodoc:
|
24
|
+
warn "'gettext/utils.rb' is deprecated. Use gettext/tools.rb."
|
25
|
+
create_mofiles_org(:verbose => verbose,
|
26
|
+
:po_root => podir,
|
27
|
+
:mo_root => targetdir,
|
28
|
+
:mo_root_rule => targetpath_rule)
|
29
|
+
end
|
30
|
+
|
31
|
+
# Deprecated. Use gettext/tools instead.
|
32
|
+
def update_pofiles(textdomain, files, app_version, po_root = "po", refpot = "tmp.pot") # :nodoc:
|
33
|
+
warn "'gettext/utils.rb' is deprecated. Use gettext/tools.rb."
|
34
|
+
options = {:po_root => po_root}
|
35
|
+
update_pofiles_org(textdomain, files, app_version, options)
|
36
|
+
end
|
37
|
+
end
|
@@ -0,0 +1,12 @@
|
|
1
|
+
#! /usr/bin/ruby
|
2
|
+
=begin
|
3
|
+
version - version information of Ruby-GetText-Package
|
4
|
+
|
5
|
+
Copyright (C) 2005-2009 Masao Mutoh
|
6
|
+
|
7
|
+
You may redistribute it and/or modify it under the same
|
8
|
+
license terms as Ruby.
|
9
|
+
=end
|
10
|
+
module GetText
|
11
|
+
VERSION = "2.0.4"
|
12
|
+
end
|
data/lib/gettext.rb
ADDED
@@ -0,0 +1,312 @@
|
|
1
|
+
=begin
|
2
|
+
gettext.rb - GetText module
|
3
|
+
|
4
|
+
Copyright (C) 2001-2009 Masao Mutoh
|
5
|
+
Copyright (C) 2001-2003 Masahiro Sakai
|
6
|
+
|
7
|
+
Masao Mutoh <mutomasa at gmail.com>
|
8
|
+
Masahiro Sakai <s01397ms@sfc.keio.ac.jp>
|
9
|
+
|
10
|
+
You may redistribute it and/or modify it under the same
|
11
|
+
license terms as Ruby.
|
12
|
+
|
13
|
+
$Id: gettext.rb,v 1.46 2008/09/13 18:23:55 mutoh Exp $
|
14
|
+
=end
|
15
|
+
|
16
|
+
if Object.constants.include? "Gem"
|
17
|
+
begin
|
18
|
+
begin
|
19
|
+
gem 'locale', '>=2.0.4'
|
20
|
+
rescue Gem::LoadError
|
21
|
+
end
|
22
|
+
rescue NoMethodError
|
23
|
+
else LoadError
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
require 'locale'
|
28
|
+
raise "Install locale as gem or uninstall old gettext" unless Locale.respond_to? :candidates
|
29
|
+
|
30
|
+
require 'gettext/version'
|
31
|
+
require 'gettext/textdomain_manager'
|
32
|
+
|
33
|
+
module GetText
|
34
|
+
# If the textdomain isn't bound when calling GetText.textdomain, this error is raised.
|
35
|
+
class NoboundTextDomainError < RuntimeError
|
36
|
+
def initialize(domainname)
|
37
|
+
@domainname = domainname
|
38
|
+
end
|
39
|
+
def message
|
40
|
+
"#{@domainname} is not bound."
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
extend self
|
45
|
+
|
46
|
+
def self.included(mod) #:nodoc:
|
47
|
+
mod.extend self
|
48
|
+
end
|
49
|
+
|
50
|
+
# bindtextdomain(domainname, options = {})
|
51
|
+
#
|
52
|
+
# Bind a textdomain(%{path}/%{locale}/LC_MESSAGES/%{domainname}.mo) to
|
53
|
+
# your program.
|
54
|
+
# Normally, the texdomain scope becomes the class/module(and parent
|
55
|
+
# classes/included modules).
|
56
|
+
#
|
57
|
+
# * domainname: the textdomain name.
|
58
|
+
# * options: options as an Hash.
|
59
|
+
# * :path - the path to the mo-files. When the value is nil, it will search default paths such as
|
60
|
+
# /usr/share/locale, /usr/local/share/locale)
|
61
|
+
# * :output_charset - The output charset. Same with GetText.set_output_charset. Usually, L10n
|
62
|
+
# library doesn't use this option. Application may use this once.
|
63
|
+
# * Returns: the GetText::TextDomainManager.
|
64
|
+
#
|
65
|
+
def bindtextdomain(domainname, *options)
|
66
|
+
bindtextdomain_to(self, domainname, *options)
|
67
|
+
end
|
68
|
+
|
69
|
+
# Includes GetText module and bind a textdomain to a class.
|
70
|
+
# * klass: the target ruby class.
|
71
|
+
# * domainname: the textdomain name.
|
72
|
+
# * options: options as an Hash. See GetText.bindtextdomain.
|
73
|
+
def bindtextdomain_to(klass, domainname, *options)
|
74
|
+
if options[0].kind_of? Hash
|
75
|
+
opts = options[0]
|
76
|
+
else
|
77
|
+
# for backward compatibility.
|
78
|
+
opts = {}
|
79
|
+
opts[:path] = options[0] if options[0]
|
80
|
+
opts[:output_charset] = options[2] if options[2]
|
81
|
+
end
|
82
|
+
unless (klass.kind_of? GetText or klass.include? GetText)
|
83
|
+
klass.__send__(:include, GetText)
|
84
|
+
end
|
85
|
+
TextDomainManager.bind_to(klass, domainname, opts)
|
86
|
+
end
|
87
|
+
|
88
|
+
# Binds a existed textdomain to your program.
|
89
|
+
# This is the same function with GetText.bindtextdomain but simpler(and faster) than bindtextdomain.
|
90
|
+
# Note that you need to call GetText.bindtextdomain first. If the domainname hasn't bound yet,
|
91
|
+
# raises GetText::NoboundTextDomainError.
|
92
|
+
# * domainname: a textdomain name.
|
93
|
+
# * Returns: the GetText::TextDomainManager.
|
94
|
+
def textdomain(domainname) #:nodoc:
|
95
|
+
textdomain_to(self, domainname)
|
96
|
+
end
|
97
|
+
|
98
|
+
# Includes GetText module and bind an exsited textdomain to a class.
|
99
|
+
# See textdomain for more detail.
|
100
|
+
# * klass: the target ruby class.
|
101
|
+
# * domainname: the textdomain name.
|
102
|
+
|
103
|
+
def textdomain_to(klass, domainname) #:nodoc:
|
104
|
+
domain = TextDomainManager.textdomain_pool(domainname)
|
105
|
+
raise NoboundTextDomainError.new(domainname) unless domain
|
106
|
+
bindtextdomain_to(klass, domainname)
|
107
|
+
end
|
108
|
+
|
109
|
+
# call-seq:
|
110
|
+
# gettext(msgid)
|
111
|
+
# _(msgid)
|
112
|
+
#
|
113
|
+
# Translates msgid and return the message.
|
114
|
+
# This doesn't make a copy of the message.
|
115
|
+
#
|
116
|
+
# You need to use String#dup if you want to modify the return value
|
117
|
+
# with destructive functions.
|
118
|
+
#
|
119
|
+
# (e.g.1) _("Hello ").dup << "world"
|
120
|
+
#
|
121
|
+
# But e.g.1 should be rewrite to:
|
122
|
+
#
|
123
|
+
# (e.g.2) _("Hello %{val}") % {:val => "world"}
|
124
|
+
#
|
125
|
+
# Because the translator may want to change the position of "world".
|
126
|
+
#
|
127
|
+
# * msgid: the message id.
|
128
|
+
# * Returns: localized text by msgid. If there are not binded mo-file, it will return msgid.
|
129
|
+
def gettext(msgid)
|
130
|
+
TextDomainManager.translate_singluar_message(self, msgid)
|
131
|
+
end
|
132
|
+
|
133
|
+
# call-seq:
|
134
|
+
# sgettext(msgid, div = '|')
|
135
|
+
# s_(msgid, div = '|')
|
136
|
+
#
|
137
|
+
# Translates msgid, but if there are no localized text,
|
138
|
+
# it returns a last part of msgid separeted "div".
|
139
|
+
#
|
140
|
+
# * msgid: the message id.
|
141
|
+
# * separator: separator or nil for no seperation.
|
142
|
+
# * Returns: the localized text by msgid. If there are no localized text,
|
143
|
+
# it returns a last part of the msgid separeted by "seperator".
|
144
|
+
# <tt>Movie|Location -> Location</tt>
|
145
|
+
# See: http://www.gnu.org/software/gettext/manual/html_mono/gettext.html#SEC151
|
146
|
+
def sgettext(msgid, seperator = "|")
|
147
|
+
TextDomainManager.translate_singluar_message(self, msgid, seperator)
|
148
|
+
end
|
149
|
+
|
150
|
+
# call-seq:
|
151
|
+
# pgettext(msgctxt, msgid)
|
152
|
+
# p_(msgctxt, msgid)
|
153
|
+
#
|
154
|
+
# Translates msgid with msgctxt. This methods is similer with s_().
|
155
|
+
# e.g.) p_("File", "New") == s_("File|New")
|
156
|
+
# p_("File", "Open") == s_("File|Open")
|
157
|
+
#
|
158
|
+
# * msgctxt: the message context.
|
159
|
+
# * msgid: the message id.
|
160
|
+
# * Returns: the localized text by msgid. If there are no localized text,
|
161
|
+
# it returns msgid.
|
162
|
+
# See: http://www.gnu.org/software/autoconf/manual/gettext/Contexts.html
|
163
|
+
def pgettext(msgctxt, msgid)
|
164
|
+
TextDomainManager.translate_singluar_message(self, "#{msgctxt}\004#{msgid}", "\004")
|
165
|
+
end
|
166
|
+
|
167
|
+
# call-seq:
|
168
|
+
# ngettext(msgid, msgid_plural, n)
|
169
|
+
# ngettext(msgids, n) # msgids = [msgid, msgid_plural]
|
170
|
+
# n_(msgid, msgid_plural, n)
|
171
|
+
# n_(msgids, n) # msgids = [msgid, msgid_plural]
|
172
|
+
#
|
173
|
+
# The ngettext is similar to the gettext function as it finds the message catalogs in the same way.
|
174
|
+
# But it takes two extra arguments for plural form.
|
175
|
+
#
|
176
|
+
# * msgid: the singular form.
|
177
|
+
# * msgid_plural: the plural form.
|
178
|
+
# * n: a number used to determine the plural form.
|
179
|
+
# * Returns: the localized text which key is msgid_plural if n is plural(follow plural-rule) or msgid.
|
180
|
+
# "plural-rule" is defined in po-file.
|
181
|
+
def ngettext(msgid, msgid_plural, n = nil)
|
182
|
+
TextDomainManager.translate_plural_message(self, msgid, msgid_plural, n)
|
183
|
+
end
|
184
|
+
|
185
|
+
# call-seq:
|
186
|
+
# nsgettext(msgid, msgid_plural, n, div = "|")
|
187
|
+
# nsgettext(msgids, n, div = "|") # msgids = [msgid, msgid_plural]
|
188
|
+
# ns_(msgid, msgid_plural, n, div = "|")
|
189
|
+
# ns_(msgids, n, div = "|") # msgids = [msgid, msgid_plural]
|
190
|
+
#
|
191
|
+
# The nsgettext is similar to the ngettext.
|
192
|
+
# But if there are no localized text,
|
193
|
+
# it returns a last part of msgid separeted "div".
|
194
|
+
#
|
195
|
+
# * msgid: the singular form with "div". (e.g. "Special|An apple")
|
196
|
+
# * msgid_plural: the plural form. (e.g. "%{num} Apples")
|
197
|
+
# * n: a number used to determine the plural form.
|
198
|
+
# * Returns: the localized text which key is msgid_plural if n is plural(follow plural-rule) or msgid.
|
199
|
+
# "plural-rule" is defined in po-file.
|
200
|
+
def nsgettext(msgid, msgid_plural, n="|", seperator = "|")
|
201
|
+
TextDomainManager.translate_plural_message(self, msgid, msgid_plural, n, seperator)
|
202
|
+
end
|
203
|
+
|
204
|
+
# call-seq:
|
205
|
+
# npgettext(msgctxt, msgid, msgid_plural, n)
|
206
|
+
# npgettext(msgctxt, msgids, n) # msgids = [msgid, msgid_plural]
|
207
|
+
# np_(msgctxt, msgid, msgid_plural, n)
|
208
|
+
# np_(msgctxt, msgids, n) # msgids = [msgid, msgid_plural]
|
209
|
+
#
|
210
|
+
# The npgettext is similar to the nsgettext function.
|
211
|
+
# e.g.) np_("Special", "An apple", "%{num} Apples", num) == ns_("Special|An apple", "%{num} Apples", num)
|
212
|
+
# * msgctxt: the message context.
|
213
|
+
# * msgid: the singular form.
|
214
|
+
# * msgid_plural: the plural form.
|
215
|
+
# * n: a number used to determine the plural form.
|
216
|
+
# * Returns: the localized text which key is msgid_plural if n is plural(follow plural-rule) or msgid.
|
217
|
+
# "plural-rule" is defined in po-file.
|
218
|
+
def npgettext(msgctxt, msgids, arg2 = nil, arg3 = nil)
|
219
|
+
if msgids.kind_of?(Array)
|
220
|
+
msgid = msgids[0]
|
221
|
+
msgid_ctxt = "#{msgctxt}\004#{msgid}"
|
222
|
+
msgid_plural = msgids[1]
|
223
|
+
opt1 = arg2
|
224
|
+
opt2 = arg3
|
225
|
+
else
|
226
|
+
msgid = msgids
|
227
|
+
msgid_ctxt = "#{msgctxt}\004#{msgid}"
|
228
|
+
msgid_plural = arg2
|
229
|
+
opt1 = arg3
|
230
|
+
opt2 = nil
|
231
|
+
end
|
232
|
+
|
233
|
+
msgstr = TextDomainManager.translate_plural_message(self, msgid_ctxt, msgid_plural, opt1, opt2)
|
234
|
+
if msgstr == msgid_ctxt
|
235
|
+
msgid
|
236
|
+
else
|
237
|
+
msgstr
|
238
|
+
end
|
239
|
+
end
|
240
|
+
|
241
|
+
# makes dynamic translation messages readable for the gettext parser.
|
242
|
+
# <tt>_(fruit)</tt> cannot be understood by the gettext parser. To help the parser find all your translations,
|
243
|
+
# you can add <tt>fruit = N_("Apple")</tt> which does not translate, but tells the parser: "Apple" needs translation.
|
244
|
+
# * msgid: the message id.
|
245
|
+
# * Returns: msgid.
|
246
|
+
def N_(msgid)
|
247
|
+
msgid
|
248
|
+
end
|
249
|
+
|
250
|
+
# This is same function as N_ but for ngettext.
|
251
|
+
# * msgid: the message id.
|
252
|
+
# * msgid_plural: the plural message id.
|
253
|
+
# * Returns: msgid.
|
254
|
+
def Nn_(msgid, msgid_plural)
|
255
|
+
[msgid, msgid_plural]
|
256
|
+
end
|
257
|
+
|
258
|
+
# Sets charset(String) such as "euc-jp", "sjis", "CP932", "utf-8", ...
|
259
|
+
# You shouldn't use this in your own Libraries.
|
260
|
+
# * charset: an output_charset
|
261
|
+
# * Returns: self
|
262
|
+
def set_output_charset(charset)
|
263
|
+
TextDomainManager.output_charset = charset
|
264
|
+
self
|
265
|
+
end
|
266
|
+
|
267
|
+
# Gets the current output_charset which is set using GetText.set_output_charset.
|
268
|
+
# * Returns: output_charset.
|
269
|
+
def output_charset
|
270
|
+
TextDomainManager.output_charset
|
271
|
+
end
|
272
|
+
|
273
|
+
# Set the locale. This value forces the locale whole the programs.
|
274
|
+
# This method calls Locale.set_app_language_tags, Locale.default, Locale.current.
|
275
|
+
# Use Locale methods if you need to handle locales more flexible.
|
276
|
+
def set_locale(lang)
|
277
|
+
Locale.set_app_language_tags(lang)
|
278
|
+
Locale.default = lang
|
279
|
+
Locale.current = lang
|
280
|
+
end
|
281
|
+
|
282
|
+
# Set the locale to the current thread.
|
283
|
+
# Note that if #set_locale is set, this value is ignored.
|
284
|
+
# If you need, set_locale(nil); set_current_locale(lang)
|
285
|
+
def set_current_locale(lang)
|
286
|
+
Locale.current = lang
|
287
|
+
end
|
288
|
+
|
289
|
+
def locale
|
290
|
+
Locale.current[0]
|
291
|
+
end
|
292
|
+
|
293
|
+
alias :locale= :set_locale #:nodoc:
|
294
|
+
alias :current_locale= :set_current_locale #:nodoc:
|
295
|
+
alias :_ :gettext #:nodoc:
|
296
|
+
alias :n_ :ngettext #:nodoc:
|
297
|
+
alias :s_ :sgettext #:nodoc:
|
298
|
+
alias :ns_ :nsgettext #:nodoc:
|
299
|
+
alias :np_ :npgettext #:nodoc:
|
300
|
+
|
301
|
+
alias :output_charset= :set_output_charset #:nodoc:
|
302
|
+
|
303
|
+
unless defined? XX
|
304
|
+
# This is the workaround to conflict p_ methods with the xx("double x") library.
|
305
|
+
# http://rubyforge.org/projects/codeforpeople/
|
306
|
+
alias :p_ :pgettext #:nodoc:
|
307
|
+
end
|
308
|
+
|
309
|
+
# for backward compatibility
|
310
|
+
alias :set_locale_all :set_locale #:nodoc:
|
311
|
+
alias :setlocale :set_locale #:nodoc:
|
312
|
+
end
|