@@ -12,12 +12,14 @@ import {
12
12
isCompoundField ,
13
13
formats ,
14
14
} from ' ./card-api' ;
15
- import { getField } from ' @cardstack/runtime-common' ;
15
+ import { CardContextName , getField } from ' @cardstack/runtime-common' ;
16
16
import type { ComponentLike } from ' @glint/template' ;
17
17
import { CardContainer } from ' @cardstack/boxel-ui/components' ;
18
18
import Modifier from ' ember-modifier' ;
19
19
import { initSharedState } from ' ./shared-state' ;
20
20
import { eq } from ' @cardstack/boxel-ui/helpers' ;
21
+ import { consume } from ' ember-provide-consume-context' ;
22
+ import Component from ' @glimmer/component' ;
21
23
22
24
interface BoxComponentSignature {
23
25
Args: { Named: { format? : Format ; displayContainer? : boolean } };
@@ -26,6 +28,33 @@ interface BoxComponentSignature {
26
28
27
29
export type BoxComponent = ComponentLike <BoxComponentSignature >;
28
30
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
+
29
58
const componentCache = initSharedState (
30
59
' componentCache' ,
31
60
() => new WeakMap <Box <BaseDef >, BoxComponent >(),
@@ -36,7 +65,6 @@ export function getBoxComponent(
36
65
defaultFormat : Format ,
37
66
model : Box <BaseDef >,
38
67
field : Field | undefined ,
39
- context : CardContext = {},
40
68
): BoxComponent {
41
69
let stable = componentCache .get (model );
42
70
if (stable ) {
@@ -46,13 +74,6 @@ export function getBoxComponent(
46
74
| { fields: FieldsTypeFor <BaseDef >; format: Format }
47
75
| undefined ;
48
76
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
-
56
77
function lookupFormat(userFormat : Format | undefined ): {
57
78
Implementation: BaseDefComponent ;
58
79
fields: FieldsTypeFor <BaseDef >;
@@ -78,12 +99,7 @@ export function getBoxComponent(
78
99
if (internalFieldsCache ?.format === format ) {
79
100
fields = internalFieldsCache .fields ;
80
101
} else {
81
- fields = fieldsComponentsFor (
82
- {},
83
- model ,
84
- defaultFieldFormat (format ),
85
- context ,
86
- );
102
+ fields = fieldsComponentsFor ({}, model , defaultFieldFormat (format ));
87
103
internalFieldsCache = { fields , format };
88
104
}
89
105
@@ -97,26 +113,57 @@ export function getBoxComponent(
97
113
let component: TemplateOnlyComponent <{
98
114
Args: { format? : Format ; displayContainer? : boolean };
99
115
}> = <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 }}
120
167
<f.Implementation
121
168
@ cardOrField ={{card }}
122
169
@ model ={{model.value }}
@@ -127,38 +174,9 @@ export function getBoxComponent(
127
174
@ fieldName ={{model.name }}
128
175
@ context ={{context }}
129
176
/>
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 >
162
180
<style >
163
181
.field-component-card.embedded-format {
164
182
padding : var (--boxel-sp );
@@ -187,7 +205,6 @@ export function getBoxComponent(
187
205
component ,
188
206
model ,
189
207
defaultFieldFormat (defaultFormat ),
190
- context ,
191
208
);
192
209
193
210
// This cast is safe because we're returning a proxy that wraps component.
@@ -212,7 +229,6 @@ function fieldsComponentsFor<T extends BaseDef>(
212
229
target : object ,
213
230
model : Box <T >,
214
231
defaultFormat : Format ,
215
- context ? : CardContext ,
216
232
): FieldsTypeFor <T > {
217
233
// This is a cache of the fields we've already created components for
218
234
// so that they do not get recreated
@@ -248,7 +264,6 @@ function fieldsComponentsFor<T extends BaseDef>(
248
264
let result = field .component (
249
265
model as unknown as Box <BaseDef >,
250
266
defaultFormat ,
251
- context ,
252
267
);
253
268
stableComponents .set (property , result );
254
269
return result ;
@@ -302,11 +317,10 @@ export function getPluralViewComponent(
302
317
field : Field <typeof BaseDef >,
303
318
boxedElement : Box <BaseDef >,
304
319
) => typeof BaseDef ,
305
- context ? : CardContext ,
306
320
): BoxComponent {
307
321
let getComponents = () =>
308
322
model .children .map ((child ) =>
309
- getBoxComponent (cardTypeFor (field , child ), format , child , field , context ),
323
+ getBoxComponent (cardTypeFor (field , child ), format , child , field ),
310
324
); // Wrap the the components in a function so that the template is reactive to changes in the model (this is essentially a helper)
311
325
let pluralViewComponent: TemplateOnlyComponent <BoxComponentSignature > =
312
326
<template >
0 commit comments