-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathCNHValid.PRW
143 lines (106 loc) · 5.04 KB
/
CNHValid.PRW
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
#INCLUDE "PROTHEUS.CH"
//====================================================================================================================\
/*/{Protheus.doc}CNHValid
====================================================================================================================
@description
Validação do número da CNH
@author Thiago Mota
@author <mota.thiago@totvs.com.br>
@author <tgmspawn@gmail.com>
@version 1.0
@since 27/06/2018
@obs
Calcular validade do número da CNH, considerando a regra exposta na "RESOLUÇÃO Nº 192, DE 30 DE MARÇO DE 2006."
Fonte: http://www.denatran.gov.br/download/Resolucoes/resolucao_192_06.doc
Art. 2º. O documento de Habilitação terá 2 (dois) números de identificação nacional e 1 (um) número de
identificação estadual, que são:
I – o primeiro número de identificação nacional – Registro Nacional, será gerado pelo sistema informatizado da Base
Índice Nacional de Condutores – BINCO, composto de 9 (nove) caracteres mais 2 (dois) dígitos verificadores de
segurança, sendo único para cada condutor e o acompanhará durante toda a sua existência como condutor, não sendo
permitida a sua reutilização para outro condutor.
II – o segundo número de identificação nacional – Número do Espelho da CNH, será formado por 8 (oito) caracteres
mais 1 (um) dígito verificador de segurança, autorizado e controlado pelo órgão máximo executivo de trânsito da
União, e identificará cada espelho de CNH expedida.
a) O dígito verificador será calculado pela rotina denominada de “módulo 11” e sempre que o resto da divisão for
zero (0) ou um (1), o dígito verificador será zero (0);
III – o número de identificação estadual será o número do formulário RENACH, documento de coleta de dados do
candidato/condutor gerado a cada serviço, composto, obrigatoriamente, por 11 (onze) caracteres, sendo as duas
primeiras posições formadas pela sigla da Unidade de Federação expedidora, facultada a utilização da última posição
como dígito verificador de segurança.
a) O número do formulário RENACH identificará a Unidade da Federação onde o condutor foi habilitado ou realizou
alterações de dados no seu prontuário pela última vez.
b) O Formulário RENACH que dá origem às informações na BINCO e autorização para a impressão da CNH deverá ficar
arquivado em segurança, no órgão ou entidade executivo de trânsito do Estado ou do Distrito Federal.
/*/
//===================================================================================================================\
User Function CNHValid( cNumCNH )
Local lRet:= .F.
Local nY
Local cNumValid:= ""
Local cDig:= ""
Local xMod
Default cNumCNH:= &(ReadVar())
cNumCNH:= AllTrim(cNumCNH)
If Len(cNumCNH) == 11
cNumValid:= Left(cNumCNH, 9)
For nY:= 1 To 2
xMod:= Mod11(cNumValid + cDig,2,10)
If ValType(xMod) == "L" .And. ! xMod
Return .F.
Else
// Alguns casos tem dígito 1, em outros não batem o 1 é convertido para 0 conforme fiz a regulamentação
If xMod == "1" .And. Substr(cNumCNH, If(nY==1,10,11), 1) == "0"
xMod:= "0"
EndIf
cDig:= xMod + cDig // O dígito verificador da CNH é invertido
EndIf
Next nY
cNumValid+= cDig
If cNumCNH == cNumValid
lRet:= .T.
Else
HELP(' ',1,"CNH Invalido" ,,"Dígito verificador da CNH inválido",2,0,,,,,, "")
EndIf
Else
HELP(' ',1,"CNH Invalido" ,,"Número da CNH deve ter 11 caracteres",2,0,,,,,, "")
EndIf
Return ( lRet )
// FIM da Funcao CNHValid
//======================================================================================================================
//====================================================================================================================\
/*/{Protheus.doc}Mod11
====================================================================================================================
@description
Calcula o Módulo 11 de uma String
@author Thiago Mota
@author <mota.thiago@totvs.com.br>
@author <tgmspawn@gmail.com>
@version 1.0
@since 27/06/2018
@obs
Método:
DV módulo 11 corresponde ao resto da divisão por 11 do somatório da multiplicação de cada algarismo da base
respectivamente por 9, 8, 7, 6, 5, 4, 3, 2, 1 e 0, a partir da unidade. O resto 10 é considerado 0.
Seguindo a resolução, o resto 1 também deve ser considerado 0 (porém, existem casos de CNH válida com dígito 1).
/*/
//===================================================================================================================\
Static Function Mod11( cStr, nMultIni, nMultFim )
Local i, nModulo := 0, cChar, nMult
nMultIni := Iif( nMultIni==Nil,2,nMultIni )
nMultFim := Iif( nMultFim==Nil,9,nMultFim )
nMult := nMultIni
cStr := AllTrim(cStr)
For i := Len(cStr) to 1 Step -1
cChar := Substr(cStr,i,1)
If isAlpha( cChar )
Help(" ", 1, "ONLYNUM")
Return .f.
End
nModulo += Val(cChar)*nMult
nMult:= IIf(nMult==nMultfim,nMultIni,nMult+1)
Next
nRest := nModulo % 11
nRest := IIf(nRest==0 .or. nRest==1,0,11-nRest)
Return(Str(nRest,1))
// FIM da Funcao Mod11
//======================================================================================================================