-
Notifications
You must be signed in to change notification settings - Fork 628
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for translations #33
Comments
Idea: Don't handle text at all in SixtyFPS, and only use symbols in It gives the user more choice, but more importantly (imo) means that i18n can be handled by a library specifically designed for it. |
In my experience, the process of translation usually involves some "local" tooling where the the strings to be translated are placed in some third-party format (I've seen excel spreadsheets!), sent to people typically external to the project and the result is imported with the same "local" tooling. The library part of run-time injection of these translations is fairly straight-forward. In some scenarios it makes sense to just select one language at compile time but more commonly I think it makes sense to have "language packs" in the form of external files that can be mmap()'ed. Do you have specific libraries in mind for translation? I think the symbol approach is indeed something some users are preferring for UI translation. I personally prefer the literals in the source, but I think both are valid approaches. |
For string in rust, we could use https://github.com/woboq/tr For string in .60, i'd like to also use a gettext based approach, with a How would that look like? For simple string that's easy: Idea: Text {
// %0 %1, ....
text: @tr("Hello %0, my name is %1", root.your_name, root.my_name);
// same substitution language as rust: (this include using {0} and {1}
text: @tr("Hello {}, my name is {}", root.your_name, root.my_name);
// Should we or should we not allow expressions like in normal strings?
// maybe limit to simple property access?
text: @tr("Hello \{root.your_name}, my name is \{root.my_name}");
} In fact, i'm tempted to use the same syntax as the Should we also offer a We need an code extractor. we could modity |
Hello, I couldn't help too much on the technical side, but I see a few points about localization: On the code side, let's not forget support about developer notes and contexts, so useful for translators ;-) We all hate translating blindly. On the translator side, which localization file format to use ? @tronical I found OrbTk using Ron with an exotic translation file . I wouldn't use this for reasons cited earlier. I think that if too much liberties are given to the translation workflow, the library dedicated to translation would become difficult to maintain. Keep it simple and stupid.
The first would need a system like Qt's resource files to include statically images, translation files, and others in the executable. I know Qt is compiling them in a precompilation stage, so I'm not certain of how to declare/compile them in Rust. Dedicated library pointing to a resource file on which are listed resources ? Finally, Using this library to access these resources ? Lot of rambling, sorry... It's probably already implemented or on the way. Having the C++/Qt mindset, I'm still struggling (a lot) with Rust :-) |
I’d suggest using the Qt L10N file format »TS«, which means we pretty much get a decent and OSS GUI L10N tool for free. Also +1 for optional asset bundling and having a proper resource management system. Having a resource management system also allows for hot reloading of assets. This may even be attractive for L10N work if we used plain-text L10N file formats like Fluent’s |
Live reload/preview is a neat idea, in the spirit of the live preview when editing the markup files :) |
In that case i’d say all in on Project Fluent and |
Hello, I think dynamic load/unload/reload of assets is interesting for i10n files. It's a really different can of worms when I compare with static bundling of assets. Yet, both ways can coexist. There are already libs for static assets. Maybe we can implement step by step:
This way, there is something quick and not so dirty to begin quickly with the translation implementation. Then, an evolution with the dynamic loading of assets. Dynamic refresh can be a feature of dynamic loading, not specifically of the translation system. If not existing, I suggest creating an issue specifically for static assets and dynamic assets. Cheers |
Currently, if one would want to do translation, it is possible with a global object like so:
Then the other elements can use the Example of project using that: (although it is using an old version that did not have global object accessible from rust at the time) |
Hi there. I'm the maintainer and one of the authors of the Fluent system, as well as a co-creator of ICU4X, and contributor to Unicode Message Format 2. It's a bit concerning for me how far along Slint is without any notion of I18n/L10n in place. Such approach often leads to trying to plaster i18n/l10n on top of a non-i18n system leading to suboptimal architectural choices. I've spent a lot of time building I18n UI architectures (including multi-modal for VUI/GUI combos) and I encourage you to look at Fluent or Message Format 2 as a basic l10n system and bind it deeply into your Widget model. You can read more about my approach in raphlinus/crochet#7 and unicode-org/message-format-wg#118 . In any case, I advise against treating l10n as something you can just |
Thank you for taking the time to look at Slint and commenting here! I'm intrigued by the concept of a "localisation unit". In the current model for translations in Slint that we have in mind, each translation is a binding that's automatically kept up-to-date. And based on our experience with KDE and Qt, we've seen this work. But it's still relatively hard for translators to get enough context about where the strings are really used, to create the best possible translation. I'm rather intrigued by the idea of enhancing the DSL (in our case) in a way that we could perhaps extract more structural and relational information for translators to see. That would indeed require a message format beyond the dumb I reckon we might do this in stages though. |
UMF2 looks very intriguing, indeed, even better than Fluent. I also assume it’ll pair well with ICU4x, for obvious reasons, so… guess that’s the only sane target to aim for regarding I18N. I was quite surprised seeing a v1.0 release with this here and the accessibility issue still open, given how deeply proper I18N integrates into pretty much everything. |
Prototype in #2662 We decided to go with a gettext approach, because we feel like it is better to have the original ebglish string in the .slint file Regarding the comments that suggest fluent, I think this is valid and it is entirely possible to use fluent with Slint, see One way could just to have a generic callback that forward to fluent
or even some possibly auto-generated global:
|
I don’t quite get it, to be honest. Fluent and UMF2 have been designed to tackle ages old, well known, limitations and weaknesses of gettext-like libraries. Namely the limitations that real human text is dependent on the parameters you try to render in between. Things like auto pluralisation. And these two are existing standards with existing implementations, one directly by Unicode. You’re working on a new GUI framework that has the opportunity to »Do Things Right™« from the get go, unlike, say, Qt, which has been around since before Fluent existed. And having mentioned Qt, it is not uncommon to pack the base translations right into the shipped executable, so there can’t be a case where default translations are missing. Slint can do the exact same, so where exactly is the value in having »original English« in the source code? If anything, that’s a downside, because in a professional setting, your editors (the people, not the software) now have to touch UI source code to fix mistakes in the default translation. And even worse, changing the default translation in source code now also means »fixing« all translation keys in all translation files. There’s a reason Qt at some point added a feature to generically look translations up by an unchanging key instead of plain English. I’m surprised and confused. |
@Evrey Thank you for sharing your concerns and insights! But gettext remains a state-of-the-art solution widely adopted in the industry and is even integrated into the glibc. Regarding Slint's formatting layer, we are still in the process of finalizing it. For the MVP, we plan to use numbered placeholders like Slint aims to support MCU and no_std runtimes, so we will need an option to read the translations at compile time to embed them into the binary as well. The main reason we want to use gettext is because we want the original in the .slint files because it offers convenience for UI developers who deal with numerous strings. The intention is to simplify the process by enabling developers to place strings within quotes without the need for extra message IDs or separate files. The concerns about fixing typos invalidating the string are effectively addressed by existing gettext tooling. These tools handle scenarios where the original source changes, ensuring translations remain intact while marking the translation dirty as changes to the original source often require corresponding adjustments in the translations.
I'm not aware of that feature. Anyway, i have read https://github.com/projectfluent/fluent/wiki/Fluent-vs-gettext#social-contract and I disagree with it: Strings in the source code are easier to write maintain (as they are in the right context) |
I personally prefer key to plain English I have used fluent in work, and feels the flexibility and customizability be unexpectedly necessary. Except for plural nouns, verb tense and many others are a must. I even defined several custom functions, one to join a list of numbers to string, with different delimiter in different language; one to convert discount number, e.g. |
Released a blog post about this: https://slint-ui.com/blog/translation-infrastructure |
Initial translation infrastructure was merged |
Hi @ogoffart - one correction for your blog post. Your snippet of "this is how it would look in Fluent" is architecturally wrong. As explained in the documents listed above Fluent actively discourages the |
@zbraniecki I'm not sure how that would work with Slint though, there is no DOM with Slint, so using fluent dom is out of question. Where do they discourages this usages? And if the snippet i used in the blog is wrong, what could you imagine a write syntax be? |
I just meant that in the example you should have Label.l10nId and assign a translation unit to a UI element, not a translation string to a single attribute. (in fluent a single message may have a value and multiple attributes - a binding of a single message with multiple attributes to a single element with multiple translatable attributes ) |
But they are not translators! I can't imagine having a UI designer complete all the i18n work.
I disagree with this. |
Indeed, never said they are. They need to put the right context and have decent string, but that is indeed hard.
This is taken care of by the msgctx of gettext, which by default with Slint is the component name, but can also be customized with So the translators don't just see "Ok", they see ("LogoutDialog", "Ok") and ("RemoveDialog", "Ok") |
This ticket tracks the ability to translate a Slint user interface in a way that allows annotating translatable strings,
Proposal.
@tr(...)
macro with the same syntax as the rusttr!
macro from the tr crate. (cf RFC: design of the tr! macro woboq/tr#1 )Workaround until this is implemented.
Put all the string in a global object and do the translation in native code
See #33 (comment)
The text was updated successfully, but these errors were encountered: