Skip to content

Commit 29270c1

Browse files
committed
Show ApplyButton's preparing state when a command is being prepared
1 parent 040d46f commit 29270c1

File tree

8 files changed

+456
-99
lines changed

8 files changed

+456
-99
lines changed

packages/host/app/components/ai-assistant/apply-button/index.gts

+86-2
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,12 @@ import { eq } from '@cardstack/boxel-ui/helpers';
55
import { CheckMark, Exclamation } from '@cardstack/boxel-ui/icons';
66
import { setCssVar } from '@cardstack/boxel-ui/modifiers';
77

8-
export type ApplyButtonState = 'ready' | 'applying' | 'applied' | 'failed';
8+
export type ApplyButtonState =
9+
| 'ready'
10+
| 'applying'
11+
| 'applied'
12+
| 'failed'
13+
| 'preparing';
914

1015
interface Signature {
1116
Element: HTMLButtonElement | HTMLDivElement;
@@ -34,6 +39,18 @@ const AiAssistantApplyButton: TemplateOnlyComponent<Signature> = <template>
3439
<CheckMark width='16' height='16' />
3540
{{else if (eq @state 'failed')}}
3641
<Exclamation width='16' height='16' />
42+
{{else if (eq @state 'preparing')}}
43+
<BoxelButton
44+
@kind='secondary-dark'
45+
@size='small'
46+
class='apply-button'
47+
tabindex='-1'
48+
{{setCssVar boxel-button-text-color='var(--boxel-200)'}}
49+
data-test-apply-state='preparing'
50+
...attributes
51+
>
52+
Working…
53+
</BoxelButton>
3754
{{/if}}
3855
</div>
3956
{{/if}}
@@ -67,7 +84,68 @@ const AiAssistantApplyButton: TemplateOnlyComponent<Signature> = <template>
6784
width: 58px;
6885
border-radius: 100px;
6986
}
70-
.state-indicator:not(.applying) {
87+
88+
.state-indicator.preparing {
89+
width: 78px;
90+
padding: 1px;
91+
border-radius: 100px;
92+
}
93+
.state-indicator.preparing .apply-button {
94+
border: 0;
95+
min-width: 74px;
96+
}
97+
.state-indicator.preparing .apply-button:focus {
98+
--boxel-button-color: inherit;
99+
filter: none;
100+
cursor: not-allowed;
101+
}
102+
.state-indicator.preparing::before {
103+
content: '';
104+
position: absolute;
105+
top: -105px;
106+
left: -55px;
107+
width: 250px;
108+
height: 250px;
109+
background: conic-gradient(
110+
#ffcc8f 0deg,
111+
#ff3966 45deg,
112+
#ff309e 90deg,
113+
#aa1dc9 135deg,
114+
#d7fad6 180deg,
115+
#5fdfea 225deg,
116+
#3d83f2 270deg,
117+
#5145e8 315deg,
118+
#ffcc8f 360deg
119+
);
120+
z-index: -1;
121+
animation: spin 4s infinite linear;
122+
}
123+
124+
.state-indicator.preparing::after {
125+
content: '';
126+
position: absolute;
127+
top: 1px;
128+
left: 1px;
129+
right: 1px;
130+
bottom: 1px;
131+
background: var(--ai-bot-message-background-color);
132+
border-radius: inherit;
133+
z-index: -1;
134+
}
135+
136+
.state-indicator.preparing {
137+
position: relative;
138+
display: inline-block;
139+
border-radius: 3rem;
140+
color: white;
141+
background: var(--boxel-700);
142+
border: none;
143+
cursor: pointer;
144+
z-index: 1;
145+
overflow: hidden;
146+
}
147+
148+
.state-indicator:not(.applying):not(.preparing) {
71149
width: 1.5rem;
72150
aspect-ratio: 1;
73151
border-radius: 50%;
@@ -77,6 +155,12 @@ const AiAssistantApplyButton: TemplateOnlyComponent<Signature> = <template>
77155
background-color: var(--boxel-error-200);
78156
border-color: var(--boxel-error-200);
79157
}
158+
159+
@keyframes spin {
160+
to {
161+
transform: rotate(360deg);
162+
}
163+
}
80164
</style>
81165
</template>;
82166

packages/host/app/components/ai-assistant/apply-button/usage.gts

+5-2
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,9 @@ export default class AiAssistantApplyButtonUsage extends Component {
2323
this.state = 'failed';
2424
break;
2525
case 'failed':
26+
this.state = 'preparing';
27+
break;
28+
case 'preparing':
2629
this.state = 'ready';
2730
break;
2831
}
@@ -31,7 +34,7 @@ export default class AiAssistantApplyButtonUsage extends Component {
3134
<FreestyleUsage @name='AiAssistant::ApplyButton'>
3235
<:description>
3336
Displays button for applying change proposed by AI Assistant. Includes
34-
ready, applying, applied and failed states.
37+
ready, applying, applied, failed, and preparing states.
3538
</:description>
3639
<:example>
3740
<div class='example-container'>
@@ -45,7 +48,7 @@ export default class AiAssistantApplyButtonUsage extends Component {
4548
<Args.String
4649
@name='state'
4750
@value={{this.state}}
48-
@options={{array 'ready' 'applying' 'applied' 'failed'}}
51+
@options={{array 'ready' 'applying' 'applied' 'failed' 'preparing'}}
4952
@description='Button state'
5053
@onInput={{fn (mut this.state)}}
5154
/>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import { TemplateOnlyComponent } from '@ember/component/template-only';
2+
3+
import ApplyButton from '../ai-assistant/apply-button';
4+
5+
interface Signature {
6+
Element: HTMLDivElement;
7+
}
8+
9+
const RoomMessageCommand: TemplateOnlyComponent<Signature> = <template>
10+
<div ...attributes>
11+
<div class='command-button-bar'>
12+
<ApplyButton @state='preparing' data-test-command-apply='preparing' />
13+
</div>
14+
</div>
15+
16+
{{! template-lint-disable no-whitespace-for-layout }}
17+
{{! ignore the above error because ember-template-lint complains about the whitespace in the multi-line comment below }}
18+
<style scoped>
19+
.command-button-bar {
20+
display: flex;
21+
justify-content: flex-end;
22+
gap: var(--boxel-sp-xs);
23+
margin-top: var(--boxel-sp);
24+
}
25+
</style>
26+
</template>;
27+
28+
export default RoomMessageCommand;

0 commit comments

Comments
 (0)