polen 0.10.0-next.11 → 0.10.0-next.13
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/api/builder/builder.js +1 -1
- package/build/api/builder/builder.js.map +1 -1
- package/build/api/config/load.js +1 -1
- package/build/api/config/load.js.map +1 -1
- package/build/api/vite/plugins/build.js +1 -1
- package/build/api/vite/plugins/build.js.map +1 -1
- package/build/api/vite/plugins/core.d.ts.map +1 -1
- package/build/api/vite/plugins/core.js +0 -2
- package/build/api/vite/plugins/core.js.map +1 -1
- package/build/api/vite/plugins/pages.js +1 -1
- package/build/api/vite/plugins/pages.js.map +1 -1
- package/build/exports/components.d.ts +4 -1
- package/build/exports/components.d.ts.map +1 -1
- package/build/exports/components.js +4 -1
- package/build/exports/components.js.map +1 -1
- package/build/lib/demos/config-schema.d.ts +6 -6
- package/build/lib/github-actions/runner.js +2 -2
- package/build/lib/github-actions/runner.js.map +1 -1
- package/build/lib/graphql-document/$$.d.ts +5 -0
- package/build/lib/graphql-document/$$.d.ts.map +1 -0
- package/build/lib/graphql-document/$$.js +5 -0
- package/build/lib/graphql-document/$$.js.map +1 -0
- package/build/lib/graphql-document/$.d.ts +2 -0
- package/build/lib/graphql-document/$.d.ts.map +1 -0
- package/build/lib/graphql-document/$.js +2 -0
- package/build/lib/graphql-document/$.js.map +1 -0
- package/build/lib/graphql-document/analysis.d.ts +44 -0
- package/build/lib/graphql-document/analysis.d.ts.map +1 -0
- package/build/lib/graphql-document/analysis.js +361 -0
- package/build/lib/graphql-document/analysis.js.map +1 -0
- package/build/lib/graphql-document/components/CopyButton.d.ts +19 -0
- package/build/lib/graphql-document/components/CopyButton.d.ts.map +1 -0
- package/build/lib/graphql-document/components/CopyButton.js +43 -0
- package/build/lib/graphql-document/components/CopyButton.js.map +1 -0
- package/build/lib/graphql-document/components/GraphQLDocument.d.ts +38 -0
- package/build/lib/graphql-document/components/GraphQLDocument.d.ts.map +1 -0
- package/build/lib/graphql-document/components/GraphQLDocument.js +130 -0
- package/build/lib/graphql-document/components/GraphQLDocument.js.map +1 -0
- package/build/lib/graphql-document/components/GraphQLDocumentWithSchema.d.ts +7 -0
- package/build/lib/graphql-document/components/GraphQLDocumentWithSchema.d.ts.map +1 -0
- package/build/lib/graphql-document/components/GraphQLDocumentWithSchema.js +45 -0
- package/build/lib/graphql-document/components/GraphQLDocumentWithSchema.js.map +1 -0
- package/build/lib/graphql-document/components/GraphQLIdentifierPopover.d.ts +33 -0
- package/build/lib/graphql-document/components/GraphQLIdentifierPopover.d.ts.map +1 -0
- package/build/lib/graphql-document/components/GraphQLIdentifierPopover.js +48 -0
- package/build/lib/graphql-document/components/GraphQLIdentifierPopover.js.map +1 -0
- package/build/lib/graphql-document/components/IdentifierLink.d.ts +39 -0
- package/build/lib/graphql-document/components/IdentifierLink.d.ts.map +1 -0
- package/build/lib/graphql-document/components/IdentifierLink.js +75 -0
- package/build/lib/graphql-document/components/IdentifierLink.js.map +1 -0
- package/build/lib/graphql-document/components/graphql-document-styles.d.ts +5 -0
- package/build/lib/graphql-document/components/graphql-document-styles.d.ts.map +1 -0
- package/build/lib/graphql-document/components/graphql-document-styles.js +167 -0
- package/build/lib/graphql-document/components/graphql-document-styles.js.map +1 -0
- package/build/lib/graphql-document/components/index.d.ts +6 -0
- package/build/lib/graphql-document/components/index.d.ts.map +1 -0
- package/build/lib/graphql-document/components/index.js +6 -0
- package/build/lib/graphql-document/components/index.js.map +1 -0
- package/build/lib/graphql-document/example.d.ts +25 -0
- package/build/lib/graphql-document/example.d.ts.map +1 -0
- package/build/lib/graphql-document/example.js +140 -0
- package/build/lib/graphql-document/example.js.map +1 -0
- package/build/lib/graphql-document/graphql-document.d.ts +35 -0
- package/build/lib/graphql-document/graphql-document.d.ts.map +1 -0
- package/build/lib/graphql-document/graphql-document.js +36 -0
- package/build/lib/graphql-document/graphql-document.js.map +1 -0
- package/build/lib/graphql-document/hooks/use-tooltip-state.d.ts +43 -0
- package/build/lib/graphql-document/hooks/use-tooltip-state.d.ts.map +1 -0
- package/build/lib/graphql-document/hooks/use-tooltip-state.js +132 -0
- package/build/lib/graphql-document/hooks/use-tooltip-state.js.map +1 -0
- package/build/lib/graphql-document/positioning-simple.d.ts +68 -0
- package/build/lib/graphql-document/positioning-simple.d.ts.map +1 -0
- package/build/lib/graphql-document/positioning-simple.js +197 -0
- package/build/lib/graphql-document/positioning-simple.js.map +1 -0
- package/build/lib/graphql-document/schema-context.d.ts +8 -0
- package/build/lib/graphql-document/schema-context.d.ts.map +1 -0
- package/build/lib/graphql-document/schema-context.js +11 -0
- package/build/lib/graphql-document/schema-context.js.map +1 -0
- package/build/lib/graphql-document/schema-integration-example.d.ts +27 -0
- package/build/lib/graphql-document/schema-integration-example.d.ts.map +1 -0
- package/build/lib/graphql-document/schema-integration-example.js +297 -0
- package/build/lib/graphql-document/schema-integration-example.js.map +1 -0
- package/build/lib/graphql-document/schema-integration.d.ts +135 -0
- package/build/lib/graphql-document/schema-integration.d.ts.map +1 -0
- package/build/lib/graphql-document/schema-integration.js +328 -0
- package/build/lib/graphql-document/schema-integration.js.map +1 -0
- package/build/lib/graphql-document/types.d.ts +117 -0
- package/build/lib/graphql-document/types.d.ts.map +1 -0
- package/build/lib/graphql-document/types.js +2 -0
- package/build/lib/graphql-document/types.js.map +1 -0
- package/build/template/components/ArgumentAnnotation.js +10 -0
- package/build/template/components/ArgumentAnnotation.js.map +1 -0
- package/build/template/components/ArgumentList.js +9 -0
- package/build/template/components/ArgumentList.js.map +1 -0
- package/build/template/components/ArgumentListAnnotation.js +15 -0
- package/build/template/components/ArgumentListAnnotation.js.map +1 -0
- package/build/template/components/Changelog.js +44 -0
- package/build/template/components/Changelog.js.map +1 -0
- package/build/template/components/{CodeBlock.jsx → CodeBlock.js} +4 -5
- package/build/template/components/{CodeBlock.jsx.map → CodeBlock.js.map} +1 -1
- package/build/template/components/{CodeBlockEnhancer.jsx → CodeBlockEnhancer.js} +4 -4
- package/build/template/components/CodeBlockEnhancer.js.map +1 -0
- package/build/template/components/DeprecationReason.js +9 -0
- package/build/template/components/DeprecationReason.js.map +1 -0
- package/build/template/components/Description.js +9 -0
- package/build/template/components/Description.js.map +1 -0
- package/build/template/components/Field.js +14 -0
- package/build/template/components/Field.js.map +1 -0
- package/build/template/components/{FieldList.jsx → FieldList.js} +4 -5
- package/build/template/components/FieldList.js.map +1 -0
- package/build/template/components/{FieldListSection.jsx → FieldListSection.js} +4 -6
- package/build/template/components/FieldListSection.js.map +1 -0
- package/build/template/components/HamburgerMenu.js +30 -0
- package/build/template/components/HamburgerMenu.js.map +1 -0
- package/build/template/components/Link.d.ts +1 -1
- package/build/template/components/{Link.jsx → Link.js} +4 -5
- package/build/template/components/Link.js.map +1 -0
- package/build/template/components/Logo.js +20 -0
- package/build/template/components/Logo.js.map +1 -0
- package/build/template/components/MDXComponents.d.ts +11 -0
- package/build/template/components/MDXComponents.d.ts.map +1 -0
- package/build/template/components/MDXComponents.js +70 -0
- package/build/template/components/MDXComponents.js.map +1 -0
- package/build/template/components/{Markdown.jsx → Markdown.js} +3 -2
- package/build/template/components/Markdown.js.map +1 -0
- package/build/template/components/MissingSchema.d.ts +1 -1
- package/build/template/components/MissingSchema.d.ts.map +1 -1
- package/build/template/components/MissingSchema.js +5 -0
- package/build/template/components/MissingSchema.js.map +1 -0
- package/build/template/components/NamedType.js +12 -0
- package/build/template/components/NamedType.js.map +1 -0
- package/build/template/components/NotFound.js +7 -0
- package/build/template/components/NotFound.js.map +1 -0
- package/build/template/components/{RadixLink.jsx → RadixLink.js} +1 -1
- package/build/template/components/RadixLink.js.map +1 -0
- package/build/template/components/TestComponent.d.ts +5 -0
- package/build/template/components/TestComponent.d.ts.map +1 -0
- package/build/template/components/TestComponent.js +7 -0
- package/build/template/components/TestComponent.js.map +1 -0
- package/build/template/components/Texts/{MinorHeading.jsx → MinorHeading.js} +4 -3
- package/build/template/components/Texts/MinorHeading.js.map +1 -0
- package/build/template/components/Texts/texts.js +1 -1
- package/build/template/components/Texts/texts.js.map +1 -1
- package/build/template/components/ThemeToggle.js +9 -0
- package/build/template/components/ThemeToggle.js.map +1 -0
- package/build/template/components/{TypeAnnotation.jsx → TypeAnnotation.js} +8 -18
- package/build/template/components/TypeAnnotation.js.map +1 -0
- package/build/template/components/TypeFieldsLinkList.js +9 -0
- package/build/template/components/TypeFieldsLinkList.js.map +1 -0
- package/build/template/components/TypeIndex.js +17 -0
- package/build/template/components/TypeIndex.js.map +1 -0
- package/build/template/components/content/$$.d.ts +1 -0
- package/build/template/components/content/$$.d.ts.map +1 -1
- package/build/template/components/content/$$.js +1 -0
- package/build/template/components/content/$$.js.map +1 -1
- package/build/template/components/content/GraphQLDocumentWithSchema.d.ts +8 -0
- package/build/template/components/content/GraphQLDocumentWithSchema.d.ts.map +1 -0
- package/build/template/components/content/GraphQLDocumentWithSchema.js +16 -0
- package/build/template/components/content/GraphQLDocumentWithSchema.js.map +1 -0
- package/build/template/components/content/GraphQLDocumentWrapper.d.ts +7 -0
- package/build/template/components/content/GraphQLDocumentWrapper.d.ts.map +1 -0
- package/build/template/components/content/GraphQLDocumentWrapper.js +62 -0
- package/build/template/components/content/GraphQLDocumentWrapper.js.map +1 -0
- package/build/template/components/graphql/graphql.d.ts +2 -2
- package/build/template/components/graphql/graphql.js +3 -0
- package/build/template/components/graphql/graphql.js.map +1 -0
- package/build/template/components/graphql/index.d.ts +1 -1
- package/build/template/components/graphql/index.js +1 -1
- package/build/template/components/graphql/index.js.map +1 -1
- package/build/template/components/graphql/{type-kind-icon.jsx → type-kind-icon.js} +3 -2
- package/build/template/components/graphql/type-kind-icon.js.map +1 -0
- package/build/template/components/graphql/type-link.js +11 -0
- package/build/template/components/graphql/type-link.js.map +1 -0
- package/build/template/components/sidebar/Sidebar.d.ts +1 -1
- package/build/template/components/sidebar/Sidebar.d.ts.map +1 -1
- package/build/template/components/sidebar/Sidebar.js +11 -0
- package/build/template/components/sidebar/Sidebar.js.map +1 -0
- package/build/template/components/sidebar/{SidebarItem.jsx → SidebarItem.js} +15 -32
- package/build/template/components/sidebar/SidebarItem.js.map +1 -0
- package/build/template/components/sidebar/ToggleButton.d.ts +1 -1
- package/build/template/components/sidebar/ToggleButton.d.ts.map +1 -1
- package/build/template/components/sidebar/ToggleButton.js +5 -0
- package/build/template/components/sidebar/ToggleButton.js.map +1 -0
- package/build/template/contexts/{ThemeContext.jsx → ThemeContext.js} +3 -4
- package/build/template/contexts/{ThemeContext.jsx.map → ThemeContext.js.map} +1 -1
- package/build/template/entry.client.d.ts +1 -0
- package/build/template/entry.client.d.ts.map +1 -1
- package/build/template/{entry.client.jsx → entry.client.js} +5 -6
- package/build/template/entry.client.js.map +1 -0
- package/build/template/routes/changelog.d.ts +1 -1
- package/build/template/routes/{changelog.jsx → changelog.js} +5 -4
- package/build/template/routes/changelog.js.map +1 -0
- package/build/template/routes/{index.jsx → index.js} +3 -2
- package/build/template/routes/index.js.map +1 -0
- package/build/template/routes/reference.$type.$field.d.ts +1 -1
- package/build/template/routes/{reference.$type.$field.jsx → reference.$type.$field.js} +6 -5
- package/build/template/routes/reference.$type.$field.js.map +1 -0
- package/build/template/routes/reference.$type.d.ts +1 -1
- package/build/template/routes/{reference.$type.jsx → reference.$type.js} +6 -5
- package/build/template/routes/reference.$type.js.map +1 -0
- package/build/template/routes/reference.d.ts +2 -2
- package/build/template/routes/{reference.jsx → reference.js} +6 -7
- package/build/template/routes/reference.js.map +1 -0
- package/build/template/routes/root.d.ts +2 -2
- package/build/template/routes/root.d.ts.map +1 -1
- package/build/template/routes/{root.jsx → root.js} +46 -88
- package/build/template/routes/root.js.map +1 -0
- package/build/template/routes.js +5 -0
- package/build/template/routes.js.map +1 -0
- package/build/template/server/app.js +1 -1
- package/build/template/server/app.js.map +1 -1
- package/build/template/server/{render-page.jsx → render-page.js} +3 -4
- package/build/template/server/render-page.js.map +1 -0
- package/build/template/server/ssg/generate.js +1 -1
- package/build/template/server/ssg/generate.js.map +1 -1
- package/build/template/server/ssg/get-route-paths.js +1 -1
- package/build/template/server/ssg/get-route-paths.js.map +1 -1
- package/build/template/server/view.js +1 -1
- package/build/template/server/view.js.map +1 -1
- package/package.json +57 -8
- package/src/api/vite/plugins/core.ts +0 -3
- package/src/api/vite/plugins/pages.ts +1 -1
- package/src/exports/components.ts +4 -1
- package/src/lib/graphql-document/$$.ts +4 -0
- package/src/lib/graphql-document/$.test.ts +132 -0
- package/src/lib/graphql-document/$.ts +1 -0
- package/src/lib/graphql-document/README.md +102 -0
- package/src/lib/graphql-document/analysis.ts +415 -0
- package/src/lib/graphql-document/components/CopyButton.tsx +76 -0
- package/src/lib/graphql-document/components/GraphQLDocument.tsx +250 -0
- package/src/lib/graphql-document/components/GraphQLDocument.unit.test.ts +188 -0
- package/src/lib/graphql-document/components/GraphQLDocumentWithSchema.tsx +70 -0
- package/src/lib/graphql-document/components/GraphQLIdentifierPopover.tsx +197 -0
- package/src/lib/graphql-document/components/IdentifierLink.tsx +160 -0
- package/src/lib/graphql-document/components/graphql-document-styles.ts +167 -0
- package/src/lib/graphql-document/components/index.ts +5 -0
- package/src/lib/graphql-document/demo.md +155 -0
- package/src/lib/graphql-document/example.ts +163 -0
- package/src/lib/graphql-document/graphql-document.ts +37 -0
- package/src/lib/graphql-document/hooks/use-tooltip-state.test.ts +76 -0
- package/src/lib/graphql-document/hooks/use-tooltip-state.ts +191 -0
- package/src/lib/graphql-document/positioning-simple.test.ts +252 -0
- package/src/lib/graphql-document/positioning-simple.ts +271 -0
- package/src/lib/graphql-document/schema-context.tsx +20 -0
- package/src/lib/graphql-document/schema-integration-example.ts +341 -0
- package/src/lib/graphql-document/schema-integration.test.ts +365 -0
- package/src/lib/graphql-document/schema-integration.ts +497 -0
- package/src/lib/graphql-document/types.ts +129 -0
- package/src/template/components/ArgumentAnnotation.tsx +1 -1
- package/src/template/components/ArgumentList.tsx +1 -1
- package/src/template/components/ArgumentListAnnotation.tsx +2 -2
- package/src/template/components/CodeBlockEnhancer.tsx +21 -21
- package/src/template/components/DeprecationReason.tsx +1 -1
- package/src/template/components/Description.tsx +1 -1
- package/src/template/components/Field.tsx +4 -4
- package/src/template/components/FieldList.tsx +1 -1
- package/src/template/components/FieldListSection.tsx +1 -1
- package/src/template/components/Link.tsx +2 -2
- package/src/template/components/MDXComponents.tsx +101 -0
- package/src/template/components/NamedType.tsx +2 -2
- package/src/template/components/TestComponent.tsx +6 -0
- package/src/template/components/TypeAnnotation.tsx +1 -1
- package/src/template/components/TypeFieldsLinkList.tsx +1 -1
- package/src/template/components/TypeIndex.tsx +1 -1
- package/src/template/components/content/$$.ts +1 -0
- package/src/template/components/content/GraphQLDocumentWithSchema.tsx +18 -0
- package/src/template/components/content/GraphQLDocumentWrapper.tsx +82 -0
- package/src/template/components/graphql/graphql.tsx +2 -2
- package/src/template/components/graphql/index.ts +1 -1
- package/src/template/components/graphql/type-link.tsx +2 -2
- package/src/template/entry.client.tsx +2 -2
- package/src/template/routes/changelog.tsx +1 -1
- package/src/template/routes/reference.$type.$field.tsx +3 -3
- package/src/template/routes/reference.$type.tsx +3 -3
- package/src/template/routes/reference.tsx +3 -3
- package/src/template/routes/root.tsx +36 -25
- package/src/template/routes.tsx +1 -1
- package/src/template/server/app.ts +1 -1
- package/src/template/server/ssg/generate.ts +1 -1
- package/src/template/server/ssg/get-route-paths.ts +1 -1
- package/src/template/server/view.ts +1 -1
- package/src/template/styles/code-block.css +32 -0
- package/build/template/components/ArgumentAnnotation.jsx +0 -16
- package/build/template/components/ArgumentAnnotation.jsx.map +0 -1
- package/build/template/components/ArgumentList.jsx +0 -16
- package/build/template/components/ArgumentList.jsx.map +0 -1
- package/build/template/components/ArgumentListAnnotation.jsx +0 -23
- package/build/template/components/ArgumentListAnnotation.jsx.map +0 -1
- package/build/template/components/Changelog.jsx +0 -68
- package/build/template/components/Changelog.jsx.map +0 -1
- package/build/template/components/CodeBlockEnhancer.jsx.map +0 -1
- package/build/template/components/DeprecationReason.jsx +0 -10
- package/build/template/components/DeprecationReason.jsx.map +0 -1
- package/build/template/components/Description.jsx +0 -10
- package/build/template/components/Description.jsx.map +0 -1
- package/build/template/components/Field.jsx +0 -22
- package/build/template/components/Field.jsx.map +0 -1
- package/build/template/components/FieldList.jsx.map +0 -1
- package/build/template/components/FieldListSection.jsx.map +0 -1
- package/build/template/components/HamburgerMenu.jsx +0 -53
- package/build/template/components/HamburgerMenu.jsx.map +0 -1
- package/build/template/components/Link.jsx.map +0 -1
- package/build/template/components/Logo.jsx +0 -29
- package/build/template/components/Logo.jsx.map +0 -1
- package/build/template/components/Markdown.jsx.map +0 -1
- package/build/template/components/MissingSchema.jsx +0 -4
- package/build/template/components/MissingSchema.jsx.map +0 -1
- package/build/template/components/NamedType.jsx +0 -17
- package/build/template/components/NamedType.jsx.map +0 -1
- package/build/template/components/NotFound.jsx +0 -26
- package/build/template/components/NotFound.jsx.map +0 -1
- package/build/template/components/RadixLink.jsx.map +0 -1
- package/build/template/components/Texts/MinorHeading.jsx.map +0 -1
- package/build/template/components/ThemeToggle.jsx +0 -10
- package/build/template/components/ThemeToggle.jsx.map +0 -1
- package/build/template/components/TypeAnnotation.jsx.map +0 -1
- package/build/template/components/TypeFieldsLinkList.jsx +0 -17
- package/build/template/components/TypeFieldsLinkList.jsx.map +0 -1
- package/build/template/components/TypeIndex.jsx +0 -27
- package/build/template/components/TypeIndex.jsx.map +0 -1
- package/build/template/components/graphql/graphql.jsx +0 -3
- package/build/template/components/graphql/graphql.jsx.map +0 -1
- package/build/template/components/graphql/type-kind-icon.jsx.map +0 -1
- package/build/template/components/graphql/type-link.jsx +0 -16
- package/build/template/components/graphql/type-link.jsx.map +0 -1
- package/build/template/components/sidebar/Sidebar.jsx +0 -15
- package/build/template/components/sidebar/Sidebar.jsx.map +0 -1
- package/build/template/components/sidebar/SidebarItem.jsx.map +0 -1
- package/build/template/components/sidebar/ToggleButton.jsx +0 -6
- package/build/template/components/sidebar/ToggleButton.jsx.map +0 -1
- package/build/template/entry.client.jsx.map +0 -1
- package/build/template/routes/changelog.jsx.map +0 -1
- package/build/template/routes/index.jsx.map +0 -1
- package/build/template/routes/reference.$type.$field.jsx.map +0 -1
- package/build/template/routes/reference.$type.jsx.map +0 -1
- package/build/template/routes/reference.jsx.map +0 -1
- package/build/template/routes/root.jsx.map +0 -1
- package/build/template/routes.jsx +0 -5
- package/build/template/routes.jsx.map +0 -1
- package/build/template/server/render-page.jsx.map +0 -1
@@ -0,0 +1,76 @@
|
|
1
|
+
/**
|
2
|
+
* Unit tests for tooltip state management hook
|
3
|
+
*
|
4
|
+
* @vitest-environment jsdom
|
5
|
+
*/
|
6
|
+
|
7
|
+
import { act, renderHook } from '@testing-library/react'
|
8
|
+
import { describe, expect, it, vi } from 'vitest'
|
9
|
+
import { useTooltipState } from './use-tooltip-state.ts'
|
10
|
+
|
11
|
+
describe('useTooltipState', () => {
|
12
|
+
it('shows tooltip after hover delay', () => {
|
13
|
+
vi.useFakeTimers()
|
14
|
+
const { result } = renderHook(() => useTooltipState({ showDelay: 300 }))
|
15
|
+
|
16
|
+
act(() => {
|
17
|
+
result.current.onHoverStart('field-1')
|
18
|
+
})
|
19
|
+
expect(result.current.isOpen('field-1')).toBe(false)
|
20
|
+
|
21
|
+
act(() => {
|
22
|
+
vi.advanceTimersByTime(300)
|
23
|
+
})
|
24
|
+
expect(result.current.isOpen('field-1')).toBe(true)
|
25
|
+
vi.useRealTimers()
|
26
|
+
})
|
27
|
+
|
28
|
+
it('hides tooltip after hover end delay', () => {
|
29
|
+
vi.useFakeTimers()
|
30
|
+
const { result } = renderHook(() => useTooltipState())
|
31
|
+
|
32
|
+
// Show tooltip
|
33
|
+
act(() => {
|
34
|
+
result.current.onHoverStart('field-1')
|
35
|
+
vi.advanceTimersByTime(300)
|
36
|
+
})
|
37
|
+
|
38
|
+
// Trigger hide
|
39
|
+
act(() => {
|
40
|
+
result.current.onHoverEnd('field-1')
|
41
|
+
})
|
42
|
+
expect(result.current.isOpen('field-1')).toBe(true) // Still open during delay
|
43
|
+
|
44
|
+
act(() => {
|
45
|
+
vi.advanceTimersByTime(200)
|
46
|
+
})
|
47
|
+
expect(result.current.isOpen('field-1')).toBe(false)
|
48
|
+
vi.useRealTimers()
|
49
|
+
})
|
50
|
+
|
51
|
+
it('pins and unpins tooltip on toggle', () => {
|
52
|
+
const { result } = renderHook(() => useTooltipState())
|
53
|
+
|
54
|
+
act(() => {
|
55
|
+
result.current.onTogglePin('field-1')
|
56
|
+
})
|
57
|
+
expect(result.current.isPinned('field-1')).toBe(true)
|
58
|
+
|
59
|
+
act(() => {
|
60
|
+
result.current.onTogglePin('field-1')
|
61
|
+
})
|
62
|
+
expect(result.current.isPinned('field-1')).toBe(false)
|
63
|
+
})
|
64
|
+
|
65
|
+
it('allows multiple pins when enabled', () => {
|
66
|
+
const { result } = renderHook(() => useTooltipState())
|
67
|
+
|
68
|
+
act(() => {
|
69
|
+
result.current.onTogglePin('field-1')
|
70
|
+
result.current.onTogglePin('field-2')
|
71
|
+
})
|
72
|
+
|
73
|
+
expect(result.current.isPinned('field-1')).toBe(true)
|
74
|
+
expect(result.current.isPinned('field-2')).toBe(true)
|
75
|
+
})
|
76
|
+
})
|
@@ -0,0 +1,191 @@
|
|
1
|
+
/**
|
2
|
+
* State management for GraphQL document tooltips
|
3
|
+
*
|
4
|
+
* Handles hover delays, pinning, and multiple tooltip coordination
|
5
|
+
*/
|
6
|
+
|
7
|
+
import { React } from '#dep/react/index'
|
8
|
+
|
9
|
+
export interface TooltipState {
|
10
|
+
/** Currently visible tooltip (via hover) */
|
11
|
+
hoveredId: string | null
|
12
|
+
/** Set of pinned tooltip IDs */
|
13
|
+
pinnedIds: Set<string>
|
14
|
+
/** ID pending show (waiting for delay) */
|
15
|
+
pendingShowId: string | null
|
16
|
+
/** ID pending hide (waiting for grace period) */
|
17
|
+
pendingHideId: string | null
|
18
|
+
}
|
19
|
+
|
20
|
+
export interface UseTooltipStateOptions {
|
21
|
+
/** Delay before showing tooltip on hover (ms) */
|
22
|
+
showDelay?: number
|
23
|
+
/** Delay before hiding tooltip on mouse leave (ms) */
|
24
|
+
hideDelay?: number
|
25
|
+
/** Whether to allow multiple pinned tooltips */
|
26
|
+
allowMultiplePins?: boolean
|
27
|
+
}
|
28
|
+
|
29
|
+
export interface UseTooltipStateReturn {
|
30
|
+
/** Check if a tooltip should be visible */
|
31
|
+
isOpen: (id: string) => boolean
|
32
|
+
/** Check if a tooltip is pinned */
|
33
|
+
isPinned: (id: string) => boolean
|
34
|
+
/** Handle hover start */
|
35
|
+
onHoverStart: (id: string) => void
|
36
|
+
/** Handle hover end */
|
37
|
+
onHoverEnd: (id: string) => void
|
38
|
+
/** Handle click (toggle pin) */
|
39
|
+
onTogglePin: (id: string) => void
|
40
|
+
/** Handle tooltip content hover (cancels hide) */
|
41
|
+
onTooltipHover: (id: string) => void
|
42
|
+
/** Unpin a specific tooltip */
|
43
|
+
unpin: (id: string) => void
|
44
|
+
/** Unpin all tooltips */
|
45
|
+
unpinAll: () => void
|
46
|
+
}
|
47
|
+
|
48
|
+
export const useTooltipState = (options: UseTooltipStateOptions = {}): UseTooltipStateReturn => {
|
49
|
+
const {
|
50
|
+
showDelay = 300,
|
51
|
+
hideDelay = 200,
|
52
|
+
allowMultiplePins = true,
|
53
|
+
} = options
|
54
|
+
|
55
|
+
const [hoveredId, setHoveredId] = React.useState<string | null>(null)
|
56
|
+
const [pinnedIds, setPinnedIds] = React.useState<Set<string>>(new Set())
|
57
|
+
const [pendingShowId, setPendingShowId] = React.useState<string | null>(null)
|
58
|
+
const [pendingHideId, setPendingHideId] = React.useState<string | null>(null)
|
59
|
+
|
60
|
+
// Timer refs
|
61
|
+
const showTimerRef = React.useRef<NodeJS.Timeout | null>(null)
|
62
|
+
const hideTimerRef = React.useRef<NodeJS.Timeout | null>(null)
|
63
|
+
|
64
|
+
// Clear any pending timers
|
65
|
+
const clearTimers = React.useCallback(() => {
|
66
|
+
if (showTimerRef.current) {
|
67
|
+
clearTimeout(showTimerRef.current)
|
68
|
+
showTimerRef.current = null
|
69
|
+
}
|
70
|
+
if (hideTimerRef.current) {
|
71
|
+
clearTimeout(hideTimerRef.current)
|
72
|
+
hideTimerRef.current = null
|
73
|
+
}
|
74
|
+
setPendingShowId(null)
|
75
|
+
setPendingHideId(null)
|
76
|
+
}, [])
|
77
|
+
|
78
|
+
// Check if tooltip should be visible
|
79
|
+
const isOpen = React.useCallback((id: string): boolean => {
|
80
|
+
return hoveredId === id || pinnedIds.has(id)
|
81
|
+
}, [hoveredId, pinnedIds])
|
82
|
+
|
83
|
+
// Check if tooltip is pinned
|
84
|
+
const isPinned = React.useCallback((id: string): boolean => {
|
85
|
+
return pinnedIds.has(id)
|
86
|
+
}, [pinnedIds])
|
87
|
+
|
88
|
+
// Handle hover start
|
89
|
+
const onHoverStart = React.useCallback((id: string) => {
|
90
|
+
// Don't show if already pinned
|
91
|
+
if (pinnedIds.has(id)) return
|
92
|
+
|
93
|
+
// Cancel any pending hide for this ID
|
94
|
+
if (pendingHideId === id) {
|
95
|
+
clearTimeout(hideTimerRef.current!)
|
96
|
+
hideTimerRef.current = null
|
97
|
+
setPendingHideId(null)
|
98
|
+
return
|
99
|
+
}
|
100
|
+
|
101
|
+
// Clear any other pending operations
|
102
|
+
clearTimers()
|
103
|
+
|
104
|
+
// Schedule show
|
105
|
+
setPendingShowId(id)
|
106
|
+
showTimerRef.current = setTimeout(() => {
|
107
|
+
setHoveredId(id)
|
108
|
+
setPendingShowId(null)
|
109
|
+
}, showDelay)
|
110
|
+
}, [pinnedIds, pendingHideId, clearTimers, showDelay])
|
111
|
+
|
112
|
+
// Handle hover end
|
113
|
+
const onHoverEnd = React.useCallback((id: string) => {
|
114
|
+
// Don't hide if pinned
|
115
|
+
if (pinnedIds.has(id)) return
|
116
|
+
|
117
|
+
// Cancel pending show if still waiting
|
118
|
+
if (pendingShowId === id) {
|
119
|
+
clearTimeout(showTimerRef.current!)
|
120
|
+
showTimerRef.current = null
|
121
|
+
setPendingShowId(null)
|
122
|
+
return
|
123
|
+
}
|
124
|
+
|
125
|
+
// Only hide if currently showing this tooltip
|
126
|
+
if (hoveredId === id) {
|
127
|
+
setPendingHideId(id)
|
128
|
+
hideTimerRef.current = setTimeout(() => {
|
129
|
+
// First set hovered to null to trigger close animation
|
130
|
+
setHoveredId(null)
|
131
|
+
setPendingHideId(null)
|
132
|
+
}, hideDelay)
|
133
|
+
}
|
134
|
+
}, [pinnedIds, pendingShowId, hoveredId, hideDelay])
|
135
|
+
|
136
|
+
// Handle tooltip content hover (cancels hide)
|
137
|
+
const onTooltipHover = React.useCallback((id: string) => {
|
138
|
+
if (pendingHideId === id) {
|
139
|
+
clearTimeout(hideTimerRef.current!)
|
140
|
+
hideTimerRef.current = null
|
141
|
+
setPendingHideId(null)
|
142
|
+
}
|
143
|
+
}, [pendingHideId])
|
144
|
+
|
145
|
+
// Toggle pin state
|
146
|
+
const onTogglePin = React.useCallback((id: string) => {
|
147
|
+
clearTimers()
|
148
|
+
|
149
|
+
setPinnedIds((prev: Set<string>) => {
|
150
|
+
const next = new Set(prev)
|
151
|
+
if (next.has(id)) {
|
152
|
+
// Unpin
|
153
|
+
next.delete(id)
|
154
|
+
setHoveredId(null) // Also clear hover state
|
155
|
+
} else {
|
156
|
+
// Pin
|
157
|
+
if (!allowMultiplePins) {
|
158
|
+
next.clear() // Clear other pins
|
159
|
+
}
|
160
|
+
next.add(id)
|
161
|
+
setHoveredId(null) // Clear hover state since it's now pinned
|
162
|
+
}
|
163
|
+
return next
|
164
|
+
})
|
165
|
+
}, [clearTimers, allowMultiplePins])
|
166
|
+
|
167
|
+
// Unpin specific tooltip
|
168
|
+
const unpin = React.useCallback((id: string) => {
|
169
|
+
setPinnedIds((prev: Set<string>) => {
|
170
|
+
const next = new Set(prev)
|
171
|
+
next.delete(id)
|
172
|
+
return next
|
173
|
+
})
|
174
|
+
}, [])
|
175
|
+
|
176
|
+
// Unpin all tooltips
|
177
|
+
const unpinAll = React.useCallback(() => {
|
178
|
+
setPinnedIds(new Set())
|
179
|
+
}, [])
|
180
|
+
|
181
|
+
return {
|
182
|
+
isOpen,
|
183
|
+
isPinned,
|
184
|
+
onHoverStart,
|
185
|
+
onHoverEnd,
|
186
|
+
onTogglePin,
|
187
|
+
onTooltipHover,
|
188
|
+
unpin,
|
189
|
+
unpinAll,
|
190
|
+
}
|
191
|
+
}
|
@@ -0,0 +1,252 @@
|
|
1
|
+
/**
|
2
|
+
* @vitest-environment jsdom
|
3
|
+
*/
|
4
|
+
|
5
|
+
import { beforeEach, describe, expect, it } from 'vitest'
|
6
|
+
import { createSimpleOverlay, createSimplePositionCalculator } from './positioning-simple.ts'
|
7
|
+
import type { Identifier } from './types.ts'
|
8
|
+
|
9
|
+
// Helper to create test identifier
|
10
|
+
const createTestIdentifier = (
|
11
|
+
name: string,
|
12
|
+
line: number,
|
13
|
+
column: number,
|
14
|
+
kind: Identifier['kind'] = 'Field',
|
15
|
+
): Identifier => ({
|
16
|
+
name,
|
17
|
+
kind,
|
18
|
+
position: {
|
19
|
+
start: column - 1,
|
20
|
+
end: column - 1 + name.length,
|
21
|
+
line,
|
22
|
+
column,
|
23
|
+
},
|
24
|
+
schemaPath: [name],
|
25
|
+
context: { selectionPath: [] },
|
26
|
+
})
|
27
|
+
|
28
|
+
describe('Simple Positioning Engine', () => {
|
29
|
+
describe('SimplePositionCalculator', () => {
|
30
|
+
const calculator = createSimplePositionCalculator()
|
31
|
+
|
32
|
+
it('should wrap identifiers in spans', () => {
|
33
|
+
const container = document.createElement('div')
|
34
|
+
container.innerHTML = `
|
35
|
+
<pre class="shiki">
|
36
|
+
<code>
|
37
|
+
<span class="line">query GetUser {</span>
|
38
|
+
<span class="line"> user {</span>
|
39
|
+
<span class="line"> name</span>
|
40
|
+
<span class="line"> }</span>
|
41
|
+
<span class="line">}</span>
|
42
|
+
</code>
|
43
|
+
</pre>
|
44
|
+
`
|
45
|
+
|
46
|
+
const identifiers = [
|
47
|
+
createTestIdentifier('query', 1, 1),
|
48
|
+
createTestIdentifier('user', 2, 3, 'Field'),
|
49
|
+
createTestIdentifier('name', 3, 5, 'Field'),
|
50
|
+
]
|
51
|
+
|
52
|
+
calculator.prepareCodeBlock(container, identifiers)
|
53
|
+
|
54
|
+
// Check that identifiers were wrapped
|
55
|
+
const wrappedElements = container.querySelectorAll('[data-graphql-id]')
|
56
|
+
expect(wrappedElements.length).toBe(3)
|
57
|
+
|
58
|
+
// Check first wrapped element
|
59
|
+
const firstWrapped = wrappedElements[0] as HTMLElement
|
60
|
+
expect(firstWrapped.textContent).toBe('query')
|
61
|
+
expect(firstWrapped.getAttribute('data-graphql-name')).toBe('query')
|
62
|
+
expect(firstWrapped.getAttribute('data-graphql-kind')).toBe('Field')
|
63
|
+
})
|
64
|
+
|
65
|
+
it('should handle multiple identifiers in same line', () => {
|
66
|
+
const container = document.createElement('div')
|
67
|
+
container.innerHTML = `
|
68
|
+
<pre class="shiki">
|
69
|
+
<code>
|
70
|
+
<span class="line">query GetUserById($id: ID!) {</span>
|
71
|
+
</code>
|
72
|
+
</pre>
|
73
|
+
`
|
74
|
+
|
75
|
+
const identifiers = [
|
76
|
+
createTestIdentifier('query', 1, 1),
|
77
|
+
createTestIdentifier('GetUserById', 1, 7),
|
78
|
+
createTestIdentifier('$id', 1, 19, 'Variable'),
|
79
|
+
createTestIdentifier('ID', 1, 24, 'Type'),
|
80
|
+
]
|
81
|
+
|
82
|
+
calculator.prepareCodeBlock(container, identifiers)
|
83
|
+
|
84
|
+
const wrappedElements = container.querySelectorAll('[data-graphql-id]')
|
85
|
+
expect(wrappedElements.length).toBe(4)
|
86
|
+
})
|
87
|
+
|
88
|
+
it('should get positions of wrapped identifiers', () => {
|
89
|
+
const container = document.createElement('div')
|
90
|
+
container.innerHTML = `
|
91
|
+
<pre class="shiki">
|
92
|
+
<code>
|
93
|
+
<span class="line"><span data-graphql-id="0-user-Field" data-graphql-name="user" data-graphql-kind="Field" data-graphql-start="0" data-graphql-end="4" data-graphql-line="1" data-graphql-column="1" data-graphql-path="user">user</span> {</span>
|
94
|
+
</code>
|
95
|
+
</pre>
|
96
|
+
`
|
97
|
+
|
98
|
+
// Mock getBoundingClientRect
|
99
|
+
container.getBoundingClientRect = () => ({
|
100
|
+
top: 0,
|
101
|
+
left: 0,
|
102
|
+
right: 100,
|
103
|
+
bottom: 100,
|
104
|
+
width: 100,
|
105
|
+
height: 100,
|
106
|
+
x: 0,
|
107
|
+
y: 0,
|
108
|
+
toJSON: () => ({}),
|
109
|
+
})
|
110
|
+
|
111
|
+
const userSpan = container.querySelector('[data-graphql-id]') as HTMLElement
|
112
|
+
userSpan.getBoundingClientRect = () => ({
|
113
|
+
top: 10,
|
114
|
+
left: 20,
|
115
|
+
right: 60,
|
116
|
+
bottom: 30,
|
117
|
+
width: 40,
|
118
|
+
height: 20,
|
119
|
+
x: 20,
|
120
|
+
y: 10,
|
121
|
+
toJSON: () => ({}),
|
122
|
+
})
|
123
|
+
|
124
|
+
const positions = calculator.getIdentifierPositions(container)
|
125
|
+
|
126
|
+
expect(positions.size).toBe(1)
|
127
|
+
const result = positions.get('0-user-Field')
|
128
|
+
expect(result).toBeDefined()
|
129
|
+
expect(result!.position).toEqual({
|
130
|
+
top: 10,
|
131
|
+
left: 20,
|
132
|
+
width: 40,
|
133
|
+
height: 20,
|
134
|
+
})
|
135
|
+
expect(result!.identifier.name).toBe('user')
|
136
|
+
expect(result!.identifier.kind).toBe('Field')
|
137
|
+
})
|
138
|
+
|
139
|
+
it('should skip already wrapped identifiers', () => {
|
140
|
+
const container = document.createElement('div')
|
141
|
+
container.innerHTML = `
|
142
|
+
<pre class="shiki">
|
143
|
+
<code>
|
144
|
+
<span class="line"><span data-graphql-id="existing">user</span> {</span>
|
145
|
+
</code>
|
146
|
+
</pre>
|
147
|
+
`
|
148
|
+
|
149
|
+
const identifiers = [
|
150
|
+
createTestIdentifier('user', 1, 1),
|
151
|
+
]
|
152
|
+
|
153
|
+
calculator.prepareCodeBlock(container, identifiers)
|
154
|
+
|
155
|
+
// Should still only have one wrapped element
|
156
|
+
const wrappedElements = container.querySelectorAll('[data-graphql-id]')
|
157
|
+
expect(wrappedElements.length).toBe(1)
|
158
|
+
expect(wrappedElements[0]!.getAttribute('data-graphql-id')).toBe('existing')
|
159
|
+
})
|
160
|
+
|
161
|
+
it('should handle empty lines gracefully', () => {
|
162
|
+
const container = document.createElement('div')
|
163
|
+
container.innerHTML = `
|
164
|
+
<pre class="shiki">
|
165
|
+
<code>
|
166
|
+
<span class="line">query {</span>
|
167
|
+
<span class="line"></span>
|
168
|
+
<span class="line"> user</span>
|
169
|
+
</code>
|
170
|
+
</pre>
|
171
|
+
`
|
172
|
+
|
173
|
+
const identifiers = [
|
174
|
+
createTestIdentifier('query', 1, 1),
|
175
|
+
createTestIdentifier('user', 3, 3),
|
176
|
+
]
|
177
|
+
|
178
|
+
expect(() => {
|
179
|
+
calculator.prepareCodeBlock(container, identifiers)
|
180
|
+
}).not.toThrow()
|
181
|
+
|
182
|
+
const wrappedElements = container.querySelectorAll('[data-graphql-id]')
|
183
|
+
expect(wrappedElements.length).toBe(2)
|
184
|
+
})
|
185
|
+
})
|
186
|
+
|
187
|
+
describe('createSimpleOverlay', () => {
|
188
|
+
it('should create positioned overlay element', () => {
|
189
|
+
const position = { top: 10, left: 20, width: 40, height: 20 }
|
190
|
+
const identifier = createTestIdentifier('user', 1, 1)
|
191
|
+
|
192
|
+
const overlay = createSimpleOverlay(position, identifier)
|
193
|
+
|
194
|
+
expect(overlay.style.position).toBe('absolute')
|
195
|
+
expect(overlay.style.top).toBe('10px')
|
196
|
+
expect(overlay.style.left).toBe('20px')
|
197
|
+
expect(overlay.style.width).toBe('40px')
|
198
|
+
expect(overlay.style.height).toBe('20px')
|
199
|
+
expect(overlay.style.cursor).toBe('pointer')
|
200
|
+
expect(overlay.getAttribute('data-graphql-overlay')).toBe('true')
|
201
|
+
expect(overlay.getAttribute('data-graphql-name')).toBe('user')
|
202
|
+
expect(overlay.getAttribute('data-graphql-kind')).toBe('Field')
|
203
|
+
})
|
204
|
+
|
205
|
+
it('should handle click events', () => {
|
206
|
+
const position = { top: 10, left: 20, width: 40, height: 20 }
|
207
|
+
const identifier = createTestIdentifier('user', 1, 1)
|
208
|
+
let clickedIdentifier: Identifier | null = null
|
209
|
+
|
210
|
+
const overlay = createSimpleOverlay(position, identifier, {
|
211
|
+
onClick: (id) => {
|
212
|
+
clickedIdentifier = id
|
213
|
+
},
|
214
|
+
})
|
215
|
+
|
216
|
+
// Simulate click
|
217
|
+
const event = new MouseEvent('click')
|
218
|
+
overlay.dispatchEvent(event)
|
219
|
+
|
220
|
+
expect(clickedIdentifier).toBe(identifier)
|
221
|
+
})
|
222
|
+
|
223
|
+
it('should handle hover events', () => {
|
224
|
+
const position = { top: 10, left: 20, width: 40, height: 20 }
|
225
|
+
const identifier = createTestIdentifier('user', 1, 1)
|
226
|
+
let hoveredIdentifier: Identifier | null = null
|
227
|
+
|
228
|
+
const overlay = createSimpleOverlay(position, identifier, {
|
229
|
+
onHover: (id) => {
|
230
|
+
hoveredIdentifier = id
|
231
|
+
},
|
232
|
+
})
|
233
|
+
|
234
|
+
// Simulate hover
|
235
|
+
const event = new MouseEvent('mouseenter')
|
236
|
+
overlay.dispatchEvent(event)
|
237
|
+
|
238
|
+
expect(hoveredIdentifier).toBe(identifier)
|
239
|
+
})
|
240
|
+
|
241
|
+
it('should apply custom className', () => {
|
242
|
+
const position = { top: 10, left: 20, width: 40, height: 20 }
|
243
|
+
const identifier = createTestIdentifier('user', 1, 1)
|
244
|
+
|
245
|
+
const overlay = createSimpleOverlay(position, identifier, {
|
246
|
+
className: 'custom-overlay-class',
|
247
|
+
})
|
248
|
+
|
249
|
+
expect(overlay.className).toBe('custom-overlay-class')
|
250
|
+
})
|
251
|
+
})
|
252
|
+
})
|