Skip to content

Commit 7eb13f3

Browse files
authored
Merge pull request #1191 from cardstack/dynamic-scope
Use dynamic context for "card context"
2 parents f5c66a1 + 91cd9f7 commit 7eb13f3

18 files changed

+172
-151
lines changed

packages/base/card-api.gts

+17-36
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ import {
2222
isNotLoadedError,
2323
isNotReadyError,
2424
CardError,
25+
CardContextName,
2526
NotLoaded,
2627
NotReady,
2728
getField,
@@ -116,7 +117,6 @@ export interface CardContext {
116117
};
117118
};
118119
}>;
119-
renderedIn?: Component<any>;
120120
}
121121

122122
function isNotLoadedValue(val: any): val is NotLoadedValue {
@@ -284,11 +284,7 @@ export interface Field<
284284
): Promise<any>;
285285
emptyValue(instance: BaseDef): any;
286286
validate(instance: BaseDef, value: any): void;
287-
component(
288-
model: Box<BaseDef>,
289-
defaultFormat: Format,
290-
context?: CardContext,
291-
): BoxComponent;
287+
component(model: Box<BaseDef>, defaultFormat: Format): BoxComponent;
292288
getter(instance: BaseDef): BaseInstanceType<CardT>;
293289
queryableValue(value: any, stack: BaseDef[]): SearchT;
294290
queryMatcher(
@@ -754,12 +750,8 @@ class Contains<CardT extends FieldDefConstructor> implements Field<CardT, any> {
754750
);
755751
}
756752

757-
component(
758-
model: Box<BaseDef>,
759-
format: Format,
760-
context?: CardContext,
761-
): BoxComponent {
762-
return fieldComponent(this, model, format, context);
753+
component(model: Box<BaseDef>, format: Format): BoxComponent {
754+
return fieldComponent(this, model, format);
763755
}
764756
}
765757

@@ -1050,18 +1042,14 @@ class LinksTo<CardT extends CardDefConstructor> implements Field<CardT> {
10501042
return fieldInstance;
10511043
}
10521044

1053-
component(
1054-
model: Box<CardDef>,
1055-
format: Format,
1056-
context?: CardContext,
1057-
): BoxComponent {
1045+
component(model: Box<CardDef>, format: Format): BoxComponent {
10581046
if (format === 'edit' && !this.computeVia) {
10591047
let innerModel = model.field(
10601048
this.name as keyof BaseDef,
10611049
) as unknown as Box<CardDef | null>;
1062-
return getLinksToEditor(innerModel, this, context);
1050+
return getLinksToEditor(innerModel, this);
10631051
}
1064-
return fieldComponent(this, model, format, context);
1052+
return fieldComponent(this, model, format);
10651053
}
10661054
}
10671055

@@ -1436,11 +1424,7 @@ class LinksToMany<FieldT extends CardDefConstructor>
14361424
return fieldInstances;
14371425
}
14381426

1439-
component(
1440-
model: Box<CardDef>,
1441-
format: Format,
1442-
context?: CardContext,
1443-
): BoxComponent {
1427+
component(model: Box<CardDef>, format: Format): BoxComponent {
14441428
let fieldName = this.name as keyof BaseDef;
14451429
let arrayField = model.field(
14461430
fieldName,
@@ -1463,7 +1447,6 @@ class LinksToMany<FieldT extends CardDefConstructor>
14631447
field: this,
14641448
format: renderFormat ?? format,
14651449
cardTypeFor,
1466-
context,
14671450
});
14681451
}
14691452
}
@@ -1472,7 +1455,6 @@ function fieldComponent(
14721455
field: Field<typeof BaseDef>,
14731456
model: Box<BaseDef>,
14741457
defaultFormat: Format,
1475-
context?: CardContext,
14761458
): BoxComponent {
14771459
let fieldName = field.name as keyof BaseDef;
14781460
let card: typeof BaseDef;
@@ -1483,7 +1465,7 @@ function fieldComponent(
14831465
(model.value[fieldName]?.constructor as typeof BaseDef) ?? field.card;
14841466
}
14851467
let innerModel = model.field(fieldName) as unknown as Box<BaseDef>;
1486-
return getBoxComponent(card, defaultFormat, innerModel, field, context);
1468+
return getBoxComponent(card, defaultFormat, innerModel, field);
14871469
}
14881470

14891471
// our decorators are implemented by Babel, not TypeScript, so they have a
@@ -1661,13 +1643,8 @@ export class BaseDef {
16611643
return _createFromSerialized(this, data, doc, relativeTo, identityContext);
16621644
}
16631645

1664-
static getComponent(
1665-
card: BaseDef,
1666-
format: Format,
1667-
field?: Field,
1668-
context?: CardContext,
1669-
) {
1670-
return getComponent(card, format, field, context);
1646+
static getComponent(card: BaseDef, format: Format, field?: Field) {
1647+
return getComponent(card, format, field);
16711648
}
16721649

16731650
static assignInitialFieldValue(
@@ -2739,15 +2716,13 @@ export function getComponent(
27392716
model: BaseDef,
27402717
format: Format,
27412718
field?: Field,
2742-
context?: CardContext,
27432719
): BoxComponent {
27442720
let box = Box.create(model);
27452721
let boxComponent = getBoxComponent(
27462722
model.constructor as BaseDefConstructor,
27472723
format,
27482724
box,
27492725
field,
2750-
context,
27512726
);
27522727
return boxComponent;
27532728
}
@@ -3102,3 +3077,9 @@ type ElementType<T> = T extends (infer V)[] ? V : never;
31023077
function makeRelativeURL(maybeURL: string, opts?: SerializeOpts): string {
31033078
return opts?.maybeRelativeURL ? opts.maybeRelativeURL(maybeURL) : maybeURL;
31043079
}
3080+
3081+
declare module 'ember-provide-consume-context/context-registry' {
3082+
export default interface ContextRegistry {
3083+
[CardContextName]: CardContext;
3084+
}
3085+
}

packages/base/field-component.gts

+86-72
Original file line numberDiff line numberDiff line change
@@ -12,12 +12,14 @@ import {
1212
isCompoundField,
1313
formats,
1414
} from './card-api';
15-
import { getField } from '@cardstack/runtime-common';
15+
import { CardContextName, getField } from '@cardstack/runtime-common';
1616
import type { ComponentLike } from '@glint/template';
1717
import { CardContainer } from '@cardstack/boxel-ui/components';
1818
import Modifier from 'ember-modifier';
1919
import { initSharedState } from './shared-state';
2020
import { eq } from '@cardstack/boxel-ui/helpers';
21+
import { consume } from 'ember-provide-consume-context';
22+
import Component from '@glimmer/component';
2123

2224
interface BoxComponentSignature {
2325
Args: { Named: { format?: Format; displayContainer?: boolean } };
@@ -26,6 +28,33 @@ interface BoxComponentSignature {
2628

2729
export type BoxComponent = ComponentLike<BoxComponentSignature>;
2830

31+
interface CardContextConsumerSignature {
32+
Blocks: { default: [CardContext] };
33+
}
34+
35+
// cardComponentModifier, when provided, is used for the host environment to get access to card's rendered elements
36+
const DEFAULT_CARD_CONTEXT = {
37+
cardComponentModifier: class NoOpModifier extends Modifier<any> {
38+
modify() {}
39+
},
40+
actions: undefined,
41+
};
42+
43+
export class CardContextConsumer extends Component<CardContextConsumerSignature> {
44+
@consume(CardContextName) declare dynamicCardContext: CardContext;
45+
46+
get context(): CardContext {
47+
return {
48+
...DEFAULT_CARD_CONTEXT,
49+
...this.dynamicCardContext,
50+
};
51+
}
52+
53+
<template>
54+
{{yield this.context}}
55+
</template>
56+
}
57+
2958
const componentCache = initSharedState(
3059
'componentCache',
3160
() => new WeakMap<Box<BaseDef>, BoxComponent>(),
@@ -36,7 +65,6 @@ export function getBoxComponent(
3665
defaultFormat: Format,
3766
model: Box<BaseDef>,
3867
field: Field | undefined,
39-
context: CardContext = {},
4068
): BoxComponent {
4169
let stable = componentCache.get(model);
4270
if (stable) {
@@ -46,13 +74,6 @@ export function getBoxComponent(
4674
| { fields: FieldsTypeFor<BaseDef>; format: Format }
4775
| undefined;
4876

49-
// cardComponentModifier, when provided, is used for the host environment to get access to card's rendered elements
50-
let cardComponentModifier =
51-
context.cardComponentModifier ??
52-
class NoOpModifier extends Modifier<any> {
53-
modify() {}
54-
};
55-
5677
function lookupFormat(userFormat: Format | undefined): {
5778
Implementation: BaseDefComponent;
5879
fields: FieldsTypeFor<BaseDef>;
@@ -78,12 +99,7 @@ export function getBoxComponent(
7899
if (internalFieldsCache?.format === format) {
79100
fields = internalFieldsCache.fields;
80101
} else {
81-
fields = fieldsComponentsFor(
82-
{},
83-
model,
84-
defaultFieldFormat(format),
85-
context,
86-
);
102+
fields = fieldsComponentsFor({}, model, defaultFieldFormat(format));
87103
internalFieldsCache = { fields, format };
88104
}
89105

@@ -97,26 +113,57 @@ export function getBoxComponent(
97113
let component: TemplateOnlyComponent<{
98114
Args: { format?: Format; displayContainer?: boolean };
99115
}> = <template>
100-
{{#let
101-
(lookupFormat @format) (if (eq @displayContainer false) false true)
102-
as |f displayContainer|
103-
}}
104-
{{#if (isCard model.value)}}
105-
<CardContainer
106-
@displayBoundaries={{displayContainer}}
107-
class='field-component-card
108-
{{f.format}}-format display-container-{{displayContainer}}'
109-
{{cardComponentModifier
110-
card=model.value
111-
format=f.format
112-
fieldType=field.fieldType
113-
fieldName=field.name
114-
}}
115-
data-test-card-format={{f.format}}
116-
data-test-field-component-card
117-
{{! @glint-ignore Argument of type 'unknown' is not assignable to parameter of type 'Element'}}
118-
...attributes
119-
>
116+
<CardContextConsumer as |context|>
117+
{{#let
118+
(lookupFormat @format) (if (eq @displayContainer false) false true)
119+
as |f displayContainer|
120+
}}
121+
{{#if (isCard model.value)}}
122+
<CardContainer
123+
@displayBoundaries={{displayContainer}}
124+
class='field-component-card
125+
{{f.format}}-format display-container-{{displayContainer}}'
126+
{{context.cardComponentModifier
127+
card=model.value
128+
format=f.format
129+
fieldType=field.fieldType
130+
fieldName=field.name
131+
}}
132+
data-test-card-format={{f.format}}
133+
data-test-field-component-card
134+
{{! @glint-ignore Argument of type 'unknown' is not assignable to parameter of type 'Element'}}
135+
...attributes
136+
>
137+
<f.Implementation
138+
@cardOrField={{card}}
139+
@model={{model.value}}
140+
@fields={{f.fields}}
141+
@format={{f.format}}
142+
@displayContainer={{@displayContainer}}
143+
@set={{model.set}}
144+
@fieldName={{model.name}}
145+
@context={{context}}
146+
/>
147+
</CardContainer>
148+
{{else if (isCompoundField model.value)}}
149+
<div
150+
data-test-compound-field-format={{f.format}}
151+
data-test-compound-field-component
152+
{{! @glint-ignore Argument of type 'unknown' is not assignable to parameter of type 'Element'}}
153+
...attributes
154+
>
155+
<f.Implementation
156+
@cardOrField={{card}}
157+
@model={{model.value}}
158+
@fields={{f.fields}}
159+
@format={{f.format}}
160+
@displayContainer={{@displayContainer}}
161+
@set={{model.set}}
162+
@fieldName={{model.name}}
163+
@context={{context}}
164+
/>
165+
</div>
166+
{{else}}
120167
<f.Implementation
121168
@cardOrField={{card}}
122169
@model={{model.value}}
@@ -127,38 +174,9 @@ export function getBoxComponent(
127174
@fieldName={{model.name}}
128175
@context={{context}}
129176
/>
130-
</CardContainer>
131-
{{else if (isCompoundField model.value)}}
132-
<div
133-
data-test-compound-field-format={{f.format}}
134-
data-test-compound-field-component
135-
{{! @glint-ignore Argument of type 'unknown' is not assignable to parameter of type 'Element'}}
136-
...attributes
137-
>
138-
<f.Implementation
139-
@cardOrField={{card}}
140-
@model={{model.value}}
141-
@fields={{f.fields}}
142-
@format={{f.format}}
143-
@displayContainer={{@displayContainer}}
144-
@set={{model.set}}
145-
@fieldName={{model.name}}
146-
@context={{context}}
147-
/>
148-
</div>
149-
{{else}}
150-
<f.Implementation
151-
@cardOrField={{card}}
152-
@model={{model.value}}
153-
@fields={{f.fields}}
154-
@format={{f.format}}
155-
@displayContainer={{@displayContainer}}
156-
@set={{model.set}}
157-
@fieldName={{model.name}}
158-
@context={{context}}
159-
/>
160-
{{/if}}
161-
{{/let}}
177+
{{/if}}
178+
{{/let}}
179+
</CardContextConsumer>
162180
<style>
163181
.field-component-card.embedded-format {
164182
padding: var(--boxel-sp);
@@ -187,7 +205,6 @@ export function getBoxComponent(
187205
component,
188206
model,
189207
defaultFieldFormat(defaultFormat),
190-
context,
191208
);
192209

193210
// This cast is safe because we're returning a proxy that wraps component.
@@ -212,7 +229,6 @@ function fieldsComponentsFor<T extends BaseDef>(
212229
target: object,
213230
model: Box<T>,
214231
defaultFormat: Format,
215-
context?: CardContext,
216232
): FieldsTypeFor<T> {
217233
// This is a cache of the fields we've already created components for
218234
// so that they do not get recreated
@@ -248,7 +264,6 @@ function fieldsComponentsFor<T extends BaseDef>(
248264
let result = field.component(
249265
model as unknown as Box<BaseDef>,
250266
defaultFormat,
251-
context,
252267
);
253268
stableComponents.set(property, result);
254269
return result;
@@ -302,11 +317,10 @@ export function getPluralViewComponent(
302317
field: Field<typeof BaseDef>,
303318
boxedElement: Box<BaseDef>,
304319
) => typeof BaseDef,
305-
context?: CardContext,
306320
): BoxComponent {
307321
let getComponents = () =>
308322
model.children.map((child) =>
309-
getBoxComponent(cardTypeFor(field, child), format, child, field, context),
323+
getBoxComponent(cardTypeFor(field, child), format, child, field),
310324
); // Wrap the the components in a function so that the template is reactive to changes in the model (this is essentially a helper)
311325
let pluralViewComponent: TemplateOnlyComponent<BoxComponentSignature> =
312326
<template>

0 commit comments

Comments
 (0)