/Languages/IronPython/IronPython/Runtime/FormattingHelper.cs

http://github.com/IronLanguages/main · C# · 149 lines · 93 code · 12 blank · 44 comment · 27 complexity · 57d057d1021808684ca679e0a83842bc MD5 · raw file

  1. /* ****************************************************************************
  2. *
  3. * Copyright (c) Microsoft Corporation.
  4. *
  5. * This source code is subject to terms and conditions of the Apache License, Version 2.0. A
  6. * copy of the license can be found in the License.html file at the root of this distribution. If
  7. * you cannot locate the Apache License, Version 2.0, please send an email to
  8. * dlr@microsoft.com. By using this source code in any fashion, you are agreeing to be bound
  9. * by the terms of the Apache License, Version 2.0.
  10. *
  11. * You must not remove this notice, or any other, from this software.
  12. *
  13. *
  14. * ***************************************************************************/
  15. using System;
  16. using System.Collections.Generic;
  17. using System.Globalization;
  18. using System.Linq;
  19. using System.Text;
  20. using System.Threading;
  21. namespace IronPython.Runtime
  22. {
  23. internal static class FormattingHelper
  24. {
  25. private static NumberFormatInfo _invariantCommaSeperatorInfo;
  26. /// <summary>
  27. /// Helper NumberFormatInfo for use by int/BigInteger __format__ routines
  28. /// for width specified leading zero support that contains ','s every 3 digits.
  29. /// i.e. For use by d/g/G format specifiers. NOT for use by n format specifiers.
  30. /// </summary>
  31. public static NumberFormatInfo InvariantCommaNumberInfo {
  32. get {
  33. if (_invariantCommaSeperatorInfo == null) {
  34. Interlocked.CompareExchange(
  35. ref _invariantCommaSeperatorInfo,
  36. new NumberFormatInfo()
  37. {
  38. NumberGroupSeparator = ",",
  39. NumberDecimalSeparator = ".",
  40. NumberGroupSizes = new int[] {3}
  41. },
  42. null
  43. );
  44. }
  45. return _invariantCommaSeperatorInfo;
  46. }
  47. }
  48. public static string/*!*/ ToCultureString<T>(T/*!*/ val, NumberFormatInfo/*!*/ nfi, StringFormatSpec spec) {
  49. string digits;
  50. string separator = nfi.NumberGroupSeparator;
  51. int[] separatorLocations = nfi.NumberGroupSizes;
  52. digits = val.ToString();
  53. // If we're adding leading zeros, we need to know how
  54. // many we need.
  55. int width = spec.Width ?? 0;
  56. int fillerLength = Math.Max(width - digits.Length, 0);
  57. bool addLeadingZeros = (spec.Fill ?? '\0') == '0' && width > digits.Length;
  58. int beginningOfDigits = fillerLength;
  59. int remainingWidth = 0;
  60. if (addLeadingZeros) {
  61. // If we're adding leading zeros, add more than necessary
  62. // we'll trim off the extra (if any) later.
  63. digits = digits.Insert(0, new string('0', fillerLength));
  64. }
  65. if (separatorLocations.Length > 0) {
  66. StringBuilder res = new StringBuilder(digits);
  67. int curGroup = 0, curDigit = digits.Length - 1;
  68. while (curDigit > 0) {
  69. // insert the separator
  70. int groupLen = separatorLocations[curGroup];
  71. if (groupLen == 0) {
  72. break;
  73. }
  74. curDigit -= groupLen;
  75. if (curDigit >= 0) {
  76. res.Insert(curDigit + 1, separator);
  77. // Once we have advanced left of the last of
  78. // the original digits, we need to adjust the
  79. // index that tracks the first original digit index.
  80. if (addLeadingZeros && curDigit < fillerLength) {
  81. beginningOfDigits++;
  82. // The remaining width is the format width minus the length
  83. // of the expanded original digits:
  84. remainingWidth = Math.Max(width - (res.Length - beginningOfDigits), 0);
  85. // If we've run out of room, then no need to insert
  86. // anymore commas into leading zeros.
  87. if (remainingWidth == 0) {
  88. break;
  89. }
  90. }
  91. }
  92. // advance the group
  93. if (curGroup + 1 < separatorLocations.Length) {
  94. if (separatorLocations[curGroup + 1] == 0) {
  95. // last value doesn't propagate
  96. break;
  97. }
  98. curGroup++;
  99. }
  100. }
  101. if (addLeadingZeros && res.Length > width) {
  102. // The remaining width is the format width minus the length
  103. // of the expanded original digits:
  104. remainingWidth = Math.Max(width - (res.Length - beginningOfDigits), 0);
  105. if (remainingWidth > 0) {
  106. // Index that points at the beginning of the requested width:
  107. var beginningOfMaximumWidth = beginningOfDigits - remainingWidth;
  108. // If the maximum width stops at a character that is part of the
  109. // separator then keep looking to the left until we find a character
  110. // that isn't. After all, it would be pretty weird to produce:
  111. // 000,xxx,xxx,xxx. So, produce 0,000,xxx,xxx,xxx instead.
  112. // (Just like CPython)
  113. if (separator.IndexOf(res[beginningOfMaximumWidth]) != -1) {
  114. for (int i = beginningOfMaximumWidth - 1; i >= 0; i--) {
  115. if (separator.IndexOf(res[i]) == -1) {
  116. res.Remove(0, i);
  117. break;
  118. }
  119. }
  120. }
  121. else {
  122. res.Remove(0, beginningOfMaximumWidth);
  123. }
  124. }
  125. else {
  126. // If we ran out of remainingWidth just formatting
  127. // the actual digits, then remove any extra leading zeros
  128. // we added.
  129. res.Remove(0, beginningOfDigits);
  130. }
  131. }
  132. digits = res.ToString();
  133. }
  134. return digits;
  135. }
  136. }
  137. }