diff --git a/src/components/config/VisualConfigEditor.module.scss b/src/components/config/VisualConfigEditor.module.scss index 832148b..d75402a 100644 --- a/src/components/config/VisualConfigEditor.module.scss +++ b/src/components/config/VisualConfigEditor.module.scss @@ -408,10 +408,9 @@ .floatingSidebarContainer { position: fixed; - left: var(--visual-config-floating-left, 16px); - top: var(--visual-config-floating-top, 120px); - width: var(--visual-config-floating-width, 280px); - max-height: var(--visual-config-floating-max-height, calc(100vh - 136px)); + left: 0; + top: 0; + will-change: transform, width, max-height; z-index: 45; opacity: 0; pointer-events: none; @@ -429,9 +428,7 @@ padding: 12px; border-radius: 26px; border: 1px solid color-mix(in srgb, var(--border-color) 84%, transparent); - background: color-mix(in srgb, var(--bg-primary) 76%, transparent); - backdrop-filter: blur(14px); - -webkit-backdrop-filter: blur(14px); + background: color-mix(in srgb, var(--bg-primary) 96%, transparent); box-shadow: 0 24px 56px -34px rgba(0, 0, 0, 0.42); -ms-overflow-style: none; scrollbar-width: none; diff --git a/src/components/config/VisualConfigEditor.tsx b/src/components/config/VisualConfigEditor.tsx index d4751a6..122c166 100644 --- a/src/components/config/VisualConfigEditor.tsx +++ b/src/components/config/VisualConfigEditor.tsx @@ -392,10 +392,9 @@ export function VisualConfigEditor({ if (!floatingElement) return undefined; const clearFloatingStyles = () => { - floatingElement.style.removeProperty('--visual-config-floating-left'); - floatingElement.style.removeProperty('--visual-config-floating-top'); - floatingElement.style.removeProperty('--visual-config-floating-width'); - floatingElement.style.removeProperty('--visual-config-floating-max-height'); + floatingElement.style.removeProperty('transform'); + floatingElement.style.removeProperty('width'); + floatingElement.style.removeProperty('max-height'); floatingElement.style.removeProperty('opacity'); floatingElement.style.removeProperty('pointer-events'); }; @@ -405,7 +404,8 @@ export function VisualConfigEditor({ return undefined; } - const getHeaderHeight = () => { + /* ---- Cache header height – recomputed only on resize ---- */ + const computeHeaderHeight = () => { const header = document.querySelector('.main-header') as HTMLElement | null; if (header) return header.getBoundingClientRect().height; @@ -413,8 +413,14 @@ export function VisualConfigEditor({ const parsed = Number.parseFloat(raw); return Number.isFinite(parsed) ? parsed : 64; }; + let headerHeight = computeHeaderHeight(); + + /* ---- Cache content scroller – resolved once ---- */ + const contentScroller = document.querySelector('.content') as HTMLElement | null; + + /* ---- Cache floating height from previous frame ---- */ + let cachedFloatingHeight = floatingElement.getBoundingClientRect().height || 200; - const getContentScroller = () => document.querySelector('.content') as HTMLElement | null; let frameId = 0; const updateFloatingPosition = () => { @@ -422,10 +428,9 @@ export function VisualConfigEditor({ const anchorRect = anchorElement.getBoundingClientRect(); const workspaceRect = workspaceElement.getBoundingClientRect(); - const floatingHeight = floatingElement.getBoundingClientRect().height; - const stickyTop = getHeaderHeight() + 20; + const stickyTop = headerHeight + 20; const viewportPadding = 16; - const maxTop = workspaceRect.bottom - floatingHeight; + const maxTop = workspaceRect.bottom - cachedFloatingHeight; const unclampedTop = Math.min(Math.max(anchorRect.top, stickyTop), maxTop); const top = Math.max(unclampedTop, viewportPadding); const left = Math.max(anchorRect.left, viewportPadding); @@ -436,10 +441,9 @@ export function VisualConfigEditor({ const maxHeight = Math.max(window.innerHeight - top - viewportPadding, 160); const isVisible = workspaceRect.bottom > stickyTop + 24 && anchorRect.top < window.innerHeight; - floatingElement.style.setProperty('--visual-config-floating-left', `${left}px`); - floatingElement.style.setProperty('--visual-config-floating-top', `${top}px`); - floatingElement.style.setProperty('--visual-config-floating-width', `${width}px`); - floatingElement.style.setProperty('--visual-config-floating-max-height', `${maxHeight}px`); + floatingElement.style.transform = `translate3d(${left}px, ${top}px, 0)`; + floatingElement.style.width = `${width}px`; + floatingElement.style.maxHeight = `${maxHeight}px`; floatingElement.style.opacity = isVisible ? '1' : '0'; floatingElement.style.pointerEvents = isVisible ? 'auto' : 'none'; }; @@ -449,10 +453,15 @@ export function VisualConfigEditor({ frameId = requestAnimationFrame(updateFloatingPosition); }; + const handleResize = () => { + headerHeight = computeHeaderHeight(); + cachedFloatingHeight = floatingElement.getBoundingClientRect().height || cachedFloatingHeight; + requestPositionUpdate(); + }; + requestPositionUpdate(); - const contentScroller = getContentScroller(); - window.addEventListener('resize', requestPositionUpdate); + window.addEventListener('resize', handleResize); window.addEventListener('scroll', requestPositionUpdate, { passive: true }); contentScroller?.addEventListener('scroll', requestPositionUpdate, { passive: true }); @@ -460,12 +469,11 @@ export function VisualConfigEditor({ typeof ResizeObserver === 'undefined' ? null : new ResizeObserver(requestPositionUpdate); resizeObserver?.observe(anchorElement); resizeObserver?.observe(workspaceElement); - resizeObserver?.observe(floatingElement); return () => { if (frameId) cancelAnimationFrame(frameId); resizeObserver?.disconnect(); - window.removeEventListener('resize', requestPositionUpdate); + window.removeEventListener('resize', handleResize); window.removeEventListener('scroll', requestPositionUpdate); contentScroller?.removeEventListener('scroll', requestPositionUpdate); clearFloatingStyles();