DEV Community

Cover image for Máscara de Input para duração de tempo (horas acima de 24)
Vinicius Basilio
Vinicius Basilio

Posted on

Máscara de Input para duração de tempo (horas acima de 24)

Olá, pessoal! Vou escrever o post em português e traduzir para inglês. Há uns meses para um projeto do meu trabalho estava escrevendo uma calculadora de horas. Nossos projetos excedem a duração de vinte e quatro horas diversas vezes, então precisava de um input que pudesse passar.

Depois de muito tempo pesquisando achei umas coisas, mas nada pronto. Copiei (sim, não sei RegEx até hoje) uma RegEx por aí e ela me ajudou por um tempo, mas olhar aquela calculadora era saber que não tinha feito o meu trabalho completo. Enfim, hoje deu um estalo e fui remixar esse projeto e achei uma gambiarra. Segue o código para que você não sofra como eu!

interface props {
  hour: (n: any) => void;
}

import { useState } from "react";
const Input = ({ hour }: props) => {
  const [useArray, setArray] = useState([]);
  const [useAcc, setAcc] = useState(0);
  const [useValue, setValue] = useState("00:00:00");

  const mask = (value: string) => {
    const s = String(value);
    return s
      .replace(/\D/g, "")
      .replace(/(\d{2})(\d)/, "$1:$2")
      .replace(/(\d{2})(\d)/, "$1:$2")
      .replace(/(\d{2})(\d)/, "$1:$2");
  };

  const masking = (n: string) => {
    const r = mask(n).split(":");
    const pattern = ["00", "00", "00"];
    const correctedValue = pattern.map((n: number | string, i: number) =>
      r[i] != null ? r[i].padEnd(2, "0") : n
    );
    setValue(correctedValue.join(":"));
    hour(correctedValue.join(":"));
  };

  const maskingTest = (e: any) => {
    const arr: any = useArray;
    const expr = e.key;

    if (expr == "Backspace" && useAcc != 0) {
      arr.pop();
      setAcc(useAcc - 1);
      return masking(arr.join());
    }

    if (useAcc == 6 && expr != "Backspace") return;
    if (isNaN(expr)) return;
    arr[useAcc] = expr;
    setArray(arr);
    setAcc(useAcc + 1);
    return masking(arr.join());
  };

  return (
    <>
      <input
        className={`
          rounded-md
          p-2
          text-sm
          bg-neutral-700
          focus:outline-none
          w-40
          `}
        type="text"
        onKeyUp={(e) => maskingTest(e)}
        value={useValue}
      />
    </>
  );
};

export default Input;

Enter fullscreen mode Exit fullscreen mode

Não use onChange

O problema do onChange é que você não quer o que o input mude! Nós queremos criar uma sombra que vai pemitir rastrear o estado original do input e reescrever uma versão modificada pela nossa RegEx. O React acha isso uma péssima ideia, e o motivo é que ele tem dificuldade de rastrear as alterações na DOM com isso, por isso é uma gambiarra. Mas em meus teste nada deu errado, então até agora tudo bem.

Vamos usar um array para armazenar o texto real digitado pelo usuário. Cada tecla vai ser adicionada como um novo elemento, vamos verificar se a tecla é "Backspace" para apagar, se não for, vamos deixar passar. Agora vamos verificar se deixamos passar um número ou outra coisa. Ao mesmo tempo, temos que alimentar um acumulador, ele serve para nos dizer onde estamos no array, é com ele que vamos limitar o tamanho do input. Ah, caso você esteja cogitando em usar useRef, o React não recomenda leitura e escrita nela durante o render.

Depois vamos pegar o padrão que criamos de display de horas e vamos e conferir o texto que a RegEx exportou para nós, assim podemos substituir nosso padrão pelo texto que ela gerou, caso algo esteja com menos de dois dígitos, vamos corrigir com padEnd; caso o texto seja menor que o padrão vamos usar as partes que restarem dele para completar.

E pronto! Agora é só usar hour das props passadas e resgatar o valor usando um setState a partir do elemento pai.

Esse código ainda precisa de muitas melhorias. Fala aí nos comentários como posso fazer isso. A url do codeSandBox =>

Valeu!

Top comments (0)