diff options
author | Timmy Willison <timmywillisn@gmail.com> | 2016-03-03 18:29:45 -0500 |
---|---|---|
committer | Timmy Willison <timmywillisn@gmail.com> | 2016-03-07 11:23:33 -0500 |
commit | 63397aaaeaca78dc79e7d9a0860e529c2b7194ec (patch) | |
tree | 3fb8ba74025560826bbc2d0f4631537e0314f1f4 | |
parent | c5c30735311c74c60689fcabdcf2cb192524000e (diff) | |
download | jquery-63397aaaeaca78dc79e7d9a0860e529c2b7194ec.tar.gz jquery-63397aaaeaca78dc79e7d9a0860e529c2b7194ec.zip |
Core: restore enumeration behavior in isPlainObject
Fixes gh-2968
Close gh-2970
-rw-r--r-- | src/core.js | 9 | ||||
-rw-r--r-- | test/unit/core.js | 16 |
2 files changed, 20 insertions, 5 deletions
diff --git a/src/core.js b/src/core.js index f27306587..967fa8e26 100644 --- a/src/core.js +++ b/src/core.js @@ -223,6 +223,7 @@ jQuery.extend( { }, isPlainObject: function( obj ) { + var key; // Not plain objects: // - Any object or value whose internal [[Class]] property is not "[object Object]" @@ -237,9 +238,11 @@ jQuery.extend( { return false; } - // If the function hasn't returned already, we're confident that - // |obj| is a plain object, created by {} or constructed with new Object - return true; + // Own properties are enumerated firstly, so to speed up, + // if last one is own, then all properties are own + for ( key in obj ) {} + + return key === undefined || hasOwn.call( obj, key ); }, isEmptyObject: function( obj ) { diff --git a/test/unit/core.js b/test/unit/core.js index accdae8e1..e33f65358 100644 --- a/test/unit/core.js +++ b/test/unit/core.js @@ -268,13 +268,21 @@ QUnit.test( "type for `Symbol`", function( assert ) { } ); QUnit.asyncTest( "isPlainObject", function( assert ) { - assert.expect( 15 ); + assert.expect( 19 ); - var pass, iframe, doc, + var pass, iframe, doc, parentObj, childObj, deep, fn = function() {}; // The use case that we want to match assert.ok( jQuery.isPlainObject( {} ), "{}" ); + assert.ok( jQuery.isPlainObject( new window.Object() ), "new Object" ); + + parentObj = { foo: "bar" }; + childObj = Object.create( parentObj ); + + assert.ok( !jQuery.isPlainObject( childObj ), "isPlainObject(Object.create({}))" ); + childObj.bar = "foo"; + assert.ok( !jQuery.isPlainObject( childObj ), "isPlainObject(Object.create({}))" ); // Not objects shouldn't be matched assert.ok( !jQuery.isPlainObject( "" ), "string" ); @@ -302,6 +310,10 @@ QUnit.asyncTest( "isPlainObject", function( assert ) { // Again, instantiated objects shouldn't be matched assert.ok( !jQuery.isPlainObject( new fn() ), "new fn" ); + // Deep object + deep = { "foo": { "baz": true }, "foo2": document }; + assert.ok( jQuery.isPlainObject( deep ), "Object with objects is still plain" ); + // DOM Element assert.ok( !jQuery.isPlainObject( document.createElement( "div" ) ), "DOM Element" ); |