liquid_glass_widgets 0.17.1 copy "liquid_glass_widgets: ^0.17.1" to clipboard
liquid_glass_widgets: ^0.17.1 copied to clipboard

A Flutter UI kit implementing the iOS 26 Liquid Glass design language. Features shader-based glassmorphism, physics-driven jelly animations, and dynamic lighting.

0.17.1 #

๐Ÿ› Fix โ€” platformViewBackdrop toggle no longer snaps the selected-tab indicator (#112 by @jfhair) #

Toggling platformViewBackdrop at runtime (e.g. switching between a map tab and a Flutter-content tab) caused the selected-indicator pill to snap to the new tab instead of sliding. The spring animation controllers inside the indicator subtree were being re-seeded at their already-settled value because the toggle added or removed the LiquidGlassBlendGroup wrapper in AdaptiveLiquidGlassLayer, changing the child's depth in the element tree and forcing Flutter to discard and re-inflate the whole subtree via initState.

Fix: AdaptiveLiquidGlassLayer is now a StatefulWidget that holds a stable GlobalKey. The child is always wrapped in a KeyedSubtree with that key, so the element identity is preserved across the wrapper toggle โ€” the indicator's AnimationControllers survive and the morph continues correctly.

No API changes. No breaking changes.


0.17.0 #

๐Ÿ”ฌ iOS 26 Concave Lens Pinch โ€” All Four Pill Widgets #

The indicatorPinchStrength concave lens warp is now unified across all four interactive pill widgets. During a drag the pill edges curve inward (iOS 26 "through a lens" effect). Fully tunable โ€” 0.0 disables it, 1.0 is maximum distortion.

New parameters #

  • GlassTabBar.indicatorPinchStrength (default 0.4)
  • GlassTabBar.indicatorExpansion (default EdgeInsets.symmetric(horizontal: 12, vertical: 8))
  • GlassSegmentedControl.indicatorPinchStrength (default 0.4)
  • GlassSegmentedControl.indicatorExpansion (default EdgeInsets.symmetric(horizontal: 12, vertical: 8))
  • AnimatedGlassIndicator exported from the public API โ€” enables baseIndicatorSettings.copyWith(...) from app code.

Changed defaults (AnimatedGlassIndicator.baseIndicatorSettings) #

  • glassColor: alpha: 0.15 โ†’ alpha: 0.0 โ€” glass pill no longer applies a white tint overlay by default.
  • chromaticAberration: GlassDefaults.chromaticAberration โ†’ 0.15 โ€” the iridescent rim fringe is now explicitly set for iOS 26 parity.

Bug fixes #

  • GlassSegmentedControl refraction โ€” labels are now refracted through the glass pill at GlassQuality.premium (was rendered in wrong z-order).
  • GlassTabBar indicator radius โ€” resting pill now inherits the tab bar's borderRadius (was hardcoded 16 px).
  • AnimatedGlassIndicator settings merge โ€” partial indicatorSettings overrides no longer silently reset chromaticAberration.
  • Pinch lens jitter at rest โ€” icon and label content no longer shimmers through the lens when the pill settles. Root cause: the jelly spring's micro-oscillations (ยฑ10 % of thickness) were directly amplified into the UV warp. Fixed by applying a quadratic ease-out to the pinch multiplier (1 โˆ’ (1 โˆ’ fade)ยฒ), compressing the near-settled oscillation range โ‰ˆ10ร—.

Try it โ€” Indicator Parity demo #

The example app includes a live Indicator Parity demo (Demos โ†’ Indicator Parity) with all four pill widgets side-by-side and real-time sliders for pinchStrength, indicatorExpansion, and chromaticAberration. Use it to tune parameters before writing any code.

๐ŸŒ‘ Apple Dimming Layer โ€” LiquidGlassSettings.backerColor (#111 by @jfhair) #

New optional backerColor on LiquidGlassSettings โ€” a shape-matched color pad composited behind the glass, giving a control's content contrast over rich or colorful backdrops (video, maps, photography) where the glass tint alone can't. This is Apple's "dimming layer" guidance from the Human Interface Guidelines (Materials section) and the pattern behind SwiftUI's clear Glass variant.

LiquidGlassSettings(
  glassColor: Color(0x20FFFFFF),
  backerColor: Color(0x59000000), // ~35% black โ€” Apple's starting point
)
  • backerColor (Color?, default null) โ€” the color's alpha is the dimming opacity. null means no backer, so all existing recipes are untouched.
  • Rendered at the widget level (like shadow) and clipped to the glass shape via ClipRRect, so it composites correctly even over a PlatformView โ€” maps, video โ€” where a shader-side tint cannot reach.
  • Applies in both light and dark mode, and for flat-edge shapes (a bar over a map is a primary use case).
  • Skipped on the grouped path (like shadow) โ€” inserting a Stack between grouped glass and its shared layer would break metaball morphing.
  • lerp fades backerColor smoothly from transparent when one side is null, rather than snapping at the midpoint.

Migration #

All four widgets share the same tuning API:

indicatorPinchStrength: 0.4,
indicatorExpansion: EdgeInsets.symmetric(horizontal: 12, vertical: 8),
indicatorSettings: AnimatedGlassIndicator.baseIndicatorSettings
    .copyWith(chromaticAberration: 0.15),

0.16.3 #

โœจ GlassTintBlend โ€” selectable tint blending path (#107 by @jfhair) #

New GlassTintBlend enum on LiquidGlassSettings to explicitly control how glassColor blends with the refracted backdrop, instead of relying entirely on the chroma heuristic.

  • GlassTintBlend.auto โ€” the default. Existing chroma gate, byte-for-byte unchanged behavior.
  • GlassTintBlend.luminosity โ€” always preserve backdrop luminosity. For near-neutral tints that need to keep the glassy look rather than flattening to a film.
  • GlassTintBlend.flat โ€” always impose the tint's brightness. For dimming layers, backing scrims, or deliberate frost-film surfaces.

Fully threaded through LiquidGlassSettings (copyWith, lerp, props), both the Premium and Standard shader paths, and preserved through AdaptiveGlass elevation rebuilds. Frosted fallback renders flat by construction and ignores the setting.

โœจ GlassScrollEdgeEffect.bottomFadeInset (#109 by @jfhair) #

New optional bottomFadeInset parameter (default 0.0) that lifts the bottom fade off the widget's true bottom edge by the specified logical pixels. Fixes cases where the scroll viewport extends below the visible area โ€” such as a bottom sheet whose content box overflows past the screen bottom โ€” causing the bottom fade to anchor off-screen and never appear.

No breaking changes. All new parameters are optional with safe defaults.


0.16.2 #

๐Ÿ› Bug Fix โ€” GlassMenu / GlassPopover rebuild on keyboard open/close #

GlassMenu and GlassPopover were rebuilding on every keyboard open/close event, even when closed. Caused by MediaQuery.of(context) in didChangeDependencies subscribing to viewInsets. Fixed by switching to scoped accessors (disableAnimationsOf, maybeSizeOf, textScalerOf). Regression tests added.

โœจ Content-luminance scroll-edge scrim (#106 by @jfhair) #

The continuous companion to the contentAwareBrightness discrete lever. Scroll-edge fades now track content luminance and dissolve toward a dark color as dark content scrolls under the bars โ€” matching the native App Store early-darkening behaviour.

  • GlassContentAwareScope.register() gains onLuminanceChanged (brightness callback is now optional). Per-rect mean luminance is delivered from the existing single capture; deliveries are gated on >0.005 movement.
  • GlassScrollEdgeEffect.contentAwareFade โ€” each edge band registers with the scope and lerps toward darkFadeColor as content darkens (luminanceDarkBelow / luminanceLightAbove thresholds, 280 ms ease-out). Inert without a scope.
  • GlassScaffold.contentAwareEdgeFade โ€” one flag to enable both bars, composing with contentAwareBrightness.
  • Latent wrap-order fix (0.16.0): GlassScaffold was wrapping the body in GlassContentAwareContent after the edge fade, so the fade overlays were inside the sampled region. With the adaptive scrim this is a feedback loop. Wrap order corrected + regression test added.

๐Ÿ› GlassModalSheet handle-drag fixes (#106) #

  • Handle drag no longer fights the inner scroll. _handleDragActive notifier is set on pointer-down (before the inner Scrollable can claim slop), disabling inner scroll for the gesture lifetime. Fixes a freeze when dragging the handle over a PlatformView.
  • dragIndicatorColor now actually reaches the drag indicator โ€” it was silently wired into _SheetLayout but never forwarded to _GlassDragIndicator.

๐Ÿ› GlassEffect โ€” defer capture when boundary is mid-paint (#106) #

toImageSync called during a dirty repaint boundary spammed [GlassEffect] toImageSync failed in debug and dropped the frame's capture. Guarded with debugNeedsPaint check (release-safe, same pattern as GlassScrollEdgeEffect).

No breaking changes. New parameters are all optional with safe defaults โ€” existing code compiles and behaves identically without changes.


0.16.1 #

๐ŸŽ iOS 26 Indicator Defaults โ€” Parity Calibration #

Three indicator defaults have been updated across GlassBottomBar and GlassSearchableBottomBar to better match the iOS 26 bottom-bar pill out of the box. No API changes โ€” all parameters remain fully configurable.

Changed defaults #

indicatorPinchStrength โ€” 1.0 โ†’ 0.4

The previous default of 1.0 applied the maximum concave lens / pinch effect during drag. iOS 26's actual pinch is more restrained โ€” 0.4 produces the characteristic "through a lens" look without over-distorting the edges.

To restore the previous behaviour:

GlassBottomBar(
  indicatorPinchStrength: 1.0,
  ...
)

indicatorExpansion โ€” EdgeInsets.all(8) โ†’ EdgeInsets.symmetric(horizontal: 12, vertical: 8)

The indicator pill in iOS 26 bottom bars is slightly wider than it is tall โ€” a subtle "landing pad" shape that reads as a rounded rectangle rather than a near-circle. The new default matches this proportion.

To restore the previous behaviour:

GlassBottomBar(
  indicatorExpansion: const EdgeInsets.all(8.0),
  ...
)

AnimatedGlassIndicator chromatic aberration โ€” 0.0 โ†’ 0.15

The indicator's internal _baseGlassSettings now sets chromaticAberration: 0.15. Real iOS 26 glass has a faint iridescent rainbow fringe at the rim. At 0.15 the effect is a whisper โ€” visible up close, subliminal during normal use.

To disable the aberration pass a full indicatorSettings override:

GlassBottomBar(
  indicatorSettings: LiquidGlassSettings(
    chromaticAberration: 0.0,
    // include other fields you need
  ),
  ...
)

Affected widgets #

  • GlassBottomBar โ€” indicatorPinchStrength and indicatorExpansion
  • GlassSearchableBottomBar โ€” indicatorPinchStrength and indicatorExpansion
  • All widgets using AnimatedGlassIndicator โ€” chromaticAberration baseline

GlassTabBar and GlassSegmentedControl retain their existing expansion defaults (EdgeInsets.all(8.0)) as their geometry is different from a bottom navigation bar.


0.16.0 #

๐ŸŽจ Content-Aware Light/Dark Adaptation #

Glass bars now automatically adapt their icon and label colors to match the content scrolling behind them โ€” light glyphs over dark content, dark glyphs over light content โ€” with a smooth cross-fade transition. This matches the iOS 26 behaviour where navigation chrome remains legible regardless of what is visible underneath.

Core engine contributed by @jfhair in PR #103.

New widgets #

  • GlassContentAwareScope โ€” wraps a screen and owns the sampling engine. Captures the content boundary at scroll rate (~5 fps), divides each registered control's rectangle into voting cells, and delivers per-control brightness verdicts via WCAG contrast ratios and dual-threshold hysteresis.
  • GlassContentAwareContent โ€” marks the sampled content region. Installs a RepaintBoundary that the scope captures. Controls must be outside this region (e.g. in Scaffold.bottomNavigationBar with extendBody: true).
  • GlassContentAwareBrightness โ€” per-control consumer that cross-fades between the light and dark GlassThemeVariant via GlassThemeVariant.lerp. Supports an external brightnessOverride (for PlatformView escape hatches), configurable grid dimensions, and per-control flip duration/curve overrides.

New parameters on existing widgets #

  • GlassBottomBar โ€” adaptiveBrightness, brightnessOverride, onBrightnessChanged. Set adaptiveBrightness: true to opt in.
  • GlassSearchableBottomBar โ€” same three parameters. Both bars automatically wrap in GlassContentAwareBrightness when enabled.
  • GlassScaffold โ€” contentAwareBrightness. When true, the scaffold automatically wraps the body in GlassContentAwareContent and the entire layout in GlassContentAwareScope. One flag, no manual widget wiring.

New API surface #

  • GlassThemeVariant.lerp(a, b, t) โ€” interpolates settings, glow colors, quality, and border radius between two theme variants. Used internally by the cross-fade but available to consumers building custom transitions.
  • GlassThemeSettings.lerp(a, b, t) โ€” interpolates all 9 glass setting fields (thickness, blur, glassColor, lightAngle, lightIntensity, etc.).
  • resolveBarLabelColor(context, brightness) โ€” shared utility for bars to resolve label color from CupertinoTheme given an overridden brightness.

Usage #

The recommended path โ€” one flag on GlassScaffold, one on the bar:

GlassScaffold(
  contentAwareBrightness: true,
  bottomBar: GlassBottomBar(
    adaptiveBrightness: true,
    onBrightnessChanged: (b) => /* flip your own icon colors */,
    tabs: [...],
    selectedIndex: _index,
    onTabSelected: (i) => setState(() => _index = i),
  ),
  body: CustomScrollView(...),
)

For custom layouts without GlassScaffold, use the standalone widgets directly:

GlassContentAwareScope(
  child: Scaffold(
    extendBody: true,
    body: GlassContentAwareContent(
      child: ListView(...),
    ),
    bottomNavigationBar: GlassBottomBar(
      adaptiveBrightness: true,
      ...
    ),
  ),
)

Bug fix #

  • GlassThemeVariant.== / hashCode missing borderRadius โ€” fixed a pre-existing bug where two variants differing only in borderRadius compared equal. This caused stale radius during content-aware cross-fades where intermediate lerped variants would not trigger rebuilds.

Polish #

  • _sample() error reporting โ€” the bare catch (_) in the sampling pipeline now reports to FlutterError inside an assert closure. Errors are still suppressed in release builds but surface in debug mode so programming mistakes are visible.
  • Removed redundant setState โ€” _GlassContentAwareBrightnessState._setBrightness no longer calls setState since AnimatedBuilder already listens to the animation controller and rebuilds on every tick.

Example app #

  • Content-Aware Brightness demo (new) โ€” dedicated showcase with alternating light and dark content bands that force visible bar flips during scrolling. Available in the Examples tab.

0.15.7 #

๐ŸŒ™ Adaptive Brightness Fix #

Fixed a bug in LightweightLiquidGlass where the shader's internal brightness estimation (backdropLuma) was incorrectly reading from the OS-level MediaQuery.platformBrightnessOf(context) rather than the inherited Flutter Theme.of(context).brightness.

This ensures that glass surfaces now correctly switch to Light Mode parameters (such as the legibility veil) when the app itself overrides the theme to Light Mode, even if the user's physical device remains in Dark Mode.


0.15.6 #

๐ŸŒซ๏ธ Scroll Edge Fade โ€” Perceptual Gradient Curve #

Replaced the 2-stop linear alpha gradient in GlassScrollEdgeEffect with a multi-stop eased curve that matches the perceptual dissolve of iOS 26.

  • 5-stop gradient profiles for both soft and hard styles โ€” eliminates the "denser in the centre" banding and the visible seam at the fade boundary.
  • hard style reworked: steeper hold โ†’ sharper drop curve instead of just compressing the soft profile. Height multiplier relaxed from 0.33ร— to 0.5ร—.
  • Example app: Nav Patterns demo now fully brightness-aware (adaptive text colours, GlassStatusBarStyle.auto, adaptive solid-bar colour).

No API changes. No breaking changes.


0.15.5 #

โœจ Whiten Strength โ€” Light-Mode Legibility Veil #

Opt-in whitening ("legibility veil") lifts glass toward white for legibility over busy light backgrounds โ€” modelling iOS 26's light-mode glass.

  • LiquidGlassSettings.whitenStrength (0.0โ€“1.0, default 0.0): lifts the finished glass toward white as the last step of the render. A single control-wide value with no spatial seams or halo artifacts.

  • LiquidGlassSettings.whitenGated (default true): when gated, the lift scales by per-pixel luminance so bright content lifts to white while dark content (text, icons) stays crisp. Ungated applies the lift uniformly โ€” useful for dark-mode frost effects.

  • Consistent across all three quality tiers from one knob: Premium (fragment shader), Standard (tint lerp), and Minimal (frosted fallback) all render the same whitenStrength value consistently.

  • GlassSearchableBottomBar whiten-at-bottom: when a scrollController is provided, the bar animates its whitening toward full white as the page nears the scroll bottom โ€” the iOS light-mode behaviour where content crowding under a bar gets the strongest legibility lift.

  • Example app: Buttons & Shadows demo now includes a real-time whiten slider with side-by-side comparison cards and scroll-to-bottom boost preview.

Contributed by @jfhair in PR #100.

๐ŸŽฌ Scale-with-Morph โ€” Cohesive Overlay Content Reveal #

Menu items and popover content now scale in alongside the liquid morph animation instead of popping in at the tail. The glass container and its content feel like a single continuous motion.

  • Items enter the tree at 30% morph progress (down from 94%) and scale from 0.5ร— to 1.0ร— via an easeOut curve alongside opacity. Text and icons visually grow with the expanding glass container.
  • Applied to both GlassMenu and GlassPopover for consistent overlay behaviour across the widget family. Content scales in on open and scales back out on close โ€” the morph animation is symmetrical in both directions.
  • No new API surface. No breaking changes. Purely visual polish.

GlassMenu scale-with-morph contributed by @F1orian in PR #97.


0.15.4 #

โšก Render Pipeline Performance Pass โ€” FPS & Battery #

Internal optimizations targeting the render object paint() hot path and background capture lifecycle. Zero API changes, zero visual changes. All 2,050 tests passing.

๐ŸŽฏ Bottom Bar Lateral Sway #

  • Subtle lateral sway on fast pill drags โ€” GlassBottomBar and GlassSearchableBottomBar now respond with a near-subliminal horizontal shift when the interactive pill is flicked quickly, matching iOS 26 bottom bar physics. Velocity-gated (slow drags produce no movement) and spring-animated back to center on release.

GPU allocation pressure (FPS) #

  • Cached ImageFilter in _RenderLightweightGlass โ€” the composed blur+saturation ImageFilter (and its 20-element ColorFilter.matrix list) was previously reconstructed on every paint() frame for every visible glass widget. With 5 glass cards at 60 fps, that's ~300 list allocations/second hitting the GC. The filter is now cached on the render object and only rebuilt when blur or saturation changes.
  • Cached ImageFilter in _RenderInteractiveIndicator โ€” same optimization for the interactive indicator shader path (GlassEffect). The brightness+blur composed filter is now cached and invalidated only when blurSigma changes.
  • Cached ImageFilter.blur in RenderLiquidGlassLayer โ€” the premium glass layer's BackdropFilterLayer blur filter was previously recreated on every paintLiquidGlass() call, even when the blur sigma hadn't changed. During jelly/morph animations (60 fps), this creates an ImageFilter.blur object per frame per premium glass layer. Now cached and reused.
  • Cached _shapesWithGeometry list โ€” the premium glass paint() method previously allocated a new list on every frame to collect active geometry shapes. It now reuses a cleared instance list, eliminating another per-frame allocation on the hot path.
  • Cached light angle trigonometry โ€” both _RenderLightweightGlass and _RenderInteractiveIndicator now cache cos(lightAngle) and -sin(lightAngle) instead of recomputing them on every _updateShaderUniforms() call. Matches the caching pattern already used by the premium LiquidGlassRenderObject._cachedLightDir.
  • Conditional alwaysNeedsCompositing โ€” both _RenderLightweightGlass and _RenderInteractiveIndicator previously returned true unconditionally, forcing the framework to create a compositing layer even in the fallback path (null shader or zero blur). Now returns true only when a BackdropFilterLayer will actually be pushed. Reduces layer tree depth on low-end devices.

Battery life #

  • Background Ticker auto-suspend โ€” LightweightLiquidGlass runs a Ticker to capture the background behind each glass widget via toImageSync. Previously, this ticker ran at 60 fps permanently โ€” even when the background hadn't changed for minutes (e.g. a static bottom bar over a static page). After 3 consecutive no-change frames (~50 ms), the ticker now self-suspends. It restarts automatically when didUpdateWidget detects a configuration change. For a bottom bar with 5 tab pills, this eliminates 300 unnecessary method calls/second while idle.

GC pressure on frame timing path #

  • Optimised GlassQualityAdapter._percentile โ€” the Phase 3 runtime percentile computation previously called _window.toList() + List.from(data)..sort() every 120 frames, creating two heap-allocated lists on the frame timing callback path. Now reuses a pre-allocated sort buffer (List<int>.filled) that is overwritten in-place. Eliminates GC pressure from the very code path that monitors for GC-induced frame drops.

Community contribution #

  • Collapsed search indicator stretch feedback โ€” the collapsed indicator in GlassSearchableBottomBar now wraps in LiquidStretch, giving it the same press-scale physics as the normal tab indicator. Contributed by @g3mf0r in PR #96.

Widget tree rebuild reduction #

  • Shared spring listener in GlassSearchableBottomBar โ€” the three morph- animation AnimationControllers (tab width, search left, search width) each had their own anonymous addListener(() => setState(() {})) callback. During a pill morph all three springs tick every frame, producing three independent setState calls per frame. Now all three share a single named _onSpringTick callback โ€” Flutter coalesces the dirty mark so only one rebuild fires. Also fixes a subtle listener leak: anonymous lambdas can't be matched by removeListener, but named methods can.

Consistency fix #

  • GlassEffect.preWarm() dummy image โ€” changed from async await picture.toImage(1, 1) to synchronous picture.toImageSync(1, 1), matching the LightweightLiquidGlass pattern. Eliminates a 1-frame async initialization delay for a 1ร—1 transparent texture. Added picture.dispose() to prevent a minor leak.

0.15.3 #

platformViewBackdrop โ€” Premium Glass Over iOS PlatformViews #

New opt-in flag that lets glass bars render correctly over native iOS views (Google Maps, Apple Maps, WebView, MapLibre, video players) while keeping the premium indicator animations alive. Previously, developers had to downgrade to GlassQuality.standard on iOS โ€” now they can stay on premium.

Contributed by @jfhair in PR #94.

New parameter: platformViewBackdrop #

  • GlassBottomBar โ€” new platformViewBackdrop parameter. When true, the bar background renders via live BackdropFilter (the premium shader's toImageSync cannot capture a UIKitView), while the premium indicator refracts the bar's own icon layer instead of the un-capturable backdrop.
  • GlassSearchableBottomBar โ€” same parameter, applied to both the tab indicator and the search pill. The collapsed search button automatically falls back to GlassQuality.standard over a PlatformView.
  • AdaptiveGlass / AdaptiveGlass.grouped() โ€” new platformViewBackdrop parameter that forces the BackdropFilter rendering path even at GlassQuality.premium.
  • AdaptiveLiquidGlassLayer โ€” new platformViewBackdrop parameter that skips the LiquidGlassBlendGroup wrapper when rendering over a PlatformView.

Usage #

GlassBottomBar(
  quality: GlassQuality.premium,
  platformViewBackdrop: Platform.isIOS, // โ† one line fix
  tabs: myTabs,
  selectedIndex: _index,
  onTabSelected: (i) => setState(() => _index = i),
)

Example app #

  • Google Maps demo โ€” updated to use platformViewBackdrop: Platform.isIOS instead of the old Platform.isIOS ? GlassQuality.standard : GlassQuality.premium workaround. The bar now stays at GlassQuality.premium on all platforms.

0.15.2 #

GPU SDF Shadows โ€” Light Mode Performance & Metaball Support #

Replaced the previous inverse-clip shadow system with GPU SDF shadows that render directly from the merged geometry matte. This is both a performance improvement and a visual correctness fix โ€” shadows now correctly follow the liquid metaball shape during morph animations, which was impossible with the old per-element ClipPath approach.

Light mode shadows #

  • GPU SDF shadow pass โ€” RenderLiquidGlassLayer.paintGlass now includes a Pass 0 that draws each BoxShadow using the geometry matte image. The matte is drawn with ImageFilter.blur for the shadow spread, tinted via ColorFilter.mode, and then the interior is punched out with BlendMode.dstOut so the glass backdrop filter never samples its own shadow. This replaces the widget-level _InversePillClipper / _InverseOvalClipper ClipPath approach.
  • Shadows plumbed through the full stack โ€” LiquidGlassSettings.effectiveShadow is now resolved at the AdaptiveGlass and AdaptiveLiquidGlassLayer level and passed through LiquidGlass.withOwnLayer โ†’ LiquidGlassLayer โ†’ _RawShapes โ†’ RenderLiquidGlassLayer. Dark mode and flat-edge shapes (borderRadius: 0) correctly receive an empty shadow list.
  • Removed old clip-based shadow system โ€” deleted _InversePillClipper, _InverseOvalClipper from GlassBottomBar, and the equivalent clippers from GlassSearchableBottomBar. Removed _wrapWithLightModeShadow from AdaptiveGlass. This eliminates ~140 lines of widget-level shadow plumbing and the associated ClipPath compositing cost.
  • Geometry image exposed to subclasses โ€” LiquidGlassRenderObject now exposes geometryImage and geometryLocalBounds as @protected getters so RenderLiquidGlassLayer can access the matte for shadow rendering.

GlassMenu overlay rendering #

  • AdaptiveLiquidGlassLayer in menu overlay โ€” the morph overlay now uses AdaptiveLiquidGlassLayer instead of raw LiquidGlassLayer, ensuring correct quality resolution and backdrop blur when the menu is rendered in premium mode.

Example app #

  • Buttons & Shadows demo โ€” locked to Light Mode (Brightness.light) to serve as the definitive showcase for GPU SDF shadows. Moved from the Demos tab to the Examples tab with updated card gradient and subtitle.
  • Apple Messages demo โ€” menu trigger buttons now use useOwnLayer: true to correctly enter the GPU SDF shadow pipeline. Menu glass settings are brightness-aware (Colors.white12 dark / Color(0x99FFFFFF) light).
  • Surfaces demo โ€” _buildDemoBackground() now accepts BuildContext and returns a light frosted gradient in Light Mode instead of the hardcoded dark navy gradient that caused unreadable text.

Bug fixes #

  • Fixed unused import snap_rect_to_pixels.dart in liquid_glass_layer.dart.
  • Fixed unused isDark variable in glass_menu_internal.dart.
  • Removed unnecessary package:flutter/material.dart import from adaptive_liquid_glass_layer.dart (already provided by cupertino.dart).
  • Fixed duplicate doc comment in glass_bottom_bar.dart.
  • Fixed release-mode crash in GlassScrollEdgeEffect._captureBackground โ€” RenderRepaintBoundary.toImage() throws synchronously on layer! when called before the first paint completes (layer not yet composited). The debugNeedsPaint guard only runs inside assert() and is stripped in release builds. Fixed by deferring the initial capture to a post-frame callback and wrapping toImage() in try/catch as a safety net. Falls back to the solid-colour gradient overlay on failure. (reported by @RuslanTsitser via #93)
  • Updated GlassBottomBar golden test reference image.

0.15.1 #

Full Light & Dark Mode โ€” Complete Adaptive UI Kit #

liquid_glass_widgets is now a fully adaptive iOS 26 UI kit. Every widget โ€” buttons, menus, search bars, sliders, switches, sheets, toasts, chips, form fields, and all surface bars โ€” automatically resolves the correct glass color, rim lighting, shadow, text, and icon values for both Light and Dark mode. No manual configuration required.

This release closes the last remaining light-mode rendering gaps and ships a reference implementation in the Apple Messages demo that matches the iOS 26 Messages app in both modes. Run the example app to toggle light and dark mode for all demo pages.

New features #

  • Configurable glass shadow โ€” LiquidGlassSettings.shadowElevation scales the light-mode drop shadow (0.0 = off, 1.0 = default, 2.0 = double). LiquidGlassSettings.shadow accepts a custom List<BoxShadow> for full control. Both flow through globalSettings (theme-level) and per-widget settings:.
  • GlassShadow constants โ€” centralised shadow values (GlassShadow.elevation, .contact, .defaults, .scaled(double)) exported for custom widget authors.
  • GlassButtonGroup.icons() โ€” introduced a lightweight group constructor for iOS 26 style segmented icon toolbars. Uses a single GlassButton parent to drive cohesive group-level stretch/glow interaction, while children are rendered as zero-overhead stateless tap targets.
  • GlassMenuController โ€” imperative controller for GlassMenu with open(), close(), and isOpen. Drives the menu programmatically instead of (or in addition to) tapping the trigger โ€” useful for gesture-arena-driven menus. (contributed by @F1orian)
  • GlassMenu.showDismissBarrier โ€” when false, suppresses the full-screen tap-to-dismiss barrier so an external gesture owner can keep receiving pointer events while the menu is open. Defaults to true. (contributed by @F1orian)
  • GlassMenu.morphFromZero โ€” when true, the menu body lerps from a zero-size point at the trigger center instead of from the trigger's own dimensions, suppressing the spawn blob (Blob A). For invisible or zero-sized triggers. (contributed by @F1orian)
  • GlassMenuController.setFollowOffset(Offset) โ€” nudges the open menu to track a moving anchor in real-time. The offset is added to the captured trigger position each frame and reset on the next open(). (contributed by @F1orian)

Light & dark mode improvements #

  • Complete adaptive rendering โ€” all widgets now resolve glass color, rim borders, shadows, text, and icon colors from CupertinoTheme.brightnessOf(context). Switching between light and dark mode requires zero widget-level changes.
  • Light-mode rim borders โ€” removed the heavy dark rim border on glass surfaces in light mode. Dark mode rim lighting remains fully intact and unchanged.
  • Light-mode drop shadows โ€” added inverse-clipped drop shadows to glass surfaces in light mode (cards, standalone buttons, bottom bars). Shadows render outside the glass boundary so the backdrop filter doesn't blur them. Note: morphing elements like the search pill do not have shadows to prevent animation artifacts.
  • Standard glass white frost โ€” standard quality glass in light mode now correctly renders as clean frosted white instead of muddy grey.
  • Dynamic color resolution โ€” improved internal text and icon styling to accurately resolve CupertinoColors against the active theme brightness.
  • GlassSearchBar and GlassTextField default colors now brightness-aware โ€” default text, icon, and glow colors now resolve dynamically against CupertinoTheme.brightnessOf(context) instead of being hardcoded to white. This ensures correct contrast in both light and dark mode.

Bug fixes #

  • Fixed missing } else { in lightweight_glass.frag that caused PATH B (standard widgets) to run inside PATH A.
  • Fixed GlassSheet sharp corners on macOS by decoupling top and bottom border radius.
  • Fixed GlassMenu selection pill alignment and hit-test accuracy when system text scaler is active.
  • Fixed GlassChip resolving with invisible white text and icons in light mode.
  • Fixed Apple Podcasts demo incorrectly forcing dark-mode glass colors in light mode.
  • Fixed GlassFormField label (Colors.white) and helper text (Color(0x99FFFFFF)) being invisible in light mode โ€” now resolves from CupertinoColors.label / .secondaryLabel
  • Fixed GlassPicker value text (Colors.white) and chevron icon (Colors.white70) being invisible in light mode โ€” now resolves from CupertinoColors.label / .secondaryLabel.
  • Fixed GlassPasswordField lock and eye toggle icons (Colors.white70) being invisible in light mode โ€” now resolves from CupertinoColors.secondaryLabel.
  • Fixed GlassActionSheet forcing dark card background (Colors.black @ 0.65), dividers, and pressed highlights regardless of brightness โ€” now resolves to light frosted glass in light mode.
  • Fixed GlassToast forcing dark pill (Colors.black @ 0.7) and white text regardless of brightness โ€” background and text now resolve from brightness for correct contrast in both modes.
  • Removed unnecessary import 'package:flutter/material.dart' from GlassFormField, GlassPicker, GlassPasswordField, and GlassToast.
  • Fixed GlassMenu morph size going negative during spring undershoot on small triggers โ€” currentWidth/currentHeight now clamped to >= 0 to prevent debug BoxConstraints assertion. (contributed by @F1orian)
  • Fixed GlassMenu sub-pixel Impeller crash โ€” body container is replaced with SizedBox.shrink() when dimensions fall below 1.0 logical pixel, preventing 0-area matte rasterisation. (contributed by @F1orian)
  • Fixed GlassIconButton bypassing theme quality chain โ€” quality now passes null through to GlassButton.custom() so GlassThemeHelpers.resolveQuality can resolve from ancestor layers, theme, then widget default. (contributed by @F1orian)

โš ๏ธ Breaking โ€” Removed frosted well overlay from GlassTextField #

GlassTextField no longer applies a hidden internal darkening overlay ("frosted well") on top of the glass surface. Previously, a DecoratedBox with Colors.black.withValues(alpha: 0.12) plus a top-edge gradient was composited inside the glass shape to simulate an iOS 26 recessed input tray. This has been removed entirely.

Why: The frosted well fought against user-specified glassColor, making text fields appear darker/muddier than buttons with identical settings. It also caused visible nested border artifacts when the overlay's BorderRadius.circular() didn't match the glass surface's LiquidRoundedSuperellipse shape โ€” especially with useOwnLayer: true and padding: EdgeInsets.zero.

Design philosophy: GlassTextField is a hero surface (search bars, compose bars, app bar inputs) โ€” not a form field nested inside glass cards. glassColor is now the single source of truth for appearance, consistent with every other glass widget.

Migration: If you relied on the frosted well for visual distinction inside a grouped layout, set a slightly darker glassColor on the text field explicitly. Most users will see cleaner, more predictable text fields with no action required.

Example app #

  • Input demos โ€” replaced GlassCard wrappers around form fields with flat CupertinoColors.systemFill containers. Glass text fields now render as standalone hero surfaces (useOwnLayer: true) inside flat-colored form sections, demonstrating the correct pattern. Glass-in-glass nesting is an anti-pattern.
  • Apple Messages demo โ€” fully adaptive light and dark mode implementation. Light mode uses the iOS system grouped background (#F2F2F7), brightness-aware white glass tint, and dynamic layer separation to enable shadows. Dark mode retains liquid blending via the shared AdaptiveLiquidGlassLayer. Press interactions use persistPressOnDrag: true and an elevated ambientBaseLight in light mode so the pressed state remains visible while holding and dragging off the button edge.

0.15.0 #

โš ๏ธ Breaking โ€” API Cleanup & Standardisation #

Removes Material-leaning widgets and thin wrappers that don't map to real iOS 26 components. Standardises the public API for v1.0 readiness. This is a breaking release โ€” users on ^0.14.0 will not auto-upgrade.

Deleted widgets #

Widget Migration
GlassPanel Replace with GlassCard(padding: EdgeInsets.all(24))
GlassWizard / GlassWizardStep No direct replacement โ€” use GlassStepper for sequential steps
GlassSideBar / GlassSideBarItem No direct replacement โ€” a proper UISplitViewController-style sidebar is planned for post-1.0
GlassSnackBar Replace with GlassToast (identical API โ€” GlassSnackBar was documented as an alias)

glassSettings โ†’ settings rename #

The glassSettings parameter on 8 widgets has been renamed to settings for consistency. The Glass prefix on the widget name already identifies the settings type โ€” glassSettings was redundant.

Affected widgets: GlassBottomBar, GlassSearchableBottomBar, GlassSegmentedControl, GlassButtonGroup, GlassMenu, GlassPopover, GlassToolbar, GlassPicker.

Migration: Find-and-replace glassSettings: โ†’ settings: in your code.

GlassSheet.show() API Cleanup #

Removed dead Material parameters from GlassSheet.show() that have no effect in the liquid glass rendering pipeline:

  • backgroundColor
  • elevation
  • shape
  • clipBehavior
  • constraints

Migration: Remove these parameters from your GlassSheet.show() calls. Use settings to configure the glass visual appearance, and margin / padding / borderRadius to control the layout and shape.

๐ŸŽจ Content Colour Audit โ€” Adaptive Light/Dark Mode #

All hardcoded Colors.white / Colors.black in widget content layers replaced with CupertinoColors adaptive equivalents. Widgets now render correctly in both light and dark mode without requiring a MaterialApp ancestor.

Affected widgets: GlassDialog, GlassActionSheet, all interactive and surface widgets.

GlassPage is now fully safe to use inside a pure CupertinoApp โ€” the Theme.of guard no longer throws when no Material ancestor is present.

๐Ÿ› Bug Fixes #

GlassSheet sharp corners (All Platforms) #

Fixed a bug where GlassSheet would render with completely sharp, square corners on all devices (including iOS and Android). The internal SafeArea wrapper was consuming the bottom safe area before the adaptive radius calculation could read it, causing it to incorrectly fallback to 0.0 everywhere. GlassSheet now decouples its top and bottom border radii, defaulting to topBorderRadius: 32.0, and smartly assigning bottomBorderRadius to 32.0 when floating (with margin) or 0.0 when docked.

โœจ New โ€” GlassButtonStyle.prominent #

A new button style matching iOS 26's .prominentGlass / .glassProminent configuration โ€” thicker, more opaque glass for primary call-to-action buttons.

GlassButton(
  style: GlassButtonStyle.prominent,
  icon: Icon(CupertinoIcons.plus),
  onTap: () {},
)

โœจ New โ€” GlassGroupedSection #

A convenience wrapper that groups GlassListTiles inside a GlassCard, automatically applying isLast: true to the final tile to suppress its bottom divider โ€” the most common source of bugs.

GlassGroupedSection(
  header: Text('Network'),
  children: [
    GlassListTile(title: Text('Wi-Fi')),
    GlassListTile(title: Text('Bluetooth')),
    GlassListTile(title: Text('VPN')), // isLast applied automatically
  ],
)

โœจ New โ€” GlassPageControl #

iOS UIPageControl equivalent โ€” dot indicators for paged content (carousels, onboarding, gallery pages). Features animated capsule-shaped active dot with glass treatment and optional tap-to-navigate.

GlassPageControl(
  count: 5,
  currentPage: _currentPage,
  onPageChanged: (page) => _pageController.animateToPage(page,
    duration: Duration(milliseconds: 300),
    curve: Curves.easeInOut,
  ),
)

๐Ÿ“– README โ€” Glass vs Content design guide #

Added a design philosophy section explaining iOS 26's glass hierarchy: glass for navigation chrome and controls, opaque for content areas. Repositioned GlassContainer as an advanced primitive in the widget catalogue.

What stays #

  • GlassBackdropScope โ€” deprecated no-op stays until 1.0.0 as previously committed in the 0.14.0 changelog. Zero cost to keep.
  • All other widgets โ€” no changes.

๐Ÿ› Fix โ€” GlassMenu auto-scrolling with large system text #

When the user increased system text size, GlassMenu would become scrollable even without menuHeight being set. The height calculation now accounts for textScaler so the menu sizes correctly at any accessibility text scale.

๐Ÿ› Fix โ€” GlassMenu selection pill misalignment at large text scale #

The sliding selection highlight and hit-test math used the nominal 44px item height for positioning, while the actual GlassMenuItem widgets grew taller via ConstrainedBox when system text was scaled up. This caused the pill to drift out of alignment and sometimes produce a "double highlight" effect. All layout math (_getItemOffset, _calculateIndexFromPosition, _isScrollable) now uses the TextScaler-aware height calculation.

๐Ÿ› Fix โ€” GlassMenuItem / GlassMenuDivider / GlassMenuLabel Light Mode #

These widgets previously hardcoded Colors.white for text, icons, and divider colours, making them invisible on light backgrounds. They now inherit from CupertinoTheme.of(context):

Widget Colour Source
GlassMenuItem theme.textTheme.textStyle.color โ†’ CupertinoColors.label
GlassMenuDivider theme.textTheme.tabLabelTextStyle.color at 15 % opacity
GlassMenuLabel theme.textTheme.tabLabelTextStyle.color at 45 % opacity

Custom iconColor, titleStyle, and color parameters still take priority over the theme default โ€” zero breaking changes.

๐Ÿงน Core โ€” CupertinoApp compatibility #

Removed all Theme.of(context) dependencies from the core library, replacing them with CupertinoTheme and defaultTargetPlatform. This ensures glass widgets adapt correctly to dark mode and background colours in pure CupertinoApp structures without falling back to Material defaults.

๐Ÿงน Example app โ€” CupertinoApp migration #

All 16 standalone demos and the main showcase app have been migrated from MaterialApp to CupertinoApp. This is the correct root widget for an iOS 26 glass library โ€” it provides CupertinoTheme to the entire widget tree, which glass widgets depend on for colour resolution.

A Theme(data: ThemeData.dark(...)) builder is injected below CupertinoApp so that any Scaffold widgets in demo pages continue to receive proper Material theming (background colour, text defaults).

๐Ÿงน Example app โ€” GlassMenu demo controls #

Added text-scale slider (1.0ร—โ€“3.0ร—) and light/dark theme toggle to the menu demo. Also added GlassMenuLabel, GlassMenuDivider, and a destructive GlassMenuItem to the menu items so their theme colour inheritance can be visually verified.


0.14.2 #

๐Ÿงน Material Artifact Purge #

Replaced internal Material primitives with Cupertino-native equivalents for better iOS 26 fidelity. No public API changes.

  • Tap feedback: InkWell โ†’ GestureDetector + iOS-style opacity highlight in GlassListTile and GlassActionSheet.
  • Icons: Icons.chevron_right, Icons.add, Icons.remove, Icons.info_outline โ†’ CupertinoIcons equivalents in GlassListTile and GlassStepper.
  • Colours: Hardcoded Colors.green / Colors.red โ†’ CupertinoColors.systemGreen / CupertinoColors.destructiveRed across GlassSwitch, GlassBadge, GlassWizard, GlassDialog, GlassMenuItem, and GlassFormField.

0.14.1 #

๐Ÿ› Fixed โ€” GlassScaffold black background #

GlassScaffold with no background: widget was forcing the inner Scaffold to Colors.transparent, rendering black instead of Theme.scaffoldBackgroundColor. Added a backgroundColor: Color? parameter; the scaffold now defaults to the theme colour when no explicit background is provided.

โœจ Improved โ€” Cancel icon size & customisation #

The dismiss ร— button in GlassSearchableBottomBar was hardcoded to size: 16, which felt undersized compared to iOS 26. Default bumped to 24. Both GlassSearchBar and GlassSearchBarConfig now expose cancelIconSize and cancelIcon so the icon size and glyph are fully overridable.

โœจ New โ€” GlassPopover #

A new overlay widget that presents custom content in a glass container with the same liquid morph animation as GlassMenu. Use it for tooltips, mini forms, preview cards, colour pickers, or any content that should appear in a glass popover โ€” without being limited to a list of menu items.

GlassPopover shares GlassMenu's liquid teardrop expansion, spring physics, auto-positioning, and screen-edge clamping. The contentBuilder receives a close callback so popover content can dismiss itself.

GlassPopover(
  trigger: GlassIconButton(
    icon: Icon(CupertinoIcons.info_circle),
    onPressed: null, // handled by GlassPopover
  ),
  contentBuilder: (context, close) => Padding(
    padding: EdgeInsets.all(16),
    child: Column(
      mainAxisSize: MainAxisSize.min,
      children: [
        Text('Profile Details',
          style: TextStyle(color: Colors.white, fontSize: 16)),
        SizedBox(height: 12),
        GlassButton(onTap: close, child: Text('Done')),
      ],
    ),
  ),
)

Key parameters:

  • contentBuilder โ€” builder for custom popover content, receives a close callback
  • trigger / triggerBuilder โ€” the widget that opens the popover on tap
  • popoverWidth / popoverHeight โ€” size control (height defaults to intrinsic)
  • alignment โ€” manual GlassMenuAlignment, or null for auto-detect
  • autoAdjustToScreen โ€” screen-edge clamping (on by default, with 12px padding)
  • barrierDismissible โ€” whether tapping outside closes the popover (default: true)
  • onOpen / onClose โ€” lifecycle callbacks
  • Full LiquidStretch and GlassGlow support

0.14.0 #

โš ๏ธ Breaking โ€” GlassAppBar redesigned as pure layout widget #

GlassAppBar has been fundamentally redesigned from a StatefulWidget with its own glass rendering to a lightweight StatelessWidget that serves purely as a layout container. This matches iOS 26's actual navigation bar pattern where the bar itself is transparent and glass effects belong on individual interactive elements.

Removed parameters:

  • settings โ€” glass surface settings (the bar no longer renders glass)
  • quality โ€” glass quality selector
  • useOwnLayer โ€” layer isolation control
  • scrollController โ€” scroll-driven glass transition
  • scrollEdgeThreshold โ€” scroll fade threshold

New parameter:

  • buttonSettings: LiquidGlassSettings? โ€” provides default glass settings to all descendant GlassButton widgets via DefaultButtonSettings (a new InheritedWidget). Individual buttons can still override with their own settings.

Migration:

// Before (0.13.x) โ€” scroll-driven glass bar:
GlassAppBar(
  title: Text('Title'),
  scrollController: _ctrl,
  scrollEdgeThreshold: 50.0,
  settings: LiquidGlassSettings(blur: 15),
  quality: GlassQuality.premium,
)

// After (0.14.0) โ€” transparent layout bar:
GlassAppBar(
  title: Text('Title'),
  buttonSettings: LiquidGlassSettings(
    glassColor: Color(0x33FFFFFF),
    thickness: 20,
  ),
)
// Use GlassScaffold.header + headerScrollController for fading headers.

Apps that relied on GlassAppBar.scrollController for scroll-driven glass should migrate to GlassScaffold with its header / headerScrollController parameters (see below).

โœจ New โ€” GlassScaffold โ€” one-widget page layout #

A brand new scaffold that replaces the manual assembly of GlassPage + Scaffold + GlassScrollEdgeEffect + Stack with a single widget. Handles z-ordering (bars always above body), edge fading, safe-area padding, and glass layer setup automatically.

header + headerScrollController + headerFadeDistance #

A fixed header widget positioned below the status bar that fades out as the user scrolls โ€” matching the iOS 26 large-title pattern used by Apple Music ("Listen Now"), Apple Podcasts, and similar apps.

GlassScaffold(
  header: Text('Listen Now', style: largeTitle),
  headerScrollController: _scrollController,
  headerFadeDistance: 60.0,
  body: CustomScrollView(
    controller: _scrollController,
    slivers: [...],
  ),
)

The header is wrapped in IgnorePointer only when fully faded (opacity == 0), so tappable elements remain interactive at full visibility.

bodyOverlays #

Optional list of widgets rendered between the body and the navigation bars in z-order. Designed for floating elements like Apple Music's play-bar pill that need to render above scrollable content but below the bottom bar.

GlassScaffold(
  bodyOverlays: [
    AnimatedPositioned(
      bottom: isMiniMode ? miniBottom : aboveBarBottom,
      left: 20, right: 20,
      child: PlayBarPill(),
    ),
  ],
  body: scrollContent,
)

AnnotatedRegion status bar fix #

GlassScaffold now wraps the internal Scaffold in an AnnotatedRegion<SystemUiOverlayStyle> so that the statusBarStyle setting correctly overrides CupertinoPageRoute's own region. Previously, pages without a GlassAppBar could lose their status bar icon styling when pushed via CupertinoPageRoute.

Improved isolation strategy #

GlassScaffold now wraps app bar and bottom bar in GlassIsolationScope(isolated: false, defaultQuality: GlassQuality.premium) instead of the previous GlassIsolationScope(isolated: true). This means bar buttons join the page-level glass blend group (shared backdrop capture) rather than creating their own layers โ€” saving GPU cost. Stack paint order already guarantees bars render on top of body content.

Stable ValueKeys on stack children #

All conditional children (header, app bar, bottom bar) now have explicit ValueKeys so Flutter tracks them by identity rather than position. This prevents unmount/remount cycles (and loss of animation state) when toggling the header on and off.

See example/lib/demos/nav_bar_patterns_demo.dart for complete GlassScaffold usage patterns.

โœจ New โ€” GlassIsolationScope.defaultQuality #

GlassIsolationScope gains a defaultQuality: GlassQuality? parameter that provides a quality hint for descendants without requiring explicit quality: on each widget. This is separate from isolated โ€” a scope can be de-isolated (for grouping) while still providing a premium quality default.

GlassThemeHelpers.resolveQuality now checks scopeDefault and skips inherited page-level quality when a scope provides a defaultQuality. This fixes a regression where bar buttons inside GlassScaffold would inherit the page's standard quality instead of getting the bar's intended premium.

โœจ New โ€” DefaultButtonSettings InheritedWidget #

A new InheritedWidget in glass_app_bar.dart that provides default LiquidGlassSettings to descendant GlassButton widgets. GlassButton now checks DefaultButtonSettings.of(context) as a fallback between explicit settings and inherited layer settings.

โœจ New โ€” AdaptiveGlass interactive auto-promotion #

AdaptiveGlass now automatically promotes interactive elements (isInteractive: true) to their own compositing layer in premium mode. This gives buttons the prominent independent refraction that matches iOS 26's button design, rather than softly blending them into the page blend group. The own-layer wrapper de-isolates children via GlassIsolationScope(isolated: false) so nested glass (e.g. tab items inside a bottom bar) groups correctly with the button's layer.

๐Ÿ› Fix โ€” GlassBottomBar extra button z-order #

Fixed a compositing z-order issue where the jelly tab indicator would incorrectly render behind the extra trailing button. The internal layout has been restructured from a Row to a Stack with explicit paint ordering โ€” the extra button is painted first (bottom of z-order), and the tab indicator is painted last (top). This matches the existing pattern in GlassSearchableBottomBar.

๐Ÿ› Fix โ€” GlassSearchableBottomBar pill z-order #

Fixed the paint order of the search pill and tab pill in GlassSearchableBottomBar. The search pill is now painted first (bottom of stack) and the tab pill last (top), so the glass indicator correctly renders on top when it overlaps the search pill during tab transitions.

๐Ÿ› Fix โ€” mounted guards in gesture handlers #

Added if (mounted) guards before every setState() call in TabDragGestureMixin, TabIndicator, and SearchableTabIndicator. Pointer events (onPointerDown, onPointerUp, onPointerCancel) and drag gesture callbacks can fire after the widget has been unmounted during rapid tab switching or page transitions, causing setState() called on disposed widget exceptions in debug mode. These guards are zero-cost safety nets.

๐Ÿ”„ Changed โ€” Nav bar patterns demo #

Removed three GlassAppBar demo patterns that used the now-deleted scroll-driven glass API:

  • Scroll-Driven Glass (transparent โ†’ glass on scroll)
  • Static Glass (always-on glass surface)
  • Large Title Glass on Scroll (combined collapsing + glass materialisation)

Added a new "Fade Header (No App Bar)" pattern demonstrating the GlassScaffold.header fade approach (Apple Music / Podcasts style).

๐Ÿ”„ Changed โ€” Example app restructured #

Showcase app uses GlassScaffold #

The main showcase app (main.dart) now uses GlassScaffold instead of manually composing GlassPage + Scaffold + bottomNavigationBar. Added a fourth tab ("Examples") and redesigned the Widgets tab with a 2-column card grid.

Apple demos migrated to GlassScaffold #

All four Apple demo apps have been migrated from manual Scaffold + Stack + GlassScrollEdgeEffect composition to GlassScaffold:

  • Apple Messages โ€” GlassScaffold handles positioning, z-ordering, and edge fading automatically. 8 lines changed.
  • Apple Music โ€” uses GlassScaffold.header for the "Listen Now" fade header, bodyOverlays for the floating play pill. Major simplification (294 lines changed).
  • Apple News โ€” migrated to GlassScaffold. Removed unnecessary nested Stack. 57 lines changed.
  • Apple Podcasts โ€” uses GlassScaffold.bodyOverlays for the mini-player pill. 372 lines changed.

Category pages re-indented #

All 6 showcase category pages (containers, feedback, input, interactive, overlays, surfaces) have been re-indented for consistency. No functional changes.

๐Ÿ› Fix โ€” Impeller backdrop visual corruption #

Fixed a critical rendering issue where GlassAppBar buttons would lose their glass background (rendering as invisible) when multiple LiquidGlassLayers shared a single root BackdropGroup (via GlassBackdropScope). On Impeller, sharing a BackdropKey across multiple RepaintBoundary layers caused the engine to bind stale or incorrect backdrop textures.

Root cause: The shared BackdropGroup architecture assumed that all glass surfaces could safely share a single GPU backdrop capture. On Impeller's tile-based renderer, this created a race between layers trying to use the same BackdropKey, causing visual corruption (backgrounds vanishing, ghost artifacts during route transitions).

Fix: Each LiquidGlassLayer now creates its own isolated BackdropGroup โ†’ RepaintBoundary subtree. This ensures every glass surface captures only its own clipped bounding box โ€” a ~30ร— reduction in GPU bandwidth compared to the previous full-screen capture, and eliminates cross-layer texture conflicts entirely.

โš ๏ธ Deprecated โ€” GlassBackdropScope #

GlassBackdropScope is now a no-op and can be safely removed from your widget tree. Each glass layer manages its own backdrop isolation automatically. GlassPage continues to work exactly as before โ€” no migration needed for apps using GlassPage.

Apps using GlassBackdropScope directly will see a deprecation warning but will compile and run without issues. Remove the widget at your convenience; it will be deleted in 1.0.0.

๐Ÿ”„ Changed โ€” GlassInteractionSettings is now the canonical interaction API #

GlassInteractionSettings (introduced in 0.13.0) is now the recommended way to configure all interaction physics app-wide. Pass it via GlassThemeData.interaction inside LiquidGlassWidgets.wrap():

runApp(LiquidGlassWidgets.wrap(
  child: const MyApp(),
  theme: GlassThemeData(
    interaction: GlassInteractionSettings(
      stretch: 0.2,              // subtler drag-following globally
      interactionScale: 1.03,    // less scale-up on press
      resistance: 0.01,          // drag damping factor
      anchorStretch: true,       // iOS 26 rubber-band from anchor
      anchorStretchSettings: AnchorStretchSettings(
        intensity: 0.8,
        squashFactor: 0.15,
      ),
    ),
  ),
));

Set stretch: 0.0 to disable drag-following app-wide while keeping press-scale. Individual widgets can still override any value via their own stretch: / interactionScale: parameters.


0.13.0 #

โœจ New โ€” Anchor Stretch, Ambient Light, and Interaction Physics #

AnchorStretchSettings โ€” fine-tuned stretch feel #

A new configuration class for LiquidStretch that controls how widgets deform when pressed and dragged. Widgets now stretch from their anchor point toward the drag direction โ€” matching iOS 26 button physics where the surface rubber-bands from its resting position rather than free-following the finger.

GlassButton(
  anchorStretchSettings: AnchorStretchSettings(
    intensity: 0.8,       // more stretchy
    squashFactor: 0.3,    // perpendicular compression
    translationDamping: 0.15, // center-shift toward finger
    bounciness: 0.2,      // elastic snap-back overshoot
  ),
)

All parameters have sensible defaults matching iOS 26 behaviour. Most developers won't need to change them.

GlassButton.ambientBaseLight โ€” surface luminosity #

A subtle white overlay (default 0.08) applied during press/drag interactions. Simulates iOS 26 surface luminosity โ€” when the directional glow tracks off-edge, the button still maintains a faint lit appearance rather than going completely dark.

GlassButton.persistPressOnDrag #

Controls whether the pressed visual state persists when the user's finger drags outside the button bounds. Defaults to true, matching iOS 26 behaviour where buttons stay visually pressed during drag-off and only release on pointer-up. Set to false for the traditional behaviour where leaving the hit-test area cancels the press.

GlassPage.settings โ€” page-level glass configuration #

GlassPage now accepts an optional settings: LiquidGlassSettings parameter and internally wraps its child in an AdaptiveLiquidGlassLayer. This means all glass widgets inside the page (GlassAppBar, GlassCard, GlassButton, etc.) automatically inherit the page's glass settings โ€” no need to set useOwnLayer: true or pass settings: to each widget individually.

GlassPage(
  settings: LiquidGlassSettings(
    glassColor: Color.fromRGBO(28, 28, 30, 0.8),
    thickness: 30,
    blur: 4,
  ),
  child: Scaffold(...),
)

When settings is null, the layer inherits from GlassTheme or uses defaults.

GlassScaffold โ€” comprehensive structural layer #

A new structural widget that coordinates interactions between GlassAppBar, GlassBottomBar, and the page body. It automatically handles edge-to-edge layout, scroll bleeding, and isolates glass rendering layers for maximum performance. This replaces the need to manually compose AdaptiveLiquidGlassLayer and GlassIsolationScope at the page root.

GlassInteractionSettings Theme #

Added a dedicated theme extension to globally configure interaction physics across all glass widgets โ€” stretch, press scale, drag resistance, anchor stretch, and anchor stretch fine-tuning. No more passing interaction parameters to every widget manually.

GlassThemeData(
  interaction: GlassInteractionSettings(
    stretch: 0.2,              // subtler drag-following globally
    interactionScale: 1.03,    // less scale-up on press
    resistance: 0.01,          // drag damping factor
    anchorStretch: true,       // iOS 26 rubber-band from anchor
    anchorStretchSettings: AnchorStretchSettings(
      intensity: 0.8,
      squashFactor: 0.15,
    ),
  ),
)

Set stretch: 0.0 to disable drag-following app-wide while keeping press-scale. Individual widgets can still override any value.

๐Ÿ› Fixes & Improvements #

Wide Button Stretch Distortion #

Fixed a rendering artifact where very wide GlassButtons would exhibit severe shape distortion at the extreme edges during horizontal stretch physics.ts.

GlassSearchBarConfig.cursorColor โ€” cursor follows Flutter theme by default #

Thanks to @jfhair for PR #71. ๐Ÿ™

GlassSearchableBottomBar's expanded search field now exposes a cursorColor knob via GlassSearchBarConfig, and the default behaviour aligns with Flutter convention โ€” the cursor follows the standard theme-resolution chain (Theme.of(context).textSelectionTheme.cursorColor โ†’ CupertinoTheme.primaryColor on iOS โ†’ Theme.of(context).colorScheme.primary) rather than being hard-coupled to textColor.

Apps that want the previous behaviour โ€” cursor matching textColor โ€” can opt in explicitly:

GlassSearchBarConfig(
  textColor: Colors.white,
  cursorColor: Colors.white,  // โ† previously implicit
)

โš  Breaking change. Apps that set a textColor and rely on the cursor implicitly matching it will see their cursor colour change to whatever their Theme.of(context).colorScheme.primary is (typically Colors.blue if untouched). Two-line migration above.

๐Ÿ› Fix โ€” Premium stretch edge clipping #

Premium-quality GlassButton could exhibit jagged rasterization edges during stretch deformation. The Impeller LiquidGlassLayer rasterizes at its native resolution, which doesn't perfectly align with the deformed shape boundary during stretch.

Fixed by wrapping the premium glass surface in a vector ClipPath at the shape boundary. This renders at screen resolution every frame while preserving full refraction, chromatic aberration, and 3D specular โ€” no quality downgrade needed.

๐Ÿ› Fix โ€” Mali GPU crash guard #

render_liquid_glass_geometry.dart now guards against zero and negative dimensions in both render() and renderAsync(). During jelly animations or rapid layout transitions (modal expansion, tab switching), matteBounds can momentarily collapse to zero dimensions โ€” producing an invalid GPU texture request that crashes Mali drivers.

The fix returns a minimal 1ร—1 fallback cache for zero-dimension frames. Additionally, matte.toImageSync() is wrapped in a try/catch to handle Mali driver failures gracefully โ€” returning a safe fallback instead of crashing the app. The next paint frame rebuilds the geometry with valid dimensions automatically.

๐Ÿ› Fix โ€” Searchable bottom bar collapsed shape #

The collapsed search button and tab indicator in GlassSearchableBottomBar used LiquidRoundedSuperellipse even when collapsed to a square. A superellipse with borderRadius: 32 on a 50ร—50 square has subtle flat segments between the arcs โ€” invisible at rest but clearly distorted during stretch deformation.

Fixed by switching to LiquidOval when constraints are square (within 2px tolerance), which renders a mathematically perfect circle that stretches uniformly. During the collapse animation (when the width is still wider than the height), the superellipse is used to avoid a squashed oval appearance.

๐ŸŽจ Visual โ€” iOS 26 thin glass defaults #

The three default theme variants have been standardised to match the thin, refractive glass aesthetic of iOS 26:

Property Dark Light Minimal
thickness 40 โ†’ 10 20 โ†’ 12 30 โ†’ 10
blur 5 โ†’ 4 6 โ†’ 5 12 โ†’ 8
lightIntensity 1.5 โ†’ 0.7 1.2 โ†’ 0.85 unchanged
lightAngle unset โ†’ 135ยฐ unset โ†’ 135ยฐ unchanged
chromaticAberration unset โ†’ 0.01 0.3 โ†’ 0.02 unchanged

โš  Visual change. Apps using GlassThemeVariant.dark, .light, or .minimal without explicit LiquidGlassSettings overrides will see thinner, subtler glass. This is intentional โ€” the previous defaults were heavier than the native iOS 26 aesthetic. If you prefer the heavier look, set explicit thickness and blur values in your GlassThemeData.

โš ๏ธ Semi-Breaking โ€” GlassAppBar transparent by default #

GlassAppBar now renders a transparent navigation bar by default โ€” no glass surface, no specular rim. This matches iOS 26's actual navigation bar pattern where the glass effect is on individual buttons, not the bar itself.

Previously, GlassAppBar always wrapped its content in an AdaptiveGlass surface with LiquidGlassSettings(blur: 15). This created a visible glass rectangle behind the title and actions โ€” a Material-style app bar with glass paint, not an iOS 26 navigation bar. A better version of this to come next

To opt in to a glass background (e.g. for scroll-edge transitions), pass explicit settings:

// Before (0.12.x) โ€” glass was always on:
GlassAppBar(title: Text('Title'))

// After (0.13.0) โ€” transparent by default:
GlassAppBar(title: Text('Title'))  // transparent, iOS 26 style

// Opt-in glass background:
GlassAppBar(
  title: Text('Title'),
  settings: LiquidGlassSettings(blur: 15, thickness: 10),
)

Additionally, quality now defaults to null (inherits from ambient scope) instead of GlassQuality.premium.

๐ŸŽจ Visual โ€” Specular rim refinement #

Standard/minimal quality glass surfaces now render a more refined specular inner-border rim:

  • True inner border โ€” the specular stroke is now clipped to its inner half via _ShapeClip, creating an optically correct glass-edge reflection instead of a center-straddling stroke that bleeds outside the shape boundary.
  • Organic overlay blending โ€” BlendMode.overlay replaces BlendMode.srcOver, so the rim reacts to the background colour underneath rather than appearing as a flat white line. Darker backgrounds produce subtler rims; lighter backgrounds produce brighter ones.
  • Flat-edge suppression โ€” shapes with borderRadius: 0 (used by GlassAppBar, GlassSideBar, GlassToolbar) no longer render the specular rim. On full-width flat surfaces, the rim looked like a Material divider line rather than an internal glass reflection.

Only affects standard and minimal quality. Premium quality uses Impeller's native LiquidGlassLayer which has its own refraction-based edge rendering.

โšก Performance โ€” Quality adapter tuning #

  • Faster degradation โ€” Phase 3 runtime monitoring now triggers a quality step-down after 2 consecutive over-budget windows (previously 3), reducing reaction time from ~6 seconds to ~4 seconds. This means devices that genuinely can't sustain their assigned quality level are protected sooner.
  • Documentation updated โ€” removed provisional calibration warnings from warmup threshold docs. The 20ms premium / 28ms standard thresholds are now considered validated.

โš ๏ธ Semi-Breaking โ€” LiquidStretch.resistance default #

The default resistance value for LiquidStretch has changed from 0.08 to 0.01. This makes all stretch interactions feel more responsive and fluid โ€” closer to the iOS 26 native feel. The previous value was overly dampened.

All widgets using LiquidStretch without explicitly setting resistance (including GlassButton, GlassCard, GlassContainer, GlassMenu) will feel stretchier. To restore the previous behaviour:

LiquidStretch(
  resistance: 0.08, // previous default
  child: ...,
)

๐Ÿงช Tests โ€” 1898 passing (+124 new) #

  • New GlassButton tests: persistPressOnDrag true/false behaviour, default values, cancel paths.
  • New GlassSearchBarConfig.cursorColor tests: default null, explicit value, independence from textColor.
  • Updated glass_quality_adapter tests for degradeWindowCount: 2.
  • Updated stretch tests for new resistance default.
  • Updated GlassAppBar defaults test for transparent-by-default change.
  • Updated golden tests for specular rim flat-edge suppression.

๐Ÿ“ฆ Example app #

  • Keypad lock screen demo โ€” new full-screen demo showcasing GlassButton in a PIN-entry layout.
  • Restructured showcase pages โ€” all category pages (containers, feedback, input, interactive, overlays, surfaces) reorganised for cleaner presentation. More work to come...

0.12.8 #

๐Ÿ› Fix โ€” GlassTextField reverted to v0.12.4 + icon drift fix #

  • Reverted to v0.12.4 โ€” restored exact line-count and layout logic. The v0.12.6โ€“0.12.7 changes introduced regressions (line breaks at wrong character boundary, icons pinned to container bottom).
  • Fixed icon drift under system text scaling โ€” in fixed-height mode, icons are now always centred relative to the container rather than relative to the text row. This prevents icons from shifting position when users change system text scaling. In dynamic-height mode (no height parameter), iconAlignment is respected as before.

Thanks @g3mf0r for the detailed testing.


0.12.7 #

๐Ÿ› Fix โ€” GlassTextField icon alignment (retained) + line-count regression fix #

  • iconAlignment: .end no longer drifts under system Large Text. The Center widget wraps only the TextField, not the entire icon Row, so CrossAxisAlignment.end/.start works correctly against the full container height. (retained from 0.12.6)
  • Reverted line-count measurement back to renderBox.size.width (the v0.12.4 approach). The v0.12.6 RenderEditable width change caused line breaks to fire a couple of characters early. Thanks @g3mf0r for catching this.

0.12.6 #

๐Ÿ› Fix โ€” GlassTextField icon alignment and line-count accuracy #

  • iconAlignment: .end no longer drifts under system Large Text. The Center widget now wraps only the TextField, not the entire icon Row, so CrossAxisAlignment.end/.start works correctly against the full container height.
  • Line-count measurement is now pixel-perfect. _measureLineCount walks the render tree to find the actual RenderEditable and uses its layout width (which accounts for the internal _caretMargin โ‰ˆ 3 px). Falls back gracefully if the render walk fails.

0.12.5 #

โœจ New โ€” GlassMenu.onClose callback #

Added onClose: VoidCallback? to GlassMenu. Fires when a close is triggered (barrier tap, trigger re-tap, or item selection), before the animation completes. Useful for synchronising external state such as a GlassMorphController.

Thanks to @g3mf0r for the contribution (#67).


0.12.4 #

๐Ÿ› Fix โ€” GlassTextField layout and reactivity #

onLineCountChanged fires correctly under fixed-height constraints #

The onLineCountChanged callback silently stopped firing after the first measurement when the field was inside a fixed-height container (e.g. SizedBox(height: 46) or height: 46 on the field itself). The internal guard used size == _lastTextFieldSize โ€” but a fixed outer height keeps the RenderBox size constant, so the guard always exited early after the first call. The fix replaces the size-equality guard with a (text, constrainedWidth) guard: the callback fires whenever the text content or available wrapping width changes, regardless of what the outer height is doing.

This also resolves the stale-state reactivity bug where _lines stored in State stopped updating borderRadius after re-opening the keyboard.

Placeholder and text stay vertically centred under system Large Text #

When height is specified (fixed-height mode), the outer padding's vertical component was applied inside the SizedBox, pushing placeholder text and icons downward when the user enabled a large system font. The field now strips vertical padding in fixed-height mode and centres the text row via Align, matching the behaviour of GlassSearchBar. The padding parameter's horizontal values are unchanged.

โœจ New โ€” bottom panel for GlassTextField and GlassTextArea #

Both widgets now accept an optional bottom widget that renders below the text area inside the same glass card. Use it to build the "rich composer" pattern โ€” a text input on top with an action bar, attachment strip, or formatting toolbar below, all sharing one glass surface:

GlassTextField(
  maxLines: 5,
  minHeight: 44,
  maxHeight: 160,
  bottom: Padding(
    padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4),
    child: Row(
      children: [
        IconButton(icon: Icon(Icons.attach_file), onPressed: _attach),
        const Spacer(),
        IconButton(icon: Icon(Icons.send), onPressed: _send),
      ],
    ),
  ),
)

The panel renders inside the glass surface alongside the text area. Callers can add a Divider between the text area and the panel if a visual separator is desired. Not available on GlassTextField.search (that constructor is single-line only; bottom is always null there).


0.12.3 #

๐ŸŽจ Visual โ€” Slider & Switch thumb refraction tuning #

  • GlassSlider thumb โ€” increased refractiveIndex (1.15 โ†’ 1.3) and thickness (10 โ†’ 13) for a more pronounced glass lens feel. Reduced glassColor alpha (0.1 โ†’ 0.08), lightIntensity (2.0 โ†’ 1.8 premium), baseAlphaMultiplier (0.2 โ†’ 0.08 premium), and edgeAlphaMultiplier (0.4 โ†’ 0 premium) for a cleaner, more transparent thumb.
  • GlassSwitch thumb โ€” reduced refractiveIndex (1.15 โ†’ 1.12) and glassColor alpha (0.1 โ†’ 0.08) for subtler refraction.
  • Material fade via Opacity widget โ€” slider thumb now uses widget-level Opacity (matching GlassSwitch pattern) instead of color alpha for the press-down fade. Critical for Impeller: properly removes the child from the compositing tree so native refraction shows through.

โšก Jelly physics โ€” spring-based velocity #

  • GlassSlider jelly โ€” replaced raw _velocity tracking with a SingleSpringController feeding buildJellyTransform. Produces smooth squash/stretch with natural deceleration and elastic bounce-back, matching the tab bar / bottom bar pill feel. maxDistortion raised (0.25 โ†’ 0.6), velocityScale lowered (30 โ†’ 2) to account for the spring's normalised 0โ†’1 position range.

0.12.2 #

โœจ New โ€” GlassTextField enhancements #

Three community-requested features for GlassTextField (and GlassTextArea):

Explicit size properties #

height, minHeight, and maxHeight give direct control over the field's dimensions โ€” no wrapping SizedBox needed:

// Fixed height โ€” matches GlassSearchBar's 44pt:
GlassTextField(height: 44, placeholder: 'Search')

// Constrained range โ€” grows with content:
GlassTextField(minHeight: 44, maxHeight: 200, maxLines: 10)

height is mutually exclusive with minHeight/maxHeight (assertion enforced).

onLineCountChanged callback #

Fires whenever the number of rendered lines changes (accounting for text wrapping, not \n characters). Also fires on initial build. Uses the TextField's own RenderBox height โ€” no external TextPainter math, so it works correctly with text scaling and system accessibility settings.

GlassTextField(
  maxLines: 6,
  onLineCountChanged: (lines) {
    setState(() => _borderRadius = lines > 1 ? 8.0 : 20.0);
  },
)

iconAlignment parameter #

Controls where prefix/suffix icons sit when the field spans multiple lines:

// Pin send button to bottom โ€” chat composer pattern:
GlassTextField(
  maxLines: 6,
  iconAlignment: CrossAxisAlignment.end,
  suffixIcon: Icon(Icons.send),
)

Accepts CrossAxisAlignment.start (top), .center (default), or .end (bottom). No visible effect on single-line fields.

All three features are forwarded through GlassTextArea.

GlassTextField.search named constructor #

A new convenience constructor that pre-configures GlassTextField with compact search-bar defaults: height: 44, iconSpacing: 8, padding: EdgeInsets.symmetric(horizontal: 12, vertical: 8), borderRadius: 22, and textInputAction: TextInputAction.search. Eliminates the boilerplate previously required to match GlassSearchBar visuals:

GlassTextField.search(
  placeholder: 'Search messagesโ€ฆ',
  prefixIcon: Icon(CupertinoIcons.search, size: 20),
  useOwnLayer: true,
)

GlassSearchBar now uses this constructor internally, reducing duplicated decoration code.

๐Ÿ› Fix #

  • onLineCountChanged now fires on programmatic controller changes โ€” Previously the callback only responded to physical keyboard input (TextField.onChanged). Setting controller.text = '...' or calling controller.clear() (e.g. in a chat "Send" handler) did not re-measure the line count. The widget now actively listens to the TextEditingController and re-measures on any text mutation.

  • Suffix icon spacing now respects iconSpacing โ€” The gap before the suffix icon was hard-coded to 12px while the prefix icon correctly used widget.iconSpacing. Both sides now use the same parameter.

0.12.1 #

๐Ÿ› Fix โ€” eliminate rectangular blur halo over PlatformViews (iOS Impeller) #

LightweightLiquidGlass and _FrostedFallback previously wrapped their glass surface in ClipPath(ShapeBorderClipper(...)). When the parent was an iOS PlatformView (e.g. mapbox_maps_flutter MapWidget, video_player on iOS), the descendant BackdropFilter's rectangular blur output leaked past the rounded clip โ€” visible as a faint square halo around the rounded glass shape, most obvious when light backdrop content scrolled underneath.

Flutter framework PR #177551 (merged Dec 2025, shipped in 3.41.0-0.0.pre+) fixed this at the engine level by forwarding ClipRRect clip data to the iOS PlatformView mutator stack โ€” but only ClipRRect, not ClipPath, even when the path inside is mathematically a rounded rect.

This release routes shapes that resolve to a RoundedRectangleBorder (i.e. LiquidRoundedSuperellipse, LiquidVerticalRoundedSuperellipse) through ClipRRect instead of ClipPath. The engine fix now triggers and the halo disappears for those shapes.

LiquidOval is intentionally NOT routed through ClipRRect โ€” empirically the engine fix doesn't forward ClipRRect with circular(double.infinity) nor a LayoutBuilder-computed finite radius on the LiquidOval path. Callers that need a halo-free circular glass surface over a PlatformView should use LiquidRoundedSuperellipse(borderRadius: size / 2) instead, which renders identically on a square widget and triggers the engine fix.

Closes upstream Flutter #175048 and #115926 for liquid_glass_widgets users.

Based on PR #61 by @jfhair.

0.12.0 #

โœจ New #

LiquidGlassWidgets.wrap() โ€” theme parameter #

wrap() now accepts an optional GlassThemeData? theme parameter. When provided, it wraps the child in a GlassTheme โ€” eliminating the need for a separate GlassTheme widget in your tree.

LiquidGlassSettings.standardOpacityMultiplier #

A new multiplier applied to the glass colour alpha when rendering in Standard mode. This allows tuning Standard 2D compositing opacity to achieve more parity with Premium 3D volumetric refraction without needing separate colour values for each mode.

LiquidGlassSettings(
  glassColor: Colors.white.withValues(alpha: 0.3),
  standardOpacityMultiplier: 0.4, // Standard renders at 0.3 ร— 0.4 = 0.12 alpha
)

Defaults to 1.0 (no change). Fully interpolated via LiquidGlassSettings.lerp() and wired through copyWith().

GlassPage โ€” full-screen glass scaffold #

A new full-screen scaffold widget for glass-based layouts. Handles background imagery, status bar styling, and background sampling in a single widget.

enableBackgroundSampling defaults to true when a background widget is provided, and false otherwise โ€” so the common case just works without extra configuration.

GlassPage(
  background: Image.asset('assets/wallpaper.jpg'),
  child: Scaffold(...),
)

Export hygiene #

  • glass_page.dart now uses a show clause: only GlassPage and GlassStatusBarStyle are exported (internal state classes are no longer public).
  • liquid_glass_scope.dart now uses a show clause: only LiquidGlassScope, GlassBackgroundSource, and GlassRefractionSource are exported.

๐ŸŽจ Visual โ€” Standard/Premium parity improvements #

Shader composite improvements (lightweight_glass.frag) #

The Standard-tier lightweight shader composite logic has been improved for closer visual parity with the Premium Impeller path. Shader rim constants are unchanged from 0.11.0 โ€” AdaptiveGlass normalization now handles Premium โ†’ Standard scaling in Dart space instead:

  • PATH A (background texture): now uses applyGlassColorLW() โ€” a luminosity-preserving glass tint that matches Premium's colour handling for both chromatic (mint, bronze) and achromatic (white, grey) glass colours.
  • PATH A ambient darkening: ambientStrength ร— 0.25 + 0.08 creates the glass shadow effect that visually separates glass from non-glass, matching what Premium achieves through blur compositing.
  • PATH A adaptive rim colour: mix(bgRgb, white, 0.7) brightens the background at the edge, matching Premium's getHighlightColor.
  • PATH A edge-zone refraction: indicator-style background warping at rounded corners using smoothstep edge zone with quadratic falloff โ€” the same proven approach as interactive_indicator.frag but scaled for containers. Zero transcendentals (polynomial smoothstep + multiplies only). Flat interior pixels naturally produce zero offset. Currently active on surface widgets (GlassBottomBar, GlassTabBar, GlassSideBar, GlassToolbar) when a backgroundKey is provided; GlassCard and GlassButton use PATH B and will benefit once AdaptiveGlass gains scope-aware background key passthrough.
  • PATH A unified interactive glow: uGlowIntensity press-feedback now applies in PATH A, closing an architectural gap where switch/slider thumbs inside background-sampled containers had no glow.
  • Volumetric depth gradient: subtle top-to-bottom ambient shading (+vertCoord ร— 0.04) in both PATH A and PATH B creates a natural 3D anchored depth feel, simulating light entering from above. Cost: one multiply + one add per fragment.
  • PATH B frost floor: 8% minimum material alpha ensures glass surfaces are always visible when glassColor.a = 0 (Premium default), preventing invisible glass in SrcOver compositing.
  • PATH B contrast-adaptive rim: shifts rim colour toward mid-grey on bright backgrounds so white-on-white borders remain distinguishable.
  • Directional rim bonus: a small 0.15 ร— directionalInfluence ร— lightIntensity term adds subtle lit-side variation on top of the constant rim base โ€” matching how Premium's 3D bevel naturally brightens toward the light source.

Interactive widget transparency tuning #

  • GlassSwitch standard thumb: baseAlphaMultiplier: 0.0, edgeAlphaMultiplier: 0.15 โ€” fully transparent body with subtle edge presence for a cleaner glass look.
  • GlassSlider standard thumb: baseAlphaMultiplier: 0.08, edgeAlphaMultiplier: 0.1 โ€” minimal body opacity with soft edge glow.

Elevated widget predictability #

Removed the arbitrary +0.2 alpha boost on elevated widgets inside AdaptiveGlass. Elevation is now expressed purely through the shader's densityFactor physics, making the opacity response predictable and proportional to user settings.

Interactive widget normalisation (GlassEffect) #

Standard-tier interactive indicators (slider thumbs, switch thumbs, segmented control pills) now apply the same normalisation as AdaptiveGlass โ€” thickness ร— 0.4, lightIntensity ร— 0.6 โ€” preventing the 2D shader from rendering these elements heavier than their Premium counterparts.

๐Ÿ“ฆ Example app #

  • Quality comparison demo background image bundled as a local asset (example/assets/mountain_landscape.jpg) โ€” eliminates network dependency and first-frame loading flash.

0.11.0 #

โœจ New โ€” Liquid Morph Engine (new architectural system) #

This release introduces the Liquid Morph Engine โ€” a standalone, reusable physics and animation system for iOS 26-style liquid glass morphing. It lives in lib/engine/ and is fully decoupled from any specific widget.

GlassMenu is the first consumer of the engine. Future widgets (sheets, cards, buttons) will use the same engine to achieve consistent, physics-correct liquid glass transitions throughout the library.

Documentation: docs/LIQUID_MORPH_ENGINE.md โ€” full guide covering GlassMorphController, LiquidMorphState, LiquidMorphPhysics, and how to integrate the engine into your own custom widgets.

Core engine types #

Type Role
GlassMorphController Lifecycle owner โ€” manages the spring, exposes open() / close()
LiquidMorphState Immutable value object โ€” one per frame, contains all render values
MorphPhase Semantic lifecycle enum โ€” tells you where in the animation you are
MorphSpeed Enum โ€” controls spring stiffness without exposing raw physics constants
LiquidMorphPhysics Internal stateless math engine

How it works #

Two conceptual "blobs" drive every morphing animation:

  • Blob A (anchor) โ€” the ghost trigger that shrinks away over the first 40 % of the animation, cleanly breaking the liquid bridge.
  • Blob B (body) โ€” travels from the trigger centre to the widget centre along a J-curve overshoot trajectory, expanding from trigger size to target size.

The SDF metaball shader creates the teardrop neck between the blobs automatically โ€” there is no explicit neck geometry. LiquidMorphPhysics.compute() determines each blob's position, scale, and blend factor on every frame.

GlassMenu โ€” first engine consumer #

  • Teardrop open animation ยท The menu grows from the trigger point along the dual-curve SDF path, producing the iOS 26 "bubble emerging from button" effect.
  • Rubber-band close physics ยท On dismiss the teardrop recoils with a critically-damped spring + overshoot tail, matching the tactile snap of native iOS context menus.
  • Velocity-bump alignment ยท Spring initial velocity is seeded from touch velocity at release โ€” fast flicks close snappily, slow releases settle deliberately.
  • Handoff latching ยท Re-opening during a close animation inherits the in-flight velocity and reverses smoothly โ€” no pop or cut.
  • Blob scaling ยท Blob sizes scale relative to trigger size and computed menu height, so short and tall menus receive proportionally correct teardrop curvature.

See it live: The Apple Messages demo (cd example && flutter run -t lib/apple_messages/apple_messages_demo.dart) showcases the morphing engine in a real-world context โ€” tap the menu or Edit button at the top of the screen to trigger the GlassMenu with full teardrop open/close physics.

Spring physics refinements #

  • Critical damping (ฮถ = 1.0) on all spring controllers prevents oscillation on rapid successive opens.
  • interactionScale, stretch, and stretchResistance integrate into the morphing path via the same spring solver used by LiquidStretch.

๐Ÿ› Fixes #

  • GlassMenu โ€” safe area / notch clipping on iOS and Android ยท Menu position and maximum height were computed from MediaQuery.padding, which is consumed by ancestor SafeArea widgets and reports 0 inside a fully-safe tree. Switched to View.of(context).padding (raw hardware insets) so the menu is always clamped correctly regardless of SafeArea nesting depth. Fixes the menu appearing under the Dynamic Island on iPhone 14 Pro and similar devices.

  • GlassMenu (scrollable) โ€” scrolling now works on large menus ยท Menus with more items than fit on screen can now be scrolled reliably.

๐Ÿ—‚ Example restructure โ€” demos/ suite #

The example/ package has been reorganised for a cleaner public-facing demo experience:

  • New example/lib/demos/ folder containing seven self-contained, copy-pasteable demos:

    • glass_menu_demo.dart โ€” all 9 GlassMenuAlignment positions, scrollable item list
    • glass_tab_bar_scrollable_demo.dart โ€” scrollable GlassTabBar with dynamic tab add
    • glass_modal_sheet_demo.dart โ€” all sheet states (peek / half / full), Apple Maps peek style
    • glass_bottom_bar_demo.dart โ€” magic-lens masking with GlassBottomBar
    • bottom_bar_tab_width_demo.dart โ€” tabWidth on both bar variants side-by-side
    • searchable_bar_demo.dart โ€” GlassSearchableBottomBar edge cases
    • shape_debug_demo.dart โ€” GlassButton shape visualiser
  • Apple Messages demo (example/lib/apple_messages/) โ€” showcases the Liquid Morph Engine in a full real-world context; tap the menu or Edit button at the top to trigger GlassMenu.

  • example/lib/modal_sheet_showcase/ removed (file moved to demos/glass_modal_sheet_demo.dart).

  • Experimental scratchpad scripts moved to git-ignored example/lib/playground/.


0.10.10 #

Thanks to @g3mf0r for PR #55. ๐Ÿ™

โœจ New #

  • GlassMenu โ€” menuAlignment enum ยท A new GlassMenuAlignment enum (10 values: none, topLeft, topCenter, topRight, centerLeft, center, centerRight, bottomLeft, bottomCenter, bottomRight) lets you pin the menu to a specific edge or corner of its trigger instead of relying solely on auto-detection. The enum is now part of the public API surface exported from glass_menu.dart.

  • GlassMenu โ€” autoAdjustToScreen with menuPadding ยท When autoAdjustToScreen: true, the new menuPadding: EdgeInsets? parameter applies additional inset constraints so the menu body never clips against device edges.

  • GlassMenu โ€” itemBorderRadius ยท Controls the corner radius of individual menu item cells, independent of the outer menuBorderRadius.

๐Ÿ› Fixes #

  • GlassTabBar โ€” multi-tab drag jump ยท Dragging the indicator across multiple tab widths in a single gesture now snaps to the correct distant tab. The previous implementation only incremented/decremented by ยฑ1 regardless of drag distance, causing the indicator to teleport unexpectedly when the finger crossed more than one tab boundary.

  • GlassTabBar โ€” glass refraction during indicator drag ยท Refraction and shadow effects are correctly suppressed during the drag animation and restored on settlement, eliminating a visual glitch where the glass distortion would persist after releasing the indicator.

๐Ÿงช Tests #

  • Added 4 new GlassMenu tests covering GlassMenuAlignment enum values, menuAlignment parameter, autoAdjustToScreen + menuPadding, and itemBorderRadius.
  • Added 2 new GlassTabBar tests covering multi-tab drag jump (left and right) to prevent regression of the PR #55 fix.

0.10.9 #

Thanks to @g3mf0r for PR #54. ๐Ÿ™

โœจ New #

  • GlassTabBar (scrollable) โ€” jelly physics on indicator drag ยท The scrollable indicator pill now feeds real-time drag velocity into the liquid glass shader, producing the same organic stretch-and-settle effect that fixed-mode tabs already had.

๐Ÿ› Fixes #

  • LiquidGlassWidgets.wrap โ€” adaptiveQuality: true without adaptiveConfig permanently locks to standard ยท The default fallback config was created with initialQuality: GlassQuality.standard, which the adapter treats as a skip-Phase-2 signal โ€” immediately jumping to Phase 3 at standard without ever running the warmup benchmark. On capable devices (including the iPhone simulator on Apple Silicon) this prevented the adapter from ever discovering that the device can sustain premium. Fixed by removing the erroneous initialQuality from the fallback; Phase 2 now always runs when no explicit quality is provided.

  • GlassTabBar (scrollable) โ€” indicator overflows bar on low tab counts ยท The right drag boundary was computed as viewMax instead of viewMax - indicatorWidth, allowing the pill to slide outside the bar when there were only 2โ€“3 wide tabs. Corrected to viewMax - targetWidth.

  • GlassTabBar (fixed) โ€” tiny accidental drags switch tabs ยท Tab switching on drag-end now requires either a displacement greater than 20 % of the tab width or a flick velocity above 400 px/s, preventing unintended switches from small incidental movements.

  • GlassTabBar (scrollable) โ€” flick gesture ignored in scrollable mode ยท A horizontal flick with sufficient velocity now overrides the nearest-tab distance calculation and advances the indicator in the flick direction, matching the fixed-mode behaviour.


0.10.8 #

Thanks to @g3mf0r for PR #52. ๐Ÿ™

๐Ÿ› Fixes #

  • GlassTabBar โ€” indicator drag drift on desktop/web ยท The indicator position was accumulated via delta.dx additions each frame, causing the pill to visually lag behind the pointer on desktop platforms where pointer events arrive at a higher frequency than the frame budget. Fixed by computing position from the absolute global pointer coordinate on every update event, eliminating accumulated drift.

  • GlassTabBar (scrollable) โ€” tab labels hidden behind indicator pill ยท The SingleChildScrollView (tab labels) and the background pill were inserted in the wrong stack order โ€” labels were painted first, then the pill on top, obscuring them. Fixed by inserting the pill before the labels so labels always paint above the pill (correct z-order).

  • GlassTabBar (scrollable) โ€” indicator fly-off past bar edges ยท The indicator pill had no boundary clamping in scrollable mode, allowing it to animate outside the visible bar area. Drag offset is now clamped to [scrollOffset - 35 %, scrollOffset + screen + 35 %].

โœจ New #

  • DividerSettings โ€” new optional dividerSettings parameter on GlassTabBar. Renders animated vertical dividers between tabs with configurable thickness, indent, endIndent, custom decoration, animation duration/curve, and an isHideAutomatically flag that fades out dividers adjacent to the active tab. Includes a copyWith helper for convenient inline customisation.

  • Grab-to-drag in scrollable mode โ€” the indicator pill in scrollable mode can now be directly grabbed and dragged to a new tab. Uses a GestureArenaTeam (HorizontalDragGestureRecognizer as captain + TapGestureRecognizer) to correctly win the arena against the SingleChildScrollView when the initial touch is within the active indicator's bounds. The scroll view retains natural scrolling behaviour when touching outside the pill.

  • indicatorShadow โ€” new optional indicatorShadow: List<BoxShadow>? parameter on GlassTabBar. Applies a drop shadow to the resting (solid-colour) indicator pill, improving contrast in light-mode themes where the pill and track share similar colours. The shadow is automatically suppressed during the liquid glass drag animation so it does not interact with the backdrop blur, and restored when the pill returns to its idle state.


0.10.7 #

Thanks to @yukinoaruu for PR #51. ๐Ÿ™

๐Ÿ› Fixes #

  • GlassMenu โ€” trigger button dead zone after closing ยท After closing the menu, the trigger button would ignore taps for the duration of the closing spring animation (~95% of travel), forcing the user to wait several seconds before being able to reopen it. Fixed by separating the visual-hide threshold (0.05) from the input-block threshold (0.80) into two independent booleans: isButtonVisible and isMenuBlocking. The button now becomes tappable again as soon as the animation drops below 80%, and the morphing glass overlay wraps in IgnorePointer(ignoring: value < 0.8) to prevent the contracting container from consuming the tap instead.

0.10.6 #

๐Ÿ› Fixes #

  • GlassBottomBar โ€” extraButton causes bar to float in the middle of the screen ยท Wrapped the inner Row in a SizedBox(height: barHeight) so the Scaffold.bottomNavigationBar slot always receives an explicit tight height constraint. Without this, the Expanded child introduced by extraButton propagated an unbounded height through LiquidGlassLayer, causing Flutter to render the bar centred on screen instead of pinned to the bottom edge.

0.10.5 #

โœจ New #

  • SearchableBottomBarController โ€” added openSearch(), closeSearch(), and isSearchOpen getter for programmatic search control. Previously the only way to open search was by driving isSearchActive from parent state.
  • GlassTabBar โ€” added maskingQuality parameter (MaskingQuality.high / MaskingQuality.off), matching the existing GlassBottomBar API. Set to off to disable the 8 px jelly-bloom expansion on lower-end devices.
  • GlassSlider โ€” added interactionBehavior, glowColor, and glowRadius for consistent drag-glow customisation across all interactive widgets.
  • GlassSegmentedControl โ€” same interactionBehavior, glowColor, glowRadius params added for API parity with GlassSlider and GlassTextField.
  • LiquidGlassWidgets.respectsAccessibility โ€” deprecated alias added pointing to respectSystemAccessibility. Will be removed in v1.0.

๐Ÿ› Fixes #

  • GlassTextField โ€” fixed a use-after-dispose crash when focusNode cycled null โ†’ external โ†’ null. The widget now tracks ownership with an explicit _ownsNode flag and correctly creates a fresh internal node on each transition.
  • GlassTabBar โ€” resolved scrollable-mode visual glitches. The indicator now stays perfectly glued to the active tab during scrolling without drifting, uses native "snappy" spring physics for consistent feel, and implements a three-layer rendering architecture so the solid indicator pill cleanly clips at the rounded viewport corners while the 8px jelly bloom expands freely over the tab bar boundaries.

0.10.4 #

A huge, heartfelt thank-you to @yukinoaruu for PR #49. ๐Ÿ™

We made a mess of the original 0.10.3 merge of his work โ€” introducing regressions that broke the very things he had so carefully built. He came straight back, fixed every issue, and did it with incredible patience and generosity. This release is entirely his. If you are enjoying GlassMenu, it is because of him.

๐Ÿ› Fixes (regressions from 0.10.3 merge) #

  • GlassMenu โ€” full-list rebuilds on every pointer event ยท Restored the _cachedWrappedItems mechanism that was accidentally dropped. The previous merge caused the entire wrapped-item list to be recreated on each pointer move, resetting pressed/hover states mid-gesture and tanking performance on menus with many items.
  • GlassMenu โ€” selection and hover state precision ยท Migrated to a ValueNotifier system (_hoveredIndexNotifier, _isDraggingNotifier). Individual menu items now rebuild in isolation instead of triggering a full setState on the entire menu tree, keeping animations at a steady 60 fps.
  • GlassMenu โ€” ghosting on selection pill ยท Fixed a double-background artifact where the selected GlassMenuItem painted its own hover fill on top of the sliding pill, producing a faint ghost ring. Selected items now transition to Colors.transparent instantly.
  • GlassMenu โ€” disabled items could be tapped ยท Tapping a disabled item no longer calls onTap or closes the menu. The pill highlight correctly skips disabled items during pointer tracking.
  • GlassMenu โ€” double onTap firing ยท Removed a redundant onTap callback in the internal wrapped-item builder that was causing every selection to fire twice.
  • GlassMenu โ€” RangeError when item list shrinks while open ยท didUpdateWidget now clears _hoveredIndex when items.length decreases, preventing an out-of-bounds crash when the pill tried to measure a deleted item.
  • GlassMenuItem โ€” disabled opacity ยท Disabled items now render at Opacity(0.4) to match the design spec and test expectations.
  • GlassMenuLabel โ€” hybrid title/child API ยท GlassMenuLabel now accepts either a title String (rendered as stylised uppercase) or an arbitrary child Widget, enabling diverse non-interactive content beyond simple section headers.
  • GlassMenuLabel โ€” height default ยท Default height set to 30.0 so the selection pill cannot drift when items with non-standard font sizes are mixed in.
  • GlassMenu โ€” glowIntensity parameter ยท Added glowIntensity and wired it through to GlassContainer, completing the full interaction-glow parameter surface.
  • GlassMenu โ€” glowOnTapOnly default corrected ยท Default changed to true to prevent a permanently stuck glow artefact during scroll and drag gestures.
  • GlassMenu โ€” stretch parameter rename ยท Renamed allowPositiveXStretch / allowNegativeXStretch / allowPositiveYStretch / allowNegativeYStretch to allowPositiveX / allowNegativeX / allowPositiveY / allowNegativeY to align with the LiquidStretch API surface.
  • GlassMenu โ€” compositing architecture ยท Removed redundant RepaintBoundary nodes that were leaving descendant glass layers DETACHED from the compositor scene, and moved GlassGlow inside GlassContainer's clip subtree to prevent glow bleed onto the background.

โš ๏ธ Breaking โ€” GlassMenu stretch parameter renames #

The four optional stretch-axis override parameters introduced in 0.10.3 have been renamed:

0.10.3 name 0.10.4 name
allowPositiveXStretch allowPositiveX
allowNegativeXStretch allowNegativeX
allowPositiveYStretch allowPositiveY
allowNegativeYStretch allowNegativeY

All four remain optional with null defaults (auto-inferred from menu position). Only code explicitly passing the old names needs updating.

0.10.3 #

Big thanks to @yukinoaruu for PR #47 โ€” a comprehensive interaction engine upgrade for GlassMenu that brings it more in line with iOS 26 context menu behaviour. ๐Ÿ™

โœจ Features #

Heterogeneous menu items โ€” GlassMenuDivider and GlassMenuLabel #

Menus now accept any Widget, enabling iOS 26-style section grouping:

GlassMenu(items: [
  const GlassMenuLabel(title: 'Actions'),    // renders as 'ACTIONS'
  GlassMenuItem(title: 'Save', onTap: () {}),
  const GlassMenuDivider(),
  GlassMenuItem(title: 'Delete', isDestructive: true, onTap: () {}),
])

GlassMenuLabel exposes a height parameter (default 30.0) so custom font sizes don't drift the selection-pill position.

GlassMenuItem โ€” rich content #

Six new parameters: subtitle, enabled, titleStyle, subtitleStyle, iconColor, iconSize.

Scroll-aware selection pill #

A sliding highlight follows the pointer and disappears automatically when the user starts scrolling (10 px drag-slop guard + ScrollNotification listener).

Elastic stretch and scroll-safe glow #

GlassMenu now wraps in LiquidStretch for spring physics on drag. glowOnTapOnly: true (the new default) suppresses the glass glare after a drag, preventing a stuck-glow artefact during list scrolling. Full parameter surface: interactionScale, stretch, stretchResistance, stretchAxis, allowPositiveX/NegativeX/Y.

New primitives on GlassGlow and GlassContainer #

GlassGlow.enabled, GlassGlow.glowOnTapOnly, and GlassContainer.glowIntensity are now available for custom integrations.

๐Ÿ› Fixes #

  • GlassMenu โ€” double BackdropFilter ยท Removed an extra blur layer above GlassContainer that doubled the blur sigma and created an over-frosted ring.
  • GlassMenu โ€” DETACHED compositing layers ยท Removed the outer RepaintBoundary wrapping _buildMorphingContainer. When GlassContainer(useOwnLayer: true) installs a BackdropFilter layer it forces compositing on the entire subtree; a RepaintBoundary above it fought the compositor for OffsetLayer ownership, leaving descendant RepaintBoundary nodes (i.e. each GlassMenuItem's glass layer) DETACHED from the scene. GlassGlow and GlassContainer already own their compositing layers โ€” no extra boundary is needed. Separately, Opacity widgets at >= 1.0 are now skipped entirely so no gratuitous OpacityLayer is inserted when compositing is already being forced by a BackdropFilter descendant.
  • GlassMenu โ€” layout overflow during open animation ยท GlassContainer(height: currentHeight) propagated tight height constraints through its entire subtree during the morph. When menu items became visible (previously at value > 0.65), the container was only ~114 px tall while 3 items needed 132 px, causing the Column inside Positioned.fill to overflow by 18 px. Fixed by deferring content rendering until value โ‰ฅ 0.85, exactly when currentHeight becomes null and the container sizes naturally โ€” no tight-constraint cascade possible.
  • GlassMenu โ€” interaction glow bleeds onto background ยท GlassGlow previously wrapped GlassContainer from the outside. _RenderGlassGlowLayer.paint() called canvas.drawCircle() over the full overlay canvas with no shape boundary, causing the radial gradient to paint beyond the menu's glass shape onto the background. Fixed by moving GlassGlow inside GlassContainer's clipBehavior: Clip.antiAlias subtree โ€” matching the architecture used by GlassButton.
  • GlassMenuItem โ€” AnimatedScale layout overflow on press ยท AnimatedScale (backed by RenderTransform) retains the pre-scale layout size during a 0.98-scale press animation, causing a spurious overflow against the menu's bounded Positioned.fill. Fixed by wrapping AnimatedScale in SizedBox(height: effectiveHeight) to isolate the transform's layout footprint.
  • GlassMenu โ€” selection pill layout exception ยท AnimatedPositioned is now inside a bounded SizedBox(height: totalH) โ†’ Stack, preventing a debug-mode layout exception and an out-of-bounds pill position when scrolled.
  • GlassMenu โ€” RangeError on item removal ยท didUpdateWidget clears _hoveredIndex when items.length shrinks while the menu is open.
  • GlassMenu โ€” GlassMenuItem state flicker ยท Wrapped items cached; only rebuilt when widget.items changes, preventing pressed/hover resets during the 60 fps spring ticker.
  • GlassGlow โ€” permanently muted glow ยท didUpdateWidget resets _glowSuppressed when glowOnTapOnly is toggled off.
  • GlassMenuItem โ€” desktop hover state leak ยท dispose() clears _isHovered.
  • Impeller โ€” extreme-stretch glyph-bounds crash ยท Scale determinant clamped before reaching the shader.
  • Android โ€” negative safe-area assertion ยท sysBottom > 25 guard added.

โš ๏ธ Semi-Breaking #

GlassMenu.items changed from List<GlassMenuItem> to List<Widget>. Existing code compiles unchanged โ€” only typed List<GlassMenuItem> variable declarations need widening.

๐Ÿงช Tests โ€” 1,648 passing #


0.10.2 #

Fixes #

  • GlassTabBar (scrollable) โ€” indicator clipping ยท Migrated the selected-tab indicator to an overlay architecture outside SingleChildScrollView, eliminating clip artifacts during scrolling and preserving the full iOS 26 glass bloom expansion.
  • GlassTabBar (scrollable) โ€” tap fires on scroll ยท onTabSelected no longer fires when the user scrolls the tab bar; selection is now only triggered on confirmed taps.
  • GlassTabBar (scrollable) โ€” bloom activates on scroll ยท The pressed indicator bloom no longer activates when scrolling the tab bar content.
  • GlassTabBar (scrollable) โ€” indicator pulsates on transition ยท Fixed a threshold bug that caused the bloom to flicker during tab-switch animations.
  • GlassTabBar (scrollable) โ€” scroll into view ยท Tapping or programmatically selecting a partially-visible tab now smoothly scrolls it fully into view.
  • GlassAdaptiveScope โ€” Android false-negative quality downgrade ยท Mid-range Android devices with Impeller/Vulkan can report inflated warmup P75 values (17โ€“18 ms) due to GPU clock-scaling and JIT shader cache warm-up โ€” not actual slowness. The previous premium threshold of < 16 ms (the raw 60-fps frame budget) was too strict and incorrectly demoted capable hardware to standard or minimal. Thanks @hank205 for the detailed diagnostic log. ๐Ÿ™
  • GlassModalSheet / .show() / GlassModalSheetScaffold โ€” dragIndicatorWidth ยท The drag handle pill width was previously hardcoded at 36 (iOS native). A new dragIndicatorWidth parameter lets you customise it โ€” e.g. 64 for sheets where a more prominent handle better suits the layout. Defaults to 36, no breaking change. Thanks @jfhair (#46). ๐Ÿ™

Changes #

  • GlassQualityAdapter / GlassAdaptiveScopeConfig / GlassAdaptiveScope โ€” configurable warmup thresholds ยท Two new parameters let you tune (and help us calibrate) the Phase 2 warmup classification thresholds:
    • warmupPremiumThresholdMs โ€” P75 below this โ†’ premium. Default raised from 16.0 to 20.0 to account for Android GPU warm-up inflation. (Calibration status: 1 device report โ€” please share yours!)
    • warmupStandardThresholdMs โ€” P75 at or below this (and above premium) โ†’ standard. Default 28.0. (Calibration status: provisional โ€” no real-device data for this band yet.)
    • skipInitialFrames raised from 60 โ†’ 90 (โ‰ˆ1.5 s at 60 Hz) to give Android more time for GPU clocks and shader caches to settle before the benchmark begins.

Phase 3 hysteresis remains the safety net. If a device cannot sustain its warmup-assigned quality, it steps down automatically within ~6 seconds โ€” the new thresholds only affect the initial classification, not runtime correction.

โš  Community calibration needed โ€” especially for warmupStandardThresholdMs. If your device produces a warmup P75 in the 20โ€“28 ms range, please enable debugLogDiagnostics: true and post your P75 + device model to the Threshold Calibration Discussion.

0.10.1 #

Big thanks to @yukinoaruu (#43) and @jfhair (#44, #45) for three excellent contributions this release. ๐Ÿ™

Fixes #

  • GlassModalSheet โ€” child State preservation ยท Removed GlobalObjectKey from the internal Focus bridge. The key was changing every rebuild, quietly tearing down child State (scroll positions, controllers, etc.) on each expand/collapse. (#44)
  • GlassModalSheet โ€” onStateChanged skipped on slow drag ยท Introduced _settledState to track the last published state separately from the in-flight animation target. Side-effects (haptics, callbacks, scroll-to-top) now fire reliably after a drag that crosses a snap threshold mid-gesture. (#45)
  • GlassModalSheet โ€” ghosting and jitter ยท Fixed visual artefacts during sheet transitions. (#43)
  • GlassModalSheet โ€” element subtree stability ยท LiquidStretch now always returns a consistent widget type regardless of interactionScale/stretch values, preventing a full subtree teardown on the frame the sheet reaches full expansion.
  • LightweightLiquidGlass โ€” null-shader passthrough ยท The widget tree shape is now stable while the fragment shader loads asynchronously; a tinted passthrough is painted instead of switching widget types.

0.10.0 #

โš ๏ธ Breaking โ€” Pre-v1 API Cleanup #

LiquidGlassWidgets.wrap() โ€” child is now a required named parameter #

Before:

LiquidGlassWidgets.wrap(const MyApp(), adaptiveQuality: true)

After:

LiquidGlassWidgets.wrap(child: const MyApp(), adaptiveQuality: true)

This aligns with Flutter widget conventions where child is always named.

GlassModalSheetScaffold โ€” parameter renames #

Old New Reason
background: body: Matches Flutter Scaffold.body โ€” it's the primary content, not a visual property
sheetChild: sheet: Cleaner, matches Flutter naming patterns

Before:

GlassModalSheetScaffold(
  background: MyMapWidget(),
  sheetChild: MySheetContent(),
)

After:

GlassModalSheetScaffold(
  body: MyMapWidget(),
  sheet: MySheetContent(),
)

GlassQualityAdapter.skipStaticProbeForTesting โ€” @visibleForTesting annotated #

The static field is now annotated @visibleForTesting. Production code referencing it will receive an analyzer hint. Usage in test files is unchanged.

๐Ÿ› Fix โ€” Android glass fallback on capable devices #

GlassQualityAdapter was applying the static probe result (GlassQuality.minimal) without respecting minQuality. On some Android devices ImageFilter.isShaderFilterSupported returns a false negative, causing the glass shader to be skipped even though the hardware supports it โ€” the only workaround being adaptiveQuality: false.

minQuality is now honoured as a true floor even when the static probe fires:

// Prevents fallback on Android devices with a false-negative static probe
LiquidGlassWidgets.wrap(
  child: const MyApp(),
  adaptiveQuality: true,
  adaptiveConfig: const GlassAdaptiveScopeConfig(
    minQuality: GlassQuality.standard,
  ),
)

โœจ New โ€” Community contributions #

GlassSearchBarConfig.searchIcon โ€” custom search icon (PR #41) #

Thanks to @jfhair for PR #41.

The search pill now accepts a fully custom Widget in place of the default CupertinoIcons.search glyph:

GlassSearchBarConfig(
  onSearchToggle: (active) { โ€ฆ },
  searchIcon: const Icon(CupertinoIcons.sparkles, color: Colors.white),
)

When searchIcon is null (default) the behaviour is unchanged.

indicatorExpansion โ€” tunable jelly-stretch on bottom bars (PR #40) #

Thanks to @jfhair for PR #40.

Both GlassBottomBar and GlassSearchableBottomBar now expose indicatorExpansion to control how far the active-tab pill stretches during a drag gesture:

GlassBottomBar(
  tabs: myTabs,
  selectedIndex: _index,
  onTabSelected: _onTab,
  indicatorExpansion: 8,   // default 14; lower = tighter morph
)

GlassModalSheet โ€” two-phase organic interpolation (PR #39) #

Thanks to @yukinoaruu for PR #39.

The sheet's corner-radius animation now uses a two-phase curve that separates the rapid initial expansion from the final settle, eliminating the snapping artifacts that were visible at the half โ†’ full transition on some devices.

The fix also corrects resolveAdaptiveRadius to use logical screen height (MediaQuery.size.height) instead of viewPadding.top as the primary Pro Max detector, preventing false-positive 54 dp radii on some non-Pro-Max iPhones with unusually high status-bar padding.

Asymmetric top/bottom corner radii in premium pipeline (PR #42) #

Thanks to @jfhair for PR #42.

LiquidVerticalRoundedSuperellipse now feeds independent top/bottom corner radii into the premium SDF shader via a 7-float-per-shape stride, enabling sheets that hug the device chassis curve at the bottom while keeping a tighter radius at the top โ€” matching the Apple Music / Apple Maps card style:

LiquidGlass(
  shape: const LiquidVerticalRoundedSuperellipse(
    topRadius: 20,
    bottomRadius: 54, // tracks iPhone 15 Pro Max chassis
  ),
  child: myContent,
)

Shader note: all shaders continue to pass glslangValidator SPIR-V validation. The new stride-7 path is gated on type == 3 in sdf.glsl and leaves the existing stride-6 path untouched.


0.9.6 #

๐Ÿ› Fix โ€” GlassModalSheet interaction glow in full state #

Thanks to @yukinoaruu for PR #38.

  • Haptic & glow suppression in full state: HapticFeedback.selectionClick() and _saturationController.forward() were firing on every touch when the sheet was in SheetState.full โ€” where the glass surface is fully opaque and neither effect is visible. Both are now gated on !isFull, eliminating spurious haptic feedback and redundant animation ticks.
  • Background glass hides when content glass is active: Added an Opacity(0.0) on the background AdaptiveGlass layer when expandProgress > 0.98 and maintainContentGlass is enabled. Prevents "glass on glass" shader conflicts in Premium mode at full expansion.
  • Interaction glow threshold tightened: GlassGlow pulse guard lowered from expandProgress < 0.98 to < 0.9 to match the existing saturation gate โ€” consistent behaviour across all glow signals.
  • GlassModalSheet geometry defaults refined: topBorderRadius defaults to 56 (was null), horizontalMargin to 5.0 (was 8.0), bottomMargin to 6.0 (was 8.0) for tighter, more native-feeling geometry.
  • InteractionNotification exported: InteractionNotification is now part of the public API surface, enabling consumers to dispatch Smart Silence events from their own widgets.
  • Corner radius tuning: GlassThemeHelpers.resolveAdaptiveRadius values updated to 54 / 46 / 46 (Pro Max / Pro / Notch) for a more conservative, closer-to-system look.

๐Ÿงช Tests โ€” Coverage improvements (1,573 tests) #

Extended branch coverage across five previously under-tested subsystems. Full test count grew from 1,491 โ†’ 1,573 (+82 tests).


0.9.5 #

โœจ Feature โ€” Asymmetric corner radii & floating peek geometry for GlassModalSheet #

Thanks to @yukinoaruu for PR #37.

  • Shader fix: lightweight_glass.frag now supports per-quadrant corner radii via a new uData6 uniform (slots 24โ€“27). A sentinel of uCornerRadius = -1.0 enables asymmetric mode; all existing symmetric shapes fall through unchanged.
  • Clip gap fix: ClipPath geometry on the Skia/Web path is now aligned to RoundedRectangleBorder (circular arc) to match the shader SDF โ€” eliminates the sub-pixel transparent notch at sheet corners.
  • Peek geometry: Five new optional params on GlassModalSheet / GlassModalSheetScaffold โ€” peekWidth, peekHorizontalMargin, peekBottomMargin, peekTopBorderRadius, peekBottomRadius โ€” for Apple Maps-style floating pill peek states.
  • Cleanup: forceSpecularRim removed from AdaptiveGlass, GlassSheet, GlassModalSheet, and GlassModalSheetScaffold. The shader renders the specular rim natively; no migration needed.

๐Ÿ› Fix โ€” GlassSearchableBottomBar dismiss pill focus & keyboard restoration #

The dismiss (ร—) pill was calling FocusScope.of(context).unfocus() which left the FocusNode in a "previously focused" state. This caused Flutter to restore the keyboard on back-navigation, and made the first post-dismiss tap get swallowed by focus routing.

Fixed by:

  • Replacing FocusScope.unfocus() with FocusManager.instance.primaryFocus?.unfocus() in the DismissPill, fully clearing focus state.
  • The ร— button now only dismisses the keyboard โ€” it does not collapse the search state. This matches the real Apple Music / Apple News behaviour where the search bar remains visible (unfocused/ready) after tapping ร—. The caller explicitly collapses search by tapping the home pill or switching tabs.
  • onCancelTap fires first (before the unfocus) so callers can react (clear results, analytics, etc.) before focus is released.

A new onCancelTap: VoidCallback? on GlassSearchBarConfig gives callers a hook into the ร— tap.

โœจ Demo โ€” Apple Music mini-player refinements #

High-fidelity improvements to the Apple Music demo to match the real Apple Music app:

  • Play pill visibility: The floating play pill now stays visible when the search bar is in the "search ready" state (keyboard dismissed). It only hides when the keyboard is actively up (_searchFieldFocused), matching real Apple Music behaviour.
  • Dynamic icon colour: collapsedLogoBuilder now shows the selected (red) icon in scroll-collapse mini mode and the unselected (white) icon when search is active, via a static _kTabs field so tab definitions aren't duplicated.
  • Play pill positioning: aboveBarBottom is now responsive to the bar's current height โ€” switching to collapsedNavBarH when search is active so the pill doesn't drop excessively when the bar shrinks.
  • Play pill animates on search from mini mode: When search is activated from the scroll-collapsed mini state, the play pill animates from the mini gap position back to its full-width position above the expanded search bar, matching real Apple Music.
  • Home pill restores full bar from any state: Tapping the home pill now always calls _dismissMiniMode() when in mini mode โ€” whether arriving from scroll-collapse or from search โ€” scrolling to top and restoring the full 3-tab bar.
  • Library default preserved: collapsedLogoBuilder in the library remains unselectedIconColor โ€” the Apple Music colour logic is isolated to the demo's GlassSearchBarConfig.
  • Multi-tab scroll fix: _dismissMiniMode now uses _activeScrollController (per active tab) instead of hardcoding the home tab's controller, fixing a bug where tapping Radio/Library in mini mode would leave the bar stuck.

0.9.4 #

โœจ Feature โ€” GlassSearchableBottomBar programmatic interaction callbacks #

Addresses two community-requested quality-of-life gaps for GlassSearchableBottomBar.

1. onBarTap โ€” tap-to-restore after scroll-to-hide #

A new onBarTap: VoidCallback? parameter on GlassSearchableBottomBar fires whenever the user taps anywhere on the bar. The callback is wired through a translucent GestureDetector wrapper, so all internal handlers (tab selection, search toggle, indicator drag) continue to work normally โ€” there is zero interference.

Primary use-case is restoring the bar after a scroll-to-hide animation that is managed in the caller's code:

GlassSearchableBottomBar(
  onBarTap: () => setState(() => _barVisible = true),
  ...
)

When onBarTap is null (the default) no extra widget is inserted into the tree โ€” zero overhead.

2. onSearchFieldTap โ€” detect taps on the active search field #

A new onSearchFieldTap: VoidCallback? parameter on GlassSearchBarConfig, passed directly to TextField.onTap. Fires on every tap of the expanded search field body, including re-focus taps after the keyboard was dismissed.

Useful for navigating to a dedicated search screen, showing a suggestion overlay, or logging an analytics event without needing to own the FocusNode:

GlassSearchBarConfig(
  onSearchToggle: ...,
  onSearchFieldTap: () {
    showSuggestions();
    analytics.log('search_field_tapped');
  },
)

Zero breaking changes. Both parameters are optional with null defaults.


0.9.3 #

โœจ Feature โ€” GlassModalSheet system & rendering performance refinement #

Big thanks to @yukinoaruu for PR #33 โ€” a comprehensive and beautifully engineered contribution that brings a whole new class of interactive modal sheet to the library.

1. GlassModalSheet system #

A new, comprehensive modal sheet implementation supporting three interactive states: peek, half, and full.

  • Physics-Driven Transitions: Spring physics for fluid, organic state changes.
  • Asymmetric Geometry: Morphs from a rounded floating pill to a sharp-bottomed full-screen container using the new LiquidVerticalRoundedSuperellipse.
  • Isolated Mechanics: Logic separated into a robust state machine (glass_modal_sheet_state.dart) and physics handler (glass_modal_sheet_mechanics.dart) โ€” a clean architectural blueprint for future complex components.

2. Device-Aware Adaptive Radius #

An intelligent radius resolution algorithm that infers the ideal corner curvature from the device's physical safe area โ€” Dynamic Island vs. Notch vs. Android Home Bar โ€” automatically matching glass curvature to device hardware without manual updates.

3. Advanced Visual Feedback โ€” Pulse System #

A global pulse synchronisation system in the rendering layer allows GlassModalSheet to trigger coordinated saturation and lighting pulses during high-velocity interactions, giving the glass surface a "living", organic feel.

4. Smart Silence โ€” suppressInteractionOnChildren #

InteractionNotification support prevents the "double-reacting" artifact where both a button and the sheet scale simultaneously on a single tap. Child buttons/switches can seamlessly suppress the parent sheet's scaling and glow effects when tapped.

5. New shapes & LiquidStretch constraints #

  • LiquidVerticalRoundedSuperellipse: Enables asymmetric corner radii (top-rounded, bottom-flat) essential for the modal sheet's full-screen morphing animation.
  • Axis constraints: allowPositive / allowNegative pivot support prevents the sheet from "collapsing" downward when dragged โ€” it only stretches upward as a tactile response.

Documentation & Testing #

  • docs/assets/GLASS_MODAL_SHEETS_GUIDE.md โ€” comprehensive developer guide covering the full parameter surface and state behaviours.
  • test/widgets/overlays/glass_modal_sheet_test.dart โ€” 679 lines of rigorous unit and widget tests covering state transitions, gesture arena logic, and physics edge cases.

Zero breaking changes. GlassModalSheet is additive โ€” all existing GlassSheet usages are unaffected.


๐Ÿ› Fix โ€” Selected icon colour washed out by glass indicator #

A huge shoutout and thanks to @jfhair for spotting this issue and putting together PR #29 โ€” it was a fantastic catch, and you had exactly the right instinct on the fix!

The active-tab icon was visually muted ("dull") at rest because the AnimatedGlassIndicator glass lens was painting over the icon layer. Simply moving the indicator behind the icons restores vibrancy but kills the refraction effect โ€” the glass shader needs icons beneath it to warp them as the pill moves.

The fix uses a split-pass sandwich: the pill's solid background renders below the icons (full vibrancy at rest), while the glass shader renders above them (refraction preserved during animation). Both GlassBottomBar and GlassSearchableBottomBar are updated. Zero breaking changes.


๐Ÿ› Fix โ€” GlassSheet specular rim artifact & washed-out inner elements #

Inspired by @yukinoaruu's work in PR #33, who introduced the forceSpecularRim flag and first surfaced this class of visual fidelity issue with the lightweight glass renderer.

The problem & fix #

On the Skia/Web (lightweight) rendering path, a refractiveIndex of 0.7 on a large GlassSheet produced a hard, visible border around the sheet โ€” a bright "line" that looked like an artifact rather than a premium glass surface.

Lowering it globally to fix the sheets caused components inside the sheet to lose their specular highlights and become washed out.

We've introduced semantic preset separation via two distinct RecommendedGlassSettings presets to solve this:

  • RecommendedGlassSettings.overlay (refractiveIndex: 0.7): For cards, buttons, and small interactive widgets.
  • RecommendedGlassSettings.sheet (refractiveIndex: 0.15): For large bottom sheets and modal overlays.

All GlassSheet.show() calls in the demo app now use the sheet preset, while every GlassButton.custom and GlassCard inside a sheet explicitly passes settings: RecommendedGlassSettings.overlay. The package-level default for GlassSheet (glass_sheet_defaults.dart) has also been updated to use refractiveIndex: 0.15 for a better out-of-the-box experience.

Zero breaking changes.


0.9.2 #

๐Ÿ› Fix โ€” GlassSwitch initial-state bloom anchor & polish #

  • First-click bloom anchored correctly. A switch initialised with value: true now anchors the bloom to the right edge on the very first tap, matching all subsequent interactions. Previously _isMovingForward was hardcoded to true at construction regardless of widget.value.

  • _justEndedDrag race condition eliminated. The flag is now consumed atomically inside didUpdateWidget rather than being reset one frame later via addPostFrameCallback, preventing a rare double-bloom after a drag toggle.

  • Floating-point guard hardened. Animation controller resets now use >= 0.99 instead of == 1.0, making the bloom sequence robust against sub-epsilon drift during rapid consecutive toggles.

  • Dead code removed (glassOverlay no-op widget).

  • Haptic feedback added. GlassSwitch now emits HapticFeedback.lightImpact() on tap-toggle, when the thumb crosses the 50 % midpoint during a drag, and on drag-release snap (when the midpoint was never crossed, e.g. a fast flick). Opt out with enableHaptics: false.

  • 3 new regression tests added; GlassSwitch test count now 24.

Zero breaking changes.


0.9.1 #

๐Ÿ› Fix โ€” Adaptive quality system calibration #

Three coordinated improvements to GlassAdaptiveScope / GlassQualityAdapter that prevent modern flagship devices from being incorrectly demoted to standard quality during app startup.

1. Startup-skip window (skipInitialFrames = 60) #

Phase 2 now discards the first 60 frames (โ‰ˆ 1 second at 60 Hz) before collecting warmup data. Those frames capture shader compilation, the first route transition, and provider/localisation initialisation โ€” all artificially inflated and unrepresentative of steady-state glass rendering. Discarding them means the warmup benchmark reflects actual glass workload, not cold-start overhead.

The constant is tunable for testing: GlassQualityAdapter.skipInitialFrames = 0.

2. Raised premium threshold: 12 ms โ†’ 16 ms #

The old threshold of 12 ms (75 % of a 60 fps frame budget) was too tight. The new threshold is 16 ms โ€” one full 60 fps frame budget โ€” which has a cleaner semantic meaning: "can the device render a premium glass frame within the 60 fps budget at P75? Yes โ†’ premium."

P75 raster time Before After
< 12 ms premium โ€”
< 16 ms standard premium
16โ€“20 ms standard standard
> 20 ms minimal minimal

3. allowStepUp defaults to true #

Previously allowStepUp defaulted to false, meaning a Phase 2 decision could never be corrected at runtime. If Phase 2 still makes a conservative call (e.g. on a device under thermal load at startup), Phase 3 can now self-correct after 10 consecutive under-budget windows (โ‰ˆ 20 seconds) + an 8-second cooldown.

The step-up is deliberately slow and invisible to users. Set allowStepUp: false explicitly if you need to lock quality for the session.

Zero breaking changes (adaptive fix) #

All three changes are additive or alter defaults in a user-beneficial direction. Explicit constructor overrides (allowStepUp: false, skipInitialFrames, custom threshold via targetFrameMs) continue to take precedence.


๐Ÿ› Fix โ€” GlassSwitch drag interaction #

GlassSwitch now supports tap and horizontal drag simultaneously without either interaction interfering with the other.

What was fixed:

  • Tap animation restored โ€” registering both onTap and onHorizontalDrag* on the same GestureDetector caused Flutter's gesture arena to drop one interaction after the first touch. Taps now use onTapDown / onTapUp so they share the gesture stream cleanly with drags.
  • Slow drag no longer cancels โ€” Flutter fires onTapCancel before confirming a horizontal drag, which was deflating the "liquid bloom" pill prematurely. _onDragStart now stops any in-progress deflation and restores the plump state immediately.
  • Animation resets between interactions โ€” the thickness animation controller was left at 1.0 after its first cycle and silently skipped the bloom on subsequent taps. It now resets to 0.0 before each new forward pass.

Gesture behaviour unchanged from the user's perspective: tap = full liquid jump animation; drag = thumb tracks finger with symmetric pill stretch; flick = velocity-based snap.

Zero breaking changes #

No API changes. All existing GlassSwitch usages continue to work without modification.


๐Ÿ› Fix โ€” interactionGlowColor now reads from GlassThemeData #

GlassBottomBar and GlassSearchableBottomBar (including its collapsedLogoBuilder state and SearchPill) previously used a hardcoded white glow (0x33FFFFFF) when no explicit interactionGlowColor was set, silently ignoring any GlassThemeData override on the ancestor tree.

Resolution order is now:

interactionGlowColor param โ†’ GlassThemeData.glowColorsFor(context).primary โ†’ internal fallback

This means setting the primary glow color in GlassThemeData now takes effect on the press-interaction highlight across both bar variants, including the collapsed logo pill, without requiring any code changes at the call site.

Affected widgets #

Widget Location
GlassBottomBar TabIndicator interaction glow
GlassSearchableBottomBar SearchableTabIndicator (normal + collapsed/logo state)
GlassSearchableBottomBar SearchPill expanded glow

Zero breaking changes #

Explicit interactionGlowColor parameters continue to win with highest priority. This only changes what happens when the parameter is left null.


โœจ Feature โ€” glowBlurRadius, glowSpreadRadius, glowOpacity on GlassGlowColors #

Three new appearance fields on GlassGlowColors give fine-grained control over the shape of the directional press-glow across all glass widgets:

Field Type Default Effect
glowBlurRadius double 4.0 Gaussian blur sigma via MaskFilter.blur โ€” softens the glow edge into a natural liquid-glass halo
glowSpreadRadius double 0 Extra circle radius as a fraction of the layer's shortest side
glowOpacity double 1 Master opacity multiplier (0โ€“1) applied on top of the glow color's own alpha

Usage #

Set them globally via GlassThemeData to affect all glass widgets at once:

GlassTheme(
  data: GlassThemeData(
    light: GlassThemeVariant(
      glowColors: GlassGlowColors(
        primary: Color(0x55FFFFFF),
        glowBlurRadius: 8,       // soft, diffuse halo
        glowSpreadRadius: 0.15,  // bleeds 15 % beyond touch radius
        glowOpacity: 0.75,       // 75 % of the color's own alpha
      ),
    ),
  ),
  child: ...,
)

Or override per-widget via GlassButton.glowBlurRadius / glowSpreadRadius / glowOpacity โ€” widget-level values take precedence over the theme.

Defaults preserve existing visual behaviour #

glowSpreadRadius and glowOpacity default to 0 and 1 respectively, preserving previous rendering. glowBlurRadius defaults to 4.0 โ€” a soft, natural halo that better fits the liquid-glass aesthetic. MaskFilter.blur is guarded at zero so there is no GPU cost when the value is left at 0. Set glowBlurRadius: 0 explicitly for a hard-edge disc.

Affected widgets #

All widgets that render GlassGlow consume these fields, including: GlassButton, GlassBottomBar, GlassSearchableBottomBar (both the tab pill and the search pill), GlassSlider, GlassSwitch.

Zero breaking changes #

Existing code that does not set these fields continues to render identically. copyWith, ==, and hashCode all include the three new fields.


0.9.0 #

โœจ New โ€” tabWidth on GlassBottomBar #

tabWidth is now available on both GlassBottomBar and GlassSearchableBottomBar. Both bar variants share identical compact-sizing semantics and the same default.

API #

GlassBottomBar(
  // Default (no tabWidth): expand โ€” pill fills available space.
  // tabWidth: 88.0 โ†’ iOS 26 compact sizing
  tabWidth: 88.0,
  ...
)
tabWidth Behaviour 2 tabs 3 tabs 4 tabs
null (default) Expand โ€” fills available space fills bar fills bar fills bar
88.0 Compact โ€” iOS 26 style 176 px 264 px 352 px

The pill is automatically clamped so it never overflows its container, regardless of how many tabs are present or how narrow the screen is.

Zero breaking changes #

tabWidth defaults to null (expand) on both GlassBottomBar and GlassSearchableBottomBar. Existing code that does not pass tabWidth continues to behave exactly as before โ€” the tab pill fills the bar. Pass tabWidth: 88.0 to opt-in to iOS 26 compact sizing.

Shared infrastructure (internal) #

  • bar_layout_utils.dart โ€” new pure-Dart file containing resolveTabPillWidth. Both GlassBottomBar and SearchableBottomBarController delegate to this single function, eliminating two separate inline implementations of the same arithmetic.
  • kBottomBarGlassDefaults โ€” the 9-field LiquidGlassSettings constant that was previously copy-pasted into both bar state classes is now defined once in bottom_bar_internal.dart and referenced from both locations.

Production hardening #

  • Extra button pinned to trailing edge in GlassBottomBar. Previously the extra button sat immediately adjacent to the tab pill when using compact tabWidth sizing, leaving empty space to its right. It is now always pinned to the far-right edge (using Expanded + Align(centerRight)) to match the searchable bar's layout. The maxTabW arithmetic is unchanged; only the Row structure changed. Works correctly in both compact and expand modes.
  • resolveTabPillWidth guards against negative maxAvailable values (math.max(0.0, maxAvailable) before the clamp) to prevent a RangeError in unusual layout constraint environments.
  • Both constructors now assert tabWidth == null || tabWidth > 0 โ€” passing a negative value previously produced a zero-width pill silently.
  • Golden regression sentinel added for tabWidth: null (expand mode), so a layout regression in legacy behaviour is caught by the pixel-test suite.

Example #

example/lib/tab_width_demo.dart โ€” covers both GlassBottomBar and GlassSearchableBottomBar via a Bar variant chip, with live metrics showing the computed pill width in real time.


0.8.4 #

CI & Tooling #

  • CI: Multi-platform test matrix. The CI pipeline now runs the full test suite on ubuntu-latest, macos-latest, and windows-latest across both stable and beta Flutter channels. Previously only macos-latest / stable was tested, which silently allowed the three Windows shader regressions shipped in 0.7.9โ€“0.7.12. Fail-fast is disabled so all platform failures are visible in a single run.

  • CI: Windows shader validation gate. glslangValidator (the same SPIR-V compiler core Flutter uses on Windows) now runs in CI on every push and PR via the shader-validation job. Any shader that would produce a "index expression must be constant" or "loop bounds must be compile-time constants" error is caught before it reaches main. Previously this check only ran locally via bash scripts/validate_shaders.sh on macOS.

  • CI: pub.dev publish dry-run gate. A dedicated pub-check job runs dart pub publish --dry-run on every push and PR. Catches missing dartdoc comments, pubspec.yaml issues, platform declaration gaps, and score regressions before they land in a release.

  • CI: Coverage threshold guard (โ‰ฅ 90 % effective). The pipeline now fails if effective line coverage drops below 90 % on the stable channel. Effective coverage is computed after stripping lib/src/renderer/* โ€” 16 GPU CustomPainter / RenderObject files that cannot execute in a headless VM (no GPU rasterizer; documented as untestable in ARCHITECTURE.md). Current effective coverage is 91.8 % (4 146 / 4 514 lines). A .codecov.yml config now mirrors this exclusion so the pub.dev / GitHub badge agrees with the CI gate rather than showing the raw ~81 % figure that included the untestable renderer paths.

  • CI: Run concurrency cancel. Added concurrency group so redundant in-progress runs on the same branch are cancelled automatically, saving CI minutes on rapid-push workflows.

  • Tooling: scripts/validate_shaders.sh cross-platform update. The shader validation script now resolves glslangValidator / glslangValidator.exe automatically, works on Windows (Git for Windows bash), and prints correct install instructions for macOS (brew), Ubuntu (apt-get), and Windows (choco / winget). Path resolution is now robust regardless of which directory the script is called from.

GlassAdaptiveScope Diagnostics (experimental) #

  • GlassAdaptiveDiagnostic โ€” rich quality change event. A new immutable data class is emitted whenever GlassAdaptiveScope changes quality tier. It carries the full context of why the change happened: from/to quality, reason (warmupComplete, thermalDegradation, thermalRecovery, restoredFromCache, staticProbe), phase, and the P75/P95 raster timing that triggered the decision.

  • GlassAdaptiveScope.onDiagnostic โ€” a new optional callback that receives a GlassAdaptiveDiagnostic alongside the existing onQualityChanged. The old callback is unchanged โ€” this is purely additive.

  • GlassAdaptiveScope.debugLogDiagnostics: true โ€” zero-wiring diagnostic mode. Add this flag to print a structured console block on every quality change in debug builds (no-op in profile/release). Designed to lower the barrier for community threshold calibration reports:

    โ”Œโ”€ ๐Ÿ“Š GlassAdaptiveScope โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
    โ”‚  Change  : premium โ†’ standard
    โ”‚  Reason  : warmupComplete
    โ”‚  Phase   : runtime
    โ”‚  P75     : 14.2 ms
    โ”‚  Frames  : 10
    โ”‚
    โ”‚  ๐Ÿ“ฌ Post to: github.com/sdegenaar/liquid_glass_widgets/discussions
    โ””โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
    
  • GlassQualityChangeReason enum โ€” exported publicly so analytics pipelines can filter on specific event types (e.g. only log warmupComplete and skip restoredFromCache noise).

  • Adapter diagnostic tracking โ€” GlassQualityAdapter now records lastP75Ms, lastP95Ms, lastFramesMeasured, and lastChangeReason before every quality decision so the scope can snapshot them synchronously before the async addPostFrameCallback gap.

Bug Fixes #

  • FIX: Refraction inverted on Android (Pixel 7, Mali GPU, OpenGL ES emulator). On all devices where Impeller uses the OpenGL ES backend, the liquid glass refraction effect appeared to bend inward rather than outward โ€” content beneath the glass lens distorted toward the centre instead of away from it. The glass bottom bar, segmented control indicator, and all premium-quality glass surfaces were affected.

    Root cause: OpenGL ES stores render-to-texture outputs with a bottom-left Y origin (Y increases upward), whereas Flutter's widget coordinate system uses Y-down. The shaders already flip screenUV.y and geometryUV.y with 1.0 โˆ’ y to compensate when sampling textures. However, the displacement vector (in liquid_glass_final_render.frag) and edgeOffsetLogical (in interactive_indicator.frag) were computed in Flutter's Y-down space and added directly to the Y-up UV without correcting the Y component. A positive Y displacement (outward at the bottom edge) therefore moved the sample toward the centre in UV space โ€” the exact opposite of the intended direction.

    Fix: Under #ifdef IMPELLER_TARGET_OPENGLES, negate the Y component of the displacement/offset vector before applying it to the sampled UV. This re-aligns the Y-down displacement with the Y-up UV coordinate space.

    The Metal (iOS/macOS) and Vulkan (Samsung S22 / Adreno / AMD Xclipse) code paths are unchanged โ€” the fix is gated entirely by IMPELLER_TARGET_OPENGLES and verified against both a Pixel 7 API 35 emulator and a physical Samsung Galaxy S22.


0.8.3 #

Performance & Bug Fixes #

  • GlassBottomBar / GlassSearchableBottomBar โ€” glass lens now correctly refracts active tab icons. Previously the selected icon layer was rendered above the AnimatedGlassIndicator in a separate compositor layer, making it invisible to the BackdropFilter. The glass pill swept over a blank canvas, producing a flat, unrefracted active icon. Both the selected and unselected icon layers are now combined into a single RepaintBoundary placed behind the glass lens, so all icon colours are physically sampled and warped by the chromatic aberration as the pill moves โ€” matching iOS 26 behaviour.

  • Performance improvement. The fix eliminates 5โ€“9 redundant GPU compositor layers per bar render frame: the per-tab RepaintBoundary nodes on both the selected and unselected icon rows have been removed in favour of a single shared compositor texture for the entire icon canvas. Fewer texture uploads, one BackdropFilter sample โ€” net improvement at 120 Hz.


0.8.2 #

Bug Fixes #

  • GlassQuality.premium no longer crashes outside a LiquidGlassLayer. Previously caused an opaque Null check operator crash. Now throws a descriptive AssertionError in debug builds and falls back gracefully (renders child without glass) in release. Fix: add useOwnLayer: true to any standalone GlassButton using premium quality.

  • GlassBottomBar / GlassSearchableBottomBar โ€” repeat-tap on active tab now fires onTabSelected (#22). Previously the index != widget.tabIndex guard silently suppressed callbacks when the user tapped the already-selected tab, making it impossible to implement scroll-to-top or refresh-on-retap patterns. The guard has been removed; onTabSelected is now always called once per gesture lifecycle regardless of whether the tab index changes.

  • GlassBottomBar / GlassSearchableBottomBar โ€” drag-end snaps to correct tab (#23). A coordinate-space mismatch in _onDragEnd caused the indicator to snap to the wrong tab: dragging to the centre of a 5-tab bar landed on tab 3 instead of tab 2. The fix corrects the inversion formula to i = round(relX ร— (n โˆ’ 1)), which is the exact inverse of the alignment space computeAlignment(i, n) = โˆ’1 + 2i/(nโˆ’1).

  • GlassBottomBar / GlassSearchableBottomBar โ€” onTabSelected no longer fires twice per tap. BottomBarTabItem had its own onTap: () => onTabSelected(i) callback that fired independently of the outer TabIndicator's onTapDown handler, causing every tap to call onTabSelected twice. The item-level callback is now null; the outer indicator is the single source of truth for all selection events.

    Credit: These interaction fixes were identified and originally patched by @qinshah in PR #23. The implementation was refactored to preserve the existing jelly physics, desktop tap support, and fling-based navigation that the PR removed, and extended to cover GlassSearchableBottomBar with shared logic via the new internal TabDragGestureMixin.

API #

  • GlassSearchBarConfig.expandWhenActive (new). Controls whether the search pill expands when isSearchActive is true. Default true โ€” no change needed for standard usage. Set to false for advanced layouts (e.g. Apple Music Play Pill pattern) where the search pill should remain compact while isSearchActive drives a non-search transition independently.

Examples #

  • apple_music_demo โ€” added as a reference for the Play Pill pattern: a floating GlassButton (useOwnLayer: true, GlassQuality.premium) that animates between a full-screen player and a mini-mode docked pill using AnimatedPositioned + AnimatedOpacity, synchronized with GlassSearchableBottomBar's spring morph via expandWhenActive.

0.8.1 #

New Features #

GlassInteractionBehavior โ€” precise, orthogonal control of press interactions #

A new first-class enum that independently controls the two dimensions of press feedback on GlassBottomBar, GlassSearchableBottomBar, and GlassTextField (as well as its derivative inputs):

Value Glow Scale
none โœ— โœ—
glowOnly โœ“ โœ—
scaleOnly โœ— โœ“
full (default) โœ“ โœ“

The glow is the iOS 26-style directional light spotlight that follows the touch position across the glass surface. The scale is the spring-physics size pulse on press.

// Glow only โ€” light follows your finger, no bounce:
GlassBottomBar(
  interactionBehavior: GlassInteractionBehavior.glowOnly,
  ...
)

// Scale only โ€” spring bounce, no glow:
GlassSearchableBottomBar(
  interactionBehavior: GlassInteractionBehavior.scaleOnly,
  pressScale: 1.06,
  ...
)

// Disable both for a completely static bar:
GlassBottomBar(
  interactionBehavior: GlassInteractionBehavior.none,
  ...
)

Zero overhead when disabled. When interactionBehavior suppresses glow (none or scaleOnly), the GlassGlow sensor widget is removed from the tree entirely โ€” saving 3 widget allocations and 3 RenderBox nodes per tab indicator per frame. Scale is resolved at build time to a scalar 1.0 with no animation controller overhang.

New parameters on GlassBottomBar, GlassSearchableBottomBar, and GlassTextField #

GlassTextField now shares the same interactionBehavior API as the bar-family widgets. The scale dimension maps onto the subtle press-bounce animation (field squishes slightly when pressed down); the glow dimension is the directional spotlight that tracks touch position across the glass surface.

GlassPasswordField and GlassTextArea delegate to GlassTextField and inherit the new parameter automatically.

Parameter Widget(s) Type Default
interactionBehavior All three GlassInteractionBehavior .full
pressScale Bar widgets / Inputs double 1.04 (bars) / 1.03 (inputs)
interactionGlowColor Bar widgets Color? null (theme default)
glowColor GlassTextField Color? null (~12% white)
interactionGlowRadius Bar widgets double 1.5
glowRadius GlassTextField double 1.5

All defaults preserve existing 0.8.0 visual behaviour โ€” no migration required.

Migration from enableGlow / enableFocusAnimation

GlassTextField.enableGlow and GlassTextField.enableFocusAnimation have been replaced by interactionBehavior. The mapping is direct:

// Before (0.8.0):
GlassTextField(enableGlow: false, enableFocusAnimation: false)

// After (0.8.1):
GlassTextField(interactionBehavior: GlassInteractionBehavior.none)

// Before: glow only
GlassTextField(enableGlow: true, enableFocusAnimation: false)
// After:
GlassTextField(interactionBehavior: GlassInteractionBehavior.glowOnly)

Bug Fixes #

  • FIX: SearchPill was silently ignoring interactionBehavior. The interactionGlowColor parameter was never passed to the SearchPill constructor, so the search pill always rendered with a visible glow regardless of the bar's interactionBehavior setting. The glow was hardcoded to Color(0x1FFFFFFF) even when behavior = none.

  • FIX: SearchPillState had no glow short-circuit on the expanded pill path. Added _wrapWithGlow helper (matching the pattern already in TabIndicatorState and SearchableTabIndicatorState) to skip GlassGlow allocation when glow is suppressed.


0.8.0 #

New Features #

GlassAdaptiveScope (experimental) โ€” automatic runtime quality adaptation #

A new scope widget that automatically adjusts GlassQuality for its subtree based on real raster performance observed from SchedulerBinding frame timings. Handles the three device scenarios that are impossible to test on a developer device:

  • Broken / slow shader drivers (e.g. Pixel 4a, Galaxy A22 class): detected synchronously at startup via ImageFilter.isShaderFilterSupported and capped immediately to minimal.
  • Warm-up jank ("wrong quality at startup"): resolved by a ~180-frame benchmark that measures real P75 raster durations and sets the initial quality tier before the user notices.
  • Thermal throttling ("fine at launch, janky after 10 minutes"): detected and corrected by a continuous runtime hysteresis engine.

Three-phase adaptation:

Phase Trigger Action
Phase 1 โ€” Static probe Mount Forces minimal on unsupported hardware; caps at standard on web
Phase 2 โ€” Warm-up First ~180 frames (~3 s at 60 fps) Sets initial quality from real P75 raster durations
Phase 3 โ€” Runtime hysteresis Ongoing Degrades after 3 bad windows; recovers after 10 good windows (8 s cooldown)

The scope acts as a quality ceiling โ€” widgets with an explicit quality: parameter are unaffected. The ceiling is enforced by GlassThemeHelpers.resolveQuality, which reads GlassAdaptiveScopeData from the nearest ancestor scope.

// Per-screen control:
GlassAdaptiveScope(
  child: Scaffold(...),
)

// Advanced โ€” conservative start for fragmented Android market:
GlassAdaptiveScope(
  initialQuality: GlassQuality.standard, // earn your way up to premium
  allowStepUp: true,
  onQualityChanged: (from, to) => analytics.log('glass_quality_changed'),
  child: child,
)

Experimental in 0.8.0. GlassAdaptiveScope and GlassAdaptiveScopeConfig are annotated @experimental. The three-phase adaptation logic is architecturally sound and fully tested, but the Phase 2 timing thresholds (P75 < 12 ms โ†’ premium, 12โ€“20 ms โ†’ standard, > 20 ms โ†’ minimal) have been validated by reasoning, not yet by broad real-device data across the Android fragmentation landscape.

How to enable it: LiquidGlassWidgets.wrap(myApp, adaptiveQuality: true) (opt-in, default false).

If you observe unexpected behaviour โ€” quality too low on a mid-range device, or stuck at standard on a flagship โ€” please file an issue with your device model and raster timings from Flutter DevTools. Your data will be used to tune the thresholds for a future release.

GlassAdaptiveScopeConfig (experimental) โ€” portable configuration value object #

Bundles all GlassAdaptiveScope parameters into a single const-constructible, equality-comparable value object. Used by LiquidGlassWidgets.wrap() and useful for passing scope configuration through APIs that cannot accept widget parameters directly.

const config = GlassAdaptiveScopeConfig(
  initialQuality: GlassQuality.standard,
  allowStepUp: true,
  targetFrameMs: 8, // 120 Hz ProMotion
);

API Refactor โ€” initialize() and wrap() separation #

The responsibilities of initialize() and wrap() have been clarified and made consistent with the broader Flutter ecosystem (cf. easy_localization, MaterialApp):

Method Responsibility
initialize() Async platform / engine setup only (shader prewarming, Impeller pipeline, debug monitor)
wrap() Widget-tree composition and all behavioral configuration

wrap() โ€” new parameters #

runApp(LiquidGlassWidgets.wrap(
  const MyApp(),
  respectSystemAccessibility: false, // moved from initialize()
  adaptiveQuality: true,             // new โ€” inserts GlassAdaptiveScope
  adaptiveConfig: GlassAdaptiveScopeConfig(
    initialQuality: GlassQuality.standard,
    allowStepUp: true,
  ),
));

Scope nesting order inserted by wrap() #

GlassAdaptiveScope โ†’ GlassBackdropScope โ†’ child

Breaking Changes #

initialize(respectSystemAccessibility:) removed #

respectSystemAccessibility has moved from initialize() to wrap().

Migration (one-line change):

// Before (0.7.x):
await LiquidGlassWidgets.initialize(respectSystemAccessibility: false);
runApp(LiquidGlassWidgets.wrap(const MyApp()));

// After (0.8.0):
await LiquidGlassWidgets.initialize();
runApp(LiquidGlassWidgets.wrap(const MyApp(), respectSystemAccessibility: false));

The LiquidGlassWidgets.respectSystemAccessibility getter and setter remain available as an escape hatch for tests and advanced runtime overrides. In production code, set it through wrap().

Bug Fixes #

Glass invisible on white / light backgrounds (transparency regression) #

  • FIX: Standalone glass widgets (GlassButton, GlassContainer, GlassTextField, GlassCard, and all widgets that delegate to them) rendered with zero opacity on light backgrounds when no explicit settings: were provided. Root cause: these widgets fell through to InheritedLiquidGlass.ofOrDefault(), which returns LiquidGlassSettings() โ€” a default with glassColor: Color(0x00FFFFFF) (alpha = 0). The lightweight shader computes body tint = glassColor.alpha ร— 0.15, so 0 ร— 0.15 = 0 โ€” the glass body was literally transparent regardless of thickness or blur.

    Fix: Replaced all InheritedLiquidGlass.ofOrDefault() call sites with the new GlassThemeHelpers.resolveSettings(), which traverses the full 5-level priority chain:

    1. Widget-level settings: parameter (explicit wins)
    2. InheritedLiquidGlass โ€” nearest parent AdaptiveLiquidGlassLayer
    3. LiquidGlassWidgets.globalSettings โ€” app-level override
    4. GlassThemeData โ€” brightness-aware theme variant (light / dark)
    5. LiquidGlassSettings() โ€” absolute last resort

    Standalone widgets now correctly resolve to the theme's glassColor and are always visible out of the box.

Light theme defaults rebalanced #

  • TWEAK: GlassThemeVariant.light updated for an icy-frosted aesthetic that reads clearly on white backgrounds:

    Property Before After
    blur 10.0 6.0
    glassColor 0x73FFFFFF (45% neutral white) 0x4AD2DCF0 (~29% cool blue-white)
    chromaticAberration 0.1 0.3
    thickness 16.0 20.0
    lightIntensity 1.0 1.2

    The cool blue-white tint (D2DCF0) matches the icy tone of iOS 26 frosted glass. Blur 6 gives visible background diffusion without obscuring content.

API #

GlassBackdropScope now exported from the main barrel #

  • FIX: GlassBackdropScope was missing from liquid_glass_widgets.dart. Consumers had to use the internal path package:liquid_glass_widgets/widgets/shared/glass_backdrop_scope.dart, which is fragile and undocumented. It is now a first-class public export.

    Migration โ€” update any direct internal imports:

    // Before (workaround, fragile):
    import 'package:liquid_glass_widgets/widgets/shared/glass_backdrop_scope.dart';
    
    // After (correct):
    import 'package:liquid_glass_widgets/liquid_glass_widgets.dart';
    
  • CHORE: add CI and Codecov badges.

0.7.16 #

Bug Fixes #

  • FIX: GlassSearchableBottomBar โ€” memory leak when controller was swapped at runtime. The old controller's listener was never removed before attaching to the new controller. Now correctly removed in didUpdateWidget.
  • FIX: DraggableIndicatorPhysics โ€” velocity NaN/Infinity guard. A zero-size render box (e.g. during widget tree warm-up) could produce Infinity or NaN for velocityX, which propagated into the spring physics and caused erratic snapping. Now clamped to 0 when the box has no size.

Refactor (zero breaking changes) #

  • REFACTOR: Extracted GlassSearchBarConfig from glass_searchable_bottom_bar.dart into a dedicated file lib/widgets/surfaces/shared/glass_search_bar_config.dart. Resolves a circular import between the public widget and its internal sub-widgets. GlassSearchBarConfig is re-exported from the barrel file โ€” no consumer-facing API change.
  • REFACTOR: Extracted _TabIndicator / _TabIndicatorState from glass_bottom_bar.dart into shared/bottom_bar_internal.dart as TabIndicator / TabIndicatorState (package-internal, not exported). Follows the same pattern used for GlassSearchableBottomBar. glass_bottom_bar.dart reduced from 1,406 โ†’ ~895 lines.
  • REFACTOR: Extracted _TabBarContent, _TabBarContentState, and _TabItem from glass_tab_bar.dart into shared/tab_bar_internal.dart. glass_tab_bar.dart reduced from 728 โ†’ ~310 lines. Architecture is now consistent across all bar-family widgets.

Test Coverage #

  • TEST: Reached 91.85% effective coverage (up from 89.6% in 0.7.15 โ€” excluding GPU/shader renderer paths that are physically untestable in a headless VM). Total: 1,031 tests, all passing, 0 analyzer warnings.
  • TEST: New test/widgets/surfaces/glass_bottom_bar_drag_test.dart โ€” 7 regression tests covering _onDragEnd physics snapping, _onDragCancel (mid-drag and no-drag), slow drags, fast flings, and full-bar sweeps. These paths are the highest-risk regressions in navigation UX.

0.7.15 #

Bug Fixes #

  • FIX: lib/theme/glass_theme_settings.dart was accidentally omitted from version control in 0.7.14. All consumers of GlassThemeSettings received a compile error (type 'GlassThemeSettings' is not a subtype). This release commits the missing file. No API change โ€” GlassThemeSettings was already exported from liquid_glass_widgets.dart.
  • FIX: GlassPerformanceMonitor._emitWarning โ€” division-by-zero crash when rasterBudget was sub-millisecond (< 1 ms). Protected with a max(1, ...) guard.

Refactor (zero breaking changes) #

  • REFACTOR: Consolidated 18 quality-resolution chains (widgetQuality ?? inherited?.quality ?? themeData.qualityFor(context) ?? GlassQuality.standard) into a single canonical helper: GlassThemeHelpers.resolveQuality(context, widgetQuality: ..., fallback: ...). Surface widgets (GlassAppBar, GlassToolbar, GlassBottomBar, GlassSearchableBottomBar, GlassSideBar) pass fallback: GlassQuality.premium to preserve their documented defaults. All other widgets default to GlassQuality.standard.
  • REFACTOR: Extracted _buildIconShadows from BottomBarTabItem to a @visibleForTesting top-level function buildIconShadows(...) in bottom_bar_internal.dart. No behaviour change โ€” enables isolated unit testing of the shadow-outline geometry.

Test Coverage #

  • TEST: Reached 90%+ effective test coverage (90.15% โ€” excluding src/renderer GPU/shader layer where headless simulation is impossible). Total: 949 tests, all passing.
  • TEST: New test/theme/glass_theme_helpers_test.dart โ€” 5 widget tests covering all 4 priority levels of GlassThemeHelpers.resolveQuality().
  • TEST: New test/widgets/surfaces/build_icon_shadows_test.dart โ€” 6 unit tests covering buildIconShadows(): null thickness, active-icon suppression, shadow count, 45ยฐ offset math, and color propagation.
  • TEST: Added test/theme/, test/renderer/, test/types/, test/constants/, test/utils/, and test/widgets/ test suites (committed for the first time โ€” these were written during the 0.7.13โ€“0.7.14 coverage push but never staged).

0.7.14 #

Bug Fixes #

  • FIX: GlassSearchableBottomBar โ€” extraButton now fades out smoothly when search activates instead of being visually clipped/shrunk between the collapsing tab pill and the expanding search pill. Layout space is still reserved during the morph (no pills jump), only the visual opacity transitions. Taps on the extra button are also correctly blocked while hidden. Fades in when search closes.
  • FIX: GlassSearchableBottomBar โ€” spring morph animations no longer produce a visible jump when reversing direction. Previously the three spring controllers (tabW, searchLeft, searchW) were each started in separate addPostFrameCallback calls, introducing a 1-frame desync at reversal. All three are now started in a single batched callback, so the morph is perfectly synchronized in both directions.
  • FIX: Indicator fade animation in GlassBottomBar / GlassSearchableBottomBar โ€” replaced Opacity wrapper with LiquidGlassSettings.visibility fading. Wrapping a BackdropFilter in Opacity composites into an offscreen buffer, breaking backdrop sampling and causing the indicator to snap in/out instead of fading. The visibility path is a single GPU pass โ€” no offscreen buffer โ€” improving drag animation performance and working uniformly for all blur values.
  • FIX: GlassBottomBar, GlassSearchableBottomBar, GlassAppBar, GlassToolbar, and GlassSideBar resolved to GlassQuality.standard instead of their documented GlassQuality.premium default. Fixed by setting quality: null in the built-in light/dark variants so each widget's documented default is respected.
  • FIX: Setting any property in GlassThemeVariant.settings silently zeroed out all unset properties (e.g. setting only thickness: 50 also reset glassColor to fully transparent). Fixed by introducing GlassThemeSettings: a parallel class with all-nullable fields that merges onto each widget's own defaults. Only the fields you explicitly set are applied; everything else inherits from the widget. GlassThemeVariant.settings now accepts GlassThemeSettings?.
  • FIX: GlassSearchableBottomBar โ€” multiple layout-math regressions in the morph animation corrected:
    • Reserved layout width now correctly scales to min(size, searchBarHeight) during search, eliminating the bloated gap when searchBarHeight < barHeight.
    • Extra button rendered width now matches the layout reserve (extraTargetW), preventing a 14 px overflow into the search pill when searchBarHeight < barHeight.
    • Restored + widget.spacing in targetSearchLeft; an erroneous tabToNextGap variable had suppressed the gap between the tab pill and search pill when no extra button was present.
    • collapseOnSearchFocus now exclusively controls visibility/opacity โ€” it no longer affects layout geometry. Toggling it mid-animation no longer triggers the spring or causes the button to jump inside the collapsed tab circle.
  • FIX: BottomBarTabItem โ€” removed a fixed vertical: 4 padding wrapping the tab column. The padding consumed constraint space before FittedBox could scale, causing a 2 px RenderFlex overflow when the bar morphed to searchBarHeight.

New #

  • NEW: GlassThemeSettings โ€” a partial settings type for use in GlassThemeVariant. Accepts the same parameters as LiquidGlassSettings but all are nullable. Only non-null fields override the target widget's defaults, enabling precise single-property theme overrides without disturbing others.
  • NEW: GlassTabPillAnchor enum + GlassSearchableBottomBar.tabPillAnchor โ€” controls how the tab pill is anchored during the morph animation. GlassTabPillAnchor.start (default) preserves existing left-anchor behaviour. GlassTabPillAnchor.center makes both edges collapse symmetrically from the pill's centre for a more balanced look. The search pill position adjusts automatically in center mode.
  • NEW: GlassSearchBarConfig.showsCancelButton now defaults to true. Tapping the dismiss pill unfocuses the keyboard and collapses search, matching the system-level behaviour seen across iOS apps (Weather, App Store, Apple News). Pass showsCancelButton: false to opt out.
  • NEW: GlassSearchBarConfig.collapsedTabWidth is now nullable. When omitted, the collapsed tab pill automatically matches GlassSearchableBottomBar.searchBarHeight, ensuring it morphs into a geometric circle with no leftover horizontal margin. Pass an explicit value to override.
  • NEW: GlassBottomBarExtraButton.collapseOnSearchFocus (default true) โ€” controls whether the extra button collapses when the search field is focused. When true, the button fades out and its layout space spring-animates to zero, giving the search input the full available width (matching native iOS behaviour). When false, the button remains fully visible and tappable alongside the search input โ€” useful for contextually relevant actions like a Filter button that applies to search results.
  • EXAMPLE: searchable_bar_repro.dart added to the example app โ€” exercises GlassSearchableBottomBar edge cases (extra-button fade, spring desync, bar-height scale, dismiss pill) in isolation. Run standalone: flutter run -t example/lib/searchable_bar_repro.dart.

0.7.13 #

New โ€” GlassQuality.minimal #

  • FEAT: GlassQuality.minimal โ€” third quality tier: a crisp frosted glass surface with zero custom fragment shader execution on any platform. Uses BackdropFilter blur

    • Rec. 709 saturation matrix + a light-angle specular rim stroke. No refraction warping or chromatic aberration โ€” a deliberately flat, clean aesthetic that looks excellent on any background and never adds GPU shader cost.

    Two distinct use cases:

    Device fallback โ€” for hardware where even [standard] is too heavy: very old Android devices with limited shader driver support, or any device where ImageFilter.isShaderFilterSupported returns false.

    GPU budget management โ€” for shader-dense screens: use [minimal] for background panels, list cards, and decorative containers while keeping [standard] or [premium] on the focal element. A screen with 15 glass list cards running [minimal] fires zero shader invocations during scroll โ€” only BackdropFilter compositing.

    AdaptiveGlass(
      quality: GlassQuality.minimal,
      child: child,
    )
    
  • FEAT: GlassThemeVariant.minimal โ€” static preset that applies .minimal quality globally via GlassThemeData:

    GlassTheme(
      data: GlassThemeData(
        light: GlassThemeVariant.minimal,
        dark:  GlassThemeVariant.minimal,
      ),
      child: child,
    )
    

New โ€” GlassPerformanceMonitor #

  • FEAT: Debug/profile-only performance monitor that watches raster frame durations while GlassQuality.premium surfaces are active. When frames exceed the GPU budget for 60 consecutive frames, a single FlutterError is emitted with actionable guidance (specific widget parameters, device compatibility notes, and alternative quality tiers).

    Zero production overhead โ€” the monitor never registers a callback in release builds. Enabled by default in debug/profile builds via LiquidGlassWidgets.initialize():

    // Default: auto-enabled in debug/profile, zero-cost in release
    await LiquidGlassWidgets.initialize();
    
    // Opt out:
    await LiquidGlassWidgets.initialize(enablePerformanceMonitor: false);
    
    // Custom thresholds (advanced):
    GlassPerformanceMonitor.rasterBudget = const Duration(microseconds: 8333); // 120 fps
    GlassPerformanceMonitor.sustainedFrameThreshold = 120; // 2 seconds at 60 fps
    

    The monitor correctly attributes slowdowns to premium glass by counting active GlassQuality.premium surfaces. It stays silent when no premium widgets are mounted, avoiding false positives from other parts of the app.


0.7.12 #

Bug Fixes #

  • FIX: Interactive blend-group stretch asymmetry โ€” LiquidStretch now expands geometry symmetrically from the widget centre, fixing the left-leans-in / right-resists imbalance during touch-drag on button groups.

  • FIX: Erroneous highlight bias โ€” removed a legacy shader hack that skewed surface normals horizontally. Normals are now derived accurately from the SDF gradient, eliminating optical hotspots that made straight groups look crooked.

  • PERF: Zero-jitter animation bounds โ€” geometry texture mapping is now strictly bound to the physical size it was rasterised for, stopping frame-lag wobble when buttons change scale during interactive drags.

  • FIX: Theme quality cascade โ€” audited 15+ widgets (GlassBottomBar, GlassSwitch, GlassTextField, and others) that were silently overriding the global GlassThemeVariant quality setting with GlassQuality.premium. All widgets now correctly inherit and respect the global quality profile, protecting frame rate and thermal limits on older devices (e.g. iPhone 12 and below).

  • FIX: Zero-thickness blur โ€” setting thickness: 0 no longer makes the glass fully transparent. Backdrop blur now renders correctly on glass surfaces regardless of geometric thickness, restoring backward-compatible behaviour.

  • FEAT: GlassSearchBarConfig.focusNode โ€” optional FocusNode for GlassSearchBarConfig. When provided, the caller has full programmatic focus control (requestFocus(), unfocus(), addListener()) independent of autoFocusOnExpand. The widget adopts the caller-provided node without disposing it (caller owns lifecycle), matching Flutter's own TextField.focusNode contract.

  • FEAT: GlassSearchBar.focusNode โ€” same FocusNode support added to the standalone GlassSearchBar for consistency. GlassTextField already had this.

  • FIX: ExtraButtonPosition โ€” new enum on GlassBottomBarExtraButton. Set .position = ExtraButtonPosition.afterSearch to pin the extra button to the right of the search pill. Spring geometry calculations reserve space correctly to prevent RenderFlex overflows during expand/collapse. Default is ExtraButtonPosition.beforeSearch โ€” fully backwards-compatible.

  • FIX: Windows / SkSL shader compilation โ€” eliminated all dynamic array index expressions from sdf.glsl. The previous getShapeSDFFromArray(int index) computed offsets at runtime, which SkSL/glslang on Windows rejects with "index expression must be constant". Replaced with literal-indexed sdf0()โ€ฆsdf15() helpers and a fully-unrolled sceneSDF for 1โ€“16 shapes. MAX_SHAPES stays 16; no API or visual change.

  • TOOLING: scripts/validate_shaders.sh โ€” macOS script that validates all shaders against Windows/SkSL compiler rules using glslangValidator. Run bash scripts/validate_shaders.sh before releasing. Requires brew install glslang (one-time).


0.7.11 #

Bug Fixes #

  • FIX: Windows/Android build failure โ€” three shader compilation errors on the SPIR-V/glslang path: loop bounds must be compile-time constants; dFdx/dFdy on a scalar float is rejected by glslang (geometry shader now uses #ifdef IMPELLER_TARGET_METAL to keep hardware derivatives on iOS/macOS and fall back to ยฑ0.5 px finite differences on Vulkan/OpenGL ES); global non-constant initialisers at file scope in liquid_glass_final_render.frag moved into main().

  • FIX: Blend-group asymmetry โ€” the liquid-glass merge neck between grouped buttons leaned toward the left button. Fixed with a bidirectional smooth-union pass (Lโ†’R + Rโ†’L, averaged 50/50) that cancels the directional bias exactly.


0.7.10 #

Bug Fixes #

  • FIX: Windows build (flutter build windows) โ€” two shader issues fatal on SkSL/glslang but silently accepted on Metal: no match for min(int, int) (replaced with a ternary) and global non-constant initialisers (moved into main()). No visual change on any platform.

0.7.9 #

Bug Fixes #

  • FIX: Windows build failure โ€” uShapeData[MAX_SHAPES * 6] was passed as a by-value function parameter, which glslang rejects. Fixed by accessing it as a global uniform. No visual change.

Tweaks #

  • TWEAK: GlassSearchableBottomBar iOS 26 Apple News parity โ€” animated inline ร— clear button replaces microphone when text is present; simplified hit-testing layout replaces Overlay layers; guaranteed GPU liquid-glass merging between the search and dismiss pills in a single shader pass.

0.7.8 #

Tweaks #

  • TWEAK: GlassThemeVariant.light now defaults to a cool-tinted glassColor (Color(0x32D2DCF0)), stronger refractiveIndex, and boosted ambientStrength to ensure premium specular rendering and visible refraction on flat white backgrounds.

Examples #

  • Apple News demo โ€” replaced Image.network calls with pre-sized bundled assets (example/assets/news_images/) to fix Impeller GPU command-buffer overflow on iOS 26 physical devices.
  • Apple News demo โ€” collapsedLogoBuilder now mirrors the active tab icon instead of a static badge.

0.7.7 #

Refactor #

  • Internal: Removed GlassIndicatorTapMixin and migrated GlassTabBar and GlassSegmentedControl fully to raw Listener pointer events, matching GlassBottomBar's robust drag-cancel and press-and-hold handling. No API change.

0.7.6 #

Bug Fixes #

  • FIX: LiquidGlassBlendGroup asymmetry โ€” left buttons attracted their neighbours more strongly than right buttons in groups of 3+. Fixed with a bidirectional smooth-union pass (Lโ†’R + Rโ†’L, averaged 50/50). Two-shape groups are mathematically identical to before.

  • FIX: GlassButtonGroup โ€” glass effect could bleed as a dark rectangle on Impeller with GlassQuality.premium and useOwnLayer: true. A ClipRRect(antiAlias) now hard-clips the bleed at the superellipse boundary without forcing a quality downgrade.


0.7.5 #

Bug Fixes #

  • FIX: GlassBottomBar / GlassSearchableBottomBar โ€” added HitTestBehavior.opaque to the root GestureDetector so the full bar height reliably consumes pointer events on simulator and desktop.

  • FIX: GlassSearchableBottomBar โ€” keyboard no longer flickers on physical devices; focus is requested after the expansion animation completes.

  • FIX: GlassSearchableBottomBar โ€” dead zone at expanded search pill edges resolved; the full glass surface now claims taps and routes them to the search field.

New โ€” GlassSearchBarConfig parameters #

Seven new parameters (all backwards-compatible):

Parameter Type Default Description
autoFocusOnExpand bool false Keyboard opens automatically on expand.
trailingBuilder WidgetBuilder? null Replaces the mic icon with any custom widget.
textInputAction TextInputAction? null Keyboard action key (search, done, go, โ€ฆ).
keyboardType TextInputType? null Keyboard layout (url, emailAddress, โ€ฆ).
autocorrect bool true Disable for codes, usernames, etc.
enableSuggestions bool true Controls QuickType bar on iOS.
onTapOutside TapRegionCallback? null Called when user taps outside the field.

0.7.4 #

New Components #

  • GlassSearchableBottomBar โ€” GlassBottomBar with a morphing search pill that shares the same AdaptiveLiquidGlassLayer as the tab pill, producing iOS 26 liquid-merge blending. When isSearchActive is true the tab pill collapses and the search pill expands via spring animation. Configured via GlassSearchBarConfig.

Examples #

  • Apple News demo (example/lib/apple_news/apple_news_demo.dart) โ€” iOS 26 Apple News replica showcasing GlassSearchableBottomBar.

Visual Fixes #

  • FIX: Default glow color on press changed from iOS system blue to a brightness-adaptive neutral white (~35% light / ~22% dark), matching iOS 26 glass press behaviour.

0.7.3 #

Performance #

  • PERF: Deleted unused rotate2d() from render.glsl โ€” it was compiled into every shader binary but never called.
  • PERF: Eliminated a redundant normalize() in interactive_indicator.frag by reusing an already-computed length.
  • PERF: Removed a no-op canvas.save()/canvas.restore() pair in GlassGlow paint.

Bug Fixes #

  • FIX: GlassGlow tracking โ€” glow gradient is now correctly recreated each frame when glowOffset changes, fixing the spotlight freezing at its initial position.
  • FIX: Glow on Skia/Web โ€” LightweightLiquidGlass now wraps in GlassGlowLayer, giving the Skia path the same light-follows-touch behaviour as Impeller.
  • FIX: Glow on first touch โ€” spotlight now appears immediately at the tap position instead of sliding in from the widget's top-left corner.
  • FIX: Glow tracking inside button groups โ€” converted from widget-local to global coordinates so the spotlight correctly follows touches regardless of nesting depth.
  • FIX: Glow radius on wide buttons โ€” switched from shortestSide to โˆš(width ร— height) so the spotlight scales proportionally to the button area.

0.7.2 #

Performance & Polish #

  • PERF: Lightweight shader (lightweight_glass.frag) โ€” reduced ALU instruction count ~10โ€“15 ops per fragment; restored the normalZ Fresnel ramp to sqrt(1 โˆ’ dot(n,n)).
  • PERF: Impeller final render shader โ€” eliminated length()/normalize() from anisotropic specular; made getHeight() fully branchless; collapsed four step() multiplications into one.
  • PERF: Dart side โ€” cached light direction trig in LiquidGlassRenderObject (only recomputed when lightAngle changes); changed GlassGroupLink.shapeEntries from List to Iterable to eliminate per-frame heap allocation.
  • FIX: Adjusted GlassBottomBar, GlassTabBar, and GlassSegmentedControl spring from 500ms bouncySpring to 350ms snappySpring, matching iOS 26 segment-indicator physics.

0.7.1 #

Bug Fixes #

  • FIX: GlassBottomBar, GlassTabBar, GlassSegmentedControl โ€” rapid taps no longer prematurely snap the indicator, killing spring physics. Removed pixel-snapping from onHorizontalDragDown so taps correctly use spatial distance for the iOS 26 jump animation.

0.7.0 #

New Components #

  • GlassDivider โ€” iOS 26-style hairline separator, horizontal and vertical. Theme-adaptive opacity (dark: 20% white / light: 10% black).
  • GlassListTile โ€” iOS 26 Settings-style row with leading icon, title, subtitle, trailing widget, and automatic grouped dividers. Use inside a zero-padding GlassCard. Convenience constants: GlassListTile.chevron, GlassListTile.infoButton.
  • GlassStepper โ€” iOS 26 UIStepper equivalent. Compact โˆ’/+ glass pill with auto-repeat on hold, min/max clamping, wraps cycling, fractional step, and haptic feedback.
  • GlassWizard + GlassWizardStep โ€” multi-step flow with numbered indicators, checkmarks, and expandable step content.

Accessibility #

  • GlassAccessibilityScope โ€” reads platform Reduce Motion and Reduce Transparency preferences and propagates them to all glass widgets in its subtree:
    • Reduce Motion: spring animations snap instantly.
    • Reduce Transparency: replaces the full glass shader pipeline with a plain BackdropFilter(blur) + frosted container.
  • Semantics updated across all remaining widgets to match iOS UIAccessibility conventions.

Performance #

  • PERF: GlassSpecularSharpness enum โ€” replaces pow(lightCatch, exponent) (two transcendentals per fragment) with a pure squaring chain in lightweight_glass.frag. Zero transcendentals. Default: .medium.
  • PERF: pow(x, 1.5) โ†’ xยทโˆšx in Impeller edge lighting โ€” sqrt() is a single hardware SFU instruction.
  • PERF: Anisotropic specular and Fresnel rim brightening ported from the Impeller path to lightweight_glass.frag, closing the largest visual gap between rendering paths.
  • PERF: Content-adaptive glass strength โ€” intensity auto-adjusts based on backdrop luminance on Impeller, or MediaQuery.platformBrightness on Skia/Web.

Developer Experience #

  • GlassRefractionSource โ€” renamed from LiquidGlassBackground to better reflect its role. LiquidGlassBackground remains as a deprecated typedef (removed in 1.0.0).
  • Synchronous background capture โ€” rebuilt using boundary.toImageSync() on native (zero CPUโ†”GPU readback) and async toImage() on web.

0.6.1 #

Visual Quality #

  • FIX: True surface normal storage in geometry texture โ€” the geometry pass now stores the SDF-gradient-derived surface normal instead of the refraction displacement vector. The render shader decodes and recomputes displacement via refract(). Specular highlights on blended glass shapes (e.g. two overlapping pills) now correctly follow true surface curvature rather than the refraction direction. Single-shape surfaces are visually identical to 0.6.0.
  • FIX: Anisotropic specular highlights (Impeller) โ€” specular lobe stretched 20% along the surface tangent, producing the horizontal oval highlight that matches iOS 26.
  • FIX: Fresnel edge luminosity ramp (Impeller) โ€” gentle brightness ramp at grazing angles matching iOS 26's centre-to-edge luminosity gradient.
  • FIX: Luminosity-preserving glass tint in lightweight shader โ€” replaced additive tint with the same applyGlassColor() model as the Impeller path: achromatic glass lifts toward white, chromatic glass shifts hue while preserving luminance.

Performance #

  • PERF: Branchless smoothUnion โ€” eliminated a conditional branch that caused warp divergence when glass shapes transition between merged and separate.
  • PERF: if/else if dispatch in shape SDF โ€” GPU now short-circuits after the first type match; default changed to 0.0 for a clearly visible failure mode.
  • PERF: Single texture fetch when chromatic aberration is disabled โ€” interactive_indicator.frag previously sampled the background three times unconditionally; 66% fewer texture reads in the common case.
  • PERF: Flat-interior early-exit in final render shader โ€” pixels where normalXY โ‰ˆ 0 skip refract() and all texture samples, replaced with a single background sample. Lossless.

0.6.0 #

Breaking Changes #

  • LiquidGlassLayer.useBackdropGroup removed. Glass layers now automatically detect a BackdropGroup ancestor. Remove useBackdropGroup: true from any LiquidGlassLayer(...) calls.

New Features #

  • LiquidGlassWidgets.wrap() โ€” wraps your app in a GlassBackdropScope in one line:
    runApp(LiquidGlassWidgets.wrap(const MyApp()));
    
  • GlassMotionScope โ€” drives glass specular angle from any Stream<double> (e.g. device gyroscope). No new dependencies required.

Performance #

  • PERF: GlassBackdropScope auto-activation โ€” glass layers automatically share a single GPU backdrop capture when a scope ancestor is present.
  • PERF: Local-space geometry rasterization โ€” geometry texture cached until pill size or shape changes, eliminating per-frame rebuilds during animation.
  • PERF: Shader UV bounds check โ€” discards fragments where geometry UV falls outside [0, 1], preventing the thin "protruding line" artifact during jelly-physics expansion.

Visual #

  • FIX: Refraction UV โ€” uses uSize uniform (always valid on first frame) instead of textureSize() which returns (0,0) on the first frame in Impeller.
  • FIX: precision highp float in final render shader (was mediump, risking colour banding on mobile).
  • FIX: iOS 26 glass tint model โ€” preserves backdrop luminance while shifting chroma. Replaces Photoshop Overlay mode.
  • FIX: Leading-dot rim artifact โ€” x / (1 + x) soft-clamping on highlight intensity prevents bright corner artifact during drag.
  • FIX: Impeller indicator clipping โ€” jelly physics animations no longer clip at the static bounding box (clipExpansion parameter added).
  • FIX: Web & WASM โ€” removed dart:io imports from shader resolution logic.

Dependencies #

  • Removed motor dependency โ€” replaced with self-contained glass_spring.dart. Zero third-party runtime dependencies beyond the Flutter SDK.

0.5.0 #

Breaking Changes #

LiquidGlass removed from the public API. It was inadvertently exposed and silently renders nothing on Skia/Web. Use AdaptiveGlass instead:

// Before
LiquidGlass(settings: LiquidGlassSettings(...), child: ...)

// After
AdaptiveGlass(settings: LiquidGlassSettings(...), child: ...)

LiquidGlassLayer, LiquidGlassBlendGroup, LiquidGlassSettings, LiquidShape, GlassGlow, and debugPaintLiquidGlassGeometry remain public.

New Features #

  • GlassBackdropScope โ€” halves GPU blur capture cost when multiple glass surfaces are on screen simultaneously. Wrap your MaterialApp or Scaffold to activate:
GlassBackdropScope(
  child: MaterialApp(
    home: Scaffold(
      appBar: GlassAppBar(...),
      bottomNavigationBar: GlassBottomBar(...),
    ),
  ),
)

Renderer #

The renderer from liquid_glass_renderer (whynotmake.it, MIT) is now vendored directly, giving full control over the rendering pipeline with no user-facing API changes.


0.4.1 #

Bug Fixes #

  • FIX: GlassBottomBar and other surfaces now correctly respond to dynamic glassSettings changes on GlassQuality.standard โ€” AdaptiveGlass in grouped mode now inherits settings from InheritedLiquidGlass instead of using empty defaults.
  • FIX: Luminance-aware ambient floor for white glass on GlassQuality.standard โ€” high-opacity white glass no longer renders as dark grey.

New #

  • FEAT: GlassBottomBar.iconLabelSpacing โ€” configurable vertical gap between tab icon and label (default: 4.0). Thanks @baneizalfe (#11).

Breaking Changes #

Library-wide IconData โ†’ Widget API migration. All icon parameters now accept any Widget:

// Before
GlassButton(icon: CupertinoIcons.heart, onTap: () {})

// After
GlassButton(icon: Icon(CupertinoIcons.heart), onTap: () {})
// Or any custom widget:
GlassButton(icon: SvgPicture.asset('assets/heart.svg'), onTap: () {})

GlassBottomBarTab.selectedIcon renamed to activeIcon to match Flutter's BottomNavigationBarItem convention.


0.4.0 #

New Components #

  • GlassMenu / GlassMenuItem / GlassPullDownButton โ€” iOS 26 morphing context menu with spring physics and position-aware expansion.
  • GlassButtonGroup โ€” joined-style container for related actions (e.g. Bold/Italic/Underline toolbar).
  • GlassFormField / GlassPasswordField / GlassTextArea / GlassPicker โ€” full iOS 26 input suite.
  • GlassSideBar โ€” vertical navigation surface with header, footer, and scrollable items.
  • GlassToolbar โ€” standard iOS-style action toolbar.
  • GlassTabBar โ€” horizontal tab navigation bar with animated indicator and scrollable mode for 5+ tabs.
  • GlassProgressIndicator โ€” circular and linear variants (indeterminate and determinate), iOS 26 specs.
  • GlassToast / GlassSnackBar โ€” 5 notification types, 3 positions, auto-dismiss, swipe-to-dismiss.
  • GlassBadge โ€” count and dot status badges, 4 positions.
  • GlassActionSheet โ€” iOS-style bottom-anchored action list.

Performance #

  • Universal Platform Support โ€” AdaptiveGlass and AdaptiveLiquidGlassLayer introduced. All 26 widgets deliver consistent glass quality on Web, Skia, and Impeller.
  • Batch-blur optimisation โ€” glass containers share a single BackdropFilter (was: one per widget). ~5ร— faster in common multi-widget layouts.
  • Impeller pipeline warm-up โ€” shaders pre-compile at startup to eliminate first-frame jank.

Theme System #

  • GlassTheme / GlassThemeData / GlassThemeVariant โ€” global styling and quality inheritance across all widgets. Set once, inherited everywhere.

0.3.0 โ€” 0.1.0 #

Early access and preview releases establishing the core widget library, initial glass rendering pipeline (LiquidGlass, LiquidGlassLayer, LiquidGlassBlendGroup), and foundational components (GlassBottomBar, GlassButton, GlassSwitch, GlassCard, GlassSearchBar, GlassSlider, GlassChip, GlassSegmentedControl, GlassSheet, GlassDialog, GlassIconButton).

161
likes
160
points
25.4k
downloads

Documentation

API reference

Publisher

unverified uploader

Weekly Downloads

A Flutter UI kit implementing the iOS 26 Liquid Glass design language. Features shader-based glassmorphism, physics-driven jelly animations, and dynamic lighting.

Repository (GitHub)
View/report issues

Topics

#liquid-glass #widget #glassmorphism #flutter #apple

License

MIT (license)

Dependencies

equatable, flutter, flutter_shaders, logging, meta

More

Packages that depend on liquid_glass_widgets