# Como trabalhar em um cliente webapp traduzido O nosso cliente web app com base em React que alimenta nossa plataforma de aprendizado foi criado usando o Gatsby. Ele é traduzido em vários idiomas do mundo todo usando [react-i18next](https://react.i18next.com/) e [i18next](https://www.i18next.com/). Você pode aprender mais sobre configurar o cliente localmente para desenvolvimento seguindo [nosso guia de configuração aqui](how-to-setup-freecodecamp-locally.md). Por padrão, a aplicação está disponível somente em inglês. Assim que você tiver o projeto configurado localmente, você poderá seguir essa documentação para rodar o cliente no idioma de sua escolha a partir de uma lista de idiomas disponíveis. Isso pode ser de grande ajuda quando você estiver trabalhando em uma nova feature que envolva tradução, em especial, e que necessite que você valide, por exemplo, uma label de botão em um idioma diferente. > [!TIP] Você não precisa seguir esse documento para traduzir o currículo do freeCodeCamp ou contribuir com a documentação. Em vez disso, leia [esse guia aqui](how-to-translate-files.md). Vamos entender como o framework i18n e suas ferramentas funcionam. ## Estrutura de arquivos A maioria dos arquivos para tradução da plataforma ficam localizados na pasta [`client/i18n`](https://github.com/freeCodeCamp/freeCodeCamp/tree/main/client/i18n). Cada idioma tem uma pasta contendo arquivos JSON com as traduções. ```console config/i18n └── all-langs.js ... client/i18n ├── configForTests.js ├── config.js ├── locales │   ├── chinese │   │   ├── intro.json │   │   ├── links.json │   │   ├── meta-tags.json │   │   ├── motivation.json │   │   ├── translations.json │   │   └── trending.json ... ... │   ├── dothraki │   │   ├── intro.json │   │   ├── links.json │   │   ├── meta-tags.json │   │   ├── motivation.json │   │   ├── translations.json │   │   └── trending.json ... ... │   ├── english │   │   ├── intro.json │   │   ├── links.json │   │   ├── meta-tags.json │   │   ├── motivation.json │   │   ├── translations.json │   │   └── trending.json │   └── espanol │   ├── intro.json │   ├── links.json │   ├── meta-tags.json │   ├── motivation.json │   ├── translations.json │   └── trending.json ├── locales.test.js ├── schema-validation.js └── validate-keys.js ``` Alguns desses arquivos estão traduzidos na nossa plataforma de tradução (Crowdin), outros não. **Arquivos traduzidos na nossa plataforma de tradução:** - O arquivo `translations.json` contém a maioria do texto que aparece nos elementos de interface de usuário. As chaves são usadas na base de código para obter o texto correto para qualquer idioma definido. Este arquivo precisa ter exatamente as mesmas chaves em todos os idiomas. - O arquivo `intro.json` contém os pares chave-valor para o texto de introdução nas páginas de certificação. Se você deseja adicionar/atualizar traduções para as chaves, [leia este guia aqui](how-to-translate-files.md). **Arquivos NÃO traduzidos na nossa plataforma de tradução:** - Os arquivos `motivation.json` não precisam ter as mesmas citações, elogios ou comprimento de array. Apenas a mesma estrutura do JSON. - O arquivo `trending.json` contém os títulos e links para os artigos de notícias populares no rodapé do site. - O arquivo `meta-tags.json` contém as informações para a tag meta do nosso site. Mudanças nesses arquivos são tipicamente feitos pelo time da staff. Se você vir algo fora do ordinário, nós recomendamos que você nos contate na [sala de chat dos contribuidores](https://chat.freecodecamp.org/channel/contributors). ## Testando o cliente web em um idioma mundial Você pode testar o client app em qualquer linguagem disponível nessa [lista de idiomas aqui](https://github.com/freeCodeCamp/freeCodeCamp/blob/6b4a6a02568b809fc216ea8566ff5df446d1da4e/config/i18n/all-langs.js#L5). ```js const availableLangs = { client: ['english', 'espanol', 'chinese'], ... }; ``` Se você estiver testando um idioma novo, crie uma pasta com o nome do idioma como título próximo dos outros idiomas e copie os arquivos JSON de outro idioma para a sua pasta. Adicione o idioma para o array `client` como mostrado acima no arquivo [`config/i18n/all-langs.js`](https://github.com/freeCodeCamp/freeCodeCamp/blob/main/config/i18n/all-langs.js). A seguir, siga as instruções nos comentários dentro do mesmo arquivo para adicionar/atualizar o rest das variáveis se necessário. Finalmente, defina a variável `CLIENT_LOCALE` no seu arquivo `.env` para o local que você deseja fazer o build quando você estiver pronto(a). ## Como estruturar os componentes Se você estiver trabalhando em uma feature ou em um bug para o cliente web, por exemplo: adicionado novos itens de IU na pagina de configuração, você deve seguir as diretrizes abaixo. Elas vão te ajudar a preparar os componentes para tradução em todos os idiomas mundiais suportados. ### Componente funcional ```js import { useTranslation } from 'react-i18next'; // in the render method: const { t } = useTranslation(); // call the "t" function with a key from the JSON file:

{t('key')}

; // more details below ``` ### Componente de classe ```js import { withTranslation } from 'react-i18next'; // withTranslation adds the "t" function to props: const { t } = this.props; // call the "t" function with a key from the JSON file:

{t('key')}

// more details below // export without redux: export default withTranslation()(Component); // or with redux: export default connect(...)(withTranslation()(Component)); ``` ## Traduza usando a função "t" ### Tradução básica ```js // in the component:

{t('p1')}

// in the JSON file: { "p1": "My paragraph" } // output:

My paragraph

``` ### Com dados dinâmicos ```js // in the component: const username = 'moT';

{t('welcome', { username: username })}

// in the JSON file: { "welcome": "Welcome {{username}}" } // output:

Welcome moT

``` O exemplo acima passa um objeto para a função `t` com uma variável `username`. A variável será usada no valor JSON onde `{{username}}` está. ## Traduza com o componente `Trans` A regra geral é usar a função "t" quando puder. Mas existe o componente `Trans` para quando isso não for o suficiente, geralmente quando você tem elementos embutidos no texto. Você pode usar o componente `Trans` com qualquer tipo de componente React. ### Elementos básicos aninhados ```js // in the component: import { Trans } from 'react-i18next'

fcc.greeting

// in the JSON file: { "fcc": { "greeting": "Welcome to freeCodeCamp" } } // output:

Welcome to freeCodeCamp

``` Você pode colocar a chave dentro das tags do componente como no exemplo acima se o texto contém tags "simples" sem atributos. `br`, `strong`, `i` e `p` são padrões, mas essa lista pode ser expandida na configuração do i18n. ### Elementos complexos aninhados Em outras situações, você vai querer ter um texto específico dentro de outro elemento, uma tag "a" é um bom exemplo: ```js // in the component:

placeholder

// in the JSON file: { "check-forum": "Check out <0>our forum." } // output:

Check out our forum

``` No exemplo acima, a chave é definida nos atributos do componente `Trans`. O `<0>` e `` no JSON representam o primeiro filho do componente, neste caso, o elemento âncora. Se há mais filhos, eles apenas vão se somar ali usando a mesma sintaxe. Você pode encontrar os filhos de um componente na ferramenta de desenvolvedor React fazendo uma inspeção. `placeholder` está ali pois o linter estava reclamando sobre um elemento `` vazio. ### Com uma variável ```js // in the component: const email = 'team@freecodecamp.org';

{{ email }}

// in the JSON file: { "fcc": { "email": "Send us an email at: <0>{{email}}" } } // output:

Send us an email at: team@freecodecamp.org

``` No exemplo acima, a chave e a variável estão configuradas nos atributos do componente `Trans`. `{{ email }}` precisa estar em algum lugar no componente `Trans` também, não importa aonde. ## Mudando texto Para mudar o texto no lado do cliente, vá para o arquivo `.json`, ache a chave usada no componente React, e mude o valor para o novo texto que você deseja. Você deve pesquisar na base do código por aquela chave para ter certeza de que não está sendo usada em outro lugar. Ou, se está, se as mudanças fazem sentido em todos os lugares. ## Adicionando texto Se o texto que você deseja adicionar ao cliente existe no arquivo `.json`, use a chave existente. Caso contrário, crie uma nova chave. O arquivo em inglês é a "fonte da verdade" para todos os arquivos `.json` que compartilham o mesmo nome. Se você precisa adicionar uma nova chave, a adicione lá. Então, adicione a chave em **todos** os arquivos `translations.json`. > [!NOTE] Use texto em inglês para todos os idiomas se o arquivo está sendo traduzido via Crowdin. Os testes vão falhar se você não fizer isso. Seria bom manter as chaves na mesma ordem entre todos os arquivos também. Além disso, tente colocar toda a pontuação, espaçamento, citações e tudo mais nos arquivos JSON, não nos componentes ou arquivos de servidor. > [!NOTE] O underscore (`_`) é um caractere reservado para chaves dos arquivos que ficam do lado do cliente. Veja [a documentação](https://www.i18next.com/translation-function/plurals) para saber como são usados. ## Documentação útil - [documentação react-i18next](https://react.i18next.com/latest/usetranslation-hook) - [documentação i18next](https://www.i18next.com/translation-function/essentials)