-
Notifications
You must be signed in to change notification settings - Fork 27.4k
angular.extend does not copy getters/setters #8573
Comments
See a proper version here: https://github.com/decafjs/decaf/blob/master/builtins/decaf.js#L17 extend() method |
Same bug exists in jQuery. |
While you're looking at it, you may want to consider ES6 Proxy support, too |
Angular doesn't need to detect getters/setters as it currently doesn't use getters/setters anywhere yet. This also applies to ES6 proxies, which don't really exist in very many browsers (v8 used to, as far as I can recall, implement Mozilla's old Proxy API, but this is no longer the case, if it ever was and I'm not misremembering). Personally, I'd love to add these features, but I don't think we'd have an actual use for them in core at this time, and it would hurt execution speed even more. If you need features like this, why not implement them in your application code? |
The method does not perform its function as advertised. It does not truly copy properties from the src objects to the dst. The jQuery implementation is many lines longer than the angular.extend() implementation. They implement deep copy and all sorts of bounds checking and even calls itself recursively. ES6 Proxy has been in Firefox since version 24. jQuery.extend = jQuery.fn.extend = function() {
var options, name, src, copy, copyIsArray, clone,
target = arguments[0] || {},
i = 1,
length = arguments.length,
deep = false;
// Handle a deep copy situation
if ( typeof target === "boolean" ) {
deep = target;
// Skip the boolean and the target
target = arguments[ i ] || {};
i++;
}
// Handle case when target is a string or something (possible in deep copy)
if ( typeof target !== "object" && !jQuery.isFunction(target) ) {
target = {};
}
// Extend jQuery itself if only one argument is passed
if ( i === length ) {
target = this;
i--;
}
for ( ; i < length; i++ ) {
// Only deal with non-null/undefined values
if ( (options = arguments[ i ]) != null ) {
// Extend the base object
for ( name in options ) {
src = target[ name ];
copy = options[ name ];
// Prevent never-ending loop
if ( target === copy ) {
continue;
}
// Recurse if we're merging plain objects or arrays
if ( deep && copy && ( jQuery.isPlainObject(copy) ||
(copyIsArray = jQuery.isArray(copy)) ) ) {
if ( copyIsArray ) {
copyIsArray = false;
clone = src && jQuery.isArray(src) ? src : [];
} else {
clone = src && jQuery.isPlainObject(src) ? src : {};
}
// Never move original objects, clone them
target[ name ] = jQuery.extend( deep, clone, copy );
// Don't bring in undefined values
} else if ( copy !== undefined ) {
target[ name ] = copy;
}
}
}
}
// Return the modified object
return target;
}; |
Thanks! |
Now, if you use
The I'm not sure if there is a way to make the |
Actually, if you make a getter or setter |
Here's how it's done right. https://github.com/decafjs/decaf/blob/master/builtins/decaf.js#L47 On Wed, Aug 19, 2015 at 3:13 PM, Sean Moran notifications@github.com
|
I was not proposing an alternative to |
defineProperty() isn't the only way to define a getter or setter. var o = { _x: 10, get x() { return this._x; } }; I'm reasonably sure you don't want angular.extend() to copy this._x (10) IMO On Wed, Aug 19, 2015 at 4:05 PM, Sean Moran notifications@github.com
|
I am aware that Then again, when a getter/setter is created at the time the object is created, it is already enumerable, so I don't know why these getters/setters can not be extended by default, but they can't. |
The angular implementation of extend() is calling the getter and/or setter A true copy means it copies the members: variables and functions. A getter Getters and setters are pretty slick. If you want to do a true extend(), The key to making extend() work is in that decafjs code. On Wed, Aug 19, 2015 at 5:08 PM, Sean Moran notifications@github.com
|
Angular exposes some of the helper functions that it uses internally. This
is the case for `extend`, `copy` and many others.
There are other libraries that specialize in these functions, keep their
focus is there and can do a better job. Eg, angular.copy does not handle
many of the browser specific quirks with native objects and probably will
never do.
It is not in the best interest of most users to make these helper functions
big nor slow, as these are used internally and any change in that direction
can have a direct impact in download size and performance. At the same
time, apps that need the most accurate version, should be better served
with other libraries.
Now, anybody is welcome to create a PR that adds support for
getters/setters. If the changes are small and the performance is about the
same, then it should get merge without any issues. If the change makes some
function big or slow, then it will need an excellent use case or a large
community supporting this trade off (in these later cases, it should get
eventually solved).
Anybody who is interested in getting this fixed, create a PR and let's move
the discussion there.
|
So why does angular provide an |
I think what @Igalfaso is trying to say is that angular tries to make its built-in functions as small and fast as it needs for its own internal purposes. For example, it does not need its |
Related to #5085 |
@mschwartz thank you for this: var o = { _x: 10, get x() { return this._x; } }; |
Given the Angular team's general stance about the utility functions it provides, would it make sense to close this issue or maybe have a doc update that warns that it should not be relied upon as a robust solution for all conditions? |
I'm really just a user, but I for one would have appreciated a doc reference pre-empting my comment here. |
@wesleycho is right. Closing as won't fix. |
resulting o object does not have the get x() method, it has 'x'
The text was updated successfully, but these errors were encountered: