@@ -4,9 +4,10 @@ import { action } from '@ember/object';
4
4
import type Owner from ' @ember/owner' ;
5
5
import { service } from ' @ember/service' ;
6
6
import Component from ' @glimmer/component' ;
7
+ import { tracked } from ' @glimmer/tracking' ;
7
8
8
9
import Folder from ' @cardstack/boxel-icons/folder' ;
9
- import { restartableTask , task } from ' ember-concurrency' ;
10
+ import { task } from ' ember-concurrency' ;
10
11
import perform from ' ember-concurrency/helpers/perform' ;
11
12
import window from ' ember-window-mock' ;
12
13
import { TrackedObject } from ' tracked-built-ins' ;
@@ -18,13 +19,19 @@ import {
18
19
CardHeader ,
19
20
} from ' @cardstack/boxel-ui/components' ;
20
21
import { eq , or , MenuItem } from ' @cardstack/boxel-ui/helpers' ;
21
- import { Eye , IconCode , IconLink } from ' @cardstack/boxel-ui/icons' ;
22
+ import {
23
+ Eye ,
24
+ IconCode ,
25
+ IconLink ,
26
+ IconPlusThin ,
27
+ } from ' @cardstack/boxel-ui/icons' ;
22
28
23
29
import {
24
30
cardTypeDisplayName ,
25
31
cardTypeIcon ,
26
32
chooseCard ,
27
33
type Query ,
34
+ type LooseSingleCardDocument ,
28
35
type ResolvedCodeRef ,
29
36
} from ' @cardstack/runtime-common' ;
30
37
@@ -33,6 +40,7 @@ import { getCard } from '@cardstack/host/resources/card-resource';
33
40
import { getCodeRef , type CardType } from ' @cardstack/host/resources/card-type' ;
34
41
import { ModuleContentsResource } from ' @cardstack/host/resources/module-contents' ;
35
42
43
+ import type CardService from ' @cardstack/host/services/card-service' ;
36
44
import type OperatorModeStateService from ' @cardstack/host/services/operator-mode-state-service' ;
37
45
import type RealmService from ' @cardstack/host/services/realm' ;
38
46
import type RealmServerService from ' @cardstack/host/services/realm-server' ;
@@ -87,7 +95,7 @@ const SelectedItem: TemplateOnlyComponent<{ Args: { title?: string } }> =
87
95
< /style >
88
96
</template >;
89
97
90
- const BeforeOptions: TemplateOnlyComponent <{ Args : {} }> = <template >
98
+ const BeforeOptions: TemplateOnlyComponent = <template >
91
99
<div class =' before-options' >
92
100
<span class =' title' >
93
101
Recent
@@ -109,13 +117,23 @@ const BeforeOptions: TemplateOnlyComponent<{ Args: {} }> = <template>
109
117
interface AfterOptionsSignature {
110
118
Args: {
111
119
chooseCard: () => void ;
120
+ createNew: () => void ;
121
+ createNewIsRunning? : boolean ;
112
122
};
113
123
}
114
124
const AfterOptions: TemplateOnlyComponent <AfterOptionsSignature > = <template >
115
125
<div class =' after-options' >
116
126
<span class =' title' >
117
127
Action
118
128
</span >
129
+ <button class =' action' {{on ' click' @ createNew}} data-test-create-instance >
130
+ {{#if @ createNewIsRunning }}
131
+ <LoadingIndicator class =' action-running' />
132
+ {{else }}
133
+ <IconPlusThin width =' 16px' height =' 16px' />
134
+ {{/if }}
135
+ New card instance
136
+ </button >
119
137
<button
120
138
class =' action'
121
139
{{on ' click' @ chooseCard}}
@@ -132,7 +150,6 @@ const AfterOptions: TemplateOnlyComponent<AfterOptionsSignature> = <template>
132
150
border-top : var (--boxel-border );
133
151
background-color : var (--boxel-light );
134
152
padding : var (--boxel-sp-xs );
135
- margin-top : var (--boxel-sp-xxs );
136
153
gap : var (--boxel-sp-xxs );
137
154
}
138
155
.title {
@@ -152,6 +169,9 @@ const AfterOptions: TemplateOnlyComponent<AfterOptionsSignature> = <template>
152
169
.action :hover {
153
170
background-color : var (--boxel-100 );
154
171
}
172
+ .action-running {
173
+ --boxel-loading-indicator-size : 16px ;
174
+ }
155
175
< /style >
156
176
</template >;
157
177
@@ -193,6 +213,8 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
193
213
@ afterOptionsComponent ={{component
194
214
AfterOptions
195
215
chooseCard =( perform this . chooseCard)
216
+ createNew =( perform this . createNew)
217
+ createNewIsRunning =this . createNew.isRunning
196
218
}}
197
219
data-test-instance-chooser
198
220
as | card |
@@ -284,9 +306,6 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
284
306
height : var (--boxel-form-control-height );
285
307
box-shadow : 0 5px 10px 0 rgba (0 0 0 / 40% );
286
308
}
287
- :deep(.instances-dropdown-content > .ember-power-select-options ) {
288
- max-height : 20rem ;
289
- }
290
309
:deep(
291
310
.boxel-select__dropdown .ember-power-select-option [aria-current = ' true' ]
292
311
),
@@ -355,10 +374,12 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
355
374
< /style >
356
375
</template >
357
376
377
+ @service private declare cardService: CardService ;
358
378
@service private declare operatorModeStateService: OperatorModeStateService ;
359
379
@service private declare realm: RealmService ;
360
380
@service private declare realmServer: RealmServerService ;
361
381
@service declare recentFilesService: RecentFilesService ;
382
+ @tracked newCardJSON: LooseSingleCardDocument | undefined ;
362
383
private playgroundSelections: Record <
363
384
string , // moduleId
364
385
{ cardId: string ; format: Format }
@@ -411,7 +432,8 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
411
432
412
433
private cardResource = getCard (
413
434
this ,
414
- () => this .playgroundSelections [this .args .moduleId ]?.cardId ,
435
+ () =>
436
+ this .newCardJSON ?? this .playgroundSelections [this .args .moduleId ]?.cardId ,
415
437
{ isAutoSave : () => true },
416
438
);
417
439
@@ -458,6 +480,7 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
458
480
}
459
481
460
482
private persistSelections = (cardId : string , format = this .format ) => {
483
+ this .newCardJSON = undefined ;
461
484
this .playgroundSelections [this .args .moduleId ] = { cardId , format };
462
485
window .localStorage .setItem (
463
486
PlaygroundSelections ,
@@ -477,7 +500,7 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
477
500
this .persistSelections (this .card .id , format );
478
501
}
479
502
480
- private chooseCard = restartableTask (async () => {
503
+ private chooseCard = task (async () => {
481
504
let chosenCard: CardDef | undefined = await chooseCard ({
482
505
filter: { type: this .args .codeRef },
483
506
});
@@ -488,6 +511,23 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
488
511
}
489
512
});
490
513
514
+ // TODO: convert this to @action once we no longer need to await below
515
+ private createNew = task (async () => {
516
+ this .newCardJSON = {
517
+ data: {
518
+ meta: {
519
+ adoptsFrom: this .args .codeRef ,
520
+ realmURL: this .operatorModeStateService .realmURL .href ,
521
+ },
522
+ },
523
+ };
524
+ await this .cardResource .loaded ; // TODO: remove await when card-resource is refactored
525
+ if (this .card ) {
526
+ this .recentFilesService .addRecentFileUrl (` ${this .card .id }.json ` );
527
+ this .persistSelections (this .card .id , ' edit' ); // open new instance in playground in edit format
528
+ }
529
+ });
530
+
491
531
private get canEdit() {
492
532
return (
493
533
this .format !== ' edit' &&
0 commit comments