Skip to content

Commit

Permalink
[wip] starting to refactor toward decorator-driven fields
Browse files Browse the repository at this point in the history
  • Loading branch information
ef4 committed Jun 4, 2024
1 parent 48bae41 commit 45c93ee
Show file tree
Hide file tree
Showing 2 changed files with 112 additions and 46 deletions.
112 changes: 84 additions & 28 deletions packages/base/card-api.gts
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,7 @@ import {
} from '@cardstack/runtime-common';
import type { ComponentLike } from '@glint/template';
import { initSharedState } from './shared-state';
import { tracked } from '@glimmer/tracking';

export { primitive, isField, type BoxComponent };
export const serialize = Symbol.for('cardstack-serialize');
Expand Down Expand Up @@ -293,7 +294,7 @@ export interface Field<
card: CardT;
name: string;
fieldType: FieldType;
computeVia: undefined | string | (() => unknown);
isComputed: boolean;
description: undefined | string;
// there exists cards that we only ever run in the host without
// the isolated renderer (RoomField), which means that we cannot
Expand Down Expand Up @@ -424,7 +425,7 @@ class ContainsMany<FieldT extends FieldDefConstructor>
readonly fieldType = 'containsMany';
constructor(
private cardThunk: () => FieldT,
readonly computeVia: undefined | string | (() => unknown),
readonly isComputed: boolean,
readonly name: string,
readonly description: string | undefined,
readonly isUsed: undefined | true,
Expand Down Expand Up @@ -653,7 +654,7 @@ class Contains<CardT extends FieldDefConstructor> implements Field<CardT, any> {
readonly fieldType = 'contains';
constructor(
private cardThunk: () => CardT,
readonly computeVia: undefined | string | (() => unknown),
readonly isComputed: boolean,
readonly name: string,
readonly description: string | undefined,
readonly isUsed: undefined | true,
Expand Down Expand Up @@ -799,7 +800,7 @@ class LinksTo<CardT extends CardDefConstructor> implements Field<CardT> {
readonly fieldType = 'linksTo';
constructor(
private cardThunk: () => CardT,
readonly computeVia: undefined | string | (() => unknown),
readonly isComputed: boolean,
readonly name: string,
readonly description: string | undefined,
readonly isUsed: undefined | true,
Expand Down Expand Up @@ -1116,7 +1117,7 @@ class LinksToMany<FieldT extends CardDefConstructor>
readonly fieldType = 'linksToMany';
constructor(
private cardThunk: () => FieldT,
readonly computeVia: undefined | string | (() => unknown),
readonly isComputed: boolean,
readonly name: string,
readonly description: string | undefined,
readonly isUsed: undefined | true,
Expand Down Expand Up @@ -1571,25 +1572,75 @@ export function containsMany<FieldT extends FieldDefConstructor>(
}
containsMany[fieldType] = 'contains-many' as FieldType;

interface BabelDecoratorDescriptor {
configurable?: boolean;
enumerable?: boolean;
writable?: boolean;
get?(): any;
set?(v: any): void;
initializer?: null | (() => any);
value?: any;
}

type BabelDecorator = (
target: object,
prop: string | symbol,
desc: BabelDecoratorDescriptor,
) => BabelDecoratorDescriptor | null | undefined | void;

interface TrackedDescriptor<T> {
get(): T;
set(value: T): void;
}

function isTrackedDescriptor<T>(desc: any): desc is TrackedDescriptor<T> {
return Boolean(desc?.get && desc?.set);
}

export function contains<FieldT extends FieldDefConstructor>(
field: FieldT,
options?: Options,
): BaseInstanceType<FieldT> {
return {
setupField(fieldName: string) {
return makeDescriptor(
new Contains(
cardThunk(field),
options?.computeVia,
fieldName,
options?.description,
options?.isUsed,
),
);
},
} as any;
): PropertyDecorator {
// our decorators are implemented by Babel, not TypeScript, so they have a
// different signature than Typescript thinks they do.
let decorator: BabelDecorator = function (
target,
key,
desc,
): PropertyDescriptor {
if (typeof key !== 'string') {
throw new Error(`"contains" decorator only supports string field names`);
}
let isComputed = Boolean(desc.get);
let result: PropertyDescriptor;
if (isComputed) {
result = desc;
} else {
const trackedDesc = tracked(target, key, desc)!;

if (!isTrackedDescriptor(trackedDesc)) {
throw new Error(`bug: got unexpected result from @glimmer/tracking`);
}
result = {
get() {
return trackedDesc.get.call(this);
},
set(value) {
trackedDesc.set.call(this, value);
},
};
}
(result.get as any)[isField] = new Contains(
cardThunk(field),
isComputed,
key,
options?.description,
options?.isUsed,
);
return result;
};
return decorator as unknown as PropertyDecorator;
}
contains[fieldType] = 'contains' as FieldType;

export function linksTo<CardT extends CardDefConstructor>(
cardOrThunk: CardT | (() => CardT),
Expand Down Expand Up @@ -2038,13 +2089,19 @@ export class CardDef extends BaseDef {
[isSavedInstance] = false;
[realmInfo]: RealmInfo | undefined = undefined;
[realmURL]: URL | undefined = undefined;
@field id = contains(IDField);
@field title = contains(StringField);
@field description = contains(StringField);
@contains(IDField) declare id: BaseInstanceType<typeof IDField>;
@contains(StringField) declare title: BaseInstanceType<typeof StringField>;
@contains(StringField) declare description: BaseInstanceType<
typeof StringField
>;

// TODO: this will probably be an image or image url field card when we have it
// UPDATE: we now have a Base64ImageField card. we can probably refactor this
// to use it directly now (or wait until a better image field comes along)
@field thumbnailURL = contains(MaybeBase64Field);
@contains(MaybeBase64Field) declare thumbnailURL: BaseInstanceType<
typeof MaybeBase64Field
>;

static displayName = 'Card';
static isCardDef = true;

Expand Down Expand Up @@ -2382,7 +2439,7 @@ function serializedGet<CardT extends BaseDefConstructor>(
`tried to serializedGet field ${fieldName} which does not exist in card ${model.constructor.name}`,
);
}
return field.serialize(peekAtField(model, fieldName), doc, visited, opts);
return field.serialize((model as any)[fieldName], doc, visited, opts);
}

async function getDeserializedValue<CardT extends BaseDefConstructor>({
Expand Down Expand Up @@ -2707,11 +2764,10 @@ async function _updateFromSerialized<T extends BaseDefConstructor>(
`cannot change the id for saved instance ${originalId}`,
);
}
let deserialized = getDataBucket(instance);

// Before updating field's value, we also have to make sure
// the subscribers also subscribes to a new value.
let existingValue = deserialized.get(field.name as string);
let existingValue = (instance as any)[field.name];
if (
isCardOrField(existingValue) ||
isArrayOfCardOrField(existingValue) ||
Expand All @@ -2720,7 +2776,7 @@ async function _updateFromSerialized<T extends BaseDefConstructor>(
) {
applySubscribersToInstanceValue(instance, field, existingValue, value);
}
deserialized.set(field.name as string, value);
(instance as any)[field.name] = value;
logger.log(recompute(instance));
}
if (isCardInstance(instance) && resource.id != null) {
Expand Down
46 changes: 28 additions & 18 deletions packages/host/tests/integration/components/computed-test.gts
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import {
linksToMany,
} from '../../helpers/base-realm';
import { renderCard } from '../../helpers/render-component';
import { BaseInstanceType } from 'https://cardstack.com/base/card-api';

let loader: Loader;

Expand All @@ -53,26 +54,35 @@ module('Integration | computeds', function (hooks) {
async () => await loader.import(`${baseRealm.url}card-api`),
);

test('can render a synchronous computed field', async function (assert) {
class Person extends CardDef {
@field firstName = contains(StringField);
@field lastName = contains(StringField);
@field fullName = contains(StringField, {
computeVia: function (this: Person) {
QUnit.only(
'can render a synchronous computed field',
async function (assert) {
class Person extends CardDef {
@contains(StringField) declare firstName: BaseInstanceType<
typeof StringField
>;

@contains(StringField) declare lastName: BaseInstanceType<
typeof StringField
>;

@contains(StringField)
get fullName(): BaseInstanceType<typeof StringField> {
return `${this.firstName} ${this.lastName}`;
},
});
static isolated = class Isolated extends Component<typeof this> {
<template>
<@fields.fullName />
</template>
};
}
}

let mango = new Person({ firstName: 'Mango', lastName: 'Abdel-Rahman' });
let root = await renderCard(loader, mango, 'isolated');
assert.strictEqual(root.textContent!.trim(), 'Mango Abdel-Rahman');
});
static isolated = class Isolated extends Component<typeof this> {
<template>
<@fields.fullName />
</template>
};
}

let mango = new Person({ firstName: 'Mango', lastName: 'Abdel-Rahman' });
let root = await renderCard(loader, mango, 'isolated');
assert.strictEqual(root.textContent!.trim(), 'Mango Abdel-Rahman');
},
);

test('can render a synchronous computed field (using a string in `computeVia`)', async function (assert) {
class Person extends CardDef {
Expand Down

0 comments on commit 45c93ee

Please sign in to comment.