Passed
Push — nullBounds ( b3e7e4...f3529c )
by no
03:35
created

DecimalParser::__construct()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 5
Code Lines 3

Duplication

Lines 0
Ratio 0 %

Importance

Changes 4
Bugs 0 Features 0
Metric Value
c 4
b 0
f 0
dl 0
loc 5
rs 9.4285
cc 2
eloc 3
nc 2
nop 2
1
<?php
2
3
namespace ValueParsers;
4
5
use DataValues\DecimalMath;
6
use DataValues\DecimalValue;
7
use DataValues\IllegalValueException;
8
9
/**
10
 * ValueParser that parses the string representation of a decimal number.
11
 *
12
 * @since 0.1
13
 *
14
 * @license GPL-2.0+
15
 * @author Daniel Kinzler
16
 */
17
class DecimalParser extends StringValueParser {
18
19
	const FORMAT_NAME = 'decimal';
20
21
	/**
22
	 * @var DecimalMath
23
	 */
24
	private $math;
25
26
	/**
27
	 * @var null|NumberUnlocalizer
28
	 */
29
	private $unlocalizer;
30
31
	/**
32
	 * @since 0.1
33
	 *
34
	 * @param ParserOptions|null $options
35
	 * @param NumberUnlocalizer|null $unlocalizer
36
	 */
37
	public function __construct( ParserOptions $options = null, NumberUnlocalizer $unlocalizer = null ) {
38
		parent::__construct( $options );
39
40
		$this->unlocalizer = $unlocalizer ?: new BasicNumberUnlocalizer();
41
	}
42
43
	/**
44
	 * @return DecimalMath
45
	 */
46
	private function getMath() {
47
		if ( $this->math === null ) {
48
			$this->math = new DecimalMath();
49
		}
50
51
		return $this->math;
52
	}
53
54
	/**
55
	 * Splits the exponent from the scientific notation of a decimal number.
56
	 *
57
	 * @since 0.5
58
	 *
59
	 * @example splitDecimalExponent( '1.2' )  is  array( '1.2', 0 )
60
	 * @example splitDecimalExponent( '1.2e3' )  is  array( '1.2', 3 )
61
	 * @example splitDecimalExponent( '1.2e-2' )  is  array( '1.2', -2 )
62
	 *
63
	 * @param string $valueString A decimal string, possibly using scientific notation.
64
	 *
65
	 * @return array list( $decimal, $exponent ) A pair of the decimal value without the
66
	 *         decimal exponent, and the decimal exponent as an integer. If $valueString
67
	 *         does not use scientific notation, $exponent will be 0.
68
	 */
69
	public function splitDecimalExponent( $valueString ) {
70
		if ( preg_match( '/^(.*)(?:[eE]|x10\^)([-+]?[\d,]+)$/', $valueString, $matches ) ) {
71
			$exponent = (int)str_replace( ',', '', $matches[2] );
72
			return array( $matches[1], $exponent );
73
		}
74
75
		return array( $valueString, 0 );
76
	}
77
78
	/**
79
	 * Applies a decimal exponent, by shifting the decimal point in the decimal string
80
	 * representation of the value.
81
	 *
82
	 * @since 0.5
83
	 *
84
	 * @example applyDecimalExponent( new DecimalValue( '1.2' ), 0 )  is  new DecimalValue( '1.2' )
85
	 * @example applyDecimalExponent( new DecimalValue( '1.2' ), 3 )  is  new DecimalValue( '1200' )
86
	 * @example applyDecimalExponent( new DecimalValue( '1.2' ), -2 )  is  new DecimalValue( '0.012' )
87
	 *
88
	 * @param DecimalValue $decimal
89
	 * @param int $exponent
90
	 *
91
	 * @return DecimalValue
92
	 */
93
	public function applyDecimalExponent( DecimalValue $decimal, $exponent ) {
94
		if ( $exponent !== 0 ) {
95
			$math = $this->getMath();
96
			$decimal = $math->shift( $decimal, $exponent );
97
		}
98
99
		return $decimal;
100
	}
101
102
	/**
103
	 * Creates a DecimalValue from a given string.
104
	 *
105
	 * The decimal notation for the value is based on ISO 31-0, with some modifications:
106
	 * - the decimal separator is '.' (period). Comma is not used anywhere.
107
	 * - leading and trailing as well as any internal whitespace is ignored
108
	 * - the following characters are ignored: comma (","), apostrophe ("'").
109
	 * - scientific (exponential) notation is supported using the pattern /e[-+]\d+/
110
	 * - the number may start (or end) with a decimal point.
111
	 * - leading zeroes are stripped, except directly before the decimal point
112
	 * - trailing zeroes are stripped, except directly after the decimal point
113
	 * - zero is always positive.
114
	 *
115
	 * @see StringValueParser::stringParse
116
	 *
117
	 * @since 0.1
118
	 *
119
	 * @param string $value
120
	 *
121
	 * @return DecimalValue
122
	 * @throws ParseException
123
	 */
124
	protected function stringParse( $value ) {
125
		$rawValue = $value;
126
127
		$value = $this->unlocalizer->unlocalizeNumber( $value );
128
129
		//handle scientific notation
130
		list( $value, $exponent ) = $this->splitDecimalExponent( $value );
131
132
		$value = $this->normalizeDecimal( $value );
133
134
		if ( $value === '' ) {
135
			throw new ParseException( 'Decimal value must not be empty', $rawValue, self::FORMAT_NAME );
136
		}
137
138
		try {
139
			$decimal = new DecimalValue( $value );
140
			$decimal = $this->applyDecimalExponent( $decimal, $exponent );
141
142
			return $decimal;
143
		} catch ( IllegalValueException $ex ) {
144
			throw new ParseException( $ex->getMessage(), $rawValue, self::FORMAT_NAME );
145
		}
146
	}
147
148
	/**
149
	 * Normalize a decimal string.
150
	 *
151
	 * @param string $number
152
	 *
153
	 * @return string
154
	 */
155
	private function normalizeDecimal( $number ) {
156
		// strip fluff
157
		$number = preg_replace( '/[ \r\n\t\'_,`]+/u', '', $number );
158
159
		// strip leading zeros
160
		$number = preg_replace( '/^([-+]?)0+([^0]|0$)/', '$1$2', $number );
161
162
		// fix leading decimal point
163
		$number = preg_replace( '/^([-+]?)\./', '${1}0.', $number );
164
165
		// strip trailing decimal point
166
		$number = preg_replace( '/\.$/', '', $number );
167
168
		// add leading sign
169
		$number = preg_replace( '/^(?=\d)/', '+', $number );
170
171
		// make "negative" zero positive
172
		$number = preg_replace( '/^-(0+(\.0+)?)$/', '+$1', $number );
173
174
		return $number;
175
	}
176
177
}
178