O que mudou
Infra (por app):
i18n/locales.ts — lista de locales (pt, en), default pt, labels para o seletor
i18n/request.ts — lê o cookie NEXT_LOCALE, carrega as mensagens
messages/pt.json + messages/en.json — todas as strings extraídas
next.config.ts — envolvido com withNextIntl (operator-pwa: withPWA(withNextIntl(...)))
app/layout.tsx — <html lang={locale}> dinâmico, NextIntlClientProvider
app/language-switcher.tsx — seletor PT | EN (cookie + router.refresh())
23 ficheiros de UI atualizados — todos os textos visíveis agora usam t('...') ou getTranslations.
Datas no relatório passaram de toLocaleString('pt-PT') fixo para useFormatter() do next-intl — localizam-se automaticamente.
Plurais em ICU no sync-chip: {count, plural, one {# pedido...} other {# pedidos...}}.
Resultado dos testes:
pnpm test:e2e — 3/3 ✓
pnpm test:e2e:auth — 4/4 ✓
tsc --noEmit em ambas as apps — limpo ✓
Para adicionar uma língua futura: criar messages/<locale>.json + adicionar o locale a i18n/locales.ts em cada app. O seletor aparece automaticamente.
71 lines
2.5 KiB
JSON
71 lines
2.5 KiB
JSON
{
|
|
"metadata": {
|
|
"title": "FieldOps — Operador",
|
|
"description": "Consola de operador industrial.",
|
|
"appName": "FieldOps Operador"
|
|
},
|
|
"common": {
|
|
"enter": "Entrar",
|
|
"entering": "A entrar…",
|
|
"status": {
|
|
"open": "Aberto",
|
|
"claimed": "Em curso",
|
|
"resolved": "Resolvido"
|
|
}
|
|
},
|
|
"errors": {
|
|
"title500": "500",
|
|
"message500": "Ocorreu um erro inesperado.",
|
|
"retry": "Tentar novamente",
|
|
"title404": "404",
|
|
"message404": "Página não encontrada.",
|
|
"backHome": "Voltar ao início"
|
|
},
|
|
"auth": {
|
|
"pickerTitle": "Quem és tu?",
|
|
"pickerSubtitle": "Escolhe o teu perfil para continuar.",
|
|
"noOperators": "Nenhum operador encontrado. Execute pnpm db:seed.",
|
|
"back": "Voltar",
|
|
"operatorSelected": "Operador selecionado",
|
|
"invalidPin": "PIN incorreto ou conta bloqueada. Tente novamente.",
|
|
"unexpectedError": "Erro inesperado. Tente novamente.",
|
|
"deleteDigit": "Apagar",
|
|
"switchOperator": "Trocar"
|
|
},
|
|
"home": {
|
|
"operator": "Operador",
|
|
"myRequests": "Os meus pedidos",
|
|
"requestMaintenance": "Pedir manutenção",
|
|
"noRequests": "Nenhum pedido ainda."
|
|
},
|
|
"sync": {
|
|
"deadLetters": "{count, plural, one {# pedido com erro — contacta o supervisor.} other {# pedidos com erro — contacta o supervisor.}}",
|
|
"pending": "{count, plural, one {# pedido por enviar} other {# pedidos por enviar}}",
|
|
"synced": "Tudo sincronizado",
|
|
"requestFailed": "Pedido {id}… falhou — contacta o supervisor.",
|
|
"close": "Fechar"
|
|
},
|
|
"maintenance": {
|
|
"newTitle": "Novo pedido de manutenção",
|
|
"workstationLabel": "Posto",
|
|
"workstationRequired": "*",
|
|
"workstationLoading": "A carregar postos…",
|
|
"workstationPlaceholder": "Seleciona um posto…",
|
|
"photoLabel": "Foto (opcional)",
|
|
"photoPreview": "Pré-visualização",
|
|
"photoButton": "Tirar / escolher foto",
|
|
"descriptionLabel": "Descrição",
|
|
"descriptionRequired": "*",
|
|
"descriptionPlaceholder": "Descreve o problema…",
|
|
"photoError": "Não foi possível processar a foto. Tenta de novo.",
|
|
"saveError": "Erro ao guardar pedido. Tenta de novo.",
|
|
"submit": "Enviar pedido",
|
|
"submitting": "A guardar…",
|
|
"sentTitle": "Pedido enviado",
|
|
"pendingTitle": "Pedido em fila",
|
|
"sentMessage": "A equipa de manutenção foi notificada e irá tratar do problema.",
|
|
"pendingMessage": "Será enviado assim que a ligação for restabelecida.",
|
|
"backHome": "Voltar ao início"
|
|
}
|
|
}
|