Implement Object.{entries, values} built-in methods (#3993)
Object.keys also updated to follow the ES11 standard. JerryScript-DCO-1.0-Signed-off-by: Roland Takacs rtakacs@inf.u-szeged.hu
This commit is contained in:
@@ -357,6 +357,18 @@ typedef enum
|
|||||||
* the fast access mode array back to normal array */
|
* the fast access mode array back to normal array */
|
||||||
} ecma_list_properties_options_t;
|
} ecma_list_properties_options_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enumerable property name listing options.
|
||||||
|
*/
|
||||||
|
typedef enum
|
||||||
|
{
|
||||||
|
ECMA_ENUMERABLE_PROPERTY_KEYS, /**< List only property names */
|
||||||
|
ECMA_ENUMERABLE_PROPERTY_VALUES, /**< List only property values */
|
||||||
|
ECMA_ENUMERABLE_PROPERTY_ENTRIES, /**< List both propery names and values */
|
||||||
|
|
||||||
|
ECMA_ENUMERABLE_PROPERTY__COUNT /**< Number of enumerable property listing types */
|
||||||
|
} ecma_enumerable_property_names_options_t;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* List enumerable properties and include the prototype chain.
|
* List enumerable properties and include the prototype chain.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -227,11 +227,10 @@ ecma_builtin_helper_get_to_locale_string_at_index (ecma_object_t *obj_p, /**< th
|
|||||||
} /* ecma_builtin_helper_get_to_locale_string_at_index */
|
} /* ecma_builtin_helper_get_to_locale_string_at_index */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The Object.keys and Object.getOwnPropertyNames routine's common part.
|
* The Object's 'getOwnPropertyNames' routine.
|
||||||
*
|
*
|
||||||
* See also:
|
* See also:
|
||||||
* ECMA-262 v5, 15.2.3.4 steps 2-5
|
* ECMA-262 v5, 15.2.3.4 steps 2-5
|
||||||
* ECMA-262 v5, 15.2.3.14 steps 3-6
|
|
||||||
*
|
*
|
||||||
* @return ecma value - Array of property names.
|
* @return ecma value - Array of property names.
|
||||||
* Returned value must be freed with ecma_free_value.
|
* Returned value must be freed with ecma_free_value.
|
||||||
|
|||||||
@@ -14,6 +14,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "ecma-alloc.h"
|
#include "ecma-alloc.h"
|
||||||
|
#include "ecma-array-object.h"
|
||||||
#include "ecma-builtin-helpers.h"
|
#include "ecma-builtin-helpers.h"
|
||||||
#include "ecma-builtins.h"
|
#include "ecma-builtins.h"
|
||||||
#include "ecma-conversion.h"
|
#include "ecma-conversion.h"
|
||||||
@@ -58,6 +59,8 @@ enum
|
|||||||
ECMA_OBJECT_ROUTINE_GET_OWN_PROPERTY_SYMBOLS,
|
ECMA_OBJECT_ROUTINE_GET_OWN_PROPERTY_SYMBOLS,
|
||||||
ECMA_OBJECT_ROUTINE_GET_PROTOTYPE_OF,
|
ECMA_OBJECT_ROUTINE_GET_PROTOTYPE_OF,
|
||||||
ECMA_OBJECT_ROUTINE_KEYS,
|
ECMA_OBJECT_ROUTINE_KEYS,
|
||||||
|
ECMA_OBJECT_ROUTINE_VALUES,
|
||||||
|
ECMA_OBJECT_ROUTINE_ENTRIES,
|
||||||
|
|
||||||
/* These should be in this order. */
|
/* These should be in this order. */
|
||||||
ECMA_OBJECT_ROUTINE_FREEZE,
|
ECMA_OBJECT_ROUTINE_FREEZE,
|
||||||
@@ -695,19 +698,35 @@ ecma_builtin_object_object_is_extensible (ecma_object_t *obj_p) /**< routine's a
|
|||||||
} /* ecma_builtin_object_object_is_extensible */
|
} /* ecma_builtin_object_object_is_extensible */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The Object object's 'keys' routine
|
* Common implementation of the Object object's 'keys', 'values', 'entries' routines
|
||||||
*
|
*
|
||||||
* See also:
|
* See also:
|
||||||
* ECMA-262 v5, 15.2.3.14
|
* ECMA-262 v11, 19.1.2.17
|
||||||
|
* ECMA-262 v11, 19.1.2.22
|
||||||
|
* ECMA-262 v11, 19.1.2.5
|
||||||
*
|
*
|
||||||
* @return ecma value
|
* @return ecma value
|
||||||
* Returned value must be freed with ecma_free_value.
|
* Returned value must be freed with ecma_free_value.
|
||||||
*/
|
*/
|
||||||
static ecma_value_t
|
static ecma_value_t
|
||||||
ecma_builtin_object_object_keys (ecma_object_t *obj_p) /**< routine's argument */
|
ecma_builtin_object_object_keys_values_helper (ecma_object_t *obj_p, /**< routine's first argument */
|
||||||
|
ecma_enumerable_property_names_options_t option) /**< listing option */
|
||||||
{
|
{
|
||||||
return ecma_builtin_helper_object_get_properties (obj_p, ECMA_LIST_ENUMERABLE);
|
/* 2. */
|
||||||
} /* ecma_builtin_object_object_keys */
|
ecma_collection_t *props_p = ecma_op_object_get_enumerable_property_names (obj_p, option);
|
||||||
|
|
||||||
|
if (props_p == NULL)
|
||||||
|
{
|
||||||
|
return ECMA_VALUE_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
ecma_value_t *names_buffer_p = props_p->buffer_p;
|
||||||
|
/* 3. */
|
||||||
|
ecma_value_t array_value = ecma_op_create_array_object (names_buffer_p, props_p->item_count, false);
|
||||||
|
ecma_collection_free (props_p);
|
||||||
|
|
||||||
|
return array_value;
|
||||||
|
} /* ecma_builtin_object_object_keys_values_helper */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The Object object's 'getOwnPropertyDescriptor' routine
|
* The Object object's 'getOwnPropertyDescriptor' routine
|
||||||
@@ -1228,7 +1247,7 @@ ecma_builtin_object_dispatch_routine (uint16_t builtin_routine_id, /**< built-in
|
|||||||
JERRY_ASSERT (builtin_routine_id == ECMA_OBJECT_ROUTINE_DEFINE_PROPERTIES);
|
JERRY_ASSERT (builtin_routine_id == ECMA_OBJECT_ROUTINE_DEFINE_PROPERTIES);
|
||||||
return ecma_builtin_object_object_define_properties (obj_p, arg2);
|
return ecma_builtin_object_object_define_properties (obj_p, arg2);
|
||||||
}
|
}
|
||||||
else if (builtin_routine_id <= ECMA_OBJECT_ROUTINE_KEYS)
|
else if (builtin_routine_id <= ECMA_OBJECT_ROUTINE_ENTRIES)
|
||||||
{
|
{
|
||||||
#if ENABLED (JERRY_ESNEXT)
|
#if ENABLED (JERRY_ESNEXT)
|
||||||
ecma_value_t object = ecma_op_to_object (arg1);
|
ecma_value_t object = ecma_op_to_object (arg1);
|
||||||
@@ -1266,10 +1285,16 @@ ecma_builtin_object_dispatch_routine (uint16_t builtin_routine_id, /**< built-in
|
|||||||
result = ecma_builtin_object_object_get_own_property_symbols (obj_p);
|
result = ecma_builtin_object_object_get_own_property_symbols (obj_p);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case ECMA_OBJECT_ROUTINE_ENTRIES:
|
||||||
|
case ECMA_OBJECT_ROUTINE_VALUES:
|
||||||
#endif /* ENABLED (JERRY_ESNEXT) */
|
#endif /* ENABLED (JERRY_ESNEXT) */
|
||||||
case ECMA_OBJECT_ROUTINE_KEYS:
|
case ECMA_OBJECT_ROUTINE_KEYS:
|
||||||
{
|
{
|
||||||
result = ecma_builtin_object_object_keys (obj_p);
|
JERRY_ASSERT (builtin_routine_id - ECMA_OBJECT_ROUTINE_KEYS < ECMA_ENUMERABLE_PROPERTY__COUNT);
|
||||||
|
|
||||||
|
const int option = builtin_routine_id - ECMA_OBJECT_ROUTINE_KEYS;
|
||||||
|
result = ecma_builtin_object_object_keys_values_helper (obj_p,
|
||||||
|
(ecma_enumerable_property_names_options_t) option);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ECMA_OBJECT_ROUTINE_GET_OWN_PROPERTY_DESCRIPTOR:
|
case ECMA_OBJECT_ROUTINE_GET_OWN_PROPERTY_DESCRIPTOR:
|
||||||
|
|||||||
@@ -53,6 +53,10 @@ ROUTINE (LIT_MAGIC_STRING_PREVENT_EXTENSIONS_UL, ECMA_OBJECT_ROUTINE_PREVENT_EXT
|
|||||||
ROUTINE (LIT_MAGIC_STRING_IS_SEALED_UL, ECMA_OBJECT_ROUTINE_IS_SEALED, 1, 1)
|
ROUTINE (LIT_MAGIC_STRING_IS_SEALED_UL, ECMA_OBJECT_ROUTINE_IS_SEALED, 1, 1)
|
||||||
ROUTINE (LIT_MAGIC_STRING_IS_FROZEN_UL, ECMA_OBJECT_ROUTINE_IS_FROZEN, 1, 1)
|
ROUTINE (LIT_MAGIC_STRING_IS_FROZEN_UL, ECMA_OBJECT_ROUTINE_IS_FROZEN, 1, 1)
|
||||||
ROUTINE (LIT_MAGIC_STRING_IS_EXTENSIBLE, ECMA_OBJECT_ROUTINE_IS_EXTENSIBLE, 1, 1)
|
ROUTINE (LIT_MAGIC_STRING_IS_EXTENSIBLE, ECMA_OBJECT_ROUTINE_IS_EXTENSIBLE, 1, 1)
|
||||||
|
#if ENABLED (JERRY_ESNEXT)
|
||||||
|
ROUTINE (LIT_MAGIC_STRING_ENTRIES, ECMA_OBJECT_ROUTINE_ENTRIES, 1, 1)
|
||||||
|
ROUTINE (LIT_MAGIC_STRING_VALUES, ECMA_OBJECT_ROUTINE_VALUES, 1, 1)
|
||||||
|
#endif /* ENABLED (JERRY_ESNEXT) */
|
||||||
ROUTINE (LIT_MAGIC_STRING_KEYS, ECMA_OBJECT_ROUTINE_KEYS, 1, 1)
|
ROUTINE (LIT_MAGIC_STRING_KEYS, ECMA_OBJECT_ROUTINE_KEYS, 1, 1)
|
||||||
ROUTINE (LIT_MAGIC_STRING_GET_OWN_PROPERTY_DESCRIPTOR_UL, ECMA_OBJECT_ROUTINE_GET_OWN_PROPERTY_DESCRIPTOR, 2, 2)
|
ROUTINE (LIT_MAGIC_STRING_GET_OWN_PROPERTY_DESCRIPTOR_UL, ECMA_OBJECT_ROUTINE_GET_OWN_PROPERTY_DESCRIPTOR, 2, 2)
|
||||||
#if ENABLED (JERRY_ESNEXT)
|
#if ENABLED (JERRY_ESNEXT)
|
||||||
|
|||||||
@@ -1983,6 +1983,118 @@ ecma_op_object_is_prototype_of (ecma_object_t *base_p, /**< base object */
|
|||||||
} while (true);
|
} while (true);
|
||||||
} /* ecma_op_object_is_prototype_of */
|
} /* ecma_op_object_is_prototype_of */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Object's EnumerableOwnPropertyNames operation
|
||||||
|
*
|
||||||
|
* See also:
|
||||||
|
* ECMA-262 v11, 7.3.23
|
||||||
|
*
|
||||||
|
* @return NULL - if operation fails
|
||||||
|
* collection of property names / values / name-value pairs - otherwise
|
||||||
|
*/
|
||||||
|
ecma_collection_t *
|
||||||
|
ecma_op_object_get_enumerable_property_names (ecma_object_t *obj_p, /**< routine's first argument */
|
||||||
|
ecma_enumerable_property_names_options_t option) /**< listing option */
|
||||||
|
{
|
||||||
|
/* 2. */
|
||||||
|
ecma_collection_t *prop_names_p = ecma_op_object_get_property_names (obj_p, ECMA_LIST_NO_OPTS);
|
||||||
|
|
||||||
|
#if ENABLED (JERRY_BUILTIN_PROXY)
|
||||||
|
if (JERRY_UNLIKELY (prop_names_p == NULL))
|
||||||
|
{
|
||||||
|
return prop_names_p;
|
||||||
|
}
|
||||||
|
#endif /* ENABLED (JERRY_BUILTIN_PROXY) */
|
||||||
|
|
||||||
|
ecma_value_t *names_buffer_p = prop_names_p->buffer_p;
|
||||||
|
/* 3. */
|
||||||
|
ecma_collection_t *properties_p = ecma_new_collection ();
|
||||||
|
|
||||||
|
/* 4. */
|
||||||
|
for (uint32_t i = 0; i < prop_names_p->item_count; i++)
|
||||||
|
{
|
||||||
|
/* 4.a */
|
||||||
|
JERRY_ASSERT (ecma_is_value_string (names_buffer_p[i]));
|
||||||
|
|
||||||
|
ecma_string_t *key_p = ecma_get_string_from_value (names_buffer_p[i]);
|
||||||
|
|
||||||
|
/* 4.a.i */
|
||||||
|
ecma_property_descriptor_t prop_desc;
|
||||||
|
ecma_value_t status;
|
||||||
|
|
||||||
|
#if ENABLED (JERRY_BUILTIN_PROXY)
|
||||||
|
if (ECMA_OBJECT_IS_PROXY (obj_p))
|
||||||
|
{
|
||||||
|
status = ecma_proxy_object_get_own_property_descriptor (obj_p, key_p, &prop_desc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
#endif /* ENABLED (JERRY_BUILTIN_PROXY) */
|
||||||
|
{
|
||||||
|
status = ecma_op_object_get_own_property_descriptor (obj_p, key_p, &prop_desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ECMA_IS_VALUE_ERROR (status))
|
||||||
|
{
|
||||||
|
ecma_collection_free (prop_names_p);
|
||||||
|
ecma_collection_free (properties_p);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 4.a.ii */
|
||||||
|
if ((prop_desc.flags & ECMA_PROP_IS_ENUMERABLE) != 0)
|
||||||
|
{
|
||||||
|
/* 4.a.ii.1 */
|
||||||
|
if (option == ECMA_ENUMERABLE_PROPERTY_KEYS)
|
||||||
|
{
|
||||||
|
ecma_collection_push_back (properties_p, ecma_copy_value (names_buffer_p[i]));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* 4.a.ii.2.a */
|
||||||
|
ecma_value_t value = ecma_op_object_get (obj_p, key_p);
|
||||||
|
|
||||||
|
if (ECMA_IS_VALUE_ERROR (value))
|
||||||
|
{
|
||||||
|
ecma_collection_free (prop_names_p);
|
||||||
|
ecma_collection_free (properties_p);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 4.a.ii.2.b */
|
||||||
|
if (option == ECMA_ENUMERABLE_PROPERTY_VALUES)
|
||||||
|
{
|
||||||
|
ecma_collection_push_back (properties_p, value);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* 4.a.ii.2.c.i */
|
||||||
|
JERRY_ASSERT (option == ECMA_ENUMERABLE_PROPERTY_ENTRIES);
|
||||||
|
|
||||||
|
/* 4.a.ii.2.c.ii */
|
||||||
|
ecma_object_t *entry_p = ecma_op_new_fast_array_object (2);
|
||||||
|
ecma_fast_array_set_property (entry_p, 0, names_buffer_p[i]);
|
||||||
|
ecma_fast_array_set_property (entry_p, 1, value);
|
||||||
|
ecma_free_value (value);
|
||||||
|
|
||||||
|
/* 4.a.ii.2.c.iii */
|
||||||
|
ecma_collection_push_back (properties_p, ecma_make_object_value (entry_p));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ecma_is_value_true (status))
|
||||||
|
{
|
||||||
|
ecma_free_property_descriptor (&prop_desc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ecma_collection_free (prop_names_p);
|
||||||
|
|
||||||
|
return properties_p;
|
||||||
|
} /* ecma_op_object_get_enumerable_property_names */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get collection of property names
|
* Get collection of property names
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -68,6 +68,8 @@ ecma_value_t ecma_op_object_get_own_property_descriptor (ecma_object_t *object_p
|
|||||||
ecma_value_t ecma_op_object_has_instance (ecma_object_t *obj_p, ecma_value_t value);
|
ecma_value_t ecma_op_object_has_instance (ecma_object_t *obj_p, ecma_value_t value);
|
||||||
ecma_value_t ecma_op_object_is_prototype_of (ecma_object_t *base_p, ecma_object_t *target_p);
|
ecma_value_t ecma_op_object_is_prototype_of (ecma_object_t *base_p, ecma_object_t *target_p);
|
||||||
ecma_collection_t * ecma_op_object_get_property_names (ecma_object_t *obj_p, uint32_t opts);
|
ecma_collection_t * ecma_op_object_get_property_names (ecma_object_t *obj_p, uint32_t opts);
|
||||||
|
ecma_collection_t * ecma_op_object_get_enumerable_property_names (ecma_object_t *obj_p,
|
||||||
|
ecma_enumerable_property_names_options_t option);
|
||||||
|
|
||||||
lit_magic_string_id_t ecma_object_get_class_name (ecma_object_t *obj_p);
|
lit_magic_string_id_t ecma_object_get_class_name (ecma_object_t *obj_p);
|
||||||
bool ecma_object_class_is (ecma_object_t *object_p, uint32_t class_id);
|
bool ecma_object_class_is (ecma_object_t *object_p, uint32_t class_id);
|
||||||
|
|||||||
@@ -0,0 +1,134 @@
|
|||||||
|
// Copyright JS Foundation and other contributors, http://js.foundation
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
var object = {};
|
||||||
|
var symbol = Symbol("symbol");
|
||||||
|
|
||||||
|
Object.defineProperties(object, {
|
||||||
|
a: {
|
||||||
|
value: 42,
|
||||||
|
enumerable: true
|
||||||
|
},
|
||||||
|
b: {
|
||||||
|
value: "foo",
|
||||||
|
enumerable: false
|
||||||
|
},
|
||||||
|
[symbol]: {
|
||||||
|
value: "symbol",
|
||||||
|
enumerable: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Object.keys and Object.entries should have the same keys
|
||||||
|
var keys = Object.keys(object);
|
||||||
|
var entries = Object.entries(object);
|
||||||
|
|
||||||
|
assert(keys.length === entries.length);
|
||||||
|
for (let i = 0; i < keys.length; i++) {
|
||||||
|
assert(keys[i] === entries[i][0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test object entries
|
||||||
|
var entries = Object.entries(object);
|
||||||
|
assert(entries instanceof Array);
|
||||||
|
assert(entries.length === 1);
|
||||||
|
assert(entries[0].length === 2);
|
||||||
|
assert(entries[0][0] === "a");
|
||||||
|
assert(entries[0][1] === 42);
|
||||||
|
|
||||||
|
// Test array entries
|
||||||
|
var array = [1, 2, "three"];
|
||||||
|
var entries = Object.entries(array);
|
||||||
|
|
||||||
|
assert(entries instanceof Array);
|
||||||
|
assert(entries.length === array.length);
|
||||||
|
|
||||||
|
for (let i = 0; i < entries.length; i++) {
|
||||||
|
assert(entries[i][0] === i + "");
|
||||||
|
assert(entries[i][1] === array[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test prototype chain
|
||||||
|
function Parent() {}
|
||||||
|
Parent.prototype.inheritedMethod = function() {};
|
||||||
|
|
||||||
|
function method() {};
|
||||||
|
function Child() {
|
||||||
|
this.prop = 5;
|
||||||
|
this.method = method;
|
||||||
|
}
|
||||||
|
|
||||||
|
Child.prototype = new Parent;
|
||||||
|
Child.prototype.prototypeMethod = function() {};
|
||||||
|
|
||||||
|
var entries = Object.entries (new Child());
|
||||||
|
assert(entries.length === 2);
|
||||||
|
assert(entries[0][0] === "prop");
|
||||||
|
assert(entries[0][1] === 5);
|
||||||
|
assert(entries[1][0] === "method");
|
||||||
|
assert(entries[1][1] === method);
|
||||||
|
|
||||||
|
// Test with primitive values
|
||||||
|
var entries = Object.entries(true);
|
||||||
|
assert(entries instanceof Array);
|
||||||
|
assert(entries.length === 0);
|
||||||
|
|
||||||
|
try {
|
||||||
|
Object.entries(undefined);
|
||||||
|
assert(false);
|
||||||
|
} catch (e) {
|
||||||
|
assert(e instanceof TypeError)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
Object.entries(null);
|
||||||
|
assert(false);
|
||||||
|
} catch (e) {
|
||||||
|
assert(e instanceof TypeError)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test proxy object
|
||||||
|
var object = {};
|
||||||
|
|
||||||
|
Object.defineProperties(object, {
|
||||||
|
a: {
|
||||||
|
value: "foo",
|
||||||
|
enumerable: false
|
||||||
|
},
|
||||||
|
b: {
|
||||||
|
value: "bar",
|
||||||
|
enumerable: true,
|
||||||
|
writable: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var proxy = new Proxy(object, {
|
||||||
|
getOwnPropertyDescriptor: function(o, v) {
|
||||||
|
handlers.push("D");
|
||||||
|
return Object.getOwnPropertyDescriptor(o, v);
|
||||||
|
},
|
||||||
|
get: function(o, v) {
|
||||||
|
handlers.push("G");
|
||||||
|
return o[v];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var handlers = [];
|
||||||
|
var entries = Object.entries(proxy);
|
||||||
|
|
||||||
|
assert(entries.length === 1);
|
||||||
|
assert(entries[0][0] === "b");
|
||||||
|
assert(entries[0][1] === "bar");
|
||||||
|
assert(handlers.length === 3);
|
||||||
|
assert(handlers.toString() === "D,D,G");
|
||||||
@@ -0,0 +1,46 @@
|
|||||||
|
// Copyright JS Foundation and other contributors, http://js.foundation
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
var object = {};
|
||||||
|
|
||||||
|
Object.defineProperties(object, {
|
||||||
|
a: {
|
||||||
|
value: "foo",
|
||||||
|
enumerable: false
|
||||||
|
},
|
||||||
|
b: {
|
||||||
|
value: "bar",
|
||||||
|
enumerable: true,
|
||||||
|
writable: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var proxy = new Proxy(object, {
|
||||||
|
getOwnPropertyDescriptor: function(o, v) {
|
||||||
|
handlers.push("D");
|
||||||
|
return Object.getOwnPropertyDescriptor(o, v);
|
||||||
|
},
|
||||||
|
get: function(o, v) {
|
||||||
|
handlers.push("G");
|
||||||
|
return o[v];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var handlers = [];
|
||||||
|
var keys = Object.keys(proxy);
|
||||||
|
|
||||||
|
assert(keys.length === 1);
|
||||||
|
assert(keys[0] === "b");
|
||||||
|
assert(handlers.length === 2);
|
||||||
|
assert(handlers.toString() === "D,D");
|
||||||
@@ -0,0 +1,127 @@
|
|||||||
|
// Copyright JS Foundation and other contributors, http://js.foundation
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
// Test array
|
||||||
|
var arr = ["a", "b", "c"];
|
||||||
|
var values = Object.values(arr);
|
||||||
|
|
||||||
|
assert(values.indexOf("a") === 0);
|
||||||
|
assert(values.indexOf("b") === 1);
|
||||||
|
assert(values.indexOf("c") === 2);
|
||||||
|
assert(values.length === 3);
|
||||||
|
|
||||||
|
// Test object
|
||||||
|
var obj = {key1: "a", key3: "b", key2: "c", key4: "d", key5: "e"};
|
||||||
|
var values = Object.values(obj);
|
||||||
|
|
||||||
|
assert(values.indexOf("a") === 0);
|
||||||
|
assert(values.indexOf("b") === 1);
|
||||||
|
assert(values.indexOf("c") === 2);
|
||||||
|
assert(values.indexOf("d") === 3);
|
||||||
|
assert(values.indexOf("e") === 4);
|
||||||
|
assert(values.length === 5);
|
||||||
|
|
||||||
|
var obj2 = {};
|
||||||
|
Object.defineProperties(obj2, {
|
||||||
|
key_one: {enumerable: true, value: "one"},
|
||||||
|
key_two: {enumerable: false, value: "two"},
|
||||||
|
});
|
||||||
|
|
||||||
|
var values = Object.values(obj2);
|
||||||
|
// values should contain: key_one
|
||||||
|
assert(values.indexOf("one") === 0);
|
||||||
|
assert(values.indexOf("two") === -1);
|
||||||
|
assert(values.length === 1);
|
||||||
|
|
||||||
|
// Test prototype chain
|
||||||
|
function Parent() {}
|
||||||
|
Parent.prototype.inheritedMethod = function() {};
|
||||||
|
|
||||||
|
function method() {};
|
||||||
|
function Child() {
|
||||||
|
this.prop = 5;
|
||||||
|
this.method = method;
|
||||||
|
}
|
||||||
|
|
||||||
|
Child.prototype = new Parent;
|
||||||
|
Child.prototype.prototypeMethod = function() {};
|
||||||
|
|
||||||
|
var values = Object.values(new Child());
|
||||||
|
|
||||||
|
assert(values.indexOf(5) === 0);
|
||||||
|
assert(values.indexOf(method) === 1);
|
||||||
|
assert(values.length === 2);
|
||||||
|
|
||||||
|
// Check enumerable properties
|
||||||
|
var o = {};
|
||||||
|
|
||||||
|
Object.defineProperty(o, "a", {
|
||||||
|
value: "OK a",
|
||||||
|
writable: true,
|
||||||
|
enumerable: true,
|
||||||
|
configurable: true
|
||||||
|
});
|
||||||
|
|
||||||
|
Object.defineProperty(o, "b", {
|
||||||
|
value: "NOT_OK",
|
||||||
|
writable: true,
|
||||||
|
enumerable: false,
|
||||||
|
configurable: true
|
||||||
|
});
|
||||||
|
|
||||||
|
Object.defineProperty(o, "c", {
|
||||||
|
value: "OK c",
|
||||||
|
writable: true,
|
||||||
|
enumerable: true,
|
||||||
|
configurable: true
|
||||||
|
});
|
||||||
|
|
||||||
|
var values = Object.values(o);
|
||||||
|
assert(values.length === 2);
|
||||||
|
assert(values[0] === "OK a");
|
||||||
|
assert(values[1] === "OK c");
|
||||||
|
|
||||||
|
// Test proxy object
|
||||||
|
var object = {};
|
||||||
|
|
||||||
|
Object.defineProperties(object, {
|
||||||
|
a: {
|
||||||
|
value: "foo",
|
||||||
|
enumerable: false
|
||||||
|
},
|
||||||
|
b: {
|
||||||
|
value: "bar",
|
||||||
|
enumerable: true,
|
||||||
|
writable: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var proxy = new Proxy(object, {
|
||||||
|
getOwnPropertyDescriptor: function(o, v) {
|
||||||
|
handlers.push("D");
|
||||||
|
return Object.getOwnPropertyDescriptor(o, v);
|
||||||
|
},
|
||||||
|
get: function(o, v) {
|
||||||
|
handlers.push("G");
|
||||||
|
return o[v];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var handlers = [];
|
||||||
|
var values = Object.values(proxy);
|
||||||
|
|
||||||
|
assert(values.length === 1);
|
||||||
|
assert(values[0] === "bar")
|
||||||
|
assert(handlers.length === 3);
|
||||||
|
assert(handlers.toString() === "D,D,G");
|
||||||
Reference in New Issue
Block a user