Skip to content

Commit 2d55163

Browse files
committed
Merge branch 'main' into cs-7598-ai-assistant-chat-should-automatically-display-return-value
2 parents 4d942c1 + 48bc199 commit 2d55163

23 files changed

+348
-169
lines changed

packages/base/command.gts

+5
Original file line numberDiff line numberDiff line change
@@ -114,6 +114,7 @@ export class SendAiAssistantMessageResult extends CardDef {
114114
@field eventId = contains(StringField);
115115
}
116116

117+
<<<<<<< HEAD
117118
export class GetBoxelUIStateResult extends CardDef {
118119
@field submode = contains(StringField);
119120
//TODO expand this to include more of the UI state:
@@ -129,3 +130,7 @@ export class LegacyGenerateAppModuleResult extends CardDef {
129130
@field moduleId = contains(StringField);
130131
@field source = contains(StringField);
131132
}
133+
134+
export class OpenAiAssistantRoomInput extends CardDef {
135+
@field roomId = contains(StringField);
136+
}

packages/experiments-realm/AiAppGenerator/create-boxel-app-command.ts

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ export default class CreateBoxelApp extends Command<
2222
this.commandContext,
2323
undefined,
2424
);
25+
2526
let { productRequirements: prdCard, roomId } =
2627
await createPRDCommand.execute(input);
2728

packages/experiments-realm/AiAppGenerator/create-product-requirements-command.ts

+9
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ import ReloadCardCommand from '@cardstack/boxel-host/commands/reload-card';
1414
import CreateAIAssistantRoomCommand from '@cardstack/boxel-host/commands/create-ai-assistant-room';
1515
import AddSkillsToRoomCommand from '@cardstack/boxel-host/commands/add-skills-to-room';
1616
import SendAiAssistantMessageCommand from '@cardstack/boxel-host/commands/send-ai-assistant-message';
17+
import OpenAiAssistantRoomCommand from '@cardstack/boxel-host/commands/open-ai-assistant-room';
1718

1819
export class CreateProductRequirementsInput extends CardDef {
1920
@field targetAudience = contains(StringField);
@@ -77,6 +78,14 @@ export default class CreateProductRequirementsInstance extends Command<
7778
let { roomId } = await createRoomCommand.execute({
7879
name: 'Product Requirements Doc Creation',
7980
});
81+
82+
let openAiAssistantRoomCommand = new OpenAiAssistantRoomCommand(
83+
this.commandContext,
84+
);
85+
await openAiAssistantRoomCommand.execute({
86+
roomId,
87+
});
88+
8089
let addSkillsToRoomCommand = new AddSkillsToRoomCommand(
8190
this.commandContext,
8291
);

packages/experiments-realm/ai-app-generator.gts

+2
Original file line numberDiff line numberDiff line change
@@ -362,6 +362,7 @@ class DashboardTab extends GlimmerComponent<{
362362
throw new Error('Missing commandContext');
363363
}
364364
let command = new CreateBoxelApp(commandContext, undefined);
365+
365366
this.isGenerating = true;
366367
try {
367368
await command.execute(
@@ -471,6 +472,7 @@ class RequirementsTab extends GlimmerComponent<{
471472
if (!this.cardRef) {
472473
throw new Error('Can not create a card without a card ref.');
473474
}
475+
474476
await this.args.context?.actions?.createCard?.(
475477
this.cardRef,
476478
this.args.currentRealm,

packages/host/app/commands/index.ts

+5
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import { VirtualNetwork } from '@cardstack/runtime-common';
22

33
import * as AddSkillsToRoomCommandModule from './add-skills-to-room';
44
import * as CreateAIAssistantRoomCommandModule from './create-ai-assistant-room';
5+
import * as OpenAiAssistantRoomCommandModule from './open-ai-assistant-room';
56
import * as PatchCardCommandModule from './patch-card';
67
import * as ReloadCardCommandModule from './reload-card';
78
import * as SaveCardCommandModule from './save-card';
@@ -32,6 +33,10 @@ export function shimHostCommands(virtualNetwork: VirtualNetwork) {
3233
'@cardstack/boxel-host/commands/save-card',
3334
SaveCardCommandModule,
3435
);
36+
virtualNetwork.shimModule(
37+
'@cardstack/boxel-host/commands/open-ai-assistant-room',
38+
OpenAiAssistantRoomCommandModule,
39+
);
3540
virtualNetwork.shimModule(
3641
'@cardstack/boxel-host/commands/send-ai-assistant-message',
3742
SendAiAssistantMessageModule,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { service } from '@ember/service';
2+
3+
import type * as BaseCommandModule from 'https://cardstack.com/base/command';
4+
5+
import HostBaseCommand from '../lib/host-base-command';
6+
7+
import MatrixService from '../services/matrix-service';
8+
import OperatorModeStateService from '../services/operator-mode-state-service';
9+
10+
export default class OpenAiAssistantRoomCommand extends HostBaseCommand<
11+
BaseCommandModule.OpenAiAssistantRoomInput,
12+
undefined
13+
> {
14+
@service private declare operatorModeStateService: OperatorModeStateService;
15+
@service private declare matrixService: MatrixService;
16+
17+
async getInputType() {
18+
let commandModule = await this.loadCommandModule();
19+
const { OpenAiAssistantRoomInput } = commandModule;
20+
return OpenAiAssistantRoomInput;
21+
}
22+
23+
protected async run(
24+
input: BaseCommandModule.OpenAiAssistantRoomInput,
25+
): Promise<undefined> {
26+
this.operatorModeStateService.aiAssistantOpen = true;
27+
this.matrixService.currentRoomId = input.roomId;
28+
}
29+
}

packages/host/app/commands/show-card.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@ export default class ShowCardCommand extends HostBaseCommand<
3232
1,
3333
);
3434
let newStackItem = await this.operatorModeStateService.createStackItem(
35-
input.cardToShow,
35+
new URL(input.cardToShow.id),
3636
newStackIndex,
3737
);
3838
this.operatorModeStateService.addItemToStack(newStackItem);

packages/host/app/components/ai-assistant/panel.gts

+12-12
Original file line numberDiff line numberDiff line change
@@ -179,9 +179,9 @@ export default class AiAssistantPanel extends Component<Signature> {
179179
</div>
180180
{{else if this.isReady}}
181181
{{! below if statement is covered in 'isReady' check above but added due to glint not realizing it }}
182-
{{#if this.currentRoomId}}
182+
{{#if this.matrixService.currentRoomId}}
183183
<Room
184-
@roomId={{this.currentRoomId}}
184+
@roomId={{this.matrixService.currentRoomId}}
185185
@monacoSDK={{this.monacoSDK}}
186186
/>
187187
{{/if}}
@@ -362,7 +362,6 @@ export default class AiAssistantPanel extends Component<Signature> {
362362
@service private declare router: RouterService;
363363
@service private declare commandService: CommandService;
364364

365-
@tracked private currentRoomId: string | undefined;
366365
@tracked private isShowingPastSessions = false;
367366
@tracked private roomToRename: SessionRoomData | undefined = undefined;
368367
@tracked private roomToDelete: SessionRoomData | undefined = undefined;
@@ -403,14 +402,14 @@ export default class AiAssistantPanel extends Component<Signature> {
403402
]);
404403
}
405404
if (roomToEnter) {
406-
this.currentRoomId = roomToEnter.roomId;
405+
this.matrixService.currentRoomId = roomToEnter.roomId;
407406
return;
408407
}
409408
}
410409

411410
let latestRoom = this.latestRoom;
412411
if (latestRoom) {
413-
this.currentRoomId = latestRoom.roomId;
412+
this.matrixService.currentRoomId = latestRoom.roomId;
414413
return;
415414
}
416415

@@ -423,8 +422,8 @@ export default class AiAssistantPanel extends Component<Signature> {
423422
}
424423

425424
private get roomResource() {
426-
return this.currentRoomId
427-
? this.roomResources.get(this.currentRoomId)
425+
return this.matrixService.currentRoomId
426+
? this.roomResources.get(this.matrixService.currentRoomId)
428427
: undefined;
429428
}
430429

@@ -464,7 +463,7 @@ export default class AiAssistantPanel extends Component<Signature> {
464463
} catch (e) {
465464
console.log(e);
466465
this.displayRoomError = true;
467-
this.currentRoomId = undefined;
466+
this.matrixService.currentRoomId = undefined;
468467
}
469468
});
470469

@@ -538,11 +537,10 @@ export default class AiAssistantPanel extends Component<Signature> {
538537

539538
@action
540539
private enterRoom(roomId: string, hidePastSessionsList = true) {
541-
this.currentRoomId = roomId;
540+
this.matrixService.currentRoomId = roomId;
542541
if (hidePastSessionsList) {
543542
this.hidePastSessions();
544543
}
545-
window.localStorage.setItem(CurrentRoomIdPersistenceKey, roomId);
546544
}
547545

548546
@action private setRoomToRename(room: SessionRoomData) {
@@ -591,7 +589,7 @@ export default class AiAssistantPanel extends Component<Signature> {
591589
window.localStorage.removeItem(NewSessionIdPersistenceKey);
592590
}
593591

594-
if (this.currentRoomId === roomId) {
592+
if (this.matrixService.currentRoomId === roomId) {
595593
window.localStorage.removeItem(CurrentRoomIdPersistenceKey);
596594
if (this.latestRoom) {
597595
this.enterRoom(this.latestRoom.roomId, false);
@@ -624,7 +622,9 @@ export default class AiAssistantPanel extends Component<Signature> {
624622

625623
private get isReady() {
626624
return Boolean(
627-
this.currentRoomId && this.maybeMonacoSDK && this.doCreateRoom.isIdle,
625+
this.matrixService.currentRoomId &&
626+
this.maybeMonacoSDK &&
627+
this.doCreateRoom.isIdle,
628628
);
629629
}
630630
}

packages/host/app/components/operator-mode/card-preview-panel/index.gts

+3-1
Original file line numberDiff line numberDiff line change
@@ -93,7 +93,9 @@ export default class CardPreviewPanel extends Component<Signature> {
9393
}
9494

9595
openInInteractMode = task(async () => {
96-
await this.operatorModeStateService.openCardInInteractMode(this.args.card);
96+
await this.operatorModeStateService.openCardInInteractMode(
97+
new URL(this.args.card.id),
98+
);
9799
});
98100

99101
<template>

packages/host/app/components/operator-mode/copy-button.gts

+7-7
Original file line numberDiff line numberDiff line change
@@ -32,10 +32,6 @@ const LEFT = 0;
3232
const RIGHT = 1;
3333

3434
export default class CopyButton extends Component<Signature> {
35-
@service declare loaderService: LoaderService;
36-
@service declare cardService: CardService;
37-
@service declare operatorModeStateService: OperatorModeStateService;
38-
3935
<template>
4036
{{#if (gt this.stacks.length 1)}}
4137
{{#if this.state}}
@@ -102,15 +98,19 @@ export default class CopyButton extends Component<Signature> {
10298
</style>
10399
</template>
104100

105-
get stacks() {
101+
@service private declare loaderService: LoaderService;
102+
@service private declare cardService: CardService;
103+
@service private declare operatorModeStateService: OperatorModeStateService;
104+
105+
private get stacks() {
106106
return this.operatorModeStateService.state?.stacks ?? [];
107107
}
108108

109-
get buttonKind() {
109+
private get buttonKind() {
110110
return this.args.isCopying ? 'primary-dark' : 'primary';
111111
}
112112

113-
get state() {
113+
private get state() {
114114
// Need to have 2 stacks in order for a copy button to exist
115115
if (this.stacks.length < 2) {
116116
return undefined;

packages/host/app/components/operator-mode/interact-submode.gts

+27-26
Original file line numberDiff line numberDiff line change
@@ -198,30 +198,28 @@ export default class InteractSubmode extends Component<Signature> {
198198
doc.data.meta.realmURL = opts.realmURL.href;
199199
}
200200

201-
let newCard = await here.cardService.createFromSerialized(
202-
doc.data,
203-
doc,
204-
relativeTo,
205-
);
206-
207201
let newItem = new StackItem({
208202
owner: here,
209-
card: newCard,
203+
newCard: { doc, relativeTo },
210204
format: opts?.cardModeAfterCreation ?? 'edit',
211205
request: new Deferred(),
212206
isLinkedCard: opts?.isLinkedCard,
213207
stackIndex,
214208
});
215209

216-
// TODO: it is important saveModel happens after newItem because it
217-
// looks like perhaps there is a race condition (or something else) when a
218-
// new linked card is created, and when it is added to the stack and closed
219-
// - the parent card is not updated with the new linked card
220-
await here.cardService.saveModel(newCard);
221-
222210
await newItem.ready();
211+
if (newItem.cardError) {
212+
console.error(
213+
`Encountered error creating card:\n${JSON.stringify(
214+
doc,
215+
null,
216+
2,
217+
)}\nError: ${JSON.stringify(newItem.cardError, null, 2)}`,
218+
);
219+
return undefined;
220+
}
223221
here.addToStack(newItem);
224-
return newCard;
222+
return newItem.card;
225223
},
226224
viewCard: async (
227225
cardOrURL: CardDef | URL,
@@ -233,9 +231,7 @@ export default class InteractSubmode extends Component<Signature> {
233231
}
234232
let newItem = new StackItem({
235233
owner: here,
236-
...(cardOrURL instanceof URL
237-
? { url: cardOrURL }
238-
: { card: cardOrURL }),
234+
url: cardOrURL instanceof URL ? cardOrURL : new URL(cardOrURL.id),
239235
format,
240236
stackIndex,
241237
});
@@ -561,19 +557,25 @@ export default class InteractSubmode extends Component<Signature> {
561557
private openSelectedSearchResultInStack = restartableTask(
562558
async (cardId: string) => {
563559
let waiterToken = waiter.beginAsync();
560+
let url = new URL(cardId);
564561
try {
565562
let searchSheetTrigger = this.searchSheetTrigger; // Will be set by showSearchWithTrigger
566-
let card = await this.cardService.getCard(cardId);
567-
if (!card) {
568-
return;
569-
}
570563

571564
// In case the left button was clicked, whatever is currently in stack with index 0 will be moved to stack with index 1,
572565
// and the card will be added to stack with index 0. shiftStack executes this logic.
573566
if (
574567
searchSheetTrigger ===
575568
SearchSheetTriggers.DropCardToLeftNeighborStackButton
576569
) {
570+
let newItem = new StackItem({
571+
owner: this,
572+
url,
573+
format: 'isolated',
574+
stackIndex: 0,
575+
});
576+
// it's important that we await the stack item readiness _before_
577+
// we mutate the stack, otherwise there are very odd visual artifacts
578+
await newItem.ready();
577579
for (
578580
let stackIndex = this.stacks.length - 1;
579581
stackIndex >= 0;
@@ -584,15 +586,14 @@ export default class InteractSubmode extends Component<Signature> {
584586
stackIndex + 1,
585587
);
586588
}
587-
await this.publicAPI(this, 0).viewCard(card, 'isolated');
588-
589+
this.addToStack(newItem);
589590
// In case the right button was clicked, the card will be added to stack with index 1.
590591
} else if (
591592
searchSheetTrigger ===
592593
SearchSheetTriggers.DropCardToRightNeighborStackButton
593594
) {
594595
await this.publicAPI(this, this.stacks.length).viewCard(
595-
card,
596+
url,
596597
'isolated',
597598
);
598599
} else {
@@ -606,15 +607,15 @@ export default class InteractSubmode extends Component<Signature> {
606607
numberOfStacks === 0 ||
607608
this.operatorModeStateService.stackIsEmpty(stackIndex)
608609
) {
609-
await this.publicAPI(this, 0).viewCard(card, 'isolated');
610+
await this.publicAPI(this, 0).viewCard(url, 'isolated');
610611
} else {
611612
stack = this.operatorModeStateService.rightMostStack();
612613
if (stack) {
613614
let bottomMostItem = stack[0];
614615
if (bottomMostItem) {
615616
let stackItem = new StackItem({
616617
owner: this,
617-
card,
618+
url,
618619
format: 'isolated',
619620
stackIndex,
620621
});

0 commit comments

Comments
 (0)