Domphy

Interpolation & Formatting

Basic interpolation

Pass values as the second argument to t():

// translations/en.json
{
  "greeting": "Hello, {{name}}!",
  "itemsLeft": "{{count}} items left",
  "welcomeBack": "Welcome back, {{firstName}} {{lastName}}"
}
t("greeting", { name: "Alice" })            // "Hello, Alice!"
t("itemsLeft", { count: 3 })                // "3 items left"
t("welcomeBack", { firstName: "John", lastName: "Doe" })

Nested key access

Use dot notation for nested translation keys:

{
  "profile": {
    "title": "Your Profile",
    "bio": "Tell us about {{name}}"
  }
}
t("profile.title")               // "Your Profile"
t("profile.bio", { name: "you" })  // "Tell us about you"

Number formatting

Use Intl.NumberFormat for locale-aware numbers — pass the formatted string as an interpolation value:

function formatNumber(value: number, locale: string): string {
  return new Intl.NumberFormat(locale).format(value)
}

// In component
{ span: (l) => t("price", { amount: formatNumber(1234567.89, locale.get(l)) }) }
// en-US: "Price: 1,234,567.89"
// de-DE: "Price: 1.234.567,89"

Or register a custom formatter in the i18n instance:

const { t } = createI18n({
  locale: "en",
  messages: { en: () => import("./en.json") },
  formatters: {
    number: (value, locale) => new Intl.NumberFormat(locale).format(Number(value)),
    currency: (value, locale) =>
      new Intl.NumberFormat(locale, { style: "currency", currency: "USD" }).format(Number(value)),
  },
})

// In translations:
// { "price": "Price: {{amount, currency}}" }
t("price", { amount: 42.5 })   // "Price: $42.50"

Date formatting

const { t } = createI18n({
  locale: "en",
  formatters: {
    date: (value, locale) =>
      new Intl.DateTimeFormat(locale, { dateStyle: "medium" }).format(new Date(String(value))),
    time: (value, locale) =>
      new Intl.DateTimeFormat(locale, { timeStyle: "short" }).format(new Date(String(value))),
    datetime: (value, locale) =>
      new Intl.DateTimeFormat(locale, { dateStyle: "short", timeStyle: "short" }).format(new Date(String(value))),
  },
})

// { "lastSeen": "Last seen {{date, date}}" }
t("lastSeen", { date: "2025-06-26" })   // "Last seen Jun 26, 2025"

Relative time

Use Intl.RelativeTimeFormat for "2 hours ago", "in 3 days":

function relativeTime(date: Date, locale: string): string {
  const seconds = Math.floor((Date.now() - date.getTime()) / 1000)
  const rtf = new Intl.RelativeTimeFormat(locale, { numeric: "auto" })

  if (Math.abs(seconds) < 60)   return rtf.format(-Math.round(seconds), "second")
  if (Math.abs(seconds) < 3600) return rtf.format(-Math.round(seconds / 60), "minute")
  if (Math.abs(seconds) < 86400) return rtf.format(-Math.round(seconds / 3600), "hour")
  return rtf.format(-Math.round(seconds / 86400), "day")
}

const Timestamp = (date: Date) => ({
  time: (l) => relativeTime(date, locale.get(l)),
  dateTime: date.toISOString(),
})

Escaping

Use {{- variable}} (with dash) to disable HTML escaping for safe HTML in translations:

{
  "linkText": "Read the <a href='{{url}}'>docs</a>",
  "escaped": "Use &lt;div&gt; tags"
}
t("linkText", { url: "https://domphy.com", interpolation: { escapeValue: false } })

Only disable escaping for translations you control — never for user-supplied content.

Context (gender / form variants)

Use context to select a translation variant based on gender or other categorical data:

{
  "greeting_male":   "He joined",
  "greeting_female": "She joined",
  "greeting":        "They joined"
}
t("greeting", { context: "male" })    // "He joined"
t("greeting", { context: "female" })  // "She joined"
t("greeting")                          // "They joined" (default)

Combining context and count

{
  "items_male_one":     "{{name}} has {{count}} item",
  "items_male_other":   "{{name}} has {{count}} items",
  "items_female_one":   "{{name}} has {{count}} item",
  "items_female_other": "{{name}} has {{count}} items"
}
t("items", { count: 2, context: "female", name: "Alice" })
// "Alice has 2 items"

Default values

Provide a fallback if a key is missing (useful during development):

t("missing.key", { defaultValue: "Fallback text" })

Typed interpolation

With TypeScript, interpolation values can be typed:

import type { TFunction } from "@domphy/i18n"

function render(t: TFunction) {
  // t expects exactly { name: string } for this key
  return { span: t("greeting", { name: "Alice" }) }
}