diff --git a/example/src/routes/_layout.svelte b/example/src/routes/_layout.svelte
index 7771265..36f972f 100644
--- a/example/src/routes/_layout.svelte
+++ b/example/src/routes/_layout.svelte
@@ -1,10 +1,11 @@
diff --git a/src/client/includes/loaderQueue.ts b/src/client/includes/loaderQueue.ts
index 988c541..37e0d97 100644
--- a/src/client/includes/loaderQueue.ts
+++ b/src/client/includes/loaderQueue.ts
@@ -4,11 +4,9 @@ import {
$dictionary,
addMessages,
} from '../stores/dictionary'
-import { getCurrentLocale } from '../stores/locale'
+import { getCurrentLocale, getFallbacksOf } from '../stores/locale'
import { $isLoading } from '../stores/loading'
-import { getAllFallbackLocales } from './utils'
-
type Queue = Set
const loaderQueue: Record = {}
@@ -25,7 +23,7 @@ function getLocaleQueue(locale: string) {
}
function getLocalesQueues(locale: string) {
- return getAllFallbackLocales(locale)
+ return getFallbacksOf(locale)
.reverse()
.map<[string, MessagesLoader[]]>(localeItem => {
const queue = getLocaleQueue(localeItem)
@@ -35,7 +33,7 @@ function getLocalesQueues(locale: string) {
}
export function hasLocaleQueue(locale: string) {
- return getAllFallbackLocales(locale)
+ return getFallbacksOf(locale)
.reverse()
.some(getLocaleQueue)
}
diff --git a/src/client/includes/lookup.ts b/src/client/includes/lookup.ts
index 39d0ec4..9f786c2 100644
--- a/src/client/includes/lookup.ts
+++ b/src/client/includes/lookup.ts
@@ -2,8 +2,7 @@
import resolvePath from 'object-resolve-path'
import { hasLocaleDictionary } from '../stores/dictionary'
-
-import { getFallbackLocale } from './utils'
+import { getFallbackOf } from '../stores/locale'
const lookupCache: Record> = {}
@@ -34,6 +33,6 @@ export const lookupMessage = (
return addToCache(
path,
locale,
- lookupMessage(dictionary, path, getFallbackLocale(locale))
+ lookupMessage(dictionary, path, getFallbackOf(locale))
)
}
diff --git a/src/client/includes/utils.ts b/src/client/includes/utils.ts
index fdfe24c..e3335d1 100644
--- a/src/client/includes/utils.ts
+++ b/src/client/includes/utils.ts
@@ -16,15 +16,6 @@ export function lower(str: string) {
return str.toLocaleLowerCase()
}
-export function getFallbackLocale(locale: string) {
- const index = locale.lastIndexOf('-')
- return index > 0 ? locale.slice(0, index) : null
-}
-
-export function getAllFallbackLocales(locale: string) {
- return locale.split('-').map((_, i, arr) => arr.slice(0, i + 1).join('-'))
-}
-
const getFromURL = (urlPart: string, key: string) => {
const keyVal = urlPart
.substr(1)
@@ -49,12 +40,11 @@ export const getClientLocale = ({
pathname,
hostname,
default: defaultLocale,
+ fallback = defaultLocale,
}: GetClientLocaleOptions) => {
let locale
- if (typeof window === 'undefined') {
- return defaultLocale
- }
+ if (typeof window === 'undefined') return fallback
if (hostname) {
locale = getFirstMatch(window.location.hostname, hostname)
@@ -88,5 +78,5 @@ export const getClientLocale = ({
if (locale) return locale
}
- return defaultLocale
+ return fallback
}
diff --git a/src/client/index.ts b/src/client/index.ts
index 24f88df..7b08c6c 100644
--- a/src/client/index.ts
+++ b/src/client/index.ts
@@ -1,22 +1,18 @@
import merge from 'deepmerge'
-import { GetClientLocaleOptions, MessageObject } from './types'
-import { getClientLocale } from './includes/utils'
-import { $locale } from './stores/locale'
+import { MessageObject } from './types'
// defineMessages allow us to define and extract dynamic message ids
export function defineMessages(i: Record) {
return i
}
-export function waitInitialLocale(options: GetClientLocaleOptions | string) {
- if (typeof options === 'string') {
- return $locale.set(options)
- }
- return $locale.set(getClientLocale(options))
-}
-
-export { $locale as locale, loadLocale as preloadLocale } from './stores/locale'
+export {
+ $locale as locale,
+ setInitialLocale,
+ // @deprecated
+ setInitialLocale as waitInitialLocale,
+} from './stores/locale'
export {
$dictionary as dictionary,
$locales as locales,
@@ -26,7 +22,7 @@ export { $isLoading as isLoading } from './stores/loading'
export { $format as format, $format as _, $format as t } from './stores/format'
// utilities
-export { getClientLocale, merge }
+export { merge }
export { customFormats, addCustomFormats } from './includes/formats'
export {
flushQueue as waitLocale,
diff --git a/src/client/stores/dictionary.ts b/src/client/stores/dictionary.ts
index 04428b8..ba2b729 100644
--- a/src/client/stores/dictionary.ts
+++ b/src/client/stores/dictionary.ts
@@ -3,22 +3,25 @@ import { writable, derived } from 'svelte/store'
import { LocaleDictionary } from '../types/index'
-let dictionary: LocaleDictionary
+import { getFallbackOf } from './locale'
+let dictionary: LocaleDictionary
const $dictionary = writable({})
-$dictionary.subscribe(newDictionary => {
- dictionary = newDictionary
-})
-function getDictionary() {
+export function getDictionary() {
return dictionary
}
-function hasLocaleDictionary(locale: string) {
+export function hasLocaleDictionary(locale: string) {
return locale in dictionary
}
-function addMessages(locale: string, ...partials: LocaleDictionary[]) {
+export function getAvailableLocale(locale: string): string | null {
+ if (locale in dictionary || locale == null) return locale
+ return getAvailableLocale(getFallbackOf(locale))
+}
+
+export function addMessages(locale: string, ...partials: LocaleDictionary[]) {
$dictionary.update(d => {
dictionary[locale] = merge.all([dictionary[locale] || {}].concat(partials))
return d
@@ -29,10 +32,6 @@ const $locales = derived([$dictionary], ([$dictionary]) =>
Object.keys($dictionary)
)
-export {
- $dictionary,
- $locales,
- getDictionary,
- hasLocaleDictionary,
- addMessages,
-}
+$dictionary.subscribe(newDictionary => (dictionary = newDictionary))
+
+export { $dictionary, $locales }
diff --git a/src/client/stores/format.ts b/src/client/stores/format.ts
index cf86055..78b4f10 100644
--- a/src/client/stores/format.ts
+++ b/src/client/stores/format.ts
@@ -3,13 +3,7 @@ import { derived } from 'svelte/store'
import { Formatter, MessageObject } from '../types'
import { lookupMessage } from '../includes/lookup'
import { hasLocaleQueue } from '../includes/loaderQueue'
-import {
- getAllFallbackLocales,
- capital,
- upper,
- lower,
- title,
-} from '../includes/utils'
+import { capital, upper, lower, title } from '../includes/utils'
import {
getMessageFormatter,
getTimeFormatter,
@@ -18,7 +12,7 @@ import {
} from '../includes/formats'
import { getDictionary, $dictionary } from './dictionary'
-import { getCurrentLocale, $locale } from './locale'
+import { getCurrentLocale, getFallbacksOf, $locale } from './locale'
const formatMessage: Formatter = (id, options = {}) => {
if (typeof id === 'object') {
@@ -38,7 +32,7 @@ const formatMessage: Formatter = (id, options = {}) => {
if (!message) {
console.warn(
- `[svelte-i18n] The message "${id}" was not found in "${getAllFallbackLocales(
+ `[svelte-i18n] The message "${id}" was not found in "${getFallbacksOf(
locale
).join('", "')}". ${
hasLocaleQueue(getCurrentLocale())
diff --git a/src/client/stores/locale.ts b/src/client/stores/locale.ts
index 949ea87..c924da4 100644
--- a/src/client/stores/locale.ts
+++ b/src/client/stores/locale.ts
@@ -1,30 +1,54 @@
import { writable } from 'svelte/store'
-import { getFallbackLocale, getAllFallbackLocales } from '../includes/utils'
import { flushQueue, hasLocaleQueue } from '../includes/loaderQueue'
+import { getClientLocale } from '../includes/utils'
+import { GetClientLocaleOptions } from '../types'
-import { getDictionary } from './dictionary'
+import { getAvailableLocale } from './dictionary'
+let fallback: string = null
let current: string
const $locale = writable(null)
-function getCurrentLocale() {
- return current
+export function getFallbackLocale() {
+ return fallback
+}
+
+export function setfallbackLocale(locale: string) {
+ fallback = locale
+}
+
+export function isFallbackLocaleOf(localeA: string, localeB: string) {
+ return localeB.indexOf(localeA) === 0
+}
+
+export function getFallbackOf(locale: string) {
+ const index = locale.lastIndexOf('-')
+ if (index > 0) return locale.slice(0, index)
+ if (fallback && !isFallbackLocaleOf(locale, fallback)) return fallback
+ return null
}
-function getAvailableLocale(locale: string): string | null {
- if (locale in getDictionary() || locale == null) return locale
- return getAvailableLocale(getFallbackLocale(locale))
+export function getFallbacksOf(locale: string): string[] {
+ const locales = locale
+ .split('-')
+ .map((_, i, arr) => arr.slice(0, i + 1).join('-'))
+
+ if (fallback != null && !isFallbackLocaleOf(locale, fallback)) {
+ return locales.concat(getFallbacksOf(fallback))
+ }
+ return locales
+}
+
+function getCurrentLocale() {
+ return current
}
-function loadLocale(localeToLoad: string) {
- return Promise.all(
- getAllFallbackLocales(localeToLoad).map(localeItem =>
- flushQueue(localeItem)
- .then(() => [localeItem, { err: undefined }])
- .catch(e => [localeItem, { err: e }])
- )
- )
+export function setInitialLocale(options: GetClientLocaleOptions) {
+ if (typeof options.fallback === 'string') {
+ setfallbackLocale(options.fallback)
+ }
+ return $locale.set(getClientLocale(options))
}
$locale.subscribe((newLocale: string) => {
@@ -37,17 +61,13 @@ $locale.subscribe((newLocale: string) => {
const localeSet = $locale.set
$locale.set = (newLocale: string): void | Promise => {
- if (getAvailableLocale(newLocale)) {
- if (hasLocaleQueue(newLocale)) {
- return flushQueue(newLocale).then(() => localeSet(newLocale))
- }
- return localeSet(newLocale)
+ if (getAvailableLocale(newLocale) && hasLocaleQueue(newLocale)) {
+ return flushQueue(newLocale).then(() => localeSet(newLocale))
}
-
- throw Error(`[svelte-i18n] Locale "${newLocale}" not found.`)
+ return localeSet(newLocale)
}
$locale.update = (fn: (locale: string) => void | Promise) =>
localeSet(fn(current))
-export { $locale, loadLocale, flushQueue, getCurrentLocale }
+export { $locale, flushQueue, getCurrentLocale }
diff --git a/test/client/index.test.ts b/test/client/index.test.ts
index bf2e2d5..43140e7 100644
--- a/test/client/index.test.ts
+++ b/test/client/index.test.ts
@@ -4,13 +4,12 @@ import {
dictionary,
locale,
format,
- getClientLocale,
addCustomFormats,
customFormats,
- preloadLocale,
register,
waitLocale,
} from '../../src/client'
+import { getClientLocale } from '../../src/client/includes/utils'
global.Intl = require('intl')
@@ -42,19 +41,9 @@ describe('locale', () => {
await locale.set('en-US')
expect(currentLocale).toBe('en-US')
})
-
- it("should throw an error if locale doesn't exist", () => {
- expect(() => locale.set('FOO')).toThrow()
- })
})
describe('dictionary', () => {
- it('load a locale and its derived locales if dictionary is a loader', async () => {
- const loaded = await preloadLocale('pt-PT')
- expect(loaded[0][0]).toEqual('pt')
- expect(loaded[1][0]).toEqual('pt-PT')
- })
-
it('load a partial dictionary and merge it with the existing one', async () => {
await locale.set('en')
register('en', () => import('../fixtures/partials/en.json'))
@@ -96,7 +85,8 @@ describe('formatting', () => {
expect(_({ id: 'switch.lang' })).toBe('Switch language')
})
- it('should translate to passed locale', () => {
+ it('should translate to passed locale', async () => {
+ await waitLocale('pt-BR')
expect(_('switch.lang', { locale: 'pt' })).toBe('Trocar idioma')
})