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

Argument with custom type is not passed to the formatter function #747

Open
riolly opened this issue Nov 4, 2023 · 3 comments
Open

Argument with custom type is not passed to the formatter function #747

riolly opened this issue Nov 4, 2023 · 3 comments
Labels
bug Something isn't working

Comments

@riolly
Copy link

riolly commented Nov 4, 2023

Version

5.26.2

Describe the bug

Following this instruction failed to pass an argument of custom type to the formatter function.

Reproduction

// en/index.ts
dateRange: "{0: DateRange|formatDateRange}",

// custom-types.ts
export interface DateRange {
  from: Date;
  to: Date;
}

// formatters.ts
formatDateRange: (value) => {
  console.log(value) // correctly typed but undefined 
  // using date-fns
  return `${format(value.from, "MM/dd/yyyy", {
    locale: localeDate[locale],
  })} - ${format(value.to, "MM/dd/yyyy", {
    locale: localeDate[locale],
  })}`;
},
  
// app.js
LL.dateRange({ from: new Date(), to: new Date() })

Logs

Cannot read property 'from' of undefined

Value correctly typed but the value undefined

Config

{
   "baseLocale": "en",
   "adapter": "react",
   "$schema": "https://unpkg.com/[email protected]/schema/typesafe-i18n.json"
}

Additional information

No response

@riolly riolly added the bug Something isn't working label Nov 4, 2023
@riolly riolly changed the title Can't pass multiple argument to formatter Argument with custom type is not passed to the formatter Nov 4, 2023
@riolly riolly changed the title Argument with custom type is not passed to the formatter Argument with custom type is not passed to the formatter function Nov 4, 2023
@vladev
Copy link

vladev commented Feb 10, 2024

I've stumbled upon this as well.

// en/component/index.ts
totalPrice: '{value:Currency|currency}',
// custom-types.ts
export type Currency = { value: number; currency: string }
// formatters.ts
currency: (value: Currency) =>
  number(locale, {
    style: 'currency',
    currency: value.currency,
    maximumFractionDigits: 2,
  })(value.value)

@jamespamplin
Copy link

I also got hit with this - and also with a currency formatter similar to @vladev.

// i18n/custom-types.ts
export type MonetaryAmount = {
  currency: string
  value: number
}

// i18n/en/index.ts
const en = {
  money: "{0:MonetaryAmount|currency}",
}

// i18n/formatters.ts
const formatters: Formatters = {
  currency: ({ value, currency }: MonetaryAmount): string => {
    const formatter = new Intl.NumberFormat(locale, {
    style: "currency",
    currency,
    minimumFractionDigits: 0,
  }

  return formatter.format(value)
}

// component.ts
function Money({value, valueCurrency}: {value: number, valueCurrency: string}) {
  const { LL } = useI18nContext()
  // below is correctly typed, but throws TypeError: Cannot read property 'value' of undefined
  return <Text>{LL.money({ value, currency: valueCurrency })}</Text>
}

Even though this was passing typescript, the call to LL was passing the argument as a "keyed argument". Passing the index through in an object works around the issue, but fails typescript: LL.money({ 0: { value, currency: valueCurrency }})

As a workaround, specifying an argument key in the translation passes typescript and works as expected at runtime:

// i18n/en/index.ts
const en = {
  money: "{money:MonetaryAmount|currency}",
}

// component.ts
function Money({value, valueCurrency}: {value: number, valueCurrency: string}) {
  const { LL } = useI18nContext()
  // now compiles in typescript and works as expected at runtime
  return <Text>{LL.money({ money: { value, currency: valueCurrency }})}</Text>
}

So this is only an issue if you use a single indexed argument in the translation.

@Nicolab
Copy link

Nicolab commented Jan 7, 2025

Thanks for your examples. Additionally, if this can help other devs:

// custom-types.ts

export type CurrencyOptions = {
  value: number;
  currency?: string,
  display?: 'symbol' | 'code',
  minDigits?: number,
  maxDigits?: number,
}
// formatters.ts

import type { FormattersInitializer } from 'typesafe-i18n'
import { number, time } from 'typesafe-i18n/formatters'
import type { CurrencyOptions } from './custom-types.js'
import type { Formatters, Locales } from './i18n-types.js'

export const initFormatters: FormattersInitializer<Locales, Formatters> = (locale: Locales) => {
	const formatters: Formatters = {
		timeShort: time(locale, { timeStyle: 'short' }),
		timeLong: time(locale, { timeStyle: 'long' }),
		dateLong: time(locale, { dateStyle: 'long' }),
		dateShort: time(locale, { dateStyle: 'short' }),
		dateMedium: time(locale, { dateStyle: 'medium' }),
		dateFull: time(locale, { dateStyle: 'full' }),
		// currency: number(locale, { style: 'currency', currency: 'EUR' }),
		currency: (options: CurrencyOptions) => {
			const {
				value,
				currency = 'EUR',
				display = 'symbol',
				minDigits = 2,
				maxDigits = 2
			} = options || {};

			return number(locale, {
				style: 'currency',
				currency,
				currencyDisplay: display,
				minimumFractionDigits: minDigits,
				maximumFractionDigits: maxDigits
			})(value);
		},
	}

	return formatters
}
// en/index.ts

const en : BaseTranslation = {
  moneyFromCurrency: '{options:CurrencyOptions|currency}',
} satisfies Translation

Usage:

// Minimal (using default values)
t.moneyFromCurrency({ options: { value: 10 } })

// Full args used
t.moneyFromCurrency({ options: { value: 10, currency: 'EUR', display: 'symbol', minDigits: 0, maxDigits: 0 } })

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

No branches or pull requests

4 participants