From 6c18989c2bc987a7e7df04821dc35f526cfc605e Mon Sep 17 00:00:00 2001 From: Luke Melia Date: Wed, 20 Mar 2024 16:47:38 -0400 Subject: [PATCH] Keep AI Assistant button visible when panel open, in a new active state - add isHidden arg to ResizablePanel - add splattributes to ResizablePanel --- .../resizable-panel-group/index.gts | 1 + .../resizable-panel-group/panel.gts | 42 ++++++++++---- .../resizable-panel-group/usage.gts | 35 ++++-------- .../components/resizable-panel-group-test.gts | 54 ++++++++++++++++++ .../ai-assist-button-active-bg.webp | Bin 0 -> 270 bytes .../ai-assist-button-active-bg@2x.webp | Bin 0 -> 442 bytes .../ai-assist-button-active-bg@3x.webp | Bin 0 -> 740 bytes .../ai-assistant/ai-assist-icon-bw.png | Bin 0 -> 320 bytes .../ai-assistant/ai-assist-icon-bw@2x.png | Bin 0 -> 479 bytes .../ai-assistant/ai-assist-icon-bw@3x.png | Bin 0 -> 627 bytes .../app/components/ai-assistant/button.gts | 30 +++++++++- .../app/components/ai-assistant/panel.gts | 1 + .../components/operator-mode/stack-item.gts | 17 ++++-- .../operator-mode/submode-layout.gts | 43 ++++++++------ packages/host/ember-cli-build.js | 4 ++ 15 files changed, 169 insertions(+), 58 deletions(-) create mode 100644 packages/host/app/components/ai-assistant/ai-assist-button-active-bg.webp create mode 100644 packages/host/app/components/ai-assistant/ai-assist-button-active-bg@2x.webp create mode 100644 packages/host/app/components/ai-assistant/ai-assist-button-active-bg@3x.webp create mode 100644 packages/host/app/components/ai-assistant/ai-assist-icon-bw.png create mode 100644 packages/host/app/components/ai-assistant/ai-assist-icon-bw@2x.png create mode 100644 packages/host/app/components/ai-assistant/ai-assist-icon-bw@3x.png diff --git a/packages/boxel-ui/addon/src/components/resizable-panel-group/index.gts b/packages/boxel-ui/addon/src/components/resizable-panel-group/index.gts index 3ad54370f0..a9def6f6fc 100644 --- a/packages/boxel-ui/addon/src/components/resizable-panel-group/index.gts +++ b/packages/boxel-ui/addon/src/components/resizable-panel-group/index.gts @@ -245,6 +245,7 @@ export default class ResizablePanelGroup extends Component { (panelContext) => panelContext.lengthPx, ); + this.panelRatios = []; for (let index = 0; index < panelLengths.length; index++) { let panelLength = panelLengths[index]; if (panelLength == undefined) { diff --git a/packages/boxel-ui/addon/src/components/resizable-panel-group/panel.gts b/packages/boxel-ui/addon/src/components/resizable-panel-group/panel.gts index 18489b8939..ae75533bf6 100644 --- a/packages/boxel-ui/addon/src/components/resizable-panel-group/panel.gts +++ b/packages/boxel-ui/addon/src/components/resizable-panel-group/panel.gts @@ -1,8 +1,10 @@ import { registerDestructor } from '@ember/destroyable'; +import { action } from '@ember/object'; import { scheduleOnce } from '@ember/runloop'; import { htmlSafe } from '@ember/template'; import Component from '@glimmer/component'; import { tracked } from '@glimmer/tracking'; +import { modifier } from 'ember-modifier'; import createRef from 'ember-ref-bucket/modifiers/create-ref'; import cssVars from '../../helpers/css-var.ts'; @@ -22,6 +24,7 @@ interface Signature { Args: { collapsible?: boolean; //default true defaultLengthFraction: number; + isHidden?: boolean; //default false isLastPanel: (panelId: number) => boolean; lengthPx?: number; minLengthPx?: number; @@ -43,6 +46,16 @@ interface Signature { Element: HTMLDivElement; } +let managePanelRegistration = modifier( + (_element, [panel, isHidden]: [Panel, boolean | undefined]) => { + if (isHidden) { + scheduleOnce('afterRender', panel, panel.unregisterPanel); + } else { + scheduleOnce('afterRender', panel, panel.registerPanel); + } + }, +); + export default class Panel extends Component { ); + await sleep(100); // let didResizeModifier run + assert.hasNumericStyle('.panel-1-content', 'height', 218, 1); + assert.hasNumericStyle('.panel-2-content', 'height', 0, 0); + this.renderController.isPanel2Hidden = false; + await sleep(100); // let didResizeModifier run + assert.hasNumericStyle('.panel-1-content', 'height', 156, 1); + assert.hasNumericStyle('.panel-2-content', 'height', 62, 1); + this.renderController.isPanel2Hidden = true; + await sleep(100); // let didResizeModifier run + assert.hasNumericStyle('.panel-1-content', 'height', 218, 1); + assert.hasNumericStyle('.panel-2-content', 'height', 0, 0); + }); }); diff --git a/packages/host/app/components/ai-assistant/ai-assist-button-active-bg.webp b/packages/host/app/components/ai-assistant/ai-assist-button-active-bg.webp new file mode 100644 index 0000000000000000000000000000000000000000..cef6c30e6c3674823941080a575dfe7cb2c5f3d7 GIT binary patch literal 270 zcmV+p0rCD)Nk&En0RRA3MM6+kP&gp^0001R2mqY{DkuOb06t|Tk42;+AsVe#s6Ylp zv;a<=ps8N^{=LS<#86OofELx2mUK;G+|PcOVtzw7bagn+a literal 0 HcmV?d00001 diff --git a/packages/host/app/components/ai-assistant/ai-assist-button-active-bg@2x.webp b/packages/host/app/components/ai-assistant/ai-assist-button-active-bg@2x.webp new file mode 100644 index 0000000000000000000000000000000000000000..6d534d1acfd5005c5ea69a9a3bdfbd2efaf58680 GIT binary patch literal 442 zcmV;r0Y&~&Nk&Gp0RRA3MM6+kP&go_0RRB75CEM4Do_AW06uLhkwv5=p_i-%BuECt zvH(t*v_Ga(_D<~K=agPC-_b|8BP@|}^0D@z$I^b}j_1)X^;Yyp_ibKeDWnRyVg`eJ zv4h+S0F`dQgx0j_&LvqUR-v8V+Hx6Egt;{+dGEVYu#s@bw3EfgZ^|IuVW7uRVa0fTZ1`Rqd3NVnF7Wa$>maz?lre@pj8A_ zwRXOMdFGtY4(XKk=y#vc_UBJZ80#xbNQfn}-7Pba;2B*zc3K$Sb>NxOGXCmuO2JC?S_m%z;z8ET`b;eIj!MT>h1m{1E=p?>7z4q zdep`pjJqcH^D>Z^?#OJE)Jw@FC=B(a5qL!2JLk!A|6C4ohks$VP8MEWtXyJ+&D|M# ksD6tzyp2^703}g)vsIjgSfsZDY8=SGnAa%bSe!-x0LYiz7XSbN literal 0 HcmV?d00001 diff --git a/packages/host/app/components/ai-assistant/ai-assist-button-active-bg@3x.webp b/packages/host/app/components/ai-assistant/ai-assist-button-active-bg@3x.webp new file mode 100644 index 0000000000000000000000000000000000000000..9580d05658e043aea1849a9fa0763c568ba49e0b GIT binary patch literal 740 zcmVh)K#of`TrrK-pU-pll=rTSy9$A$Y)92i7J>;u@+0=`ZEyehXTp9){Bf^& z8SPgX1AjPo+pNEPKfZw!7!tWvQwC1nve^wz{wD5Fjc#&~uLhpvt{KfSyEPqJc$viO z1xh0&*y`wpq^ZjDdH?^+_1MQYfB^nW`5oWLrTTQ&^hOdt`^JC%!wdi3LdVV~B`+l1 z5zGhj%$0`ar%KqkA_DvFn9GMMKcSN8#iDP)a0czdx=8@4)P~MRVmYuIA9e8XP8{sh zX)X{;Q*9a@SNq5K$Q2Erfx7v3NZ#=UWAuOR-tPA$>#)0Tvay>D+C5b6#dfk=6S4sD zJpJngM7J6;&P6FNh-GO}qh=7Q`d{f2x^O$2FN-a&+^XiajSznYA>TS|#l<5m*)MrA?J~^F*BUEHO?3whgSCOt!pfYYj(kn`k&eFG z1C`pEcI2|KW306|>-@d0zlwZ#io4?(=zhKt2#pLBNjF=w W$E9=Eh=?j~Sml+qe_e5i0Du4^z;!49 literal 0 HcmV?d00001 diff --git a/packages/host/app/components/ai-assistant/ai-assist-icon-bw.png b/packages/host/app/components/ai-assistant/ai-assist-icon-bw.png new file mode 100644 index 0000000000000000000000000000000000000000..c54c819f3c0e38c0d247981b1f361a2fb8658feb GIT binary patch literal 320 zcmeAS@N?(olHy`uVBq!ia0vp^QXtI23?!pd0{;Rjg8-ipSN8%RAd`5Y>hr}CsG7ed z$S;`T!2yQ=g@z9X2?YZ4_Fve*-*gx#wZYTHF+}2Wasm+i`yc=Ee?2oBM*`z#nFTi6 z{{KJ!=l^_$r-uLj?|=3GKYPHvKmY$b|NsA=^I^f2|Nl=Y{QqCgJj47iP#w>||Nq08 zZmM+xWy2E^m?oW{BiZoef4{WEj;D@12L4PdmI)sFy@m0_<7@37?40f#;R!jtphSxy zOZT;uDqF0|z1S8nvC0n3`h&~v9M58&^j59X;S XM#em=htp+NfdawP)z4*}Q$iB}fa`d8 literal 0 HcmV?d00001 diff --git a/packages/host/app/components/ai-assistant/ai-assist-icon-bw@2x.png b/packages/host/app/components/ai-assistant/ai-assist-icon-bw@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..f3066b1f2de7fc42449d6bde4b0aa733515bb417 GIT binary patch literal 479 zcmeAS@N?(olHy`uVBq!ia0vp^CLqkg3?x4-dwml~83*`;xVjhk0GT9#+m3o#Kn(&V zL4Lsu3+_L-Q1GEaz#@OcfrJSPT6I&~7#JA6JzX3_JdSUj7P+lifybfIK>O6c@ZYOK zIrkY%S^RU6=5>}6MFJ~Zf#dT& z+jTsD{?t}r|9$Ozr>IcnNi9p1H~WMs#PXyqB>zX#}C(|4M(?#^_H_48}*_;Muwo7y1j)4NgSOqapjchQNBySK$e zBZbE&= knMQNARSPe^-FxyA|I+#(vDt<7R-icYboFyt=akR{05B-NmjD0& literal 0 HcmV?d00001 diff --git a/packages/host/app/components/ai-assistant/ai-assist-icon-bw@3x.png b/packages/host/app/components/ai-assistant/ai-assist-icon-bw@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..ce7a33216b4b3164a74e2e8f7986049422a9c554 GIT binary patch literal 627 zcmeAS@N?(olHy`uVBq!ia0vp^ejv=k3?yq>Z1sSYZh%jSt9yYDkVzz1yDe@xP$^$Y zkY6x^!v}$a0}o~^7&Js)*swp}rkH_&@wumqV~EG`()%?Qd*U*!KlHnw{m<&}5#KEcp+w!ExUCo8d>@^ojObSn1G z(;a4!Uzs+3 { } diff --git a/packages/host/app/components/ai-assistant/panel.gts b/packages/host/app/components/ai-assistant/panel.gts index 66f824290c..230e7f30f9 100644 --- a/packages/host/app/components/ai-assistant/panel.gts +++ b/packages/host/app/components/ai-assistant/panel.gts @@ -169,6 +169,7 @@ export default class AiAssistantPanel extends Component { grid-template-rows: auto 1fr; background-color: var(--boxel-ai-purple); border: none; + border-radius: 0; color: var(--boxel-light); height: 100%; position: relative; diff --git a/packages/host/app/components/operator-mode/stack-item.gts b/packages/host/app/components/operator-mode/stack-item.gts index c831bc72fc..0d58343509 100644 --- a/packages/host/app/components/operator-mode/stack-item.gts +++ b/packages/host/app/components/operator-mode/stack-item.gts @@ -319,10 +319,19 @@ export default class OperatorModeStackItem extends Component { }); private calculateLastSavedMsg() { - this.lastSavedMsg = - this.lastSaved != null - ? `Saved ${formatDistanceToNow(this.lastSaved, { addSuffix: true })}` - : undefined; + // runs frequently, so only change a tracked property if the value has changed + if (this.lastSaved == null) { + if (this.lastSavedMsg) { + this.lastSavedMsg = undefined; + } + } else { + let savedMessage = `Saved ${formatDistanceToNow(this.lastSaved, { + addSuffix: true, + })}`; + if (this.lastSavedMsg != savedMessage) { + this.lastSavedMsg = savedMessage; + } + } } private doWithStableScroll = restartableTask( diff --git a/packages/host/app/components/operator-mode/submode-layout.gts b/packages/host/app/components/operator-mode/submode-layout.gts index 3ec15e71e3..ca3d3b0a97 100644 --- a/packages/host/app/components/operator-mode/submode-layout.gts +++ b/packages/host/app/components/operator-mode/submode-layout.gts @@ -9,7 +9,7 @@ import onClickOutside from 'ember-click-outside/modifiers/on-click-outside'; import { ResizablePanelGroup } from '@cardstack/boxel-ui/components'; import type { PanelContext } from '@cardstack/boxel-ui/components'; -import { and, not } from '@cardstack/boxel-ui/helpers'; +import { and, cn, not } from '@cardstack/boxel-ui/helpers'; import AiAssistantButton from '@cardstack/host/components/ai-assistant/button'; import AiAssistantPanel from '@cardstack/host/components/ai-assistant/panel'; @@ -152,8 +152,10 @@ export default class SubmodeLayout extends Component {