ArrayInstance.cs 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. using System.Linq;
  2. using Jint.Native.Object;
  3. using Jint.Runtime;
  4. using Jint.Runtime.Descriptors;
  5. namespace Jint.Native.Array
  6. {
  7. public class ArrayInstance : ObjectInstance
  8. {
  9. private readonly Engine _engine;
  10. public ArrayInstance(Engine engine) : base(engine)
  11. {
  12. _engine = engine;
  13. }
  14. public override string Class
  15. {
  16. get
  17. {
  18. return "Array";
  19. }
  20. }
  21. public override bool DefineOwnProperty(string propertyName, PropertyDescriptor desc, bool throwOnError)
  22. {
  23. var oldLenDesc = GetOwnProperty("length").As<DataDescriptor>();
  24. var oldLen = TypeConverter.ToNumber(oldLenDesc.Value);
  25. if (propertyName == "length")
  26. {
  27. var descData = desc as DataDescriptor;
  28. if (descData == null)
  29. {
  30. return base.DefineOwnProperty("length", desc, throwOnError);
  31. }
  32. var newLenDesc = new DataDescriptor(desc);
  33. uint newLen = TypeConverter.ToUint32(descData.Value);
  34. if (newLen != TypeConverter.ToNumber(descData.Value))
  35. {
  36. throw new JavaScriptException(_engine.RangeError);
  37. }
  38. newLenDesc.Value = newLen;
  39. if (newLen >= oldLen)
  40. {
  41. return base.DefineOwnProperty("length", newLenDesc, throwOnError);
  42. }
  43. if (!oldLenDesc.WritableIsSet)
  44. {
  45. if (throwOnError)
  46. {
  47. throw new JavaScriptException(_engine.TypeError);
  48. }
  49. return false;
  50. }
  51. var newWritable = true;
  52. if (!newLenDesc.WritableIsSet)
  53. {
  54. newWritable = false;
  55. newLenDesc.Writable = true;
  56. }
  57. var succeeded = base.DefineOwnProperty("length", newLenDesc, throwOnError);
  58. if (!succeeded)
  59. {
  60. return false;
  61. }
  62. // in the case of sparse arrays, treat each concrete element instead of
  63. // iterating over all indexes
  64. if (Properties.Count < oldLen - newLen)
  65. {
  66. var keys = Properties.Keys.ToArray();
  67. foreach (var key in keys)
  68. {
  69. uint index;
  70. // is it the index of the array
  71. if (uint.TryParse(key, out index) && index >= newLen && index < oldLen)
  72. {
  73. var deleteSucceeded = Delete(key, false);
  74. if (!deleteSucceeded)
  75. {
  76. newLenDesc.Value = index + 1;
  77. if (!newWritable)
  78. {
  79. newLenDesc.Writable = false;
  80. }
  81. base.DefineOwnProperty("length", newLenDesc, false);
  82. if (throwOnError)
  83. {
  84. throw new JavaScriptException(_engine.TypeError);
  85. }
  86. return false;
  87. }
  88. }
  89. }
  90. }
  91. else
  92. {
  93. while (newLen < oldLen)
  94. {
  95. // algorithm as per the spec
  96. oldLen--;
  97. var deleteSucceeded = Delete(TypeConverter.ToString(oldLen), false);
  98. if (!deleteSucceeded)
  99. {
  100. newLenDesc.Value = oldLen + 1;
  101. if (!newWritable)
  102. {
  103. newLenDesc.Writable = false;
  104. }
  105. base.DefineOwnProperty("length", newLenDesc, false);
  106. if (throwOnError)
  107. {
  108. throw new JavaScriptException(_engine.TypeError);
  109. }
  110. return false;
  111. }
  112. }
  113. }
  114. if (!newWritable)
  115. {
  116. DefineOwnProperty("length", new DataDescriptor(null) {Writable = false}, false);
  117. }
  118. return true;
  119. }
  120. else if (IsArrayIndex(propertyName))
  121. {
  122. var index = TypeConverter.ToUint32(propertyName);
  123. if (index >= oldLen && !oldLenDesc.WritableIsSet)
  124. {
  125. if (throwOnError)
  126. {
  127. throw new JavaScriptException(_engine.TypeError);
  128. }
  129. return false;
  130. }
  131. var succeeded = base.DefineOwnProperty(propertyName, desc, false);
  132. if (!succeeded)
  133. {
  134. if (throwOnError)
  135. {
  136. throw new JavaScriptException(_engine.TypeError);
  137. }
  138. return false;
  139. }
  140. if (index >= oldLen)
  141. {
  142. oldLenDesc.Value = index + 1;
  143. base.DefineOwnProperty("length", oldLenDesc, false);
  144. }
  145. return true;
  146. }
  147. return base.DefineOwnProperty(propertyName, desc, false);
  148. }
  149. public static bool IsArrayIndex(object p)
  150. {
  151. return TypeConverter.ToString(TypeConverter.ToUint32(p)) == TypeConverter.ToString(p) && TypeConverter.ToUint32(p) != uint.MaxValue;
  152. }
  153. }
  154. }