Passed
Push — master ( 3d1adc...86db89 )
by João
02:21
created

calcvICMSDif()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 5
Code Lines 2

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 3
CRAP Score 1

Importance

Changes 0
Metric Value
cc 1
eloc 2
nc 1
nop 1
dl 0
loc 5
ccs 3
cts 3
cp 1
crap 1
rs 10
c 0
b 0
f 0
1
<?php
2
3
declare(strict_types=1);
4
5
namespace Gbbs\NfeCalculos;
6
7
use Exception;
8
use Gbbs\NfeCalculos\Exception\InvalidCSTException;
9
use Gbbs\NfeCalculos\Exception\NotImplementedCSTException;
10
11
class ICMS
12
{
13
    public $orig;  // Origem da mercadoria
14
    public $CST;  // Tributação do ICMS
15
    public $modBC;  // Modalidade de determinação da BC do ICMS
16
    public $pRedBC;  // Percentual da Redução de BC
17
    public $vBC;  // Valor da BC do ICMS
18
    public $pICMS;  // Alíquota do imposto
19
    public $vICMS;  // Valor do ICMS
20
    public $modBCST;  // Modalidade de determinação da BC do ICMS ST
21
    public $pMVAST;  // Percentual da margem de valor Adicionado do ICMS ST
22
    public $pRedBCST;  // Percentual da Redução de BC do ICMS ST
23
    public $vBCST;  // Valor da BC do ICMS ST
24
    public $pICMSST;  // Alíquota do imposto do ICMS ST
25
    public $vICMSST;  // Valor do ICMS ST
26
    public $UFST;  // UF para qual é devido o ICMS ST
27
    public $pBCop;  // Percentual da BC operação própria
28
    public $vBCSTRet;  // Valor da BC do ICMS Retido Anteriormente
29
    public $vICMSSTRet;  // Valor do ICMS Retido Anteriormente
30
    public $motDesICMS;  // Motivo da desoneração do ICMS
31
    public $vBCSTDest;  // Valor da BC do ICMS ST da UF destino
32
    public $vICMSSTDest;  // Valor do ICMS ST da UF destino
33
    public $pCredSN;  // Alíquota aplicável de cálculo do crédito (Simples Nacional)
34
    public $vCredICMSSN;  // Valor crédito do ICMS que pode ser aproveitado nos termos do art. 23 da LC 123 (SIMPLES NACIONAL)
35
    public $vICMSDeson;  // Valor do ICMS da desoneração
36
    public $vICMSOp;  // Valor do ICMS da Operação
37
    public $pDif;  // percentual do diferimento
38
    public $vICMSDif;  // Valor do ICMS Diferido
39
    public $vBCFCP;  // Valor da Base de Cálculo do FCP
40
    public $pFCP;  // Percentual do FCP
41
    public $vFCP;  // Valor do FCP
42
    public $vBCFCPST;  // Valor da Base de Cálculo do FCP retido por Substituição Tributária
43
    public $pFCPST;  // Percentual do FCP retido por Substituição Tributária.
44
    public $vFCPST;  // Valor do FCP retido por Substituição Tributária
45
    public $vBCFCPSTRet;  // Valor da BC do FCP retido anteriormente por Substituição Tributária
46
    public $pFCPSTRet;  // Alíquota do FCP retido anteriormente por Substituição Tributária
47
    public $vFCPSTRet;  // Valor do FCP retido anteriormente por Substituição Tributária
48
    public $pST;  // Alíquota suportada pelo Consumidor Final
49
}
50
51
/**
52
 * @param ICMS $ICMS
53
 * @param string $ufOrigem
54
 * @param string $ufDestino
55
 * @param float $reducao
56
 * @return ICMS
57
 * @throws NotImplementedCSTException|InvalidCSTException|Exception
58
 */
59 1
function calcularICMS(ICMS $ICMS, string $ufOrigem, string $ufDestino, float $reducao = null): ICMS
60
{
61
    $notImplemented = [
62 14
        '20', '30', '40', '50', '60', '70', '101', '103',
63
        '200', '201', '202', '203', '300', '400', '500', '900'
64
    ];
65 14
    if (in_array($ICMS->CST, $notImplemented, true)) {
66 1
        throw new NotImplementedCSTException($ICMS->CST);
67
    }
68 13
    if ($reducao === null) {
69 12
        $ICMS->pICMS = pICMSFromUFs($ufOrigem, $ufDestino);
70 12
        $ICMS->pICMSST = pICMSSTFromUFs($ufOrigem, $ufDestino);
71
    } else {
72 1
        $ICMS->pICMS = $reducao;
73 1
        $ICMS->pICMSST = $reducao;
74
    }
75
    $calculosCST = [
76 13
        '00' => 'Gbbs\NfeCalculos\calcCST00',
77
        '10' => 'Gbbs\NfeCalculos\calcCST10',
78
        '41' => 'Gbbs\NfeCalculos\calcCST41',
79
        '51' => 'Gbbs\NfeCalculos\calcCST51',
80
        '90' => 'Gbbs\NfeCalculos\calcCST90',
81
        '102' => 'Gbbs\NfeCalculos\calcCSOSN102',
82
    ];
83 13
    if (array_key_exists($ICMS->CST, $calculosCST)) {
84 12
        return $calculosCST[$ICMS->CST]($ICMS);
85
    }
86 1
    throw new InvalidCSTException($ICMS->CST);
87
}
88
89
/**
90
 * Subtrai do valor da BC do ICMS ST o percentual da redução de BC do ICMS ST
91
 * @param $ICMS
92
 * @return float
93
 */
94 1
function calcularReducaoValorBCST(ICMS $ICMS): float
95
{
96 2
    return $ICMS->vBCST * (1 - $ICMS->pRedBCST / 100);
97
}
98
99
/**
100
 * Calcula o Valor do ICMS
101
 * @param $ICMS
102
 * @return float
103
 */
104 1
function calcvICMS(ICMS $ICMS): float
105
{
106 6
    return round($ICMS->vBC * $ICMS->pICMS / 100, 2);
107
}
108
109
110
/**
111
 * Calcula o valor do ICMS Diferido
112
 * @param ICMS $ICMS
113
 * @return float
114
 */
115 1
function calcvICMSDif(ICMS $ICMS): float
116
{
117 1
    $vICMS = calcvICMS($ICMS);
118
119 1
    return $vICMS - $ICMS->vBC * ($ICMS->pICMS - ($ICMS->pICMS * ($ICMS->pDif / 100))) / 100;
120
}
121
122
/**
123
 * @param $ICMS
124
 * @return ICMS
125
 * @throws Exception
126
 */
127 1
function calcCST00(ICMS $ICMS): ICMS
128
{
129 3
    if ($ICMS->modBC !== 0) {
130 1
        throw new Exception('modBC ' . $ICMS->modBC . ' not implemented');
131
    }
132 2
    $calculado = new ICMS();
133 2
    $calculado->orig = $ICMS->orig;
134 2
    $calculado->CST = $ICMS->CST;
135 2
    $calculado->modBC = $ICMS->modBC;
136 2
    $calculado->vBC = $ICMS->vBC;
137 2
    $calculado->pICMS = $ICMS->pICMS;
138 2
    $calculado->vICMS = calcvICMS($ICMS);
139
140 2
    return $calculado;
141
}
142
143
/**
144
 * @param $ICMS
145
 * @return ICMS
146
 * @throws Exception
147
 */
148 1
function calcCST10(ICMS $ICMS): ICMS
149
{
150 3
    if ($ICMS->modBCST !== 4) {
151 1
        throw new Exception('modBCST ' . $ICMS->modBCST . ' not implemented');
152
    }
153 2
    $calculado = new ICMS();
154 2
    $calculado->orig = $ICMS->orig;
155 2
    $calculado->CST = $ICMS->CST;
156 2
    $calculado->modBC = $ICMS->modBC;
157 2
    $calculado->vBC = $ICMS->vBC;
158 2
    $calculado->pICMS = $ICMS->pICMS;
159 2
    $calculado->modBCST = $ICMS->modBCST;
160 2
    $calculado->pMVAST = $ICMS->pMVAST;
161 2
    $calculado->pRedBCST = $ICMS->pRedBCST;
162 2
    $calculado->pICMSST = $ICMS->pICMSST;
163 2
    $calculado->vICMS = calcvICMS($ICMS);
164 2
    $calculado->vBCST = calcularReducaoValorBCST($ICMS) * (1 + $ICMS->pMVAST / 100);
165 2
    $calculado->vICMSST = $ICMS->pMVAST === 0.0
166 1
        ? 0.0
167 1
        : round(($calculado->vBCST * (1 - $ICMS->pRedBCST / 100)) * $ICMS->pICMSST / 100 - $calculado->vICMS, 2);
168
169 2
    return $calculado;
170
}
171
172
/**
173
 * @param $ICMS
174
 * @return ICMS
175
 * @throws Exception
176
 */
177 1
function calcCST41(ICMS $ICMS): ICMS
178
{
179 2
    $calculado = new ICMS();
180 2
    $calculado->orig = $ICMS->orig;
181 2
    $calculado->CST = $ICMS->CST;
182 2
    $calculado->vBC = 0.0;
183 2
    $calculado->vICMS = 0.0;
184 2
    $calculado->pICMS = 0.0;
185 2
    $calculado->vICMSDeson = $ICMS->vBC * ($ICMS->pICMS / 100);
186 2
    $calculado->motDesICMS = $ICMS->motDesICMS;
187
188 2
    return $calculado;
189
}
190
191
/**
192
 * @param $ICMS
193
 * @return ICMS
194
 * @throws Exception
195
 */
196 1
function calcCST51(ICMS $ICMS): ICMS
197
{
198 2
    if ($ICMS->modBC !== 0) {
199 1
        throw new Exception('modBC ' . $ICMS->modBC . ' not implemented');
200
    }
201 1
    $calculado = new ICMS();
202 1
    $calculado->orig = '0';
203 1
    $calculado->CST = $ICMS->CST;
204 1
    $calculado->modBC = $ICMS->modBC;
205 1
    $calculado->vBC = $ICMS->vBC;
206 1
    $calculado->pICMS = $ICMS->pICMS - ($ICMS->pICMS * $ICMS->pDif) / 100;
207 1
    $calculado->vICMS = calcvICMS($calculado);
208 1
    $calculado->pDif = $ICMS->pDif;
209 1
    $calculado->vICMSDif = calcvICMSDif($ICMS);
210
211 1
    return $calculado;
212
}
213
214
/**
215
 * @param $ICMS
216
 * @return ICMS
217
 * @throws Exception
218
 */
219 1
function calcCST90(ICMS $ICMS): ICMS
220
{
221 1
    $calculado = new ICMS();
222 1
    $calculado->orig = $ICMS->orig;
223 1
    $calculado->CST = $ICMS->CST;
224 1
    $calculado->modBC = $ICMS->modBC;
225 1
    $calculado->vBC = $ICMS->vBC;
226 1
    $calculado->pICMS = $ICMS->pICMS;
227 1
    $calculado->vICMS = calcvICMS($ICMS);
228
229 1
    return $calculado;
230
}
231
232
/**
233
 * @param $ICMS
234
 * @return ICMS
235
 */
236 1
function calcCSOSN102(ICMS $ICMS): ICMS
237
{
238 1
    $calculado = new ICMS();
239 1
    $calculado->orig = $ICMS->orig;
240 1
    $calculado->CST = $ICMS->CST;
241 1
    return $calculado;
242
}
243
244
/**
245
 * @param string $ufOrigem
246
 * @param string $ufDestino
247
 * @throws Exception
248
 * @return float
249
 */
250 1
function pICMSFromUFs(string $ufOrigem, string $ufDestino): float
251
{
252 13
    $path = realpath(__DIR__ . '/../storage') . '/';
253 13
    $picmsFile = file_get_contents($path . 'picms.json');
254 13
    $picmsList = json_decode($picmsFile, true);
255 13
    if ($ufDestino === '99') {
256 1
        return 0.0;
257
    }
258 12
    foreach ($picmsList as $picms) {
259 12
        if ($picms['uf'] === $ufOrigem) {
260 11
            return (float) $picms['uf' . $ufDestino];
261
        }
262
    }
263 1
    throw new Exception('UF inexistente: ' . $ufOrigem . ' - ' . $ufDestino);
264
}
265
266
/**
267
 * @param string $ufOrigem
268
 * @param string $ufDestino
269
 * @throws Exception
270
 * @return float
271
 */
272 1
function pICMSSTFromUFs(string $ufOrigem, string $ufDestino): float
273
{
274 13
    $path = realpath(__DIR__ . '/../storage') . '/';
275 13
    $picmsstFile = file_get_contents($path . 'picmsst.json');
276 13
    $picmsstList = json_decode($picmsstFile, true);
277 13
    if ($ufDestino === '99') {
278 1
        return 0.0;
279
    }
280 12
    foreach ($picmsstList as $picmsst) {
281 12
        if ($picmsst['uf'] === $ufOrigem) {
282 11
            return (float) $picmsst['uf' . $ufDestino];
283
        }
284
    }
285 1
    throw new Exception('UF inexistente: ' . $ufOrigem . ' - ' . $ufDestino);
286
}
287