package br.com.caelum.stella.validation.ie; import java.text.ParseException; import java.util.regex.Pattern; import javax.swing.text.MaskFormatter; import br.com.caelum.stella.DigitoGenerator; import br.com.caelum.stella.DigitoPara; import br.com.caelum.stella.MessageProducer; import br.com.caelum.stella.SimpleMessageProducer; /** * <p> * Documentação de referência: * </p> * <a href="http://www.pfe.fazenda.sp.gov.br/consist_ie.shtm">Secretaria da * Fazenda do Estado de São Paulo</a> <a * href="http://www.sintegra.gov.br/Cad_Estados/cad_AP.html">SINTEGRA - ROTEIRO * DE CRÍTICA DA INSCRIÇÃO ESTADUAL </a> * */ public class IEAmapaValidator extends AbstractIEValidator { public static final Pattern FORMATED = Pattern.compile("(03)[.](\\d{3})\\.?(\\d{3})[-](\\d{1})"); public static final Pattern UNFORMATED = Pattern.compile("(03)(\\d{3})(\\d{3})(\\d{1})"); /** * Este considera, por padrão, que as cadeias estão formatadas e utiliza um * {@linkplain SimpleMessageProducer} para geração de mensagens. */ public IEAmapaValidator() { super(true); } /** * O validador utiliza um {@linkplain SimpleMessageProducer} para geração de * mensagens. * * @param isFormatted * considerar cadeia formatada quando <code>true</code> */ public IEAmapaValidator(boolean isFormatted) { super(isFormatted); } public IEAmapaValidator(MessageProducer messageProducer, boolean isFormatted) { super(messageProducer, isFormatted); } @Override protected Pattern getUnformattedPattern() { return UNFORMATED; } @Override protected Pattern getFormattedPattern() { return FORMATED; } @Override protected boolean hasValidCheckDigits(String unformattedIE) { String iESemDigito = unformattedIE.substring(0, unformattedIE.length() - 1); String digito = unformattedIE.substring(unformattedIE.length() - 1); String digitoCalculado = calculaDigito(iESemDigito); return digito.equals(digitoCalculado); } private String calculaDigito(String iESemDigito) { int ie = Integer.parseInt(iESemDigito); /* * http://www.sintegra.gov.br/Cad_Estados/cad_AP.html * * De 03000001X a 03017000X => CASO 1 (p=5; d=0) * * De 03017001X a 03019022X => CASO 2 (p=9; d=1) * * De 03019023X em diante => CASO 3 (p=0; d=0) */ String p = "0"; String d = "0"; if ((3000001 <= ie) && (ie <= 3017000)) { p = "5"; } else if ((3017001 <= ie) && (ie <= 3019022)) { p = "9"; d = "1"; } return new DigitoPara(iESemDigito).addDigito(p).comMultiplicadoresDeAte(1, 9).complementarAoModulo() .trocandoPorSeEncontrar("0", 10).trocandoPorSeEncontrar(d, 11).calcula(); } private String formata(String valor) { try { final MaskFormatter formatador = new MaskFormatter("##.###.###-#"); formatador.setValidCharacters("1234567890"); formatador.setValueContainsLiteralCharacters(false); return formatador.valueToString(valor); } catch (ParseException e) { throw new RuntimeException("Valor gerado não bate com o padrão: " + valor, e); } } @Override public String generateRandomValid() { final String ieSemDigito = "03" + new DigitoGenerator().generate(6); final String ieComDigito = ieSemDigito + calculaDigito(ieSemDigito); if (isFormatted) { return formata(ieComDigito); } return ieComDigito; } }