Criando esquemas de validação com Yup

Criando esquemas de validação com Yup
Luan Alves
Luan Alves

Compartilhe

Imagine que você está desenvolvendo um aplicativo web e se depara com a necessidade de validar campos de formulários.

Bom, lidar com esse tipo de validação é algo extremamente comum em diversas páginas na web.

Quem nunca teve que preencher um formulário para se cadastrar em uma rede social ou realizar uma compra online?

E certamente já aconteceu de você preencher algum campo de forma incorreta. Acertei? É, acontece com todo mundo.

Vou te mostrar um exemplo prático disso. Vamos lá?

Formulário de validação com Yup

Uma das melhores formas de entender um assunto é através de sua aplicação prática. Então, vamos pensar na seguinte situação:

Imagem animada de gif contendo página de desconto para passagens de ônibus, com o título: “ClickBônus” e abaixo o texto “Preencha o formulário e participe do nosso Clube de Vantagens!”, ao lado esquerdo. No lado direito um formulário com os campos “Nome”, “Email”, “Telefone” e “Cep” e um botão escrito “Quero participar”. No gif é escrita a palavra “teste” no campo de “E-mail”, então após o botão “Quero participar” ser clicado aparece o aviso “Inclua um "@" no endereço de email. Falta um "@" em "teste".

No exemplo da imagem, as pessoas estão preenchendo o campo de e-mail de forma incorreta. Assim, ao clicar no botão que envia os dados do formulário, surge o aviso indicado a forma correta de preencher aquele campo.

Essa regra de validação é aplicada através do input com valor “email” no atributo type.

Existem mais atributos HTML que também orientam na validação de campos de formulário como, por exemplo:

  • O atributo required pode ser utilizado a quase qualquer input para exigir que o usuário preencha um campo antes de enviar o formulário.

  • O atributo type especifica o tipo de dado esperado em um input, como email, number, url, entre outros, você também pode ler o artigo “Recebendo dados de usuário na Web com inputs”, escrito por mim, para conhecer mais sobre essa tag.

  • Os atributos minlength e maxlength permitem validar a quantidade mínima e máxima de caracteres que é possível preencher em um campo, podendo ser utilizados nos inputs de tipo text, search, url, tel, email, and password.

  • Os atributos min e max definem os valores mínimos e máximos para inputs que recebem valores numéricos como number, date e outros.

  • O atributo pattern é uma poderosa ferramenta para inserir expressões regulares, conhecidas também como “regex” ou “regexp”. As expressões regulares permitem definir padrões de caracteres de forma personalizada. Como por exemplo, você pode conferir no artigo “Formatando CPF com ajuda das Expressões Regulares”, escrito pelo instrutor Felipe Nascimento.

Para validar campos de formulários você também pode utilizar mais atributos HTML em conjunto com JavaScript e React.

Entretanto, esse processo pode se tornar complexo. Afinal de contas, as aplicações em React costumam ter uma grande quantidade de troca de dados.

Isso pode acontecer através de solicitações de API, envios de formulários ou objetos personalizados para lidar com os estados.

Banner promocional da Alura, com um design futurista em tons de azul, apresentando o texto

Como tornar a validação um processo mais simples e padronizado?

Para tornar a implementação e manutenção de validação um processo mais simples, podemos utilizar o Yup.

O Yup é, de forma geral, um construtor de schemas(esquemas) para análise e validação de valores em campos.

O conceito de esquema se refere a definir as regras que campos de formulários devem seguir e centralizá-las em um único lugar.

Por exemplo, num esquema você pode definir as regras sobre a quantidade máxima de caracteres em cada campo, se é possível escrever letras e números ou apenas números, se é obrigatório ou opcional preencher um campo.

E para cada regra, você define como será a mensagem de alerta orientando a pessoa usuária no preenchimento correto.

Agora, que você já sabe o que é e para quê serve, vamos entender como você pode implementar o Yup no seu projeto.

Como integrar o Yup ao seu projeto

O comando abaixo adiciona o yup como dependência do seu projeto

npm install yup
  1. Importando o Yup no componente de formulário com o comando:
import * as Yup from 'yup';
  1. Crie uma variável ou um hook personalizado, para inserir os esquemas de validação, como por exemplo:
export const useValidation = () => { }
  1. Defina a estrutura do esquema:

A estrutura de um esquema começa pelo método object() que define um objeto de esquema. Cada vez que o método é acionado, é criada uma instância do objeto de esquema e moldado pelo método shape(), que funciona como uma atribuição do esquema ao objeto, no seu interior é onde você define o conjunto de propriedade + métodos para validação.

export const useValidation = () => {
    return yup.object().shape({})
}

Comece validando o primeiro campo do formulário, neste exemplo será o campo de “Nome”


export const useValidation = () => {
    return yup.object().shape({
        nome: yup.string(“No campo nome não é permitido inserir números.”).required('Por favor, digite o seu nome!'),
    })
}

O método string() valida se o valor enviado na propriedade nome é do tipo string.

O método required() valida se o valor enviado não é nulo ou vazio, tornando o campo com preenchimento obrigatório.

No parâmetro do método, você pode inserir uma mensagem orientando o usuário no preenchimento correto. Caso não seja inserida nenhuma mensagem, o Yup já possui mensagens de forma padrão para a maioria dos métodos, geralmente em inglês.

Também é possível utilizar outros métodos do Yup para definir regras de validação, separei alguns para você conferir:

  • number(): Valida se os caracteres inseridos são números.
  • date(): Valida se o valor é uma url válida.
  • url(): Valida se o valor é uma URL válida.
  • `max(n)``: Valida se o valor é menor ou igual a n.
  • min(n): Valida se o valor é maior ou igual a n.
  • oneOf([array]): Valida se o valor está presente no array.
  • positive(): O valor deve ser um número positivo.
  • integer(): Valida se o número é um número inteiro.

Existe uma grande quantidade de métodos que o Yup oferece, você pode conferir todos eles na documentação.

  1. Defina os métodos e regras para todos os campos do formulário:

export const useValidation = () => {
    return yup.object().shape({
        nome: yup.string().required('Por favor, digite o seu nome!'),
        email: yup.string().email('Por favor, digite um e-mail válido!').required('Por favor digite seu e-mail'),
        telefone: yup.string().required('Por favor, informe o seu telefone'),
        cep: yup.string().required('Por favor, informe o seu código postal (CEP)'),
    })
}

O Yup usa o padrão de projeto builder(builder design pattern), ele é frequentemente utilizado em situações em que é preciso construir objetos complexos com muitas opções e configurações diferentes, separando a construção de um objeto de sua representação.

Por exemplo, temos o objeto principal que é const Formulario = () => {}, nele, são inseridos os componentes do formulário e outros objetos que vão interagir entre si:

Objeto com o estado inicial do formulário:


    const [formData, setFormData] = useState({
        nome: '',
        email: '',
        telefone: '',
        cep: '',
    })

Objeto com estado de erros:


    const [erros, setErros] = useState({
        nome: '',
        email: '',
        telefone: '',
        cep: '',
    })

Agora, podemos integrar o esquema de validação ao formulário. Para isso, é necessário aplicar a lógica de validação.

  1. Crie uma função assíncrona para tratar a submissão do formulário:
    const yupSchema = useValidation()

    const submeterFormulario = async (event) => {
        event.preventDefault()

        try {
            await yupSchema.validate(formData)
        } catch (error) {
            console.error(error)
        }

        console.log("Validando dados: ", formData)
    }

Com o Yup é possível realizar validações síncronas ou assíncronas, porém, a validação síncrona só funciona se não houver testes assíncronos configurados. Neste caso, estamos utilizando a segunda opção como exemplo.

O bloco try encapsula a chamada à função yupSchema.validate(), e no seu parâmetro é passado o objeto no qual será aplicado o esquema de validação formData.

Se a validação for bem-sucedida, o código dentro do bloco catch não é executado. Se a validação falhar, o bloco catch captura o erro e verifica se é um erro de validação Yup.

Se for um erro de validação Yup, o código percorre cada erro e atualiza o estado dos erros com as mensagens específicas.

Agora é possível submeter o formulário. Teste enviar os dados sem preencher nenhum campo para a validação do Yup acontecer:

A imagem mostra uma tela de erro em uma aplicação web, especificamente uma mensagem de ValidationError que indica que o campo "CEP" (código postal) precisa ser informado ou corrigido. Isso acontece no contexto de um formulário online denominado "ClickBônus", onde os usuários são convidados a se inscreverem para participar de um clube de vantagens. O formulário requer que o usuário preencha campos como nome, e-mail, telefone e CEP. Ao lado, há uma ilustração de uma personagem feminina com presentes, o que sugere benefícios ou recompensas.

O console informa o objeto formData com os campos vazios, além disso, também informa “ValidationError: Por favor, informe o seu código postal (CEP)”.

Nesse caso, a validação não ocorreu como esperado, pois logo na primeira validação(cep), foi retornado o erro e parou, sem validar os outros campos que também estavam vazios. E agora, como resolver?

Como validar erros aninhados

Para realizar a validação de vários campos ao mesmo tempo, você pode utilizar o ValidationError, que foi lançado no console no último exemplo.

Ele é um objeto que é gerado toda vez que a validação não ocorre como esperado, e contém algumas propriedades, como, por exemplo:

  • value: Representa o valor do campo que foi validado.
  • path: Representa uma string, indicando onde o erro foi gerado.
  • errors: Representa um array de mensagens de erros.
  • inner: Representa um array de ValidationErrors.

A propriedade inner pode ser utilizada em conjunto com abortEarly: Um booleano que, caso tenha o valor false irá retornar todos os erros, sem interromper no primeiro. Caso tenha o valor true o abortEarly vai retornar apenas o primeiro erro de validação encontrado.

Por exemplo:


    const yupSchema = useValidation()

    const submeterFormulario = async (event) => {
        event.preventDefault();

        setSucesso('')
        setErros({
            nome: '',
            email: '',
            telefone: '',
            cep: '',
        })
        try {
            await yupSchema.validate(formData, {
                abortEarly: false
            })
            setSucesso('Cadastro enviado!')
        } catch (error) {

            if (error instanceof yup.ValidationError) {

                error.inner.forEach(validationFailed => {

                    setErros(oldState => {
                        return {
                            ...oldState,
                            [validationFailed.path]: validationFailed.errors
                        }
                    })

                })

            } else {
                // Se necessário, você pode tratar outros tipos de erro
                console.error("Erro inesperado:", error);
            }
        }
    }

E, então, é possível testar e validar o formulário. Esse é um passo a passo focado para validar este formulário.

No entanto, é bom lembrar que os métodos para aplicar validação mudam sempre de acordo com cada projeto e também podem ter abordagens diferentes para projetos similares.

Por esse motivo, recomendo também utilizar a documentação oficial do Yup como base para conhecer outras formas de uso dessa ferramenta fantástica.

Você também pode realizar o download do projeto completo utilizado como exemplo neste artigo ou conferir o seu código no repositório.

E agora é sua vez de praticar personalizando o formulário com outras formas de validação para fixar o conhecimento.

Conclusão

O tópico de formulários no front-end é bem extenso e tem muitos recursos bacanas para serem utilizados.

Você pode aprender mais na prática realizando a nova Formação Formulários React.

Ainda sobre o assunto de formulários, caso queira conhecer sobre o tema de máscaras, que é um recurso muito útil e interessante para você aplicar nos seus projetos, recomendo a leitura do artigo “Criando uma máscara de Telefone com Javascript”, escrito pelo instrutor Felipe Nascimento.

Bons estudos e boa leitura!

Luan Alves
Luan Alves

Sou Luan Alves, estudante de Análise e Desenvolvimento de Sistemas, instrutor de Desenvolvimento Front-End no Grupo Alura. Estou aqui para ajudar a tirar suas dúvidas, aprender e compartilhar conhecimento. :)

Veja outros artigos sobre Front-end