Formatter.php 95 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\i18n;
  8. use Closure;
  9. use DateInterval;
  10. use DateTime;
  11. use DateTimeInterface;
  12. use DateTimeZone;
  13. use IntlDateFormatter;
  14. use NumberFormatter;
  15. use Yii;
  16. use yii\base\Component;
  17. use yii\base\InvalidArgumentException;
  18. use yii\base\InvalidConfigException;
  19. use yii\helpers\ArrayHelper;
  20. use yii\helpers\FormatConverter;
  21. use yii\helpers\Html;
  22. use yii\helpers\HtmlPurifier;
  23. use yii\helpers\Url;
  24. /**
  25. * Formatter provides a set of commonly used data formatting methods.
  26. *
  27. * The formatting methods provided by Formatter are all named in the form of `asXyz()`.
  28. * The behavior of some of them may be configured via the properties of Formatter. For example,
  29. * by configuring [[dateFormat]], one may control how [[asDate()]] formats the value into a date string.
  30. *
  31. * Formatter is configured as an application component in [[\yii\base\Application]] by default.
  32. * You can access that instance via `Yii::$app->formatter`.
  33. *
  34. * The Formatter class is designed to format values according to a [[locale]]. For this feature to work
  35. * the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) has to be installed.
  36. * Most of the methods however work also if the PHP intl extension is not installed by providing
  37. * a fallback implementation. Without intl month and day names are in English only.
  38. * Note that even if the intl extension is installed, formatting date and time values for years >=2038 or <=1901
  39. * on 32bit systems will fall back to the PHP implementation because intl uses a 32bit UNIX timestamp internally.
  40. * On a 64bit system the intl formatter is used in all cases if installed.
  41. *
  42. * > Note: The Formatter class is meant to be used for formatting values for display to users in different
  43. * > languages and time zones. If you need to format a date or time in machine readable format, use the
  44. * > PHP [date()](https://www.php.net/manual/en/function.date.php) function instead.
  45. *
  46. * @author Qiang Xue <qiang.xue@gmail.com>
  47. * @author Enrica Ruedin <e.ruedin@guggach.com>
  48. * @author Carsten Brandt <mail@cebe.cc>
  49. * @since 2.0
  50. */
  51. class Formatter extends Component
  52. {
  53. /**
  54. * @since 2.0.13
  55. */
  56. const UNIT_SYSTEM_METRIC = 'metric';
  57. /**
  58. * @since 2.0.13
  59. */
  60. const UNIT_SYSTEM_IMPERIAL = 'imperial';
  61. /**
  62. * @since 2.0.13
  63. */
  64. const FORMAT_WIDTH_LONG = 'long';
  65. /**
  66. * @since 2.0.13
  67. */
  68. const FORMAT_WIDTH_SHORT = 'short';
  69. /**
  70. * @since 2.0.13
  71. */
  72. const UNIT_LENGTH = 'length';
  73. /**
  74. * @since 2.0.13
  75. */
  76. const UNIT_WEIGHT = 'mass';
  77. /**
  78. * @var string|null the text to be displayed when formatting a `null` value.
  79. * Defaults to `'<span class="not-set">(not set)</span>'`, where `(not set)`
  80. * will be translated according to [[locale]].
  81. */
  82. public $nullDisplay;
  83. /**
  84. * @var array the text to be displayed when formatting a boolean value. The first element corresponds
  85. * to the text displayed for `false`, the second element for `true`.
  86. * Defaults to `['No', 'Yes']`, where `Yes` and `No`
  87. * will be translated according to [[locale]].
  88. */
  89. public $booleanFormat;
  90. /**
  91. * @var string|null the locale ID that is used to localize the date and number formatting.
  92. * For number and date formatting this is only effective when the
  93. * [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is installed.
  94. * If not set, [[\yii\base\Application::language]] will be used.
  95. */
  96. public $locale;
  97. /**
  98. * @var string|null the language code (e.g. `en-US`, `en`) that is used to translate internal messages.
  99. * If not set, [[locale]] will be used (without the `@calendar` param, if included).
  100. *
  101. * @since 2.0.28
  102. */
  103. public $language;
  104. /**
  105. * @var string|null the time zone to use for formatting time and date values.
  106. *
  107. * This can be any value that may be passed to [date_default_timezone_set()](https://www.php.net/manual/en/function.date-default-timezone-set.php)
  108. * e.g. `UTC`, `Europe/Berlin` or `America/Chicago`.
  109. * Refer to the [php manual](https://www.php.net/manual/en/timezones.php) for available time zones.
  110. * If this property is not set, [[\yii\base\Application::timeZone]] will be used.
  111. *
  112. * Note that the default time zone for input data is assumed to be UTC by default if no time zone is included in the input date value.
  113. * If you store your data in a different time zone in the database, you have to adjust [[defaultTimeZone]] accordingly.
  114. */
  115. public $timeZone;
  116. /**
  117. * @var string the time zone that is assumed for input values if they do not include a time zone explicitly.
  118. *
  119. * The value must be a valid time zone identifier, e.g. `UTC`, `Europe/Berlin` or `America/Chicago`.
  120. * Please refer to the [php manual](https://www.php.net/manual/en/timezones.php) for available time zones.
  121. *
  122. * It defaults to `UTC` so you only have to adjust this value if you store datetime values in another time zone in your database.
  123. *
  124. * Note that a UNIX timestamp is always in UTC by its definition. That means that specifying a default time zone different from
  125. * UTC has no effect on date values given as UNIX timestamp.
  126. *
  127. * @since 2.0.1
  128. */
  129. public $defaultTimeZone = 'UTC';
  130. /**
  131. * @var string the default format string to be used to format a [[asDate()|date]].
  132. * This can be "short", "medium", "long", or "full", which represents a preset format of different lengths.
  133. *
  134. * It can also be a custom format as specified in the [ICU manual](https://unicode-org.github.io/icu/userguide/format_parse/datetime/).
  135. * Alternatively this can be a string prefixed with `php:` representing a format that can be recognized by the
  136. * PHP [date()](https://www.php.net/manual/en/function.date.php)-function.
  137. *
  138. * For example:
  139. *
  140. * ```php
  141. * 'MM/dd/yyyy' // date in ICU format
  142. * 'php:m/d/Y' // the same date in PHP format
  143. * ```
  144. */
  145. public $dateFormat = 'medium';
  146. /**
  147. * @var string the default format string to be used to format a [[asTime()|time]].
  148. * This can be "short", "medium", "long", or "full", which represents a preset format of different lengths.
  149. *
  150. * It can also be a custom format as specified in the [ICU manual](https://unicode-org.github.io/icu/userguide/format_parse/datetime/).
  151. * Alternatively this can be a string prefixed with `php:` representing a format that can be recognized by the
  152. * PHP [date()](https://www.php.net/manual/en/function.date.php)-function.
  153. *
  154. * For example:
  155. *
  156. * ```php
  157. * 'HH:mm:ss' // time in ICU format
  158. * 'php:H:i:s' // the same time in PHP format
  159. * ```
  160. */
  161. public $timeFormat = 'medium';
  162. /**
  163. * @var string the default format string to be used to format a [[asDatetime()|date and time]].
  164. * This can be "short", "medium", "long", or "full", which represents a preset format of different lengths.
  165. *
  166. * It can also be a custom format as specified in the [ICU manual](https://unicode-org.github.io/icu/userguide/format_parse/datetime/).
  167. *
  168. * Alternatively this can be a string prefixed with `php:` representing a format that can be recognized by the
  169. * PHP [date()](https://www.php.net/manual/en/function.date.php) function.
  170. *
  171. * For example:
  172. *
  173. * ```php
  174. * 'MM/dd/yyyy HH:mm:ss' // date and time in ICU format
  175. * 'php:m/d/Y H:i:s' // the same date and time in PHP format
  176. * ```
  177. */
  178. public $datetimeFormat = 'medium';
  179. /**
  180. * @var \IntlCalendar|int|null the calendar to be used for date formatting. The value of this property will be directly
  181. * passed to the [constructor of the `IntlDateFormatter` class](https://www.php.net/manual/en/intldateformatter.create.php).
  182. *
  183. * Defaults to `null`, which means the Gregorian calendar will be used. You may also explicitly pass the constant
  184. * `\IntlDateFormatter::GREGORIAN` for Gregorian calendar.
  185. *
  186. * To use an alternative calendar like for example the [Jalali calendar](https://en.wikipedia.org/wiki/Jalali_calendar),
  187. * set this property to `\IntlDateFormatter::TRADITIONAL`.
  188. * The calendar must then be specified in the [[locale]], for example for the persian calendar the configuration for the formatter would be:
  189. *
  190. * ```php
  191. * 'formatter' => [
  192. * 'locale' => 'fa_IR@calendar=persian',
  193. * 'calendar' => \IntlDateFormatter::TRADITIONAL,
  194. * ],
  195. * ```
  196. *
  197. * Available calendar names can be found in the [ICU manual](https://unicode-org.github.io/icu/userguide/datetime/calendar/).
  198. *
  199. * Since PHP 5.5 you may also use an instance of the [[\IntlCalendar]] class.
  200. * Check the [PHP manual](https://www.php.net/manual/en/intldateformatter.create.php) for more details.
  201. *
  202. * If the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is not available, setting this property will have no effect.
  203. *
  204. * @see https://www.php.net/manual/en/intldateformatter.create.php
  205. * @see https://www.php.net/manual/en/class.intldateformatter.php#intl.intldateformatter-constants.calendartypes
  206. * @see https://www.php.net/manual/en/class.intlcalendar.php
  207. * @since 2.0.7
  208. */
  209. public $calendar;
  210. /**
  211. * @var string|null the character displayed as the decimal point when formatting a number.
  212. * If not set, the decimal separator corresponding to [[locale]] will be used.
  213. * If [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is not available, the default value is '.'.
  214. */
  215. public $decimalSeparator;
  216. /**
  217. * @var string|null the character displayed as the decimal point when formatting a currency.
  218. * If not set, the currency decimal separator corresponding to [[locale]] will be used.
  219. * If [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is not available, setting this property will have no effect.
  220. * @since 2.0.35
  221. */
  222. public $currencyDecimalSeparator;
  223. /**
  224. * @var string|null the character displayed as the thousands separator (also called grouping separator) character when formatting a number.
  225. * If not set, the thousand separator corresponding to [[locale]] will be used.
  226. * If [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is not available, the default value is ','.
  227. */
  228. public $thousandSeparator;
  229. /**
  230. * @var array a list of name value pairs that are passed to the
  231. * intl [NumberFormatter::setAttribute()](https://www.php.net/manual/en/numberformatter.setattribute.php) method of all
  232. * the number formatter objects created by [[createNumberFormatter()]].
  233. * This property takes only effect if the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is installed.
  234. *
  235. * Please refer to the [PHP manual](https://www.php.net/manual/en/class.numberformatter.php#intl.numberformatter-constants.unumberformatattribute)
  236. * for the possible options.
  237. *
  238. * For example to adjust the maximum and minimum value of fraction digits you can configure this property like the following:
  239. *
  240. * ```php
  241. * [
  242. * NumberFormatter::MIN_FRACTION_DIGITS => 0,
  243. * NumberFormatter::MAX_FRACTION_DIGITS => 2,
  244. * ]
  245. * ```
  246. */
  247. public $numberFormatterOptions = [];
  248. /**
  249. * @var array a list of name value pairs that are passed to the
  250. * intl [NumberFormatter::setTextAttribute()](https://www.php.net/manual/en/numberformatter.settextattribute.php) method of all
  251. * the number formatter objects created by [[createNumberFormatter()]].
  252. * This property takes only effect if the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is installed.
  253. *
  254. * Please refer to the [PHP manual](https://www.php.net/manual/en/class.numberformatter.php#intl.numberformatter-constants.unumberformattextattribute)
  255. * for the possible options.
  256. *
  257. * For example to change the minus sign for negative numbers you can configure this property like the following:
  258. *
  259. * ```php
  260. * [
  261. * NumberFormatter::NEGATIVE_PREFIX => 'MINUS',
  262. * ]
  263. * ```
  264. */
  265. public $numberFormatterTextOptions = [];
  266. /**
  267. * @var array a list of name value pairs that are passed to the
  268. * intl [NumberFormatter::setSymbol()](https://www.php.net/manual/en/numberformatter.setsymbol.php) method of all
  269. * the number formatter objects created by [[createNumberFormatter()]].
  270. * This property takes only effect if the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is installed.
  271. *
  272. * Please refer to the [PHP manual](https://www.php.net/manual/en/class.numberformatter.php#intl.numberformatter-constants.unumberformatsymbol)
  273. * for the possible options.
  274. *
  275. * For example to choose a custom currency symbol, e.g. [U+20BD](https://unicode-table.com/en/20BD/) instead of `руб.` for Russian Ruble:
  276. *
  277. * ```php
  278. * [
  279. * NumberFormatter::CURRENCY_SYMBOL => '₽',
  280. * ]
  281. * ```
  282. *
  283. * @since 2.0.4
  284. */
  285. public $numberFormatterSymbols = [];
  286. /**
  287. * @var string|null the 3-letter ISO 4217 currency code indicating the default currency to use for [[asCurrency]].
  288. * If not set, the currency code corresponding to [[locale]] will be used.
  289. * Note that in this case the [[locale]] has to be specified with a country code, e.g. `en-US` otherwise it
  290. * is not possible to determine the default currency.
  291. */
  292. public $currencyCode;
  293. /**
  294. * @var int the base at which a kilobyte is calculated (1000 or 1024 bytes per kilobyte), used by [[asSize]] and [[asShortSize]].
  295. * Defaults to 1024.
  296. */
  297. public $sizeFormatBase = 1024;
  298. /**
  299. * @var string default system of measure units. Defaults to [[UNIT_SYSTEM_METRIC]].
  300. * Possible values:
  301. * - [[UNIT_SYSTEM_METRIC]]
  302. * - [[UNIT_SYSTEM_IMPERIAL]]
  303. *
  304. * @see asLength
  305. * @see asWeight
  306. * @since 2.0.13
  307. */
  308. public $systemOfUnits = self::UNIT_SYSTEM_METRIC;
  309. /**
  310. * @var array configuration of weight and length measurement units.
  311. * This array contains the most usable measurement units, but you can change it
  312. * in case you have some special requirements.
  313. *
  314. * For example, you can add smaller measure unit:
  315. *
  316. * ```php
  317. * $this->measureUnits[self::UNIT_LENGTH][self::UNIT_SYSTEM_METRIC] = [
  318. * 'nanometer' => 0.000001
  319. * ]
  320. * ```
  321. * @see asLength
  322. * @see asWeight
  323. * @since 2.0.13
  324. */
  325. public $measureUnits = [
  326. self::UNIT_LENGTH => [
  327. self::UNIT_SYSTEM_IMPERIAL => [
  328. 'inch' => 1,
  329. 'foot' => 12,
  330. 'yard' => 36,
  331. 'chain' => 792,
  332. 'furlong' => 7920,
  333. 'mile' => 63360,
  334. ],
  335. self::UNIT_SYSTEM_METRIC => [
  336. 'millimeter' => 1,
  337. 'centimeter' => 10,
  338. 'meter' => 1000,
  339. 'kilometer' => 1000000,
  340. ],
  341. ],
  342. self::UNIT_WEIGHT => [
  343. self::UNIT_SYSTEM_IMPERIAL => [
  344. 'grain' => 1,
  345. 'drachm' => 27.34375,
  346. 'ounce' => 437.5,
  347. 'pound' => 7000,
  348. 'stone' => 98000,
  349. 'quarter' => 196000,
  350. 'hundredweight' => 784000,
  351. 'ton' => 15680000,
  352. ],
  353. self::UNIT_SYSTEM_METRIC => [
  354. 'gram' => 1,
  355. 'kilogram' => 1000,
  356. 'ton' => 1000000,
  357. ],
  358. ],
  359. ];
  360. /**
  361. * @var array The base units that are used as multipliers for smallest possible unit from [[measureUnits]].
  362. * @since 2.0.13
  363. */
  364. public $baseUnits = [
  365. self::UNIT_LENGTH => [
  366. self::UNIT_SYSTEM_IMPERIAL => 12, // 1 feet = 12 inches
  367. self::UNIT_SYSTEM_METRIC => 1000, // 1 meter = 1000 millimeters
  368. ],
  369. self::UNIT_WEIGHT => [
  370. self::UNIT_SYSTEM_IMPERIAL => 7000, // 1 pound = 7000 grains
  371. self::UNIT_SYSTEM_METRIC => 1000, // 1 kilogram = 1000 grams
  372. ],
  373. ];
  374. /**
  375. * @var bool whether the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is loaded.
  376. */
  377. private $_intlLoaded = false;
  378. /**
  379. * @var \ResourceBundle cached ResourceBundle object used to read unit translations
  380. */
  381. private $_resourceBundle;
  382. /**
  383. * @var array cached unit translation patterns
  384. */
  385. private $_unitMessages = [];
  386. /**
  387. * {@inheritdoc}
  388. */
  389. public function init()
  390. {
  391. if ($this->timeZone === null) {
  392. $this->timeZone = Yii::$app->timeZone;
  393. }
  394. if ($this->locale === null) {
  395. $this->locale = Yii::$app->language;
  396. }
  397. if ($this->language === null) {
  398. $this->language = strtok($this->locale, '@');
  399. }
  400. if ($this->booleanFormat === null) {
  401. $this->booleanFormat = [Yii::t('yii', 'No', [], $this->language), Yii::t('yii', 'Yes', [], $this->language)];
  402. }
  403. if ($this->nullDisplay === null) {
  404. $this->nullDisplay = '<span class="not-set">' . Yii::t('yii', '(not set)', [], $this->language) . '</span>';
  405. }
  406. $this->_intlLoaded = extension_loaded('intl');
  407. if (!$this->_intlLoaded) {
  408. if ($this->decimalSeparator === null) {
  409. $this->decimalSeparator = '.';
  410. }
  411. if ($this->thousandSeparator === null) {
  412. $this->thousandSeparator = ',';
  413. }
  414. }
  415. }
  416. /**
  417. * Formats the value based on the given format type.
  418. * This method will call one of the "as" methods available in this class to do the formatting.
  419. * For type "xyz", the method "asXyz" will be used. For example, if the format is "html",
  420. * then [[asHtml()]] will be used. Format names are case insensitive.
  421. * @param mixed $value the value to be formatted.
  422. * @param string|array|Closure $format the format of the value, e.g., "html", "text" or an anonymous function
  423. * returning the formatted value.
  424. *
  425. * To specify additional parameters of the formatting method, you may use an array.
  426. * The first element of the array specifies the format name, while the rest of the elements will be used as the
  427. * parameters to the formatting method. For example, a format of `['date', 'Y-m-d']` will cause the invocation
  428. * of `asDate($value, 'Y-m-d')`.
  429. *
  430. * The anonymous function signature should be: `function($value, $formatter)`,
  431. * where `$value` is the value that should be formatted and `$formatter` is an instance of the Formatter class,
  432. * which can be used to call other formatting functions.
  433. * The possibility to use an anonymous function is available since version 2.0.13.
  434. * @return string the formatting result.
  435. * @throws InvalidArgumentException if the format type is not supported by this class.
  436. */
  437. public function format($value, $format)
  438. {
  439. if ($format instanceof Closure) {
  440. return $format($value, $this);
  441. }
  442. if (is_array($format)) {
  443. if (!isset($format[0])) {
  444. throw new InvalidArgumentException('The $format array must contain at least one element.');
  445. }
  446. $f = $format[0];
  447. $format[0] = $value;
  448. $params = $format;
  449. $format = $f;
  450. } else {
  451. $params = [$value];
  452. }
  453. $method = 'as' . $format;
  454. if ($this->hasMethod($method)) {
  455. return call_user_func_array([$this, $method], $params);
  456. }
  457. throw new InvalidArgumentException("Unknown format type: $format");
  458. }
  459. // simple formats
  460. /**
  461. * Formats the value as is without any formatting.
  462. * This method simply returns back the parameter without any format.
  463. * The only exception is a `null` value which will be formatted using [[nullDisplay]].
  464. * @param mixed $value the value to be formatted.
  465. * @return string the formatted result.
  466. */
  467. public function asRaw($value)
  468. {
  469. if ($value === null) {
  470. return $this->nullDisplay;
  471. }
  472. return $value;
  473. }
  474. /**
  475. * Formats the value as an HTML-encoded plain text.
  476. * @param string|null $value the value to be formatted.
  477. * @return string the formatted result.
  478. */
  479. public function asText($value)
  480. {
  481. if ($value === null) {
  482. return $this->nullDisplay;
  483. }
  484. return Html::encode($value);
  485. }
  486. /**
  487. * Formats the value as an HTML-encoded plain text with newlines converted into breaks.
  488. * @param string|null $value the value to be formatted.
  489. * @return string the formatted result.
  490. */
  491. public function asNtext($value)
  492. {
  493. if ($value === null) {
  494. return $this->nullDisplay;
  495. }
  496. return nl2br(Html::encode($value));
  497. }
  498. /**
  499. * Formats the value as HTML-encoded text paragraphs.
  500. * Each text paragraph is enclosed within a `<p>` tag.
  501. * One or multiple consecutive empty lines divide two paragraphs.
  502. * @param string|null $value the value to be formatted.
  503. * @return string the formatted result.
  504. */
  505. public function asParagraphs($value)
  506. {
  507. if ($value === null) {
  508. return $this->nullDisplay;
  509. }
  510. return str_replace('<p></p>', '', '<p>' . preg_replace('/\R{2,}/u', "</p>\n<p>", Html::encode($value)) . '</p>');
  511. }
  512. /**
  513. * Formats the value as HTML text.
  514. * The value will be purified using [[HtmlPurifier]] to avoid XSS attacks.
  515. * Use [[asRaw()]] if you do not want any purification of the value.
  516. * @param string|null $value the value to be formatted.
  517. * @param array|null $config the configuration for the HTMLPurifier class.
  518. * @return string the formatted result.
  519. */
  520. public function asHtml($value, $config = null)
  521. {
  522. if ($value === null) {
  523. return $this->nullDisplay;
  524. }
  525. return HtmlPurifier::process($value, $config);
  526. }
  527. /**
  528. * Formats the value as a mailto link.
  529. * @param string|null $value the value to be formatted.
  530. * @param array $options the tag options in terms of name-value pairs. See [[Html::mailto()]].
  531. * @return string the formatted result.
  532. */
  533. public function asEmail($value, $options = [])
  534. {
  535. if ($value === null) {
  536. return $this->nullDisplay;
  537. }
  538. return Html::mailto(Html::encode($value), $value, $options);
  539. }
  540. /**
  541. * Formats the value as an image tag.
  542. * @param mixed $value the value to be formatted.
  543. * @param array $options the tag options in terms of name-value pairs. See [[Html::img()]].
  544. * @return string the formatted result.
  545. */
  546. public function asImage($value, $options = [])
  547. {
  548. if ($value === null) {
  549. return $this->nullDisplay;
  550. }
  551. return Html::img($value, $options);
  552. }
  553. /**
  554. * Formats the value as a hyperlink.
  555. * @param mixed $value the value to be formatted.
  556. * @param array $options the tag options in terms of name-value pairs. See [[Html::a()]]. Since 2.0.43 there is
  557. * a special option available `scheme` - if set it won't be passed to [[Html::a()]] but it will control the URL
  558. * protocol part of the link by normalizing URL and ensuring that it uses specified scheme. See [[Url::ensureScheme()]].
  559. * If `scheme` is not set the original behavior is preserved which is to add "http://" prefix when "://" string is
  560. * not found in the $value.
  561. * @return string the formatted result.
  562. */
  563. public function asUrl($value, $options = [])
  564. {
  565. if ($value === null) {
  566. return $this->nullDisplay;
  567. }
  568. $url = $value;
  569. $scheme = ArrayHelper::remove($options, 'scheme');
  570. if ($scheme === null) {
  571. if (strpos($url, '://') === false) {
  572. $url = 'http://' . $url;
  573. }
  574. } else {
  575. $url = Url::ensureScheme($url, $scheme);
  576. }
  577. return Html::a(Html::encode($value), $url, $options);
  578. }
  579. /**
  580. * Formats the value as a boolean.
  581. * @param mixed $value the value to be formatted.
  582. * @return string the formatted result.
  583. * @see booleanFormat
  584. */
  585. public function asBoolean($value)
  586. {
  587. if ($value === null) {
  588. return $this->nullDisplay;
  589. }
  590. return $value ? $this->booleanFormat[1] : $this->booleanFormat[0];
  591. }
  592. // date and time formats
  593. /**
  594. * Formats the value as a date.
  595. * @param int|string|DateTime|DateTimeInterface|null $value the value to be formatted. The following
  596. * types of value are supported:
  597. *
  598. * - an integer representing a UNIX timestamp. A UNIX timestamp is always in UTC by its definition.
  599. * - a string that can be [parsed to create a DateTime object](https://www.php.net/manual/en/datetime.formats.php).
  600. * The timestamp is assumed to be in [[defaultTimeZone]] unless a time zone is explicitly given.
  601. * - a PHP [DateTime](https://www.php.net/manual/en/class.datetime.php) object. You may set the time zone
  602. * for the DateTime object to specify the source time zone.
  603. *
  604. * The formatter will convert date values according to [[timeZone]] before formatting it.
  605. * If no timezone conversion should be performed, you need to set [[defaultTimeZone]] and [[timeZone]] to the same value.
  606. * Also no conversion will be performed on values that have no time information, e.g. `"2017-06-05"`.
  607. *
  608. * @param string|null $format the format used to convert the value into a date string.
  609. * If null, [[dateFormat]] will be used.
  610. *
  611. * This can be "short", "medium", "long", or "full", which represents a preset format of different lengths.
  612. * It can also be a custom format as specified in the [ICU manual](https://unicode-org.github.io/icu/userguide/format_parse/datetime/).
  613. *
  614. * Alternatively this can be a string prefixed with `php:` representing a format that can be recognized by the
  615. * PHP [date()](https://www.php.net/manual/en/function.date.php)-function.
  616. *
  617. * @return string the formatted result.
  618. * @throws InvalidArgumentException if the input value can not be evaluated as a date value.
  619. * @throws InvalidConfigException if the date format is invalid.
  620. * @see dateFormat
  621. */
  622. public function asDate($value, $format = null)
  623. {
  624. if ($format === null) {
  625. $format = $this->dateFormat;
  626. }
  627. return $this->formatDateTimeValue($value, $format, 'date');
  628. }
  629. /**
  630. * Formats the value as a time.
  631. * @param int|string|DateTime|DateTimeInterface|null $value the value to be formatted. The following
  632. * types of value are supported:
  633. *
  634. * - an integer representing a UNIX timestamp. A UNIX timestamp is always in UTC by its definition.
  635. * - a string that can be [parsed to create a DateTime object](https://www.php.net/manual/en/datetime.formats.php).
  636. * The timestamp is assumed to be in [[defaultTimeZone]] unless a time zone is explicitly given.
  637. * - a PHP [DateTime](https://www.php.net/manual/en/class.datetime.php) object. You may set the time zone
  638. * for the DateTime object to specify the source time zone.
  639. *
  640. * The formatter will convert date values according to [[timeZone]] before formatting it.
  641. * If no timezone conversion should be performed, you need to set [[defaultTimeZone]] and [[timeZone]] to the same value.
  642. *
  643. * @param string|null $format the format used to convert the value into a date string.
  644. * If null, [[timeFormat]] will be used.
  645. *
  646. * This can be "short", "medium", "long", or "full", which represents a preset format of different lengths.
  647. * It can also be a custom format as specified in the [ICU manual](https://unicode-org.github.io/icu/userguide/format_parse/datetime/).
  648. *
  649. * Alternatively this can be a string prefixed with `php:` representing a format that can be recognized by the
  650. * PHP [date()](https://www.php.net/manual/en/function.date.php)-function.
  651. *
  652. * @return string the formatted result.
  653. * @throws InvalidArgumentException if the input value can not be evaluated as a date value.
  654. * @throws InvalidConfigException if the date format is invalid.
  655. * @see timeFormat
  656. */
  657. public function asTime($value, $format = null)
  658. {
  659. if ($format === null) {
  660. $format = $this->timeFormat;
  661. }
  662. return $this->formatDateTimeValue($value, $format, 'time');
  663. }
  664. /**
  665. * Formats the value as a datetime.
  666. * @param int|string|DateTime|DateTimeInterface|null $value the value to be formatted. The following
  667. * types of value are supported:
  668. *
  669. * - an integer representing a UNIX timestamp. A UNIX timestamp is always in UTC by its definition.
  670. * - a string that can be [parsed to create a DateTime object](https://www.php.net/manual/en/datetime.formats.php).
  671. * The timestamp is assumed to be in [[defaultTimeZone]] unless a time zone is explicitly given.
  672. * - a PHP [DateTime](https://www.php.net/manual/en/class.datetime.php) object. You may set the time zone
  673. * for the DateTime object to specify the source time zone.
  674. *
  675. * The formatter will convert date values according to [[timeZone]] before formatting it.
  676. * If no timezone conversion should be performed, you need to set [[defaultTimeZone]] and [[timeZone]] to the same value.
  677. *
  678. * @param string|null $format the format used to convert the value into a date string.
  679. * If null, [[datetimeFormat]] will be used.
  680. *
  681. * This can be "short", "medium", "long", or "full", which represents a preset format of different lengths.
  682. * It can also be a custom format as specified in the [ICU manual](https://unicode-org.github.io/icu/userguide/format_parse/datetime/).
  683. *
  684. * Alternatively this can be a string prefixed with `php:` representing a format that can be recognized by the
  685. * PHP [date()](https://www.php.net/manual/en/function.date.php)-function.
  686. *
  687. * @return string the formatted result.
  688. * @throws InvalidArgumentException if the input value can not be evaluated as a date value.
  689. * @throws InvalidConfigException if the date format is invalid.
  690. * @see datetimeFormat
  691. */
  692. public function asDatetime($value, $format = null)
  693. {
  694. if ($format === null) {
  695. $format = $this->datetimeFormat;
  696. }
  697. return $this->formatDateTimeValue($value, $format, 'datetime');
  698. }
  699. /**
  700. * @var array map of short format names to IntlDateFormatter constant values.
  701. */
  702. private $_dateFormats = [
  703. 'short' => 3, // IntlDateFormatter::SHORT,
  704. 'medium' => 2, // IntlDateFormatter::MEDIUM,
  705. 'long' => 1, // IntlDateFormatter::LONG,
  706. 'full' => 0, // IntlDateFormatter::FULL,
  707. ];
  708. /**
  709. * @param int|string|DateTime|DateTimeInterface|null $value the value to be formatted. The following
  710. * types of value are supported:
  711. *
  712. * - an integer representing a UNIX timestamp
  713. * - a string that can be [parsed to create a DateTime object](https://www.php.net/manual/en/datetime.formats.php).
  714. * The timestamp is assumed to be in [[defaultTimeZone]] unless a time zone is explicitly given.
  715. * - a PHP [DateTime](https://www.php.net/manual/en/class.datetime.php) object
  716. *
  717. * @param string $format the format used to convert the value into a date string.
  718. * @param string $type 'date', 'time', or 'datetime'.
  719. * @throws InvalidConfigException if the date format is invalid.
  720. * @return string the formatted result.
  721. */
  722. private function formatDateTimeValue($value, $format, $type)
  723. {
  724. $timeZone = $this->timeZone;
  725. // avoid time zone conversion for date-only and time-only values
  726. if ($type === 'date' || $type === 'time') {
  727. list($timestamp, $hasTimeInfo, $hasDateInfo) = $this->normalizeDatetimeValue($value, true);
  728. if (($type === 'date' && !$hasTimeInfo) || ($type === 'time' && !$hasDateInfo)) {
  729. $timeZone = $this->defaultTimeZone;
  730. }
  731. } else {
  732. $timestamp = $this->normalizeDatetimeValue($value);
  733. }
  734. if ($timestamp === null) {
  735. return $this->nullDisplay;
  736. }
  737. // intl does not work with dates >=2038 or <=1901 on 32bit machines, fall back to PHP
  738. $year = $timestamp->format('Y');
  739. if ($this->_intlLoaded && !(PHP_INT_SIZE === 4 && ($year <= 1901 || $year >= 2038))) {
  740. if (strncmp($format, 'php:', 4) === 0) {
  741. $format = FormatConverter::convertDatePhpToIcu(substr($format, 4));
  742. }
  743. if (isset($this->_dateFormats[$format])) {
  744. if ($type === 'date') {
  745. $formatter = new IntlDateFormatter(
  746. $this->locale,
  747. $this->_dateFormats[$format],
  748. IntlDateFormatter::NONE,
  749. $timeZone,
  750. $this->calendar
  751. );
  752. } elseif ($type === 'time') {
  753. $formatter = new IntlDateFormatter(
  754. $this->locale,
  755. IntlDateFormatter::NONE,
  756. $this->_dateFormats[$format],
  757. $timeZone,
  758. $this->calendar
  759. );
  760. } else {
  761. $formatter = new IntlDateFormatter(
  762. $this->locale,
  763. $this->_dateFormats[$format],
  764. $this->_dateFormats[$format],
  765. $timeZone,
  766. $this->calendar
  767. );
  768. }
  769. } else {
  770. $formatter = new IntlDateFormatter(
  771. $this->locale,
  772. IntlDateFormatter::NONE,
  773. IntlDateFormatter::NONE,
  774. $timeZone,
  775. $this->calendar,
  776. $format
  777. );
  778. }
  779. // make IntlDateFormatter work with DateTimeImmutable
  780. if ($timestamp instanceof \DateTimeImmutable) {
  781. $timestamp = new DateTime($timestamp->format(DateTime::ISO8601), $timestamp->getTimezone());
  782. }
  783. return $formatter->format($timestamp);
  784. }
  785. if (strncmp($format, 'php:', 4) === 0) {
  786. $format = substr($format, 4);
  787. } else {
  788. $format = FormatConverter::convertDateIcuToPhp($format, $type, $this->locale);
  789. }
  790. if ($timeZone != null) {
  791. if ($timestamp instanceof \DateTimeImmutable) {
  792. $timestamp = $timestamp->setTimezone(new DateTimeZone($timeZone));
  793. } else {
  794. $timestamp->setTimezone(new DateTimeZone($timeZone));
  795. }
  796. }
  797. return $timestamp->format($format);
  798. }
  799. /**
  800. * Normalizes the given datetime value as a DateTime object that can be taken by various date/time formatting methods.
  801. *
  802. * @param int|string|DateTime|DateTimeInterface|null $value the datetime value to be normalized. The following
  803. * types of value are supported:
  804. *
  805. * - an integer representing a UNIX timestamp
  806. * - a string that can be [parsed to create a DateTime object](https://www.php.net/manual/en/datetime.formats.php).
  807. * The timestamp is assumed to be in [[defaultTimeZone]] unless a time zone is explicitly given.
  808. * - a PHP [DateTime](https://www.php.net/manual/en/class.datetime.php) object
  809. *
  810. * @param bool $checkDateTimeInfo whether to also check if the date/time value has some time and date information attached.
  811. * Defaults to `false`. If `true`, the method will then return an array with the first element being the normalized
  812. * timestamp, the second a boolean indicating whether the timestamp has time information and third a boolean indicating
  813. * whether the timestamp has date information.
  814. * This parameter is available since version 2.0.1.
  815. * @return DateTime|array the normalized datetime value
  816. * Since version 2.0.1 this may also return an array if `$checkDateTimeInfo` is true.
  817. * The first element of the array is the normalized timestamp and the second is a boolean indicating whether
  818. * the timestamp has time information or it is just a date value.
  819. * Since version 2.0.12 the array has third boolean element indicating whether the timestamp has date information
  820. * or it is just a time value.
  821. * @throws InvalidArgumentException if the input value can not be evaluated as a date value.
  822. */
  823. protected function normalizeDatetimeValue($value, $checkDateTimeInfo = false)
  824. {
  825. // checking for DateTime and DateTimeInterface is not redundant, DateTimeInterface is only in PHP>5.5
  826. if ($value === null || $value instanceof DateTime || $value instanceof DateTimeInterface) {
  827. // skip any processing
  828. return $checkDateTimeInfo ? [$value, true, true] : $value;
  829. }
  830. if (empty($value)) {
  831. $value = 0;
  832. }
  833. try {
  834. if (is_numeric($value)) { // process as unix timestamp, which is always in UTC
  835. $timestamp = new DateTime('@' . (int) $value, new DateTimeZone('UTC'));
  836. return $checkDateTimeInfo ? [$timestamp, true, true] : $timestamp;
  837. }
  838. if (
  839. ($timestamp = DateTime::createFromFormat(
  840. 'Y-m-d|',
  841. $value,
  842. new DateTimeZone($this->defaultTimeZone))
  843. ) !== false
  844. ) { // try Y-m-d format (support invalid dates like 2012-13-01)
  845. return $checkDateTimeInfo ? [$timestamp, false, true] : $timestamp;
  846. }
  847. if (
  848. ($timestamp = DateTime::createFromFormat(
  849. 'Y-m-d H:i:s',
  850. $value,
  851. new DateTimeZone($this->defaultTimeZone))
  852. ) !== false
  853. ) { // try Y-m-d H:i:s format (support invalid dates like 2012-13-01 12:63:12)
  854. return $checkDateTimeInfo ? [$timestamp, true, true] : $timestamp;
  855. }
  856. // finally try to create a DateTime object with the value
  857. if ($checkDateTimeInfo) {
  858. $timestamp = new DateTime($value, new DateTimeZone($this->defaultTimeZone));
  859. $info = date_parse($value);
  860. return [
  861. $timestamp,
  862. !($info['hour'] === false && $info['minute'] === false && $info['second'] === false),
  863. !($info['year'] === false && $info['month'] === false && $info['day'] === false && empty($info['zone'])),
  864. ];
  865. }
  866. return new DateTime($value, new DateTimeZone($this->defaultTimeZone));
  867. } catch (\Exception $e) {
  868. throw new InvalidArgumentException("'$value' is not a valid date time value: " . $e->getMessage()
  869. . "\n" . print_r(DateTime::getLastErrors(), true), $e->getCode(), $e);
  870. }
  871. }
  872. /**
  873. * Formats a date, time or datetime in a float number as UNIX timestamp (seconds since 01-01-1970).
  874. * @param int|string|DateTime|DateTimeInterface|null $value the value to be formatted. The following
  875. * types of value are supported:
  876. *
  877. * - an integer representing a UNIX timestamp
  878. * - a string that can be [parsed to create a DateTime object](https://www.php.net/manual/en/datetime.formats.php).
  879. * The timestamp is assumed to be in [[defaultTimeZone]] unless a time zone is explicitly given.
  880. * - a PHP [DateTime](https://www.php.net/manual/en/class.datetime.php) object
  881. *
  882. * @return string the formatted result.
  883. */
  884. public function asTimestamp($value)
  885. {
  886. if ($value === null) {
  887. return $this->nullDisplay;
  888. }
  889. $timestamp = $this->normalizeDatetimeValue($value);
  890. return number_format($timestamp->format('U'), 0, '.', '');
  891. }
  892. /**
  893. * Formats the value as the time interval between a date and now in human readable form.
  894. *
  895. * This method can be used in three different ways:
  896. *
  897. * 1. Using a timestamp that is relative to `now`.
  898. * 2. Using a timestamp that is relative to the `$referenceTime`.
  899. * 3. Using a `DateInterval` object.
  900. *
  901. * @param int|string|DateTime|DateTimeInterface|DateInterval|null $value the value to be formatted. The following
  902. * types of value are supported:
  903. *
  904. * - an integer representing a UNIX timestamp
  905. * - a string that can be [parsed to create a DateTime object](https://www.php.net/manual/en/datetime.formats.php).
  906. * The timestamp is assumed to be in [[defaultTimeZone]] unless a time zone is explicitly given.
  907. * - a PHP [DateTime](https://www.php.net/manual/en/class.datetime.php) object
  908. * - a PHP DateInterval object (a positive time interval will refer to the past, a negative one to the future)
  909. *
  910. * @param int|string|DateTime|DateTimeInterface|null $referenceTime if specified the value is used as a reference time instead of `now`
  911. * when `$value` is not a `DateInterval` object.
  912. * @return string the formatted result.
  913. * @throws InvalidArgumentException if the input value can not be evaluated as a date value.
  914. */
  915. public function asRelativeTime($value, $referenceTime = null)
  916. {
  917. if ($value === null) {
  918. return $this->nullDisplay;
  919. }
  920. if ($value instanceof DateInterval) {
  921. $interval = $value;
  922. } else {
  923. $timestamp = $this->normalizeDatetimeValue($value);
  924. $timeZone = new DateTimeZone($this->timeZone);
  925. if ($referenceTime === null) {
  926. $dateNow = new DateTime('now', $timeZone);
  927. } else {
  928. $dateNow = $this->normalizeDatetimeValue($referenceTime);
  929. $dateNow->setTimezone($timeZone);
  930. }
  931. $dateThen = $timestamp->setTimezone($timeZone);
  932. $interval = $dateThen->diff($dateNow);
  933. }
  934. if ($interval->invert) {
  935. if ($interval->y >= 1) {
  936. return Yii::t('yii', 'in {delta, plural, =1{a year} other{# years}}', ['delta' => $interval->y], $this->language);
  937. }
  938. if ($interval->m >= 1) {
  939. return Yii::t('yii', 'in {delta, plural, =1{a month} other{# months}}', ['delta' => $interval->m], $this->language);
  940. }
  941. if ($interval->d >= 1) {
  942. return Yii::t('yii', 'in {delta, plural, =1{a day} other{# days}}', ['delta' => $interval->d], $this->language);
  943. }
  944. if ($interval->h >= 1) {
  945. return Yii::t('yii', 'in {delta, plural, =1{an hour} other{# hours}}', ['delta' => $interval->h], $this->language);
  946. }
  947. if ($interval->i >= 1) {
  948. return Yii::t('yii', 'in {delta, plural, =1{a minute} other{# minutes}}', ['delta' => $interval->i], $this->language);
  949. }
  950. if ($interval->s == 0) {
  951. return Yii::t('yii', 'just now', [], $this->language);
  952. }
  953. return Yii::t('yii', 'in {delta, plural, =1{a second} other{# seconds}}', ['delta' => $interval->s], $this->language);
  954. }
  955. if ($interval->y >= 1) {
  956. return Yii::t('yii', '{delta, plural, =1{a year} other{# years}} ago', ['delta' => $interval->y], $this->language);
  957. }
  958. if ($interval->m >= 1) {
  959. return Yii::t('yii', '{delta, plural, =1{a month} other{# months}} ago', ['delta' => $interval->m], $this->language);
  960. }
  961. if ($interval->d >= 1) {
  962. return Yii::t('yii', '{delta, plural, =1{a day} other{# days}} ago', ['delta' => $interval->d], $this->language);
  963. }
  964. if ($interval->h >= 1) {
  965. return Yii::t('yii', '{delta, plural, =1{an hour} other{# hours}} ago', ['delta' => $interval->h], $this->language);
  966. }
  967. if ($interval->i >= 1) {
  968. return Yii::t('yii', '{delta, plural, =1{a minute} other{# minutes}} ago', ['delta' => $interval->i], $this->language);
  969. }
  970. if ($interval->s == 0) {
  971. return Yii::t('yii', 'just now', [], $this->language);
  972. }
  973. return Yii::t('yii', '{delta, plural, =1{a second} other{# seconds}} ago', ['delta' => $interval->s], $this->language);
  974. }
  975. /**
  976. * Represents the value as duration in human readable format.
  977. *
  978. * @param DateInterval|string|int|null $value the value to be formatted. Acceptable formats:
  979. * - [DateInterval object](https://www.php.net/manual/ru/class.dateinterval.php)
  980. * - integer - number of seconds. For example: value `131` represents `2 minutes, 11 seconds`
  981. * - ISO8601 duration format. For example, all of these values represent `1 day, 2 hours, 30 minutes` duration:
  982. * `2015-01-01T13:00:00Z/2015-01-02T13:30:00Z` - between two datetime values
  983. * `2015-01-01T13:00:00Z/P1D2H30M` - time interval after datetime value
  984. * `P1D2H30M/2015-01-02T13:30:00Z` - time interval before datetime value
  985. * `P1D2H30M` - simply a date interval
  986. * `P-1D2H30M` - a negative date interval (`-1 day, 2 hours, 30 minutes`)
  987. *
  988. * @param string $implodeString will be used to concatenate duration parts. Defaults to `, `.
  989. * @param string $negativeSign will be prefixed to the formatted duration, when it is negative. Defaults to `-`.
  990. * @return string the formatted duration.
  991. * @since 2.0.7
  992. */
  993. public function asDuration($value, $implodeString = ', ', $negativeSign = '-')
  994. {
  995. if ($value === null) {
  996. return $this->nullDisplay;
  997. }
  998. if ($value instanceof DateInterval) {
  999. $isNegative = $value->invert;
  1000. $interval = $value;
  1001. } elseif (is_numeric($value)) {
  1002. $isNegative = $value < 0;
  1003. $zeroDateTime = (new DateTime())->setTimestamp(0);
  1004. $valueDateTime = (new DateTime())->setTimestamp(abs($value));
  1005. $interval = $valueDateTime->diff($zeroDateTime);
  1006. } elseif (strncmp($value, 'P-', 2) === 0) {
  1007. $interval = new DateInterval('P' . substr($value, 2));
  1008. $isNegative = true;
  1009. } else {
  1010. $interval = new DateInterval($value);
  1011. $isNegative = $interval->invert;
  1012. }
  1013. $parts = [];
  1014. if ($interval->y > 0) {
  1015. $parts[] = Yii::t('yii', '{delta, plural, =1{1 year} other{# years}}', ['delta' => $interval->y], $this->language);
  1016. }
  1017. if ($interval->m > 0) {
  1018. $parts[] = Yii::t('yii', '{delta, plural, =1{1 month} other{# months}}', ['delta' => $interval->m], $this->language);
  1019. }
  1020. if ($interval->d > 0) {
  1021. $parts[] = Yii::t('yii', '{delta, plural, =1{1 day} other{# days}}', ['delta' => $interval->d], $this->language);
  1022. }
  1023. if ($interval->h > 0) {
  1024. $parts[] = Yii::t('yii', '{delta, plural, =1{1 hour} other{# hours}}', ['delta' => $interval->h], $this->language);
  1025. }
  1026. if ($interval->i > 0) {
  1027. $parts[] = Yii::t('yii', '{delta, plural, =1{1 minute} other{# minutes}}', ['delta' => $interval->i], $this->language);
  1028. }
  1029. if ($interval->s > 0) {
  1030. $parts[] = Yii::t('yii', '{delta, plural, =1{1 second} other{# seconds}}', ['delta' => $interval->s], $this->language);
  1031. }
  1032. if ($interval->s === 0 && empty($parts)) {
  1033. $parts[] = Yii::t('yii', '{delta, plural, =1{1 second} other{# seconds}}', ['delta' => $interval->s], $this->language);
  1034. $isNegative = false;
  1035. }
  1036. return empty($parts) ? $this->nullDisplay : (($isNegative ? $negativeSign : '') . implode($implodeString, $parts));
  1037. }
  1038. // number formats
  1039. /**
  1040. * Formats the value as an integer number by removing any decimal digits without rounding.
  1041. *
  1042. * Since 2.0.16 numbers that are mispresented after normalization are formatted as strings using fallback function
  1043. * without [PHP intl extension](https://www.php.net/manual/en/book.intl.php) support. For very big numbers it's
  1044. * recommended to pass them as strings and not use scientific notation otherwise the output might be wrong.
  1045. *
  1046. * @param mixed $value the value to be formatted.
  1047. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1048. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1049. * @return string the formatted result.
  1050. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1051. */
  1052. public function asInteger($value, $options = [], $textOptions = [])
  1053. {
  1054. if ($value === null) {
  1055. return $this->nullDisplay;
  1056. }
  1057. $normalizedValue = $this->normalizeNumericValue($value);
  1058. if ($this->isNormalizedValueMispresented($value, $normalizedValue)) {
  1059. return $this->asIntegerStringFallback((string) $value);
  1060. }
  1061. if ($this->_intlLoaded) {
  1062. $f = $this->createNumberFormatter(NumberFormatter::DECIMAL, null, $options, $textOptions);
  1063. $f->setAttribute(NumberFormatter::FRACTION_DIGITS, 0);
  1064. if (($result = $f->format($normalizedValue, NumberFormatter::TYPE_INT64)) === false) {
  1065. throw new InvalidArgumentException('Formatting integer value failed: ' . $f->getErrorCode() . ' ' . $f->getErrorMessage());
  1066. }
  1067. return $result;
  1068. }
  1069. return number_format((int) $normalizedValue, 0, $this->decimalSeparator, $this->thousandSeparator);
  1070. }
  1071. /**
  1072. * Formats the value as a decimal number.
  1073. *
  1074. * Property [[decimalSeparator]] will be used to represent the decimal point. The
  1075. * value is rounded automatically to the defined decimal digits.
  1076. *
  1077. * Since 2.0.16 numbers that are mispresented after normalization are formatted as strings using fallback function
  1078. * without [PHP intl extension](https://www.php.net/manual/en/book.intl.php) support. For very big numbers it's
  1079. * recommended to pass them as strings and not use scientific notation otherwise the output might be wrong.
  1080. *
  1081. * @param mixed $value the value to be formatted.
  1082. * @param int|null $decimals the number of digits after the decimal point.
  1083. * If not given, the number of digits depends in the input value and is determined based on
  1084. * `NumberFormatter::MIN_FRACTION_DIGITS` and `NumberFormatter::MAX_FRACTION_DIGITS`, which can be configured
  1085. * using [[$numberFormatterOptions]].
  1086. * If the PHP intl extension is not available, the default value is `2`.
  1087. * If you want consistent behavior between environments where intl is available and not, you should explicitly
  1088. * specify a value here.
  1089. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1090. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1091. * @return string the formatted result.
  1092. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1093. * @see decimalSeparator
  1094. * @see thousandSeparator
  1095. */
  1096. public function asDecimal($value, $decimals = null, $options = [], $textOptions = [])
  1097. {
  1098. if ($value === null) {
  1099. return $this->nullDisplay;
  1100. }
  1101. $normalizedValue = $this->normalizeNumericValue($value);
  1102. if ($this->isNormalizedValueMispresented($value, $normalizedValue)) {
  1103. return $this->asDecimalStringFallback((string) $value, $decimals);
  1104. }
  1105. if ($this->_intlLoaded) {
  1106. $f = $this->createNumberFormatter(NumberFormatter::DECIMAL, $decimals, $options, $textOptions);
  1107. if (($result = $f->format($normalizedValue)) === false) {
  1108. throw new InvalidArgumentException('Formatting decimal value failed: ' . $f->getErrorCode() . ' ' . $f->getErrorMessage());
  1109. }
  1110. return $result;
  1111. }
  1112. if ($decimals === null) {
  1113. $decimals = 2;
  1114. }
  1115. return number_format($normalizedValue, $decimals, $this->decimalSeparator, $this->thousandSeparator);
  1116. }
  1117. /**
  1118. * Formats the value as a percent number with "%" sign.
  1119. *
  1120. * Since 2.0.16 numbers that are mispresented after normalization are formatted as strings using fallback function
  1121. * without [PHP intl extension](https://www.php.net/manual/en/book.intl.php) support. For very big numbers it's
  1122. * recommended to pass them as strings and not use scientific notation otherwise the output might be wrong.
  1123. *
  1124. * @param mixed $value the value to be formatted. It must be a factor e.g. `0.75` will result in `75%`.
  1125. * @param int|null $decimals the number of digits after the decimal point.
  1126. * If not given, the number of digits depends in the input value and is determined based on
  1127. * `NumberFormatter::MIN_FRACTION_DIGITS` and `NumberFormatter::MAX_FRACTION_DIGITS`, which can be configured
  1128. * using [[$numberFormatterOptions]].
  1129. * If the PHP intl extension is not available, the default value is `0`.
  1130. * If you want consistent behavior between environments where intl is available and not, you should explicitly
  1131. * specify a value here.
  1132. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1133. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1134. * @return string the formatted result.
  1135. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1136. */
  1137. public function asPercent($value, $decimals = null, $options = [], $textOptions = [])
  1138. {
  1139. if ($value === null) {
  1140. return $this->nullDisplay;
  1141. }
  1142. $normalizedValue = $this->normalizeNumericValue($value);
  1143. if ($this->isNormalizedValueMispresented($value, $normalizedValue)) {
  1144. return $this->asPercentStringFallback((string) $value, $decimals);
  1145. }
  1146. if ($this->_intlLoaded) {
  1147. $f = $this->createNumberFormatter(NumberFormatter::PERCENT, $decimals, $options, $textOptions);
  1148. if (($result = $f->format($normalizedValue)) === false) {
  1149. throw new InvalidArgumentException('Formatting percent value failed: ' . $f->getErrorCode() . ' ' . $f->getErrorMessage());
  1150. }
  1151. return $result;
  1152. }
  1153. if ($decimals === null) {
  1154. $decimals = 0;
  1155. }
  1156. $normalizedValue *= 100;
  1157. return number_format($normalizedValue, $decimals, $this->decimalSeparator, $this->thousandSeparator) . '%';
  1158. }
  1159. /**
  1160. * Formats the value as a scientific number.
  1161. *
  1162. * @param mixed $value the value to be formatted.
  1163. * @param int|null $decimals the number of digits after the decimal point.
  1164. * If not given, the number of digits depends in the input value and is determined based on
  1165. * `NumberFormatter::MIN_FRACTION_DIGITS` and `NumberFormatter::MAX_FRACTION_DIGITS`, which can be configured
  1166. * using [[$numberFormatterOptions]].
  1167. * If the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is not available, the default value
  1168. * depends on your PHP configuration.
  1169. * If you want consistent behavior between environments where intl is available and not, you should explicitly
  1170. * specify a value here.
  1171. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1172. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1173. * @return string the formatted result.
  1174. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1175. */
  1176. public function asScientific($value, $decimals = null, $options = [], $textOptions = [])
  1177. {
  1178. if ($value === null) {
  1179. return $this->nullDisplay;
  1180. }
  1181. $value = $this->normalizeNumericValue($value);
  1182. if ($this->_intlLoaded) {
  1183. $f = $this->createNumberFormatter(NumberFormatter::SCIENTIFIC, $decimals, $options, $textOptions);
  1184. if (($result = $f->format($value)) === false) {
  1185. throw new InvalidArgumentException('Formatting scientific number value failed: ' . $f->getErrorCode() . ' ' . $f->getErrorMessage());
  1186. }
  1187. return $result;
  1188. }
  1189. if ($decimals !== null) {
  1190. return sprintf("%.{$decimals}E", $value);
  1191. }
  1192. return sprintf('%.E', $value);
  1193. }
  1194. /**
  1195. * Formats the value as a currency number.
  1196. *
  1197. * This function does not require the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) to be installed
  1198. * to work, but it is highly recommended to install it to get good formatting results.
  1199. *
  1200. * Since 2.0.16 numbers that are mispresented after normalization are formatted as strings using fallback function
  1201. * without PHP intl extension support. For very big numbers it's recommended to pass them as strings and not use
  1202. * scientific notation otherwise the output might be wrong.
  1203. *
  1204. * @param mixed $value the value to be formatted.
  1205. * @param string|null $currency the 3-letter ISO 4217 currency code indicating the currency to use.
  1206. * If null, [[currencyCode]] will be used.
  1207. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1208. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1209. * @return string the formatted result.
  1210. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1211. * @throws InvalidConfigException if no currency is given and [[currencyCode]] is not defined.
  1212. */
  1213. public function asCurrency($value, $currency = null, $options = [], $textOptions = [])
  1214. {
  1215. if ($value === null) {
  1216. return $this->nullDisplay;
  1217. }
  1218. $normalizedValue = $this->normalizeNumericValue($value);
  1219. if ($this->isNormalizedValueMispresented($value, $normalizedValue)) {
  1220. return $this->asCurrencyStringFallback((string) $value, $currency);
  1221. }
  1222. if ($this->_intlLoaded) {
  1223. $currency = $currency ?: $this->currencyCode;
  1224. // currency code must be set before fraction digits
  1225. // https://www.php.net/manual/en/numberformatter.formatcurrency.php#114376
  1226. if ($currency && !isset($textOptions[NumberFormatter::CURRENCY_CODE])) {
  1227. $textOptions[NumberFormatter::CURRENCY_CODE] = $currency;
  1228. }
  1229. $formatter = $this->createNumberFormatter(NumberFormatter::CURRENCY, null, $options, $textOptions);
  1230. if ($currency === null) {
  1231. $result = $formatter->format($normalizedValue);
  1232. } else {
  1233. $result = $formatter->formatCurrency($normalizedValue, $currency);
  1234. }
  1235. if ($result === false) {
  1236. throw new InvalidArgumentException('Formatting currency value failed: ' . $formatter->getErrorCode() . ' ' . $formatter->getErrorMessage());
  1237. }
  1238. return $result;
  1239. }
  1240. if ($currency === null) {
  1241. if ($this->currencyCode === null) {
  1242. throw new InvalidConfigException('The default currency code for the formatter is not defined and the php intl extension is not installed which could take the default currency from the locale.');
  1243. }
  1244. $currency = $this->currencyCode;
  1245. }
  1246. return $currency . ' ' . $this->asDecimal($normalizedValue, 2, $options, $textOptions);
  1247. }
  1248. /**
  1249. * Formats the value as a number spellout.
  1250. *
  1251. * This function requires the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) to be installed.
  1252. *
  1253. * This formatter does not work well with very big numbers.
  1254. *
  1255. * @param mixed $value the value to be formatted
  1256. * @return string the formatted result.
  1257. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1258. * @throws InvalidConfigException when the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is not available.
  1259. */
  1260. public function asSpellout($value)
  1261. {
  1262. if ($value === null) {
  1263. return $this->nullDisplay;
  1264. }
  1265. $value = $this->normalizeNumericValue($value);
  1266. if ($this->_intlLoaded) {
  1267. $f = $this->createNumberFormatter(NumberFormatter::SPELLOUT);
  1268. if (($result = $f->format($value)) === false) {
  1269. throw new InvalidArgumentException('Formatting number as spellout failed: ' . $f->getErrorCode() . ' ' . $f->getErrorMessage());
  1270. }
  1271. return $result;
  1272. }
  1273. throw new InvalidConfigException('Format as Spellout is only supported when PHP intl extension is installed.');
  1274. }
  1275. /**
  1276. * Formats the value as a ordinal value of a number.
  1277. *
  1278. * This function requires the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) to be installed.
  1279. *
  1280. * This formatter does not work well with very big numbers.
  1281. *
  1282. * @param mixed $value the value to be formatted
  1283. * @return string the formatted result.
  1284. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1285. * @throws InvalidConfigException when the [PHP intl extension](https://www.php.net/manual/en/book.intl.php) is not available.
  1286. */
  1287. public function asOrdinal($value)
  1288. {
  1289. if ($value === null) {
  1290. return $this->nullDisplay;
  1291. }
  1292. $value = $this->normalizeNumericValue($value);
  1293. if ($this->_intlLoaded) {
  1294. $f = $this->createNumberFormatter(NumberFormatter::ORDINAL);
  1295. if (($result = $f->format($value)) === false) {
  1296. throw new InvalidArgumentException('Formatting number as ordinal failed: ' . $f->getErrorCode() . ' ' . $f->getErrorMessage());
  1297. }
  1298. return $result;
  1299. }
  1300. throw new InvalidConfigException('Format as Ordinal is only supported when PHP intl extension is installed.');
  1301. }
  1302. /**
  1303. * Formats the value in bytes as a size in human readable form for example `12 kB`.
  1304. *
  1305. * This is the short form of [[asSize]].
  1306. *
  1307. * If [[sizeFormatBase]] is 1024, [binary prefixes](https://en.wikipedia.org/wiki/Binary_prefix) (e.g. kibibyte/KiB, mebibyte/MiB, ...)
  1308. * are used in the formatting result.
  1309. *
  1310. * @param string|int|float|null $value value in bytes to be formatted.
  1311. * @param int|null $decimals the number of digits after the decimal point.
  1312. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1313. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1314. * @return string the formatted result.
  1315. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1316. * @see sizeFormatBase
  1317. * @see asSize
  1318. */
  1319. public function asShortSize($value, $decimals = null, $options = [], $textOptions = [])
  1320. {
  1321. if ($value === null) {
  1322. return $this->nullDisplay;
  1323. }
  1324. list($params, $position) = $this->formatNumber($value, $decimals, 4, $this->sizeFormatBase, $options, $textOptions);
  1325. if ($this->sizeFormatBase == 1024) {
  1326. switch ($position) {
  1327. case 0:
  1328. return Yii::t('yii', '{nFormatted} B', $params, $this->language);
  1329. case 1:
  1330. return Yii::t('yii', '{nFormatted} KiB', $params, $this->language);
  1331. case 2:
  1332. return Yii::t('yii', '{nFormatted} MiB', $params, $this->language);
  1333. case 3:
  1334. return Yii::t('yii', '{nFormatted} GiB', $params, $this->language);
  1335. case 4:
  1336. return Yii::t('yii', '{nFormatted} TiB', $params, $this->language);
  1337. default:
  1338. return Yii::t('yii', '{nFormatted} PiB', $params, $this->language);
  1339. }
  1340. } else {
  1341. switch ($position) {
  1342. case 0:
  1343. return Yii::t('yii', '{nFormatted} B', $params, $this->language);
  1344. case 1:
  1345. return Yii::t('yii', '{nFormatted} kB', $params, $this->language);
  1346. case 2:
  1347. return Yii::t('yii', '{nFormatted} MB', $params, $this->language);
  1348. case 3:
  1349. return Yii::t('yii', '{nFormatted} GB', $params, $this->language);
  1350. case 4:
  1351. return Yii::t('yii', '{nFormatted} TB', $params, $this->language);
  1352. default:
  1353. return Yii::t('yii', '{nFormatted} PB', $params, $this->language);
  1354. }
  1355. }
  1356. }
  1357. /**
  1358. * Formats the value in bytes as a size in human readable form, for example `12 kilobytes`.
  1359. *
  1360. * If [[sizeFormatBase]] is 1024, [binary prefixes](https://en.wikipedia.org/wiki/Binary_prefix) (e.g. kibibyte/KiB, mebibyte/MiB, ...)
  1361. * are used in the formatting result.
  1362. *
  1363. * @param string|int|float|null $value value in bytes to be formatted.
  1364. * @param int|null $decimals the number of digits after the decimal point.
  1365. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1366. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1367. * @return string the formatted result.
  1368. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1369. * @see sizeFormatBase
  1370. * @see asShortSize
  1371. */
  1372. public function asSize($value, $decimals = null, $options = [], $textOptions = [])
  1373. {
  1374. if ($value === null) {
  1375. return $this->nullDisplay;
  1376. }
  1377. list($params, $position) = $this->formatNumber($value, $decimals, 4, $this->sizeFormatBase, $options, $textOptions);
  1378. if ($this->sizeFormatBase == 1024) {
  1379. switch ($position) {
  1380. case 0:
  1381. return Yii::t('yii', '{nFormatted} {n, plural, =1{byte} other{bytes}}', $params, $this->language);
  1382. case 1:
  1383. return Yii::t('yii', '{nFormatted} {n, plural, =1{kibibyte} other{kibibytes}}', $params, $this->language);
  1384. case 2:
  1385. return Yii::t('yii', '{nFormatted} {n, plural, =1{mebibyte} other{mebibytes}}', $params, $this->language);
  1386. case 3:
  1387. return Yii::t('yii', '{nFormatted} {n, plural, =1{gibibyte} other{gibibytes}}', $params, $this->language);
  1388. case 4:
  1389. return Yii::t('yii', '{nFormatted} {n, plural, =1{tebibyte} other{tebibytes}}', $params, $this->language);
  1390. default:
  1391. return Yii::t('yii', '{nFormatted} {n, plural, =1{pebibyte} other{pebibytes}}', $params, $this->language);
  1392. }
  1393. } else {
  1394. switch ($position) {
  1395. case 0:
  1396. return Yii::t('yii', '{nFormatted} {n, plural, =1{byte} other{bytes}}', $params, $this->language);
  1397. case 1:
  1398. return Yii::t('yii', '{nFormatted} {n, plural, =1{kilobyte} other{kilobytes}}', $params, $this->language);
  1399. case 2:
  1400. return Yii::t('yii', '{nFormatted} {n, plural, =1{megabyte} other{megabytes}}', $params, $this->language);
  1401. case 3:
  1402. return Yii::t('yii', '{nFormatted} {n, plural, =1{gigabyte} other{gigabytes}}', $params, $this->language);
  1403. case 4:
  1404. return Yii::t('yii', '{nFormatted} {n, plural, =1{terabyte} other{terabytes}}', $params, $this->language);
  1405. default:
  1406. return Yii::t('yii', '{nFormatted} {n, plural, =1{petabyte} other{petabytes}}', $params, $this->language);
  1407. }
  1408. }
  1409. }
  1410. /**
  1411. * Formats the value as a length in human readable form for example `12 meters`.
  1412. * Check properties [[baseUnits]] if you need to change unit of value as the multiplier
  1413. * of the smallest unit and [[systemOfUnits]] to switch between [[UNIT_SYSTEM_METRIC]] or [[UNIT_SYSTEM_IMPERIAL]].
  1414. *
  1415. * @param float|int $value value to be formatted.
  1416. * @param int|null $decimals the number of digits after the decimal point.
  1417. * @param array $numberOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1418. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1419. * @return string the formatted result.
  1420. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1421. * @throws InvalidConfigException when INTL is not installed or does not contain required information.
  1422. * @see asLength
  1423. * @since 2.0.13
  1424. * @author John Was <janek.jan@gmail.com>
  1425. */
  1426. public function asLength($value, $decimals = null, $numberOptions = [], $textOptions = [])
  1427. {
  1428. return $this->formatUnit(self::UNIT_LENGTH, self::FORMAT_WIDTH_LONG, $value, $decimals, $numberOptions, $textOptions);
  1429. }
  1430. /**
  1431. * Formats the value as a length in human readable form for example `12 m`.
  1432. * This is the short form of [[asLength]].
  1433. *
  1434. * Check properties [[baseUnits]] if you need to change unit of value as the multiplier
  1435. * of the smallest unit and [[systemOfUnits]] to switch between [[UNIT_SYSTEM_METRIC]] or [[UNIT_SYSTEM_IMPERIAL]].
  1436. *
  1437. * @param float|int $value value to be formatted.
  1438. * @param int|null $decimals the number of digits after the decimal point.
  1439. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1440. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1441. * @return string the formatted result.
  1442. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1443. * @throws InvalidConfigException when INTL is not installed or does not contain required information.
  1444. * @see asLength
  1445. * @since 2.0.13
  1446. * @author John Was <janek.jan@gmail.com>
  1447. */
  1448. public function asShortLength($value, $decimals = null, $options = [], $textOptions = [])
  1449. {
  1450. return $this->formatUnit(self::UNIT_LENGTH, self::FORMAT_WIDTH_SHORT, $value, $decimals, $options, $textOptions);
  1451. }
  1452. /**
  1453. * Formats the value as a weight in human readable form for example `12 kilograms`.
  1454. * Check properties [[baseUnits]] if you need to change unit of value as the multiplier
  1455. * of the smallest unit and [[systemOfUnits]] to switch between [[UNIT_SYSTEM_METRIC]] or [[UNIT_SYSTEM_IMPERIAL]].
  1456. *
  1457. * @param float|int $value value to be formatted.
  1458. * @param int|null $decimals the number of digits after the decimal point.
  1459. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1460. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1461. * @return string the formatted result.
  1462. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1463. * @throws InvalidConfigException when INTL is not installed or does not contain required information.
  1464. * @since 2.0.13
  1465. * @author John Was <janek.jan@gmail.com>
  1466. */
  1467. public function asWeight($value, $decimals = null, $options = [], $textOptions = [])
  1468. {
  1469. return $this->formatUnit(self::UNIT_WEIGHT, self::FORMAT_WIDTH_LONG, $value, $decimals, $options, $textOptions);
  1470. }
  1471. /**
  1472. * Formats the value as a weight in human readable form for example `12 kg`.
  1473. * This is the short form of [[asWeight]].
  1474. *
  1475. * Check properties [[baseUnits]] if you need to change unit of value as the multiplier
  1476. * of the smallest unit and [[systemOfUnits]] to switch between [[UNIT_SYSTEM_METRIC]] or [[UNIT_SYSTEM_IMPERIAL]].
  1477. *
  1478. * @param float|int $value value to be formatted.
  1479. * @param int|null $decimals the number of digits after the decimal point.
  1480. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1481. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1482. * @return string the formatted result.
  1483. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1484. * @throws InvalidConfigException when INTL is not installed or does not contain required information.
  1485. * @since 2.0.13
  1486. * @author John Was <janek.jan@gmail.com>
  1487. */
  1488. public function asShortWeight($value, $decimals = null, $options = [], $textOptions = [])
  1489. {
  1490. return $this->formatUnit(self::UNIT_WEIGHT, self::FORMAT_WIDTH_SHORT, $value, $decimals, $options, $textOptions);
  1491. }
  1492. /**
  1493. * @param string $unitType one of [[UNIT_WEIGHT]], [[UNIT_LENGTH]]
  1494. * @param string $unitFormat one of [[FORMAT_WIDTH_SHORT]], [[FORMAT_WIDTH_LONG]]
  1495. * @param float|int|null $value to be formatted
  1496. * @param int|null $decimals the number of digits after the decimal point.
  1497. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1498. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1499. * @return string
  1500. * @throws InvalidConfigException when INTL is not installed or does not contain required information
  1501. */
  1502. private function formatUnit($unitType, $unitFormat, $value, $decimals, $options, $textOptions)
  1503. {
  1504. if ($value === null) {
  1505. return $this->nullDisplay;
  1506. }
  1507. $multipliers = array_values($this->measureUnits[$unitType][$this->systemOfUnits]);
  1508. list($params, $position) = $this->formatNumber(
  1509. $this->normalizeNumericValue($value) * $this->baseUnits[$unitType][$this->systemOfUnits],
  1510. $decimals,
  1511. null,
  1512. $multipliers,
  1513. $options,
  1514. $textOptions
  1515. );
  1516. $message = $this->getUnitMessage($unitType, $unitFormat, $this->systemOfUnits, $position);
  1517. return (new \MessageFormatter($this->locale, $message))->format([
  1518. '0' => $params['nFormatted'],
  1519. 'n' => $params['n'],
  1520. ]);
  1521. }
  1522. /**
  1523. * @param string $unitType one of [[UNIT_WEIGHT]], [[UNIT_LENGTH]]
  1524. * @param string $unitFormat one of [[FORMAT_WIDTH_SHORT]], [[FORMAT_WIDTH_LONG]]
  1525. * @param string|null $system either [[UNIT_SYSTEM_METRIC]] or [[UNIT_SYSTEM_IMPERIAL]]. When `null`, property [[systemOfUnits]] will be used.
  1526. * @param int $position internal position of size unit
  1527. * @return string
  1528. * @throws InvalidConfigException when INTL is not installed or does not contain required information
  1529. */
  1530. private function getUnitMessage($unitType, $unitFormat, $system, $position)
  1531. {
  1532. if (isset($this->_unitMessages[$unitType][$unitFormat][$system][$position])) {
  1533. return $this->_unitMessages[$unitType][$unitFormat][$system][$position];
  1534. }
  1535. if (!$this->_intlLoaded) {
  1536. throw new InvalidConfigException('Format of ' . $unitType . ' is only supported when PHP intl extension is installed.');
  1537. }
  1538. if ($this->_resourceBundle === null) {
  1539. try {
  1540. $this->_resourceBundle = new \ResourceBundle($this->locale, 'ICUDATA-unit');
  1541. } catch (\IntlException $e) {
  1542. throw new InvalidConfigException('Current ICU data does not contain information about measure units. Check system requirements.');
  1543. }
  1544. }
  1545. $unitNames = array_keys($this->measureUnits[$unitType][$system]);
  1546. $bundleKey = 'units' . ($unitFormat === self::FORMAT_WIDTH_SHORT ? 'Short' : '');
  1547. $unitBundle = $this->_resourceBundle[$bundleKey][$unitType][$unitNames[$position]];
  1548. if ($unitBundle === null) {
  1549. throw new InvalidConfigException(
  1550. 'Current ICU data version does not contain information about unit type "' . $unitType
  1551. . '" and unit measure "' . $unitNames[$position] . '". Check system requirements.'
  1552. );
  1553. }
  1554. $message = [];
  1555. foreach ($unitBundle as $key => $value) {
  1556. if ($key === 'dnam') {
  1557. continue;
  1558. }
  1559. $message[] = "$key{{$value}}";
  1560. }
  1561. return $this->_unitMessages[$unitType][$unitFormat][$system][$position] = '{n, plural, ' . implode(' ', $message) . '}';
  1562. }
  1563. /**
  1564. * Given the value in bytes formats number part of the human readable form.
  1565. *
  1566. * @param string|int|float $value value in bytes to be formatted.
  1567. * @param int|null $decimals the number of digits after the decimal point
  1568. * @param int $maxPosition maximum internal position of size unit, ignored if $formatBase is an array
  1569. * @param array|int $formatBase the base at which each next unit is calculated, either 1000 or 1024, or an array
  1570. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1571. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1572. * @return array [parameters for Yii::t containing formatted number, internal position of size unit]
  1573. * @throws InvalidArgumentException if the input value is not numeric or the formatting failed.
  1574. * @since 2.0.32
  1575. */
  1576. protected function formatNumber($value, $decimals, $maxPosition, $formatBase, $options, $textOptions)
  1577. {
  1578. $value = $this->normalizeNumericValue($value);
  1579. $position = 0;
  1580. if (is_array($formatBase)) {
  1581. $maxPosition = count($formatBase) - 1;
  1582. }
  1583. do {
  1584. if (is_array($formatBase)) {
  1585. if (!isset($formatBase[$position + 1])) {
  1586. break;
  1587. }
  1588. if (abs($value) < $formatBase[$position + 1]) {
  1589. break;
  1590. }
  1591. } else {
  1592. if (abs($value) < $formatBase) {
  1593. break;
  1594. }
  1595. $value /= $formatBase;
  1596. }
  1597. $position++;
  1598. } while ($position < $maxPosition + 1);
  1599. if (is_array($formatBase) && $position !== 0) {
  1600. $value /= $formatBase[$position];
  1601. }
  1602. // no decimals for smallest unit
  1603. if ($position === 0) {
  1604. $decimals = 0;
  1605. } elseif ($decimals !== null) {
  1606. $value = round($value, $decimals);
  1607. }
  1608. // disable grouping for edge cases like 1023 to get 1023 B instead of 1,023 B
  1609. $oldThousandSeparator = $this->thousandSeparator;
  1610. $this->thousandSeparator = '';
  1611. if ($this->_intlLoaded && !isset($options[NumberFormatter::GROUPING_USED])) {
  1612. $options[NumberFormatter::GROUPING_USED] = 0;
  1613. }
  1614. // format the size value
  1615. $params = [
  1616. // this is the unformatted number used for the plural rule
  1617. // abs() to make sure the plural rules work correctly on negative numbers, intl does not cover this
  1618. // https://english.stackexchange.com/questions/9735/is-1-followed-by-a-singular-or-plural-noun
  1619. 'n' => abs($value),
  1620. // this is the formatted number used for display
  1621. 'nFormatted' => $this->asDecimal($value, $decimals, $options, $textOptions),
  1622. ];
  1623. $this->thousandSeparator = $oldThousandSeparator;
  1624. return [$params, $position];
  1625. }
  1626. /**
  1627. * Normalizes a numeric input value.
  1628. *
  1629. * - everything [empty](https://www.php.net/manual/en/function.empty.php) will result in `0`
  1630. * - a [numeric](https://www.php.net/manual/en/function.is-numeric.php) string will be casted to float
  1631. * - everything else will be returned if it is [numeric](https://www.php.net/manual/en/function.is-numeric.php),
  1632. * otherwise an exception is thrown.
  1633. *
  1634. * @param mixed $value the input value
  1635. * @return float|int the normalized number value
  1636. * @throws InvalidArgumentException if the input value is not numeric.
  1637. */
  1638. protected function normalizeNumericValue($value)
  1639. {
  1640. if (empty($value)) {
  1641. return 0;
  1642. }
  1643. if (is_string($value) && is_numeric($value)) {
  1644. $value = (float) $value;
  1645. }
  1646. if (!is_numeric($value)) {
  1647. throw new InvalidArgumentException("'$value' is not a numeric value.");
  1648. }
  1649. return $value;
  1650. }
  1651. /**
  1652. * Creates a number formatter based on the given type and format.
  1653. *
  1654. * You may override this method to create a number formatter based on patterns.
  1655. *
  1656. * @param int $style the type of the number formatter.
  1657. * Values: NumberFormatter::DECIMAL, ::CURRENCY, ::PERCENT, ::SCIENTIFIC, ::SPELLOUT, ::ORDINAL
  1658. * ::DURATION, ::PATTERN_RULEBASED, ::DEFAULT_STYLE, ::IGNORE
  1659. * @param int|null $decimals the number of digits after the decimal point.
  1660. * @param array $options optional configuration for the number formatter. This parameter will be merged with [[numberFormatterOptions]].
  1661. * @param array $textOptions optional configuration for the number formatter. This parameter will be merged with [[numberFormatterTextOptions]].
  1662. * @return NumberFormatter the created formatter instance
  1663. */
  1664. protected function createNumberFormatter($style, $decimals = null, $options = [], $textOptions = [])
  1665. {
  1666. $formatter = new NumberFormatter($this->locale, $style);
  1667. // set text attributes
  1668. foreach ($this->numberFormatterTextOptions as $attribute => $value) {
  1669. $this->setFormatterTextAttribute($formatter, $attribute, $value, 'numberFormatterTextOptions', 'numberFormatterOptions');
  1670. }
  1671. foreach ($textOptions as $attribute => $value) {
  1672. $this->setFormatterTextAttribute($formatter, $attribute, $value, '$textOptions', '$options');
  1673. }
  1674. // set attributes
  1675. foreach ($this->numberFormatterOptions as $attribute => $value) {
  1676. $this->setFormatterIntAttribute($formatter, $attribute, $value, 'numberFormatterOptions', 'numberFormatterTextOptions');
  1677. }
  1678. foreach ($options as $attribute => $value) {
  1679. $this->setFormatterIntAttribute($formatter, $attribute, $value, '$options', '$textOptions');
  1680. }
  1681. if ($decimals !== null) {
  1682. $formatter->setAttribute(NumberFormatter::MAX_FRACTION_DIGITS, $decimals);
  1683. $formatter->setAttribute(NumberFormatter::MIN_FRACTION_DIGITS, $decimals);
  1684. }
  1685. // set symbols
  1686. if ($this->decimalSeparator !== null) {
  1687. $formatter->setSymbol(NumberFormatter::DECIMAL_SEPARATOR_SYMBOL, $this->decimalSeparator);
  1688. }
  1689. if ($this->currencyDecimalSeparator !== null) {
  1690. $formatter->setSymbol(NumberFormatter::MONETARY_SEPARATOR_SYMBOL, $this->currencyDecimalSeparator);
  1691. }
  1692. if ($this->thousandSeparator !== null) {
  1693. $formatter->setSymbol(NumberFormatter::GROUPING_SEPARATOR_SYMBOL, $this->thousandSeparator);
  1694. $formatter->setSymbol(NumberFormatter::MONETARY_GROUPING_SEPARATOR_SYMBOL, $this->thousandSeparator);
  1695. }
  1696. foreach ($this->numberFormatterSymbols as $symbol => $value) {
  1697. $this->setFormatterSymbol($formatter, $symbol, $value, 'numberFormatterSymbols');
  1698. }
  1699. return $formatter;
  1700. }
  1701. /**
  1702. * @param NumberFormatter $formatter
  1703. * @param mixed $attribute
  1704. * @param mixed $value
  1705. * @param string $source
  1706. * @param string $alternative
  1707. */
  1708. private function setFormatterTextAttribute($formatter, $attribute, $value, $source, $alternative)
  1709. {
  1710. if (!is_int($attribute)) {
  1711. throw new InvalidArgumentException(
  1712. "The $source array keys must be integers recognizable by NumberFormatter::setTextAttribute(). \""
  1713. . gettype($attribute) . '" provided instead.'
  1714. );
  1715. }
  1716. if (!is_string($value)) {
  1717. if (is_int($value)) {
  1718. throw new InvalidArgumentException(
  1719. "The $source array values must be strings. Did you mean to use $alternative?"
  1720. );
  1721. }
  1722. throw new InvalidArgumentException(
  1723. "The $source array values must be strings. \"" . gettype($value) . '" provided instead.'
  1724. );
  1725. }
  1726. $formatter->setTextAttribute($attribute, $value);
  1727. }
  1728. /**
  1729. * @param NumberFormatter $formatter
  1730. * @param mixed $symbol
  1731. * @param mixed $value
  1732. * @param string $source
  1733. */
  1734. private function setFormatterSymbol($formatter, $symbol, $value, $source)
  1735. {
  1736. if (!is_int($symbol)) {
  1737. throw new InvalidArgumentException(
  1738. "The $source array keys must be integers recognizable by NumberFormatter::setSymbol(). \""
  1739. . gettype($symbol) . '" provided instead.'
  1740. );
  1741. }
  1742. if (!is_string($value)) {
  1743. throw new InvalidArgumentException(
  1744. "The $source array values must be strings. \"" . gettype($value) . '" provided instead.'
  1745. );
  1746. }
  1747. $formatter->setSymbol($symbol, $value);
  1748. }
  1749. /**
  1750. * @param NumberFormatter $formatter
  1751. * @param mixed $attribute
  1752. * @param mixed $value
  1753. * @param string $source
  1754. * @param string $alternative
  1755. */
  1756. private function setFormatterIntAttribute($formatter, $attribute, $value, $source, $alternative)
  1757. {
  1758. if (!is_int($attribute)) {
  1759. throw new InvalidArgumentException(
  1760. "The $source array keys must be integers recognizable by NumberFormatter::setAttribute(). \""
  1761. . gettype($attribute) . '" provided instead.'
  1762. );
  1763. }
  1764. if (!is_int($value)) {
  1765. if (is_string($value)) {
  1766. throw new InvalidArgumentException(
  1767. "The $source array values must be integers. Did you mean to use $alternative?"
  1768. );
  1769. }
  1770. throw new InvalidArgumentException(
  1771. "The $source array values must be integers. \"" . gettype($value) . '" provided instead.'
  1772. );
  1773. }
  1774. $formatter->setAttribute($attribute, $value);
  1775. }
  1776. /**
  1777. * Checks if string representations of given value and its normalized version are different.
  1778. * @param string|float|int $value
  1779. * @param float|int $normalizedValue
  1780. * @return bool
  1781. * @since 2.0.16
  1782. */
  1783. protected function isNormalizedValueMispresented($value, $normalizedValue)
  1784. {
  1785. if (empty($value)) {
  1786. $value = 0;
  1787. }
  1788. return (string) $normalizedValue !== $this->normalizeNumericStringValue((string) $value);
  1789. }
  1790. /**
  1791. * Normalizes a numeric string value.
  1792. * @param string $value
  1793. * @return string the normalized number value as a string
  1794. * @since 2.0.16
  1795. */
  1796. protected function normalizeNumericStringValue($value)
  1797. {
  1798. $powerPosition = strrpos($value, 'E');
  1799. if ($powerPosition !== false) {
  1800. $valuePart = substr($value, 0, $powerPosition);
  1801. $powerPart = substr($value, $powerPosition + 1);
  1802. } else {
  1803. $powerPart = null;
  1804. $valuePart = $value;
  1805. }
  1806. $separatorPosition = strrpos($valuePart, '.');
  1807. if ($separatorPosition !== false) {
  1808. $integerPart = substr($valuePart, 0, $separatorPosition);
  1809. $fractionalPart = substr($valuePart, $separatorPosition + 1);
  1810. } else {
  1811. $integerPart = $valuePart;
  1812. $fractionalPart = null;
  1813. }
  1814. // truncate insignificant zeros, keep minus
  1815. $integerPart = preg_replace('/^\+?(-?)0*(\d+)$/', '$1$2', $integerPart);
  1816. // for zeros only leave one zero, keep minus
  1817. $integerPart = preg_replace('/^\+?(-?)0*$/', '${1}0', $integerPart);
  1818. if ($fractionalPart !== null) {
  1819. // truncate insignificant zeros
  1820. $fractionalPart = rtrim($fractionalPart, '0');
  1821. if (empty($fractionalPart)) {
  1822. $fractionalPart = $powerPart !== null ? '0' : null;
  1823. }
  1824. }
  1825. $normalizedValue = $integerPart;
  1826. if ($fractionalPart !== null) {
  1827. $normalizedValue .= '.' . $fractionalPart;
  1828. } elseif ($normalizedValue === '-0') {
  1829. $normalizedValue = '0';
  1830. }
  1831. if ($powerPart !== null) {
  1832. $normalizedValue .= 'E' . $powerPart;
  1833. }
  1834. return $normalizedValue;
  1835. }
  1836. /**
  1837. * Fallback for formatting value as a decimal number.
  1838. *
  1839. * Property [[decimalSeparator]] will be used to represent the decimal point. The value is rounded automatically
  1840. * to the defined decimal digits.
  1841. *
  1842. * @param string|int|float $value the value to be formatted.
  1843. * @param int|null $decimals the number of digits after the decimal point. The default value is `2`.
  1844. * @return string the formatted result.
  1845. * @see decimalSeparator
  1846. * @see thousandSeparator
  1847. * @since 2.0.16
  1848. */
  1849. protected function asDecimalStringFallback($value, $decimals = 2)
  1850. {
  1851. if (empty($value)) {
  1852. $value = 0;
  1853. }
  1854. $value = $this->normalizeNumericStringValue((string) $value);
  1855. $separatorPosition = strrpos($value, '.');
  1856. if ($separatorPosition !== false) {
  1857. $integerPart = substr($value, 0, $separatorPosition);
  1858. $fractionalPart = substr($value, $separatorPosition + 1);
  1859. } else {
  1860. $integerPart = $value;
  1861. $fractionalPart = null;
  1862. }
  1863. $decimalOutput = '';
  1864. if ($decimals === null) {
  1865. $decimals = 2;
  1866. }
  1867. $carry = 0;
  1868. if ($decimals > 0) {
  1869. $decimalSeparator = $this->decimalSeparator;
  1870. if ($this->decimalSeparator === null) {
  1871. $decimalSeparator = '.';
  1872. }
  1873. if ($fractionalPart === null) {
  1874. $fractionalPart = str_repeat('0', $decimals);
  1875. } elseif (strlen($fractionalPart) > $decimals) {
  1876. $cursor = $decimals;
  1877. // checking if fractional part must be rounded
  1878. if ((int) substr($fractionalPart, $cursor, 1) >= 5) {
  1879. while (--$cursor >= 0) {
  1880. $carry = 0;
  1881. $oneUp = (int) substr($fractionalPart, $cursor, 1) + 1;
  1882. if ($oneUp === 10) {
  1883. $oneUp = 0;
  1884. $carry = 1;
  1885. }
  1886. $fractionalPart = substr($fractionalPart, 0, $cursor) . $oneUp . substr($fractionalPart, $cursor + 1);
  1887. if ($carry === 0) {
  1888. break;
  1889. }
  1890. }
  1891. }
  1892. $fractionalPart = substr($fractionalPart, 0, $decimals);
  1893. } elseif (strlen($fractionalPart) < $decimals) {
  1894. $fractionalPart = str_pad($fractionalPart, $decimals, '0');
  1895. }
  1896. $decimalOutput .= $decimalSeparator . $fractionalPart;
  1897. }
  1898. // checking if integer part must be rounded
  1899. if ($carry || ($decimals === 0 && $fractionalPart !== null && (int) substr($fractionalPart, 0, 1) >= 5)) {
  1900. $integerPartLength = strlen($integerPart);
  1901. $cursor = 0;
  1902. while (++$cursor <= $integerPartLength) {
  1903. $carry = 0;
  1904. $oneUp = (int) substr($integerPart, -$cursor, 1) + 1;
  1905. if ($oneUp === 10) {
  1906. $oneUp = 0;
  1907. $carry = 1;
  1908. }
  1909. $integerPart = substr($integerPart, 0, -$cursor) . $oneUp . substr($integerPart, $integerPartLength - $cursor + 1);
  1910. if ($carry === 0) {
  1911. break;
  1912. }
  1913. }
  1914. if ($carry === 1) {
  1915. $integerPart = '1' . $integerPart;
  1916. }
  1917. }
  1918. if (strlen($integerPart) > 3) {
  1919. $thousandSeparator = $this->thousandSeparator;
  1920. if ($thousandSeparator === null) {
  1921. $thousandSeparator = ',';
  1922. }
  1923. $integerPart = strrev(implode(',', str_split(strrev($integerPart), 3)));
  1924. if ($thousandSeparator !== ',') {
  1925. $integerPart = str_replace(',', $thousandSeparator, $integerPart);
  1926. }
  1927. }
  1928. return $integerPart . $decimalOutput;
  1929. }
  1930. /**
  1931. * Fallback for formatting value as an integer number by removing any decimal digits without rounding.
  1932. *
  1933. * @param string|int|float $value the value to be formatted.
  1934. * @return string the formatted result.
  1935. * @since 2.0.16
  1936. */
  1937. protected function asIntegerStringFallback($value)
  1938. {
  1939. if (empty($value)) {
  1940. $value = 0;
  1941. }
  1942. $value = $this->normalizeNumericStringValue((string) $value);
  1943. $separatorPosition = strrpos($value, '.');
  1944. if ($separatorPosition !== false) {
  1945. $integerPart = substr($value, 0, $separatorPosition);
  1946. } else {
  1947. $integerPart = $value;
  1948. }
  1949. return $this->asDecimalStringFallback($integerPart, 0);
  1950. }
  1951. /**
  1952. * Fallback for formatting value as a percent number with "%" sign.
  1953. *
  1954. * Property [[decimalSeparator]] will be used to represent the decimal point. The value is rounded automatically
  1955. * to the defined decimal digits.
  1956. *
  1957. * @param string|int|float $value the value to be formatted.
  1958. * @param int|null $decimals the number of digits after the decimal point. The default value is `0`.
  1959. * @return string the formatted result.
  1960. * @since 2.0.16
  1961. */
  1962. protected function asPercentStringFallback($value, $decimals = null)
  1963. {
  1964. if (empty($value)) {
  1965. $value = 0;
  1966. }
  1967. if ($decimals === null) {
  1968. $decimals = 0;
  1969. }
  1970. $value = $this->normalizeNumericStringValue((string) $value);
  1971. $separatorPosition = strrpos($value, '.');
  1972. if ($separatorPosition !== false) {
  1973. $integerPart = substr($value, 0, $separatorPosition);
  1974. $fractionalPart = str_pad(substr($value, $separatorPosition + 1), 2, '0');
  1975. $integerPart .= substr($fractionalPart, 0, 2);
  1976. $fractionalPart = substr($fractionalPart, 2);
  1977. if ($fractionalPart === '') {
  1978. $multipliedValue = $integerPart;
  1979. } else {
  1980. $multipliedValue = $integerPart . '.' . $fractionalPart;
  1981. }
  1982. } else {
  1983. $multipliedValue = $value . '00';
  1984. }
  1985. return $this->asDecimalStringFallback($multipliedValue, $decimals) . '%';
  1986. }
  1987. /**
  1988. * Fallback for formatting value as a currency number.
  1989. *
  1990. * @param string|int|float $value the value to be formatted.
  1991. * @param string|null $currency the 3-letter ISO 4217 currency code indicating the currency to use.
  1992. * If null, [[currencyCode]] will be used.
  1993. * @return string the formatted result.
  1994. * @throws InvalidConfigException if no currency is given and [[currencyCode]] is not defined.
  1995. * @since 2.0.16
  1996. */
  1997. protected function asCurrencyStringFallback($value, $currency = null)
  1998. {
  1999. if ($currency === null) {
  2000. if ($this->currencyCode === null) {
  2001. throw new InvalidConfigException('The default currency code for the formatter is not defined.');
  2002. }
  2003. $currency = $this->currencyCode;
  2004. }
  2005. return $currency . ' ' . $this->asDecimalStringFallback($value, 2);
  2006. }
  2007. }