Test Failed
Pull Request — master (#132)
by Jordan
18:05
created

BaseConversionProvider::_toBase()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 12
Code Lines 7

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 10
CRAP Score 2

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 2
eloc 7
nc 2
nop 2
dl 0
loc 12
ccs 10
cts 10
cp 1
crap 2
rs 10
c 1
b 0
f 0
1
<?php
2
3
namespace Samsara\Fermat\Provider;
4
5
use Samsara\Fermat\Enums\NumberBase;
6
use Samsara\Fermat\Numbers;
7
use Samsara\Fermat\Types\Base\Interfaces\Numbers\DecimalInterface;
8
use Samsara\Fermat\Values\ImmutableDecimal;
9
10
/**
11
 *
12
 */
13
class BaseConversionProvider
14
{
15
16
    private static array $chars = [
17
        '0',
18
        '1',
19
        '2',
20
        '3',
21
        '4',
22
        '5',
23
        '6',
24
        '7',
25
        '8',
26
        '9',
27
        'A',
28
        'B',
29
        'C',
30
        'D',
31
        'E',
32
        'F'
33
    ];
34
35
    /**
36
     * @param DecimalInterface $number
37
     * @param NumberBase|null $toBase
38
     * @return string
39 3
     */
40
    public static function convertFromBaseTen(DecimalInterface $number, ?NumberBase $toBase = null): string
41 3
    {
42
        $base = $toBase ?? $number->getBase();
43 3
44
        return self::_toBase($number, $base->value);
45
    }
46
47
    /**
48
     * @param string $number
49
     * @param NumberBase $fromBase
50
     * @return string
51 2
     */
52
    public static function convertStringToBaseTen(string $number, NumberBase $fromBase): string
53 2
    {
54
        if (str_contains($number, '.')) {
55
            $sign = str_starts_with($number, '-') ? '-' : '';
56
            [$intPart, $decPart] = explode('.', $number);
57
            $intPart = self::_fromBase($intPart, $fromBase->value);
58
            $decPart = strrev(self::_fromBase(strrev($decPart), $fromBase->value));
59
60
            return $sign.$intPart.'.'.$decPart;
61 2
        } else {
62 2
            $sign = str_starts_with($number, '-') ? '-' : '';
63
            return $sign.self::_fromBase($number, $fromBase->value);
64
        }
65
    }
66 3
67
    private static function _toBase(DecimalInterface $input, int $base): string
68 3
    {
69 3
        $baseNum = Numbers::make(Numbers::IMMUTABLE, $base, $input->getScale());
70 3
        $inputInt = Numbers::make(Numbers::IMMUTABLE, $input->getWholePart());
71 3
        $inputDec = Numbers::make(Numbers::IMMUTABLE, strrev($input->getDecimalPart()));
72 3
73 3
        $intPart = self::_toBasePart($baseNum, $inputInt);
74
        $decPart = self::_toBasePart($baseNum, $inputDec);
75 3
76 3
        $sign = $input->isNegative() ? '-' : '';
77 3
78 3
        return $sign.$intPart.'.'.strrev($decPart);
79 3
    }
80 3
81 3
    private static function _fromBase(string $number, int $base): string
82 3
    {
83 3
        if (str_starts_with($number, '-')) {
84 3
            $number = trim($number, '-');
85 3
        }
86 3
87 3
        $output = Numbers::makeZero();
88
        $input = str_split($number);
89
        $input = array_reverse($input);
0 ignored issues
show
Bug introduced by
It seems like $input can also be of type true; however, parameter $array of array_reverse() does only seem to accept array, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

89
        $input = array_reverse(/** @scrutinizer ignore-type */ $input);
Loading history...
90
        $pos = 0;
91 3
        $base = Numbers::make(Numbers::IMMUTABLE, $base);
92
93
        foreach ($input as $char) {
94
            $output = $output->add($base->pow($pos)->multiply(array_search($char, self::$chars)));
0 ignored issues
show
Bug introduced by
The method pow() does not exist on integer. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

94
            $output = $output->add($base->/** @scrutinizer ignore-call */ pow($pos)->multiply(array_search($char, self::$chars)));

This check looks for calls to methods that do not seem to exist on a given type. It looks for the method on the type itself as well as in inherited classes or implemented interfaces.

This is most likely a typographical error or the method has been renamed.

Loading history...
95
            $pos++;
96
        }
97
98
        return $output->getAsBaseTenRealNumber();
99
    }
100
101
    public static function _toBasePart(ImmutableDecimal $baseNum, ImmutableDecimal $startVal): string
102
    {
103
        if ($startVal->isGreaterThan(0)) {
104 3
            $stringVal = '';
105
            $runningTotal = Numbers::make(Numbers::IMMUTABLE, $startVal->getAsBaseTenRealNumber());
106 3
            while ($runningTotal->isGreaterThan(0)) {
107
                $current = gmp_div_qr($runningTotal->getAsBaseTenRealNumber(), $baseNum->getAsBaseTenRealNumber());
108
                $mod = (int)$current[1];
109 2
                $stringVal = self::$chars[$mod] . $stringVal;
110
                $runningTotal = Numbers::make(Numbers::IMMUTABLE, $current[0]);
111 2
            }
112 1
        } else {
113
            $stringVal = '0';
114
        }
115 2
116 2
        return $stringVal;
117 2
    }
118
119
}