From 921632e38d150c025e4b70d849662e7e7d88409c Mon Sep 17 00:00:00 2001 From: Adam Obuchowicz Date: Mon, 19 Aug 2024 15:28:38 +0200 Subject: [PATCH] New design of CB: two modes (#10814) Fixes #10603 [Screencast from 2024-08-14 12-10-51.webm](https://github.com/user-attachments/assets/fcd5bfa4-b128-4a84-a19f-c14e78dae8c9) What is not yet implemented: the filtering. That means that spaces keep their special meaning, and we still display modules and types. The component list itself was refactored to a separate vue component. The logic of default visualization type in preview changed a bit: as now there is no selected component, we remember with what suggestion have we switched to code edit mode. --- CHANGELOG.md | 5 + app/gui2/e2e/componentBrowser.spec.ts | 18 +- app/gui2/src/bindings.ts | 3 +- app/gui2/src/components/ComponentBrowser.vue | 562 +++++------------- .../ComponentBrowser/ComponentEditor.vue | 11 + .../ComponentBrowser/ComponentList.vue | 262 ++++++++ .../components/ComponentBrowser/scrolling.ts | 20 +- app/gui2/src/components/GraphEditor.vue | 2 +- .../components/GraphEditor/NodeWidgetTree.vue | 18 +- app/gui2/src/components/SvgButton.vue | 2 +- app/gui2/src/stores/graph/graphDatabase.ts | 2 +- app/gui2/src/util/autoBlur.ts | 33 +- app/gui2/src/util/getIconName.ts | 19 + 13 files changed, 492 insertions(+), 465 deletions(-) create mode 100644 app/gui2/src/components/ComponentBrowser/ComponentList.vue diff --git a/CHANGELOG.md b/CHANGELOG.md index 9dc4cdb49984..c1f6b7eaa2d3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -3,9 +3,14 @@ #### Enso IDE - [Table Editor Widget][10774] displayed in `Table.new` component. +- [New design of Component Browser][10814] - the component list is under the + input and shown only in the initial "component browsing" mode - after picking + any suggestion with Tab or new button the mode is switched to "code editing", + where visualization preview is displayed instead. - [Drilldown for XML][10824] [10774]: https://github.com/enso-org/enso/pull/10774 +[10814]: https://github.com/enso-org/enso/pull/10814 [10824]: https://github.com/enso-org/enso/pull/10824 #### Enso Standard Library diff --git a/app/gui2/e2e/componentBrowser.spec.ts b/app/gui2/e2e/componentBrowser.spec.ts index 86e5272edc0b..63eb2f27e6f3 100644 --- a/app/gui2/e2e/componentBrowser.spec.ts +++ b/app/gui2/e2e/componentBrowser.spec.ts @@ -106,7 +106,7 @@ test('Graph Editor pans to Component Browser', async ({ page }) => { // Dragging out an edge to the bottom of the viewport; when the CB pans into view, some nodes are out of view. await page.mouse.move(100, 1100) await page.mouse.down({ button: 'middle' }) - await page.mouse.move(100, 80) + await page.mouse.move(100, 280) await page.mouse.up({ button: 'middle' }) await expect(locate.graphNodeByBinding(page, 'five')).toBeInViewport() const outputPort = await locate.outputPortCoordinates(locate.graphNodeByBinding(page, 'final')) @@ -240,14 +240,10 @@ test('Visualization preview: type-based visualization selection', async ({ page await expect(locate.componentBrowser(page)).toExist() await expect(locate.componentBrowserEntry(page)).toExist() const input = locate.componentBrowserInput(page).locator('input') - await input.fill('4') - await expect(input).toHaveValue('4') - await expect(locate.jsonVisualization(page)).toExist() await input.fill('Table.ne') await expect(input).toHaveValue('Table.ne') - // The table visualization is not currently working with `executeExpression` (#9194), but we can test that the JSON - // visualization is no longer selected. - await expect(locate.jsonVisualization(page)).toBeHidden() + await locate.componentBrowser(page).getByTestId('switchToEditMode').click() + await expect(locate.tableVisualization(page)).toBeVisible() await page.keyboard.press('Escape') await expect(locate.componentBrowser(page)).toBeHidden() await expect(locate.graphNode(page)).toHaveCount(nodeCount) @@ -258,17 +254,15 @@ test('Visualization preview: user visualization selection', async ({ page }) => const nodeCount = await locate.graphNode(page).count() await locate.addNewNodeButton(page).click() await expect(locate.componentBrowser(page)).toExist() - await expect(locate.componentBrowserEntry(page)).toExist() const input = locate.componentBrowserInput(page).locator('input') await input.fill('4') await expect(input).toHaveValue('4') - await expect(locate.jsonVisualization(page)).toExist() + await locate.componentBrowser(page).getByTestId('switchToEditMode').click() + await expect(locate.jsonVisualization(page)).toBeVisible() await expect(locate.jsonVisualization(page)).toContainText('"visualizedExpr": "4"') await locate.toggleVisualizationSelectorButton(page).click() await page.getByRole('button', { name: 'Table' }).click() - // The table visualization is not currently working with `executeExpression` (#9194), but we can test that the JSON - // visualization is no longer selected. - await expect(locate.jsonVisualization(page)).toBeHidden() + await expect(locate.tableVisualization(page)).toBeVisible() await page.keyboard.press('Escape') await expect(locate.componentBrowser(page)).toBeHidden() await expect(locate.graphNode(page)).toHaveCount(nodeCount) diff --git a/app/gui2/src/bindings.ts b/app/gui2/src/bindings.ts index f36234d2ec04..99cc69a11ca6 100644 --- a/app/gui2/src/bindings.ts +++ b/app/gui2/src/bindings.ts @@ -19,8 +19,9 @@ export const interactionBindings = defineKeybinds('current-interaction', { }) export const componentBrowserBindings = defineKeybinds('component-browser', { - applySuggestion: ['Tab'], + applySuggestionAndSwitchToEditMode: ['Tab'], acceptSuggestion: ['Enter'], + acceptCode: ['Enter'], acceptInput: ['Mod+Enter'], acceptAIPrompt: ['Tab', 'Enter'], moveUp: ['ArrowUp'], diff --git a/app/gui2/src/components/ComponentBrowser.vue b/app/gui2/src/components/ComponentBrowser.vue index 32184a2d32ca..bb35ef45b612 100644 --- a/app/gui2/src/components/ComponentBrowser.vue +++ b/app/gui2/src/components/ComponentBrowser.vue @@ -1,15 +1,24 @@ + + @@ -463,7 +362,7 @@ const handler = componentBrowserBindings.handler({
-
-
-
-
- - - - - -
-
-
-
-
-
- - - - - -
-
-
-
- - - - - -
-
-
-
- -
-
- -
-
-
- + + - -
+ +
diff --git a/app/gui2/src/components/ComponentBrowser/ComponentList.vue b/app/gui2/src/components/ComponentBrowser/ComponentList.vue new file mode 100644 index 000000000000..777e3f5c00b5 --- /dev/null +++ b/app/gui2/src/components/ComponentBrowser/ComponentList.vue @@ -0,0 +1,262 @@ + + + + + diff --git a/app/gui2/src/components/ComponentBrowser/scrolling.ts b/app/gui2/src/components/ComponentBrowser/scrolling.ts index 0400540621b6..b0b3256bb6ad 100644 --- a/app/gui2/src/components/ComponentBrowser/scrolling.ts +++ b/app/gui2/src/components/ComponentBrowser/scrolling.ts @@ -1,24 +1,20 @@ import { useApproach } from '@/composables/animation' -import { computed, ref } from 'vue' +import { ToValue } from '@/util/reactivity' +import { computed, ref, toValue } from 'vue' export type ScrollTarget = - | { type: 'bottom' } + | { type: 'top' } | { type: 'selected' } | { type: 'offset'; offset: number } -export function useScrolling( - selectedPos: { value: number }, - scrollerSize: { value: number }, - contentSize: { value: number }, - entrySize: number, -) { - const targetScroll = ref({ type: 'bottom' }) +export function useScrolling(selectedPos: ToValue) { + const targetScroll = ref({ type: 'top' }) const targetScrollPosition = computed(() => { switch (targetScroll.value.type) { case 'selected': - return Math.max(selectedPos.value - scrollerSize.value + entrySize, 0) - case 'bottom': - return contentSize.value - scrollerSize.value + return toValue(selectedPos) + case 'top': + return 0.0 case 'offset': return targetScroll.value.offset } diff --git a/app/gui2/src/components/GraphEditor.vue b/app/gui2/src/components/GraphEditor.vue index cb95fc6c671e..59cbc0e6b4ff 100644 --- a/app/gui2/src/components/GraphEditor.vue +++ b/app/gui2/src/components/GraphEditor.vue @@ -53,7 +53,7 @@ import { colorFromString } from '@/util/colors' import { partition } from '@/util/data/array' import { every, filterDefined } from '@/util/data/iterable' import { Rect } from '@/util/data/rect' -import { Err, Ok, unwrapOr, type Result } from '@/util/data/result' +import { Err, Ok, unwrapOr } from '@/util/data/result' import { Vec2 } from '@/util/data/vec2' import { computedFallback } from '@/util/reactivity' import { until } from '@vueuse/core' diff --git a/app/gui2/src/components/GraphEditor/NodeWidgetTree.vue b/app/gui2/src/components/GraphEditor/NodeWidgetTree.vue index a8a35f73a141..c16fd3fc12c6 100644 --- a/app/gui2/src/components/GraphEditor/NodeWidgetTree.vue +++ b/app/gui2/src/components/GraphEditor/NodeWidgetTree.vue @@ -8,7 +8,7 @@ import { useGraphStore, type NodeId } from '@/stores/graph' import type { NodeType } from '@/stores/graph/graphDatabase' import { Ast } from '@/util/ast' import type { Vec2 } from '@/util/data/vec2' -import { displayedIconOf } from '@/util/getIconName' +import { iconOfNode } from '@/util/getIconName' import { computed, toRef, watch } from 'vue' import { DisplayIcon } from './widgets/WidgetIcon.vue' @@ -100,21 +100,7 @@ const widgetTree = provideWidgetTree( () => emit('openFullMenu'), ) -const expressionInfo = computed(() => graph.db.getExpressionInfo(props.ast.externalId)) -const suggestionEntry = computed(() => graph.db.getNodeMainSuggestion(props.nodeId)) -const topLevelIcon = computed(() => { - switch (props.nodeType) { - default: - case 'component': - return displayedIconOf( - suggestionEntry.value, - expressionInfo.value?.methodCall?.methodPointer, - expressionInfo.value?.typename ?? 'Unknown', - ) - case 'output': - return 'data_output' - } -}) +const topLevelIcon = computed(() => iconOfNode(props.nodeId, graph.db)) watch(toRef(widgetTree, 'currentEdit'), (edit) => edit && selectNode()) diff --git a/app/gui2/src/components/SvgButton.vue b/app/gui2/src/components/SvgButton.vue index eea8f245a874..481668c709c6 100644 --- a/app/gui2/src/components/SvgButton.vue +++ b/app/gui2/src/components/SvgButton.vue @@ -15,7 +15,7 @@ const _props = defineProps<{ diff --git a/app/gui2/src/stores/graph/graphDatabase.ts b/app/gui2/src/stores/graph/graphDatabase.ts index b4d89e783874..5cfdad76244b 100644 --- a/app/gui2/src/stores/graph/graphDatabase.ts +++ b/app/gui2/src/stores/graph/graphDatabase.ts @@ -21,7 +21,7 @@ import { } from '@/util/reactivity' import * as objects from 'enso-common/src/utilities/data/object' import * as set from 'lib0/set' -import { reactive, ref, shallowReactive, watchEffect, WatchStopHandle, type Ref } from 'vue' +import { reactive, ref, shallowReactive, WatchStopHandle, type Ref } from 'vue' import type { MethodCall, StackItem } from 'ydoc-shared/languageServerTypes' import type { Opt } from 'ydoc-shared/util/data/opt' import type { ExternalId, SourceRange, VisualizationMetadata } from 'ydoc-shared/yjsModel' diff --git a/app/gui2/src/util/autoBlur.ts b/app/gui2/src/util/autoBlur.ts index ae61f5004d23..5a8df1e9374a 100644 --- a/app/gui2/src/util/autoBlur.ts +++ b/app/gui2/src/util/autoBlur.ts @@ -55,18 +55,43 @@ export function isNodeOutside(element: any, area: Opt): boolean { } /** Returns a new interaction based on the given `interaction`. The new interaction will be ended if a pointerdown event - * occurs outside the given `area` element. */ + * occurs outside the given `area` element. + * + * See also {@link cancelOnClickOutside}. + */ export function endOnClickOutside( - area: Ref, + area: Ref>, interaction: Interaction, ): Interaction { - const chainedPointerdown = interaction.pointerdown const handler = injectInteractionHandler() + return handleClickOutside(area, interaction, handler.end.bind(handler)) +} + +/** Returns a new interaction based on the given `interaction`. The new interaction will be canceled if a pointerdown event + * occurs outside the given `area` element. + * + * See also {@link endOnClickOutside}. + */ +export function cancelOnClickOutside( + area: Ref>, + interaction: Interaction, +) { + const handler = injectInteractionHandler() + return handleClickOutside(area, interaction, handler.cancel.bind(handler)) +} + +/** Common part of {@link cancelOnClickOutside} and {@link endOnClickOutside}. */ +function handleClickOutside( + area: Ref>, + interaction: Interaction, + handler: (interaction: Interaction) => void, +) { + const chainedPointerdown = interaction.pointerdown const wrappedInteraction: Interaction = { ...interaction, pointerdown: (e: PointerEvent, ...args) => { if (targetIsOutside(e, unrefElement(area))) { - handler.end(wrappedInteraction) + handler(wrappedInteraction) return false } return chainedPointerdown ? chainedPointerdown(e, ...args) : false diff --git a/app/gui2/src/util/getIconName.ts b/app/gui2/src/util/getIconName.ts index 50f9c4327eab..a12a237b8311 100644 --- a/app/gui2/src/util/getIconName.ts +++ b/app/gui2/src/util/getIconName.ts @@ -1,3 +1,5 @@ +import { NodeId } from '@/stores/graph' +import { GraphDb } from '@/stores/graph/graphDatabase' import { SuggestionKind, type SuggestionEntry, @@ -43,3 +45,20 @@ export function displayedIconOf( return DEFAULT_ICON } } + +export function iconOfNode(node: NodeId, graphDb: GraphDb) { + const expressionInfo = graphDb.getExpressionInfo(node) + const suggestionEntry = graphDb.getNodeMainSuggestion(node) + const nodeType = graphDb.nodeIdToNode.get(node)?.type + switch (nodeType) { + default: + case 'component': + return displayedIconOf( + suggestionEntry, + expressionInfo?.methodCall?.methodPointer, + expressionInfo?.typename ?? 'Unknown', + ) + case 'output': + return 'data_output' + } +}