summaryrefslogtreecommitdiff
path: root/Userland
diff options
context:
space:
mode:
authorIdan Horowitz <idan.horowitz@gmail.com>2021-07-08 04:39:37 +0300
committerLinus Groh <mail@linusgroh.de>2021-07-08 16:30:09 +0100
commitf0abcde00c457f972acb626d8d3e2b9272c2f0b8 (patch)
treee6ecad0925c4782930e01568377207b6e4fd99e6 /Userland
parent6343bfa9d7ae24b078ffc00a312a413bd45e0422 (diff)
downloadserenity-f0abcde00c457f972acb626d8d3e2b9272c2f0b8.zip
LibJS: Add %TypedArray%.prototype.lastIndexOf
Diffstat (limited to 'Userland')
-rw-r--r--Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.cpp49
-rw-r--r--Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.h1
-rw-r--r--Userland/Libraries/LibJS/Tests/builtins/TypedArray/TypedArray.prototype.lastIndexOf.js44
3 files changed, 94 insertions, 0 deletions
diff --git a/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.cpp b/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.cpp
index 6e984d93c8..693a3db543 100644
--- a/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.cpp
+++ b/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.cpp
@@ -36,6 +36,7 @@ void TypedArrayPrototype::initialize(GlobalObject& object)
define_native_function(vm.names.forEach, for_each, 1, attr);
define_native_function(vm.names.includes, includes, 1, attr);
define_native_function(vm.names.indexOf, index_of, 1, attr);
+ define_native_function(vm.names.lastIndexOf, last_index_of, 1, attr);
define_native_function(vm.names.some, some, 1, attr);
define_native_function(vm.names.join, join, 1, attr);
define_native_function(vm.names.keys, keys, 0, attr);
@@ -355,6 +356,54 @@ JS_DEFINE_NATIVE_FUNCTION(TypedArrayPrototype::index_of)
return Value(-1);
}
+// 23.2.3.17 %TypedArray%.prototype.lastIndexOf ( searchElement [ , fromIndex ] ), https://tc39.es/ecma262/#sec-%typedarray%.prototype.lastindexof
+JS_DEFINE_NATIVE_FUNCTION(TypedArrayPrototype::last_index_of)
+{
+ auto typed_array = typed_array_from(vm, global_object);
+ if (!typed_array)
+ return {};
+
+ auto length = typed_array->array_length();
+
+ if (length == 0)
+ return Value(-1);
+
+ double n;
+ if (vm.argument_count() > 1) {
+ n = vm.argument(1).to_integer_or_infinity(global_object);
+ if (vm.exception())
+ return {};
+ } else {
+ n = length - 1;
+ }
+
+ if (Value(n).is_negative_infinity())
+ return Value(-1);
+
+ i32 k;
+ if (n >= 0) {
+ k = min(n, (i32)length - 1);
+ } else {
+ auto relative_k = length + n;
+ if (relative_k < 0) // ensures we dont underflow `k`
+ relative_k = -1;
+ k = relative_k;
+ }
+
+ auto search_element = vm.argument(0);
+ for (; k >= 0; --k) {
+ auto k_present = typed_array->has_property(k);
+ if (k_present) {
+ auto element_k = typed_array->get(k);
+
+ if (strict_eq(search_element, element_k))
+ return Value(k);
+ }
+ }
+
+ return Value(-1);
+}
+
// 23.2.3.25 %TypedArray%.prototype.some ( callbackfn [ , thisArg ] ), https://tc39.es/ecma262/#sec-%typedarray%.prototype.some
JS_DEFINE_NATIVE_FUNCTION(TypedArrayPrototype::some)
{
diff --git a/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.h b/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.h
index 4c6674e959..6485d1829e 100644
--- a/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.h
+++ b/Userland/Libraries/LibJS/Runtime/TypedArrayPrototype.h
@@ -33,6 +33,7 @@ private:
JS_DECLARE_NATIVE_FUNCTION(for_each);
JS_DECLARE_NATIVE_FUNCTION(includes);
JS_DECLARE_NATIVE_FUNCTION(index_of);
+ JS_DECLARE_NATIVE_FUNCTION(last_index_of);
JS_DECLARE_NATIVE_FUNCTION(some);
JS_DECLARE_NATIVE_FUNCTION(join);
JS_DECLARE_NATIVE_FUNCTION(keys);
diff --git a/Userland/Libraries/LibJS/Tests/builtins/TypedArray/TypedArray.prototype.lastIndexOf.js b/Userland/Libraries/LibJS/Tests/builtins/TypedArray/TypedArray.prototype.lastIndexOf.js
new file mode 100644
index 0000000000..c45b105ce4
--- /dev/null
+++ b/Userland/Libraries/LibJS/Tests/builtins/TypedArray/TypedArray.prototype.lastIndexOf.js
@@ -0,0 +1,44 @@
+const TYPED_ARRAYS = [
+ Uint8Array,
+ Uint8ClampedArray,
+ Uint16Array,
+ Uint32Array,
+ Int8Array,
+ Int16Array,
+ Int32Array,
+ Float32Array,
+ Float64Array,
+];
+
+const BIGINT_TYPED_ARRAYS = [BigUint64Array, BigInt64Array];
+
+test("basic functionality", () => {
+ TYPED_ARRAYS.forEach(T => {
+ expect(T.prototype.lastIndexOf).toHaveLength(1);
+
+ const typedArray = new T(3);
+ typedArray[0] = 1;
+ typedArray[1] = 2;
+ typedArray[2] = 2;
+
+ expect(typedArray.lastIndexOf(2)).toBe(2);
+ expect(typedArray.lastIndexOf(-1)).toBe(-1);
+ expect(typedArray.lastIndexOf(Infinity)).toBe(-1);
+ expect(typedArray.lastIndexOf(2, 2)).toBe(2);
+ expect(typedArray.lastIndexOf(2, -2)).toBe(1);
+ });
+
+ BIGINT_TYPED_ARRAYS.forEach(T => {
+ expect(T.prototype.lastIndexOf).toHaveLength(1);
+
+ const typedArray = new T(3);
+ typedArray[0] = 1n;
+ typedArray[1] = 2n;
+ typedArray[2] = 2n;
+
+ expect(typedArray.lastIndexOf(2n)).toBe(2);
+ expect(typedArray.lastIndexOf(-1n)).toBe(-1);
+ expect(typedArray.lastIndexOf(2n, 2)).toBe(2);
+ expect(typedArray.lastIndexOf(2n, -2)).toBe(1);
+ });
+});