5
5
from dataclasses import dataclass
6
6
from gettext import GNUTranslations , NullTranslations
7
7
from importlib .resources .abc import Traversable
8
- from typing import Literal , NewType , Protocol , TypeAlias , cast , overload
8
+ from typing import Literal , NewType , Protocol , TypeAlias , overload
9
9
10
10
from questionpy_common .environment import (
11
11
Environment ,
@@ -207,18 +207,48 @@ def __call__(self, message: str, *, defer: bool | None = None) -> str | UserStri
207
207
"""Translate the given message.
208
208
209
209
Args:
210
- message: Gettext `msgid`.
210
+ message: Gettext `msgid`. This should also be the message in its primary language, usually english.
211
211
defer: By default, translation is deferred only when no request is being processed at the time of the
212
212
`gettext` call. This parameter can be explicitly set to `True` to force deferral or to `False` to never
213
213
defer translations. In the latter case, calling `gettext` before a request is processed
214
214
(e.g. during init) will raise an error.
215
215
"""
216
216
217
217
218
- _NGettext : TypeAlias = Callable [[str ], str ]
218
+ class _NGettext (Protocol ):
219
+ @overload
220
+ def __call__ (self , msgid1 : str , msgid2 : str , n : int , * , defer : None = None ) -> str | UserString : ...
221
+
222
+ @overload
223
+ def __call__ (self , msgid1 : str , msgid2 : str , n : int , * , defer : Literal [True ]) -> UserString : ...
224
+
225
+ @overload
226
+ def __call__ (self , msgid1 : str , msgid2 : str , n : int , * , defer : Literal [False ]) -> str : ...
227
+
228
+ def __call__ (self , msgid1 : str , msgid2 : str , n : int , * , defer : bool | None = None ) -> str | UserString :
229
+ """Translate the given message, accounting for plural forms.
230
+
231
+ Args:
232
+ msgid1: Message id in its (english) singular form, used if no plural form exists and `n == 1`.
233
+ msgid2: Message id in its (english) plural form, used if no plural form exists and `n >= 2`.
234
+ n: This number is passed through the plural formula of the active catalog to determine which form to use.
235
+ defer: By default, translation is deferred only when no request is being processed at the time of the
236
+ `gettext` call. This parameter can be explicitly set to `True` to force deferral or to `False` to never
237
+ defer translations. In the latter case, calling `gettext` before a request is processed
238
+ (e.g. during init) will raise an error.
239
+ """
240
+
241
+
242
+ _Noop : TypeAlias = Callable [[str ], str ]
243
+ class GettextFunctions (tuple [_Gettext , _Noop ], _Gettext ):
244
+ def __new__ (cls , * args , ** kwargs ):
245
+
246
+ def __init__ (self ):
247
+ super ().__init__ ()
248
+
219
249
220
250
221
- def get_for (module_name : str ) -> tuple [_Gettext , _NGettext ]:
251
+ def get_for (module_name : str ) -> tuple [_Gettext , _Noop ]:
222
252
"""Initializes i18n for the package owning the given Python module and returns the gettext-family functions.
223
253
224
254
Args:
@@ -240,10 +270,10 @@ def gettext(message: str, *, defer: bool | None = None) -> str | UserString:
240
270
request_state = _require_request_state (domain , domain_state )
241
271
return request_state .translations .gettext (message )
242
272
243
- def ngettext (message : str ) -> str :
273
+ def gettext_N (message : str ) -> str :
244
274
return message
245
275
246
- return cast (_Gettext , gettext ), ngettext
276
+ return cast (_Gettext , gettext ), gettext_N
247
277
248
278
249
279
__all__ = ["DEFAULT_CATEGORY" , "GettextDomain" , "domain_of" , "get_for" , "get_translations_of_package" ]
0 commit comments