diff options
author | Idan Horowitz <idan.horowitz@gmail.com> | 2021-07-08 04:39:37 +0300 |
---|---|---|
committer | Linus Groh <mail@linusgroh.de> | 2021-07-08 16:30:09 +0100 |
commit | f0abcde00c457f972acb626d8d3e2b9272c2f0b8 (patch) | |
tree | e6ecad0925c4782930e01568377207b6e4fd99e6 /Userland | |
parent | 6343bfa9d7ae24b078ffc00a312a413bd45e0422 (diff) | |
download | serenity-f0abcde00c457f972acb626d8d3e2b9272c2f0b8.zip |
LibJS: Add %TypedArray%.prototype.lastIndexOf
Diffstat (limited to 'Userland')
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); + }); +}); |