Skip to content
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

IntlMessageFormat and custom formatters #5

Closed
kachkaev opened this issue Sep 1, 2018 · 27 comments
Closed

IntlMessageFormat and custom formatters #5

kachkaev opened this issue Sep 1, 2018 · 27 comments

Comments

@kachkaev
Copy link

kachkaev commented Sep 1, 2018

Hi @jamuhl 👋

I'm opening this issue to discuss how custom formatters could be added to i18next-icu. It seems that MessageFormat you are leveraging has this functionality already, it's jut not exposed into the API of i18next-icu. See the docs here:

https://messageformat.github.io/messageformat/page-guide#toc11__anchor
https://messageformat.github.io/messageformat/MessageFormat.html#addFormatters

The reason why I have started exploring custom formatters is because I would like to output values with certain numbers of fractional digits, e.g. 1, 3 or 6.

Here's how my translation files could look in an ideal world:

{
  "example1": "Our value is equal to {value, numberWithThreeFractionalDigits}.",
  "example2": "Our value is roughly equal to {value, numberWithFD, 1}, but if you like things to be precise, it is {value, numberWithFD, 6}.",
}

I'm not yet sure what I'd write in addFormatters() to define numberWithThreeFractionalDigits, numberWithFD or whatever. But before I get there, it'd be great to know how addFormatters() could be called from i18next-icu. I see you are doing something smart about memorisation of IntlMessageFormat here, which needs to be taken into account:

i18next.IntlMessageFormat = IntlMessageFormat;

i18next-icu/src/index.js

Lines 45 to 53 in 133d297

let fc;
if (this.options.memoize) {
fc = utils.getPath(this.mem, `${lng}.${ns}.${key}`);
}
if (!fc) {
fc = new IntlMessageFormat(res, lng);
if (this.options.memoize && (this.options.memoizeFallback || !info || hadSuccessfulLookup)) utils.setPath(this.mem, `${lng}.${ns}.${key}`, fc);
}
return fc.format(options);

Being able to define custom formatters in the ICU plugin will make it as extensible as the main formatter. And in turn this will add even more value to the synergy i18next ecosystem creates!

@jamuhl
Copy link
Member

jamuhl commented Sep 2, 2018

hm...not sure if people add customFormatters per message or just add them like once as default to all messages...

in i18next we have the https://www.i18next.com/translation-function/formatting

we might just have some additional option added to icu init like:

i18next
  .use(ICU)
  .init({
    i18nFormat: {
      localeData: fr, // you also can pass in array of localeData

formatters: {
  upcase: function(v) { return v.toUpperCase(); },
  locale: function(v, lc) { return lc; },
  prop: function(v, lc, p) { return v[p] }
}

    }
  });

those get added to all compiled message functions here: https://github.com/i18next/i18next-icu/blob/master/src/index.js#L50

Sorry just take a quick look...not sure if i got it right how it should behave.

@kachkaev
Copy link
Author

kachkaev commented Sep 2, 2018

According to the docs, IntlMessageFormat expects a map of formatters to be the third parameter. It is not passed to the constructor at the moment:

fc = new IntlMessageFormat(res, lng);

@kachkaev
Copy link
Author

kachkaev commented Sep 2, 2018

From the user point of view the solution could look something like:

const icu = new ICU();
icu.addLocaleData(fr);
icu.addFormatters({
  upcase: function(v) { return v.toUpperCase(); },
  locale: function(v, lc) { return lc; },
  prop: function(v, lc, p) { return v[p] }
});
i18n.use(icu);

@jamuhl
Copy link
Member

jamuhl commented Sep 2, 2018

we could provide both...an additional addFormatters function and passing them via options:

this.options = utils.defaults(i18nextOptions, options, this.options || {}, getDefaults());

@kachkaev
Copy link
Author

kachkaev commented Sep 2, 2018

Yeah that'd be cool. I'll be happy to test this feature if you are fancy an experiment 😉

@jamuhl
Copy link
Member

jamuhl commented Sep 3, 2018

Will see when i find time...not hard to add this...more a problem of my time management 😭

@kachkaev
Copy link
Author

kachkaev commented Sep 3, 2018

No worries, I know your pain! Just ping me when you want to go for it and I'll find some time to give the feature a test 😉

@jamuhl
Copy link
Member

jamuhl commented Sep 3, 2018

hm...just started with it...as i thought simplest of all the shit i got planned....oh no...not really:

https://github.com/yahoo/intl-messageformat/issues/121#issuecomment-176043615

IntlMessageFormat custom formatters are no custom formatters just some...idk what they are: https://github.com/yahoo/intl-messageformat#user-defined-formats

And currently i do not expect them to do anything in this direction. So only option i currently see would be moving to messageformat

@kachkaev
Copy link
Author

kachkaev commented Sep 3, 2018

😅 oh gosh that's confusing indeed! Looks like my brain mixed-up MessageFormat formatters with IntlMessageFormat user-defined formats – I really thought they were the same thing!

Although user-defined formats in IntlMessageFormat are not functions as I originally expected, injecting them via i18next-icu may still make sense. Check out the options they pass in the example you referred to:

var msg = new IntlMessageFormat('The price is: {price, number, USD}', 'en-US', {
    number: {
        USD: {
            style   : 'currency',
            currency: 'USD'
        }
    }
});

This { style: 'currency', currency: 'USD' } looks very much like what gets passed to the browser's NumberFormat or its polyfill. This means that we can do something like:

const icu = new ICU();
icu.addLocaleData(fr);
icu.addUserDefinedFormats({
  number: {
    THREE_FRACTIONAL_DIGITS: {
      minimumSignificantDigits: 3,
      maximumSignificantDigits: 3
    },
    ROUGH: {
      maximumSignificantDigits: 1
    },
    VERY_PRECISE: {
      minimumSignificantDigits: 6,
    },
  }
});
i18n.use(icu);

And then:

{
  "example1": "Our value is equal to {value, number, THREE_FRACTIONAL_DIGITS}.",
  "example2": "Our value is roughly equal to {value, number, ROUGH}, but if you like things to be precise, it is {value, number, VERY_PRECISE}.",
}

There are plenty of options NumberFormat accepts, which makes it pretty powerful. Turns out we can cover lots of formatting cases just keeping things declarative – no need to use functions!

WDYT?

@jamuhl
Copy link
Member

jamuhl commented Sep 3, 2018

hope tests are sufficient to get how it works: 9607f39

currently running travis...will npm publish if all green

@jamuhl
Copy link
Member

jamuhl commented Sep 3, 2018

i18next-icu@0.5.0

@kachkaev
Copy link
Author

kachkaev commented Sep 3, 2018

Outstanding 🎉 I'll test your commit soon! Just one quick note before you've released this: should the method be called addFormatters or addUserDefinedFormats as I suggest in the more recent comment? The second name slightly better reflects the origin of the feature, but I don't have a strong opinion here.

@kachkaev
Copy link
Author

kachkaev commented Sep 3, 2018

Ouch, looks like I was a bit late 😅

@kachkaev
Copy link
Author

kachkaev commented Sep 3, 2018

Just noticed that the third argument in IntlMessageFormat constructor is called formats, not formatters: https://github.com/yahoo/intl-messageformat/blob/master/index.d.ts#L2

It does not influence the behaviour in any way, just makes things a bit less obscure (especially given the fact that the things with IntlMessageFormat vs MessageFormat are confusing from start 😅 ).

Let me give your new version a try first anyway! Given that it's unlikely that too many people will be use custom formats from day one, I'll be happy to see the methods renamed if you agree with me that this can positively influence the coherence of 18next-icu.

@kachkaev
Copy link
Author

kachkaev commented Sep 3, 2018

Custom formats work well when passed to the ICU constructor:

const icu = new ICU({
  formatters: {
    number: {
      THREE_FRACTION_DIGITS: {
        minimumFractionDigits: 3,
        maximumFractionDigits: 3,
      },
    },
  },
});

This is truly great! 🎉

However, when I try using addFormatters, the numbers format as if my custom format did not exist:

const icu = new ICU();
icu.addFormatters({
  number: {
    PRICE: {
      minimumFractionDigits: 3,
    },
  },
});

I could get as far as adding several console.log() calls to node_modules/i18next-icu/dist/commonjs/index.js and noticed something strange. When formatters are passed as options, this.formatters are defined in parse(). When calling addFormatters(), this.formatters are also defined in this method, but are undefined in the following parse() calls. Does it give any hints?


One thing worth noting in my most recent example is that I replaced minimumSignificantDigits: 3 with minimumFractionDigits: 3 — significant digits are both fractional and integer ones combined. I also renamed THREE_FRACTIONAL_DIGITS with THREE_FRACTION_DIGITS for naming consistency with MessageFormat.

@jamuhl
Copy link
Member

jamuhl commented Sep 4, 2018

changed test to use the add function -> works on my machine and travis...

could you take another look - as i can't reproduce that. Only thing i know is having formats on init and additional adding via add function will overwrite them.

also done renamings...hope you like that

@jamuhl
Copy link
Member

jamuhl commented Sep 4, 2018

i18next-icu@0.6.0 includes changes.

@kachkaev
Copy link
Author

kachkaev commented Sep 4, 2018

Great, thank you 🎉 👏 I'll try the changes tonight.

@kachkaev
Copy link
Author

kachkaev commented Sep 4, 2018

Hi @jamuhl, just a quick heads up. I ran out of time this evening, so have to postpone my tests for another day. Hope this can be tomorrow – I'll let you know once I get there. Sorry for this.

@jamuhl
Copy link
Member

jamuhl commented Sep 5, 2018

@kachkaev no problem...if there is something breaking it will be no big deal...i think after this i will change version to v1.0.0 anyway.

@dyniper
Copy link

dyniper commented Dec 15, 2018

I did some testing with this. The formatters passed in the ctor options works, but calling addUserDefinedFormats doesn't actually add the formats.

@kachkaev
Copy link
Author

kachkaev commented Dec 15, 2018

@dyniper this worked for me to register a custom ICU format:

import * as ICU from "i18next-icu";
const icu = new ICU({
  formats: {
    number: {
      PRICE: {
        minimumFractionDigits: 2,
        useGrouping: false,
      },
    },
  },
});

If this solution satisfies you too, let’s close this issue.

@dyniper
Copy link

dyniper commented Dec 17, 2018

Yes, this work, but if you do:

import * as ICU from "i18next-icu";
const icu = new ICU();
icu.addUserDefinedFormats({
  formats: {
    number: {
      PRICE: {
        minimumFractionDigits: 2,
        useGrouping: false,
      },
    },
  },
});

, that one doesn't work. I don't mind having to provide the formats in the ctor, but if there is a method to add them after constructing the object, it should work, otherwise, remove the method.

@jamuhl
Copy link
Member

jamuhl commented Dec 18, 2018

@dyniper PR with a fix or removing that is welcome

@dyniper
Copy link

dyniper commented Dec 20, 2018

Will put that on the pile. Really busy these days and it makes hard to do community contributions. Thanks for everything you are doing.
By the way, I'm thinking about doing a i18nFormat module based on messageformat instead (https://github.com/messageformat/messageformat) as this would provide true custom formatter.

@jamuhl
Copy link
Member

jamuhl commented Dec 20, 2018

@dyniper sounds like a good plan 👍

@jamuhl jamuhl closed this as completed Dec 20, 2018
@developerdanwu
Copy link

First of all, thanks for creating such a fantastic tool and my apologies for commenting on such an old issue. However, I was trying to use the custom formatter like the code below and one of my translators appears to want a 24 hour time whereas for the rest of the locales I want 12 hour time. Is it possible to set these custom formatters based on the locale? Eg. if user is on en it would one set of configs and another locale another set. If this is not an existing feature, I am willing to contribute and open a PR for this if given some direction on what would be the best solution. Thanks!

const icu = new ICU({
  parseLngForICU: (lng) => {
    return lng === 'en' ? 'en-GB' : lng;
  },
  memoize: true,
  formats: {
    number: {},
    date: {
      long: {
        day: 'numeric',
        month: 'short',
        year: 'numeric',
        weekday: 'long',
      },
      full: {
        hour12: true,
        day: 'numeric',
        month: 'short',
        year: 'numeric',
        weekday: 'long',
        hour: '2-digit',
        minute: '2-digit',
      },
    },
    time: {
      short: {
        hour12: true,
      },
    },
  },
});

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

4 participants