<?php namespace CommerceGuys\Intl\Tests\Formatter; use CommerceGuys\Intl\Currency\Currency; use CommerceGuys\Intl\Formatter\NumberFormatter; use CommerceGuys\Intl\NumberFormat\NumberFormat; /** * @coversDefaultClass \CommerceGuys\Intl\Formatter\NumberFormatter */ class NumberFormatterTest extends \PHPUnit_Framework_TestCase { /** * Prepare two number formats. */ protected $numberFormats = array( 'latn' => array( 'numbering_system' => 'latn', 'decimal_pattern' => '#,##0.###', 'percent_pattern' => '#,##0%', 'currency_pattern' => '¤#,##0.00', 'accounting_currency_pattern' => '¤#,##0.00;(¤#,##0.00)', ), 'beng' => array( 'numbering_system' => 'beng', 'decimal_pattern' => '#,##,##0.###', 'percent_pattern' => '#,##,##0%', 'currency_pattern' => '#,##,##0.00¤', 'accounting_currency_pattern' => '#,##,##0.00¤;(#,##,##0.00¤)', ), ); /** * Prepare two currency formats. */ protected $currencies = array( 'USD' => array( 'code' => 'USD', 'name' => 'US Dollar', 'numeric_code' => '840', 'symbol' => '$', ), 'BND' => array( 'code' => 'BND', 'name' => 'dollar Brunei', 'numeric_code' => '096', 'symbol' => 'BND', ), ); /** * @covers ::__construct * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::getNumberFormat * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat */ public function testConstructor() { $numberFormat = new NumberFormat(); $formatter = new NumberFormatter($numberFormat, NumberFormatter::DECIMAL); $this->assertSame($numberFormat, $formatter->getNumberFormat()); } /** * @covers ::__construct * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat * * @expectedException \CommerceGuys\Intl\Exception\InvalidArgumentException * @expectedExceptionMessage Unknown format style provided to NumberFormatter::__construct(). */ public function testConstructorWithInvalidStyle() { $numberFormat = new NumberFormat(); new NumberFormatter($numberFormat, 'foo'); } /** * @covers ::format * @covers ::replaceDigits * @covers ::replaceSymbols * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat * * @dataProvider numberValueProvider */ public function testFormat($number_format, $style, $value, $expected_value) { $formatter = new NumberFormatter($number_format, $style); $formattedNumber = $formatter->format($value); $this->assertSame($expected_value, $formattedNumber); } /** * @covers ::SetMinimumFractionDigits * @covers ::SetMaximumFractionDigits * @covers ::format * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::replaceDigits * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::replaceSymbols * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat */ public function testFormatFractionDigits() { $numberFormat = $this->createNumberFormat($this->numberFormats['latn']); $formatter = new NumberFormatter($numberFormat); $formatter->setMinimumFractionDigits(2); $formattedNumber = $formatter->format('12.5'); $this->assertSame('12.50', $formattedNumber); $formatter = new NumberFormatter($numberFormat); $formatter->setMaximumFractionDigits(1); $formattedNumber = $formatter->format('12.50'); $this->assertSame('12.5', $formattedNumber); $formatter = new NumberFormatter($numberFormat); $formatter->setMinimumFractionDigits(4); $formatter->setMaximumFractionDigits(5); $formattedNumber = $formatter->format('12.50000'); $this->assertSame('12.5000', $formattedNumber); } /** * @covers ::format * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::format * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat * * @expectedException \CommerceGuys\Intl\Exception\InvalidArgumentException */ public function testFormatOnlyAllowsNumbers() { $numberFormat = $this->createNumberFormat($this->numberFormats['latn']); $formatter = new NumberFormatter($numberFormat); $formatter->format('a12.34'); } /** * @covers ::formatCurrency * @covers ::replaceSymbols * @uses \CommerceGuys\Intl\Currency\Currency * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::format * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::replaceDigits * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat * * @dataProvider currencyValueProvider */ public function testFormatCurrency($number_format, $currency, $style, $value, $expected_value) { $formatter = new NumberFormatter($number_format, $style); $formattedNumber = $formatter->formatCurrency($value, $currency); $this->assertSame($expected_value, $formattedNumber); } /** * @covers ::parseCurrency * @uses \CommerceGuys\Intl\Currency\Currency * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat * * @dataProvider formattedCurrencyProvider */ public function testParseCurrency($number_format, $currency, $style, $value, $expected_value) { $formatter = new NumberFormatter($number_format, $style); $parsedNumber = $formatter->parseCurrency($value, $currency); $this->assertSame($expected_value, $parsedNumber); } /** * @covers ::getNumberFormat * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat */ public function testGetNumberFormat() { $numberFormat = $this->createNumberFormat($this->numberFormats['latn']); $formatter = new NumberFormatter($numberFormat, NumberFormatter::DECIMAL); $this->assertSame($numberFormat, $formatter->getNumberFormat()); } /** * @covers ::getMinimumFractionDigits * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat */ public function testMinimumFractionDigits() { $numberFormat = $this->createNumberFormat($this->numberFormats['latn']); // Defaults to 0 for decimal and percentage formats. $formatter = new NumberFormatter($numberFormat, NumberFormatter::DECIMAL); $this->assertEquals(0, $formatter->getMinimumFractionDigits()); $formatter = new NumberFormatter($numberFormat, NumberFormatter::PERCENT); $this->assertEquals(0, $formatter->getMinimumFractionDigits()); // Should default to null for currency formats. $formatter = new NumberFormatter($numberFormat, NumberFormatter::CURRENCY); $this->assertNull($formatter->getMinimumFractionDigits()); $formatter = new NumberFormatter($numberFormat, NumberFormatter::CURRENCY_ACCOUNTING); $this->assertNull($formatter->getMinimumFractionDigits()); } /** * @covers ::getMaximumFractionDigits * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat */ public function testMaximumFractionDigits() { $numberFormat = $this->createNumberFormat($this->numberFormats['latn']); // Defaults to 3 for decimal and percentage formats. $formatter = new NumberFormatter($numberFormat, NumberFormatter::DECIMAL); $this->assertEquals(3, $formatter->getMaximumFractionDigits()); $formatter = new NumberFormatter($numberFormat, NumberFormatter::PERCENT); $this->assertEquals(3, $formatter->getMaximumFractionDigits()); // Should default to null for currency formats. $formatter = new NumberFormatter($numberFormat, NumberFormatter::CURRENCY); $this->assertNull($formatter->getMaximumFractionDigits()); $formatter = new NumberFormatter($numberFormat, NumberFormatter::CURRENCY_ACCOUNTING); $this->assertNull($formatter->getMaximumFractionDigits()); } /** * @covers ::isGroupingUsed * @covers ::setGroupingUsed * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::format * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::replaceDigits * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::replaceSymbols * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat */ public function testGroupingUsed() { $numberFormat = $this->createNumberFormat($this->numberFormats['latn']); // The formatter groups correctly. $formatter = new NumberFormatter($numberFormat, NumberFormatter::DECIMAL); $this->assertTrue($formatter->isGroupingUsed()); $this->assertSame('10,000.9', $formatter->format('10000.90')); // The formatter respects grouping turned off. $formatter = new NumberFormatter($numberFormat, NumberFormatter::DECIMAL); $formatter->setGroupingUsed(false); $this->assertFalse($formatter->isGroupingUsed()); $this->assertSame('10000.9', $formatter->format('10000.90')); } /** * @covers ::getCurrencyDisplay * @covers ::setCurrencyDisplay * @covers ::formatCurrency * @uses \CommerceGuys\Intl\Currency\Currency * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::__construct * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::format * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::replaceDigits * @uses \CommerceGuys\Intl\Formatter\NumberFormatter::replaceSymbols * @uses \CommerceGuys\Intl\NumberFormat\NumberFormat */ public function testCurrencyDisplay() { $numberFormat = $this->createNumberFormat($this->numberFormats['latn']); $currency = $this->createCurrency($this->currencies['USD']); // Currency display defaults to symbol. $formatter = new NumberFormatter($numberFormat, NumberFormatter::CURRENCY); $this->assertSame(NumberFormatter::CURRENCY_DISPLAY_SYMBOL, $formatter->getCurrencyDisplay()); $formattedNumber = $formatter->formatCurrency('100', $currency); $this->assertSame('$100.00', $formattedNumber); // Currency display respects setting the value to currency code. $formatter = new NumberFormatter($numberFormat, NumberFormatter::CURRENCY); $formatter->setCurrencyDisplay(NumberFormatter::CURRENCY_DISPLAY_CODE); $this->assertSame(NumberFormatter::CURRENCY_DISPLAY_CODE, $formatter->getCurrencyDisplay()); $formattedNumber = $formatter->formatCurrency('100', $currency); $this->assertSame('USD100.00', $formattedNumber); } /** * Provides the number format, number style, value and expected formatted value. */ public function numberValueProvider() { return array( array($this->createNumberFormat($this->numberFormats['latn']), NumberFormatter::DECIMAL, '-50.5', '-50.5'), array($this->createNumberFormat($this->numberFormats['latn']), NumberFormatter::PERCENT, '50.5', '50.5%'), array($this->createNumberFormat($this->numberFormats['latn']), NumberFormatter::DECIMAL, '5000000.5', '5,000,000.5'), array($this->createNumberFormat($this->numberFormats['beng'], 'bn'), NumberFormatter::DECIMAL, '-50.5', '-৫০.৫'), array($this->createNumberFormat($this->numberFormats['beng'], 'bn'), NumberFormatter::PERCENT, '50.5', '৫০.৫%'), array($this->createNumberFormat($this->numberFormats['beng'], 'bn'), NumberFormatter::DECIMAL, '5000000.5', '৫০,০০,০০০.৫') ); } /** * Provides the number format, currency format, number style, value and expected formatted value. */ public function currencyValueProvider() { return array( array($this->createNumberFormat($this->numberFormats['latn']), $this->createCurrency($this->currencies['USD']), NumberFormatter::CURRENCY, '-5.05', '-$5.05'), array($this->createNumberFormat($this->numberFormats['latn']), $this->createCurrency($this->currencies['USD']), NumberFormatter::CURRENCY_ACCOUNTING, '-5.05', '($5.05)'), array($this->createNumberFormat($this->numberFormats['latn']), $this->createCurrency($this->currencies['USD']), NumberFormatter::CURRENCY, '500100.05', '$500,100.05'), array($this->createNumberFormat($this->numberFormats['beng'], 'bn'), $this->createCurrency($this->currencies['BND'], 'bn'), NumberFormatter::CURRENCY, '-50.5', '-৫০.৫০BND'), array($this->createNumberFormat($this->numberFormats['beng'], 'bn'), $this->createCurrency($this->currencies['BND'], 'bn'), NumberFormatter::CURRENCY_ACCOUNTING, '-50.5', '(৫০.৫০BND)'), array($this->createNumberFormat($this->numberFormats['beng'], 'bn'), $this->createCurrency($this->currencies['BND'], 'bn'), NumberFormatter::CURRENCY, '500100.05', '৫,০০,১০০.০৫BND'), ); } /** * Provides values for the formatted currency parser. */ public function formattedCurrencyProvider() { return array( array($this->createNumberFormat($this->numberFormats['latn']), $this->createCurrency($this->currencies['USD']), NumberFormatter::CURRENCY, '$500,100.05', '500100.05'), array($this->createNumberFormat($this->numberFormats['latn']), $this->createCurrency($this->currencies['USD']), NumberFormatter::CURRENCY, '-$1,059.59', '-1059.59'), array($this->createNumberFormat($this->numberFormats['latn']), $this->createCurrency($this->currencies['USD']), NumberFormatter::CURRENCY_ACCOUNTING, '($1,059.59)', '-1059.59'), array($this->createNumberFormat($this->numberFormats['beng'], 'bn'), $this->createCurrency($this->currencies['BND'], 'bn'), NumberFormatter::CURRENCY, '৫,০০,১০০.০৫BND', '500100.05'), ); } /** * Helper for initiating a new NumberFormat object. */ protected function createNumberFormat(array $definition, $locale = 'en') { $default = array( 'decimal_separator' => '.', 'grouping_separator' => ',', 'plus_sign' => '+', 'minus_sign' => '-', 'percent_sign' => '%' ); $format = array_merge($default, $definition); $numberFormat = new NumberFormat(); $numberFormat->setLocale($locale); $numberFormat->setNumberingSystem($format['numbering_system']); $numberFormat->setDecimalSeparator($format['decimal_separator']); $numberFormat->setGroupingSeparator($format['grouping_separator']); $numberFormat->setPlusSign($format['plus_sign']); $numberFormat->setMinusSign($format['minus_sign']); $numberFormat->setPercentSign($format['percent_sign']); $numberFormat->setDecimalPattern($format['decimal_pattern']); $numberFormat->setPercentPattern($format['percent_pattern']); $numberFormat->setCurrencyPattern($format['currency_pattern']); $numberFormat->setAccountingCurrencyPattern($format['accounting_currency_pattern']); return $numberFormat; } /** * Helper for initiating a new Currency object. */ protected function createCurrency(array $definition, $locale = 'en') { $default = array( 'fraction_digits' => 2 ); $format = array_merge($default, $definition); $currency = new Currency(); $currency->setCurrencyCode($format['code']); $currency->setName($format['name']); $currency->setNumericCode($format['numeric_code']); $currency->setFractionDigits($format['fraction_digits']); $currency->setSymbol($format['symbol']); $currency->setLocale($locale); return $currency; } }