Skip to content

Commit 426dc00

Browse files
committed
Merge branch 'main' into ECO-172-fixed-crm
2 parents f4db552 + d190264 commit 426dc00

File tree

11 files changed

+130
-24
lines changed

11 files changed

+130
-24
lines changed

.github/workflows/pr-boxel-host.yml

-2
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,6 @@ jobs:
6363
AWS_S3_BUCKET: boxel-host-preview.stack.cards
6464
AWS_REGION: us-east-1
6565
AWS_CLOUDFRONT_DISTRIBUTION: EU4RGLH4EOCHJ
66-
ENABLE_PLAYGROUND: true
6766
AI_ASSISTANT_EXPERIMENTAL_ATTACHING_FILES_ENABLED: true
6867
with:
6968
package: boxel-host
@@ -94,7 +93,6 @@ jobs:
9493
AWS_S3_BUCKET: boxel-host-preview.boxel.ai
9594
AWS_REGION: us-east-1
9695
AWS_CLOUDFRONT_DISTRIBUTION: E2PZR9CIAW093B
97-
ENABLE_PLAYGROUND: true
9896
AI_ASSISTANT_EXPERIMENTAL_ATTACHING_FILES_ENABLED: true
9997
with:
10098
package: boxel-host
Loading

packages/boxel-ui/addon/src/icons.gts

+3
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@ import IconPencilCrossedOut from './icons/icon-pencil-crossed-out.gts';
4040
import IconPencilNotCrossedOut from './icons/icon-pencil-not-crossed-out.gts';
4141
import IconPlus from './icons/icon-plus.gts';
4242
import IconPlusCircle from './icons/icon-plus-circle.gts';
43+
import IconPlusThin from './icons/icon-plus-thin.gts';
4344
import IconSearch from './icons/icon-search.gts';
4445
import IconSearchThick from './icons/icon-search-thick.gts';
4546
import IconTrash from './icons/icon-trash.gts';
@@ -100,6 +101,7 @@ export const ALL_ICON_COMPONENTS = [
100101
IconPencilNotCrossedOut,
101102
IconPlus,
102103
IconPlusCircle,
104+
IconPlusThin,
103105
IconSearch,
104106
IconSearchThick,
105107
IconTrash,
@@ -161,6 +163,7 @@ export {
161163
IconPencilNotCrossedOut,
162164
IconPlus,
163165
IconPlusCircle,
166+
IconPlusThin,
164167
IconSearch,
165168
IconSearchThick,
166169
IconTrash,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// This file is auto-generated by 'pnpm rebuild:icons'
2+
import type { TemplateOnlyComponent } from '@ember/component/template-only';
3+
4+
import type { Signature } from './types.ts';
5+
6+
const IconComponent: TemplateOnlyComponent<Signature> = <template>
7+
<svg
8+
xmlns='http://www.w3.org/2000/svg'
9+
viewBox='3.5 3.5 17 17'
10+
...attributes
11+
><path
12+
fill='var(--icon-color, currentColor)'
13+
d='M19 11h-6V5a1 1 0 0 0-2 0v6H5a1 1 0 0 0 0 2h6v6a1 1 0 0 0 2 0v-6h6a1 1 0 0 0 0-2Z'
14+
/></svg>
15+
</template>;
16+
17+
// @ts-expect-error this is the only way to set a name on a Template Only Component currently
18+
IconComponent.name = 'IconPlusThin';
19+
export default IconComponent;

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

-7
Original file line numberDiff line numberDiff line change
@@ -41,8 +41,6 @@ import RecentFiles from '@cardstack/host/components/editor/recent-files';
4141
import CodeSubmodeEditorIndicator from '@cardstack/host/components/operator-mode/code-submode/editor-indicator';
4242
import SyntaxErrorDisplay from '@cardstack/host/components/operator-mode/syntax-error-display';
4343

44-
import ENV from '@cardstack/host/config/environment';
45-
4644
import { getCard } from '@cardstack/host/resources/card-resource';
4745
import { isReady, type FileResource } from '@cardstack/host/resources/file';
4846
import {
@@ -88,8 +86,6 @@ import DetailPanel from './detail-panel';
8886
import NewFileButton from './new-file-button';
8987
import SubmodeLayout from './submode-layout';
9088

91-
const isPlaygroundEnabled = ENV.featureFlags?.ENABLE_PLAYGROUND;
92-
9389
interface Signature {
9490
Args: {
9591
saveSourceOnClose: (url: URL, content: string) => void;
@@ -469,9 +465,6 @@ export default class CodeSubmode extends Component<Signature> {
469465
}
470466

471467
private get shouldDisplayPlayground() {
472-
if (!isPlaygroundEnabled) {
473-
return false;
474-
}
475468
return isCardDef(this.selectedCardOrField?.cardOrField);
476469
}
477470

packages/host/app/components/operator-mode/code-submode/playground-panel.gts

+49-9
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,10 @@ import { action } from '@ember/object';
44
import type Owner from '@ember/owner';
55
import { service } from '@ember/service';
66
import Component from '@glimmer/component';
7+
import { tracked } from '@glimmer/tracking';
78

89
import Folder from '@cardstack/boxel-icons/folder';
9-
import { restartableTask, task } from 'ember-concurrency';
10+
import { task } from 'ember-concurrency';
1011
import perform from 'ember-concurrency/helpers/perform';
1112
import window from 'ember-window-mock';
1213
import { TrackedObject } from 'tracked-built-ins';
@@ -18,13 +19,19 @@ import {
1819
CardHeader,
1920
} from '@cardstack/boxel-ui/components';
2021
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';
2228

2329
import {
2430
cardTypeDisplayName,
2531
cardTypeIcon,
2632
chooseCard,
2733
type Query,
34+
type LooseSingleCardDocument,
2835
type ResolvedCodeRef,
2936
} from '@cardstack/runtime-common';
3037

@@ -33,6 +40,7 @@ import { getCard } from '@cardstack/host/resources/card-resource';
3340
import { getCodeRef, type CardType } from '@cardstack/host/resources/card-type';
3441
import { ModuleContentsResource } from '@cardstack/host/resources/module-contents';
3542

43+
import type CardService from '@cardstack/host/services/card-service';
3644
import type OperatorModeStateService from '@cardstack/host/services/operator-mode-state-service';
3745
import type RealmService from '@cardstack/host/services/realm';
3846
import type RealmServerService from '@cardstack/host/services/realm-server';
@@ -87,7 +95,7 @@ const SelectedItem: TemplateOnlyComponent<{ Args: { title?: string } }> =
8795
</style>
8896
</template>;
8997

90-
const BeforeOptions: TemplateOnlyComponent<{ Args: {} }> = <template>
98+
const BeforeOptions: TemplateOnlyComponent = <template>
9199
<div class='before-options'>
92100
<span class='title'>
93101
Recent
@@ -109,13 +117,23 @@ const BeforeOptions: TemplateOnlyComponent<{ Args: {} }> = <template>
109117
interface AfterOptionsSignature {
110118
Args: {
111119
chooseCard: () => void;
120+
createNew: () => void;
121+
createNewIsRunning?: boolean;
112122
};
113123
}
114124
const AfterOptions: TemplateOnlyComponent<AfterOptionsSignature> = <template>
115125
<div class='after-options'>
116126
<span class='title'>
117127
Action
118128
</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>
119137
<button
120138
class='action'
121139
{{on 'click' @chooseCard}}
@@ -132,7 +150,6 @@ const AfterOptions: TemplateOnlyComponent<AfterOptionsSignature> = <template>
132150
border-top: var(--boxel-border);
133151
background-color: var(--boxel-light);
134152
padding: var(--boxel-sp-xs);
135-
margin-top: var(--boxel-sp-xxs);
136153
gap: var(--boxel-sp-xxs);
137154
}
138155
.title {
@@ -152,6 +169,9 @@ const AfterOptions: TemplateOnlyComponent<AfterOptionsSignature> = <template>
152169
.action:hover {
153170
background-color: var(--boxel-100);
154171
}
172+
.action-running {
173+
--boxel-loading-indicator-size: 16px;
174+
}
155175
</style>
156176
</template>;
157177

@@ -193,6 +213,8 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
193213
@afterOptionsComponent={{component
194214
AfterOptions
195215
chooseCard=(perform this.chooseCard)
216+
createNew=(perform this.createNew)
217+
createNewIsRunning=this.createNew.isRunning
196218
}}
197219
data-test-instance-chooser
198220
as |card|
@@ -284,9 +306,6 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
284306
height: var(--boxel-form-control-height);
285307
box-shadow: 0 5px 10px 0 rgba(0 0 0 / 40%);
286308
}
287-
:deep(.instances-dropdown-content > .ember-power-select-options) {
288-
max-height: 20rem;
289-
}
290309
:deep(
291310
.boxel-select__dropdown .ember-power-select-option[aria-current='true']
292311
),
@@ -355,10 +374,12 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
355374
</style>
356375
</template>
357376

377+
@service private declare cardService: CardService;
358378
@service private declare operatorModeStateService: OperatorModeStateService;
359379
@service private declare realm: RealmService;
360380
@service private declare realmServer: RealmServerService;
361381
@service declare recentFilesService: RecentFilesService;
382+
@tracked newCardJSON: LooseSingleCardDocument | undefined;
362383
private playgroundSelections: Record<
363384
string, // moduleId
364385
{ cardId: string; format: Format }
@@ -411,7 +432,8 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
411432

412433
private cardResource = getCard(
413434
this,
414-
() => this.playgroundSelections[this.args.moduleId]?.cardId,
435+
() =>
436+
this.newCardJSON ?? this.playgroundSelections[this.args.moduleId]?.cardId,
415437
{ isAutoSave: () => true },
416438
);
417439

@@ -458,6 +480,7 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
458480
}
459481

460482
private persistSelections = (cardId: string, format = this.format) => {
483+
this.newCardJSON = undefined;
461484
this.playgroundSelections[this.args.moduleId] = { cardId, format };
462485
window.localStorage.setItem(
463486
PlaygroundSelections,
@@ -477,7 +500,7 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
477500
this.persistSelections(this.card.id, format);
478501
}
479502

480-
private chooseCard = restartableTask(async () => {
503+
private chooseCard = task(async () => {
481504
let chosenCard: CardDef | undefined = await chooseCard({
482505
filter: { type: this.args.codeRef },
483506
});
@@ -488,6 +511,23 @@ class PlaygroundPanelContent extends Component<PlaygroundContentSignature> {
488511
}
489512
});
490513

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+
491531
private get canEdit() {
492532
return (
493533
this.format !== 'edit' &&

packages/host/app/config/environment.d.ts

-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,6 @@ declare const config: {
2929
assetsURL: string;
3030
stripePaymentLink: string;
3131
featureFlags?: {
32-
ENABLE_PLAYGROUND: boolean;
3332
AI_ASSISTANT_EXPERIMENTAL_ATTACHING_FILES_ENABLED: boolean;
3433
};
3534
};

packages/host/config/environment.js

-3
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,6 @@ module.exports = function (environment) {
4242
resolvedBaseRealmURL:
4343
process.env.RESOLVED_BASE_REALM_URL || 'http://localhost:4201/base/',
4444
featureFlags: {
45-
ENABLE_PLAYGROUND: process.env.ENABLE_PLAYGROUND || false,
4645
AI_ASSISTANT_EXPERIMENTAL_ATTACHING_FILES_ENABLED:
4746
process.env.AI_ASSISTANT_EXPERIMENTAL_ATTACHING_FILES_ENABLED || false,
4847
},
@@ -55,7 +54,6 @@ module.exports = function (environment) {
5554
// ENV.APP.LOG_TRANSITIONS_INTERNAL = true;
5655
// ENV.APP.LOG_VIEW_LOOKUPS = true;
5756
ENV.featureFlags = {
58-
ENABLE_PLAYGROUND: true,
5957
AI_ASSISTANT_EXPERIMENTAL_ATTACHING_FILES_ENABLED: true,
6058
};
6159
}
@@ -78,7 +76,6 @@ module.exports = function (environment) {
7876
ENV.minSaveTaskDurationMs = 0;
7977
ENV.sqlSchema = sqlSchema;
8078
ENV.featureFlags = {
81-
ENABLE_PLAYGROUND: true,
8279
AI_ASSISTANT_EXPERIMENTAL_ATTACHING_FILES_ENABLED: true,
8380
};
8481
}

packages/host/tests/acceptance/code-submode/playground-test.gts

+45
Original file line numberDiff line numberDiff line change
@@ -587,6 +587,51 @@ export class BlogPost extends CardDef {
587587
]);
588588
});
589589

590+
test<TestContextWithSSE>('can create new instance', async function (assert) {
591+
window.localStorage.removeItem('recent-files');
592+
await visitOperatorMode({
593+
submode: 'code',
594+
codePath: `${testRealmURL}blog-post.gts`,
595+
});
596+
let recentFiles = JSON.parse(window.localStorage.getItem('recent-files')!);
597+
assert.deepEqual(recentFiles[0], [testRealmURL, 'blog-post.gts']);
598+
await click('[data-boxel-selector-item-text="BlogPost"]');
599+
await click('[data-test-accordion-item="playground"] button');
600+
assert
601+
.dom('[data-test-instance-chooser] [data-test-selected-item]')
602+
.doesNotExist();
603+
604+
await click('[data-test-instance-chooser]');
605+
await this.expectEvents({
606+
assert,
607+
realm,
608+
expectedNumberOfEvents: 2,
609+
callback: async () => {
610+
await click('[data-test-create-instance]');
611+
},
612+
});
613+
recentFiles = JSON.parse(window.localStorage.getItem('recent-files')!);
614+
assert.strictEqual(recentFiles.length, 2, 'recent file count is correct');
615+
let newCardId = `${recentFiles[0][0]}${recentFiles[0][1]}`.replace(
616+
'.json',
617+
'',
618+
);
619+
assert
620+
.dom('[data-test-instance-chooser] [data-test-selected-item]')
621+
.hasText('Untitled Blog Post', 'created instance is selected');
622+
assert
623+
.dom(
624+
`[data-test-playground-panel] [data-test-card="${newCardId}"][data-test-card-format="edit"]`,
625+
)
626+
.exists('new card is rendered in edit format');
627+
628+
await click('[data-test-instance-chooser]');
629+
assert
630+
.dom('[data-option-index]')
631+
.exists({ count: 1 }, 'dropdown instance count is correct');
632+
assert.dom('[data-option-index]').containsText('Blog Post');
633+
});
634+
590635
test<TestContextWithSSE>('playground preview for card with contained fields can live update when module changes', async function (assert) {
591636
// change: added "Hello" before rendering title on the template
592637
const authorCard = `import { contains, field, CardDef, Component } from "https://cardstack.com/base/card-api";

packages/runtime-common/marked-sync.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { marked } from 'marked';
22
import { sanitizeHtml } from './dompurify-runtime';
3-
import { md5 } from 'super-fast-md5';
3+
import { simpleHash } from '.';
44

55
const CODEBLOCK_KEY_PREFIX = 'codeblock_';
66

@@ -12,7 +12,7 @@ export function markedSync(markdown: string) {
1212
// markdown, please use the `CodeBlock` modifier to render the
1313
// markdown.
1414
code(code, language = '') {
15-
let id = `${CODEBLOCK_KEY_PREFIX}${md5(Date.now() + language + code)}`;
15+
let id = `${CODEBLOCK_KEY_PREFIX}${simpleHash(Date.now() + language + code)}`;
1616
// we pass the code thru using localstorage instead of in the DOM,
1717
// that way we don't have to worry about escaping code. note that the
1818
// DOM wants to render "<template>" strings when we put them in the

packages/runtime-common/utils.ts

+11
Original file line numberDiff line numberDiff line change
@@ -61,3 +61,14 @@ export function decodeWebSafeBase64(encoded: string): string {
6161

6262
return Buffer.from(base64, 'base64').toString('utf-8');
6363
}
64+
65+
// This is the djb2_xor hash function from http://www.cse.yorku.ca/~oz/hash.html
66+
export function simpleHash(str: string) {
67+
let len = str.length;
68+
let h = 5381;
69+
70+
for (let i = 0; i < len; i++) {
71+
h = (h * 33) ^ str.charCodeAt(i);
72+
}
73+
return (h >>> 0).toString(16);
74+
}

0 commit comments

Comments
 (0)