DEV: Support modifyClass for native class fields on EmberObjects (#20987)
EmberObject's `reopen` feature allows changes to be made to the prototype of the class, but it does not work with native class fields. Native class field values are set on the instance in the constructor, and therefore override any values from the prototype. This commit implements a workaround which detects possible field overrides and then sets the values during the `init()` function of the EmberObject. This isn't perfect - old field values will still be present while any constructor function is running. But in the vast majority of cases, it should provide parity with old non-native-class EmberObject properties. This commit also adds a warning when trying to override fields on non-EmberObject classes. There is no change in behavior here - we're just warning about the fact it doesn't work.
This commit is contained in:
parent
788dc0a096
commit
a19efc4304
|
@ -227,25 +227,72 @@ class PluginApi {
|
||||||
* ```
|
* ```
|
||||||
**/
|
**/
|
||||||
modifyClass(resolverName, changes, opts) {
|
modifyClass(resolverName, changes, opts) {
|
||||||
const klass = this._resolveClass(resolverName, opts);
|
const info = this._resolveClass(resolverName, opts);
|
||||||
if (!klass) {
|
if (!info) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
const klass = info.class;
|
||||||
|
|
||||||
if (canModify(klass, "member", resolverName, changes)) {
|
if (canModify(info, "member", resolverName, changes)) {
|
||||||
delete changes.pluginId;
|
delete changes.pluginId;
|
||||||
|
|
||||||
if (klass.class.reopen) {
|
const changeDescriptors = Object.getOwnPropertyDescriptors(changes);
|
||||||
klass.class.reopen(changes);
|
|
||||||
} else {
|
const fieldDescriptorEntries = Object.entries(changeDescriptors).filter(
|
||||||
|
([key, desc]) => {
|
||||||
|
if (typeof desc.value === "function" || desc.get || desc.set) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
klass.prototype.mergedProperties?.includes(key) ||
|
||||||
|
klass.prototype.concatenatedProperties?.includes(key)
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (klass.reopen) {
|
||||||
|
// klass is an EmberObject - use builtin ember 'reopen' feature
|
||||||
|
klass.reopen(changes);
|
||||||
|
|
||||||
|
if (fieldDescriptorEntries.length) {
|
||||||
|
// reopen sets these values on the prototype, but they might be overrides for
|
||||||
|
// native class fields. Those need to be overridden after the original values
|
||||||
|
// have been set on the instance by the constructor.
|
||||||
|
klass.reopen({
|
||||||
|
init() {
|
||||||
Object.defineProperties(
|
Object.defineProperties(
|
||||||
klass.class.prototype || klass.class,
|
this,
|
||||||
Object.getOwnPropertyDescriptors(changes)
|
Object.fromEntries(fieldDescriptorEntries)
|
||||||
|
);
|
||||||
|
this._super();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Not an EmberObject. We can override functions/getters/setters on the prototype
|
||||||
|
// but we don't currently have a way to override fields. Log a warning to the console
|
||||||
|
// if there are any overrides that look like fields.
|
||||||
|
|
||||||
|
Object.defineProperties(klass.prototype, changeDescriptors);
|
||||||
|
|
||||||
|
if (fieldDescriptorEntries.length) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.warn(
|
||||||
|
consolePrefix(),
|
||||||
|
`Attempted to modify fields in ${resolverName}, which is not an EmberObject. ` +
|
||||||
|
"These changes will not take effect. " +
|
||||||
|
`Fields: ${JSON.stringify(
|
||||||
|
fieldDescriptorEntries.map(([key]) => key)
|
||||||
|
)}`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return klass;
|
return info;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -4,6 +4,7 @@ import discourseComputed from "discourse-common/utils/decorators";
|
||||||
import { withPluginApi } from "discourse/lib/plugin-api";
|
import { withPluginApi } from "discourse/lib/plugin-api";
|
||||||
import { setupTest } from "ember-qunit";
|
import { setupTest } from "ember-qunit";
|
||||||
import { getOwner } from "discourse-common/lib/get-owner";
|
import { getOwner } from "discourse-common/lib/get-owner";
|
||||||
|
import Sinon from "sinon";
|
||||||
|
|
||||||
module("Unit | Utility | plugin-api", function (hooks) {
|
module("Unit | Utility | plugin-api", function (hooks) {
|
||||||
setupTest(hooks);
|
setupTest(hooks);
|
||||||
|
@ -35,6 +36,9 @@ module("Unit | Utility | plugin-api", function (hooks) {
|
||||||
|
|
||||||
test("modifyClass works with native class Ember objects", function (assert) {
|
test("modifyClass works with native class Ember objects", function (assert) {
|
||||||
class NativeTestThingy extends EmberObject {
|
class NativeTestThingy extends EmberObject {
|
||||||
|
firstField = "firstFieldValue";
|
||||||
|
otherField = "otherFieldValue";
|
||||||
|
|
||||||
@discourseComputed
|
@discourseComputed
|
||||||
prop() {
|
prop() {
|
||||||
return "howdy";
|
return "howdy";
|
||||||
|
@ -47,6 +51,8 @@ module("Unit | Utility | plugin-api", function (hooks) {
|
||||||
api.modifyClass("native-test-thingy:main", {
|
api.modifyClass("native-test-thingy:main", {
|
||||||
pluginId: "plugin-api-test",
|
pluginId: "plugin-api-test",
|
||||||
|
|
||||||
|
otherField: "new otherFieldValue",
|
||||||
|
|
||||||
@discourseComputed
|
@discourseComputed
|
||||||
prop() {
|
prop() {
|
||||||
return `${this._super(...arguments)} partner`;
|
return `${this._super(...arguments)} partner`;
|
||||||
|
@ -56,10 +62,15 @@ module("Unit | Utility | plugin-api", function (hooks) {
|
||||||
|
|
||||||
const thingy = getOwner(this).lookup("native-test-thingy:main");
|
const thingy = getOwner(this).lookup("native-test-thingy:main");
|
||||||
assert.strictEqual(thingy.prop, "howdy partner");
|
assert.strictEqual(thingy.prop, "howdy partner");
|
||||||
|
assert.strictEqual(thingy.firstField, "firstFieldValue");
|
||||||
|
assert.strictEqual(thingy.otherField, "new otherFieldValue");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("modifyClass works with native classes", function (assert) {
|
test("modifyClass works with native classes", function (assert) {
|
||||||
class ClassTestThingy {
|
class ClassTestThingy {
|
||||||
|
firstField = "firstFieldValue";
|
||||||
|
otherField = "otherFieldValue";
|
||||||
|
|
||||||
get keep() {
|
get keep() {
|
||||||
return "hey!";
|
return "hey!";
|
||||||
}
|
}
|
||||||
|
@ -69,23 +80,42 @@ module("Unit | Utility | plugin-api", function (hooks) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
getOwner(this).register("class-test-thingy:main", new ClassTestThingy(), {
|
getOwner(this).register("class-test-thingy:main", ClassTestThingy);
|
||||||
instantiate: false,
|
|
||||||
});
|
const warnStub = Sinon.stub(console, "warn");
|
||||||
|
|
||||||
withPluginApi("1.1.0", (api) => {
|
withPluginApi("1.1.0", (api) => {
|
||||||
api.modifyClass("class-test-thingy:main", {
|
api.modifyClass("class-test-thingy:main", {
|
||||||
pluginId: "plugin-api-test",
|
pluginId: "plugin-api-test",
|
||||||
|
|
||||||
|
otherField: "new otherFieldValue",
|
||||||
get prop() {
|
get prop() {
|
||||||
return "g'day";
|
return "g'day";
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const thingy = getOwner(this).lookup("class-test-thingy:main");
|
assert.strictEqual(
|
||||||
assert.strictEqual(thingy.keep, "hey!");
|
warnStub.callCount,
|
||||||
assert.strictEqual(thingy.prop, "g'day");
|
1,
|
||||||
|
"fields warning was printed to console"
|
||||||
|
);
|
||||||
|
assert.true(warnStub.args[0][1].startsWith("Attempted to modify fields"));
|
||||||
|
|
||||||
|
const thingy = new ClassTestThingy();
|
||||||
|
|
||||||
|
assert.strictEqual(thingy.keep, "hey!", "maintains unchanged base getter");
|
||||||
|
assert.strictEqual(thingy.prop, "g'day", "can override getter");
|
||||||
|
assert.strictEqual(
|
||||||
|
thingy.firstField,
|
||||||
|
"firstFieldValue",
|
||||||
|
"maintains unchanged base field"
|
||||||
|
);
|
||||||
|
assert.strictEqual(
|
||||||
|
thingy.otherField,
|
||||||
|
"otherFieldValue",
|
||||||
|
"cannot override field"
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
skip("modifyClass works with getters", function (assert) {
|
skip("modifyClass works with getters", function (assert) {
|
||||||
|
@ -95,9 +125,7 @@ module("Unit | Utility | plugin-api", function (hooks) {
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
getOwner(this).register("test-class:main", Base, {
|
getOwner(this).register("test-class:main", Base);
|
||||||
instantiate: false,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Performing this lookup triggers `factory._onLookup`. In DEBUG builds, that invokes injectedPropertyAssertion()
|
// Performing this lookup triggers `factory._onLookup`. In DEBUG builds, that invokes injectedPropertyAssertion()
|
||||||
// https://github.com/emberjs/ember.js/blob/36505f1b42/packages/%40ember/-internals/runtime/lib/system/core_object.js#L1144-L1163
|
// https://github.com/emberjs/ember.js/blob/36505f1b42/packages/%40ember/-internals/runtime/lib/system/core_object.js#L1144-L1163
|
||||||
|
|
Loading…
Reference in New Issue