0
Fork 0
mirror of https://github.com/TryGhost/Ghost.git synced 2025-01-20 22:42:53 -05:00

Fixed PromiseObject related errors thrown from members activity list (#15641)

no issue

- the `memberRecord` getter on the members activity controller was
returning the direct result of `store.findRecord()` which in reality is
a `PromiseObject` that is a type of proxy. Although templates handle
this OK, any JS code using the object needs to know that it's not a
typical type of object and has to use `.get()` for all property access
to avoid errors
- switched `memberRecord` from a getter to a resource so we have more
control over the returned object and loading lifecycle
- added a `MemberFetcher` resource class that awaits the result of the
store find and only then sets the value meaning we always have a fully
resolved model object
- being a resource the fetch will only occur when the `member` id
property changes, unlike the getter which performed the fetch every time
it was accessed
This commit is contained in:
Kevin Ansfield 2022-10-17 14:28:34 +01:00 committed by GitHub
parent 22fe1c01de
commit f78f264982
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 30 additions and 10 deletions

View file

@ -1,8 +1,10 @@
import Controller from '@ember/controller'; import Controller from '@ember/controller';
import MemberFetcher from 'ghost-admin/helpers/member-fetcher';
import {EMAIL_EVENTS, NEWSLETTER_EVENTS} from 'ghost-admin/helpers/members-event-filter'; import {EMAIL_EVENTS, NEWSLETTER_EVENTS} from 'ghost-admin/helpers/members-event-filter';
import {action} from '@ember/object'; import {action} from '@ember/object';
import {inject as service} from '@ember/service'; import {inject as service} from '@ember/service';
import {tracked} from '@glimmer/tracking'; import {tracked} from '@glimmer/tracking';
import {use} from 'ember-could-get-used-to-this';
export default class MembersActivityController extends Controller { export default class MembersActivityController extends Controller {
@service router; @service router;
@ -14,6 +16,8 @@ export default class MembersActivityController extends Controller {
@tracked excludedEvents = null; @tracked excludedEvents = null;
@tracked member = null; @tracked member = null;
@use memberRecord = new MemberFetcher(() => [this.member]);
// we don't want to show or allow filtering of certain events in some situations // we don't want to show or allow filtering of certain events in some situations
// - no member selected = don't show email events, they flood the list and the API can't paginate correctly // - no member selected = don't show email events, they flood the list and the API can't paginate correctly
// - newsletter is disabled = don't show email or newletter events // - newsletter is disabled = don't show email or newletter events
@ -35,16 +39,6 @@ export default class MembersActivityController extends Controller {
return (this.excludedEvents || '').split(',').concat(this.hiddenEvents); return (this.excludedEvents || '').split(',').concat(this.hiddenEvents);
} }
get memberRecord() {
if (!this.member) {
return null;
}
// TODO: {reload: true} here shouldn't be needed but without it
// the template renders nothing if the record is already in the store
return this.store.findRecord('member', this.member, {reload: true});
}
@action @action
changeExcludedEvents(newList) { changeExcludedEvents(newList) {
this.router.transitionTo({queryParams: {excludedEvents: newList}}); this.router.transitionTo({queryParams: {excludedEvents: newList}});

View file

@ -0,0 +1,26 @@
import {Resource} from 'ember-could-get-used-to-this';
import {inject as service} from '@ember/service';
import {tracked} from '@glimmer/tracking';
class MemberFetcher extends Resource {
@service store;
@tracked loadedMember = null;
get value() {
return this.loadedMember;
}
async setup() {
const [memberId] = this.args.positional;
if (!memberId) {
return;
}
const record = await this.store.findRecord('member', memberId);
this.loadedMember = record;
}
}
export default MemberFetcher;