Package Exports
- use-t
- use-t/lib/index.js
This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (use-t) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
🗺
use-t
Modern React translations made simple.
use-t is a lightweight, type-safe React internationalization (i18n) library that leverages React hooks and context for seamless translation management. Built with modern React patterns, it provides an intuitive API for handling translations, dynamic loading, namespaces, and complex interpolations.
✨ Features
🪝 Hook-based API — Modern React hooks with useT()
🚀 Zero dependencies — Lightweight and fast
📦 Dynamic loading — Load translations on-demand
🏷️ Namespace support — Organize translations by feature
🔧 Template literals — JSX interpolation with t.t
🌍 Fallback locales — Graceful degradation
📝 TypeScript — Full type safety and IntelliSense
⚡ Multiple APIs — Hooks, HOCs, render props, and context
Installation
npm i use-t
Quick Start
Basic Usage
import {Provider, useT} from 'use-t';
// 1. Define your translations
const translations = {
en: {
main: {
greeting: 'Hello',
welcome: 'Welcome back!'
}
},
es: {
main: {
greeting: 'Hola',
welcome: '¡Bienvenido de nuevo!'
}
}
};
// 2. Create a component that uses translations
const App = () => {
const [t, {setLocale, locale}] = useT();
return (
<div>
<h1>{t('greeting')}, World!</h1>
<p>{t('welcome')}</p>
<button onClick={() => setLocale(locale === 'en' ? 'es' : 'en')}>
Switch Language
</button>
</div>
);
};
// 3. Wrap your app with Provider
export default () => (
<Provider locale="en" map={translations}>
<App />
</Provider>
);Function-based Translations
const translations = {
en: {
main: {
userGreeting: (name) => `Hello, ${name}!`,
itemCount: (count) => `You have ${count} ${count === 1 ? 'item' : 'items'}`
}
}
};
const UserProfile = ({username, itemCount}) => {
const [t] = useT();
return (
<div>
<h2>{t('userGreeting', username)}</h2>
<p>{t('itemCount', itemCount)}</p>
</div>
);
};Template Literal Interpolation
const translations = {
en: {
main: {
welcomeMessage: (interpolate) => interpolate`Welcome ${0}, you have ${1} new messages!`
}
}
};
const Dashboard = ({user, messageCount}) => {
const [t] = useT();
return (
<div>
{/* With translation */}
{t.t('welcomeMessage')`Welcome ${user.name}, you have ${messageCount} new messages!`}
{/* Fallback if translation missing */}
{t.t('missingKey')`Default message with ${user.name}`}
</div>
);
};API Reference
import {Provider, useT, withT, Trans, Consumer, context, createTranslations} from 'use-t';| Export | Type | Description |
|---|---|---|
<Provider> |
Component | Context provider for translations |
useT() |
Hook | React hook returning [t, state] |
withT() |
HOC | Higher-order component injecting t and T props |
<Trans> |
Component | Render prop component for translations |
<Consumer> |
Component | Context consumer for provider state |
context |
Context | React context object |
createTranslations() |
Function | Create custom translation instances |
<Provider> Props
<Provider
locale="en" // Current locale (default: 'en')
defaultLocale="en" // Fallback locale (default: 'en')
ns="main" // Default namespace (default: 'main')
map={translations} // Preloaded translations
loader={loadFn} // Dynamic loader function
>useT() Hook
const [t, state] = useT(); // Default namespace
const [t, state] = useT('errors'); // Single namespace
const [t, state] = useT(['main', 'errors']); // Multiple namespacesTranslation function t:
t(key)- Simple translationt(key, ...args)- Function translation with argumentst.t(key)- Template literal translation
State object:
state.locale- Current localestate.setLocale(locale)- Change localestate.load(locale, namespace)- Preload translations
Advanced Usage
Dynamic Loading
const Provider = () => {
const loadTranslations = async (locale, namespace) => {
const response = await fetch(`/api/translations/${locale}/${namespace}`);
return response.json();
};
return (
<Provider
locale="en"
loader={loadTranslations}
map={{
en: { main: { loading: 'Loading...' } } // Initial translations
}}
>
<App />
</Provider>
);
};Namespaces
const translations = {
en: {
common: {
save: 'Save',
cancel: 'Cancel'
},
errors: {
required: 'This field is required',
invalid: 'Invalid input'
},
dashboard: {
title: 'Dashboard',
stats: 'Statistics'
}
}
};
// Use multiple namespaces
const Form = () => {
const [t] = useT(['common', 'errors']);
return (
<form>
<button type="submit">{t('save')}</button>
<button type="button">{t('cancel')}</button>
<span className="error">{t('required')}</span>
</form>
);
};
// Namespace-specific component
const Dashboard = () => {
const [t] = useT('dashboard');
return (
<div>
<h1>{t('title')}</h1>
<p>{t('stats')}</p>
</div>
);
};Complex Interpolations
const translations = {
en: {
main: {
loginFooter: (interpolate) => interpolate`
By signing in, you agree to our ${0} and ${1}.
`,
notification: (interpolate) => interpolate`
${0} sent you ${1} ${2}
`
}
}
};
const LoginForm = () => {
const [t] = useT();
return (
<div>
<form>...</form>
<p>
{t.t('loginFooter')`
By signing in, you agree to our ${<Link to="/terms">Terms</Link>} and ${<Link to="/privacy">Privacy Policy</Link>}.
`}
</p>
</div>
);
};
const NotificationItem = ({sender, count, type}) => {
const [t] = useT();
return (
<div>
{t.t('notification')`
${<strong>{sender}</strong>} sent you ${count} ${type}
`}
</div>
);
};Higher-Order Component
import {withT} from 'use-t';
const MyComponent = ({t, T, ...otherProps}) => (
<div>
<h1>{t('title')}</h1>
<button onClick={() => T.setLocale('es')}>
Español
</button>
</div>
);
export default withT(MyComponent);
// Or with specific namespace:
export default withT(MyComponent, 'dashboard');Render Props
import {Trans} from 'use-t';
const Navigation = () => (
<nav>
<Trans ns="navigation">
{(t, T) => (
<>
<a href="/">{t('home')}</a>
<a href="/about">{t('about')}</a>
<button onClick={() => T.setLocale('fr')}>
Français
</button>
</>
)}
</Trans>
</nav>
);
// String shorthand
const Title = () => <Trans>pageTitle</Trans>;
// Mixed content
const Header = () => (
<Trans>
{t => t('welcome')}!
</Trans>
);TypeScript Support
use-t is written in TypeScript and provides full type safety:
import {TranslatorFn, ProviderState, TranslationMap} from 'use-t';
// Type your translations
interface Translations {
greeting: string;
userWelcome: (name: string) => string;
itemCount: (count: number) => string;
}
const translations: TranslationMap = {
en: {
main: {
greeting: 'Hello',
userWelcome: (name: string) => `Welcome, ${name}!`,
itemCount: (count: number) => `${count} items`
} as Translations
}
};
// Typed component props
interface Props {
t: TranslatorFn;
T: ProviderState;
}
const MyComponent: React.FC<Props> = ({t, T}) => (
<div>
<h1>{t('greeting')}</h1>
<p>{t('userWelcome', 'John')}</p>
</div>
);Best Practices
Translation Organization
// ✅ Good: Organize by feature/page
const translations = {
en: {
auth: {
login: 'Log In',
signup: 'Sign Up',
forgotPassword: 'Forgot Password?'
},
dashboard: {
welcome: 'Welcome back!',
stats: 'Your Statistics'
},
common: {
save: 'Save',
cancel: 'Cancel',
loading: 'Loading...'
}
}
};
// ❌ Avoid: All translations in one namespace
const translations = {
en: {
main: {
login: 'Log In',
dashboardWelcome: 'Welcome back!',
saveButton: 'Save',
// ... hundreds of keys
}
}
};Performance Tips
// ✅ Good: Load namespaces on-demand
const LazyDashboard = () => {
const [t] = useT('dashboard'); // Only loads dashboard namespace
return <div>{t('title')}</div>;
};
// ✅ Good: Preload critical translations
<Provider
map={{
en: {
common: commonTranslations // Critical UI elements
}
}}
loader={dynamicLoader} // Non-critical loaded on-demand
>
// ✅ Good: Use default locale as fallback
<Provider
locale="fr"
defaultLocale="en" // Falls back to English if French missing
map={translations}
>Error Handling
const translations = {
en: {
main: {
// Use descriptive keys that work as fallbacks
'user.welcome': 'Welcome!',
'error.network': 'Network error occurred',
'button.save': 'Save'
}
}
};
// Keys become fallback text if translation missing
const Component = () => {
const [t] = useT();
return (
<div>
{/* Shows "Welcome!" or "user.welcome" if missing */}
<h1>{t('user.welcome')}</h1>
{/* Shows "Save" or "button.save" if missing */}
<button>{t('button.save')}</button>
</div>
);
};Custom Translation Instances
Create isolated translation contexts for libraries or complex apps:
import {createTranslations} from 'use-t';
// Create custom instance with different default namespace
const {Provider: LibProvider, useT: useLibT} = createTranslations('library');
const LibraryComponent = () => {
const [t] = useLibT();
return <div>{t('libMessage')}</div>;
};
// Use in your app
<LibProvider map={{en: {library: {libMessage: 'Hello from library!'}}}}>
<LibraryComponent />
</LibProvider>Migration from Other Libraries
From react-i18next
// react-i18next
import {useTranslation} from 'react-i18next';
const {t, i18n} = useTranslation();
t('key');
i18n.changeLanguage('es');
// use-t equivalent
import {useT} from 'use-t';
const [t, {setLocale}] = useT();
t('key');
setLocale('es');From React Intl
// React Intl
import {useIntl} from 'react-intl';
const intl = useIntl();
intl.formatMessage({id: 'key'});
// use-t equivalent
import {useT} from 'use-t';
const [t] = useT();
t('key');Troubleshooting
Translation not showing:
- Check if the key exists in your translation map
- Verify the correct namespace is being used
- Ensure Provider is wrapping your component
- Check browser console for loading errors
Dynamic loading not working:
- Verify your loader function returns a Promise
- Check network requests in browser dev tools
- Ensure proper error handling in loader
TypeScript errors:
- Import types:
import type {TranslatorFn} from 'use-t'; - Check translation map structure matches expected format
Detailed API Documentation
For comprehensive API documentation, see:
<Provider>- Context provider configurationuseT()- React hook usage and exampleswithT()- Higher-order component patterns<Trans>- Render prop component usage<Consumer>- Context consumer patternscontext- Direct context accesscreateTranslations()- Custom instances
License
Unlicense — public domain.