Class StickyHeaderContainer

java.lang.Object
com.codename1.ui.Component
com.codename1.ui.Container
com.codename1.components.StickyHeaderContainer
All Implemented Interfaces:
Animation, Editable, StyleListener, Iterable<Component>

public class StickyHeaderContainer extends Container

A scrollable container that pins the most recently scrolled-past section header to the top of its viewport, in the style of the iOS contacts list or sectioned material lists. As the next section's header rises into the pinned slot the previous header is replaced through a configurable scroll-driven transition: a directional slide where the rising header pushes the pinned one up and out, or a cover where the rising header progressively slides on top of the pinned one. The outgoing header can optionally fade out alongside either transition via setHeaderFadeOut(boolean). Transitions are driven by scroll position so the visual stays in sync with the user's gesture and there is no time-based animation that lags behind a slow drag or skips ahead on a fling.

Sections are added with addSection(header, content). The header is a real component that participates in the scroll: when it is the active section's header it is moved into a pinned overlay slot at the top of the container, and a same-sized invisible placeholder is left behind in the scroll content so nothing jumps. Because the pinned header is the same instance, action listeners and child components remain interactive while it is pinned.

StickyHeaderContainer sticky = new StickyHeaderContainer();
sticky.setTransitionStyle(StickyHeaderContainer.TRANSITION_SLIDE);
for (char c = 'A'; c <= 'Z'; c++) {
    Label header = new Label("" + c, "StickyHeader");
    Container items = new Container(BoxLayout.y());
    for (int i = 0; i < 5; i++) {
        items.add(new Label(c + " entry " + i));
    }
    sticky.addSection(header, items);
}
form.add(BorderLayout.CENTER, sticky);
  • Field Details

    • TRANSITION_NONE

      public static final int TRANSITION_NONE
      Replace the pinned header without any visible movement of the pinned header. The rising section's header stays hidden behind the pinned slot during the overlap and the swap is instant once it reaches the slot top. Generally TRANSITION_COVER is a more useful choice, since with TRANSITION_NONE the rising header disappears under the pinned one with no visual feedback that the swap is approaching. Kept for backward compatibility.
      See Also:
    • TRANSITION_SLIDE

      public static final int TRANSITION_SLIDE
      As the next section's header rises into the pinned slot from below it pushes the pinned header up and out of the slot in sync with the scroll, replacing it once the rising header reaches the top.
      See Also:
    • TRANSITION_FADE

      public static final int TRANSITION_FADE
      Convenience equivalent to TRANSITION_SLIDE combined with setHeaderFadeOut(boolean) set to true: the pinned header both slides up and fades to transparency as the rising header closes the gap. Prefer composing setTransitionStyle(TRANSITION_SLIDE) with setHeaderFadeOut(true) -- or setTransitionStyle(TRANSITION_COVER) with setHeaderFadeOut(true) -- when you want a fade-out on top of another transition. Kept for backward compatibility.
      See Also:
    • TRANSITION_COVER

      public static final int TRANSITION_COVER
      As the next section's header rises into the pinned slot from below it progressively slides on top of the pinned header, covering it from the bottom up while the pinned header stays fixed in the slot. The swap happens once the rising header reaches the slot top, at which point it has fully replaced the pinned one with no jump in position. Combine with setHeaderFadeOut(boolean) to also fade the covered header out during the overlap.
      See Also:
  • Constructor Details

    • StickyHeaderContainer

      public StickyHeaderContainer()
      Creates an empty sticky header container. Add sections via addSection(header, content).
  • Method Details

    • paint

      public void paint(Graphics g)
      Paints the COVER overlay on top of the pinned slot: when the transition style is TRANSITION_COVER and a push is in flight, the rising section's header is rendered again above the pinned host so it visibly slides over the pinned header instead of being hidden underneath it. The natural paint in the scroller is at the same viewport position, so re-painting the header after super.paint(g) simply moves it to the top of the z-order in the slot region; outside that region the second paint draws over identical scroller content and is invisible. No-op for any other transition style or when no swap is mid-flight.
      Specified by:
      paint in interface Animation
      Overrides:
      paint in class Container
    • addSection

      public StickyHeaderContainer addSection(Component header, Component content)
      Adds a section consisting of a sticky header and its content. The content may be null for a header-only section. Returns this for chaining.
    • addSection

      public StickyHeaderContainer addSection(Component header)
      Adds a header-only section.
    • getScrollContainer

      public Container getScrollContainer()
      Returns the inner scrolling container that hosts the section content. Use this to add non-section components such as a footer, or for programmatic scrolling via setScrollPosition(int).
    • getStickyHost

      public Container getStickyHost()
      Returns the overlay container that hosts the currently-pinned header. While a section is active its header lives here; otherwise the host is empty and zero-height.
    • getStickyHeaders

      public List<Component> getStickyHeaders()
      Returns an unmodifiable view of the registered sticky headers in the order they were added.
    • getActiveSectionIndex

      public int getActiveSectionIndex()
      Returns the index of the currently pinned section, or -1 if no header is currently pinned.
    • setTransitionStyle

      public void setTransitionStyle(int style)
      Selects how the pinned header is replaced when the next section rises into the slot. One of TRANSITION_NONE, TRANSITION_SLIDE (default), TRANSITION_FADE or TRANSITION_COVER. The fade-out of the pinned header is a separate, composable concern controlled by setHeaderFadeOut(boolean).
    • getTransitionStyle

      public int getTransitionStyle()
    • setHeaderFadeOut

      public void setHeaderFadeOut(boolean fade)
      When true the pinned header fades to transparency in proportion to the push progress as the next section rises into the slot. This is independent of the transition style and composes with TRANSITION_SLIDE (slide + fade), TRANSITION_COVER (cover + fade) and TRANSITION_NONE (fade only). TRANSITION_FADE implies the fade-out behaviour regardless of this flag. Defaults to false.
    • isHeaderFadeOut

      public boolean isHeaderFadeOut()
      Returns whether the pinned header is configured to fade out during a swap. See setHeaderFadeOut(boolean).
    • setTransitionDurationMillis

      public void setTransitionDurationMillis(int millis)
      Retained for API compatibility. Transitions are now scroll-driven so the per-frame duration no longer affects visuals; the value is validated and stored but otherwise unused.
    • getTransitionDurationMillis

      public int getTransitionDurationMillis()
    • isTransitionInProgress

      public boolean isTransitionInProgress()
      Returns true while the next section's header is overlapping the pinned slot, i.e. the scroll-driven transition is mid-flight.
    • getTransitionProgress

      public float getTransitionProgress()
      Returns the progress of the in-flight transition as a fraction in [0, 1]: 0 when the next section is just touching the slot from below and 1 when it has fully displaced the pinned header. Returns 0 when no transition is in progress.
    • setScrollPosition

      public void setScrollPosition(int y)
      Sets the scroll position of the inner scroll container. The value is clamped to the valid range and triggers a sticky-header recompute.
    • getScrollPosition

      public int getScrollPosition()
      Returns the current scroll position of the inner scroll container.
    • clearSections

      public void clearSections()
      Removes all sections and content from the container.
    • updateSticky

      public void updateSticky()
      Recomputes which section header should be pinned and how far the next section has displaced it. Called internally on every scroll event; call it explicitly when section content has been mutated outside of a normal layout cycle.