-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathREADME
301 lines (242 loc) · 12.2 KB
/
README
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
NAME
Locale::Wolowitz - Dead simple localization with JSON.
SYNOPSIS
# in ./i18n/locales.coll.json
{
"Welcome!": {
"he": "ברוכים הבאים!",
"es": "Bienvenido!"
},
"I'm using %1": {
"he": "אני משתמש ב%1",
"es": "Estoy usando %1"
},
"Linux": {
"he": "לינוקס"
}
}
# in your app
use Locale::Wolowitz;
my $w = Locale::Wolowitz->new( './i18n' );
print $w->loc('Welcome!', 'es'); # prints 'Bienvenido!'
print $w->loc("I'm using %1", 'he', $w->loc('Linux', 'he')); # prints "אני משתמש בלינוקס"
# you can also directly load data (useful if data is not in files, but say in database)
$w->load_structure({
hello => {
he => 'שלום',
fr => 'bonjour'
}
});
print $w->loc('hello', 'he'); # prints "שלום"
DESCRIPTION
Locale::Wolowitz is a very simple text localization system. Yes, another
localization system.
Frankly, I never realized how to use the standard Perl localization
systems such as Locale::Maketext, Gettext, Data::Localize or whatever.
It seems they are more meant to localize an application to the language
of the system on which its running, which isn't really what I need. Most
of the time, seeing as how I'm mostly writing web applications, I wish
to localize my applications/websites according to the user's wishes, not
by the system. For example, I may create a content management system
where the user can select the interface's language. Also, I grew to hate
the standard .po files, and thought using a JSON format might be more
comfortable.
Locale::Wolowitz allows you to provide different languages to end-users
of your applications. To some extent, when writing RESTful web
applications, this means you can perform language negotiation with
visitors (see Content negotiation on Wikipedia
<https://secure.wikimedia.org/wikipedia/en/w/index.php?title=Content_neg
otiation&oldid=367120431>).
Locale::Wolowitz works with JSON files. Each file can serve one or more
languages. When creating an instance of this module, you are required to
pass a path to a directory where your application's JSON localization
files are present. These are all loaded and merged into one big hash-ref
(unless you tell the module to only load a specific file), which is
stored in memory. A file with only one language has to be named
<lang>.json (where <lang> is the name of the language, you'd probably
want to use the two-letter ISO 639-1 code). A file with multiple
languages must end with .coll.json (this requirement will probably be
lifted in the future).
The basic idea is to write your application in a base language, and use
the JSON files to translate text to other languages. For example, lets
say you're writing your application in English and translating it to
Hebrew, Spanish, and Dutch. You put Spanish and Dutch translations in
one file, and since everybody hates Israel, you put Hebrew translations
alone. The Spanish and Dutch file can look like this:
# es_and_nl.coll.json
{
"Welcome!": {
"es": "Bienvenido!",
"nl": "Welkom!"
},
"I'm using %1": {
"es": "Estoy usando %1",
"nl": "Ik gebruik %1"
},
"Linux": {} // this line can also be missing entirely
}
While the Hebrew file can look like this:
# he.json
{
"Welcome!": "ברוכים הבאים!",
"I'm using %1": "אני משתמש ב%1",
"Linux": "לינוקס"
}
When loading these files, Locale::Wolowitz internally merges the two
files into one structure:
{
"Welcome!" => {
"es" => "Bienvenido!",
"nl" => "Welkom!",
"he" => "ברוכים הבאים!",
},
"I'm using %1" => {
"es" => "Estoy usando %1",
"nl" => "Ik gebruik %1",
"he" => "אני משתמש ב%1",
},
"Linux" => {
"he" => "לינוקס",
}
}
Notice the "%1" substrings above. This is a placeholder, just like in
other localization paradigms - they are replaced with content you
provide, usually dynamic content. In Locale::Wolowitz, placeholders are
written with a percent sign, followed by an integer, starting from 1
(e.g. %1, %2, %3). When passing data for the placeholders, make sure
you're passing scalars, or printable objects, otherwise you'll encounter
errors.
We can also see here that Spanish and Dutch have no translation for
"Linux". Since Linux is written "Linux" in these languages, they have no
translation. When attempting to translate a string that has no
translation to the requested language, or has no reference in the JSON
files at all, the string is simply returned as is (but placeholders will
still be replaced as expected).
Say you write your application in English (and thus 'en' is your base
language). Since Locale::Wolowitz doesn't really know what your base
language is, you can translate texts within the same language. This is
more useful when you want to give some of your strings an identifier.
For example:
"copyrights": {
"en": "Copyrights, 2010 Ido Perlmuter",
"he": "כל הזכויות שמורות, 2010 עידו פרלמוטר"
}
CONSTRUCTOR
new( [ $path / $filename, \%options ] )
Creates a new instance of this module. A path to a directory in which
JSON localization files exist, or a path to a specific localization
file, *may* be supplied. If you pass a directory, all JSON localization
files in it will be loaded and merged as described above. If you pass
one file, only that file will be loaded.
Note that "Locale::Wolowitz" will ignore dotfiles in the provided path
(e.g. hidden files, backups files, etc.).
A hash-ref of options can also be provided. The only option currently
supported is "utf8", which is on by default. If on, all JSON files are
assumed to be in UTF-8 character set and will be automatically decoded.
Provide a false value if your files are not UTF-8 encoded, for example:
Locale::Wolowitz->new( '/path/to/files', { utf8 => 0 } );
OBJECT METHODS
load_path( $path / $filename )
Receives a path to a directory in which JSON localization files exist,
or a path to a specific localization file, and loads (and merges) the
localization data from the file(s). If localization data was already
loaded previously, the structure will be merged, with the new data
taking precedence.
You can call this method and load_structure() as much as you want, the
data from each call will be merged with existing data.
load_structure( \%structure, [ $lang ] )
Receives a hash-ref of localization data similar to that in the JSON
files and loads it into the object (possibly merging with existing data,
if any). If $lang is supplied, a one-to-one structure will be assumed,
like so:
load_structure(
{ "hello" => "שלום", "world" => "עולם" },
'he'
)
Or, if $lang is not provided, the structure must be the multiple
language structure, like so:
load_structure({
"hello" => {
"he" => "שלום",
"fr" => "bonjour"
},
"world" => {
"he" => "עולם",
"fr" => "monde",
"it" => "mondo"
}
})
You can call this method and load_path() as much as you want, the data
from each call will be merged with existing data.
loc( $msg, $lang, [ @args ] )
Returns the string $msg, translated to the requested language (if such a
translation exists, otherwise no traslation occurs). Any other
parameters passed to the method (@args) are injected to the placeholders
in the string (if present).
If an argument is an array ref, it'll be replaced with a recursive call
to "loc" with its elements, with the $lang argument automatically added.
In other words, the following two statements are equivalent:
print $w->loc("I'm using %1", 'he', $w->loc('Linux', 'he'));
# same result as
print $w->loc("I'm using %1", 'he', [ 'Linux' ]);
loc_for( $lang )
Returns a function ref that is like "loc", but with the $lang curried
away.
use Locale::Wolowitz;
my $w = Locale::Wolowitz->new( './i18n' );
my $french_loc = $w->loc_for('fr');
my $german_loc = $w->loc_for('de');
print $french_loc->('Welcome!'); # equivalent to $w->loc( 'Welcome!', 'fr' )
DIAGNOSTICS
The following exceptions are thrown by this module:
"You must provide a path to localization directory."
This exception is thrown if you haven't provided the "new()"
subroutine a path to a localization file, or a directory of
localization files. Read the documentation for the "new()"
subroutine above.
"Can't open localization directory: %s" and "Can't close localization
directory: %s"
This exception is thrown if Locale::Wolowitz failed to open/close
the directory of the localization files. This will probably happen
due to permission problems. The error message should include the
actual reason for the failure.
"Path must be to a directory or a JSON file."
This exception is thrown if you passed a wrong value to the "new()"
subroutine as the path to the localization directory/file. Either
the path is wrong and thus does not exist, or the path does exist,
but is not a directory and not a file.
"Can't open localization file %s: %s" and "Can't close localization file
%s: %s"
This exception is thrown if Locale::Wolowitz fails to open/close a
specific localization file. This will usually happen because of
permission problems. The error message will include both the name of
the file, and the actual reason for the failure.
CONFIGURATION AND ENVIRONMENT
"Locale::Wolowitz" requires no configuration files or environment
variables.
DEPENDENCIES
"Locale::Wolowitz" depends on the following CPAN modules:
* Carp
* JSON::MaybeXS
"Locale::Wolowitz" recommends Cpanel::JSON::XS or JSON::XS for faster
parsing of JSON files.
INCOMPATIBILITIES WITH OTHER MODULES
None reported.
BUGS AND LIMITATIONS
No bugs have been reported.
Please report any bugs or feature requests to
"[email protected]", or through the web interface at
<http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Locale-Wolowitz>.
AUTHOR
Ido Perlmuter <[email protected]>
LICENSE AND COPYRIGHT
Copyright 2017 Ido Perlmuter
Licensed under the Apache License, Version 2.0 (the "License"); you may
not use this file except in compliance with the License. You may obtain
a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.